views:

1946

answers:

3

I am making a loading spinner with html5 canvas. I have my graphic on the canvas but when i rotate it the image rotates off the canvas. How do I tell it to spin the graphic on its center point?

<!DOCTYPE html>
<html>
 <head>
  <title>Canvas test</title>
  <script type="text/javascript">
   window.onload = function() {
    var drawingCanvas = document.getElementById('myDrawing');
    // Check the element is in the DOM and the browser supports canvas
    if(drawingCanvas && drawingCanvas.getContext) {
     // Initaliase a 2-dimensional drawing context
     var context = drawingCanvas.getContext('2d');

     //Load the image object in JS, then apply to canvas onload     
     var myImage = new Image();
     myImage.onload = function() {
      context.drawImage(myImage, 0, 0, 27, 27);
     }
     myImage.src = "img/loading.png";

     context.rotate(45);
    }
   }
  </script>
 </head>
 <body>
  <canvas id="myDrawing" width="27" height="27">
  </canvas>
 </body>
</html>
+1  A: 

rotate turns the canvas(?) around your current position, which is 0, 0 to start. you need to "move" to your desired center point, which you can accomplish with

context.translate(x,y);

after you move your reference point, you want to center your image over that point. you can do this by calling

context.drawImage(myImage, -(27/2), -(27/2), 27, 27);

this tells the browser to start drawing the image from above and to the left of your current reference point, by have the size of the image, whereas before you were starting at your reference point and drawing entirely below and to the right (all directions relative to the rotation of the canvas).

since your canvas is the size of your image, your call to translate will use the same measurement, (27/2), for x and y coordinates.

so, to put it all together

// initialization:
context.translate(27/2, 27/2);

// onload: 
context.rotate(Math.PI * 45 / 180);
context.drawImage(myImage, -(27/2), -(27/2), 27, 27);

edit: also, rotation units are radians, so you'll need to translate degrees to radians in your code.

edits for rearranging stuff.

lincolnk
A: 

I'm closer. This will rotate the image in the top left corner. How can I center it up? Also it looks like it is redrawing the image each time overlapping the previous image. How can I just have on instance of the image on the canvas? Do I need to remove it each time?

    <!DOCTYPE html>
    <html>
     <head>
      <title>Canvas test</title>
      <script type="text/javascript">
         setInterval(draw, 150);
          var degrees = 0.0;
          function draw() {
        var drawingCanvas = document.getElementById('myDrawing');
        // Check the element is in the DOM and the browser supports canvas
        if(drawingCanvas && drawingCanvas.getContext) {

         // Initaliase a 2-dimensional drawing context
         var context = drawingCanvas.getContext('2d');

        context.translate(0, 0);
        context.rotate(Math.PI * 45 / 180);

         //Load the image object in JS, then apply to canvas onload     
         var myImage = new Image();
         myImage.onload = function() {
          context.drawImage(myImage, -(50/2), -(50/2), 50, 50);
         }

         myImage.src = "spin.png";

        }
       }
      </script>
     </head>
     <body>
      <center><canvas id="myDrawing" width="50" height="50"></center>
      </canvas>
     </body>
    </html>
bill
Here is the image I am using in this example..http://devthought.com/wp-content/projects/mootools/APNG/Demo/images/zoom-spin-12.png
bill
+4  A: 

Here is the complete working example:)

<!DOCTYPE html>
<html>
    <head>
        <title>Canvas Cog</title>
        <script type="text/javascript">
            var cog = new Image();
            function init() {
                cog.src = ''; // Set source path
                setInterval(draw,10);
            }
            var rotation = 0;
            function draw(){
                var ctx = document.getElementById('myCanvas').getContext('2d');
                ctx.globalCompositeOperation = 'destination-over';
                ctx.save();
                ctx.clearRect(0,0,27,27);
                ctx.translate(13.5,13.5); // to get it in the origin
                rotation +=1;
                ctx.rotate(rotation*Math.PI/64); //rotate in origin
                ctx.translate(-13.5,-13.5); //put it back
                ctx.drawImage(cog,0,0);
                ctx.restore();
            }
            init();
        </script>
    </head>
    <body>
        <canvas width="27" height="27" id="myCanvas"></canvas>
    </body>
</html>
Bill