HTML5 Canvas Bild drehen

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

Ich habe ein Problem beim Drehen eines Bildes um 90 ° nach rechts oder links.

Ich verwende ein Bild auf der Leinwand. Auf demselben Bildschirm befinden sich mehrere Leinwandbilder, die denen des Beispiels entsprechen. Ich habe es jedoch so nah wie möglich am Projekt belassen.

Ich frage, wie kann ich das Bild um 90 ° nach links oder rechts drehen, wenn ich auf "Nach links drehen" und "Nach rechts drehen" klicke?

Ich habe mehrere Codes im Internet ausprobiert, aber keiner hat funktioniert.

Max William Vitorino
quelle

Antworten:

152

Sie können canvas 'context.translate & context.rotate verwenden, um Ihr Bild zu drehen

Geben Sie hier die Bildbeschreibung ein

Hier ist eine Funktion zum Zeichnen eines Bildes, das um die angegebenen Grade gedreht wird:

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

Hier ist Code und eine Geige: 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>
markE
quelle
Was ist also mit putImageData ()? Es scheint nicht zu funktionieren
TheRealChx101
Nun, putImageData würde funktionieren , aber es ist viel teurer, dasselbe zu tun - also wird putImageData hier nicht benötigt! ;-)
markE
15
Dieser Code funktioniert perfekt für quadratische Bilder. Bei Rechteckbildern wird das Bild falsch zugeschnitten (siehe jsfiddle.net/6ZsCz/803 ). Wenn Sie ctx.drawImage ändern (image, -image.width / 2, -image.width / 2); Um ctx.drawImage (image, -image.width / 2, -image.height / 2) zu sein, funktioniert es für Bilder aller Größen.
Steve Farthing
8
in Ordnung. Hier ist ein Beispiel, bei dem die Leinwand zum Bild passt und Teile des Bildes nicht verloren gehen: jsfiddle.net/6ZsCz/804 .
Steve Farthing
2
@markE: Zum 3. Mal in diesem Monat haben Sie mir geholfen :-D
Dr. Gianluigi Zane Zanettini
46

Schnellste Methode zur Drehung von 2D-Kontextbildern

Eine allgemeine Bilddrehung, Position und Skalierung.

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

Wenn Sie den Drehpunkt steuern möchten, verwenden Sie die nächste Funktion

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

Zurücksetzen der 2D-Kontexttransformation

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

So drehen Sie das Bild nach links (gegen den Uhrzeigersinn) um 90 Grad

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

So drehen Sie das Bild um 90 Grad nach rechts (im Uhrzeigersinn)

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

Beispiel zeichnen 500 Bilder übersetzt gedreht skaliert

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
quelle
1
Danke - ich bin glücklich - Wie schnell ist mein Computer!?!
T4NK3R
@ Blindman67 Danke für die tolle Lösung! Würden Sie bitte in den folgenden Zeilen erklären, was getan wird? Ich lutsche am mathematischen Teil: / - 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
1
@zumek spr.xrund yrrepräsentieren die gerenderte Koordinate des Sprites. Die ersten beiden Zeilen geben die Breite / Höhe des Bildes (Leinwand) mit Rändern an, mit denen sich das Sprite (von allen Seiten) von der Leinwand entfernen kann. Bei den nächsten beiden wird die Position um die Randgröße nach oben und links versetzt. Die Position der% -Zyklen (wie sich Asteroiden nach links bewegen, erscheint rechts). Die Ränder stellen sicher, dass das Sprite nicht verschwindet und dann auf der gegenüberliegenden Seite wieder angezeigt wird.
Blindman67
Das wird unterschätzt.
Andrew Shi
Gibt es eine Möglichkeit, das drawImageElement zu drehen , ohne es canvasselbst zu drehen ?
Oldboy
27

