Nitro Type - Race Result Enhancements

Shows NT Season Points earned and Skipped Characters (Nitros) Used on Race Result screen.

目前为 2022-04-09 提交的版本。查看 最新版本

// ==UserScript==
// @name         Nitro Type - Race Result Enhancements
// @version      0.4.2
// @description  Shows NT Season Points earned and Skipped Characters (Nitros) Used on Race Result screen.
// @author       Toonidy
// @match        *://*.nitrotype.com/race
// @match        *://*.nitrotype.com/race/*
// @icon         
// @grant        none
// @license      MIT
// @namespace    https://gf.qytechs.cn/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 Enhancements]${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, nitros, 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, nitros, skipped }
}

/** 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

/** 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()

			// Setup New Racer Stats Container
			const racers = raceObj.state.racers.slice().sort(sortRacersHandler)

			const dummyCell = document.createElement("div")
			dummyCell.classList.add("split-cell")

			let racerRankNewNodes = []

			const racerRankNodes = newNode.querySelectorAll(".gridTable-row")
			racerRankNodes.forEach((node, i) => {
				const r = racers[i]

				const listRow = node.querySelector(".gridTable-cell:nth-of-type(2) .split"),
					statRow = listRow.querySelector(".split-cell:nth-of-type(2)")

				// Add in the new stat fields
				const { points, skipped } = getUserRaceResult(r),
					accuracyNode = statRow.querySelector(".list .list-item:nth-of-type(2)"),
					suffixClass = accuracyNode?.querySelector("span")?.className || "tc-ts"

				const newStatRow = document.createElement("div")
				newStatRow.className = `${listRow.className} new-stat-row`
				newStatRow.append(dummyCell.cloneNode(), statRow)
				listRow.append(dummyCell.cloneNode())

				listRow.after(newStatRow)

				const skippedNode = document.createElement("div")
				skippedNode.classList.add("list-item", "skipped")
				skippedNode.innerHTML = `${r.robot ? "N/A" : skipped} <span class="${suffixClass}">Skipped</span>`

				const pointsNode = document.createElement("div")
				pointsNode.classList.add("list-item", "points")
				pointsNode.innerHTML = `${r.robot ? "N/A" : points} <span class="${suffixClass}">Points</span>`

				racerRankNewNodes[i] = [skippedNode, pointsNode]

				if (!accuracyNode) {
					if (!node.classList.contains("is-wampus") && !r.disqualified) {
						logging.warn(`Race Result")("Unable to setup new stats on row ${i}`)
					}
					return
				}
				accuracyNode.after(skippedNode, pointsNode)
			})

			/* Track new progress updates */
			server.on("update", (e) => {
				const racers = raceObj.state.racers.slice().sort(sortRacersHandler)
				racerRankNodes.forEach((node, i) => {
					const r = racers[i],
						{ points, skipped } = getUserRaceResult(r),
						[skippedNode, pointsNode] = racerRankNewNodes[i],
						accuracyNode = node.querySelector(".new-stat-row .list .list-item:nth-of-type(2)")

					if (r.disqualified || node.classList.contains("is-wampus")) {
						skippedNode.remove()
						pointsNode.remove()
						return
					}

					skippedNode.childNodes[0].textContent = `${r.robot ? "N/A" : skipped} `
					pointsNode.childNodes[0].textContent = `${r.robot ? "N/A" : points} `

					if (!accuracyNode) {
						logging.warn(`Race Result")("Unable to insert new stats back into row ${i}`)
						return
					}
					accuracyNode.after(skippedNode, pointsNode)
				})
			})
			break
		}
	}
})

resultObserver.observe(raceContainer, { childList: true })

logging.info("Init")("Race Result listener has been setup")

QingJ © 2025

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