UNPKG

matrix-react-sdk

Version:
115 lines (109 loc) 19.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = exports.DefaultOptions = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _arrays = require("../../utils/arrays"); function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /* Copyright 2024 New Vector Ltd. Copyright 2020-2023 The Matrix.org Foundation C.I.C. SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ const DefaultOptions = exports.DefaultOptions = { maxCount: 200, gravity: 0.05, maxDrift: 5 }; const KEY_FRAME_INTERVAL = 15; // 15ms, roughly class Snowfall { constructor(options) { (0, _defineProperty2.default)(this, "options", void 0); (0, _defineProperty2.default)(this, "context", null); (0, _defineProperty2.default)(this, "particles", []); (0, _defineProperty2.default)(this, "lastAnimationTime", 0); (0, _defineProperty2.default)(this, "isRunning", false); (0, _defineProperty2.default)(this, "start", async (canvas, timeout = 3000) => { if (!canvas) { return; } this.context = canvas.getContext("2d"); this.particles = []; const count = this.options.maxCount; while (this.particles.length < count) { this.particles.push(this.resetParticle({}, canvas.width, canvas.height)); } this.isRunning = true; requestAnimationFrame(this.renderLoop); if (timeout) { window.setTimeout(this.stop, timeout); } }); (0, _defineProperty2.default)(this, "stop", async () => { this.isRunning = false; }); (0, _defineProperty2.default)(this, "resetParticle", (particle, width, height) => { particle.x = Math.random() * width; particle.y = Math.random() * -height; particle.xCol = particle.x; particle.diameter = Math.random() * 7 + 4; particle.maximumDrift = Math.random() * this.options.maxDrift + 3.5; particle.gravity = this.options.gravity + Math.random() * 6 + 4; return particle; }); (0, _defineProperty2.default)(this, "renderLoop", () => { if (!this.context || !this.context.canvas) { return; } if (this.particles.length === 0) { this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height); } else { const timeDelta = Date.now() - this.lastAnimationTime; if (timeDelta >= KEY_FRAME_INTERVAL || !this.lastAnimationTime) { // Clear the screen first this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height); this.lastAnimationTime = Date.now(); this.animateAndRenderSnowflakes(); } requestAnimationFrame(this.renderLoop); } }); this.options = _objectSpread(_objectSpread({}, DefaultOptions), options); } animateAndRenderSnowflakes() { if (!this.context || !this.context.canvas) { return; } const height = this.context.canvas.height; for (const particle of (0, _arrays.arrayFastClone)(this.particles)) { particle.y += particle.gravity; // We treat the drift as a sine function to have a more fluid-like movement instead // of a pong-like movement off walls of the X column. This means that for // $x=A\sin(\frac{2\pi}{P}y)$ we use the `maximumDrift` as the amplitude (A) and a // large multiplier to create a very long waveform through P. const peakDistance = 75 * particle.maximumDrift; const PI2 = Math.PI * 2; particle.x = particle.maximumDrift * Math.sin(PI2 / peakDistance * particle.y); particle.x += particle.xCol; // bring the particle to the right place const radius = particle.diameter / 2; this.context.save(); this.context.beginPath(); this.context.ellipse(particle.x, particle.y, radius, radius, 0, 0, 360); this.context.fillStyle = "#eaeaea"; // grey so it shows up on the light theme this.context.fill(); this.context.closePath(); this.context.restore(); // Remove any dead snowflakes const maxBounds = radius * 4; // make sure it's *really* off screen if (particle.y > height + maxBounds) { const idx = this.particles.indexOf(particle); this.particles.splice(idx, 1); } } } } exports.default = Snowfall; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_arrays","require","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","default","getOwnPropertyDescriptors","defineProperties","defineProperty","DefaultOptions","exports","maxCount","gravity","maxDrift","KEY_FRAME_INTERVAL","Snowfall","constructor","options","canvas","timeout","context","getContext","particles","count","resetParticle","width","height","isRunning","requestAnimationFrame","renderLoop","window","setTimeout","stop","particle","x","Math","random","y","xCol","diameter","maximumDrift","clearRect","timeDelta","Date","now","lastAnimationTime","animateAndRenderSnowflakes","arrayFastClone","peakDistance","PI2","PI","sin","radius","save","beginPath","ellipse","fillStyle","fill","closePath","restore","maxBounds","idx","indexOf","splice"],"sources":["../../../src/effects/snowfall/index.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020-2023 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n */\nimport ICanvasEffect from \"../ICanvasEffect\";\nimport { arrayFastClone } from \"../../utils/arrays\";\n\nexport type SnowfallOptions = {\n    /**\n     * The maximum number of snowflakes to render at a given time\n     */\n    maxCount: number;\n    /**\n     * The amount of gravity to apply to the snowflakes\n     */\n    gravity: number;\n    /**\n     * The amount of drift (horizontal sway) to apply to the snowflakes. Each snowflake varies.\n     */\n    maxDrift: number;\n};\n\ntype Snowflake = {\n    x: number;\n    y: number;\n    xCol: number;\n    diameter: number;\n    maximumDrift: number;\n    gravity: number;\n};\n\nexport const DefaultOptions: SnowfallOptions = {\n    maxCount: 200,\n    gravity: 0.05,\n    maxDrift: 5,\n};\n\nconst KEY_FRAME_INTERVAL = 15; // 15ms, roughly\n\nexport default class Snowfall implements ICanvasEffect {\n    private readonly options: SnowfallOptions;\n\n    public constructor(options: { [key: string]: any }) {\n        this.options = { ...DefaultOptions, ...options };\n    }\n\n    private context: CanvasRenderingContext2D | null = null;\n    private particles: Array<Snowflake> = [];\n    private lastAnimationTime = 0;\n\n    public isRunning = false;\n\n    public start = async (canvas: HTMLCanvasElement, timeout = 3000): Promise<void> => {\n        if (!canvas) {\n            return;\n        }\n        this.context = canvas.getContext(\"2d\");\n        this.particles = [];\n        const count = this.options.maxCount;\n        while (this.particles.length < count) {\n            this.particles.push(this.resetParticle({} as Snowflake, canvas.width, canvas.height));\n        }\n        this.isRunning = true;\n        requestAnimationFrame(this.renderLoop);\n        if (timeout) {\n            window.setTimeout(this.stop, timeout);\n        }\n    };\n\n    public stop = async (): Promise<void> => {\n        this.isRunning = false;\n    };\n\n    private resetParticle = (particle: Snowflake, width: number, height: number): Snowflake => {\n        particle.x = Math.random() * width;\n        particle.y = Math.random() * -height;\n        particle.xCol = particle.x;\n        particle.diameter = Math.random() * 7 + 4;\n        particle.maximumDrift = Math.random() * this.options.maxDrift + 3.5;\n        particle.gravity = this.options.gravity + Math.random() * 6 + 4;\n        return particle;\n    };\n\n    private renderLoop = (): void => {\n        if (!this.context || !this.context.canvas) {\n            return;\n        }\n        if (this.particles.length === 0) {\n            this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height);\n        } else {\n            const timeDelta = Date.now() - this.lastAnimationTime;\n            if (timeDelta >= KEY_FRAME_INTERVAL || !this.lastAnimationTime) {\n                // Clear the screen first\n                this.context.clearRect(0, 0, this.context.canvas.width, this.context.canvas.height);\n\n                this.lastAnimationTime = Date.now();\n                this.animateAndRenderSnowflakes();\n            }\n            requestAnimationFrame(this.renderLoop);\n        }\n    };\n\n    private animateAndRenderSnowflakes(): void {\n        if (!this.context || !this.context.canvas) {\n            return;\n        }\n        const height = this.context.canvas.height;\n        for (const particle of arrayFastClone(this.particles)) {\n            particle.y += particle.gravity;\n\n            // We treat the drift as a sine function to have a more fluid-like movement instead\n            // of a pong-like movement off walls of the X column. This means that for\n            // $x=A\\sin(\\frac{2\\pi}{P}y)$ we use the `maximumDrift` as the amplitude (A) and a\n            // large multiplier to create a very long waveform through P.\n            const peakDistance = 75 * particle.maximumDrift;\n            const PI2 = Math.PI * 2;\n            particle.x = particle.maximumDrift * Math.sin((PI2 / peakDistance) * particle.y);\n            particle.x += particle.xCol; // bring the particle to the right place\n\n            const radius = particle.diameter / 2;\n            this.context.save();\n            this.context.beginPath();\n            this.context.ellipse(particle.x, particle.y, radius, radius, 0, 0, 360);\n            this.context.fillStyle = \"#eaeaea\"; // grey so it shows up on the light theme\n            this.context.fill();\n            this.context.closePath();\n            this.context.restore();\n\n            // Remove any dead snowflakes\n            const maxBounds = radius * 4; // make sure it's *really* off screen\n            if (particle.y > height + maxBounds) {\n                const idx = this.particles.indexOf(particle);\n                this.particles.splice(idx, 1);\n            }\n        }\n    }\n}\n"],"mappings":"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","ignoreList":[]}