Race Result Points Nitro Type

Shows Points earned from the race.

// ==UserScript==
// @name         Race Result Points Nitro Type
// @version      0.1.1
// @description  Shows Points earned from the race.
// @author       Nate Dogg
// @match        *://*.nitrotype.com/race
// @match        *://*.nitrotype.com/race/*
// @icon         
// @grant        none
// @namespace https://greasyfork.org/users/805959
// ==/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)),
	}
})()

/////////////
//  Utils  //
/////////////

const raceContainer = document.getElementById("raceContainer"),
	raceObj = raceContainer ? findReact(raceContainer) : null
if (!raceContainer || !raceObj) {
	logging.error("Init")("Could not find the race track")
	return
}

const currentUserID = raceObj.props.user.userID

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

			const currentUserResult = raceObj.state.racers.find((r) => r.userID === currentUserID)
			if (!currentUserResult || !currentUserResult.progress || typeof currentUserResult.place === "undefined") {
				logging.warn("Finish")("Unable to find race results")
				return
			}

			const listItemTarget = document.querySelector(
				"#raceContainer .gridTable-row.is-self .list .list-item:last-of-type"
			)
			if (!listItemTarget) {
				logging.warn("Finish")("Unable to update user's race result")
				return
			}

			const { typed, skipped, startStamp, completeStamp, errors } = currentUserResult.progress,
				wpm = Math.round((typed - skipped) / 5 / ((completeStamp - startStamp) / 6e4)),
				acc = ((1 - errors / (typed - skipped)) * 100).toFixed(2),
				points = Math.round((100 + wpm / 2) * (1 - errors / typed))

			const listContainer = listItemTarget.parentNode,
				listItemPoints = document.createElement("div")

			listItemPoints.classList.add("list-item")
			listItemPoints.innerHTML = `${points} <span class="${
				listItemTarget.querySelector("span")?.className || "tc-ts"
			}">Points</span>`
			listContainer.insertBefore(listItemPoints, listItemTarget)

			logging.info("Finish")(`Points collected ${points}`)
			break
		}
	}
})

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

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