2013-07-01 299 views
45
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/HTML5画布旋转图像

我有问题的图像旋转90°到右侧或左侧。

我在画布上使用了一个图像,同一个画面将有几个画布与示例的画布相同,但是我尽量靠近该项目。

我问,当我点击“向左旋转”和“向右旋转”时,如何将图像向左或向右旋转90°?

我在互联网上尝试了几个代码,但都没有工作。

回答

96

您可以使用画布context.translate & context.rotate做图像旋转

enter image description here

这里有一个函数来绘制由指定度旋转图像:

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(); 
} 

这里是代码和小提琴: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> 
+0

由于坊间...完美的作品! –

+0

那么putImageData()呢?它似乎没有工作 – TheRealChx101

+0

那么,putImageData *将工作*,但做同样的事情是一个更昂贵的操作 - 所以putImageData不会在这里调用! ;-) – markE

19

另一种解决方案适用于方形图像。这是一个适用于任何维度图像的解决方案。画布将始终适合图像,而不是可能导致图像部分被裁剪掉的其他解决方案。

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/

+0

工作出色,即使源代码是base64。 – Moussawi7

2

这是我做的事

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(); 
    } 
} 

使用

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

HTML

<canvas id="canva" width="350" height="350" style="border:solid thin black;"></canvas> 
1

为什么不为整个页面做。在页面加载时检测所有图像并连续旋转所有图像。

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); 

,详情请参阅theApp.start_Action所有行动开始 的HTML可以如下:

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

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

一些选项重叠旋转,边框也加入

14

最快的2D背景图像旋转方法

通用图像旋转,位置和比例。

// 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); 
} 

如果你想控制旋转点使用下一个功能

// 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); 
} 

要重置2D语境转换

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

因此,以图像旋转到左边(逆时针方向)的90度

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

因此以旋转图像向右(顺时针)90度

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

实施例绘制500个图像翻译旋转的缩放

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);

+3

这是一个非常酷的演示! – daviestar

3

这是绘制一个旋转的最简单的代码和缩放图像:

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(); 
} 
1

作为@markE提他answer

另一种方法是untranslate & unrotate绘制

后它比方面快得多保存和恢复。

这里是一个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);