Greasy Fork is available in English.

OSU navigator

Use key "z" and "x" as mouse left and right, and displays your mouse cursor as osu yellow mouse cursor

  1. /* eslint-disable no-multi-spaces */
  2.  
  3. // ==UserScript==
  4. // @name OSU navigator
  5. // @name:zh-CN 鼠标OSU化
  6. // @name:zh-TW 滑鼠OSU化
  7. // @name:ko 마우스 OSU화
  8. // @namespace OSU_NAVIGATOR
  9. // @version 0.2
  10. // @description Use key "z" and "x" as mouse left and right, and displays your mouse cursor as osu yellow mouse cursor
  11. // @description:zh-CN 使用"z"和"x"键作为鼠标左右键,并将鼠标样式显示为圆形亮黄osu光标
  12. // @description:zh-TW 使用“z”和“x”鍵作為滑鼠左右鍵,並將滑鼠樣式顯示為圓形亮黃osu光標
  13. // @description:ko "z" 및 "x" 키를 마우스 좌우 키로 사용하고 마우스 스타일을 둥근 밝은 노란색 osu 커서로 표시합니다
  14. // @author PY-DNG
  15. // @license MIT
  16. // @match http*://*/*
  17. // @icon https://api.iowen.cn/favicon/get.php?url=osu.ppy.sh
  18. // @grant none
  19. // ==/UserScript==
  20.  
  21. (function() {
  22. 'use strict';
  23.  
  24. // Arguments: level=LogLevel.Info, logContent, asObject=false
  25. // Needs one call "DoLog();" to get it initialized before using it!
  26. function DoLog() {
  27. // Global log levels set
  28. window.LogLevel = {
  29. None: 0,
  30. Error: 1,
  31. Success: 2,
  32. Warning: 3,
  33. Info: 4,
  34. }
  35. window.LogLevelMap = {};
  36. window.LogLevelMap[LogLevel.None] = {prefix: '' , color: 'color:#ffffff'}
  37. window.LogLevelMap[LogLevel.Error] = {prefix: '[Error]' , color: 'color:#ff0000'}
  38. window.LogLevelMap[LogLevel.Success] = {prefix: '[Success]' , color: 'color:#00aa00'}
  39. window.LogLevelMap[LogLevel.Warning] = {prefix: '[Warning]' , color: 'color:#ffa500'}
  40. window.LogLevelMap[LogLevel.Info] = {prefix: '[Info]' , color: 'color:#888888'}
  41. window.LogLevelMap[LogLevel.Elements] = {prefix: '[Elements]', color: 'color:#000000'}
  42.  
  43. // Current log level
  44. DoLog.logLevel = LogLevel.Info; // Info Warning Success Error
  45.  
  46. // Log counter
  47. DoLog.logCount === undefined && (DoLog.logCount = 0);
  48. if (++DoLog.logCount > 512) {
  49. console.clear();
  50. DoLog.logCount = 0;
  51. }
  52.  
  53. // Get args
  54. let level, logContent, asObject;
  55. switch (arguments.length) {
  56. case 1:
  57. level = LogLevel.Info;
  58. logContent = arguments[0];
  59. asObject = false;
  60. break;
  61. case 2:
  62. level = arguments[0];
  63. logContent = arguments[1];
  64. asObject = false;
  65. break;
  66. case 3:
  67. level = arguments[0];
  68. logContent = arguments[1];
  69. asObject = arguments[2];
  70. break;
  71. default:
  72. level = LogLevel.Info;
  73. logContent = 'DoLog initialized.';
  74. asObject = false;
  75. break;
  76. }
  77.  
  78. // Log when log level permits
  79. if (level <= DoLog.logLevel) {
  80. let msg = '%c' + LogLevelMap[level].prefix;
  81. let subst = LogLevelMap[level].color;
  82.  
  83. if (asObject) {
  84. msg += ' %o';
  85. } else {
  86. switch(typeof(logContent)) {
  87. case 'string': msg += ' %s'; break;
  88. case 'number': msg += ' %d'; break;
  89. case 'object': msg += ' %o'; break;
  90. }
  91. }
  92.  
  93. console.log(msg, subst, logContent);
  94. }
  95. }
  96. DoLog();
  97.  
  98. main();
  99. function main() {
  100. // Terminal element event listeners
  101. /*
  102. for (const elm of document.querySelectorAll('*')) {
  103. dealElement(elm);
  104. }
  105. document.addEventListener('DOMNodeInserted', (e) => {if(!e.target){debugger;}dealElement(e.target);});
  106. */
  107. document.addEventListener('mousemove', function(e) {
  108. const elm = document.elementFromPoint(e.x, e.y);
  109. removeListeners(window.OSUMouse.target);
  110. addListeners(elm);
  111. window.OSUMouse.target = elm;
  112. }, {
  113. capture: true,
  114. passive: true
  115. })
  116.  
  117. // Global event listeners
  118. document.body.onkeydown = keyDownListener;
  119. document.body.onkeyup = keyUpListener;
  120.  
  121. // Global status recorder
  122. window.OSUMouse = {
  123. ctrlKey: false,
  124. altKey: false,
  125. shiftKey: false,
  126. metaKey: false,
  127. target: document.body
  128. };
  129.  
  130. // Change cursor
  131. osuMouseCursor();
  132. }
  133.  
  134. function addListeners(elm) {
  135. elm.addEventListener('mousemove', recordMouseStatus);
  136. }
  137.  
  138. function removeListeners(elm) {
  139. // Record mouse status
  140. elm.removeEventListener('mousemove', recordMouseStatus);
  141. }
  142.  
  143. function recordMouseStatus(e) {
  144. const props = ['screenX', 'screenY', 'clientX', 'clientY', 'relatedTarget', 'region']
  145. for (const prop of props) {
  146. window.OSUMouse[prop] = e[prop];
  147. }
  148. }
  149.  
  150. function keyDownListener(e) {
  151. switch (e.key) {
  152. case 'Control':
  153. window.OSUMouse.ctrlKey = true;
  154. //DoLog(window.OSUMouse);
  155. break;
  156. case 'Shift':
  157. window.OSUMouse.shiftKey = true;
  158. //DoLog(window.OSUMouse);
  159. break;
  160. case 'Alt':
  161. window.OSUMouse.altKey = true;
  162. //DoLog(window.OSUMouse);
  163. break;
  164. case 'Meta':
  165. window.OSUMouse.metaKey = true;
  166. //DoLog(window.OSUMouse);
  167. break;
  168. case 'z':
  169. case 'Z':
  170. case 'x':
  171. case 'X':
  172. dispatchMouseDown(e.target);
  173. break;
  174. }
  175. }
  176.  
  177. function keyUpListener(e) {
  178. switch (e.key) {
  179. case 'Control':
  180. window.OSUMouse.ctrlKey = false;
  181. //DoLog(window.OSUMouse);
  182. break;
  183. case 'Shift':
  184. window.OSUMouse.shiftKey = false;
  185. //DoLog(window.OSUMouse);
  186. break;
  187. case 'Alt':
  188. window.OSUMouse.altKey = false;
  189. //DoLog(window.OSUMouse);
  190. break;
  191. case 'Meta':
  192. window.OSUMouse.metaKey = false;
  193. //DoLog(window.OSUMouse);
  194. break;
  195. case 'z':
  196. case 'Z':
  197. !inputing() && dispatchMouseLeftUp();
  198. break;
  199. case 'x':
  200. case 'X':
  201. !inputing() && dispatchMouseRightUp();
  202. break;
  203. }
  204. }
  205.  
  206. function dispatchMouseDown() {
  207. const mouseEventInit = {};
  208. for (const [key, value] of Object.entries(window.OSUMouse)) {
  209. mouseEventInit[key] = value;
  210. }
  211. mouseEventInit.bubbles = true;
  212. const focusEventInit = {relatedTarget: window.OSUMouse.relatedTarget, bubbles: true};
  213. const mouseLeft = new MouseEvent('mousedown', mouseEventInit);
  214. const focus = new FocusEvent('focus', focusEventInit);
  215. window.OSUMouse.target.dispatchEvent(focus);
  216. window.OSUMouse.target.dispatchEvent(mouseLeft);
  217. }
  218.  
  219. function dispatchMouseLeftUp() {
  220. const mouseEventInit = {};
  221. for (const [key, value] of Object.entries(window.OSUMouse)) {
  222. mouseEventInit[key] = value;
  223. }
  224. mouseEventInit.bubbles = true;
  225. const mouseRight = new MouseEvent('mouseup', mouseEventInit);
  226. const mouseclick = new MouseEvent('click', mouseEventInit);
  227. window.OSUMouse.target.dispatchEvent(mouseRight);
  228. window.OSUMouse.target.dispatchEvent(mouseclick);
  229. }
  230.  
  231. function dispatchMouseRightUp() {
  232. const mouseEventInit = {};
  233. for (const [key, value] of Object.entries(window.OSUMouse)) {
  234. mouseEventInit[key] = value;
  235. }
  236. mouseEventInit.bubbles = true;
  237. const mousecontextmenu = new MouseEvent('contentmenu', mouseEventInit);
  238. window.OSUMouse.target.dispatchEvent(mousecontextmenu);
  239. }
  240.  
  241. function inputing() {
  242. return document.activeElement && [HTMLInputElement, HTMLTextAreaElement].some((o) => (document.activeElement instanceof o));
  243. }
  244.  
  245. function osuMouseCursor() {
  246. // Cursor
  247. const OSUCursor = '';
  248. const CSSCursor = 'body {cursor: url("{C}"), auto !important;}'.replace('{C}', OSUCursor);
  249. addStyle(CSSCursor, 'osu_cursor');
  250.  
  251. /*
  252. // Canvas
  253. const canvas = document.createElement('canvas');
  254. const CSSCanvas = '#osu_cursor_canvas {position: fixed; pointer-events: none; z-index: 99999999}';
  255. const img = new Image();
  256. img.onload = function() {
  257. const ctx = canvas.getContext('2d');
  258. const half = img.width / 2;
  259. canvas.width = img.width;
  260. canvas.height = img.height;
  261. ctx.drawImage(img, 0, 0);
  262. canvas.id = 'osu_cursor_canvas';
  263. document.body.addEventListener('mousemove', (e) => {
  264. canvas.style.top = (e.clientY - half).toString() + 'px';
  265. canvas.style.left = (e.clientX - half).toString() + 'px';
  266. });
  267. document.body.appendChild(canvas);
  268. };
  269. img.src = OSUCursor;
  270. addStyle(CSSCanvas);
  271. */
  272. }
  273.  
  274. // Just stopPropagation and preventDefault
  275. function destroyEvent(e) {
  276. if (!e) {return false;};
  277. if (!e instanceof Event) {return false;};
  278. e.stopPropagation();
  279. e.preventDefault();
  280. }
  281.  
  282. // Append a style text to document(<head>) with a <style> element
  283. function addStyle(css, id) {
  284. const style = document.createElement("style");
  285. id && (style.id = id);
  286. style.textContent = css;
  287. for (const elm of document.querySelectorAll('#'+id)) {
  288. elm.parentElement && elm.parentElement.removeChild(elm);
  289. }
  290. document.head.appendChild(style);
  291. }
  292. })();