您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Shows NT Season Points earned and other player's accuracy on Race Result screen.
当前为
// ==UserScript== // @name Nitro Type - Race Result Enhancements // @version 0.1.0 // @description Shows NT Season Points earned and other player's accuracy on Race Result screen. // @author Toonidy // @match *://*.nitrotype.com/race // @match *://*.nitrotype.com/race/* // @icon  // @grant none // @license MIT // @namespace https://greasyfork.org/users/858426 // ==/UserScript== ///////////// // Utils // ///////////// /** Finds the React Component from given dom. */ const findReact = (dom, traverseUp = 0) => { const key = Object.keys(dom).find((key) => key.startsWith("__reactFiber$")) const domFiber = dom[key] if (domFiber == null) return null const getCompFiber = (fiber) => { let parentFiber = fiber?.return while (typeof parentFiber?.type == "string") { parentFiber = parentFiber?.return } return parentFiber } let compFiber = getCompFiber(domFiber) for (let i = 0; i < traverseUp && compFiber; i++) { compFiber = getCompFiber(compFiber) } return compFiber?.stateNode } /** Console logging with some prefixing. */ const logging = (() => { const logPrefix = (prefix = "") => { const formatMessage = `%c[Nitro Type Race Result Points]${prefix ? `%c[${prefix}]` : ""}` let args = [console, `${formatMessage}%c`, "background-color: #D62F3A; color: #fff; font-weight: bold"] if (prefix) { args = args.concat("background-color: #4f505e; color: #fff; font-weight: bold") } return args.concat("color: unset") } return { info: (prefix) => Function.prototype.bind.apply(console.info, logPrefix(prefix)), warn: (prefix) => Function.prototype.bind.apply(console.warn, logPrefix(prefix)), error: (prefix) => Function.prototype.bind.apply(console.error, logPrefix(prefix)), log: (prefix) => Function.prototype.bind.apply(console.log, logPrefix(prefix)), debug: (prefix) => Function.prototype.bind.apply(console.debug, logPrefix(prefix)), } })() /** Calculate User's Race score. */ const getUserRaceResult = (user) => { const { typed, skipped, startStamp, completeStamp, errors } = user.progress let endStamp = completeStamp || Date.now() const wpm = Math.round((typed - skipped) / 5 / ((endStamp - startStamp) / 6e4)), accuracy = ((1 - errors / (typed - skipped)) * 100).toFixed(2), points = Math.round((100 + wpm / 2) * (1 - errors / typed)) return [accuracy, points, wpm] } /** Sort Handler to sort by rank position. */ const sortRacersHandler = (e, t) => { // Source: https://www.nitrotype.com/dist/site/js/ra.js return e.disqualified && !t.disqualified ? 1 : (t.disqualified && !e.disqualified) || (e.progress.completeStamp && !t.progress.completeStamp) ? -1 : t.progress.completeStamp && !e.progress.completeStamp ? 1 : e.progress.completeStamp && t.progress.completeStamp ? e.progress.completeStamp < t.progress.completeStamp ? -1 : 1 : e.progress.percentageFinished === t.progress.percentageFinished ? 0 : e.progress.percentageFinished > t.progress.percentageFinished ? -1 : 1 } /////////////////// // Racing Page // /////////////////// const raceContainer = document.getElementById("raceContainer"), raceObj = raceContainer ? findReact(raceContainer) : null if (!raceContainer || !raceObj) { logging.error("Init")("Could not find the race track") return } const server = raceObj.server, currentUserID = raceObj.props.user.userID let racerStatNodes = null server.on("update", (e) => { if (!racerStatNodes) { return } const racers = raceObj.state.racers.slice().sort(sortRacersHandler) racers.forEach((r, i) => { if (!r || r.robot || r.userID === currentUserID || !racerStatNodes[r.userID]) { return } const statNodes = racerStatNodes[r.userID] statNodes.points.remove() statNodes.accuracy.remove() if (r.disqualified) { return } const [accuracy, points] = getUserRaceResult(r), listRow = document.querySelector(`#raceContainer .gridTable-row:nth-of-type(${i + 1})`), wpmNode = listRow?.querySelector(".list .list-item:nth-of-type(1)"), suffixClass = wpmNode?.querySelector("span")?.className || "tc-ts" statNodes.accuracy.innerHTML = `${accuracy} <span class="${suffixClass}">% Acc</span>` statNodes.points.innerHTML = `${points} <span class="${suffixClass}">Points</span>` wpmNode.after(statNodes.points) wpmNode.before(statNodes.accuracy) }) }) /** Mutation obverser to track whether results screen showed up. */ const resultObserver = new MutationObserver(([mutation], observer) => { for (const newNode of mutation.addedNodes) { if (newNode.classList.contains("race-results")) { observer.disconnect() let newStatNodes = {} const racers = raceObj.state.racers.slice().sort(sortRacersHandler) racers.forEach((r, i) => { if (!r || r.robot) { return } if (!r.progress) { logging.warn("Finish")("Unable to find race results", r) return } const listRow = document.querySelector(`#raceContainer .gridTable-row:nth-of-type(${i + 1})`), wpmNode = listRow?.querySelector( `.list .list-item:nth-of-type(${r.userID === currentUserID ? 2 : 1})` ), suffixClass = wpmNode?.querySelector("span")?.className || "tc-ts" if (!listRow || !wpmNode) { logging.warn("Finish")("Unable to update user's race result") return } const [accuracy, points] = getUserRaceResult(r) let statNodes = { points: document.createElement("div"), accuracy: null, } statNodes.points.classList.add("list-item") statNodes.points.innerHTML = `${points} <span class="${suffixClass}">Points</span>` wpmNode.after(statNodes.points) if (r.userID !== currentUserID) { statNodes.accuracy = document.createElement("div") statNodes.accuracy.classList.add("list-item") statNodes.accuracy.innerHTML = `${accuracy} <span class="${suffixClass}">% Acc</span>` wpmNode.before(statNodes.accuracy) newStatNodes[r.userID] = statNodes } logging.debug("Finish")( `Race Result stats updated for ${r.userID}: ${points} points / ${accuracy} % Acc` ) }) racerStatNodes = newStatNodes break } } }) resultObserver.observe(raceContainer, { childList: true }) logging.info("Init")("Race Result listener has been setup")