HTML5 Canvas旋转图像

时间:2013-07-01 19:15:51

标签: html image canvas rotation html5-canvas

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/

将图像向右或向左旋转90°时出现问题。

我在画布上使用了一个图像,同一个屏幕将有几个画布等于示例的画布,但是我把它尽可能地靠近项目。

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

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

11 个答案:

答案 0 :(得分:116)

你可以使用canvas'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>

答案 1 :(得分:30)

最快的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);

答案 2 :(得分:22)

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

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/

答案 3 :(得分:4)

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

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

答案 4 :(得分:3)

正如@markE在answer

中提到的那样
  

替代方案是翻译和翻译绘图后不旋转

它比上下文保存和恢复快得多。

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

答案 5 :(得分: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>

答案 6 :(得分:2)

@Steve Farthing的回答绝对正确。

但是,如果旋转超过4次,则图像将从两侧被裁切。 为此,您需要这样做。

git fetch + git merge

然后您将获得所需的结果。谢谢。希望这会帮助某人:)

答案 7 :(得分: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);

请查看所有操作开始的App.start_Action HTML可以如下:

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

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

还添加了一些重叠旋转,边框的选项

答案 8 :(得分:1)

这是全度图像旋转代码。我建议您在jsfiddle中检查以下示例应用程序。

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

enter image description here

此示例应用程序的处理流程为

  1. 加载图像,计算boundaryRad
  2. 创建临时画布
  3. 将画布上下文原点移动到投影矩形的联合位置
  4. 以输入度数量旋转画布上下文
  5. 使用canvas.toDataURL方法创建图像斑点
  6. 使用图像斑点,创建新的Image元素并渲染

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

答案 9 :(得分:1)

我构建了一个图像上传器,可以在背景中调整图像大小,还允许用户将图像向左或向右旋转 90 度。这不是仅用几行代码就能解决的问题,因此您需要查看 jsfiddle https://jsfiddle.net/alienbush/z02jatnr/6/。我还在下面提供了完整的代码。

主要绘图功能如下所示:

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

答案 10 :(得分:0)

要旋转和填充(拉伸以适合),试试这个小提琴,它使用画布的角作为起点而不是中心,可以旋转 90、180 和 270 度。

注意:小提琴正在拉伸 200 * 300 图像以适合 600 * 650 画布

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/