89 votes

HTML5 Canvas Rotation d'une image

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/

J'ai un problème pour faire pivoter une image de 90° vers la droite ou vers la gauche.

J'utilise une image sur la toile, le même écran aura plusieurs toiles égales à celle de l'exemple, mais je l'ai laissé aussi proche que possible du projet.

Je demande comment faire pivoter l'image de 90 ° vers la gauche ou la droite lorsque je clique sur "Rotation à gauche" et "Rotation à droite" ?

J'ai essayé plusieurs codes sur internet mais aucun n'a fonctionné.

4voto

casamia Points 176

C'est un code de rotation d'image à plein degré. Je vous recommande de vérifier l'exemple d'application ci-dessous dans le jsfiddle.

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

enter image description here

Le flux de processus de cet exemple d'application est le suivant

  1. charger l'image, calculer boundaryRad
  2. créer une toile temporaire
  3. déplace l'origine du contexte du canevas à la position conjointe du rectangle projeté
  4. fait pivoter le contexte du canevas avec la quantité de degré d'entrée
  5. utiliser la méthode canvas.toDataURL pour créer un blob d'image
  6. en utilisant le blob d'image, créez un nouvel élément Image et effectuez un rendu

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

2voto

Dr.Sai Points 267

Voici ce que j'ai fait

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

utilisation

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

HTML

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

2voto

McGrew Points 23

J'ai construit un chargeur d'images qui redimensionne l'image en arrière-plan et permet également à l'utilisateur de faire pivoter l'image de 90 degrés à gauche ou à droite. Ce n'est pas quelque chose qui peut être résolu avec seulement quelques lignes de code, donc vous voudrez vérifier le jsfiddle. https://jsfiddle.net/alienbush/z02jatnr/6/ . J'ai également inclus le code complet ci-dessous.

La fonction principale de dessin ressemble à ceci :

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

Voici le code complet :

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>

1voto

Dr.Sai Points 267

Pourquoi ne pas le faire pour la page entière. Au chargement de la page, détectez toutes les images et faites-les tourner en continu.

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

Veuillez consulter la sectionApp.start_Action où toutes les actions commencent. Le HTML peut être le suivant :

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

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

Certaines options de chevauchement des rotations et des bordures ont également été ajoutées.

1voto

Vivek Saha Points 81

La réponse de @Steve Farthing est tout à fait juste.

Mais si vous effectuez plus de 4 rotations, l'image sera rognée des deux côtés. Pour cela, vous devez faire comme ceci.

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

Vous obtiendrez alors le résultat souhaité. Merci. J'espère que cela aidera quelqu'un :)

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X