您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds customisable quick loadout change buttons on Items page.
// ==UserScript== // @name Torn Loadout Switcher // @namespace https://github.com/SOLiNARY // @version 0.6.1 // @description Adds customisable quick loadout change buttons on Items page. // @author Ramin Quluzade, Silmaril [2665762] // @license MIT // @match https://www.torn.com/item.php* // @icon https://www.google.com/s2/favicons?sz=64&domain=torn.com // @grant unsafeWindow // @grant GM_addStyle // @run-at document-start // ==/UserScript== (async function() { 'use strict'; // Change to 'false' to see only numbers, 'true' to see titles const showTitles = true; const includeLogo = false; const rfcvArg = "rfcv="; const isTampermonkeyEnabled = typeof unsafeWindow !== 'undefined'; const getEquippedItemsUrl = "/page.php?sid=itemsLoadouts&step=getEquippedItems"; let rfcv = localStorage.getItem("silmaril-loadout-switcher-rfcv") ?? null; let rfcvUpdatedThisSession = false; let mutationFound = false; let panelAdded = false; let loadoutTitles = {}; const { fetch: originalFetch } = isTampermonkeyEnabled ? unsafeWindow : window; const customFetch = async (...args) => { let [resource, config] = args; let response = await originalFetch(resource, config); if (rfcvUpdatedThisSession && Object.keys(loadoutTitles).length > 0) { return response; } let fetchUrl = response.url; if (!rfcvUpdatedThisSession){ let rfcvIdx = fetchUrl.indexOf(rfcvArg); if (rfcvIdx >= 0){ rfcv = fetchUrl.substr(rfcvIdx + rfcvArg.length); localStorage.setItem("silmaril-loadout-switcher-rfcv", rfcv); document.querySelectorAll("div.silmaril-torn-loadout-switcher-container button").forEach((button) => button.classList.remove("disabled")); rfcvUpdatedThisSession = true; } } if (Object.keys(loadoutTitles).length == 0){ if (fetchUrl.indexOf(getEquippedItemsUrl) >= 0){ const json = () => response.clone().json() .then((data) => { if (data.currentLoadouts != null){ for (let key in data.currentLoadouts) { if (data.currentLoadouts.hasOwnProperty(key)) { loadoutTitles[key] = data.currentLoadouts[key].title; } } } return data }) response.json = json; response.text = async () =>JSON.stringify(await json()); } } return response; }; if (isTampermonkeyEnabled){ unsafeWindow.fetch = customFetch; } else { window.fetch = customFetch; } const styles = ` div#loadoutsRoot p[class^=title___] { overflow-y: hidden; overflow-x: auto; } div.silmaril-torn-loadout-switcher-container { display: inline-flex; align-items: center; margin-left: 5px; } div.silmaril-torn-loadout-switcher-container a img { display: flex; height: 50px; flex-direction: row; align-content: stretch; justify-content: space-around; align-items: flex-start; } .wave-animation { position: relative; overflow: hidden; } .wave { pointer-events: none; position: absolute; width: 100%; height: 33px; background-color: transparent; opacity: 0; transform: translateX(-100%); animation: waveAnimation 3s cubic-bezier(0, 0, 0, 1); } @media (max-width: 768px) { div[class^=main___] > div[class^=content___] { margin-top: 10px; } } @keyframes waveAnimation { 0% { opacity: 1; transform: translateX(-100%); } 100% { opacity: 0; transform: translateX(100%); } } `; if (isTampermonkeyEnabled){ GM_addStyle(styles); } else { let style = document.createElement("style"); style.type = "text/css"; style.innerHTML = styles; while (document.head == null){ await sleep(50); } document.head.appendChild(style); } const setLoadoutUrl = "/page.php?sid=itemsLoadouts&step=changeLoadout&setID={loadoutId}&rfcv={rfcv}"; let selectedLoadouts = localStorage.getItem("silmaril-loadout-switcher-selected-loadouts") ?? "1,2,3"; let selectedLoadoutsArray = selectedLoadouts.split(','); const observerTarget = document.querySelector("html"); const observerConfig = { attributes: false, childList: true, characterData: false, subtree: true }; const observer = new MutationObserver(function(mutations) { mutations.forEach(function(mutationItem) { if (mutationFound || panelAdded){ observer.disconnect(); return; } let mutation = mutationItem.target; if (mutation.classList.contains("title___nIMRx")) { mutationFound = true; observer.disconnect(); const buttonContainer = document.createElement('div'); buttonContainer.className = 'silmaril-torn-loadout-switcher-container'; const waveDiv = document.createElement('div'); waveDiv.className = 'wave'; buttonContainer.appendChild(waveDiv); addLoadoutAndSettingButtons(buttonContainer); addLogo(buttonContainer); if (!panelAdded){ mutation.appendChild(buttonContainer); panelAdded = true; } } }); }); observer.observe(observerTarget, observerConfig); function addLogo(root){ if (!includeLogo){ return; } const logoLink = document.createElement('a'); logoLink.href = '/factions.php?step=profile&ID=6731'; logoLink.target = '_blank'; const logoImg = document.createElement('img'); logoImg.src = ''; logoImg.alt = 'Next Level logo'; logoLink.appendChild(logoImg); root.appendChild(logoLink); } function addLoadoutAndSettingButtons(root){ addLoadoutButtons(root); const settings = document.createElement('button'); settings.type = 'button'; settings.title = 'Settings'; settings.className = 'torn-btn'; settings.textContent = '⚙'; settings.addEventListener('click', () => { let userInput = prompt("Please, enter which loadouts from 1 to 9 you want to see, comma-separated (default: 1,2,3):", selectedLoadouts); let wave = root.querySelector("div.wave"); if (userInput !== null && userInput.length > 0) { localStorage.setItem("silmaril-loadout-switcher-selected-loadouts", userInput); selectedLoadouts = userInput; selectedLoadoutsArray = selectedLoadouts.split(','); root.querySelectorAll("button, a").forEach((item) => item.remove()); addLoadoutAndSettingButtons(root); addLogo(root); wave.style.backgroundColor = "green"; } else { wave.style.animationDuration = "3s"; wave.style.backgroundColor = "yellow"; console.error("[TornLoadoutSwitcher] User cancelled input of selected loadouts."); } wave.style.animation = 'none'; wave.offsetHeight; wave.style.animation = null; }); root.appendChild(settings); } async function addLoadoutButtons(root){ selectedLoadoutsArray.forEach((loadout) => { const button = document.createElement('button'); button.type = 'button'; button.title = showTitles ? loadout : loadoutTitles[loadout] ?? ''; button.className = rfcv === null ? 'torn-btn disabled' : 'torn-btn'; button.textContent = showTitles ? loadoutTitles[loadout] : loadout; button.setAttribute('data-loadout-number', loadout); button.addEventListener('click', () => {handleLoadoutClick(root)}); root.appendChild(button); }) } async function handleLoadoutClick(root){ let loadout = event.target.getAttribute('data-loadout-number'); if (event.target.classList.contains('disabled')){ return; } let url = setLoadoutUrl.replace("{loadoutId}", loadout).replace("{rfcv}", rfcv); await sendSetLoadoutRequest(url, root); } async function sendSetLoadoutRequest(url, root){ let wave = root.querySelector("div.wave"); await fetch(url, { method: 'GET', }) .then(response => { if (response.ok) { wave.style.backgroundColor = "green"; } else { console.error("[TornLoadoutSwitcher] Set Loadout request failed:", response); wave.style.backgroundColor = "red"; wave.style.animationDuration = "5s"; } }) .catch(error => { console.error("[TornLoadoutSwitcher] Error setting loadout:", error); wave.style.backgroundColor = "red"; wave.style.animationDuration = "5s"; }); wave.style.animation = 'none'; wave.offsetHeight; wave.style.animation = null; } function sleep(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } })();