Reference Source

src/parallax.mjs

import { Logger } from './vendor/logger.min.mjs';
import { Layer } from './layer.mjs'

class ParallaxSingleton {
	/**
	 * The version of the module.
	 */
	version = "VERSION_REPLACE_ME";
    /** The logger module this module uses to log errors / logs
     * @private
     * @type {Object}
     */
    logger = new Logger();
    /**
     * The layer class.
     * @type {Layer}
     */
    Layer = Layer;
    /**
     * An set of instances that use the parallax system.
     * @private
     * @type {Set}
     */
    instances = new Set();
    /**
     * Weakmap to store info on instances used in this module.
     * @private
     * @type {WeakMap}
     */
    instanceWeakMap = new WeakMap();
    /**
     * The last position of the camera.
     * @private
     * @type {x: number | null, y: number | null}
     */
    lastCamPos = { x: null, y: null };
    /**
     * The virtual position on the map where the layers look natural together.
     * @private
     * @type {x: number | null, y: number | null}
     */
    cameraAnchor = { x: null, y: null };
    /**
     * Whether the anchor y position is set.
     * @private
     * @type {boolean}
     */
    anchorYSet = false;
    /**
     * Whether the anchor x position is set.
     * @private
     * @type {boolean}
     */
    anchorXSet = false;
    /**
     * @private
     */
	constructor() {
        this.logger.registerType('Parallax-Module', '#ff6600');
	}
    /**
     * Gets the camera position.
     * @returns {{x: number, y: number}} - The camera position.
     */
    getCamPos() {
        const viewEye = VYLO.Client.getViewEye();
        if (viewEye) {
            return { x: viewEye.x, y: viewEye.y };
        }
        return { x: null, y: null };
    }
    /**
     * Whether the last camera position is set.
     * @returns {boolean} - Whether the last camera position is set.
     */
    hasLastCamPos() {
        return this.lastCamPos.x !== null && this.lastCamPos.y !== null;
    }
    /**
     * Sets the last camera position.
     * @param {number} pX - The last x position of the camera.
     * @param {number} pY - The last y position of the camera.
     */
    setLastCamPos(pX, pY) {
        this.lastCamPos.x = pX;
        this.lastCamPos.y = pY;
    }
    /**
     * Sets the anchor position for the parallax system.
     * @param {{ x: number, y: number }} pCameraAnchor - The virtual position on the map where the layers look natural together.
     */
    setCameraAnchor(pCameraAnchor) {
        this.setCameraAnchorX(pCameraAnchor.x);
        this.setCameraAnchorY(pCameraAnchor.y);
    }
    /**
     * Sets the anchor x position for the parallax system.
     * @param {number} pXAnchor - The x position to set the anchor to.
     */
    setCameraAnchorX(pXAnchor) {
        this.cameraAnchor.x = pXAnchor;
        this.anchorXSet = true;
    }
    /**
     * Sets the anchor y position for the parallax system.
     * @param {number} pYAnchor - The y position to set the anchor to.
     */
    setCameraAnchorY(pYAnchor) {
        this.cameraAnchor.y = pYAnchor;
        this.anchorYSet = true;
    }
    /**
     * Gets the anchor position.
     * @returns {{x: number | null, y: number | null}} - The anchor position.
     */
    getCameraAnchor() {
        return { ...this.cameraAnchor };
    }
    /**
     * Gets the anchor x position.
     * @returns {number | null} - The anchor x position.
     */
    getAnchorX() {
        return this.cameraAnchor.x;
    }
    /**
     * Gets the anchor y position.
     * @returns {number | null} - The anchor x position.
     */
    getAnchorY() {
        return this.cameraAnchor.y;
    }
    /**
     * Resets the anchor position.
     */
    resetAnchor() {
        this.resetAnchorX();
        this.resetAnchorY();
    }
    /**
     * Resets the anchor x position.
     */
    resetAnchorX() {
        this.cameraAnchor.x = null;
        this.anchorXSet = false;
    }
    /**
     * Resets the anchor y position.
     */
    resetAnchorY() {
        this.cameraAnchor.y = null;
        this.anchorYSet = false;
    }
    /**
     * Whether the anchor x position is set.
     * @returns {boolean} - Whether the anchor x position is set.
     */
    isAnchorXSet() {
        return this.anchorXSet;
    }
    /**
     * Whether the anchor y position is set.
     * @returns {boolean} - Whether the anchor y position is set.
     */
    isAnchorYSet() {
        return this.anchorYSet;
    }
    /**
     * Creates two clones of the instance to loop infinitely.
     * @private
     * @param {Diob} pInstance - The instance to base the clones off of.
     * @param {boolean} pBypassEvent - Whether to bypass the onRelocated event.
     * @returns {Diob[]} - An array of the two clones.
     */
    createLoopInstances(pInstance, pBypassEvent) {
        // Create a left and right clone
        const first = VYLO.newDiob('MapObject');
        const second = VYLO.newDiob('MapObject');

        const children = [first, second];

        first.isCullable = false;
        second.isCullable = false;

        // Make the left and right clone particle look the same as the initial instance
        first.setAppearance(pInstance);
        second.setAppearance(pInstance);

        if (!pBypassEvent) {
            // Do not mutate event if one is found. Call alongside it.
            const oldRelocatedEvent = pInstance.onRelocated;
            // When the main instance moves, move the clones with their relative position to it.
            if (typeof oldRelocatedEvent === 'function') {
                pInstance.onRelocated = (pX, pY) => {
                    oldRelocatedEvent.call(pInstance, pX, pY);
                    this.handleOnRelocated(pInstance, children);
                }
            } else {
                pInstance.onRelocated = (pX, pY) => {
                    this.handleOnRelocated(pInstance, children);
                }
            }
        }

        return children;
    }
    /**
     * Enables infinite looping for the horizontal plane.
     * @private
     * @param {Diob} pInstance - The instance to loop.
     * @param {boolean} pBypassEvent - Whether to bypass the onRelocated event.
     */
    toggleInfiniteHorizontal(pInstance, pBypassEvent) {
        const [left, right] = this.createLoopInstances(pInstance, pBypassEvent);

        // Position the left clone
        left.x = pInstance.x - pInstance.icon.width;
        left.y = pInstance.y;

        // Position the right clone
        right.x = pInstance.x + pInstance.icon.width;
        right.y = pInstance.y;

        // Store the clones in a temporary array
        const children = [left, right];
        // Loop the clones and store their relative positions to the main instance
        children.forEach((pChild) => {
            pChild.relativeX = pChild.x - pInstance.x;
            pChild.relativeY = pChild.y - pInstance.y;
        });
    }

