Race Result Points Nitro Type

Shows Points earned from the race.

  1. // ==UserScript==
  2. // @name Race Result Points Nitro Type
  3. // @version 0.1.1
  4. // @description Shows Points earned from the race.
  5. // @author Nate Dogg
  6. // @match *://*.nitrotype.com/race
  7. // @match *://*.nitrotype.com/race/*
  8. // @icon 
  9. // @grant none
  10. // @namespace https://gf.qytechs.cn/users/805959
  11. // ==/UserScript==
  12.  
  13. /////////////
  14. // Utils //
  15. /////////////
  16.  
  17. /** Finds the React Component from given dom. */
  18. const findReact = (dom, traverseUp = 0) => {
  19. const key = Object.keys(dom).find((key) => key.startsWith("__reactFiber$"))
  20. const domFiber = dom[key]
  21. if (domFiber == null) return null
  22. const getCompFiber = (fiber) => {
  23. let parentFiber = fiber?.return
  24. while (typeof parentFiber?.type == "string") {
  25. parentFiber = parentFiber?.return
  26. }
  27. return parentFiber
  28. }
  29. let compFiber = getCompFiber(domFiber)
  30. for (let i = 0; i < traverseUp && compFiber; i++) {
  31. compFiber = getCompFiber(compFiber)
  32. }
  33. return compFiber?.stateNode
  34. }
  35.  
  36. /** Console logging with some prefixing. */
  37. const logging = (() => {
  38. const logPrefix = (prefix = "") => {
  39. const formatMessage = `%c[Nitro Type Race Result Points]${prefix ? `%c[${prefix}]` : ""}`
  40. let args = [console, `${formatMessage}%c`, "background-color: #D62F3A; color: #fff; font-weight: bold"]
  41. if (prefix) {
  42. args = args.concat("background-color: #4f505e; color: #fff; font-weight: bold")
  43. }
  44. return args.concat("color: unset")
  45. }
  46. return {
  47. info: (prefix) => Function.prototype.bind.apply(console.info, logPrefix(prefix)),
  48. warn: (prefix) => Function.prototype.bind.apply(console.warn, logPrefix(prefix)),
  49. error: (prefix) => Function.prototype.bind.apply(console.error, logPrefix(prefix)),
  50. log: (prefix) => Function.prototype.bind.apply(console.log, logPrefix(prefix)),
  51. debug: (prefix) => Function.prototype.bind.apply(console.debug, logPrefix(prefix)),
  52. }
  53. })()
  54.  
  55. /////////////
  56. // Utils //
  57. /////////////
  58.  
  59. const raceContainer = document.getElementById("raceContainer"),
  60. raceObj = raceContainer ? findReact(raceContainer) : null
  61. if (!raceContainer || !raceObj) {
  62. logging.error("Init")("Could not find the race track")
  63. return
  64. }
  65.  
  66. const currentUserID = raceObj.props.user.userID
  67.  
  68. /** Mutation obverser to track whether results screen showed up. */
  69. const resultObserver = new MutationObserver(([mutation], observer) => {
  70. for (const newNode of mutation.addedNodes) {
  71. if (newNode.classList.contains("race-results")) {
  72. observer.disconnect()
  73.  
  74. const currentUserResult = raceObj.state.racers.find((r) => r.userID === currentUserID)
  75. if (!currentUserResult || !currentUserResult.progress || typeof currentUserResult.place === "undefined") {
  76. logging.warn("Finish")("Unable to find race results")
  77. return
  78. }
  79.  
  80. const listItemTarget = document.querySelector(
  81. "#raceContainer .gridTable-row.is-self .list .list-item:last-of-type"
  82. )
  83. if (!listItemTarget) {
  84. logging.warn("Finish")("Unable to update user's race result")
  85. return
  86. }
  87.  
  88. const { typed, skipped, startStamp, completeStamp, errors } = currentUserResult.progress,
  89. wpm = Math.round((typed - skipped) / 5 / ((completeStamp - startStamp) / 6e4)),
  90. acc = ((1 - errors / (typed - skipped)) * 100).toFixed(2),
  91. points = Math.round((100 + wpm / 2) * (1 - errors / typed))
  92.  
  93. const listContainer = listItemTarget.parentNode,
  94. listItemPoints = document.createElement("div")
  95.  
  96. listItemPoints.classList.add("list-item")
  97. listItemPoints.innerHTML = `${points} <span class="${
  98. listItemTarget.querySelector("span")?.className || "tc-ts"
  99. }">Points</span>`
  100. listContainer.insertBefore(listItemPoints, listItemTarget)
  101.  
  102. logging.info("Finish")(`Points collected ${points}`)
  103. break
  104. }
  105. }
  106. })
  107.  
  108. resultObserver.observe(raceContainer, { childList: true })
  109.  
  110. logging.info("Init")("Race Result listener has been setup")

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址