„Wenn ein Arbeiter seine Arbeit gut machen will, muss er zuerst seine Werkzeuge schärfen.“ – Konfuzius, „Die Gespräche des Konfuzius. Lu Linggong“
Titelseite > Programmierung > Erstellen eines Reaktionshakens zum Drehen von Bildern in einem beliebigen Winkel

Erstellen eines Reaktionshakens zum Drehen von Bildern in einem beliebigen Winkel

Veröffentlicht am 13.09.2024
Durchsuche:433

Creating a React Hook for Rotating Images at Any Angle

Bei der Webentwicklung müssen Sie möglicherweise ein Bild drehen, was in CSS einfach zu bewerkstelligen ist. Einfacher Code wie diese Transformation: rotation(90deg);. Aber was ist, wenn wir es in JS machen wollen?

TLDR

Zeichnet das Bild auf die Leinwand in der Browserumgebung und dreht es. Aber vorher müssen wir noch etwas rechnen, um das ursprüngliche Bildseitenverhältnis beizubehalten.

Kern

Angenommen, wir haben das Bild geladen, kann die Berechnung des gedrehten Bildes wie folgt erfolgen:

const { PI, sin, cos, abs } = Math;
const angle = (degree * PI) / 180;
const sinAngle = sin(angle);
const cosAngle = cos(angle);

const rotatedWidth = abs(imageWidth * cosAngle)   abs(imageHeight * sinAngle);

const rotatedHeight = abs(imageWidth * sinAngle)   abs(imageHeight * cosAngle);

Und als nächstes verwenden wir einige der Canvas-APIs, um die eigentliche Drehung durchzuführen:

const canvas = document.createElement('canvas');

const { width: canvasWidth, height: canvasHeight } = canvas;
const canvasCtx2D = canvas.getContext('2d');

canvasCtx2D.clearRect(0, 0, canvasWidth, canvasHeight);
canvasCtx2D.translate(canvasWidth / 2, canvasHeight / 2);
canvasCtx2D.rotate(angle);

canvasCtx2D.drawImage(
  image,
  -imageWidth / 2,
  -imageHeight / 2,
  imageWidth,
  imageHeight,
);

return canvas.toDataURL('image/png');

Einpacken

Wenn der Kerncode vorhanden ist, können wir einige Optimierungen vornehmen und dedizierte React-Hooks schreiben, um ihn zu verwenden:

import { useEffect, useRef, useState } from 'react';

type RotatedImage = {
  src: string;
  width: number;
  height: number;
} | null;

let canvas: HTMLCanvasElement | null = null;
let canvasCtx2D: CanvasRenderingContext2D | null = null;

const getRotatedImage = (
  image: HTMLImageElement | null,
  rotation: number,
): RotatedImage => {
  canvas ??= document.createElement('canvas');
  canvasCtx2D ??= canvas.getContext('2d');

  if (!image || !canvasCtx2D) return null;

  const { width: imageWidth, height: imageHeight, currentSrc } = image;
  const degree = rotation % 360;
  if (!degree) {
    return {
      src: currentSrc,
      width: imageWidth,
      height: imageHeight,
    };
  }

  const { PI, sin, cos, abs } = Math;
  const angle = (degree * PI) / 180;
  const sinAngle = sin(angle);
  const cosAngle = cos(angle);

  canvas.width = abs(imageWidth * cosAngle)   abs(imageHeight * sinAngle);
  canvas.height = abs(imageWidth * sinAngle)   abs(imageHeight * cosAngle);

  // The width and height of the canvas will be automatically rounded.
  const { width: canvasWidth, height: canvasHeight } = canvas;

  canvasCtx2D.clearRect(0, 0, canvasWidth, canvasHeight);
  canvasCtx2D.translate(canvasWidth / 2, canvasHeight / 2);
  canvasCtx2D.rotate(angle);

  canvasCtx2D.drawImage(
    image,
    -imageWidth / 2,
    -imageHeight / 2,
    imageWidth,
    imageHeight,
  );

  const src = canvas.toDataURL('image/png');
  canvas.width = 0;
  canvas.height = 0;

  return {
    src,
    width: canvasWidth,
    height: canvasHeight,
  };
};

const useRotateImage = (imageSrc: string, rotation?: number): RotatedImage => {
  const imageEle = useRef(null);
  const [rotatedImage, setRotatedImage] = useState(null);

  useEffect(() => {
    if (typeof rotation === 'number') {
      let currImage = imageEle.current;

      if (currImage?.currentSrc !== imageSrc) {
        currImage = new Image();
        imageEle.current = currImage;

        currImage.src = imageSrc;
      }

      currImage.decode().then(
        () => setRotatedImage(getRotatedImage(currImage, rotation)),
        () => setRotatedImage(null),
      );
    }
  }, [imageSrc, rotation]);

  return rotatedImage;
};

export default useRotateImage;

Hier verwende ich dasselbe Canvas-Element wieder, um wiederholte Erstellungen zu reduzieren. Zweitens sollte beachtet werden, dass ich seine Breite und Höhe nach jeder Drehung auf 0 setze, um die Speichernutzung zu reduzieren. Übrigens habe ich auch die Reinigung der Leinwand durchgeführt. Dies liegt daran, dass in der HTML-Spezifikation, wenn Sie die Breite und Höhe der Leinwand ändern (unabhängig davon, ob sie dieselben wie zuvor sind), die Leinwand gelöscht wird, was mit „canvasCtx2D.clearRect(0, 0, canvasWidth, canvasHeight)“ identisch ist wird von modernen Browsern unterstützt.

In useRotateImage behalte ich einen Verweis auf das Bildelement und lege den gedrehten Bildstatus nach image.decode() fest, der aufgelöst wird, nachdem die Bilddaten bereit sind.

Unten ist ein Online-Anwendungsfall:


Wenn Sie dies hilfreich fanden, denken Sie bitte darüber nach, meinen Newsletter zu abonnieren für weitere nützliche Artikel und Tools zur Webentwicklung. Danke fürs Lesen!

Freigabeerklärung Dieser Artikel ist abgedruckt unter: https://dev.to/zacharylee/creating-a-react-hook-for-rotating-images-at-any-angle-4nnb?1 Bei Verstößen wenden Sie sich bitte an Study_golang@163 .com, um es zu löschen
Neuestes Tutorial Mehr>

Haftungsausschluss: Alle bereitgestellten Ressourcen stammen teilweise aus dem Internet. Wenn eine Verletzung Ihres Urheberrechts oder anderer Rechte und Interessen vorliegt, erläutern Sie bitte die detaillierten Gründe und legen Sie einen Nachweis des Urheberrechts oder Ihrer Rechte und Interessen vor und senden Sie ihn dann an die E-Mail-Adresse: [email protected] Wir werden die Angelegenheit so schnell wie möglich für Sie erledigen.

Copyright© 2022 湘ICP备2022001581号-3