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));
}
})();