From 464ad28fa4cdce2c3f73f80e076177282a35b293 Mon Sep 17 00:00:00 2001 From: Jordan Irwin Date: Sun, 3 Dec 2023 20:00:01 -0800 Subject: [PATCH] Add function to rotate sprite images --- srcjs/stendhal.css | 4 +++ srcjs/stendhal.html | 3 ++ srcjs/stendhal/data/SpriteStore.ts | 55 ++++++++++++++++++++++++++++++ 3 files changed, 62 insertions(+) diff --git a/srcjs/stendhal.css b/srcjs/stendhal.css index e3e69bbc7d7..eebd40cbbb6 100644 --- a/srcjs/stendhal.css +++ b/srcjs/stendhal.css @@ -586,6 +586,10 @@ div.logcolR { margin: 2px; } +canvas#drawing-stage { + display: none; +} + @font-face { font-family: Amaranth; src: url("/data/font/Amaranth-Regular.ttf"); diff --git a/srcjs/stendhal.html b/srcjs/stendhal.html index d94a861ead7..c8cf877e402 100644 --- a/srcjs/stendhal.html +++ b/srcjs/stendhal.html @@ -398,6 +398,9 @@

+ + + diff --git a/srcjs/stendhal/data/SpriteStore.ts b/srcjs/stendhal/data/SpriteStore.ts index af9cfd017d4..dc04b2fc0e2 100644 --- a/srcjs/stendhal/data/SpriteStore.ts +++ b/srcjs/stendhal/data/SpriteStore.ts @@ -125,6 +125,61 @@ export class SpriteStore { }); } + /** + * Rotates an image. + * + * @param img + * Image to be rotated. + * @param angle + * Angle of rotation. + */ + private rotate(img: HTMLImageElement, angle: number) { + const canvas = document.getElementById("drawing-stage")!; + const ctx = canvas.getContext("2d")!; + // make sure working with blank canvas + ctx.clearRect(0, 0, canvas.width, canvas.height); + canvas.width = img.width; + canvas.height = img.height; + + ctx.translate(canvas.width / 2, canvas.height / 2); + ctx.rotate(angle * Math.PI / 180); + ctx.translate(-canvas.width / 2, -canvas.height / 2); + ctx.drawImage(img, 0, 0); + + img.src = canvas.toDataURL("image/png"); + } + + /** + * Retrieves a rotated image. + * + * @param filename + * Path to target image file. + * @param angle + * Angle of rotation. + * @return + * HTMLImageElement. + */ + getRotated(filename: string, angle: number): any { + if (angle == 0) { + return this.get(filename); + } + const id = filename + "-rot" + angle; + if (this.images[id]) { + return this.images[id]; + } + const img = this.get(filename).cloneNode(); + if (img.complete) { + this.rotate(img, angle); + } else { + img.onload = () => { + this.rotate(img, angle); + img.onload = undefined; + } + } + this.images[id] = img; + return img; + } + /** * Used when we only want an image if it was previously cached. *