screenshots.gif

(function ghostCursor() {
    var width = window.innerWidth;
    var height = window.innerHeight;
    var cursor = {
        x: width / 2,
        y: width / 2,
    };
    var particles = [];

    function init() {
        bindEvents();
        attachInitialParticleStyles();
        loop();
    }

    // Bind events that are needed
    function bindEvents() {
        document.addEventListener("mousemove", onMouseMove);
        document.addEventListener("touchmove", onTouchMove);
        document.addEventListener("touchstart", onTouchMove);

        window.addEventListener("resize", onWindowResize);
    }

    function onWindowResize(e) {
        width = window.innerWidth;
        height = window.innerHeight;
    }

    function onTouchMove(e) {
        if (e.touches.length > 0) {
            for (var i = 0; i < e.touches.length; i++) {
                addParticle(e.touches[i].clientX, e.touches[i].clientY);
            }
        }
    }

    function onMouseMove(e) {
        cursor.x = e.clientX;
        cursor.y = e.clientY;

        addParticle(cursor.x, cursor.y);
    }

    function addParticle(x, y) {
        var particle = new Particle();
        particle.init(x, y);
        particles.push(particle);
    }

    function updateParticles() {
        // Updated
        for (var i = 0; i < particles.length; i++) {
            particles[i].update();
        }

        // Remove dead particles
        for (var i = particles.length - 1; i >= 0; i--) {
            if (particles[i].lifeSpan < 0) {
                particles[i].die();
                particles.splice(i, 1);
            }
        }
    }

    function loop() {
        requestAnimationFrame(loop);
        updateParticles();
    }

    /**
     * Particles
     */

    function Particle() {
        // How long cursor particle will live for
        this.maxLifeSpan = 80;

        // Will begin fading at this point in the lifespan
        this.fadePoint = 50;

        this.lifeSpan = this.maxLifeSpan;

        // Init, and set properties
        this.init = function(x, y) {
            this.position = {
                x: x - 10,
                y: y - 10,
            };

            this.element = document.createElement("span");
            this.element.className = "particle-cursors";
            this.update();

            document.body.appendChild(this.element);
        };

        this.update = function() {
            this.lifeSpan--;

            this.element.style.transform = "translate3d(" + this.position.x + "px," + this.position.y + "px,0)";

            if (this.lifeSpan < this.fadePoint) {
                this.element.style.opacity = (this.lifeSpan - 1) / (this.fadePoint - 1);
            }
        };

        this.die = function() {
            this.element.parentNode.removeChild(this.element);
        };
    }

    /**
     * Utils
     */

    // Injects initial cursor styles to the head of the page.
    function attachInitialParticleStyles() {
        var initialStyles = [
            ".particle-cursors {",
            "position: absolute;",
            "display: block;",
            "pointer-events: none;",
            "z-index: 1000000;",
            "width: 12px;",
            "height: 19px;",
            "will-change: transform;",
            "background-size: contain;",
            "background-image: url('');",
            "}",
        ].join("");

        var style = document.createElement("style");
        style.type = "text/css";
        style.innerHTML = initialStyles;
        document.getElementsByTagName("head")[0].appendChild(style);
    }

    // Applies css `properties` to an element.
    function applyProperties(target, properties) {
        for (var key in properties) {
            target.style[key] = properties[key];
        }
    }

    init();
})();