/* eslint-disable no-multi-spaces */
// ==UserScript==
// @name OSU navigator
// @name:zh-CN 鼠标OSU化
// @name:zh-TW 滑鼠OSU化
// @name:ko 마우스 OSU화
// @namespace OSU_NAVIGATOR
// @version 0.2
// @description Use key "z" and "x" as mouse left and right, and displays your mouse cursor as osu yellow mouse cursor
// @description:zh-CN 使用"z"和"x"键作为鼠标左右键,并将鼠标样式显示为圆形亮黄osu光标
// @description:zh-TW 使用“z”和“x”鍵作為滑鼠左右鍵,並將滑鼠樣式顯示為圓形亮黃osu光標
// @description:ko "z" 및 "x" 키를 마우스 좌우 키로 사용하고 마우스 스타일을 둥근 밝은 노란색 osu 커서로 표시합니다
// @author PY-DNG
// @license MIT
// @match http*://*/*
// @icon https://api.iowen.cn/favicon/get.php?url=osu.ppy.sh
// @grant none
// ==/UserScript==
(function() {
'use strict';
// Arguments: level=LogLevel.Info, logContent, asObject=false
// Needs one call "DoLog();" to get it initialized before using it!
function DoLog() {
// Global log levels set
window.LogLevel = {
None: 0,
Error: 1,
Success: 2,
Warning: 3,
Info: 4,
}
window.LogLevelMap = {};
window.LogLevelMap[LogLevel.None] = {prefix: '' , color: 'color:#ffffff'}
window.LogLevelMap[LogLevel.Error] = {prefix: '[Error]' , color: 'color:#ff0000'}
window.LogLevelMap[LogLevel.Success] = {prefix: '[Success]' , color: 'color:#00aa00'}
window.LogLevelMap[LogLevel.Warning] = {prefix: '[Warning]' , color: 'color:#ffa500'}
window.LogLevelMap[LogLevel.Info] = {prefix: '[Info]' , color: 'color:#888888'}
window.LogLevelMap[LogLevel.Elements] = {prefix: '[Elements]', color: 'color:#000000'}
// Current log level
DoLog.logLevel = LogLevel.Info; // Info Warning Success Error
// Log counter
DoLog.logCount === undefined && (DoLog.logCount = 0);
if (++DoLog.logCount > 512) {
console.clear();
DoLog.logCount = 0;
}
// Get args
let level, logContent, asObject;
switch (arguments.length) {
case 1:
level = LogLevel.Info;
logContent = arguments[0];
asObject = false;
break;
case 2:
level = arguments[0];
logContent = arguments[1];
asObject = false;
break;
case 3:
level = arguments[0];
logContent = arguments[1];
asObject = arguments[2];
break;
default:
level = LogLevel.Info;
logContent = 'DoLog initialized.';
asObject = false;
break;
}
// Log when log level permits
if (level <= DoLog.logLevel) {
let msg = '%c' + LogLevelMap[level].prefix;
let subst = LogLevelMap[level].color;
if (asObject) {
msg += ' %o';
} else {
switch(typeof(logContent)) {
case 'string': msg += ' %s'; break;
case 'number': msg += ' %d'; break;
case 'object': msg += ' %o'; break;
}
}
console.log(msg, subst, logContent);
}
}
DoLog();
main();
function main() {
// Terminal element event listeners
/*
for (const elm of document.querySelectorAll('*')) {
dealElement(elm);
}
document.addEventListener('DOMNodeInserted', (e) => {if(!e.target){debugger;}dealElement(e.target);});
*/
document.addEventListener('mousemove', function(e) {
const elm = document.elementFromPoint(e.x, e.y);
removeListeners(window.OSUMouse.target);
addListeners(elm);
window.OSUMouse.target = elm;
}, {
capture: true,
passive: true
})
// Global event listeners
document.body.onkeydown = keyDownListener;
document.body.onkeyup = keyUpListener;
// Global status recorder
window.OSUMouse = {
ctrlKey: false,
altKey: false,
shiftKey: false,
metaKey: false,
target: document.body
};
// Change cursor
osuMouseCursor();
}
function addListeners(elm) {
elm.addEventListener('mousemove', recordMouseStatus);
}
function removeListeners(elm) {
// Record mouse status
elm.removeEventListener('mousemove', recordMouseStatus);
}
function recordMouseStatus(e) {
const props = ['screenX', 'screenY', 'clientX', 'clientY', 'relatedTarget', 'region']
for (const prop of props) {
window.OSUMouse[prop] = e[prop];
}
}
function keyDownListener(e) {
switch (e.key) {
case 'Control':
window.OSUMouse.ctrlKey = true;
//DoLog(window.OSUMouse);
break;
case 'Shift':
window.OSUMouse.shiftKey = true;
//DoLog(window.OSUMouse);
break;
case 'Alt':
window.OSUMouse.altKey = true;
//DoLog(window.OSUMouse);
break;
case 'Meta':
window.OSUMouse.metaKey = true;
//DoLog(window.OSUMouse);
break;
case 'z':
case 'Z':
case 'x':
case 'X':
dispatchMouseDown(e.target);
break;
}
}
function keyUpListener(e) {
switch (e.key) {
case 'Control':
window.OSUMouse.ctrlKey = false;
//DoLog(window.OSUMouse);
break;
case 'Shift':
window.OSUMouse.shiftKey = false;
//DoLog(window.OSUMouse);
break;
case 'Alt':
window.OSUMouse.altKey = false;
//DoLog(window.OSUMouse);
break;
case 'Meta':
window.OSUMouse.metaKey = false;
//DoLog(window.OSUMouse);
break;
case 'z':
case 'Z':
!inputing() && dispatchMouseLeftUp();
break;
case 'x':
case 'X':
!inputing() && dispatchMouseRightUp();
break;
}
}
function dispatchMouseDown() {
const mouseEventInit = {};
for (const [key, value] of Object.entries(window.OSUMouse)) {
mouseEventInit[key] = value;
}
mouseEventInit.bubbles = true;
const focusEventInit = {relatedTarget: window.OSUMouse.relatedTarget, bubbles: true};
const mouseLeft = new MouseEvent('mousedown', mouseEventInit);
const focus = new FocusEvent('focus', focusEventInit);
window.OSUMouse.target.dispatchEvent(focus);
window.OSUMouse.target.dispatchEvent(mouseLeft);
}
function dispatchMouseLeftUp() {
const mouseEventInit = {};
for (const [key, value] of Object.entries(window.OSUMouse)) {
mouseEventInit[key] = value;
}
mouseEventInit.bubbles = true;
const mouseRight = new MouseEvent('mouseup', mouseEventInit);
const mouseclick = new MouseEvent('click', mouseEventInit);
window.OSUMouse.target.dispatchEvent(mouseRight);
window.OSUMouse.target.dispatchEvent(mouseclick);
}
function dispatchMouseRightUp() {
const mouseEventInit = {};
for (const [key, value] of Object.entries(window.OSUMouse)) {
mouseEventInit[key] = value;
}
mouseEventInit.bubbles = true;
const mousecontextmenu = new MouseEvent('contentmenu', mouseEventInit);
window.OSUMouse.target.dispatchEvent(mousecontextmenu);
}
function inputing() {
return document.activeElement && [HTMLInputElement, HTMLTextAreaElement].some((o) => (document.activeElement instanceof o));
}
function osuMouseCursor() {
// Cursor
const OSUCursor = '';
const CSSCursor = 'body {cursor: url("{C}"), auto !important;}'.replace('{C}', OSUCursor);
addStyle(CSSCursor, 'osu_cursor');
/*
// Canvas
const canvas = document.createElement('canvas');
const CSSCanvas = '#osu_cursor_canvas {position: fixed; pointer-events: none; z-index: 99999999}';
const img = new Image();
img.onload = function() {
const ctx = canvas.getContext('2d');
const half = img.width / 2;
canvas.width = img.width;
canvas.height = img.height;
ctx.drawImage(img, 0, 0);
canvas.id = 'osu_cursor_canvas';
document.body.addEventListener('mousemove', (e) => {
canvas.style.top = (e.clientY - half).toString() + 'px';
canvas.style.left = (e.clientX - half).toString() + 'px';
});
document.body.appendChild(canvas);
};
img.src = OSUCursor;
addStyle(CSSCanvas);
*/
}
// Just stopPropagation and preventDefault
function destroyEvent(e) {
if (!e) {return false;};
if (!e instanceof Event) {return false;};
e.stopPropagation();
e.preventDefault();
}
// Append a style text to document(<head>) with a <style> element
function addStyle(css, id) {
const style = document.createElement("style");
id && (style.id = id);
style.textContent = css;
for (const elm of document.querySelectorAll('#'+id)) {
elm.parentElement && elm.parentElement.removeChild(elm);
}
document.head.appendChild(style);
}
})();