Giddy Mouse

The cursor will rotate naturally, following your movements

// ==UserScript==
// @name         Giddy Mouse
// @description  The cursor will rotate naturally, following your movements
// @author       Ko
// @namespace    u/TagProKo
// @license      MIT
// @version      2.0
// @icon         
// @include      *
// @run-at       document-idle
// ==/UserScript==

// I tried my best explaning everything using comments.

(function(){

    // Define all supported cursors, their hotspot position, their angle (in radians with respect to ->) and their data uri
    // I used the default cursors for Internet Explorer on Windows 7, including an animated 'progress' cursor (animated png)
    // The grab, grabbing and both the zoom cursors are those used by chromium/chromeos

    // To replicate my process (for example when you want to use Windows 10 cursors), use 'RealWorld Cursor Editor' to convert
    // .cur and .ani (found in c:\windows\cursors) to .png and animated .png respectively. Then encode those into a base64 data url.
    // A base64 encoder that works for animaged png's: https://onlinepngtools.com/convert-png-to-base64 (remember to select 'create valid url'

    var srcs = {
        'alias':        {x:0,y:0,   ang:-1.9634954084936207,src:''},
        'context-menu': {x:0,y:0,   ang:-1.9634954084936207,src:''},
        'copy':         {x:0,y:0,   ang:-1.9634954084936207,src:''},
        'default':      {x:0,y:0,   ang:-1.9634954084936207,src:''},
        'grab':         {x:10,y:11, ang:-1.5707963267948966,src:''},
        'grabbing':     {x:7,y:7,   ang:-1.5707963267948966,src:''},
        'help':         {x:0,y:0,   ang:-1.9634954084936207,src:''},
        'pointer':      {x:6,y:0,   ang:-1.5707963267948966,src:''},
        'progress':     {x:0,y:8,   ang:-1.9634954084936207,src:''},
        'zoom-in':      {x:6,y:6,   ang:-2.3561944901923450,src:''},
        'zoom-out':     {x:6,y:6,   ang:-2.3561944901923450,src:''},
    };

    // I find having the cursor switch to a caret over selectable text really breaks this experience, therefore 'auto' can always be the 'default' cursor.
    srcs.auto = srcs.default;

    // The rotating cursor is an IMG element, the actual cursor will be hidden
    var cursor = document.createElement('img');
    document.body.appendChild(cursor);
    cursor.id = 'cursor';



    // Get the computedCursor for the root element.
    document.documentElement.computedCursor = getComputedStyle(document.documentElement).cursor;
    // All other element's computedCursors will be calculated on the fly and then cached.



    // Create a stylesheet for our custom styling.

    var style = document.createElement('style');
    style.id = 'rotating-cursor'; // Just for anyone wondering what the extra style element is doing in their DOM.
    document.head.appendChild(style);
    var styleSheet = style.sheet;

    // The rotating cursor's styling itsself. Hidden by default, and only shown when it has the .enabled class
    styleSheet.insertRule('#cursor { position: fixed; left: 0; top: 0; z-index: 2147483647; pointer-events: none; opacity: 0; }');
    styleSheet.insertRule('#cursor.enabled { opacity: 1; }');

    // Hide the original cursor on any element
    styleSheet.insertRule(':not(.cursorcheck):not(.unsupported) { cursor: none !important; }');



    // Values for the rotating simulation
    // The position of the 'tail' of the cursor will always be stored, at a set distance (30) from the tip
    var tail = 30,
        tailX = 0,
        tailY = 0,
        angle;

    // The initial cursor, shown after the first mouse event.
    // Just to get the mouse on screen 1 frame earlier, before the computedCursor has been calculated
    var src = srcs.auto;

    function mousemove(event, skipSimulation) {

        // Whether the mouse was enabled or not *before* this mousemove
        var enabled = cursor.classList.contains('enabled');

        // Get the target's computedCursor (from cache or using the function)
        if (event.target.computedCursor) src = srcs[event.target.computedCursor];
        else setTimeout(getComputedCursor, null, event);

        // Enable or disable our rotating cursor based on whether we support it.
        if (src) cursor.classList.add('enabled');
        else return cursor.classList.remove('enabled');

        // Simulate the rotation

        if (skipSimulation) {}

        else if (enabled) {

            // Regular simulation

            var dx = event.clientX - tailX,
                dy = event.clientY - tailY;

            var distance = Math.sqrt( Math.pow(dx,2) + Math.pow(dy,2) );

            if (distance) {
                tailX = event.clientX - dx * tail / distance;
                tailY = event.clientY - dy * tail / distance;
            }

            angle = Math.atan2( dy, dx );

        } else {

            // Reset the angle when entering a supported area again

            angle = src.ang;

            tailX = event.clientX - tail * Math.cos(angle);
            tailY = event.clientY - tail * Math.sin(angle);

        }

        // After the simulation, the result is applied to the rotating cursor style.

        cursor.style.transform =
            "translate(calc(" + event.clientX + "px - 50%),calc(" + event.clientY + "px - 50%))" +
            "rotate(" + (angle - src.ang) + "rad)" +
            "translate(calc(50% - " + src.x + "px), calc(50% - " + src.y + "px))";

        // Last but not least, the right image is set.
        cursor.src = src.src;
    }



    var scrolling;

    // Events after which the cursor is to be updated.
    // When entering an element by moving the mouse, both a mousemove & mouseover event are emitted
    // Only when scrolling, the browser can sent a 'mouseover' without a 'mousemove'
    document.addEventListener('mousemove', mousemove);
    document.addEventListener('mouseover', function(event) { if (scrolling) setTimeout(mousemove,null,event); } );

    // To avoid having to run our 'mousemove' function twice,
    // we keep track of whether a mouseover is caused by scrolling or moving the mouse.
    document.addEventListener('mousemove', function() { scrolling = false; } );
    document.addEventListener('scroll', function() { scrolling = true; } );




    function getComputedCursor(event) {

        // If not cached, the computedCursor is calculated asynchroneously using this function.
        // This can take in the order of a frame to be calculated, which will cause a visible hickup if done synchroneously

        var element = event.target,
            elements = [event.target];

        // STEP 1: apply the .cursorcheck class to all ancestors until one has a computedCursor cached.
        //     This class blocks the globally set "cursor: none;"

        element.classList.add('cursorcheck');

        do {
            element = element.parentElement;
            elements.push(element);
            element.classList.add('cursorcheck');
        }
        while (!element.computedCursor);

        // STEP 2: The youngest ancestor to have a cached computedCursor will have it's cursor style overwritten to it,
        //     so that any offspring without a defined cursor style can inherit it.

        var org_cursor = element.style.cursor;
        element.style.cursor = element.computedCursor;

        // STEP 3: Since now all those ancestors have their original cursor style, we can as well cache all of them,
        //     instead of just the element the cursor hovers over at the moment.
        for (var e in elements) {
            element = elements[e];

            element.computedCursor = getComputedStyle(element).cursor;

            if (!srcs[element.computedCursor]) {
                element.classList.add('unsupported');
                element.style.cursor = event.target.computedCursor;
            }
        }

        // STEP 4: reset the overwritten cursor style, as well as removing all .cursorcheck classes again
        //     so that the original cursor will once again be hidden everywhere.
        element.style.cursor = org_cursor;

        for (e in elements) {
            elements[e].classList.remove('cursorcheck');
        }

        // Since the setTimeout will definately run before the next mousemove event,
        // we can be sure that the cursor is still in the same position.
        // To update the cursor with the newly calculated 'computedCursor', we just call mousemove again.
        // This is particullarly useful when moving just 1 pixel in a non-cached element and stopping then.

        mousemove(event, true);

    }



    // Various cases in which the cursor should disappear

    document.addEventListener('mouseleave', function() { cursor.classList.remove('enabled'); } );
    document.addEventListener('dragstart', function() { cursor.classList.remove('enabled'); } );

    // There are three more cases (of which I know) after which the cursor shoul disappear, but I don't know how to detect them:

    // 1. When the mouse enter's an iFrame
    // 2. When using the middle click to scroll
    // 3. When hovering a scrollbar

})();