@litecanvas/utils
Version:
Utilities to help build litecanvas games
187 lines (156 loc) • 3.6 kB
JavaScript
import { Vector, vec } from "../vector/index.js"
import "litecanvas"
export const ANCHOR_CENTER = /** @__PURE__ */ vec(0.5, 0.5)
export const ANCHOR_TOP_LEFT = /** @__PURE__ */ vec(0, 0)
export const ANCHOR_TOP_RIGHT = /** @__PURE__ */ vec(1, 0)
export const ANCHOR_BOT_LEFT = /** @__PURE__ */ vec(0, 1)
export const ANCHOR_BOT_RIGHT = /** @__PURE__ */ vec(1, 1)
export class Actor {
/** @type {Image|HTMLCanvasElement|OffscreenCanvas} */
sprite
/** @type {Vector} The actor position */
pos
/** @type {Vector} The actor anchor (origin) */
_o
/** @type {Vector} The actor scale */
_s
/** @type {boolean} */
flipX = false
/** @type {boolean} */
flipY = false
/** @type {number} The actor angle (in degrees) */
angle = 0
/** @type {number} The actor opacity */
opacity = 1
/** @type {boolean} If `true` the actor will not be drawn. */
hidden = false
/**
* @param {Image|HTMLCanvasElement|OffscreenCanvas} sprite
* @param {Vector} position
* @param {Vector} anchor
*/
constructor(sprite, position, anchor = ANCHOR_TOP_LEFT) {
this.sprite = sprite
this.pos = position || vec(0)
this._o = vec(anchor) // clone the anchor vector
this._s = vec(1, 1)
}
/**
* @param {number}
*/
set x(value) {
this.pos.x = value
}
/**
* @returns {number}
*/
get x() {
return this.pos.x
}
/**
* @param {number}
*/
set y(value) {
this.pos.y = value
}
/**
* @returns {number}
*/
get y() {
return this.pos.y
}
/**
* @param {Vector}
*/
set anchor(vec) {
this._o.x = vec.x
this._o.y = vec.y
}
/**
* @returns {Vector}
*/
get anchor() {
return this._o
}
/**
* @returns {number}
*/
get width() {
return this.sprite.width * this._s.x
}
/**
* @returns {number}
*/
get height() {
return this.sprite.height * this._s.y
}
/**
* @retuns {Vector}
*/
get scale() {
return this._s
}
/**
* Sets the actor scale
*
* @param {number} x
* @param {number} [y]
*/
scaleTo(x, y = x) {
this._s.x = x
this._s.y = y
}
/**
* Multiplies the actor scale
*
* @param {number} x
* @param {number} [y]
*/
scaleBy(x, y = x) {
this._s.x *= x
this._s.y *= y
}
/**
* @returns {number[]}
*/
getBounds(scaled = true) {
const w = this.sprite.width * (scaled ? this._s.x : 1)
const h = this.sprite.height * (scaled ? this._s.y : 1)
const x = this.pos.x - w * this.anchor.x
const y = this.pos.y - h * this.anchor.y
return [x, y, w, h]
}
/**
* Draw the actor
*
* @param {LitecanvasInstance} [litecanvas]
*/
draw(litecanvas = globalThis, saveContext = true) {
if (this.hidden || this.opacity <= 0) return
if (saveContext) litecanvas.push()
this.transform(litecanvas)
this.drawImage(litecanvas)
if (saveContext) litecanvas.pop()
}
/**
* @param {LitecanvasInstance} litecanvas
*/
transform(litecanvas) {
litecanvas.translate(this.pos.x, this.pos.y)
litecanvas.rotate(litecanvas.deg2rad(this.angle))
litecanvas.scale(
(this.flipX ? -1 : 1) * this._s.x,
(this.flipY ? -1 : 1) * this._s.y
)
}
/**
* @param {LitecanvasInstance} litecanvas
*/
drawImage(litecanvas, alpha = true) {
const anchor = this.anchor
const x = -this.sprite.width * (this.flipX ? 1 - anchor.x : anchor.x)
const y = -this.sprite.height * (this.flipY ? 1 - anchor.y : anchor.y)
if (alpha) litecanvas.alpha(this.opacity)
litecanvas.image(x, y, this.sprite)
}
}