97
jQuery('#carregar').click(function() {
  var canvas    = document.getElementById('canvas');
  var image   = document.getElementById('image');
  var element = canvas.getContext("2d");
  element.clearRect(0, 0, canvas.width, canvas.height);
  element.drawImage(image, 0, 0, 300, 300);
});

jsfiddle.net/braziel/nWyDE/

I have a problem to rotate an image 90 ° to the right or to the left.

I use an image on the canvas, the same screen will have several canvas equal to that of the example, but I left it as close as possible to the project.

I ask, how do I rotate the image 90 ° to the left or right when I click "Rotate Left" and "Rotate Right"?

I tried several codes on the internet but none worked.

Mosh Feu
  • 28,354
  • 16
  • 88
  • 135
Max William Vitorino
  • 1,200
  • 2
  • 11
  • 19

11 Answers11

185

You can use canvas’ context.translate & context.rotate to rotate your image

enter image description here

Here’s a function to draw an image that is rotated by the specified degrees:

function drawRotated(degrees){
    context.clearRect(0,0,canvas.width,canvas.height);

    // save the unrotated context of the canvas so we can restore it later
    // the alternative is to untranslate & unrotate after drawing
    context.save();

    // move to the center of the canvas
    context.translate(canvas.width/2,canvas.height/2);

    // rotate the canvas to the specified degrees
    context.rotate(degrees*Math.PI/180);

    // draw the image
    // since the context is rotated, the image will be rotated also
    context.drawImage(image,-image.width/2,-image.width/2);

    // we’re done with the rotating so restore the unrotated context
    context.restore();
}

Here is code and a Fiddle: http://jsfiddle.net/m1erickson/6ZsCz/

<!doctype html>
<html>
<head>
<link rel="stylesheet" type="text/css" media="all" href="css/reset.css" /> <!-- reset css -->
<script type="text/javascript" src="http://code.jquery.com/jquery.min.js"></script>

<style>
    body{ background-color: ivory; }
    canvas{border:1px solid red;}
</style>
    
<script>
$(function(){

    var canvas=document.getElementById("canvas");
    var ctx=canvas.getContext("2d");

    var angleInDegrees=0;

    var image=document.createElement("img");
    image.onload=function(){
        ctx.drawImage(image,canvas.width/2-image.width/2,canvas.height/2-image.width/2);
    }
    image.src="houseicon.png";

    $("#clockwise").click(function(){ 
        angleInDegrees+=30;
        drawRotated(angleInDegrees);
    });

    $("#counterclockwise").click(function(){ 
        angleInDegrees-=30;
        drawRotated(angleInDegrees);
    });

    function drawRotated(degrees){
        ctx.clearRect(0,0,canvas.width,canvas.height);
        ctx.save();
        ctx.translate(canvas.width/2,canvas.height/2);
        ctx.rotate(degrees*Math.PI/180);
        ctx.drawImage(image,-image.width/2,-image.width/2);
        ctx.restore();
    }

       
}); // end $(function(){});
</script>

</head>

<body>
    <canvas id="canvas" width=300 height=300></canvas><br>
    <button id="clockwise">Rotate right</button>
    <button id="counterclockwise">Rotate left</button>
</body>
</html>
Lee Taylor
  • 7,761
  • 16
  • 33
  • 49
