Using dojox.gfx to draw a movable shape

Lets walk through how to use dojox.gfx to make drawings embedded in web pages that users can manipulate by dragging them around with their mouse (using dojo version 1.0.0).

Step 1: Draw a rectangle:


In the page <HEAD> are the required includes to load the relevant dojo resources. At a minimum for these drawings are dojo.js and dojox.gfx. (Other dojo code may need additional dojo.require statments especially dojo.parser, and may need dojo.css as well):

<HEAD>
<script type='text/javascript' 
   src='../dojo/1.0.0/dojo/dojo.js' 
   djConfig='parseOnLoad: true'>
</script>
<script type='text/javascript'>
   dojo.require('dojox.gfx');   // omit this and the rectangle doesn't draw
</script>
</HEAD>

In the <BODY>, create a DOM node that can be used as a drawing surface. Then in a script create a drawing surface (a canvas), and draw the rectangle on that surface.

<body>
<div id='drawing_surface'></div>

<script type='text/javascript'>
    // createSurface(parentNode, width, height);
    surface = dojox.gfx.createSurface(dojo.byId('drawing_surface'),'640px', '110px');
    // draw rectangle on the surface
    // createRect({}) parameters
    //             x: horizontal offset from left side of surface
    //             y: vertical offset from top of surface
    //             height, width: dimensions of rectangle
    rectangle = surface.createRect({ x: 20, y: 5, height: 100, width: 150})
                    .setFill([0,0,255,0.3])   
                    .setStroke({ color: "blue", width: 2})
                  ;
</script>
</body>

Step 2: Make the rectangle movable:


Simply add dojox.gfx.move to the includes in the <HEAD>

<style type='text/css'>
   @import '../dojo/1.0.0/dojo/resources/dojo.css';
</style>
<script type='text/javascript' 
   src='../dojo/1.0.0/dojo/dojo.js' 
   djConfig='parseOnLoad: true'>
</script>
<script type='text/javascript'>
   dojo.require('dojox.gfx');   // omit this and the rectangle doesn't draw
   dojo.require('dojox.gfx.move');  // omit this and the rectangle doesn't move
   dojo.require('dojo.parser');  // parser and dojo.css aren't required
</script>

And make the rectangle movable by using the Movable constructor:

<div id='drawing_surface'></div>

<script type='text/javascript'>
    surface = dojox.gfx.createSurface(dojo.byId('drawing_surface'),'500px', '300px');
    rectangle = surface.createRect({ x: 20, y: 10, height: 100, width: 100})
                    .setFill([0,0,255,0.3])   
                    .setStroke({ color: "blue", width: 2})
                  ;
    // make the rectangle draggable
    new dojox.gfx.Moveable(rectangle);
</script>

Step 3: Limit the motion of the rectangle:


The <HEAD> contains the same set of included resources.

<style type='text/css'>
   @import '../dojo/1.0.0/dojo/resources/dojo.css';
</style>
<script type='text/javascript' 
     src='../dojo/1.0.0/dojo/dojo.js' 
     djConfig='parseOnLoad: true'>
</script>
<script type='text/javascript'>
   dojo.require('dojox.gfx');   // omit this and the rectangle doesn't draw
   dojo.require('dojox.gfx.move');  // omit this and the rectangle does't move
   dojo.require('dojo.parser');
</script>

Now things get more complex. We want to make the rectangle dragabble only to the edges of the drawing surface (so that it stops moving rather than vanishing at the edge of the surrface). We will do this by using a custom Mover class rather than the default dojox.gfx.Mover. We will declare a subclass of dojox.gfx.Mover that overrides the onMouseMove method of dojox.gfx.Mover. This overridden method will need to be aware of both the limits of motion on the drawing surface and the boundaries of the rectangle. We then just need to make this rectangle movable using the custom mover:

<div id='drawing_surface'></div>

