UNPKG

@litecanvas/utils

Version:

Utilities to help build litecanvas games

259 lines (225 loc) 5.34 kB
import "litecanvas" const HALF_PI = Math.PI / 2 /** * @param {any} object * @param {string} prop * @param {number|number} toValue * @param {number} [duration] * @param {(n: number) => number} [easing] * @returns {this} */ export const tween = (object, prop, toValue, duration = 1, easing = LINEAR) => { return new TweenController(object, prop, toValue, duration, easing) } export const LINEAR = (n) => n export const EASE_IN = (n) => n * n export const EASE_OUT = (n) => -n * (n - 2.0) export const EASE_IN_OUT = (n) => { if (n < 0.5) { return 2.0 * n * n } return -2.0 * n * n + 4.0 * n - 1.0 } export const BACK_IN = (n) => n * n * n - n * Math.sin(n * Math.PI) export const BACK_OUT = (n) => { let a = 1.0 - n return 1.0 - (a * a * a - a * Math.sin(a * Math.PI)) } export const BACK_IN_OUT = (n) => { if (n < 0.5) { let a = 2.0 * n return 0.5 * (a * a * a - a * Math.sin(a * Math.PI)) } let a = 1.0 - (2.0 * n - 1.0) return 0.5 * (1.0 - (a * a * a - a * Math.sin(n * Math.PI))) + 0.5 } export const ELASTIC_IN = (n) => { return Math.sin(13 * HALF_PI * n) * Math.pow(2, 10 * (n - 1)) } export const ELASTIC_OUT = (n) => { return Math.sin(-13 * HALF_PI * (n + 1)) * Math.pow(2, -10 * n) + 1 } export const ELASTIC_IN_OUT = (n) => { if (n < 0.5) { let a = Math.sin(13 * HALF_PI * (2 * n)) let b = Math.pow(2, 10 * (2 * n - 1)) return 0.5 * a * b } let a = Math.sin(-13 * HALF_PI * (2 * n - 1 + 1)) let b = Math.pow(2, -10 * (2 * n - 1)) return 0.5 * (a * b + 2) } export const BOUNCE_IN = (n) => 1 - BOUNCE_OUT(1 - n) export const BOUNCE_OUT = (n) => { if (n < 4.0 / 11.0) { return (121.0 * n * n) / 16.0 } else if (n < 8.0 / 11.0) { return (363.0 / 40.0) * n * n - (99.0 / 10.0) * n + 17.0 / 5.0 } else if (n < 9.0 / 10.0) { return (4356.0 / 361.0) * n * n - (35442.0 / 1805.0) * n + 16061.0 / 1805.0 } return (54.0 / 5.0) * n * n - (513.0 / 25.0) * n + 268.0 / 25.0 } export const BOUNCE_IN_OUT = (n) => { if (n < 0.5) { return 0.5 * BOUNCE_IN(n * 2) } return 0.5 * BOUNCE_OUT(n * 2 - 1) + 0.5 } class TweenController { /** @type {boolean} */ running = false /** @type {*} */ _o /** @type {string} */ _p /** @type {number|number} */ _x /** @type {number} */ _d /** @type {number} */ _w /** @type {(x: number) => number} */ _e /** @type {boolean} */ _rel /** @type {Function[]} */ _cb = [] /** @type {number} */ _t = 0 /** @type {Function} */ _u = 0 /** @type {TweenController} */ _ch = this /** @type {TweenController} */ _cu = this /** @type {LitecanvasInstance} */ _lc /** * @param {*} object * @param {string} prop * @param {number|number} toValue * @param {number} duration * @param {(x: number) => number} easing */ constructor(object, prop, toValue, duration, easing) { this._o = object this._p = prop this._x = toValue this._d = duration this._e = easing this._w = 0 } /** * @param {LitecanvasInstance} [engine] * @returns {this} */ start(engine) { if (this.running) { return this } this._cu.stop(false) this._ch = this._cu = this this.running = true const fromValue = this._o[this._p] || 0 const toValue = this._rel ? fromValue + this._x : this._x this._lc = this._lc || engine || globalThis this._u = this._lc.listen("update", (dt) => { if (this._t <= this._w) { this._t += dt return } const t = this._t - this._w this._o[this._p] = this._lc.lerp(fromValue, toValue, this._e(t / this._d)) this._t += dt if (t >= this._d) { this._o[this._p] = toValue this.stop() } }) return this } /** * @param {boolean} completed if `false` don't call the `onEnd()` registered callbacks. * @returns {this} */ stop(completed = true) { if (!this._u) return this this.running = false this._u() this._t = 0 if (completed) { for (const callback of this._cb) { callback(this._o) } } return this } /** * @param {LitecanvasInstance} [engine] * @returns {this} */ restart(engine = null, completed = false) { return this.stop(completed).restart(engine) } /** * @param {Function} callback * @returns {this} */ onEnd(callback) { this._cb.push(callback) return this } /** * @param {TweenController} another * @returns {this} */ chain(another) { this._ch.onEnd(() => { this._cu = another.start(this._lc) }) this._ch = another return this } /** * @param {boolean} [flag=true] * @returns {this} */ reset() { this._cb.length = 0 return this.stop() } /** * @param {boolean} [flag=true] * @returns {this} */ relative(flag = true) { this._rel = flag return this } /** * @param {number} value * @returns {this} */ delay(value) { this._w = value return this } /** * Returns the current tween of the chain. * * @returns {this} */ get current() { return this._cu } /** * @returns {number} the current progress (0..1) */ get progress() { if (this.running && this._t > this._w) { return (this._t - this._w) / this._d } return 0 } }