您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Displays the number of daily races completed by each member of a Nitro Type team in the team roster table on the Nitro Type team page.
// ==UserScript== // @name Nitro Type - Show Daily Races // @namespace https://github.com/rickstaa/nitro-type-show-daily-races // @version 1.6.1 // @description Displays the number of daily races completed by each member of a Nitro Type team in the team roster table on the Nitro Type team page. // @author Rick Staa // @match *://*.nitrotype.com/team/* // @icon  // @grant none // @license MIT // ==/UserScript== /** * Fetches team stats from NitroType API. * @param {number} teamTAG - The TAG of the team to fetch stats for. * @returns {Promise<Object>} - A promise that resolves to an object containing the team stats. */ const fetchTeamStats = async (teamTAG) => { try { const response = await fetch( `https://www.nitrotype.com/api/v2/teams/${teamTAG}` ); return await response.json(); } catch (error) { console.error(`Error fetching team stats: ${error}`); throw error; } }; /** * Wait for an element to be available in the DOM. * @param {string} selector - The selector to wait for. * @returns {Promise<Element>} - A promise that resolves to the element. */ const waitForElm = (selector) => { return new Promise((resolve) => { const observer = new MutationObserver((mutationsList) => { for (const mutation of mutationsList) { if (mutation.type === "childList" && mutation.addedNodes.length > 0) { const element = document.querySelector(selector); if (element) { observer.disconnect(); resolve(element); } } } }); observer.observe(document.body, { childList: true, subtree: true, }); const element = document.querySelector(selector); if (element) { observer.disconnect(); resolve(element); } }); }; /** * Calculates the number of days a member has been in the group since their join date. * @param {number} joinStamp - The timestamp of the member's join date. * @returns {number} The number of days (including fractions) the member has been in the group. */ const calculateMemberDays = (joinStamp) => { const joinDate = new Date(joinStamp * 1000); const memberDays = Math.abs(Date.now() - joinDate.getTime()) / (1000 * 3600 * 24); return memberDays; }; /** * Userscript entry point. */ (() => { "use strict"; const teamID = window.location.href.split("/")[4]; let originalRows = []; /** * Add the daily races column to the team stats table when the page has loaded. */ window.addEventListener("load", async () => { const teamStatsTable = await waitForElm(".table--teamOverview"); // Retrieve current season information. const seasonInfo = NTGLOBALS.ACTIVE_SEASONS.find((s) => { const now = Date.now(); return now >= s.startStamp * 1e3 && now <= s.endStamp * 1e3; }); const DAYS_SINCE_SEASON_START = Math.abs(Date.now() - seasonInfo.startStamp * 1000) / (1000 * 60 * 60 * 24); // Retrieve team stats. const { results: teamStats } = await fetchTeamStats(teamID); // Calculate all time and season daily races. const dailyMemberRaces = teamStats.members.reduce( (acc, member) => { // Calculate all time daily races. const { displayName, username, joinStamp, played } = member; let memberName = `${displayName || username}`; memberName = memberName.replace(/[\f]+/g, ""); // Remove special characters. const memberDays = Math.max(1, calculateMemberDays(joinStamp)); const allTimeDailyRaces = played / Math.round(memberDays); acc.allTime[memberName] = Math.round(allTimeDailyRaces); // Calculate daily races performed this season for the team. const { played: seasonPlayed = played } = teamStats.season.find( (m) => m.displayName === displayName || m.username === username ) || {}; const seasonMemberDays = Math.max( 1, Math.min(memberDays, DAYS_SINCE_SEASON_START) ); const seasonRaces = Math.min(played, seasonPlayed); const seasonDailyRaces = seasonRaces / Math.round(seasonMemberDays); acc.season[memberName] = Math.round(seasonDailyRaces); return acc; }, { allTime: {}, season: {} } ); /** * Removes the daily races column from the DOM. */ const removeDailyRacesColumn = () => { document .querySelectorAll(".daily-races-header, .daily-races-column-item") .forEach((element) => element.remove()); }; /** * Adds a daily races column to the team stats table and sorts the table by daily races. */ const addDailyRacesColumn = async () => { // Check if daily races column was sorted. const oldDailyRacesHeader = document.querySelector(".daily-races-header"); const wasSorted = oldDailyRacesHeader?.classList.contains("table-filter--asc") || oldDailyRacesHeader?.classList.contains("table-filter--desc"); const sortType = oldDailyRacesHeader?.classList.contains( "table-filter--asc" ) ? "asc" : "desc"; // Remove daily races column if it already exists. await removeDailyRacesColumn(); // Add extra daily races header column. const dailyRacesHeader = document.createElement("th"); dailyRacesHeader.classList.add( "table-cell", "table-cell--lastRace", "table-filter", "daily-races-header" ); if (wasSorted) { dailyRacesHeader.classList.add(`table-filter--${sortType}`); } dailyRacesHeader.innerHTML = "Daily<br>Races"; teamStatsTable.querySelector("thead tr").appendChild(dailyRacesHeader); /** * Sort the daily races column. * @param {string} sortType The sort type to use. Either "asc" or "desc". */ const sortDailyRacesColumn = (sortType) => { if (originalRows.length === 0) { // Store original row order. originalRows = Array.from( teamStatsTable.querySelectorAll("tbody tr") ); } // Sort rows by dailyRaces in descending order. const teamStatsTableRows = Array.from( teamStatsTable.querySelectorAll("tbody tr") ); const sortedRows = teamStatsTableRows .map((row) => { const memberName = row.cells[1] .querySelector("span") .textContent.trim(); const dailyRaces = parseInt( row.cells[dailyRacesHeader.cellIndex].textContent.trim(), 10 ); return { row, memberName, dailyRaces }; }) .sort((a, b) => b.dailyRaces - a.dailyRaces) .map(({ row }) => row); // Revert the order if the sortType is ascending. if (sortType === "asc") { sortedRows.reverse(); dailyRacesHeader.classList.remove("table-filter--desc"); dailyRacesHeader.classList.add("table-filter--asc"); } else { dailyRacesHeader.classList.remove("table-filter--asc"); dailyRacesHeader.classList.add("table-filter--desc"); } // Remove the sort icon from all other columns. Array.from(teamStatsTable.querySelector("thead tr").cells).forEach( (cell) => { if (cell !== dailyRacesHeader) { cell.classList.remove("table-filter--asc", "table-filter--desc"); } } ); // Remove rows and add them back in sorted order. teamStatsTable.querySelector("tbody").append(...sortedRows); // Restore original order if other columns are clicked. Array.from(teamStatsTable.querySelector("thead tr").cells).forEach( (cell) => { if (cell !== dailyRacesHeader) { cell.addEventListener("click", () => { // Remove the sort icon. dailyRacesHeader.classList.remove( "table-filter--asc", "table-filter--desc" ); // Restore original row order. teamStatsTable.querySelector("tbody").append(...originalRows); originalRows = []; }); } } ); }; // Loop through all team members and display the daily races. const teamStatsTableRows = Array.from( teamStatsTable.querySelectorAll("tbody tr") ); teamStatsTableRows.forEach((row) => { // Retrieve member name. const memberName = row.cells[1] .querySelector("span") .textContent.trim() // Add daily races column. const dailyRacesCell = document.createElement("td"); dailyRacesCell.classList.add( "table-cell", "table-cell--lastRace", "daily-races-column-item" ); // Calculate team stats or season stats based on the table type. let dailyRaces; if (teamStatsTable.classList.contains("table--teamSeason")) { dailyRaces = dailyMemberRaces.season[memberName]; } else { dailyRaces = dailyMemberRaces.allTime[memberName]; } dailyRacesCell.textContent = dailyRaces; row.appendChild(dailyRacesCell); }); // If sorted sort the columns. if (wasSorted) { sortDailyRacesColumn(sortType); } // Add a click handler that sorts the team members by daily races. dailyRacesHeader.addEventListener("click", (event) => { const elementClasses = event.target.classList; sortDailyRacesColumn( elementClasses.contains("table-filter--desc") ? "asc" : "desc" ); // Switch sort type. }); }; // Add daily races column. await addDailyRacesColumn(); // Re-load daily races column when the table filter is changed. const tableFilterObserver = new MutationObserver(async () => { await addDailyRacesColumn(); }); // Start observing the element. tableFilterObserver.observe(teamStatsTable, { attributes: true }); }); })();