<script type='text/javascript'>
      // Create a drawing surface 640 by 140 pixels in size.
      surface = dojox.gfx.createSurface(dojo.byId('drawing_surface'),'640px', '150px');
      // Create a rectangle on the drawing surface.
      areaofinterest = surface.createRect({ x: 20, y: 10, height: 100, width: 100})
                    .setFill([0,0,255,0.3])
                    .setStroke({ color: "blue", width: 2})
                  ;
      // define limits to motion - in this case, the entire drawing surface.
      var limits = { xmin: 0, xmax: 640, ymin: 0, ymax: 150};
      
      // Extend mover with a class that overwites onMouseMove.
      // We will call the class net.aa3sd.surfaceMover to prevent naming 
      // conflicts with any existing dojo classes
      dojo.declare("net.aa3sd.surfaceMover",dojox.gfx.Mover, { 
           // Mover has several methods, we only need to overwrite onMouseMove.
           onMouseMove: function(event) {
              // getTransform() will tell us how far the shape has been moved
              // from its initial position
              transform = this.shape.getTransform();
              // If this.shape hasn't been transformed yet 
              // getTransform() will return null, so we
              // need to handle the special case of the first mouse movement.
              if (transform==null) { 
                 // We aren't dealing with rotations here, so we will 
                 // just define initial values for the translations dx and dy.
                 transform = { dx: 0, dy: 0 };
              }
              // event.layerX is the mouse position in the coordinate system of the
              // layer that was clicked on, the rectangle in this case
              var layerx = event.layerX;  
              var layery = event.layerY;
              if (this.click_on_rectangle==null) { 
                // we need to find where in the rectangle the mouse pointer is
                // otherwise the rectangle will drag untill the mouse pointer
                // hits the limits, rather than when the rectangel hits 
                // the limits.
                this.click_on_rectangle  = {  
                    x: layerx - this.shape.shape.x - transform.dx,
                    y: layery - this.shape.shape.y - transform.dy
                 }  // x,y of initial mouse click in coordinate system of rectangle
              }
              var click_on_surface = {
                 x: layerx - this.click_on_rectangle.x,
                 y: layery - this.click_on_rectangle.y 
              };   // x,y of mouse click in coordinate system of drawing surface
              var x = event.clientX;
              var y = event.clientY;
              // check to see if the edges of the rectangle are within the
              // limits of movement, if so, allow the rectangle to be moved
              if (click_on_surface.x > limits.xmin 
                & click_on_surface.y > limits.ymin 
                & click_on_surface.x < limits.xmax - areaofinterest.shape.width
                & click_on_surface.y < limits.ymax - areaofinterest.shape.height
                 ) { 
                    // move the rectangle by applying a translation
                    this.shape.applyLeftTransform({ 
                           dx: x - this.lastX, 
                           dy: y - this.lastY
                    });
                    // store the last position of the rectangle
                    this.lastX = x;
                    this.lastY = y;
              }
              dojo.stopEvent(event);
           }
           });
      // now we simply make the rectangle movable using the custom mover.
      new dojox.gfx.Moveable(areaofinterest, { mover: net.aa3sd.surfaceMover });
</script>

Debugging

Debugging javascript code can be complex, and trying to understand what different coordinate systems are being used by the drawing and the mouse movements can be challenging. A technique to help here is to display the coordinates on the page as the mouse moves. We can use this technique to get feedback on coordinates displayed in the page or to update form elements to store the transformation:


We will just add a div tag on the page where we want the feedback to appear, and then in the onMouseMove method replace the content of that div with some information about the current coordinates:

<div id='drawing_surface'></div>

<script type='text/javascript'>
      ....
      dojo.declare("net.aa3sd.surfaceMover",dojox.gfx.Mover, { 
           onMouseMove: function(event) {
              ...
              dojo.byId('temp').innerHTML = "event.layerX=" + layerx 
                  + ", event.layerY=" + layery + " event.clientX=" 
                  + x + ", event.clientY="+ y + " transform.dx=" 
                  + transform.dx + ", transform.dy=" transform.dy ;
              dojo.stopEvent(event);
           }
           });
      new dojox.gfx.Moveable(rectangle_d, { mover: net.aa3sd.surfaceMover_d });

</script>

<div id=temp></div>