Die andere Lösung eignet sich hervorragend für quadratische Bilder. Hier ist eine Lösung, die für ein Bild jeder Dimension funktioniert. Die Leinwand passt immer zum Bild und nicht zur anderen Lösung. Dies kann dazu führen, dass Teile des Bildes abgeschnitten werden.

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
quelle
Gibt es eine Möglichkeit zu drehen, drawImageohne die zu drehen canvas?
Oldboy
7

Dies ist der einfachste Code zum Zeichnen eines gedrehten und skalierten Bildes:

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();
}
CpnCrunch
quelle
Gibt es eine Möglichkeit, das drawImageElement zu drehen , ohne es canvasselbst zu drehen ? ist save restorewirklich der einzige Weg?
Oldboy
@oldboy Leinwand wird hier nicht gedreht. Die Drehung wird auf den Kontext angewendet, der innerhalb der Leinwand funktioniert
aleha
@aleha also, gibt es eine Möglichkeit, das drawImageElement direkt zu drehen, ohne das ctxselbst zu drehen ?
Oldboy
@oldboy Ich denke du hast das Konzept falsch verstanden. ctxs.drawImage zeichnet nur.
Aleha
@aleha ja, ich weiß, dass drawImagenur zeichnet. Es zeichnet ein Bild auf die Leinwand oder den Kontext. Ich frage mich, ob es möglich ist, das Bild selbst direkt zu drehen, anstatt die Leinwand oder den Kontext zu drehen ...?
Oldboy
5

Wie @markE in seiner Antwort erwähnt

Die Alternative besteht darin, die Übersetzung nach dem Zeichnen aufzuheben und aufzuheben

Es ist viel schneller als das Speichern und Wiederherstellen von Kontexten.

Hier ist ein Beispiel

// 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
quelle
Gibt es eine Möglichkeit, das drawImageElement zu drehen , ohne es canvasselbst zu drehen ?
Oldboy
Drehen Sie den drawImagedirekt *
oldboy
3

Dies ist ein Bildrotationscode mit vollem Grad. Ich empfehle Ihnen, die folgende Beispiel-App in der jsfiddle zu überprüfen.

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

Geben Sie hier die Bildbeschreibung ein

Der Prozessablauf dieser Beispiel-App ist

  1. Bild laden, BoundaryRad ​​berechnen
  2. temporäre Leinwand erstellen
  3. Verschieben Sie den Canvas-Kontextursprung an die gemeinsame Position des projizierten Rechtecks
  4. Drehen Sie den Canvas-Kontext mit dem Eingabegrad
  5. Verwenden Sie die canvas.toDataURL-Methode, um einen Bild-Blob zu erstellen
  6. Erstellen Sie mit dem Bild-Blob ein neues Bildelement und rendern Sie es

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
quelle
Du bist der beste! Ich weiß nicht, wie viel Zeit Sie dafür aufgewendet haben. Ich weiß das wirklich zu schätzen.
Aleskei Sacharow
Wie dreht man ein drawImageElement, ohne die gesamte Leinwand selbst zu drehen ?
Oldboy
2

Hier ist etwas, was ich getan habe

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

Verwendung

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

HTML

<canvas id="canva" width="350" height="350" style="border:solid thin black;"></canvas>
Dr.Sai
quelle
2

Die Antwort von @Steve Farthing ist absolut richtig.

Wenn Sie jedoch mehr als viermal drehen, wird das Bild von beiden Seiten zugeschnitten. Dafür müssen Sie so vorgehen.

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

Dann erhalten Sie das gewünschte Ergebnis. Vielen Dank. Hoffe das hilft jemandem :)

Vivek Saha
quelle
1

Warum nicht für die gesamte Seite? Erkennen Sie beim Laden der Seite alle Bilder und drehen Sie sie kontinuierlich.

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

Bitte überprüfen Sie die App.start_Action, in der alle Aktionen beginnen. Der HTML-Code kann wie folgt lauten:

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

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

Einige Optionen zum Überlappen von Rotationen und Rahmen werden ebenfalls hinzugefügt

Dr.Sai
quelle