markE
  • 102,905
  • 11
  • 164
  • 176
  • so what about putImageData( ) ? It doesn't seem to work – TheRealChx101 Apr 11 '14 at 06:25
  • Well, putImageData *would work*, but it's a much more expensive operation to do the same thing--so putImageData is not called for here! ;-) – markE Apr 11 '14 at 15:01
  • Appears you have a typo: ctx.drawImage(image,-image.width/2,-image.width/2); should be ctx.drawImage(image,-image.width/2,-image.height/2); – Steve Farthing Jul 02 '14 at 13:54
  • @SteveFarthing No typo...when rotated 90 degrees, the width becomes the height and the height becomes the width. ;-) – markE Jul 02 '14 at 14:40
  • 25
    This code works perfectly for square images. For rectangle images the image is cropped incorrectly (see http://jsfiddle.net/6ZsCz/803/ ). If you change ctx.drawImage(image,-image.width/2,-image.width/2); to be ctx.drawImage(image,-image.width/2,-image.height/2) then it works for all size images. – Steve Farthing Jul 02 '14 at 16:10
  • @SteveFarthing. Cropping is not "incorrect", it's just a design option--like overflow:hidden. :-) If your own design requires displaying the full rectangular image you could (1) increase the size of the canvas (2) decrease the size of your image when rotated at 90 or 270. Cheers! – markE Jul 02 '14 at 16:24
  • 11
    ok. Heres an example where the canvas fits to the image and parts of the image are not lost: http://jsfiddle.net/6ZsCz/804/ . – Steve Farthing Jul 02 '14 at 16:27
  • 1
    @SteveFarthing can you put this as an answer? It works flawlessly and I'm positive that anyone who looks for this functionality needs your 'corrected' solution – nicholaswmin Mar 29 '15 at 15:21
  • 2
    @markE : for the 3rd time this month you helped me out :-D – Dr. Gianluigi Zane Zanettini Jan 30 '16 at 12:05
  • Please check you fiddle, it doesn't work: `Uncaught DOMException: Failed to execute 'drawImage' on 'CanvasRenderingContext2D': The HTMLImageElement provided is in the 'broken' state.` – deathangel908 Jul 09 '17 at 20:44
  • Great job as always Mark! Updated JSFiddle with working image: https://jsfiddle.net/6ZsCz/3455/ – Ohad Schneider Nov 05 '17 at 11:49
65

Quickest 2D context image rotation method

A general purpose image rotation, position, and scale.

// no need to use save and restore between calls as it sets the transform rather 
// than multiply it like ctx.rotate ctx.translate ctx.scale and ctx.transform
// Also combining the scale and origin into the one call makes it quicker
// x,y position of image center
// scale scale of image
// rotation in radians.
function drawImage(image, x, y, scale, rotation){
    ctx.setTransform(scale, 0, 0, scale, x, y); // sets scale and origin
    ctx.rotate(rotation);
    ctx.drawImage(image, -image.width / 2, -image.height / 2);
} 

If you wish to control the rotation point use the next function

// same as above but cx and cy are the location of the point of rotation
// in image pixel coordinates
function drawImageCenter(image, x, y, cx, cy, scale, rotation){
    ctx.setTransform(scale, 0, 0, scale, x, y); // sets scale and origin
    ctx.rotate(rotation);
    ctx.drawImage(image, -cx, -cy);
} 

To reset the 2D context transform

ctx.setTransform(1,0,0,1,0,0); // which is much quicker than save and restore

Thus to rotate image to the left (anti clockwise) 90 deg

drawImage(image, canvas.width / 2, canvas.height / 2, 1, - Math.PI / 2);

Thus to rotate image to the right (clockwise) 90 deg

drawImage(image, canvas.width / 2, canvas.height / 2, 1, Math.PI / 2);

Example draw 500 images translated rotated scaled

var image = new Image;
image.src = "https://i.stack.imgur.com/C7qq2.png?s=328&g=1";
var canvas = document.createElement("canvas");
var ctx = canvas.getContext("2d");
canvas.style.position = "absolute";
canvas.style.top = "0px";
canvas.style.left = "0px";
document.body.appendChild(canvas);
var w,h;
function resize(){ w = canvas.width = innerWidth; h = canvas.height = innerHeight;}
resize();
window.addEventListener("resize",resize);
function rand(min,max){return Math.random() * (max ?(max-min) : min) + (max ? min : 0) }
function DO(count,callback){ while (count--) { callback(count) } }
const sprites = [];
DO(500,()=>{
    sprites.push({
       x : rand(w), y : rand(h),
       xr : 0, yr : 0, // actual position of sprite
       r : rand(Math.PI * 2),
       scale : rand(0.1,0.25),
       dx : rand(-2,2), dy : rand(-2,2),
       dr : rand(-0.2,0.2),
    });
});
function drawImage(image, spr){
    ctx.setTransform(spr.scale, 0, 0, spr.scale, spr.xr, spr.yr); // sets scales and origin
    ctx.rotate(spr.r);
    ctx.drawImage(image, -image.width / 2, -image.height / 2);
}
function update(){
    var ihM,iwM;
    ctx.setTransform(1,0,0,1,0,0);
    ctx.clearRect(0,0,w,h);
    if(image.complete){
      var iw = image.width;
      var ih = image.height;
      for(var i = 0; i < sprites.length; i ++){
          var spr = sprites[i];
          spr.x += spr.dx;
          spr.y += spr.dy;
          spr.r += spr.dr;
          iwM = iw * spr.scale * 2 + w;
          ihM = ih * spr.scale * 2 + h;
          spr.xr = ((spr.x % iwM) + iwM) % iwM - iw * spr.scale;
          spr.yr = ((spr.y % ihM) + ihM) % ihM - ih * spr.scale;
          drawImage(image,spr);
      }
    }    
    requestAnimationFrame(update);
}
requestAnimationFrame(update);
Blindman67
  • 51,134
  • 11
  • 73
  • 136
  • 1
    Thank you - I'm happy - How fast is my computer is !?! – T4NK3R Aug 22 '18 at 07:26
  • @Blindman67 Thanks for the great solution! Would you please kindly explain what's being done in the following lines? I suck at the math part :/ - iwM = iw * spr.scale * 2 + w; ihM = ih * spr.scale * 2 + h; spr.xr = ((spr.x % iwM) + iwM) % iwM - iw * spr.scale; spr.yr = ((spr.y % ihM) + ihM) % ihM - ih * spr.scale; – zumek Jan 12 '19 at 02:45
  • 1
    @zumek `spr.xr` and `yr` represent the rendered coordinate of the sprite. The first two lines get the width/height of image (canvas) with margins that allow sprite to move off the canvas (on all side). Next two get the position offsetting by margin size up and left. The % cycles position (like asteroids move off left appear on right) The margins ensure that you don't see the sprite disappear and then reappear on opposite side. – Blindman67 Jan 12 '19 at 03:33
  • This is underrated. –  May 29 '20 at 01:24
  • is there any way to rotate the `drawImage` element without rotating the `canvas` itself? – oldboy Dec 16 '20 at 05:30
  • in this [answer](https://stackoverflow.com/a/46811056/7543162), the `ctx` is being rotated instead of the `drawImage` element, right? is there any way to rotate the `drawImage` element directly without having to rotate the `ctx` and then unrotate it, etc? – oldboy Dec 16 '20 at 20:54
  • so the way in the answer that i linked is slower than `setTransform`, `rotate`, `drawImage`? ok, but is there any way to rotate the "pen" instead of rotating the "canvas"? – oldboy Dec 16 '20 at 23:35
  • yes, i know rotation ultimately takes place via a transform, but the transform is applied to the canvas/context. what am i not understanding? youre confusing me – oldboy Dec 16 '20 at 23:48
  • i now see whence the confusion was arising. i wasnt asking whether or not we can rotate the function `drawImage` (i know that would make zero sense), but rather the **element** or "output/application/etc" (perhaps a poor choice of words) of `drawImage`. consider my Q in the context of the following metaphor whereby the canvas/context is a piece of a paper on a desk and `drawImage` is the pen/pencil. right now, we are rotating the piece of paper, then drawing on it, then unrotating the paper. is it possible to instead rotate the pen or hand (similar to how CSS transforms are applied to eles)? – oldboy Dec 17 '20 at 01:02
  • ok. the way canvas goes about it is so much more confusing than vanilla CSS transforms. thanks for your help. – oldboy Dec 17 '20 at 01:56
  • @Blindman67 this is a nice solution, is there a non-square solution, e.g., scale.x and scale.y are different? Suppose I had an image originally of size 1800x1800 but I want to place it as 180x120 and I want to rotate it? – mshaffer Dec 25 '20 at 16:52
  • @mshaffer Yes you can. Just set the axis scales with `ctx.setTransform(scaleX, 0, 0, scaleY, x, y) ` where `scaleX` and `scaleY` replace `scale` in the `drawImage` functions. – Blindman67 Dec 25 '20 at 17:59
30

The other solution works great for square images. Here is a solution that will work for an image of any dimension. The canvas will always fit the image rather than the other solution which may cause portions of the image to be cropped off.

var canvas;

var angleInDegrees=0;

var image=document.createElement("img");
image.onload=function(){

    drawRotated(0);
}
image.src="http://greekgear.files.wordpress.com/2011/07/bob-barker.jpg";

$("#clockwise").click(function(){ 
    angleInDegrees+=90 % 360;
    drawRotated(angleInDegrees);
});

$("#counterclockwise").click(function(){ 
    if(angleInDegrees == 0)
        angleInDegrees = 270;
    else
        angleInDegrees-=90 % 360;
    drawRotated(angleInDegrees);
});

function drawRotated(degrees){
    if(canvas) document.body.removeChild(canvas);

    canvas = document.createElement("canvas");
    var ctx=canvas.getContext("2d");
    canvas.style.width="20%";

    if(degrees == 90 || degrees == 270) {
        canvas.width = image.height;
        canvas.height = image.width;
    } else {
        canvas.width = image.width;
        canvas.height = image.height;
    }

    ctx.clearRect(0,0,canvas.width,canvas.height);
    if(degrees == 90 || degrees == 270) {
        ctx.translate(image.height/2,image.width/2);
    } else {
        ctx.translate(image.width/2,image.height/2);
   }
    ctx.rotate(degrees*Math.PI/180);
    ctx.drawImage(image,-image.width/2,-image.height/2);

    document.body.appendChild(canvas);
}

http://jsfiddle.net/6ZsCz/1588/

Steve Farthing
  • 792
  • 6
  • 10
14

This is the simplest code to draw a rotated and scaled image:

function drawImage(ctx, image, x, y, w, h, degrees){
  ctx.save();
  ctx.translate(x+w/2, y+h/2);
  ctx.rotate(degrees*Math.PI/180.0);
  ctx.translate(-x-w/2, -y-h/2);
  ctx.drawImage(image, x, y, w, h);
  ctx.restore();
}
Mohammad Usman
  • 37,952
  • 20
  • 92
  • 95
CpnCrunch
  • 4,831
  • 1
  • 33
  • 31
  • 1
    is there any way to rotate the `drawImage` element without rotating the `canvas` itself? is `save restore` really the only way? – oldboy Dec 16 '20 at 05:59
  • 2
    @oldboy canvas isnt rotated here. Rotation is applied to context which works inside canvas – aleha_84 Dec 16 '20 at 10:22
  • @aleha so then, is there any way to rotate the `drawImage` element directly without rotating the `ctx` itself? – oldboy Dec 16 '20 at 20:49
  • 2
    @oldboy I think you misunderstood the concept. ctxs.drawImage only draws. – aleha_84 Dec 16 '20 at 21:04
  • @aleha yes, i know that `drawImage` only draws. it draws an image onto the canvas or context. i am wondering if it is possible to directly rotate the image itself instead of rotating the canvas or context...? – oldboy Dec 16 '20 at 23:32
6

As @markE mention in his answer

the alternative is to untranslate & unrotate after drawing

It is much faster than context save and restore.

Here is an example

// translate and rotate
this.context.translate(x,y);
this.context.rotate(radians);
this.context.translate(-x,-y);

this.context.drawImage(...);    

// untranslate and unrotate
this.context.translate(x, y);
this.context.rotate(-radians);
this.context.translate(-x,-y);
aleha_84
  • 8,309
  • 2
  • 38
  • 46
6

This is full degree image rotation code. I recommend you to check the below example app in the jsfiddle.

https://jsfiddle.net/casamia743/xqh48gno/

enter image description here

The process flow of this example app is

  1. load Image, calculate boundaryRad
  2. create temporary canvas
  3. move canvas context origin to joint position of the projected rect
  4. rotate canvas context with input degree amount
  5. use canvas.toDataURL method to make image blob
  6. using image blob, create new Image element and render

function init() {
  ...
  image.onload = function() {
     app.boundaryRad = Math.atan(image.width / image.height);
  }
  ...
}



/**
 * NOTE : When source rect is rotated at some rad or degrees, 
 * it's original width and height is no longer usable in the rendered page.
 * So, calculate projected rect size, that each edge are sum of the 
 * width projection and height projection of the original rect.
 */
function calcProjectedRectSizeOfRotatedRect(size, rad) {
  const { width, height } = size;

  const rectProjectedWidth = Math.abs(width * Math.cos(rad)) + Math.abs(height * Math.sin(rad));
  const rectProjectedHeight = Math.abs(width * Math.sin(rad)) + Math.abs(height * Math.cos(rad));

  return { width: rectProjectedWidth, height: rectProjectedHeight };
}

/**
 * @callback rotatedImageCallback
 * @param {DOMString} dataURL - return value of canvas.toDataURL()
 */

/**
 * @param {HTMLImageElement} image 
 * @param {object} angle
 * @property {number} angle.degree 
 * @property {number} angle.rad
 * @param {rotatedImageCallback} cb
 * 
 */
function getRotatedImage(image, angle, cb) {
  const canvas = document.createElement('canvas');
  const { degree, rad: _rad } = angle;

  const rad = _rad || degree * Math.PI / 180 || 0;
  debug('rad', rad);

  const { width, height } = calcProjectedRectSizeOfRotatedRect(
    { width: image.width, height: image.height }, rad
  );
  debug('image size', image.width, image.height);
  debug('projected size', width, height);

  canvas.width = Math.ceil(width);
  canvas.height = Math.ceil(height);

  const ctx = canvas.getContext('2d');
  ctx.save();

  const sin_Height = image.height * Math.abs(Math.sin(rad))
  const cos_Height = image.height * Math.abs(Math.cos(rad))
  const cos_Width = image.width * Math.abs(Math.cos(rad))
  const sin_Width = image.width * Math.abs(Math.sin(rad))

  debug('sin_Height, cos_Width', sin_Height, cos_Width);
  debug('cos_Height, sin_Width', cos_Height, sin_Width);

  let xOrigin, yOrigin;

  if (rad < app.boundaryRad) {
    debug('case1');
    xOrigin = Math.min(sin_Height, cos_Width);
    yOrigin = 0;
  } else if (rad < Math.PI / 2) {
    debug('case2');
    xOrigin = Math.max(sin_Height, cos_Width);
    yOrigin = 0;
  } else if (rad < Math.PI / 2 + app.boundaryRad) {
    debug('case3');
    xOrigin = width;
    yOrigin = Math.min(cos_Height, sin_Width);
  } else if (rad < Math.PI) {
    debug('case4');
    xOrigin = width;
    yOrigin = Math.max(cos_Height, sin_Width);
  } else if (rad < Math.PI + app.boundaryRad) {
    debug('case5');
    xOrigin = Math.max(sin_Height, cos_Width);
    yOrigin = height;
  } else if (rad < Math.PI / 2 * 3) {
    debug('case6');
    xOrigin = Math.min(sin_Height, cos_Width);
    yOrigin = height;
  } else if (rad < Math.PI / 2 * 3 + app.boundaryRad) {
    debug('case7');
    xOrigin = 0;
    yOrigin = Math.max(cos_Height, sin_Width);
  } else if (rad < Math.PI * 2) {
    debug('case8');
    xOrigin = 0;
    yOrigin = Math.min(cos_Height, sin_Width);
  }

  debug('xOrigin, yOrigin', xOrigin, yOrigin)

  ctx.translate(xOrigin, yOrigin)
  ctx.rotate(rad);
  ctx.drawImage(image, 0, 0);
  if (DEBUG) drawMarker(ctx, 'red');

  ctx.restore();

  const dataURL = canvas.toDataURL('image/jpg');

  cb(dataURL);
}

function render() {
    getRotatedImage(app.image, {degree: app.degree}, renderResultImage)
}
casamia
  • 619
  • 1
  • 10
  • 19
3

I built an image uploader that resizes the image in the background, and also allows the user to rotate the image 90 degrees left or right. This isn't something that is solved with just a few lines of code, so you will want to checkout the jsfiddle https://jsfiddle.net/alienbush/z02jatnr/6/. I also included the complete code below.

The main drawing function looks like this:

let drawOptimizedImage = function (canvas, image, maxSize, rotationDirection) {

    let degrees = updateRotationDegrees(rotationDirection)
    let newSize = determineSize(image.width, image.height, maxSize.width, maxSize.height, degrees)

    canvas.width = newSize.width
    canvas.height = newSize.height

    let ctx = canvas.getContext('2d')
    ctx.save()
    ctx.clearRect(0, 0, canvas.width, canvas.height)

    if (degrees === 0) {
        ctx.drawImage(image, 0, 0, newSize.width, newSize.height)
    } 
    else {
        ctx.translate(canvas.width / 2, canvas.height / 2)
        ctx.rotate(degrees * Math.PI / 180)

        if (Math.abs(degrees) === 180) {
            ctx.drawImage(image, -newSize.width / 2, -newSize.height / 2, newSize.width, newSize.height)
        }
        else { // 90 or 270 degrees (values for width and height are swapped for these rotation positions)
            ctx.drawImage(image, -newSize.height / 2, -newSize.width / 2, newSize.height, newSize.width)
        }
    }

    ctx.restore()
}

Here's the complete code:

let imgPreview = document.getElementById('imgPreview')
let canvas = document.getElementById('canvas')
let statusMessage = document.getElementById('statusMessage')
let img = new Image
let maxSize = {
  width: 800,
  height: 600
}
let rotationDegrees = 0

// canvas.toBlob Polyfill from https://gist.github.com/salzhrani/02a6e807f24785a4d34b
if (!HTMLCanvasElement.prototype.toBlob) {
  Object.defineProperty(HTMLCanvasElement.prototype, 'toBlob', {
    value: function(callback, type, quality) {
      var bin = atob(this.toDataURL(type, quality).split(',')[1]),
        len = bin.length,
        len32 = len >> 2,
        a8 = new Uint8Array(len),
        a32 = new Uint32Array(a8.buffer, 0, len32);

      for (var i = 0, j = 0; i < len32; i++) {
        a32[i] = bin.charCodeAt(j++) |
          bin.charCodeAt(j++) << 8 |
          bin.charCodeAt(j++) << 16 |
          bin.charCodeAt(j++) << 24;
      }

      var tailLength = len & 3;

      while (tailLength--) {
        a8[j] = bin.charCodeAt(j++);
      }

      callback(new Blob([a8], {
        'type': type || 'image/png'
      }));
    }
  });
}

document.getElementById('fileInput').addEventListener('change', function(e) {
  if (!e.target.files.length) {
    return
  }
  let file = e.target.files[0]
  if (isValidMIME(file, ['image/bmp', 'image/jpeg', 'image/png'])) {
    img.src = window.URL.createObjectURL(file)
  } else {
    alert('Invalid file type. The image file must be one of the following:  .jpg  .jpeg  .png  .bmp')
  }
})

let isValidMIME = function(file, MIMEtypes) {
  for (let i = 0; i < MIMEtypes.length; i++) {
    if (MIMEtypes[i] === file.type) {
      return true
    }
  }
  return false
}

img.addEventListener('load', function() {
  rotationDegrees = 0
  removeStatusMessage()
  drawOptimizedImage(canvas, img, maxSize)
  updateImgPreview(canvas, imgPreview)
})

let removeStatusMessage = function() {
  statusMessage.textContent = ''
  statusMessage.style.display = 'none'
}

let drawOptimizedImage = function(canvas, image, maxSize, rotationDirection) {
  let degrees = updateRotationDegrees(rotationDirection)
  let newSize = determineSize(image.width, image.height, maxSize.width, maxSize.height, degrees)

  canvas.width = newSize.width
  canvas.height = newSize.height

  let ctx = canvas.getContext('2d')
  ctx.save()
  ctx.clearRect(0, 0, canvas.width, canvas.height)

  if (degrees === 0) {
    ctx.drawImage(image, 0, 0, newSize.width, newSize.height)
  } else {
    ctx.translate(canvas.width / 2, canvas.height / 2)
    ctx.rotate(degrees * Math.PI / 180)

    if (Math.abs(degrees) === 180) {
      ctx.drawImage(image, -newSize.width / 2, -newSize.height / 2, newSize.width, newSize.height)
    } else { // 90 or 270 degrees (values for width and height are swapped for these rotation positions)
      ctx.drawImage(image, -newSize.height / 2, -newSize.width / 2, newSize.height, newSize.width)
    }
  }

  ctx.restore()
}

let updateRotationDegrees = function(rotationDirection) {
  if (rotationDirection === 'clockwise') {
    rotationDegrees += 90
  } else if (rotationDirection === 'anticlockwise') {
    rotationDegrees -= 90
  }
  if (Math.abs(rotationDegrees) === 360) {
    rotationDegrees = 0
  }
  return rotationDegrees
}

let determineSize = function(width, height, maxW, maxH, degrees) {
  let w, h;
  degrees = Math.abs(degrees)
  if (degrees === 90 || degrees === 270) { // values for width and height are swapped for these rotation positions
    w = height
    h = width
  } else {
    w = width
    h = height
  }
  if (w > h) {
    if (w > maxW) {
      h = h * maxW / w
      w = maxW
    }
  } else {
    if (h > maxH) {
      w = w * maxH / h
      h = maxH
    }
  }
  return {
    width: w,
    height: h
  }
}

let updateImgPreview = function(canvas, div) {
  if (canvas.width < div.clientWidth && canvas.height < div.clientHeight) {
    div.style.backgroundSize = 'auto'
  } else {
    div.style.backgroundSize = 'contain'
  }
  div.style.backgroundImage = 'url(' + canvas.toDataURL() + ')'
}

document.getElementById('clockwiseBtn').addEventListener('click', function() {
  removeStatusMessage()
  drawOptimizedImage(canvas, img, maxSize, 'clockwise')
  updateImgPreview(canvas, imgPreview)
})

document.getElementById('anticlockwiseBtn').addEventListener('click', function() {
  removeStatusMessage()
  drawOptimizedImage(canvas, img, maxSize, 'anticlockwise')
  updateImgPreview(canvas, imgPreview)
})

document.getElementById('uploadBtn').addEventListener('click', function() {
  let fileInput = document.getElementById('fileInput')
  if (!fileInput.files.length) {
    alert('Please choose a file first');
    return;
  }

  let formData = new FormData()
  formData.append('fileName', 'yourCustomFileNameHere')

  canvas.toBlob(function(blob) {
    formData.append('image', blob)
    let url = 'theURLtoSendTheFileTo'
    sendForm(url, formData, doAfterUploadSuccess)
  }, 'image/jpeg', 1.0)
})

let sendForm = function(url, formData, callback) {

  // Simulating upload. Use the commented code below for a real upload.
  statusMessage.style.display = 'block'
  statusMessage.textContent = 'Uploading, please wait...'
  setTimeout(callback, 2000)

  // let xhr = new XMLHttpRequest()
  // addUploadListeners(xhr)
  // xhr.open('POST', url, true)
  // xhr.onload = function () {
  //     if (xhr.status == 200) {
  //         if (callback) { callback(xhr) }
  //     }
  //     else if (xhr.status === 0) {
  //         alert('No response from server. Check network connection.')
  //     }
  //     else {
  //         alert('There was a problem uploading:  ' + xhr.statusText)
  //     }
  // }
  // statusMessage.style.display = 'block'
  // xhr.send(formData)
}

let addUploadListeners = function(xhr) {
  xhr.addEventListener('loadstart', function(e) {
    statusMessage.textContent = 'Uploading, please wait...'
  })
  xhr.addEventListener('abort', function(e) {
    statusMessage.textContent = 'Aborted upload'
  })
  xhr.addEventListener('error', function(e) {
    statusMessage.textContent = 'Error during upload'
  })
}

let doAfterUploadSuccess = function(xhr) {
  statusMessage.textContent = 'Success!'
}
body {
  background-color: lightgray;
  min-height: 100vh;
  font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
  font-size: 11pt;
  font-weight: 400;
  -webkit-font-smoothing: antialiased;
  margin: 0;
}

button {
  outline: none !important;
  border-style: none;
  background-color: rgb(115, 115, 250);
  color: white;
  padding: 10px 15px 12px 15px;
  cursor: pointer;
  border-radius: 2px;
  font: inherit;
}

.canvas {
  /* You can use display: none; to hide the canvas. The image will upload the same, whether or not the canvas is diplayed. */
  display: block;
  margin: 0px auto 40px auto;
}

.card {
  background: white;
  border-radius: 2px;
  box-shadow: 0 2px 2px 0 rgba(0, 0, 0, 0.14), 0 3px 1px -2px rgba(0, 0, 0, 0.2), 0 1px 5px 0 rgba(0, 0, 0, 0.12);
}

.img-form {
  padding: 15px 15px 0px 15px;
  width: 320px;
  margin: 0 auto;
}

.img-uploader-panel {
  padding: 10px;
  width: 340px;
  margin: 0px auto 20px auto;
}

.img-uploader-header {
  margin-top: 5px;
  text-align: center;
}

.img-preview {
  display: flex;
  background-repeat: no-repeat;
  background-position-x: center;
  background-position-y: center;
  background-size: contain;
  border: 1px solid rgb(160, 160, 160);
  width: 320px;
  height: 320px;
  margin: 0 auto;
}

.status-message {
  display: none;
  align-self: flex-end;
  margin: 0;
  line-height: 50px;
  background-color: rgba(0, 0, 0, 0.5);
  color: white;
  width: 100%;
  text-align: center;
}

.uploader-btn-panel {
  display: flex;
  padding-top: 10px;
  justify-content: center;
  align-items: center;
}

.rotate-btn {
  font-size: 32px;
  padding: 5px 12px 9px 13px;
  margin-right: 10px;
  background-color: transparent;
  color: rgba(83, 83, 249, 1);
}

.clockwise-btn {
  transform: rotate(90deg);
}

.anticlockwise-btn {
  transform: rotate(270deg);
}

.upload-btn {
  margin-left: 20px;
}
<form action="post" id="imgForm" class="img-form">
  <label for="fileInput">Add an image:</label><br>
  <input type="file" name="fileInput" id="fileInput" accept=".jpg, .jpeg, .png, .bmp">
</form><br>

<div class="img-uploader-panel card">
  <p class="img-uploader-header">Image Preview</p>
  <div id="imgPreview" class="img-preview">
    <p id="statusMessage" class="status-message"></p>
  </div>
  <div class="uploader-btn-panel">
    <button id="anticlockwiseBtn" class="anticlockwise-btn rotate-btn">&#8634;</button>
    <button id="clockwiseBtn" class="clockwise-btn rotate-btn">&#8635;</button>
    <button id="uploadBtn" class="upload-btn">Upload</button>
  </div>
</div>

<canvas id="canvas" class="canvas"></canvas>
McGrew
  • 812
  • 1
  • 9
  • 16
1

Here is Something I did

var ImgRotator = {
    angle:parseInt(45),
    image:{},
    src:"",
    canvasID:"",
    intervalMS:parseInt(500),
    jump:parseInt(5),
    start_action:function(canvasID, imgSrc, interval, jumgAngle){
        ImgRotator.jump = jumgAngle;
        ImgRotator.intervalMS = interval;
        ImgRotator.canvasID = canvasID;
        ImgRotator.src = imgSrc ;
        var image = new Image();
        var canvas = document.getElementById(ImgRotator.canvasID);
        image.onload = function() {
            ImgRotator.image = image;
            canvas.height = canvas.width = Math.sqrt( image.width* image.width+image.height*image.height);
            window.setInterval(ImgRotator.keepRotating,ImgRotator.intervalMS);
            //theApp.keepRotating();
        };
        image.src = ImgRotator.src;   
    },
    keepRotating:function(){
        ImgRotator.angle+=ImgRotator.jump;
        var canvas = document.getElementById(ImgRotator.canvasID);
        var ctx = canvas.getContext("2d");
        ctx.save();
        ctx.clearRect(0,0,canvas.width,canvas.height);
        ctx.translate(canvas.width/2,canvas.height/2);
        ctx.rotate(ImgRotator.angle*Math.PI/180); 
        ctx.drawImage(ImgRotator.image, -ImgRotator.image.width/2,-ImgRotator.image.height/2);
        ctx.restore();
    }
}

usage

ImgRotator.start_action("canva",
            "",
            500,15
            );

HTML

<canvas id="canva" width="350" height="350" style="border:solid thin black;"></canvas>
Dr.Sai
  • 303
  • 4
  • 5
1

@Steve Farthing's answer is absolutely right.

But if you rotate more than 4 times then the image will get cropped from both the side. For that you need to do like this.

$("#clockwise").click(function(){ 
    angleInDegrees+=90 % 360;
    drawRotated(angleInDegrees);
    if(angleInDegrees == 360){  // add this lines
        angleInDegrees = 0
    }
});

Then you will get the desired result. Thanks. Hope this will help someone :)

Vivek Saha
  • 91
  • 1
  • 9
1

To rotate and fill (stretch to fit), try this fiddle which uses the corner of the canvas for a start point instead of the center and can rotate 90, 180 and 270 degrees.

Note: the fiddle is stretching a 200 * 300 image to fit a 600 * 650 canvas

const canvas = document.getElementById('canvas');
const image = document.getElementById('source');

const Rotation = {
  None: 0,
  Right: 90,
  Flip: 180,
  Left: 270
};

const noRotation = () => Rotate(Rotation.None);
const rotateRight = () => Rotate(Rotation.Right);
const rotateLeft = () => Rotate(Rotation.Left);
const flip = () => Rotate(Rotation.Flip);
const Rotate = (orientation) => rotateImage(canvas, image, orientation);

function rotateImage(canvas, image, rotation){
    const rotate =  correctRotation(rotation);
  const context = canvas.getContext('2d');
  
  let offsetX = 0;
  let offsetY = 0;

    let displacementX = canvas.width;
    let displacementY = canvas.height;

  if (rotate === Rotation.Left || rotate === Rotation.Right){
    displacementX = canvas.height;
    displacementY = canvas.width;
  }

  switch(rotate) {
    case Rotation.Flip:
      offsetX = canvas.width;
      offsetY = canvas.height;
      break;

  case Rotation.Left:
      offsetY = canvas.height;
      break;

  case Rotation.Right:
        offsetX = canvas.width;
      break;
  }
  
  context.setTransform(1, 0, 0, 1, offsetX, offsetY);
  context.rotate(rotate * Math.PI / 180);
  context.drawImage(image, 0,0, displacementX, displacementY )
}

function correctRotation(rotation) {
    return !rotation ? Rotation.None    
    : rotation < Rotation.Flip ? Rotation.Right
    : rotation < Rotation.Left ? Rotation.Flip 
    : Rotation.Left;
}

image.addEventListener('load', e => noRotation() );

https://jsfiddle.net/0unyh7ck/8/

Antony Booth
  • 413
  • 4
  • 5
0

Why not do it for the entire page. At page load detect all images and continuously rotate all of them.

 var RotationCollection = {
    rotators: [],
    start_action: function (showBorders, isoverlap) {
        try {
            var canvasTemplate = '<canvas id="_ID_" width="350" height="350"  ></canvas>';

            var ja = 5;
            $.each($("img"), function (index, val) {
                var newID = "can_" + index;
                var can = canvasTemplate.replace("_ID_", newID);

                if (showBorders == true) $(can).insertAfter($(val)).css({ "border": "solid thin black", "box-shadow": "5px 5px 10px 2px black", "border-raduis": "15px" });
                else $(can).insertAfter($(val));
                $(val).remove();

                var curRot = new RotationClass(newID, $(val).attr('src'), ja  * ((0 == index % 2) ? -1 : 1), isoverlap);
                RotationCollection.rotators[index] = curRot;
                ja += 5;
                //return false;
            });
            window.setInterval(function () {
                $.each(RotationCollection.rotators, function (index, value) {
                    value.drawRotatedImage();
                })
            }, 500);
        }
        catch (err) {
            console.log(err.message);
        }
    }
};
function RotationClass(canvasID, imgSrc, jumgAngle, overlap) {
    var self = this;
    self.overlap = overlap;
    self.angle = parseInt(45);
    self.image = {};
    self.src = imgSrc;
    self.canvasID = canvasID;
    self.jump = parseInt(jumgAngle);
    self.start_action = function () {
        var image = new Image();
        var canvas = document.getElementById(self.canvasID);
        image.onload = function () {
            self.image = image;
            canvas.height = canvas.width = Math.sqrt(image.width * image.width + image.height * image.height);
            self.drawRotatedImage(self);
        };
        image.src = self.src;
    }
    self.start_action();
    this.drawRotatedImage = function () {
        var self = this;
        self.angle += self.jump;
        var canvas = document.getElementById(self.canvasID);
        var ctx = canvas.getContext("2d");
        ctx.save();
        if (self.overlap) ctx.clearRect(0, 0, canvas.width, canvas.height);
        ctx.translate(canvas.width / 2, canvas.height / 2);
        ctx.rotate(self.angle * Math.PI / 180);
        ctx.drawImage(self.image, -self.image.width / 2, -self.image.height / 2);
        ctx.restore();
    }
}
var theApp = {
    start_Action: function () {
        RotationCollection.start_action(true, true);
    }
};
$(document).ready(theApp.start_Action);

Please check out for theApp.start_Action where all action begins The HTML can be as follows:

 <p>
    Deepika Padukone.<br />
    <img alt="deepika" src="" />
</p>

<p>
    Priyanka Chopra.<br />
    <img alt="Priyanka" src="" />
</p>

Some options to overlap rotations, borders are also added

Dr.Sai
  • 303
  • 4
  • 5