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")