// ==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
})();