ShowSpinnerCount (Greasemonkey Ver)

Adds spinner count to beatmap pages

  1. // ==UserScript==
  2. // @name ShowSpinnerCount (Greasemonkey Ver)
  3. // @namespace osu
  4. // @version 1.0.1
  5. // @description Adds spinner count to beatmap pages
  6. // @author Magnus Cosmos
  7. // @match https://osu.ppy.sh/*
  8. // @match https://lazer.ppy.sh/*
  9. // @run-at document-start
  10. // ==/UserScript==
  11.  
  12. const countSpinnerSvg = `"url(data:image/svg+xml;base64,PHN2ZyB4bWxuczpkYz0iaHR0cDovL3B1cmwub3JnL2RjL2VsZW1lbnRzLzEuMS8iIHhtbG5zOmNjPSJodHRwOi8vY3JlYXRpdmVjb21tb25zLm9yZy9ucyMiIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyIgeG1sbnM6c3ZnPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB2aWV3Qm94PSIwIDAgNTYyLjUgNTYyLjUiIGhlaWdodD0iNTYyLjUiIHdpZHRoPSI1NjIuNSIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIgdmVyc2lvbj0iMS4xIiBpZD0ic3ZnNDE1NSI+PG1ldGFkYXRhIGlkPSJtZXRhZGF0YTQxNjEiPjxyZGY6UkRGPjxjYzpXb3JrIHJkZjphYm91dD0iIj48ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD48ZGM6dHlwZSByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIi8+PGRjOnRpdGxlLz48L2NjOldvcms+PC9yZGY6UkRGPjwvbWV0YWRhdGE+PGRlZnMgaWQ9ImRlZnM0MTU5Ii8+PGcgdHJhbnNmb3JtPSJtYXRyaXgoMS4yNSwwLDAsLTEuMjUsMCw1NjIuNSkiIGlkPSJnNDE2MyI+PGcgaWQ9Imc0MTY1Ii8+PGcgaWQ9Imc0MTY3Ij48cGF0aCBpZD0icGF0aDQxNjkiIHN0eWxlPSJmaWxsOiM0NDExODg7ZmlsbC1vcGFjaXR5OjE7ZmlsbC1ydWxlOmV2ZW5vZGQ7c3Ryb2tlOm5vbmUiIGQ9Im0gNDEwLjg2MzEsMTQ1LjY5OCBjIDQzLjc5NzIsNDMuNzk3MyA0My43OTcyLDExNC44MDY3IDAsMTU4LjYwNCAwLDAgLTEwNi41NjExLDEwNi41NjExIC0xMDYuNTYxMSwxMDYuNTYxMSAtNDMuNzk3Myw0My43OTcyIC0xMTQuODA2Nyw0My43OTcyIC0xNTguNjA0LDAgMCwwIC0xMDYuNTYxMDksLTEwNi41NjExIC0xMDYuNTYxMDksLTEwNi41NjExIC00My43OTcyNTksLTQzLjc5NzMgLTQzLjc5NzI1OSwtMTE0LjgwNjcgMCwtMTU4LjYwNCAwLDAgMTA2LjU2MTA5LC0xMDYuNTYxMDkgMTA2LjU2MTA5LC0xMDYuNTYxMDkgNDMuNzk3MywtNDMuNzk3MjU5IDExNC44MDY3LC00My43OTcyNTkgMTU4LjYwNCwwIDAsMCAxMDYuNTYxMSwxMDYuNTYxMDkgMTA2LjU2MTEsMTA2LjU2MTA5IHoiLz48cGF0aCBpZD0icGF0aDQxNzEiIHN0eWxlPSJmaWxsOiNmZmRkNTU7ZmlsbC1vcGFjaXR5OjE7ZmlsbC1ydWxlOm5vbnplcm87c3Ryb2tlOm5vbmUiIGQ9Im0gMzMxLjI1LDIyNSBjIDAsLTU4LjU5MzggLTQ3LjY1NjMsLTEwNi4yNSAtMTA2LjI1LC0xMDYuMjUgLTU4LjU5MzgsMCAtMTA2LjI1LDQ3LjY1NjIgLTEwNi4yNSwxMDYuMjUgMCw1OC41OTM4IDQ3LjY1NjIsMTA2LjI1IDEwNi4yNSwxMDYuMjUgNTguNTkzNywwIDEwNi4yNSwtNDcuNjU2MiAxMDYuMjUsLTEwNi4yNSB6IE0gMzc1LDIyNSBDIDM3NSwzMDcuODEyNSAzMDcuODEyNSwzNzUgMjI1LDM3NSAxNDIuMTg3NSwzNzUgNzUsMzA3LjgxMjUgNzUsMjI1IDc1LDE0Mi4xODc1IDE0Mi4xODc1LDc1IDIyNSw3NSBjIDgyLjgxMjUsMCAxNTAsNjcuMTg3NSAxNTAsMTUwIHoiLz48ZyB0cmFuc2Zvcm09Im1hdHJpeCgwLjc1LDAsMCwtMC43NSwwLDQ1MCkiIGlkPSJnNDE3MyI+PHBhdGggaWQ9InBhdGg0MTc1IiBzdHlsZT0iZmlsbDogI2ZmZGQ1NTsiIGQ9Im0gMzAwLDIxOC43NTk4IGMgNDQuODY3NywwIDgxLjI0MDIsMzYuMzcyNSA4MS4yNDAyLDgxLjI0MDIgMCw0NC44Njc3IC0zNi4zNzI1LDgxLjI0MDIgLTgxLjI0MDIsODEuMjQwMiAtNDQuODY3NywwIC04MS4yNDAyLC0zNi4zNzI1IC04MS4yNDAyLC04MS4yNDAyIDAsLTQ0Ljg2NzcgMzYuMzcyNSwtODEuMjQwMiA4MS4yNDAyLC04MS4yNDAyIHoiLz48L2c+PC9nPjwvZz48L3N2Zz4=)"`;
  13.  
  14. window.eval(`
  15. function getReact() {
  16. return new Promise((resolve) => {
  17. const defineProperty = Object.defineProperty;
  18. Object.defineProperty = function () {
  19. defineProperty.apply(this, arguments);
  20. const prop = arguments[1];
  21. const descriptor = arguments[2];
  22. if (descriptor.get && descriptor.get.a) {
  23. if ("createElement" in descriptor.get.a) {
  24. Object.defineProperty = defineProperty;
  25. resolve(descriptor.get.a);
  26. }
  27. }
  28. }
  29. });
  30. }
  31.  
  32. getReact().then((react) => {
  33. const createElement = react.createElement;
  34. react.createElement = function() {
  35. const r = createElement.apply(this, arguments);
  36. if (arguments[1]?.className === "beatmap-basic-stats") {
  37. const countSpinners = r._owner.stateNode.renderEntry("count_spinners");
  38. countSpinners.props.children[0].props.style.backgroundImage = ${countSpinnerSvg};
  39. countSpinners.props.title = "Spinner Count";
  40. r.props.children.push(countSpinners);
  41. }
  42. return r;
  43. }
  44. });`);
  45.  
  46. function addStyle(css) {
  47. const head = document.querySelector("head");
  48. if (head) {
  49. const style = document.createElement("style");
  50. style.type = "text/css";
  51. style.innerHTML = css.replace(/;/g, " !important;");
  52. head.appendChild(style);
  53. }
  54. }
  55.  
  56. addStyle(`
  57. @media (min-width: 900px) {
  58. .beatmapset-header {
  59. grid-template-columns: 1fr 320px;
  60. }
  61. .beatmapset-info {
  62. grid-template-columns: 1fr 175px 320px;
  63. }
  64. }`);