    /**
     * Enables infinite looping for the vertical plane.
     * @private
     * @param {Diob} pInstance - The instance to loop.
     * @param {boolean} pBypassEvent - Whether to bypass the onRelocated event.
     */
    toggleInfiniteVertical(pInstance, pBypassEvent) {
        const [top, bottom] = this.createLoopInstances(pInstance, pBypassEvent);

        // Position the left clone
        top.x = pInstance.x;
        top.y = pInstance.y - pInstance.icon.height;

        // Position the right clone
        bottom.x = pInstance.x;
        bottom.y = pInstance.y + pInstance.icon.height;

        // Store the clones in a temporary array
        const children = [top, bottom];
        // Loop the clones and store their relative positions to the main instance
        children.forEach((pChild) => {
            pChild.relativeX = pChild.x - pInstance.x;
            pChild.relativeY = pChild.y - pInstance.y;
        });
    }
    /**
     * Toggle infinite looping for both the horizontal and vertical planes.
     * @private
     * @param {Diob} pInstance - The instance to loop.
     */
    toggleInfinitePlanes(pInstance) {
        this.toggleInfiniteHorizontal(pInstance, true);
        this.toggleInfiniteVertical(pInstance);
    }
    /**
     * Adds an instance to the parallax system.
     * Call this first and then add your instance to the map.
     * @private
     * @param {Object} pInstance - The instance to add to the parallax system.
     * @param {Object} pParallaxConfig - The parallax info that tells this module how to control this instance.
     * @prop {number} pParallaxConfig.x - The x multiplier for this instance. Controls how fast or slow this instance moves. -Infinity to Infinity. 1 to move with camera.
     * @prop {number} pParallaxConfig.y - The y multiplier for this instance. Controls how fast or slow this instance moves. -Infinity to Infinity. 1 to move with camera.
     * @prop {boolean} pParallaxConfig.infiniteHorizontal - Whether this instance will infiniteHorizontal endlessly.
     * @prop {boolean} pParallaxConfig.infiniteVertical - Whether this instance will infiniteVertical endlessly.
     * @prop {number} pParallaxConfig.cameraAnchorX - The x position of the camera to anchor this instance to.
     * @prop {number} pParallaxConfig.cameraAnchorY - The y position of the camera to anchor this instance to.
     * 
     * ## The following is how the speed of the parallax multipliers are factored in.  
     (x | y) < 1 = faster behind the camera eg: (-> Player goes this way = Instance goes this way <-)  
     (x | y) > 1 faster against the camera eg: (-> Player goes this way = Instance goes this way ->)  
     (x | y) = 0 = static to the camera eg: (-> Player goes this way = Instance does nothing, and moves with the camera)   
     (x | y) = 1 = moves with the camera eg: (-> Player goes this way = Instance goes this way -> at position of camera)  
     */
    add(pInstance, pParallaxConfig) {
        if (!pInstance) {
            this.logger.prefix('Parallax-Module').error('No pInstance passed!');
            return;
        }

        if (pParallaxConfig instanceof Object) {
            if (!this.instances.has(pInstance)) {
                const { x, y, mapName } = pInstance;
                // Clone the parallax object
                const parallaxConfig = { ...pParallaxConfig };
                // Set the parallax info to the instance
                this.instanceWeakMap.set(pInstance, parallaxConfig);
                this.instances.add(pInstance);
                if (typeof x === 'number' && typeof y === 'number' && typeof mapName === 'string') {
                    pInstance.setPos(x, y, mapName);
                }
                this.init(pInstance, parallaxConfig);
            }
        } else {
            this.logger.prefix('Parallax-Module').error('No pParallaxConfig passed or invalid type found!');
        }
    }
    /**
     * Initializes this instance.
     * @private
     * @param {Object} pInstance - The instance to initialize.
     * @param {Object} pParallaxConfig - The parallax info that tells this module how to control this instance.
     * @prop {number} pParallaxConfig.x - The x multiplier for this instance. Controls how fast or slow this instance moves. -Infinity to Infinity. 0 to move with camera.
     * @prop {number} pParallaxConfig.y - The y multiplier for this instance. Controls how fast or slow this instance moves. -Infinity to Infinity. 0 to move with camera.
     * @prop {boolean} pParallaxConfig.infiniteHorizontal - Whether this instance will loop endlessly.
     * @prop {boolean} pParallaxConfig.infiniteVertical - Whether this instance will loop endlessly.
     * @prop {number} pParallaxConfig.cameraAnchorX - The x position of the camera to anchor this instance to.
     * @prop {number} pParallaxConfig.cameraAnchorY - The y position of the camera to anchor this instance to.
     */
    init(pInstance, pParallaxConfig) {
        if (!VYLO) {
            this.logger.prefix('Parallax-Module').error('VYLO not found! This module depends on the VYLO object being in the global name space.');
            return;
        }

        const { x, y } = this.getCamPos();

        if (!this.hasLastCamPos()) {
            this.setLastCamPos(x, y);
        }

        // Update the instance's initial position based on the anchor position
        this.updateInstance(pInstance, x, y, { x: pParallaxConfig.cameraAnchorX, y: pParallaxConfig.cameraAnchorY });

        const { infiniteHorizontal, infiniteVertical } = pParallaxConfig;

        if (infiniteHorizontal && infiniteVertical) {
            this.toggleInfinitePlanes(pInstance);
        } else if (infiniteHorizontal) {
            this.toggleInfiniteHorizontal(pInstance);
        } else if (infiniteVertical) {
            this.toggleInfiniteVertical(pInstance);
        }
    }
    /**
     * Removes an instance to the parallax system.
     * @param {Object} pInstance - The instance to remove to the parallax system.
     */
    remove(pInstance) {
        if (!pInstance) {
            this.logger.prefix('Parallax-Module').error('No pInstance passed!');
            return;
        }

        if (this.instances.has(pInstance)) {
            this.instances.delete(pInstance);
            this.instanceWeakMap.delete(pInstance);
        }
    }
    /**
     * Updates the parallax system.
     * @param {number} pCameraX - The x position of the camera.
     * @param {number} pCameraY - The y position of the camera.
    */
    update(pCameraX = 0, pCameraY = 0) {
        for (const instance of this.instances) {
            this.updateInstance(instance, pCameraX, pCameraY);
        }

        this.lastCamPos.x = pCameraX;
        this.lastCamPos.y = pCameraY;
    }
    /**
     * Updates the instance's position based on the camera's position.
     * @private
     * @param {Diob} pInstance - The instance to update.
     * @param {number} pCameraX - The x position of the camera.
     * @param {number} pCameraY - The y position of the camera.
     * @param {{x: number | null, y: number | null}} [pAnchor] - The camera anchor position to use.
     */
    updateInstance(pInstance, pCameraX, pCameraY, pAnchor) {
        let lastCamPosX = this.lastCamPos.x;
        let lastCamPosY = this.lastCamPos.y;

        if (pAnchor) {
            const x = this.getAnchorX() || pAnchor.x;
            const y = this.getAnchorY() || pAnchor.y;

            if (typeof x === 'number') {
                lastCamPosX = x;
            }

            if (typeof y === 'number') {
                lastCamPosY = y;
            }
        }

        const parallaxConfig = this.instanceWeakMap.get(pInstance);

        // Move the instance with the camera if the parallax is set to 0
        const isBackgroundX = parallaxConfig.x === 0;
        const isBackgroundY = parallaxConfig.y === 0;

        // Position to set the instance to.
        let x;
        let y;

        if (isBackgroundX) {
            x = pCameraX - pInstance.icon.width / 2;
        } else {
            let deltaX = pCameraX - lastCamPosX;
            let distX = deltaX * parallaxConfig.x;
            x = pInstance.x + distX;
        }

        if (isBackgroundY) {
            y = pCameraY - pInstance.icon.height / 2;
        } else {
            let deltaY = pCameraY - lastCamPosY;
            let distY = deltaY * parallaxConfig.y;
            y = pInstance.y + distY;
        }

        // Set the position
        pInstance.x = x;
        pInstance.y = y;

        // Logic cannot be ran on static background instances as they should not loop
        if (!isBackgroundX && !isBackgroundY) {
            if (parallaxConfig.infiniteHorizontal) {
                // The start pos + total width
                const rightEnd = pInstance.x + pInstance.icon.width;
                // The start pos - total width / 6
                const leftEnd = pInstance.x - pInstance.icon.width / 6;
                if (pCameraX > rightEnd) {
                    pInstance.x += pInstance.icon.width;
                } else if (pCameraX < leftEnd) {
                    pInstance.x -= pInstance.icon.width;
                }
            }

            if (parallaxConfig.infiniteVertical) {
                // The start pos + total height
                const bottomEnd = pInstance.x + pInstance.icon.height;
                // The start pos - total height / 6
                const topEnd = pInstance.x - pInstance.icon.height / 6;
                if (pCameraY > bottomEnd) {
                    pInstance.y += pInstance.icon.height;
                } else if (pCameraY < topEnd) {
                    pInstance.y -= pInstance.icon.height;
                }
            }
        }
    }
    /**
     * Handles the onRelocated event for instances. Moves their children in relativity to their position.
     * @private
     * @param {Diob | MapObject} pInstance - The instance to handle the event for.
     * @param {MapObject[]} pChildren - An array of children belonging to the instance.
     */
    handleOnRelocated(pInstance, pChildren) {
        // Update the children's position when the parent moves
        pChildren.forEach((pChild) => {
            pChild.x = pInstance.x + pChild.relativeX;
            pChild.y = pInstance.y + pChild.relativeY;
            pChild.mapName = pInstance.mapName;
        });
    }
}

const Parallax = new ParallaxSingleton();
export { Parallax };