Provide you with the task of the mission before accepting, and sometimes also an additional hint.
// ==UserScript==
// @name TORN: TornTools - Mission Hints
// @namespace torntools.mission-hints
// @version 1.0.0
// @author DeKleineKobini [2114440] and the TornTools team
// @description Provide you with the task of the mission before accepting, and sometimes also an additional hint.
// @license GPL-3.0-or-later
// @icon https://www.google.com/s2/favicons?sz=64&domain=torn.com
// @supportURL https://github.com/Mephiles/torntools_extension/issues
// @match https://*.torn.com/page.php?sid=missions*
// @grant GM.getValue
// @grant GM.info
// @grant GM.setValue
// @grant GM_addStyle
// @grant unsafeWindow
// @run-at document-end
// @contributionURL https://buymeacoffee.com/dekleinekobini
// ==/UserScript==
(function() {
"use strict";
var s = new Set();
var _css = async (t) => {
if (s.has(t)) return;
s.add(t);
((c) => {
if (typeof GM_addStyle === "function") GM_addStyle(c);
else (document.head || document.documentElement).appendChild(document.createElement("style")).append(c);
})(t);
};
var FEATURE_MANAGER;
var ttStorage;
var SCRIPT_INJECTOR;
var RUNTIME_INFORMATION;
var RUNTIME_STORAGE;
function setFeatureManager(featureManager) {
FEATURE_MANAGER = featureManager;
}
function setTTStorage(storage) {
ttStorage = storage;
}
function setScriptInjector(scriptInjector) {
SCRIPT_INJECTOR = scriptInjector;
}
function setRuntimeInformation(runtimeInformation) {
RUNTIME_INFORMATION = runtimeInformation;
}
function setRuntimeStorage(runtimeStorage) {
RUNTIME_STORAGE = runtimeStorage;
}
_css(".tt-loading-placeholder{content:var(--default-preloader-url,url(https://www.torn.com/images/v2/main/ajax-loader.gif));margin:0 auto;padding:10px;display:none}.tt-loading-placeholder.active{display:block}");
(() => {
if (typeof window === "undefined" || window.location.href.endsWith("/_generated_background_page.html")) return "BACKGROUND";
else if (typeof browser === "object" && browser.action) return "POPUP";
else return "CONTENT";
})();
function sleep(millis) {
return new Promise((resolve) => setTimeout(resolve, millis));
}
function isIntNumber(number) {
if (number === null) return false;
if (number.match(/[a-zA-Z]/)) return false;
const _number = parseFloat(number.toString());
return !Number.isNaN(_number) && Number.isFinite(_number) && _number % 1 === 0;
}
function toNumericVersion(version) {
return parseInt(version.split(".").map((part) => part.padStart(3, "0")).join("").padEnd(9, "9"));
}
function isTabFocused() {
return document.hasFocus();
}
function svgImport(svgImport) {
if (typeof svgImport !== "string") return (attributes = {}) => createFallbackElement(attributes);
if (svgImport.startsWith("data:image/svg+xml")) {
const encodedData = svgImport.substring(19);
let svgContent;
try {
svgContent = decodeURIComponent(encodedData);
} catch (error) {
console.error("Failed to decode SVG data URL", error);
return (attributes = {}) => createFallbackElement(attributes);
}
return (attributes = {}) => createSvgElement(svgContent, attributes);
}
return (attributes = {}) => createSvgElement(svgImport, attributes);
}
function createFallbackElement(attributes) {
const svgNS = "http://www.w3.org/2000/svg";
const svg = document.createElementNS(svgNS, "svg");
svg.setAttribute("width", "24");
svg.setAttribute("height", "24");
svg.setAttribute("viewBox", "0 0 24 24");
Object.entries(attributes).filter(([, value]) => value !== false && value !== null && value !== void 0).map(([key, value]) => svg.setAttribute(key, String(value)));
const rect = document.createElementNS(svgNS, "rect");
rect.setAttribute("x", "0");
rect.setAttribute("y", "0");
rect.setAttribute("width", "24");
rect.setAttribute("height", "24");
rect.setAttribute("fill", "red");
svg.appendChild(rect);
return svg;
}
function createSvgElement(svgContent, attributes = {}) {
const fullAttributes = {
width: "size" in attributes ? attributes.size : "1em",
height: "size" in attributes ? attributes.size : "1em",
...attributes
};
const svg = elementBuilder({
type: "template",
html: svgContent.trim()
}).content.firstChild;
if (!isSVGElement(svg)) return createFallbackElement(fullAttributes);
Object.entries(fullAttributes).filter(([, value]) => value !== false && value !== null && value !== void 0).forEach(([key, value]) => svg.setAttribute(key, String(value)));
return svg;
}
var arrow_bend_up_left_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M236,200a12,12,0,0,1-24,0,84.09,84.09,0,0,0-84-84H61l27.52,27.51a12,12,0,0,1-17,17l-48-48a12,12,0,0,1,0-17l48-48a12,12,0,0,1,17,17L61,92h67A108.12,108.12,0,0,1,236,200Z'/%3e%3c/svg%3e";
var arrow_clockwise_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M244,56v48a12,12,0,0,1-12,12H184a12,12,0,1,1,0-24H201.1l-19-17.38c-.13-.12-.26-.24-.38-.37A76,76,0,1,0,127,204h1a75.53,75.53,0,0,0,52.15-20.72,12,12,0,0,1,16.49,17.45A99.45,99.45,0,0,1,128,228h-1.37A100,100,0,1,1,198.51,57.06L220,76.72V56a12,12,0,0,1,24,0Z'/%3e%3c/svg%3e";
var arrow_down_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M208.49,152.49l-72,72a12,12,0,0,1-17,0l-72-72a12,12,0,0,1,17-17L116,187V40a12,12,0,0,1,24,0V187l51.51-51.52a12,12,0,0,1,17,17Z'/%3e%3c/svg%3e";
var arrow_up_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M208.49,120.49a12,12,0,0,1-17,0L140,69V216a12,12,0,0,1-24,0V69L64.49,120.49a12,12,0,0,1-17-17l72-72a12,12,0,0,1,17,0l72,72A12,12,0,0,1,208.49,120.49Z'/%3e%3c/svg%3e";
var check_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M232.49,80.49l-128,128a12,12,0,0,1-17,0l-56-56a12,12,0,1,1,17-17L96,183,215.51,63.51a12,12,0,0,1,17,17Z'/%3e%3c/svg%3e";
var check_circle_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M176.49,95.51a12,12,0,0,1,0,17l-56,56a12,12,0,0,1-17,0l-24-24a12,12,0,1,1,17-17L112,143l47.51-47.52A12,12,0,0,1,176.49,95.51ZM236,128A108,108,0,1,1,128,20,108.12,108.12,0,0,1,236,128Zm-24,0a84,84,0,1,0-84,84A84.09,84.09,0,0,0,212,128Z'/%3e%3c/svg%3e";
var copy_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M216,28H88A12,12,0,0,0,76,40V76H40A12,12,0,0,0,28,88V216a12,12,0,0,0,12,12H168a12,12,0,0,0,12-12V180h36a12,12,0,0,0,12-12V40A12,12,0,0,0,216,28ZM156,204H52V100H156Zm48-48H180V88a12,12,0,0,0-12-12H100V52H204Z'/%3e%3c/svg%3e";
var info_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M108,84a16,16,0,1,1,16,16A16,16,0,0,1,108,84Zm128,44A108,108,0,1,1,128,20,108.12,108.12,0,0,1,236,128Zm-24,0a84,84,0,1,0-84,84A84.09,84.09,0,0,0,212,128Zm-72,36.68V132a20,20,0,0,0-20-20,12,12,0,0,0-4,23.32V168a20,20,0,0,0,20,20,12,12,0,0,0,4-23.32Z'/%3e%3c/svg%3e";
var spinner_gap_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M140,32V64a12,12,0,0,1-24,0V32a12,12,0,0,1,24,0Zm84,84H192a12,12,0,0,0,0,24h32a12,12,0,0,0,0-24Zm-42.26,48.77a12,12,0,1,0-17,17l22.63,22.63a12,12,0,0,0,17-17ZM128,180a12,12,0,0,0-12,12v32a12,12,0,0,0,24,0V192A12,12,0,0,0,128,180ZM74.26,164.77,51.63,187.4a12,12,0,0,0,17,17l22.63-22.63a12,12,0,1,0-17-17ZM76,128a12,12,0,0,0-12-12H32a12,12,0,0,0,0,24H64A12,12,0,0,0,76,128ZM68.6,51.63a12,12,0,1,0-17,17L74.26,91.23a12,12,0,0,0,17-17Z'/%3e%3c/svg%3e";
var warning_circle_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M128,20A108,108,0,1,0,236,128,108.12,108.12,0,0,0,128,20Zm0,192a84,84,0,1,1,84-84A84.09,84.09,0,0,1,128,212Zm-12-80V80a12,12,0,0,1,24,0v52a12,12,0,0,1-24,0Zm28,40a16,16,0,1,1-16-16A16,16,0,0,1,144,172Z'/%3e%3c/svg%3e";
var x_circle_bold_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M168.49,104.49,145,128l23.52,23.51a12,12,0,0,1-17,17L128,145l-23.51,23.52a12,12,0,0,1-17-17L111,128,87.51,104.49a12,12,0,0,1,17-17L128,111l23.51-23.52a12,12,0,0,1,17,17ZM236,128A108,108,0,1,1,128,20,108.12,108.12,0,0,1,236,128Zm-24,0a84,84,0,1,0-84,84A84.09,84.09,0,0,0,212,128Z'/%3e%3c/svg%3e";
var airplane_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M240,136v32a8,8,0,0,1-8,8,7.61,7.61,0,0,1-1.57-.16L156,161v23.73l17.66,17.65A8,8,0,0,1,176,208v24a8,8,0,0,1-11,7.43l-37-14.81L91,239.43A8,8,0,0,1,80,232V208a8,8,0,0,1,2.34-5.66L100,184.69V161L25.57,175.84A7.61,7.61,0,0,1,24,176a8,8,0,0,1-8-8V136a8,8,0,0,1,4.42-7.16L100,89.06V44a28,28,0,0,1,56,0V89.06l79.58,39.78A8,8,0,0,1,240,136Z'/%3e%3c/svg%3e";
var arrows_out_cardinal_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M96,136H64v24a8,8,0,0,1-13.66,5.66l-32-32a8,8,0,0,1,0-11.32l32-32A8,8,0,0,1,64,96v24H96a8,8,0,0,1,0,16Zm0-72h24V96a8,8,0,0,0,16,0V64h24a8,8,0,0,0,5.66-13.66l-32-32a8,8,0,0,0-11.32,0l-32,32A8,8,0,0,0,96,64Zm141.66,58.34-32-32A8,8,0,0,0,192,96v24H160a8,8,0,0,0,0,16h32v24a8,8,0,0,0,13.66,5.66l32-32A8,8,0,0,0,237.66,122.34ZM160,192H136V160a8,8,0,0,0-16,0v32H96a8,8,0,0,0-5.66,13.66l32,32a8,8,0,0,0,11.32,0l32-32A8,8,0,0,0,160,192Z'/%3e%3c/svg%3e";
var bell_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M221.8,175.94C216.25,166.38,208,139.33,208,104a80,80,0,1,0-160,0c0,35.34-8.26,62.38-13.81,71.94A16,16,0,0,0,48,200H88.81a40,40,0,0,0,78.38,0H208a16,16,0,0,0,13.8-24.06ZM128,216a24,24,0,0,1-22.62-16h45.24A24,24,0,0,1,128,216Z'/%3e%3c/svg%3e";
var bell_slash_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M221.84,192v0a1.85,1.85,0,0,1-3,.28L83.27,43.19a4,4,0,0,1,.8-6A79.55,79.55,0,0,1,129.17,24C173,24.66,207.8,61.1,208,104.92c.14,34.88,8.31,61.54,13.82,71A15.89,15.89,0,0,1,221.84,192Zm-7.92,18.62a8,8,0,0,1-11.85,10.76L182.62,200H167.16a40,40,0,0,1-78.41,0H47.91a15.78,15.78,0,0,1-13.59-7.59,16.42,16.42,0,0,1-.09-16.68c5.55-9.73,13.7-36.64,13.7-71.73A79.42,79.42,0,0,1,58.79,63.85L42,45.38A8,8,0,1,1,53.84,34.62ZM150.59,200H105.32a24,24,0,0,0,45.27,0Z'/%3e%3c/svg%3e";
var caret_down_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M213.66,101.66l-80,80a8,8,0,0,1-11.32,0l-80-80A8,8,0,0,1,48,88H208a8,8,0,0,1,5.66,13.66Z'/%3e%3c/svg%3e";
var caret_right_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M181.66,133.66l-80,80A8,8,0,0,1,88,208V48a8,8,0,0,1,13.66-5.66l80,80A8,8,0,0,1,181.66,133.66Z'/%3e%3c/svg%3e";
var caret_up_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M215.39,163.06A8,8,0,0,1,208,168H48a8,8,0,0,1-5.66-13.66l80-80a8,8,0,0,1,11.32,0l80,80A8,8,0,0,1,215.39,163.06Z'/%3e%3c/svg%3e";
var funnel_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M227.81,66.76l-.08.09L160,139.17v55.49A16,16,0,0,1,152.87,208l-32,21.34A16,16,0,0,1,96,216V139.17L28.27,66.85l-.08-.09A16,16,0,0,1,40,40H216a16,16,0,0,1,11.84,26.76Z'/%3e%3c/svg%3e";
var funnel_x_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M227.73,66.85,160,139.17v55.49A16,16,0,0,1,152.87,208l-32,21.34A16,16,0,0,1,96,216V139.17L28.27,66.85l-.08-.09A16,16,0,0,1,40,40H216a16,16,0,0,1,11.84,26.76ZM227.31,192l18.35-18.34a8,8,0,0,0-11.32-11.32L216,180.69l-18.34-18.35a8,8,0,0,0-11.32,11.32L204.69,192l-18.35,18.34a8,8,0,0,0,11.32,11.32L216,203.31l18.34,18.35a8,8,0,0,0,11.32-11.32Z'/%3e%3c/svg%3e";
var gear_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M216,130.16q.06-2.16,0-4.32l14.92-18.64a8,8,0,0,0,1.48-7.06,107.6,107.6,0,0,0-10.88-26.25,8,8,0,0,0-6-3.93l-23.72-2.64q-1.48-1.56-3-3L186,40.54a8,8,0,0,0-3.94-6,107.29,107.29,0,0,0-26.25-10.86,8,8,0,0,0-7.06,1.48L130.16,40Q128,40,125.84,40L107.2,25.11a8,8,0,0,0-7.06-1.48A107.6,107.6,0,0,0,73.89,34.51a8,8,0,0,0-3.93,6L67.32,64.27q-1.56,1.49-3,3L40.54,70a8,8,0,0,0-6,3.94,107.71,107.71,0,0,0-10.87,26.25,8,8,0,0,0,1.49,7.06L40,125.84Q40,128,40,130.16L25.11,148.8a8,8,0,0,0-1.48,7.06,107.6,107.6,0,0,0,10.88,26.25,8,8,0,0,0,6,3.93l23.72,2.64q1.49,1.56,3,3L70,215.46a8,8,0,0,0,3.94,6,107.71,107.71,0,0,0,26.25,10.87,8,8,0,0,0,7.06-1.49L125.84,216q2.16.06,4.32,0l18.64,14.92a8,8,0,0,0,7.06,1.48,107.21,107.21,0,0,0,26.25-10.88,8,8,0,0,0,3.93-6l2.64-23.72q1.56-1.48,3-3L215.46,186a8,8,0,0,0,6-3.94,107.71,107.71,0,0,0,10.87-26.25,8,8,0,0,0-1.49-7.06ZM128,168a40,40,0,1,1,40-40A40,40,0,0,1,128,168Z'/%3e%3c/svg%3e";
var info_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24Zm-4,48a12,12,0,1,1-12,12A12,12,0,0,1,124,72Zm12,112a16,16,0,0,1-16-16V128a8,8,0,0,1,0-16,16,16,0,0,1,16,16v40a8,8,0,0,1,0,16Z'/%3e%3c/svg%3e";
var plus_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M208,32H48A16,16,0,0,0,32,48V208a16,16,0,0,0,16,16H208a16,16,0,0,0,16-16V48A16,16,0,0,0,208,32ZM184,136H136v48a8,8,0,0,1-16,0V136H72a8,8,0,0,1,0-16h48V72a8,8,0,0,1,16,0v48h48a8,8,0,0,1,0,16Z'/%3e%3c/svg%3e";
var stethoscope_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M240,160a32,32,0,1,0-39.93,31,8,8,0,0,0-.07,1,32,32,0,0,1-32,32H144a32,32,0,0,1-32-32V151.48c31.47-4,56-31.47,56-64.31V40a8,8,0,0,0-8-8H136a8,8,0,0,0,0,16h16V87.17c0,26.58-21.25,48.49-47.36,48.83A48,48,0,0,1,56,88V48H72a8,8,0,0,0,0-16H48a8,8,0,0,0-8,8V88a64,64,0,0,0,56,63.49V192a48.05,48.05,0,0,0,48,48h24a48.05,48.05,0,0,0,48-48,8,8,0,0,0-.07-1A32,32,0,0,0,240,160Zm-32,8a8,8,0,1,1,8-8A8,8,0,0,1,208,168Z'/%3e%3c/svg%3e";
var table_fill_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M224,48H32a8,8,0,0,0-8,8V192a16,16,0,0,0,16,16H216a16,16,0,0,0,16-16V56A8,8,0,0,0,224,48ZM40,112H80v32H40Zm56,0H216v32H96ZM40,160H80v32H40Zm176,32H96V160H216v32Z'/%3e%3c/svg%3e";
var caret_down_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M213.66,101.66l-80,80a8,8,0,0,1-11.32,0l-80-80A8,8,0,0,1,53.66,90.34L128,164.69l74.34-74.35a8,8,0,0,1,11.32,11.32Z'/%3e%3c/svg%3e";
var eye_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M247.31,124.76c-.35-.79-8.82-19.58-27.65-38.41C194.57,61.26,162.88,48,128,48S61.43,61.26,36.34,86.35C17.51,105.18,9,124,8.69,124.76a8,8,0,0,0,0,6.5c.35.79,8.82,19.57,27.65,38.4C61.43,194.74,93.12,208,128,208s66.57-13.26,91.66-38.34c18.83-18.83,27.3-37.61,27.65-38.4A8,8,0,0,0,247.31,124.76ZM128,192c-30.78,0-57.67-11.19-79.93-33.25A133.47,133.47,0,0,1,25,128,133.33,133.33,0,0,1,48.07,97.25C70.33,75.19,97.22,64,128,64s57.67,11.19,79.93,33.25A133.46,133.46,0,0,1,231.05,128C223.84,141.46,192.43,192,128,192Zm0-112a48,48,0,1,0,48,48A48.05,48.05,0,0,0,128,80Zm0,80a32,32,0,1,1,32-32A32,32,0,0,1,128,160Z'/%3e%3c/svg%3e";
var eye_slash_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M53.92,34.62A8,8,0,1,0,42.08,45.38L61.32,66.55C25,88.84,9.38,123.2,8.69,124.76a8,8,0,0,0,0,6.5c.35.79,8.82,19.57,27.65,38.4C61.43,194.74,93.12,208,128,208a127.11,127.11,0,0,0,52.07-10.83l22,24.21a8,8,0,1,0,11.84-10.76Zm47.33,75.84,41.67,45.85a32,32,0,0,1-41.67-45.85ZM128,192c-30.78,0-57.67-11.19-79.93-33.25A133.16,133.16,0,0,1,25,128c4.69-8.79,19.66-33.39,47.35-49.38l18,19.75a48,48,0,0,0,63.66,70l14.73,16.2A112,112,0,0,1,128,192Zm6-95.43a8,8,0,0,1,3-15.72,48.16,48.16,0,0,1,38.77,42.64,8,8,0,0,1-7.22,8.71,6.39,6.39,0,0,1-.75,0,8,8,0,0,1-8-7.26A32.09,32.09,0,0,0,134,96.57Zm113.28,34.69c-.42.94-10.55,23.37-33.36,43.8a8,8,0,1,1-10.67-11.92A132.77,132.77,0,0,0,231.05,128a133.15,133.15,0,0,0-23.12-30.77C185.67,75.19,158.78,64,128,64a118.37,118.37,0,0,0-19.36,1.57A8,8,0,1,1,106,49.79,134,134,0,0,1,128,48c34.88,0,66.57,13.26,91.66,38.35,18.83,18.83,27.3,37.62,27.65,38.41A8,8,0,0,1,247.31,131.26Z'/%3e%3c/svg%3e";
var plus_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M224,128a8,8,0,0,1-8,8H136v80a8,8,0,0,1-16,0V136H40a8,8,0,0,1,0-16h80V40a8,8,0,0,1,16,0v80h80A8,8,0,0,1,224,128Z'/%3e%3c/svg%3e";
var question_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M140,180a12,12,0,1,1-12-12A12,12,0,0,1,140,180ZM128,72c-22.06,0-40,16.15-40,36v4a8,8,0,0,0,16,0v-4c0-11,10.77-20,24-20s24,9,24,20-10.77,20-24,20a8,8,0,0,0-8,8v8a8,8,0,0,0,16,0v-.72c18.24-3.35,32-17.9,32-35.28C168,88.15,150.06,72,128,72Zm104,56A104,104,0,1,1,128,24,104.11,104.11,0,0,1,232,128Zm-16,0a88,88,0,1,0-88,88A88.1,88.1,0,0,0,216,128Z'/%3e%3c/svg%3e";
var trash_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M216,48H176V40a24,24,0,0,0-24-24H104A24,24,0,0,0,80,40v8H40a8,8,0,0,0,0,16h8V208a16,16,0,0,0,16,16H192a16,16,0,0,0,16-16V64h8a8,8,0,0,0,0-16ZM96,40a8,8,0,0,1,8-8h48a8,8,0,0,1,8,8v8H96Zm96,168H64V64H192ZM112,104v64a8,8,0,0,1-16,0V104a8,8,0,0,1,16,0Zm48,0v64a8,8,0,0,1-16,0V104a8,8,0,0,1,16,0Z'/%3e%3c/svg%3e";
var x_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M205.66,194.34a8,8,0,0,1-11.32,11.32L128,139.31,61.66,205.66a8,8,0,0,1-11.32-11.32L116.69,128,50.34,61.66A8,8,0,0,1,61.66,50.34L128,116.69l66.34-66.35a8,8,0,0,1,11.32,11.32L139.31,128Z'/%3e%3c/svg%3e";
var x_circle_default = "data:image/svg+xml,%3csvg%20xmlns='http://www.w3.org/2000/svg'%20viewBox='0%200%20256%20256'%20fill='currentColor'%3e%3cpath%20d='M165.66,101.66,139.31,128l26.35,26.34a8,8,0,0,1-11.32,11.32L128,139.31l-26.34,26.35a8,8,0,0,1-11.32-11.32L116.69,128,90.34,101.66a8,8,0,0,1,11.32-11.32L128,116.69l26.34-26.35a8,8,0,0,1,11.32,11.32ZM232,128A104,104,0,1,1,128,24,104.11,104.11,0,0,1,232,128Zm-16,0a88,88,0,1,0-88,88A88.1,88.1,0,0,0,216,128Z'/%3e%3c/svg%3e";
svgImport(caret_down_default);
svgImport(eye_default);
svgImport(eye_slash_default);
svgImport(plus_default);
svgImport(trash_default);
svgImport(question_default);
svgImport(x_default);
svgImport(x_circle_default);
svgImport(arrow_bend_up_left_bold_default);
svgImport(arrow_clockwise_bold_default);
svgImport(arrow_down_bold_default);
svgImport(arrow_up_bold_default);
svgImport(check_bold_default);
svgImport(check_circle_bold_default);
svgImport(copy_bold_default);
svgImport(info_bold_default);
svgImport(warning_circle_bold_default);
svgImport(x_circle_bold_default);
svgImport(spinner_gap_bold_default);
svgImport(arrows_out_cardinal_fill_default);
svgImport(airplane_fill_default);
svgImport(bell_fill_default);
svgImport(bell_slash_fill_default);
svgImport(caret_down_fill_default);
svgImport(caret_right_fill_default);
svgImport(caret_up_fill_default);
svgImport(info_fill_default);
svgImport(funnel_fill_default);
svgImport(funnel_x_fill_default);
svgImport(gear_fill_default);
svgImport(plus_fill_default);
svgImport(stethoscope_fill_default);
svgImport(table_fill_default);
function elementBuilder(options) {
if (typeof options === "string") return document.createElement(options);
else if (typeof options === "object") {
options = {
type: "div",
id: void 0,
class: void 0,
text: void 0,
html: void 0,
value: void 0,
href: void 0,
children: [],
attributes: {},
events: {},
style: {},
dataset: {},
...options
};
const newElement = document.createElement(options.type);
if (options.id) newElement.id = options.id;
if (options.class) newElement.className = Array.isArray(options.class) ? options.class.filter((name) => !!name).join(" ") : options.class.trim();
if (options.text !== void 0) newElement.textContent = options.text.toString();
if (options.html) newElement.innerHTML = options.html;
if (options.value && "value" in newElement) if (typeof options.value === "function") newElement.value = options.value();
else newElement.value = options.value;
if (options.href && "href" in newElement) newElement.href = options.href;
for (const child of options.children.filter((child) => !!child) || []) if (typeof child === "string") newElement.appendChild(document.createTextNode(child));
else newElement.appendChild(child);
if (options.attributes) {
let attributes = options.attributes;
if (typeof attributes === "function") attributes = attributes();
for (const attribute in attributes) newElement.setAttribute(attribute, attributes[attribute].toString());
}
for (const event in options.events) newElement.addEventListener(event, options.events[event]);
for (const key in options.style) newElement.style[key] = options.style[key];
for (const key in options.dataset) if (typeof options.dataset[key] === "object") newElement.dataset[key] = JSON.stringify(options.dataset[key]);
else newElement.dataset[key] = options.dataset[key].toString();
return newElement;
} else throw new Error("Invalid options provided to newElement.");
}
function findAllElements(selector, parent = document) {
return Array.from(parent.querySelectorAll(selector));
}
function findParent(element, options = {}) {
options = {
tag: void 0,
class: void 0,
partialClass: void 0,
id: void 0,
hasAttribute: void 0,
maxAttempts: -1,
currentAttempt: 1,
...options
};
if (!element?.parentElement) return void 0;
if (options.maxAttempts !== -1 && options.currentAttempt > options.maxAttempts) return void 0;
if (options.tag && element.parentElement.tagName === options.tag) return element.parentElement;
if (options.id && element.parentElement.id === options.id) return element.parentElement;
if (options.class && (Array.isArray(options.class) && options.class.some((c) => element.parentElement.classList.contains(c)) || !Array.isArray(options.class) && element.parentElement.classList.contains(options.class))) return element.parentElement;
if (options.partialClass && Array.from(element.parentElement.classList).some((c) => c.startsWith(options.partialClass))) return element.parentElement;
if (options.hasAttribute && element.parentElement.getAttribute(options.hasAttribute) !== null) return element.parentElement;
return findParent(element.parentElement, {
...options,
currentAttempt: options.currentAttempt + 1
});
}
function isSVGElement(node) {
return node && "nodeType" in node && node.nodeType === Node.ELEMENT_NODE && "ownerSVGElement" in node;
}
var EVENT_CHANNELS = function(EVENT_CHANNELS) {
EVENT_CHANNELS["FETCH"] = "tt-fetch";
EVENT_CHANNELS["XHR"] = "tt-xhr";
EVENT_CHANNELS["CHAT_MESSAGE"] = "chat-message";
EVENT_CHANNELS["CHAT_NEW"] = "chat-box-new";
EVENT_CHANNELS["CHAT_OPENED"] = "chat-box-opened";
EVENT_CHANNELS["CHAT_PEOPLE_MENU_OPENED"] = "chat-people-menu-opened";
EVENT_CHANNELS["CHAT_SETTINGS_MENU_OPENED"] = "chat-settings-menu-opened";
EVENT_CHANNELS["CHAT_REFRESHED"] = "chat-refreshed";
EVENT_CHANNELS["CHAT_RECONNECTED"] = "chat-reconnected";
EVENT_CHANNELS["CHAT_CLOSED"] = "chat-closed";
EVENT_CHANNELS["COMPANY_EMPLOYEES_PAGE"] = "company-employees-page";
EVENT_CHANNELS["COMPANY_STOCK_PAGE"] = "company-stock-page";
EVENT_CHANNELS["FACTION_ARMORY_TAB"] = "faction-armory-tab";
EVENT_CHANNELS["FACTION_CRIMES"] = "faction-crimes";
EVENT_CHANNELS["FACTION_CRIMES2"] = "faction-crimes2";
EVENT_CHANNELS["FACTION_CRIMES2_TAB"] = "faction-crimes2-tab";
EVENT_CHANNELS["FACTION_CRIMES2_REFRESH"] = "faction-crimes2-refresh";
EVENT_CHANNELS["FACTION_GIVE_TO_USER"] = "faction-give-to-user";
EVENT_CHANNELS["FACTION_UPGRADE_INFO"] = "faction-upgrade-info";
EVENT_CHANNELS["FACTION_INFO"] = "faction-info";
EVENT_CHANNELS["FACTION_MAIN"] = "faction-main";
EVENT_CHANNELS["FACTION_NATIVE_FILTER"] = "faction-filter_native";
EVENT_CHANNELS["FACTION_NATIVE_SORT"] = "faction-sort_native";
EVENT_CHANNELS["FACTION_NATIVE_ICON_UPDATE"] = "faction-icon_update_native";
EVENT_CHANNELS["FF_SCOUTER_GAUGE"] = "ff-scouter-gauge";
EVENT_CHANNELS["ITEM_AMOUNT"] = "item-amount";
EVENT_CHANNELS["ITEM_EQUIPPED"] = "item-equipped";
EVENT_CHANNELS["ITEM_ITEMS_LOADED"] = "item-items-loaded";
EVENT_CHANNELS["ITEM_SWITCH_TAB"] = "item-switch-tab";
EVENT_CHANNELS["HOSPITAL_SWITCH_PAGE"] = "hospital-switch-page";
EVENT_CHANNELS["JAIL_SWITCH_PAGE"] = "jail-switch-page";
EVENT_CHANNELS["USERLIST_SWITCH_PAGE"] = "userlist-switch-page";
EVENT_CHANNELS["TRAVEL_SELECT_TYPE"] = "travel-select-type";
EVENT_CHANNELS["TRAVEL_SELECT_COUNTRY"] = "travel-select-country";
EVENT_CHANNELS["TRAVEL_DESTINATION_UPDATE"] = "travel-destination-update";
EVENT_CHANNELS["TRAVEL_ABROAD__SHOP_LOAD"] = "TRAVEL_ABROAD__SHOP_LOAD";
EVENT_CHANNELS["TRAVEL_ABROAD__SHOP_REFRESH"] = "TRAVEL_ABROAD__SHOP_REFRESH";
EVENT_CHANNELS["FEATURE_ENABLED"] = "feature-enabled";
EVENT_CHANNELS["FEATURE_DISABLED"] = "feature-disabled";
EVENT_CHANNELS["STATE_CHANGED"] = "state-changed";
EVENT_CHANNELS["SHOP__LOAD"] = "SHOP__LOAD";
EVENT_CHANNELS["GYM_LOAD"] = "gym-load";
EVENT_CHANNELS["GYM_TRAIN"] = "gym-train";
EVENT_CHANNELS["CRIMES_LOADED"] = "crimes-loaded";
EVENT_CHANNELS["CRIMES_CRIME"] = "crimes-crime";
EVENT_CHANNELS["CRIMES2_HOME_LOADED"] = "crimes2-home-loaded";
EVENT_CHANNELS["CRIMES2_BURGLARY_LOADED"] = "crimes2-burglary-loaded";
EVENT_CHANNELS["CRIMES2_CRIME_LOADED"] = "crimes2-crime-loaded";
EVENT_CHANNELS["MISSION_LOAD"] = "mission-load";
EVENT_CHANNELS["MISSION_REWARDS"] = "mission-rewards";
EVENT_CHANNELS["TRADE"] = "trade";
EVENT_CHANNELS["PROFILE_FETCHED"] = "profile-fetched";
EVENT_CHANNELS["FILTER_APPLIED"] = "filter-applied";
EVENT_CHANNELS["STATS_ESTIMATED"] = "stats-estimated";
EVENT_CHANNELS["SWITCH_PAGE"] = "switch-page";
EVENT_CHANNELS["AUCTION_SWITCH_TYPE"] = "auction-switch-type";
EVENT_CHANNELS["ITEMMARKET_CATEGORY_ITEMS"] = "itemmarket-category-items";
EVENT_CHANNELS["ITEMMARKET_CATEGORY_ITEMS_UPDATE"] = "itemmarket-category-items-update";
EVENT_CHANNELS["ITEMMARKET_ITEMS"] = "itemmarket-items";
EVENT_CHANNELS["ITEMMARKET_ITEMS_UPDATE"] = "itemmarket-items-update";
EVENT_CHANNELS["ITEMMARKET_ITEM_DETAILS"] = "itemmarket-item-details";
EVENT_CHANNELS["WINDOW__FOCUS"] = "WINDOW__FOCUS";
return EVENT_CHANNELS;
}({});
var ANTI_SCRAPE_EVENTS = [
"TRAVEL_ABROAD__SHOP_LOAD",
"chat-message",
"chat-box-opened",
"chat-closed",
"chat-refreshed",
"chat-reconnected",
"itemmarket-category-items",
"itemmarket-category-items-update",
"itemmarket-items",
"itemmarket-items-update"
];
var CUSTOM_LISTENERS = (() => {
const listeners = {};
for (const channel of Object.values(EVENT_CHANNELS)) listeners[channel] = [];
return listeners;
})();
function addXHRListener(callback) {
SCRIPT_INJECTOR.injectXHR();
window.addEventListener("tt-xhr", callback);
}
function triggerCustomListener(channel, payload) {
if (ANTI_SCRAPE_EVENTS.includes(channel) && !isTabFocused()) return;
for (const listener of CUSTOM_LISTENERS[channel]) listener(payload);
}
function setupMissionsPage() {
addXHRListener(async ({ detail: { page, xhr, ...detail } }) => {
if (page !== "page" || !("uri" in detail)) return;
const { uri } = detail;
let sid = new URLSearchParams(xhr.requestBody).get("sid");
if (!sid && uri && (uri.sid || uri["?sid"])) sid = uri.sid || uri["?sid"];
if (sid === "missionsRewards") new MutationObserver((_mutations, observer) => {
triggerCustomListener(EVENT_CHANNELS.MISSION_REWARDS);
observer.disconnect();
}).observe(document.querySelector("#viewMissionsRewardsContainer"), { childList: true });
else if (sid === "missions" || sid === "completeContract" || sid === "acceptMission") new MutationObserver((_mutations, observer) => {
triggerCustomListener(EVENT_CHANNELS.MISSION_LOAD);
observer.disconnect();
}).observe(document.querySelector("#missionsMainContainer"), { childList: true });
});
}
_css(".tt-mission-information{margin-top:16px}.tt-mission-information>span{display:block}.tt-mission-title{text-align:center;color:var(--tt-color-light-green);margin-bottom:4px}");
var TornToolsCache = class {
_cache;
constructor() {
this._cache = {};
}
set cache(value) {
this._cache = value || {};
}
get cache() {
return this._cache;
}
get(section, key) {
return this.getCacheValue(section, key)?.value;
}
async remove(section, key) {
if (!key) {
key = section;
section = null;
}
if (section && !this.hasValue(section, key) || !section && !this.hasValue(key.toString())) return;
if (section) delete this.cache[section][key];
else delete this.cache[key];
await ttStorage.set({ cache: this.cache });
}
hasValue(section, key) {
return this.getCacheValue(section, key) !== null;
}
getCacheValue(section, key) {
if (!key) {
key = section;
section = null;
}
let value = null;
if (section) {
if (section in this.cache && key in this.cache[section]) value = this.cache[section][key];
} else if (key in this.cache) value = this.cache[key];
if (value === null || !("value" in value)) return null;
if ("indefinite" in value) return value;
else return value.timeout > Date.now() ? value : null;
}
async set(object, ttl, section) {
return this._set(object, ttl, section);
}
setIndefinite(object, section) {
return this._set(object, null, section);
}
async _set(object, ttl, section) {
const timeout = ttl === null ? null : Date.now() + ttl;
if (section) {
if (!(section in this.cache)) this.cache[section] = {};
for (const [key, value] of Object.entries(object)) this.cache[section][key] = this.createCacheValue(value, timeout);
} else for (const [key, value] of Object.entries(object)) this.cache[key] = this.createCacheValue(value, timeout);
await ttStorage.set({ cache: this.cache });
}
createCacheValue(value, timeout) {
if (timeout === null) return {
value,
indefinite: true
};
else return {
value,
timeout
};
}
async clear(section) {
if (section) {
delete this.cache[section];
await ttStorage.set({ cache: this.cache });
} else ttStorage.set({ cache: {} }).then(() => this.cache = {});
}
async refresh() {
let hasChanged = false;
const now = Date.now();
refreshObject(this.cache);
for (const section in this.cache) if (!Object.keys(this.cache[section]).length) delete this.cache[section];
if (hasChanged) await ttStorage.set({ cache: this.cache });
function refreshObject(object) {
for (const key in object) {
const value = object[key];
if ("value" in value) {
const cacheValue = value;
if ("indefinite" in cacheValue || cacheValue.timeout > now) continue;
hasChanged = true;
delete object[key];
} else refreshObject(value);
}
}
}
};
var ttCache = new TornToolsCache();
var DefaultSetting = class {
type;
defaultValue;
constructor(type, defaultValue) {
this.type = type;
this.defaultValue = defaultValue;
}
};
var DEFAULT_STORAGE = {
version: {
current: new DefaultSetting("string", () => RUNTIME_INFORMATION.getVersion()),
initial: new DefaultSetting("string", () => RUNTIME_INFORMATION.getVersion()),
oldVersion: new DefaultSetting("string"),
showNotice: new DefaultSetting("boolean", true)
},
api: {
torn: {
key: new DefaultSetting("string"),
online: new DefaultSetting("boolean", true),
error: new DefaultSetting("string")
},
tornstats: { key: new DefaultSetting("string") },
yata: { key: new DefaultSetting("string") },
ffScouter: { key: new DefaultSetting("string") }
},
settings: {
updateNotice: new DefaultSetting("boolean", true),
featureDisplay: new DefaultSetting("boolean", true),
featureDisplayPosition: new DefaultSetting("string", "bottom-left"),
featureDisplayOnlyFailed: new DefaultSetting("boolean", false),
featureDisplayHideDisabled: new DefaultSetting("boolean", false),
featureDisplayHideEmpty: new DefaultSetting("boolean", true),
developer: new DefaultSetting("boolean", false),
formatting: {
tct: new DefaultSetting("boolean", false),
date: new DefaultSetting("string", "eu"),
time: new DefaultSetting("string", "eu")
},
sorting: { abroad: {
column: new DefaultSetting("string", ""),
order: new DefaultSetting("string", "none")
} },
notifications: {
sound: new DefaultSetting("string", "default"),
soundCustom: new DefaultSetting("string", ""),
tts: new DefaultSetting("boolean", false),
ttsVoice: new DefaultSetting("string", "default"),
ttsRate: new DefaultSetting("number", 1),
link: new DefaultSetting("boolean", true),
volume: new DefaultSetting("number", 100),
requireInteraction: new DefaultSetting("boolean", false),
types: {
global: new DefaultSetting("boolean", () => typeof Notification !== "undefined" && Notification.permission === "granted"),
events: new DefaultSetting("boolean", true),
messages: new DefaultSetting("boolean", true),
status: new DefaultSetting("boolean", true),
traveling: new DefaultSetting("boolean", true),
cooldowns: new DefaultSetting("boolean", true),
education: new DefaultSetting("boolean", true),
newDay: new DefaultSetting("boolean", true),
energy: new DefaultSetting("array", ["100%"]),
nerve: new DefaultSetting("array", ["100%"]),
happy: new DefaultSetting("array", ["100%"]),
life: new DefaultSetting("array", ["100%"]),
offline: new DefaultSetting("array", []),
chainTimerEnabled: new DefaultSetting("boolean", true),
chainBonusEnabled: new DefaultSetting("boolean", true),
leavingHospitalEnabled: new DefaultSetting("boolean", true),
landingEnabled: new DefaultSetting("boolean", true),
cooldownDrugEnabled: new DefaultSetting("boolean", true),
cooldownBoosterEnabled: new DefaultSetting("boolean", true),
cooldownMedicalEnabled: new DefaultSetting("boolean", true),
chainTimer: new DefaultSetting("array", []),
chainBonus: new DefaultSetting("array", []),
leavingHospital: new DefaultSetting("array", []),
landing: new DefaultSetting("array", []),
cooldownDrug: new DefaultSetting("array", []),
cooldownBooster: new DefaultSetting("array", []),
cooldownMedical: new DefaultSetting("array", []),
stocks: new DefaultSetting("object", {}),
missionsLimitEnabled: new DefaultSetting("boolean", false),
missionsLimit: new DefaultSetting("string", ""),
missionsExpireEnabled: new DefaultSetting("boolean", false),
missionsExpire: new DefaultSetting("array", []),
npcsGlobal: new DefaultSetting("boolean", true),
npcs: new DefaultSetting("array", []),
npcPlannedEnabled: new DefaultSetting("boolean", true),
npcPlanned: new DefaultSetting("array", []),
refillEnergyEnabled: new DefaultSetting("boolean", true),
refillEnergy: new DefaultSetting("string", ""),
refillNerveEnabled: new DefaultSetting("boolean", true),
refillNerve: new DefaultSetting("string", "")
}
},
apiUsage: {
comment: new DefaultSetting("string", "TornTools"),
delayEssential: new DefaultSetting("number", 30),
delayBasic: new DefaultSetting("number", 120),
delayPassive: new DefaultSetting("number", 3600),
delayStakeouts: new DefaultSetting("number", 30),
user: {
bars: new DefaultSetting("boolean", true),
cooldowns: new DefaultSetting("boolean", true),
travel: new DefaultSetting("boolean", true),
newevents: new DefaultSetting("boolean", true),
newmessages: new DefaultSetting("boolean", true),
refills: new DefaultSetting("boolean", true),
stocks: new DefaultSetting("boolean", true),
education: new DefaultSetting("boolean", true),
networth: new DefaultSetting("boolean", true),
inventory: new DefaultSetting("boolean", true),
jobpoints: new DefaultSetting("boolean", true),
merits: new DefaultSetting("boolean", true),
perks: new DefaultSetting("boolean", true),
icons: new DefaultSetting("boolean", true),
ammo: new DefaultSetting("boolean", true),
battlestats: new DefaultSetting("boolean", true),
crimes: new DefaultSetting("boolean", true),
workstats: new DefaultSetting("boolean", true),
skills: new DefaultSetting("boolean", true),
weaponexp: new DefaultSetting("boolean", true),
properties: new DefaultSetting("boolean", true),
calendar: new DefaultSetting("boolean", true),
organizedcrime: new DefaultSetting("boolean", true),
missions: new DefaultSetting("boolean", true),
personalstats: new DefaultSetting("boolean", true),
attacks: new DefaultSetting("boolean", true),
money: new DefaultSetting("boolean", true),
honors: new DefaultSetting("boolean", true),
medals: new DefaultSetting("boolean", true),
virus: new DefaultSetting("boolean", true)
}
},
themes: {
pages: new DefaultSetting("string", "default"),
containers: new DefaultSetting("string", "default")
},
hideIcons: new DefaultSetting("array", []),
hideCasinoGames: new DefaultSetting("array", []),
hideStocks: new DefaultSetting("array", []),
alliedFactions: new DefaultSetting("array", []),
customLinks: new DefaultSetting("array", []),
employeeInactivityWarning: new DefaultSetting("array", []),
factionInactivityWarning: new DefaultSetting("array", []),
userAlias: new DefaultSetting("array", []),
csvDelimiter: new DefaultSetting("string", ";"),
pages: {
global: {
alignLeft: new DefaultSetting("boolean", false),
hideLevelUpgrade: new DefaultSetting("boolean", false),
hideQuitButtons: new DefaultSetting("boolean", false),
hideTutorials: new DefaultSetting("boolean", false),
keepAttackHistory: new DefaultSetting("boolean", true),
miniProfileLastAction: new DefaultSetting("boolean", true),
reviveProvider: new DefaultSetting("string", ""),
pageTitles: new DefaultSetting("boolean", true),
stackingMode: new DefaultSetting("boolean", false),
noOutsideLinkAlert: new DefaultSetting("boolean", false),
urlFill: new DefaultSetting("boolean", true)
},
profile: {
avgpersonalstats: new DefaultSetting("boolean", false),
statusIndicator: new DefaultSetting("boolean", true),
idBesideProfileName: new DefaultSetting("boolean", true),
notes: new DefaultSetting("boolean", true),
showAllyWarning: new DefaultSetting("boolean", true),
ageToWords: new DefaultSetting("boolean", true),
disableAllyAttacks: new DefaultSetting("boolean", true),
box: new DefaultSetting("boolean", true),
boxStats: new DefaultSetting("boolean", true),
boxSpy: new DefaultSetting("boolean", true),
boxStakeout: new DefaultSetting("boolean", true),
boxAttackHistory: new DefaultSetting("boolean", true),
boxFetch: new DefaultSetting("boolean", true)
},
chat: {
fontSize: new DefaultSetting("number", 12),
searchChat: new DefaultSetting("boolean", true),
completeUsernames: new DefaultSetting("boolean", true),
highlights: new DefaultSetting("array", [{
name: "$player",
color: "#7ca900"
}]),
titleHighlights: new DefaultSetting("array", []),
tradeTimer: new DefaultSetting("boolean", true),
resizable: new DefaultSetting("boolean", true),
hideChatButton: new DefaultSetting("boolean", true),
hideChat: new DefaultSetting("boolean", false)
},
sidebar: {
notes: new DefaultSetting("boolean", true),
highlightEnergy: new DefaultSetting("boolean", true),
highlightNerve: new DefaultSetting("boolean", false),
ocTimer: new DefaultSetting("boolean", true),
oc2Timer: new DefaultSetting("boolean", true),
oc2TimerPosition: new DefaultSetting("boolean", false),
oc2TimerLevel: new DefaultSetting("boolean", true),
factionOCTimer: new DefaultSetting("boolean", false),
collapseAreas: new DefaultSetting("boolean", true),
settingsLink: new DefaultSetting("boolean", true),
hideGymHighlight: new DefaultSetting("boolean", false),
hideNewspaperHighlight: new DefaultSetting("boolean", false),
upkeepPropHighlight: new DefaultSetting("number", 0),
barLinks: new DefaultSetting("boolean", true),
pointsValue: new DefaultSetting("boolean", true),
npcLootTimes: new DefaultSetting("boolean", true),
npcLootTimesService: new DefaultSetting("string", "tornstats"),
cooldownEndTimes: new DefaultSetting("boolean", true),
companyAddictionLevel: new DefaultSetting("boolean", true),
showJobPointsToolTip: new DefaultSetting("boolean", true),
rwTimer: new DefaultSetting("boolean", true),
virusTimer: new DefaultSetting("boolean", false)
},
popup: {
dashboard: new DefaultSetting("boolean", true),
marketSearch: new DefaultSetting("boolean", true),
bazaarUsingExternal: new DefaultSetting("boolean", true),
calculator: new DefaultSetting("boolean", true),
stocksOverview: new DefaultSetting("boolean", true),
notifications: new DefaultSetting("boolean", true),
defaultTab: new DefaultSetting("string", "dashboard"),
showStakeouts: new DefaultSetting("boolean", true),
showIcons: new DefaultSetting("boolean", true),
fullBarTime: new DefaultSetting("boolean", false)
},
icon: {
global: new DefaultSetting("boolean", true),
energy: new DefaultSetting("boolean", true),
nerve: new DefaultSetting("boolean", true),
happy: new DefaultSetting("boolean", true),
life: new DefaultSetting("boolean", true),
chain: new DefaultSetting("boolean", true),
travel: new DefaultSetting("boolean", true)
},
education: {
greyOut: new DefaultSetting("boolean", true),
finishTime: new DefaultSetting("boolean", true)
},
jail: { filter: new DefaultSetting("boolean", true) },
bank: {
investmentInfo: new DefaultSetting("boolean", true),
investmentDueTime: new DefaultSetting("boolean", true)
},
home: {
networthDetails: new DefaultSetting("boolean", true),
effectiveStats: new DefaultSetting("boolean", true)
},
items: {
quickItems: new DefaultSetting("boolean", true),
values: new DefaultSetting("boolean", true),
drugDetails: new DefaultSetting("boolean", true),
marketLinks: new DefaultSetting("boolean", false),
highlightBloodBags: new DefaultSetting("string", "none"),
missingFlowers: new DefaultSetting("boolean", false),
missingPlushies: new DefaultSetting("boolean", false),
bookEffects: new DefaultSetting("boolean", true),
canGains: new DefaultSetting("boolean", true),
nerveGains: new DefaultSetting("boolean", true),
candyHappyGains: new DefaultSetting("boolean", true),
energyWarning: new DefaultSetting("boolean", true),
medicalLife: new DefaultSetting("boolean", true),
openedSupplyPackValue: new DefaultSetting("boolean", true),
hideRecycleMessage: new DefaultSetting("boolean", false),
hideTooManyItemsWarning: new DefaultSetting("boolean", false)
},
crimes: { quickCrimes: new DefaultSetting("boolean", true) },
companies: {
idBesideCompanyName: new DefaultSetting("boolean", false),
specials: new DefaultSetting("boolean", true),
autoStockFill: new DefaultSetting("boolean", true),
employeeEffectiveness: new DefaultSetting("number", 18)
},
travel: {
computer: new DefaultSetting("boolean", true),
table: new DefaultSetting("boolean", true),
cleanFlight: new DefaultSetting("boolean", false),
tabTitleTimer: new DefaultSetting("boolean", false),
travelProfits: new DefaultSetting("boolean", true),
fillMax: new DefaultSetting("boolean", true),
peopleFilter: new DefaultSetting("boolean", true),
landingTime: new DefaultSetting("boolean", true),
flyingTime: new DefaultSetting("boolean", true),
itemFilter: new DefaultSetting("boolean", true),
energyWarning: new DefaultSetting("boolean", true),
cooldownWarnings: new DefaultSetting("boolean", true),
autoTravelTableCountry: new DefaultSetting("boolean", false),
autoFillMax: new DefaultSetting("boolean", true),
efficientRehab: new DefaultSetting("boolean", true),
efficientRehabSelect: new DefaultSetting("boolean", false)
},
stocks: {
filter: new DefaultSetting("boolean", true),
acronyms: new DefaultSetting("boolean", true),
valueAndProfit: new DefaultSetting("boolean", true),
moneyInput: new DefaultSetting("boolean", true)
},
competitions: {
easterEggs: new DefaultSetting("boolean", false),
easterEggsAlert: new DefaultSetting("boolean", true)
},
events: { worth: new DefaultSetting("boolean", true) },
hospital: { filter: new DefaultSetting("boolean", true) },
auction: { filter: new DefaultSetting("boolean", true) },
api: {
autoFillKey: new DefaultSetting("boolean", true),
autoDemo: new DefaultSetting("boolean", false),
autoPretty: new DefaultSetting("boolean", true),
clickableSelections: new DefaultSetting("boolean", true)
},
forums: {
menu: new DefaultSetting("boolean", true),
hidePosts: new DefaultSetting("object", {}),
hideThreads: new DefaultSetting("object", {}),
highlightPosts: new DefaultSetting("object", {}),
highlightThreads: new DefaultSetting("object", {}),
ignoredThreads: new DefaultSetting("object", {}),
debugInfoBtn: new DefaultSetting("boolean", true),
onlyNewFeedButton: new DefaultSetting("boolean", true)
},
bazaar: {
itemsCost: new DefaultSetting("boolean", true),
worth: new DefaultSetting("boolean", true),
fillMax: new DefaultSetting("boolean", true),
maxBuyIgnoreCash: new DefaultSetting("boolean", false),
highlightSubVendorItems: new DefaultSetting("boolean", false)
},
trade: {
itemValues: new DefaultSetting("boolean", true),
openChat: new DefaultSetting("boolean", true)
},
displayCase: { worth: new DefaultSetting("boolean", true) },
shops: {
fillMax: new DefaultSetting("boolean", true),
maxBuyIgnoreCash: new DefaultSetting("boolean", false),
profit: new DefaultSetting("boolean", true),
filters: new DefaultSetting("boolean", true),
values: new DefaultSetting("boolean", true)
},
casino: {
netTotal: new DefaultSetting("boolean", true),
blackjack: new DefaultSetting("boolean", true),
highlow: new DefaultSetting("boolean", false),
highlowMovement: new DefaultSetting("boolean", true)
},
racing: {
winPercentage: new DefaultSetting("boolean", true),
upgrades: new DefaultSetting("boolean", true),
filter: new DefaultSetting("boolean", true)
},
faction: {
idBesideFactionName: new DefaultSetting("boolean", false),
csvRaidReport: new DefaultSetting("boolean", true),
csvRankedWarReport: new DefaultSetting("boolean", true),
csvWarReport: new DefaultSetting("boolean", true),
csvChainReport: new DefaultSetting("boolean", true),
csvChallengeContributions: new DefaultSetting("boolean", true),
openOc: new DefaultSetting("boolean", true),
highlightOwn: new DefaultSetting("boolean", true),
availablePlayers: new DefaultSetting("boolean", true),
recommendedNnb: new DefaultSetting("boolean", true),
ocNnb: new DefaultSetting("boolean", true),
ocTimes: new DefaultSetting("boolean", true),
ocLastAction: new DefaultSetting("boolean", true),
banker: new DefaultSetting("boolean", true),
showFullInfobox: new DefaultSetting("boolean", true),
foldableInfobox: new DefaultSetting("boolean", true),
numberMembers: new DefaultSetting("boolean", true),
warFinishTimes: new DefaultSetting("boolean", false),
memberFilter: new DefaultSetting("boolean", true),
armoryFilter: new DefaultSetting("boolean", true),
armoryWorth: new DefaultSetting("boolean", true),
upgradeRequiredRespect: new DefaultSetting("boolean", true),
memberInfo: new DefaultSetting("boolean", false),
rankedWarFilter: new DefaultSetting("boolean", true),
quickItems: new DefaultSetting("boolean", true),
stakeout: new DefaultSetting("boolean", true),
showFactionSpy: new DefaultSetting("boolean", true),
oc2Filter: new DefaultSetting("boolean", true),
warnCrime: new DefaultSetting("boolean", false),
rankedWarValue: new DefaultSetting("boolean", true),
totalChallengeContributions: new DefaultSetting("boolean", true)
},
property: {
value: new DefaultSetting("boolean", true),
happy: new DefaultSetting("boolean", true)
},
gym: {
specialist: new DefaultSetting("boolean", true),
disableStats: new DefaultSetting("boolean", true),
graph: new DefaultSetting("boolean", true),
steadfast: new DefaultSetting("boolean", true),
progress: new DefaultSetting("boolean", true)
},
missions: {
hints: new DefaultSetting("boolean", true),
rewards: new DefaultSetting("boolean", true)
},
attack: {
bonusInformation: new DefaultSetting("boolean", true),
timeoutWarning: new DefaultSetting("boolean", true),
fairAttack: new DefaultSetting("boolean", true),
weaponExperience: new DefaultSetting("boolean", true),
hideAttackButtons: new DefaultSetting("array", [])
},
city: {
items: new DefaultSetting("boolean", true),
combineDuplicates: new DefaultSetting("boolean", true)
},
joblist: { specials: new DefaultSetting("boolean", true) },
bounties: { filter: new DefaultSetting("boolean", true) },
userlist: { filter: new DefaultSetting("boolean", true) },
itemmarket: {
highlightCheapItems: new DefaultSetting("number|empty", ""),
highlightCheapItemsSound: new DefaultSetting("boolean", false),
leftBar: new DefaultSetting("boolean", false),
fillMax: new DefaultSetting("boolean", true)
},
competition: { filter: new DefaultSetting("boolean", true) },
museum: { autoFill: new DefaultSetting("boolean", true) },
enemies: { filter: new DefaultSetting("boolean", true) },
friends: { filter: new DefaultSetting("boolean", true) },
targets: { filter: new DefaultSetting("boolean", true) },
crimes2: {
burglaryFilter: new DefaultSetting("boolean", true),
value: new DefaultSetting("boolean", true)
}
},
scripts: {
noConfirm: {
itemEquip: new DefaultSetting("boolean", true),
tradeAccept: new DefaultSetting("boolean", false),
pointsMarketRemove: new DefaultSetting("boolean", false),
pointsMarketBuy: new DefaultSetting("boolean", false),
abroadItemBuy: new DefaultSetting("boolean", true)
},
achievements: {
show: new DefaultSetting("boolean", true),
completed: new DefaultSetting("boolean", false)
},
lastAction: {
factionMember: new DefaultSetting("boolean", false),
companyOwn: new DefaultSetting("boolean", false),
companyOther: new DefaultSetting("boolean", false)
},
statsEstimate: {
global: new DefaultSetting("boolean", true),
delay: new DefaultSetting("number", 1500),
cachedOnly: new DefaultSetting("boolean", true),
displayNoResult: new DefaultSetting("boolean", false),
maxLevel: new DefaultSetting("number", 100),
profiles: new DefaultSetting("boolean", true),
enemies: new DefaultSetting("boolean", true),
hof: new DefaultSetting("boolean", true),
attacks: new DefaultSetting("boolean", true),
userlist: new DefaultSetting("boolean", true),
bounties: new DefaultSetting("boolean", true),
factions: new DefaultSetting("boolean", true),
wars: new DefaultSetting("boolean", true),
abroad: new DefaultSetting("boolean", true),
competition: new DefaultSetting("boolean", true),
rankedWars: new DefaultSetting("boolean", true),
targets: new DefaultSetting("boolean", true)
},
ffScouter: {
miniProfile: new DefaultSetting("boolean", true),
profile: new DefaultSetting("boolean", true),
attack: new DefaultSetting("boolean", true),
factionList: new DefaultSetting("boolean", true),
gauge: new DefaultSetting("boolean", true)
}
},
external: {
tornstats: new DefaultSetting("boolean", false),
yata: new DefaultSetting("boolean", false),
prometheus: new DefaultSetting("boolean", false),
lzpt: new DefaultSetting("boolean", false),
tornw3b: new DefaultSetting("boolean", false),
ffScouter: new DefaultSetting("boolean", false),
tornintel: new DefaultSetting("boolean", false)
}
},
filters: {
hospital: {
enabled: new DefaultSetting("boolean", true),
timeStart: new DefaultSetting("number", 0),
timeEnd: new DefaultSetting("number", 100),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100),
faction: new DefaultSetting("string", ""),
activity: new DefaultSetting("array", []),
revivesOn: new DefaultSetting("boolean", false)
},
jail: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
faction: new DefaultSetting("string", "All"),
timeStart: new DefaultSetting("number", 0),
timeEnd: new DefaultSetting("number", 100),
levelStart: new DefaultSetting("number", 1),
levelEnd: new DefaultSetting("number", 100),
scoreStart: new DefaultSetting("number", 0),
scoreEnd: new DefaultSetting("number", 5e3),
bailCost: new DefaultSetting("number", -1)
},
racing: {
enabled: new DefaultSetting("boolean", true),
hideRaces: new DefaultSetting("array", []),
timeStart: new DefaultSetting("number", 0),
timeEnd: new DefaultSetting("number", 48),
driversMin: new DefaultSetting("number", 2),
driversMax: new DefaultSetting("number", 100),
lapsMin: new DefaultSetting("number", 1),
lapsMax: new DefaultSetting("number", 100),
track: new DefaultSetting("array", []),
name: new DefaultSetting("string", "")
},
containers: new DefaultSetting("object", {}),
travel: {
open: new DefaultSetting("boolean", false),
type: new DefaultSetting("string", "basic"),
categories: new DefaultSetting("array", []),
countries: new DefaultSetting("array", []),
hideOutOfStock: new DefaultSetting("boolean", false),
applySalesTax: new DefaultSetting("boolean", false),
sellAnonymously: new DefaultSetting("boolean", false)
},
abroadPeople: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
status: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100),
faction: new DefaultSetting("string", ""),
special: {
newPlayer: new DefaultSetting("string", "both"),
inCompany: new DefaultSetting("string", "both"),
inFaction: new DefaultSetting("string", "both"),
isDonator: new DefaultSetting("string", "both"),
hasBounties: new DefaultSetting("string", "both"),
bazaarOpen: new DefaultSetting("string", "both")
},
estimates: new DefaultSetting("array", []),
ffScoreMax: new DefaultSetting("number", null),
ffScoreMin: new DefaultSetting("number", null)
},
abroadItems: {
enabled: new DefaultSetting("boolean", true),
profitOnly: new DefaultSetting("boolean", false),
outOfStock: new DefaultSetting("boolean", false),
categories: new DefaultSetting("array", []),
taxes: new DefaultSetting("array", [])
},
trade: { hideValues: new DefaultSetting("boolean", false) },
gym: {
specialist1: new DefaultSetting("string", "none"),
specialist2: new DefaultSetting("string", "none"),
strength: new DefaultSetting("boolean", false),
speed: new DefaultSetting("boolean", false),
defense: new DefaultSetting("boolean", false),
dexterity: new DefaultSetting("boolean", false)
},
city: { highlightItems: new DefaultSetting("boolean", true) },
bounties: {
maxLevel: new DefaultSetting("number", 100),
hideUnavailable: new DefaultSetting("boolean", false)
},
userlist: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100),
special: {
fedded: new DefaultSetting("string", "both"),
fallen: new DefaultSetting("string", "both"),
traveling: new DefaultSetting("string", "both"),
newPlayer: new DefaultSetting("string", "both"),
onWall: new DefaultSetting("string", "both"),
inCompany: new DefaultSetting("string", "both"),
inFaction: new DefaultSetting("string", "both"),
isDonator: new DefaultSetting("string", "both"),
inHospital: new DefaultSetting("string", "both"),
inJail: new DefaultSetting("string", "both"),
earlyDischarge: new DefaultSetting("string", "both"),
hasBounties: new DefaultSetting("string", "both"),
bazaarOpen: new DefaultSetting("string", "both")
},
hospReason: {
attackedBy: new DefaultSetting("string", "both"),
muggedBy: new DefaultSetting("string", "both"),
hospitalizedBy: new DefaultSetting("string", "both"),
other: new DefaultSetting("string", "both")
},
estimates: new DefaultSetting("array", []),
ffScoreMax: new DefaultSetting("number", null),
ffScoreMin: new DefaultSetting("number", null)
},
stocks: {
enabled: new DefaultSetting("boolean", true),
name: new DefaultSetting("string", ""),
investment: {
owned: new DefaultSetting("string", "both"),
benefit: new DefaultSetting("string", "both"),
passive: new DefaultSetting("string", "both")
},
price: {
price: new DefaultSetting("string", "both"),
profit: new DefaultSetting("string", "both")
}
},
faction: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 1),
levelEnd: new DefaultSetting("number", 100),
lastActionStart: new DefaultSetting("number", 0),
lastActionEnd: new DefaultSetting("number", -1),
status: new DefaultSetting("array", []),
position: new DefaultSetting("string", ""),
special: {
fedded: new DefaultSetting("string", "both"),
fallen: new DefaultSetting("string", "both"),
newPlayer: new DefaultSetting("string", "both"),
inCompany: new DefaultSetting("string", "both"),
isDonator: new DefaultSetting("string", "both"),
isRecruit: new DefaultSetting("string", "both")
},
ffScoreMax: new DefaultSetting("number", null),
ffScoreMin: new DefaultSetting("number", null)
},
factionArmory: {
enabled: new DefaultSetting("boolean", true),
hideUnavailable: new DefaultSetting("boolean", false),
weapons: {
name: new DefaultSetting("string", ""),
category: new DefaultSetting("string", ""),
rarity: new DefaultSetting("string", ""),
weaponType: new DefaultSetting("string", ""),
damage: new DefaultSetting("string", ""),
accuracy: new DefaultSetting("string", ""),
weaponBonus: new DefaultSetting("array", [])
},
armor: {
name: new DefaultSetting("string", ""),
rarity: new DefaultSetting("string", ""),
defence: new DefaultSetting("string", ""),
set: new DefaultSetting("string", ""),
armorBonus: new DefaultSetting("string", "")
},
temporary: { name: new DefaultSetting("string", "") }
},
factionRankedWar: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
status: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 1),
levelEnd: new DefaultSetting("number", 100),
estimates: new DefaultSetting("array", []),
ffScoreMax: new DefaultSetting("number", null),
ffScoreMin: new DefaultSetting("number", null)
},
profile: {
relative: new DefaultSetting("boolean", false),
stats: new DefaultSetting("array", [])
},
competition: {
levelStart: new DefaultSetting("number", 1),
levelEnd: new DefaultSetting("number", 100),
estimates: new DefaultSetting("array", [])
},
shops: {
hideLoss: new DefaultSetting("boolean", false),
hideUnder100: new DefaultSetting("boolean", false)
},
auction: {
enabled: new DefaultSetting("boolean", true),
weapons: {
name: new DefaultSetting("string", ""),
category: new DefaultSetting("string", ""),
rarity: new DefaultSetting("string", ""),
weaponType: new DefaultSetting("string", ""),
damage: new DefaultSetting("string", ""),
accuracy: new DefaultSetting("string", ""),
weaponBonus: new DefaultSetting("array", []),
quality: new DefaultSetting("string", "")
},
armor: {
name: new DefaultSetting("string", ""),
rarity: new DefaultSetting("string", ""),
defence: new DefaultSetting("string", ""),
set: new DefaultSetting("string", ""),
armorBonus: new DefaultSetting("string", "")
},
items: {
name: new DefaultSetting("string", ""),
category: new DefaultSetting("string", ""),
rarity: new DefaultSetting("string", "")
}
},
enemies: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100),
estimates: new DefaultSetting("array", [])
},
friends: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100)
},
targets: {
enabled: new DefaultSetting("boolean", true),
activity: new DefaultSetting("array", []),
levelStart: new DefaultSetting("number", 0),
levelEnd: new DefaultSetting("number", 100),
estimates: new DefaultSetting("array", [])
},
burglary: {
targetName: new DefaultSetting("string", ""),
targetType: new DefaultSetting("array", [])
},
oc2: {
enabled: new DefaultSetting("boolean", true),
difficulty: new DefaultSetting("array", []),
status: new DefaultSetting("array", [])
}
},
userdata: new DefaultSetting("object", { date: -1 }),
torndata: new DefaultSetting("object", { date: -2 }),
stockdata: new DefaultSetting("object", {}),
factiondata: new DefaultSetting("object", {}),
localdata: {
tradeMessage: new DefaultSetting("number", 0),
popup: { calculatorItems: new DefaultSetting("array", []) },
vault: {
initialized: new DefaultSetting("boolean", false),
lastTransaction: new DefaultSetting("string", ""),
total: new DefaultSetting("number", 0),
user: {
initial: new DefaultSetting("number", 0),
current: new DefaultSetting("number", 0)
},
partner: {
initial: new DefaultSetting("number", 0),
current: new DefaultSetting("number", 0)
}
},
chatResize: new DefaultSetting("object", {}),
feedHidden: new DefaultSetting("object", {}),
threadsHiddenInFeed: new DefaultSetting("array", [])
},
stakeouts: new DefaultSetting("object", { list: [] }),
factionStakeouts: new DefaultSetting("object", { list: [] }),
attackHistory: {
fetchData: new DefaultSetting("boolean", true),
lastAttack: new DefaultSetting("number", 0),
history: new DefaultSetting("object", {})
},
notes: {
sidebar: {
text: new DefaultSetting("string", ""),
height: new DefaultSetting("string", "22px")
},
profile: new DefaultSetting("object", {})
},
quick: {
items: new DefaultSetting("array", []),
factionItems: new DefaultSetting("array", []),
crimes: new DefaultSetting("array", []),
jail: new DefaultSetting("array", [])
},
cache: new DefaultSetting("object", {}),
npcs: new DefaultSetting("object", {}),
notificationHistory: new DefaultSetting("array", []),
notifications: {
events: new DefaultSetting("object", {}),
messages: new DefaultSetting("object", {}),
newDay: new DefaultSetting("object", {}),
energy: new DefaultSetting("object", {}),
happy: new DefaultSetting("object", {}),
nerve: new DefaultSetting("object", {}),
life: new DefaultSetting("object", {}),
travel: new DefaultSetting("object", {}),
drugs: new DefaultSetting("object", {}),
boosters: new DefaultSetting("object", {}),
medical: new DefaultSetting("object", {}),
hospital: new DefaultSetting("object", {}),
chain: new DefaultSetting("object", {}),
chainCount: new DefaultSetting("object", {}),
stakeouts: new DefaultSetting("object", {}),
npcs: new DefaultSetting("object", {}),
offline: new DefaultSetting("object", {}),
missionsLimit: new DefaultSetting("object", {}),
missionsExpire: new DefaultSetting("object", {}),
refillEnergy: new DefaultSetting("object", {}),
refillNerve: new DefaultSetting("object", {})
},
migrations: new DefaultSetting("array", [])
};
function getDefaultStorage(defaultStorage) {
const newStorage = {};
for (const key in defaultStorage) if (typeof defaultStorage[key] === "object") {
const setting = defaultStorage[key];
if (setting instanceof DefaultSetting && "defaultValue" in setting) switch (typeof setting.defaultValue) {
case "function":
newStorage[key] = setting.defaultValue();
break;
case "boolean":
case "number":
case "string":
case "object":
newStorage[key] = setting.defaultValue;
break;
default:
newStorage[key] = setting.defaultValue;
break;
}
else newStorage[key] = getDefaultStorage(defaultStorage[key]);
} else newStorage[key] = defaultStorage[key];
return newStorage;
}
var MIGRATIONS = [
{
id: "9da14c73-0145-4b1d-90e3-0363a5b57499",
version: "9.0.0",
execute(_database, flags, _oldStorage) {
flags.updateUserdata = true;
}
},
{
id: "43fae1f2-5568-4ae5-b12f-f3625e1e58c6",
version: "9.0.0",
execute(database, _flags, _oldStorage) {
database.cache["personal-stats"] = {};
}
},
{
id: "b194a6d5-4230-4b03-8a8b-bebd7c431cc9",
version: "9.0.0",
execute(database, _flags, _oldStorage) {
database.settings.pages.api.autoDemo = false;
}
},
{
id: "b0f539ba-41f8-4eed-93e2-e8523f7c49a5",
version: "9.0.1",
execute(database, _flags, oldStorage) {
const oldCustomLinks = oldStorage?.settings?.customLinks ?? [];
database.settings.customLinks = oldCustomLinks.map((link) => {
return link.preset && link.preset !== "custom" ? {
newTab: link.newTab,
location: link.location,
name: link.name,
preset: link.preset
} : {
newTab: link.newTab,
location: link.location,
name: link.name,
href: link.href
};
});
}
},
{
id: "360b1f70-c78b-44c1-b217-24bd6b398bac",
version: "9.0.5",
execute(database, _flags, oldStorage) {
if (!oldStorage?.settings?.userAlias || Array.isArray(oldStorage.settings.userAlias)) return;
const oldUserAliases = oldStorage.settings.userAlias;
database.settings.userAlias = Object.entries(oldUserAliases).map(([id, { alias, name }]) => {
const idMatch = id.match(/^(\d+)$/);
return idMatch ? {
userId: parseInt(idMatch[0]),
userName: name,
alias
} : {
userId: -1,
userName: name,
alias,
incorrectId: id
};
});
}
},
{
id: "95c020eb-2c75-4bbe-8fe9-64f96f108f48",
version: "9.0.5",
execute(database, _flags, oldStorage) {
if (!oldStorage?.settings?.pages?.popup?.defaultTab) return;
if (oldStorage.settings.pages.popup.defaultTab === "stocks") database.settings.pages.popup.defaultTab = "stocksOverview";
else if (oldStorage.settings.pages.popup.defaultTab === "market") database.settings.pages.popup.defaultTab = "marketSearch";
}
},
{
id: "96356911-fecd-4b79-9825-ee5ad422c8fe",
version: "9.0.5",
execute(database, _flags, oldStorage) {
if (typeof oldStorage?.settings?.pages?.popup.hoverBarTime !== "boolean") return;
database.settings.pages.popup.fullBarTime = oldStorage.settings.pages.popup.hoverBarTime;
}
},
{
id: "7396191c-35a9-4d92-905a-0e411f9a6823",
version: "9.0.5",
execute(_database, _flags, _oldStorage) {
ttStorage.remove("usage");
}
},
{
id: "d3e6e03a-698d-4df4-9062-4d3c9ce9d479",
version: "9.0.5",
execute(database, _flags, oldStorage) {
if (!oldStorage?.filters?.travel?.categories?.includes("other")) return;
database.filters.travel.categories = [...oldStorage.filters.travel.categories, "defensive"];
}
},
{
id: "700848e9-ee48-42ce-b8b1-893cb471cfe4",
version: "9.0.6",
execute(_database, flags, _oldStorage) {
flags.clearCache = true;
}
},
{
id: "a1b2c3d4-e5f6-7890-abcd-ef1234567890",
version: "9.0.6",
execute(database, _flags, oldStorage) {
const oldStakeouts = oldStorage?.stakeouts;
if (!oldStakeouts || typeof oldStakeouts !== "object") return;
const reservedKeys = new Set([
"order",
"date",
"list"
]);
const oldOrder = oldStakeouts.order ?? [];
const list = [];
Object.entries(oldStakeouts).filter((entry) => !reservedKeys.has(entry[0])).forEach(([id, data]) => {
const orderIndex = oldOrder.indexOf(id);
list.push({
...data,
id: parseInt(id),
order: orderIndex !== -1 ? orderIndex : Date.now()
});
});
database.stakeouts.list = list;
}
},
{
id: "b2c3d4e5-f6a7-8901-bcde-f12345678901",
version: "9.0.6",
execute(database, _flags, oldStorage) {
const oldFactionStakeouts = oldStorage?.factionStakeouts;
if (!oldFactionStakeouts || typeof oldFactionStakeouts !== "object") return;
const reservedKeys = new Set(["date", "list"]);
const list = [];
Object.entries(oldFactionStakeouts).filter((entry) => !reservedKeys.has(entry[0])).forEach(([id, data]) => {
list.push({
...data,
id: parseInt(id),
order: Date.now()
});
});
database.factionStakeouts.list = list;
}
}
];
async function executeMigrationScripts(storage, oldStorage) {
if (RUNTIME_INFORMATION.isUserscript()) return;
const migrations = MIGRATIONS.filter(({ version }) => toNumericVersion(version) >= toNumericVersion(storage.version.initial)).filter(({ id }) => !storage.migrations.map(({ id }) => id).includes(id));
const flags = {
updateUserdata: false,
updateFactiondata: false,
updateTorndata: false,
clearCache: false
};
migrations.reverse().filter((migration) => {
migration.execute(storage, flags, oldStorage);
storage.migrations.push({ id: migration.id });
});
if (flags.updateUserdata) storage.userdata.date = 0;
if (flags.updateFactiondata) storage.factiondata.date = 0;
if (flags.updateTorndata) storage.torndata.date = 0;
if (flags.clearCache) storage.cache = {};
}
var settings;
var filters;
var version;
var api;
var userdata;
var torndata;
var stakeouts;
var attackHistory;
var notes;
var factiondata;
var quick;
var localdata;
var npcs;
var notificationHistory;
var stockdata;
var factionStakeouts;
var notifications;
var migrations;
var databaseLoaded = false;
var databaseLoading = false;
var storageListeners = {
settings: [],
filters: [],
version: [],
userdata: [],
torndata: [],
attackHistory: [],
stakeouts: [],
factionStakeouts: [],
notes: [],
factiondata: [],
localdata: [],
cache: [],
api: [],
npcs: [],
stockdata: [],
notificationHistory: [],
notifications: [],
quick: [],
migrations: []
};
async function loadDatabase(force = false) {
if (databaseLoaded && !force) return {
settings,
filters,
version,
userdata,
stakeouts,
factionStakeouts,
notes,
factiondata,
localdata,
cache: ttCache.cache,
api,
npcs,
torndata,
notificationHistory,
attackHistory,
quick,
stockdata,
notifications,
migrations
};
else if (databaseLoaded && !settings || databaseLoading) {
await sleep(75);
return await loadDatabase(force);
}
databaseLoading = true;
const database = await ttStorage.get();
populateDatabaseVariables(database);
console.log("TT - Database loaded.", database);
databaseLoaded = true;
databaseLoading = false;
return database;
}
async function migrateDatabase(force = false) {
try {
const loadedStorage = await ttStorage.get();
if (!loadedStorage || !Object.keys(loadedStorage).length) {
console.log("TT - Fresh installation detected, setting up default storage.");
await ttStorage.reset();
await loadDatabase();
return;
}
const storedVersion = loadedStorage?.version?.current || "5.0.0";
const currentVersion = RUNTIME_INFORMATION.getVersion();
console.log(`TT - Migration check: ${storedVersion} -> ${currentVersion}`);
const migratedStorage = convertStorage(loadedStorage, DEFAULT_STORAGE);
await executeMigrationScripts(migratedStorage, loadedStorage);
migratedStorage.version.current = currentVersion;
await ttStorage.set(migratedStorage);
populateDatabaseVariables(migratedStorage);
console.log("TT - Database migration completed successfully.");
} catch (error) {
console.error("TT - Database migration failed:", error);
await loadDatabase();
}
}
function convertStorage(oldStorage, defaultStorage) {
const newStorage = {};
for (const key in defaultStorage) {
if (!oldStorage) oldStorage = {};
if (!(key in oldStorage)) oldStorage[key] = {};
const defaultValue = defaultStorage[key];
if (typeof defaultValue === "object" && defaultValue !== null) if (defaultValue instanceof DefaultSetting) newStorage[key] = migrateDefaultSetting(oldStorage[key], defaultValue);
else newStorage[key] = convertStorage(oldStorage[key], defaultValue);
else newStorage[key] = oldStorage[key] ?? defaultValue;
}
return newStorage;
}
function migrateDefaultSetting(oldValue, setting) {
if (isValidSettingValue(oldValue, setting)) return oldValue;
if (setting.defaultValue) return typeof setting.defaultValue === "function" ? setting.defaultValue() : setting.defaultValue;
return null;
}
function isValidSettingValue(value, setting) {
if (setting.type === "array") return Array.isArray(value);
return setting.type.split("|").some((type) => type === "empty" && value === "" || typeof value === type);
}
function populateDatabaseVariables(database) {
settings = database.settings;
filters = database.filters;
version = database.version;
api = database.api;
userdata = database.userdata;
torndata = database.torndata;
localdata = database.localdata;
stakeouts = database.stakeouts;
attackHistory = database.attackHistory;
notes = database.notes;
factiondata = database.factiondata;
quick = database.quick;
npcs = database.npcs;
stockdata = database.stockdata;
factionStakeouts = database.factionStakeouts;
notificationHistory = database.notificationHistory;
notifications = database.notifications;
migrations = database.migrations;
ttCache.cache = database.cache;
}
function initializeDatabaseListener() {
RUNTIME_STORAGE.addChangeListener((changes, area) => {
if (area === "local") for (const key in changes) {
switch (key) {
case "settings":
settings = changes.settings.newValue;
break;
case "filters":
filters = changes.filters.newValue;
break;
case "version":
version = changes.version.newValue;
break;
case "userdata":
userdata = changes.userdata.newValue;
break;
case "api":
api = changes.api.newValue;
break;
case "torndata":
torndata = changes.torndata.newValue;
break;
case "stakeouts":
stakeouts = changes.stakeouts.newValue;
break;
case "attackHistory":
attackHistory = changes.attackHistory.newValue;
break;
case "notes":
notes = changes.notes.newValue;
break;
case "factiondata":
factiondata = changes.factiondata.newValue;
break;
case "quick":
quick = changes.quick.newValue;
break;
case "localdata":
localdata = changes.localdata.newValue;
break;
case "cache":
ttCache.cache = changes.cache.newValue;
break;
case "npcs":
npcs = changes.npcs.newValue;
break;
case "stockdata":
stockdata = changes.stockdata.newValue;
break;
case "notificationHistory":
notificationHistory = changes.notificationHistory.newValue;
break;
case "notifications":
notifications = changes.notifications.newValue;
break;
case "factionStakeouts":
factionStakeouts = changes.factionStakeouts.newValue;
break;
}
if (storageListeners[key]) storageListeners[key].forEach((listener) => listener(changes[key].oldValue, changes[key].newValue));
}
});
}
function setLocaldata(data) {
localdata = data;
}
function setFilters(data) {
filters = data;
}
function capitalizeText(text, partialOptions = {}) {
if (!{
everyWord: false,
...partialOptions
}.everyWord) return text[0].toUpperCase() + text.slice(1);
return text.trim().split(" ").map((word) => capitalizeText(word)).join(" ").trim();
}
var LINKS = {
auction: "https://www.torn.com/amarket.php",
bank: "https://www.torn.com/bank.php",
bazaar: "https://www.torn.com/bazaar.php",
bounties: "https://www.torn.com/bounties.php#!p=main",
chain: "https://www.torn.com/factions.php?step=your#/war/chain",
church: "https://www.torn.com/church.php",
committee: "https://www.torn.com/committee.php",
companies: "https://www.torn.com/companies.php",
companyEmployees: "https://www.torn.com/companies.php#/option=employees",
crimes: "https://www.torn.com/crimes.php",
donator: "https://www.torn.com/donator.php",
education: "https://www.torn.com/page.php?sid=education",
events: "https://www.torn.com/events.php#/step=all",
faction: "https://www.torn.com/factions.php",
faction__ranked_war: "https://www.torn.com/factions.php?step=your&type=1#/war/rank",
faction_oc: "https://www.torn.com/factions.php?step=your#/tab=crimes",
gym: "https://www.torn.com/gym.php",
home: "https://www.torn.com/index.php",
homepage: "https://www.torn.com/index.php",
hospital: "https://www.torn.com/hospitalview.php",
itemmarket: "https://www.torn.com/page.php?sid=ItemMarket",
items: "https://www.torn.com/item.php",
items_booster: "https://www.torn.com/item.php#boosters-items",
items_candy: "https://www.torn.com/item.php#candy-items",
items_drug: "https://www.torn.com/item.php#drugs-items",
items_medical: "https://www.torn.com/item.php#medical-items",
jailview: "https://www.torn.com/jailview.php",
jobs: "https://www.torn.com/companies.php",
loan: "https://www.torn.com/loan.php",
messages: "https://www.torn.com/messages.php",
missions: "https://www.torn.com/loader.php?sid=missions",
organizedCrimes: "https://www.torn.com/factions.php?step=your#/tab=crimes",
pc: "https://www.torn.com/pc.php",
points: "https://www.torn.com/page.php?sid=points",
pointsmarket: "https://www.torn.com/pmarket.php",
properties: "https://www.torn.com/properties.php",
property_upkeep: "https://www.torn.com/properties.php#/p=options&tab=upkeep",
property_vault: "https://www.torn.com/properties.php#/p=options&tab=vault",
raceway: "https://www.torn.com/page.php?sid=racing",
staff: "https://www.torn.com/staff.php",
stocks: "https://www.torn.com/page.php?sid=stocks",
trade: "https://www.torn.com/trade.php",
travelagency: "https://www.torn.com/page.php?sid=travel"
};
LINKS.donator, LINKS.donator, LINKS.staff, LINKS.committee, LINKS.church, LINKS.jobs, LINKS.jobs, LINKS.jobs, LINKS.jobs, LINKS.jobs, LINKS.jobs, LINKS.companies, LINKS.companies, LINKS.companies, LINKS.faction, LINKS.faction, LINKS.faction, LINKS.faction, LINKS.faction, LINKS.education, LINKS.education, LINKS.bank, LINKS.bank, LINKS.travelagency, LINKS.property_vault, LINKS.loan, LINKS.auction, LINKS.bazaar, LINKS.itemmarket, LINKS.pointsmarket, LINKS.stocks, LINKS.stocks, LINKS.trade, LINKS.homepage, LINKS.raceway, LINKS.raceway, LINKS.faction_oc, LINKS.faction_oc, LINKS.faction_oc, LINKS.faction_oc, LINKS.bounties, LINKS.bank, LINKS.auction, LINKS.auction, LINKS.hospital, LINKS.hospital, LINKS.hospital, LINKS.jailview, LINKS.hospital, LINKS.items_booster, LINKS.items_booster, LINKS.items_booster, LINKS.items_booster, LINKS.items_booster, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.items_drug, LINKS.items_drug, LINKS.items_drug, LINKS.items_drug, LINKS.items_drug, LINKS.travelagency, LINKS.travelagency, LINKS.travelagency, LINKS.travelagency, LINKS.travelagency, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.items_medical, LINKS.property_upkeep, LINKS.property_upkeep, LINKS.property_upkeep;
[
{
id: 1,
reason: "Admin"
},
{
id: 4,
reason: "NPC"
},
{
id: 7,
reason: "NPC"
},
{
id: 9,
reason: "NPC"
},
{
id: 10,
reason: "NPC"
},
{
id: 15,
reason: "NPC"
},
{
id: 17,
reason: "NPC"
},
{
id: 19,
reason: "NPC"
},
{
id: 20,
reason: "NPC"
},
{
id: 21,
reason: "NPC"
}
].map(({ id }) => id);
function is2FACheckPage() {
return !!document.querySelector(".content-wrapper.logged-out .two-factor-auth-container");
}
function getPageStatus() {
const infoMessage = document.querySelector(".content-wrapper .info-msg-cont");
if (infoMessage?.classList.contains("red")) {
if (infoMessage.textContent.includes("items in your inventory")) return { access: true };
else if (findParent(infoMessage, { class: "no-parcel-wrap" })?.style?.display === "none") return { access: true };
return {
access: false,
message: infoMessage.textContent
};
}
if (document.querySelector(".captcha")) return {
access: false,
message: "Captcha required"
};
else if (document.querySelector(".dirty-bomb")) return {
access: false,
message: "Dirty bomb screen"
};
else if (is2FACheckPage()) return {
access: false,
message: "2 Factor Authentication"
};
return { access: true };
}
var Feature = class {
name;
scope;
executionTiming;
constructor(name, scope, executionTiming = "CONTENT_LOADED") {
this.name = name;
this.scope = scope;
this.executionTiming = executionTiming;
}
precondition() {
return true;
}
initialise() {}
execute(liveReload) {}
cleanup() {}
storageKeys() {
return [];
}
requirements() {
return true;
}
shouldTriggerEvents() {
return false;
}
shouldLiveReload() {
return false;
}
requiresScreenInformation() {
return true;
}
};
function initialise() {
CUSTOM_LISTENERS[EVENT_CHANNELS.MISSION_LOAD].push(async () => {
if (!FEATURE_MANAGER.isEnabled(MissionHintsFeature)) return;
await showHints();
});
}
async function showHints() {
const MISSION_HINTS = {
a_good_day_to_get_hard: {
task: "Achieve a killstreak of 3 - 10.",
hint: "Buying losses is a valid strategy."
},
a_kimpossible_task: {
task: "Defeat (P) using only melee and temporary weapons.",
hint: "Guns can stay equipped."
},
a_problem_at_the_tracks: {
task: "Defeat 3 (P) using only fist or melee weapons.",
hint: "Other weapons can stay equipped. Using any other weapon will fail this mission!"
},
a_thor_loser: {
task: "Use Duke's hammer to hit 6 - 14 unique body parts",
hint: "Stalemating is a good way to achieve this."
},
against_the_odds: { task: "Defeat 2 (P)." },
an_honorary_degree: {
task: "Defeat (P) without using any guns",
hint: "Guns can stay equipped. Using a gun will fail this mission!"
},
army_of_one: {
task: "Attack (P) 3 times with various masks.",
hint: "Duke will send you the two masks to wear. Make sure that one of the attacks is without any mask. You just need to attack the target, defeating them isn't a requirement."
},
bakeout_breakout: { task: "Combine a fruitcake and the lock pick, and send 'special fruitcake' to someone in jail." },
bare_knuckle: {
task: "Defeat (P) with no weapons or armor equipped.",
hint: "Unequip everything. Residual effects from previous fights will fail this mission!"
},
batshit_crazy: {
task: "Inflict damage with Penelope.",
hint: "You'll receive Penelope from Duke. Amount of damage is apparently based on your maximum life."
},
battering_ram: { task: "Attack (P) 3 times." },
big_tub_of_muscle: { task: "Defeat (P) despite their gargantuan strength." },
birthday_surprise: {
task: "Obtain and send a specific item as a present to Duke.",
hint: "Place the item in an empty box, then gift wrap it to get it as a parcel."
},
bonnie_and_clyde: { task: "Defeat (P), and their spouse." },
bountiful: {
task: "Successfully claim 2 - 5 bounties",
hint: "Make sure to hospitalize the target."
},
bounty_on_the_mutiny: {
task: "Bounty (P) and wait for the bounty to be claimed.",
hint: "You cannot claim your own bounties."
},
bring_it: {
task: "Defeat Duke in a group attack",
hint: "Unlike other missions, you'll have a week to finish this one. Doesn't have to be the finishing hit, so just join when people try to loot him."
},
candy_from_babies: {
task: "Collect $50,000 - $250,000 in bounties.",
hint: "Doesn't have to be in a single bounty. Make sure to hospitalize the target(s)."
},
charity_work: {
task: "Mug 2 (P)",
hint: "Must be a successful mug. Sending them a small amount will guarantee a mug."
},
cracking_up: {
task: "Defeat and interrogate (P) for the code to unlock Duke's safe and send the content to Duke after opening it.",
hint: "Can take a few times to get the code."
},
critical_education: { task: "Achieve 3 - 9 critical hits" },
cut_them_down_to_size: { task: "Defeat any player of your level or higher." },
dirty_little_secret: {
task: "Put a bounty on (P), then attack the person who claimed it.",
hint: "If the claimer is anonymous, their ID # still shows up in the mission screen."
},
double_jeopardy: {
task: "Put a bounty on someone and defeat them.",
hint: "Bounty can be any amount as it doesn't have to be claimed."
},
drug_problem: { task: "Defeat 4 - 7 (P)." },
emotional_debt: { task: "Hit (P) with tear gas or pepper spray." },
estranged: {
task: "Injure one of (P)'s legs.",
hint: "Feet count as legs in this instance."
},
family_ties: { task: "Hospitalize (P) 3 times" },
field_trip: { task: "Win $100 - $1,000,000 on 3 casino games." },
fireworks: { task: "Expend 250 - 1250 rounds of ammunition" },
forgotten_bills: { task: "Defeat (P)" },
frenzy: {
task: "Defeat any 5 - 15 players.",
hint: "You must initiate the attacks; buying losses will not work for this mission"
},
get_things_jumping: {
task: "Deal and receive damage.",
hint: "Values are apparently based on your maximum life."
},
graffiti: {
task: "Hit (P) with pepper spray.",
hint: "Even if it's ineffective, it still counts."
},
guardian: { task: "Defeat (P)." },
hammer_time: {
task: "Defeat (P) with a hammer.",
hint: "Guns can stay equipped. Dual hammers don't count."
},
hands_off: { task: "Defeat 3 - 5 (P)." },
hare_meet_tortoise: {
task: "Defeat (P) despite their lightning fast speed.",
hint: "Flash and smoke grenades will reduce their speed."
},
hide_and_seek: {
task: "Find (P) from 3 - 5 listed and defeat them.",
hint: "Given clues make it easy to identify the target."
},
hiding_in_plain_view: { task: "Defeat (P) in a random country." },
high_fliers: { task: "Defeat 3 (P) in random countries" },
hobgoblin: { task: "Defeat a player of your choice 5 times" },
immovable_object: { task: "Defeat (P) despite their impenetrable defense." },
inside_job: {
task: "Attack (P) and secrete an item on them.",
hint: "Duke will send you the item. The Secrete option will appear after defeating (P)."
},
introduction_duke: { task: "Complete 10 Duke contracts." },
keeping_up_appearances: {
task: "Mug (P) and send them back the money.",
hint: "Must be a successful mug. Sending them a small amount will guarantee a mug."
},
kiss_of_death: { task: "Defeat (P) and use the kiss option." },
lack_of_awareness: { task: "Defeat (P)." },
lost_and_found: { task: "Put (P) in the hospital for 12 hours." },
loud_and_clear: {
task: "Use 3 - 11 explosive grenades.",
hint: "Some explosive grenades are 'Flash Grenade', 'HEG', 'Grenade', <a href='https://wiki.torn.com/wiki/Explosive_Grenade' target='_blank'>and more</a>."
},
loyal_customer: { task: "Defeat (P)." },
make_it_slow: {
task: "Defeat (P) in no fewer than 5 - 9 turns in a single attack.",
hint: "Survive for said amount of turns then defeat the target. You can keep retrying if it fails."
},
marriage_counseling: { task: "Defeat (P)'s spouse." },
massacrist: { task: "Defeat (P)." },
meeting_the_challenge: { task: "Mug people for a total of $10,000 - $16,000,000." },
motivator: {
task: "Lose or stalemate to (P) on the first attempt.",
hint: "You can get your health low by using the wrong blood bag. Make yourself weak by unequipping armor and equip a rusty sword."
},
new_kid_on_the_block: { task: "Defeat 5 players." },
no_man_is_an_island: {
task: "Mug 2 out of 3 (P).",
hint: "You can select which of the targets to hit, as long as you mug 2 different ones."
},
no_second_chances: { task: "Defeat (P) on the first attempt." },
out_of_the_frying_pan: { task: "Go to the jail, use Felovax to go to the hospital, then use Zylkene." },
painleth_dentitht: {
task: "Defeat (P) with a baseball bat.",
hint: "Other weapons can stay equipped."
},
party_tricks: { task: "Defeat (P) despite their nimble dexterity." },
pass_the_word: {
task: "Send a message including keyword to (P).",
hint: "It's easy to achieve this by copying the mission description."
},
peak_experience: { task: "Defeat (P)." },
proof_of_the_pudding: {
task: "Use a specific weapon on (P), then send the weapon to them.",
hint: "You don't have to send the exact same gun you used to attack with, just same type."
},
rabbit_response: {
task: "Defeat 3 (P) within 30 - 10 minutes.",
hint: "Timer starts after attacking one of the targets, so make sure they are all out of the hospital."
},
reconstruction: {
task: "Equip kitchen knife and leather gloves, defeat (P) then dump both items.",
hint: "Don't have to actually use the kitchen knife."
},
red_faced: { task: "Defeat (P) with a trout on the finishing hit." },
rising_costs: {
task: "Hit (P) with a brick.",
hint: "Brick has to hit, a miss won't count."
},
rolling_in_it: {
task: "Mug (P).",
hint: "Must be a successful mug. Sending them a small amount will guarantee a mug."
},
safari: {
task: "Defeat (P) with a rifle in South Africa.",
hint: "All hits need to be done with a rifle weapon type. Other weapons can stay equipped."
},
scammer: {
task: "Defeat (P).",
hint: "Target might have some nice cash on them, mugging could be beneficial."
},
sellout_slayer: {
task: "Buy a gun, use the gun on any 2 - 6 players, then sell it again.",
hint: "Not every non-melee weapon is a gun. As example, a blowgun might not work."
},
sending_a_message: { task: "Defeat (P)." },
show_some_muscle: {
task: "Attack (P).",
hint: "You just need to attack the target, defeating them isn't a requirement."
},
sleep_aid: { task: "Defeat (P)." },
some_people: { task: "Send any item as a parcel to (P)." },
standard_routine: { task: "Defeat (P) with a clubbed weapon, fists or kick." },
stomach_upset: { task: "Injure (P)'s stomach." },
swan_step_too_far: {
task: "Get an item from the dump and defeat its previous owner.",
hint: "You can keep searching till you find an item previously owned by someone you can actually defeat."
},
the_executive_game: {
task: "Defeat (P) using only fists or kick.",
hint: "Weapons can stay equipped."
},
the_tattoo_artist: {
task: "Defeat (P) using only a slashing or piercing weapon.",
hint: "Guns can stay equipped."
},
three_peat: { task: "Leave any player, mug any player and hospitalize any player." },
training_day: { task: "Use 250 - 1,250 energy in the gym." },
tree_huggers: { task: "Defeat 5 - 8 (P)." },
undercutters: { task: "Defeat 3 (P)." },
unwanted_attention: { task: "Hospitalize 4 (P)." },
withdrawal: {
task: "Injure (P)'s both arms.",
hint: "Hands count as arms in this case."
},
wrath_of_duke: { task: "Defeat 4 (P)." }
};
for (const context of findAllElements(".giver-cont-wrap > div[id^=mission]:not(.tt-modified)")) {
const title = context.querySelector(".title-black").childNodes[0].wholeText.replace(/\n/g, "").trim();
const key = transformTitle(title);
let task, hint;
if (key in MISSION_HINTS) {
const mission = MISSION_HINTS[key];
task = mission.task;
hint = mission.hint ? elementBuilder({
type: "span",
html: mission.hint
}) : null;
} else if (title.includes("{name}")) {
task = "You are using a conflicting script.";
hint = "Please remove the script that changes the mission title or contact the TornTools developers.";
} else {
task = "Couldn't find information for this mission.";
hint = "Contact the TornTools developers.";
}
const children = [elementBuilder({
type: "h6",
class: "tt-mission-title",
text: "TornTools Mission Information"
}), elementBuilder({
type: "span",
children: [elementBuilder({
type: "b",
text: "Task: "
}), task]
})];
if (hint) children.push(elementBuilder("br"), elementBuilder({
type: "span",
children: [elementBuilder({
type: "b",
text: "Hint: "
}), hint]
}));
context.querySelector(".max-height-fix").appendChild(elementBuilder({
type: "div",
class: "tt-mission-information",
children
}));
context.classList.add("tt-modified");
}
function transformTitle(title) {
return title.toLowerCase().replaceAll(" ", "_").replaceAll(":", "").replaceAll("-", "_").replaceAll("!", "").replaceAll(",", "");
}
}
function removeHints() {
for (const context of findAllElements(".giver-cont-wrap > div[id^=mission].tt-modified")) context.classList.remove(".tt-modified");
for (const information of findAllElements(".tt-mission-information")) information.remove();
}
var MissionHintsFeature = class extends Feature {
constructor() {
super("Mission Hints", "missions");
}
precondition() {
return getPageStatus().access;
}
isEnabled() {
return settings.pages.missions.hints;
}
initialise() {
initialise();
}
async execute() {
await showHints();
}
cleanup() {
removeHints();
}
storageKeys() {
return ["settings.pages.missions.hints"];
}
};
var RequestListenerInjector = class {
injectListeners;
id;
constructor(injectListeners) {
this.injectListeners = injectListeners;
this.id = capitalizeText(injectListeners.name);
}
inject() {
if (this.isInjected()) return;
this.injectListeners();
this.setInjected();
}
isInjected() {
return document.documentElement.dataset[`tt${this.id}`] === "true";
}
setInjected() {
document.documentElement.dataset[`tt${this.id}`] = "true";
}
};
function injectFetchListeners() {
const oldFetch = SCRIPT_INJECTOR.getWindow().fetch;
SCRIPT_INJECTOR.getWindow().fetch = (input, init) => new Promise((resolve, reject) => {
oldFetch(input, init).then(async (response) => {
const page = response.url.substring(response.url.indexOf("torn.com/") + 9, response.url.indexOf(".php"));
let json = {};
try {
json = await response.clone().json();
} catch {}
let body = null;
if (init) {
body = init.body;
if (body !== null && typeof body === "object" && body?.constructor?.name === "FormData") {
const newBody = {};
for (const [key, value] of [...body]) if (isIntNumber(value)) newBody[key] = parseFloat(value);
else newBody[key] = value;
body = newBody;
}
}
const url = response.url || input;
const detail = {
page,
json,
text: await response.clone().text(),
fetch: {
url,
body,
status: response.status
}
};
window.dispatchEvent(new CustomEvent("tt-fetch", { detail }));
resolve(response);
}).catch((error) => {
reject(error);
});
});
}
function injectXhrListeners() {
const oldXHROpen = window.XMLHttpRequest.prototype.open;
const oldXHRSend = window.XMLHttpRequest.prototype.send;
window.XMLHttpRequest.prototype.open = function(method, url) {
let params = this["params"] ?? {};
if ("xhrOpenAdjustments" in window && typeof window.xhrOpenAdjustments === "object") for (const key in window.xhrOpenAdjustments) {
if (typeof window.xhrOpenAdjustments[key] !== "function") continue;
const adjustments = window.xhrOpenAdjustments[key]({ ...this }, method, url);
method = adjustments.method;
url = adjustments.url;
params = {
...params,
...adjustments.params || {}
};
}
this["method"] = method;
this["url"] = url;
this["params"] = params;
this.addEventListener("readystatechange", function() {
if (this.readyState > 3 && this.status === 200) {
const page = this.responseURL.substring(this.responseURL.indexOf("torn.com/") + 9, this.responseURL.indexOf(".php"));
let json, uri;
if (isJsonString(this.response)) json = JSON.parse(this.response);
else uri = getUrlParams(this.responseURL);
let text;
if (this.responseType === "" || this.responseType === "text") text = this.responseText;
window.dispatchEvent(new CustomEvent("tt-xhr", { detail: {
page,
json,
uri,
xhr: {
requestBody: this["requestBody"],
response: this.response,
responseType: this.responseType,
responseText: text,
responseURL: this.responseURL
}
} }));
}
});
arguments[0] = method;
arguments[1] = url;
return oldXHROpen.apply(this, arguments);
};
window.XMLHttpRequest.prototype.send = function(body) {
this["params"] = this["params"] ?? {};
if ("xhrSendAdjustments" in window && typeof window.xhrSendAdjustments === "object") for (const key in window.xhrSendAdjustments) {
if (typeof window.xhrSendAdjustments[key] !== "function") continue;
body = window.xhrSendAdjustments[key]({ ...this }, body);
}
this["requestBody"] = body;
arguments[0] = body;
return oldXHRSend.apply(this, arguments);
};
}
function getUrlParams(url, prop) {
if (!url) url = location.href;
const definitions = decodeURIComponent(url.slice(url.indexOf("?") + 1)).split("&");
const params = {};
definitions.forEach((val) => {
const parts = val.split("=", 2);
params[parts[0]] = parts[1];
});
return prop && prop in params ? params[prop] : params;
}
function isJsonString(str) {
if (!str || str === "") return false;
try {
JSON.parse(str);
} catch {
return false;
}
return true;
}
var TornToolsStorage = class {
change(object) {
return new Promise(async (resolve) => {
const keys = Object.keys(object);
for (const key of keys) {
const data = this.recursive(await this.get(key), object[key]);
await this.set({ [key]: data });
}
resolve();
});
}
recursive(parent, toChange) {
for (const key in toChange) if (parent && typeof parent === "object" && !Array.isArray(parent[key]) && key in parent && typeof toChange[key] === "object" && !Array.isArray(toChange[key]) && toChange[key] !== null) parent[key] = this.recursive(parent[key], toChange[key]);
else if (parent && typeof parent === "object") {
const value = toChange[key];
parent[key] = Array.isArray(value) ? Array.from(value) : value;
} else parent = { [key]: toChange[key] };
return parent;
}
};
var TTScriptStorage = class extends TornToolsStorage {
prefix;
constructor(prefix) {
super();
this.prefix = prefix;
}
storageKey(key) {
return key === "cache" ? key : `${this.prefix}_${key}`;
}
async get(key) {
if (Array.isArray(key)) return await Promise.all(key.map((k) => this.storageKey(k)).map((k) => GM.getValue(k)));
else if (key) return await GM.getValue(this.storageKey(key));
else {
const storageKeys = Object.keys(DEFAULT_STORAGE);
const storageValues = await this.get(storageKeys);
return storageKeys.reduce((total, k, i) => {
total[k] = storageValues[i];
return total;
}, {});
}
}
async set(object) {
await Promise.all(Object.entries(object).map(([key, value]) => {
UserscriptRuntimeStorage.callback({ [key]: {
newValue: value,
oldValue: null
} }, "local");
return GM.setValue(this.storageKey(key), value);
}));
}
remove(_key) {
throw new Error("Method not implemented.");
}
clear() {
throw new Error("Method not implemented.");
}
reset(_key) {
throw new Error("Method not implemented.");
}
getSize() {
throw new Error("Method not implemented.");
}
};
_css(".tt-hidden{display:none!important}.tt-black-overlay{z-index:100;background-color:#00000059;width:100%;height:100%;position:fixed;top:0;left:0}.no-margin{margin:0}.tt-delimiter{border-top:#ccc;border-left:none;border-right:none;border-top:1px solid var(--sidebar-horizontal-divider-bg-color);border-bottom:#fff;border-bottom:1px solid var(--sidebar-horizontal-divider-shadow-color);height:0;margin-bottom:5px;overflow:hidden}.tt-overlay{z-index:1000000;background-color:#00000059;width:100%;height:100%;position:fixed;top:0;left:0}.tt-overlay-item,.tt-overlay-item-notbroken{z-index:999999999;position:relative}.tt-overlay-item .tt-overlay-ignore{z-index:0;pointer-events:none}.tt-overlay-item .tt-overlay-ignore:before{content:\"\";z-index:1000000;background-color:#00000059;width:100%;height:100%;position:absolute;top:0;left:0}.relative{position:relative}.flex-break{border:0;height:0;margin:0;flex-basis:100%!important}.mt10{margin-top:10px}.mb10{margin-bottom:10px}.t-flex{display:flex}[class*=torn-icon-]{vertical-align:middle;background:url(https://www.torn.com/images/v2/city/location_icons_34x34px.svg) no-repeat;width:34px;height:34px;display:inline-block}.torn-icon-item-market{background-position:-68px -34px}.tt-sidebar-area{margin-top:2px;overflow:hidden}.tt-sidebar-area>div{cursor:pointer;vertical-align:top;background-color:var(--default-bg-panel-color);border-top-right-radius:5px;border-bottom-right-radius:5px;position:relative;overflow:hidden}.tt-sidebar-area a{color:var(--default-content-font-color);justify-content:flex-start;align-items:center;height:100%;text-decoration:none;display:flex;overflow:hidden}.tt-sidebar-area a span{float:none;vertical-align:middle;margin-left:10px;display:inline-block}.tt-button-link{cursor:pointer;color:var(--default-blue-color)}.tt-btn{background-color:var(--tt-color-light-green);color:#000;border-radius:6px;width:fit-content}.tt-btn:not([disabled]){cursor:pointer}.tt-btn[disabled]{cursor:not-allowed;opacity:.4}.tt-msg-box{background:var(--info-msg-grey-gradient);box-shadow:var(--info-msg-box-shadow);color:var(--info-msg-font-color);border-radius:5px;margin-top:10px;font-size:0;line-height:16px}.tt-msg-box .tt-msg-div{background:var(--info-msg-horizontal-gradient);border-radius:5px;justify-content:flex-start;display:flex}.tt-msg-box .tt-msg{vertical-align:middle;background-color:var(--default-bg-panel-active-color);background:var(--info-msg-delimiter-gradient);border-radius:0 5px 5px 0;flex-grow:1;width:1px;height:auto}.tt-msg-box .tt-content{vertical-align:middle;color:var(--info-msg-font-color);background-color:var(--default-bg-panel-active-color);background:var(--info-msg-bg-gradient);border-radius:0 5px 5px 0;padding:10px;font-size:13px;position:relative}.tt-message-box{color:var(--info-msg-font-color);box-shadow:var(--info-msg-box-shadow);border-radius:5px;margin-top:10px;font-size:13px;display:flex}.tt-message-box .tt-message-icon-wrap{background:var(--info-msg-grey-gradient);border-radius:5px 0 0 5px;width:34px}.tt-message-box .tt-message-icon{background:var(--info-msg-horizontal-gradient);border-radius:5px 0 0 5px;justify-content:center;width:34px;height:100%;display:flex}.tt-message-box .tt-svg{width:34px;height:34px}.tt-message-box .tt-message-wrap{background-color:var(--default-bg-panel-active-color);background:var(--info-msg-bg-gradient);border-radius:0 5px 5px 0;flex-grow:1;align-items:center;padding:10px;display:flex}.tt-message-box .tt-message{flex-grow:1}.tt-svg{width:128px;height:128px}.tt-svg .tt-svg-upper{stroke:#000;fill:#000}.tt-svg .tt-svg-lower{stroke:#568725;fill:#568725}#sidebarroot .pill{cursor:pointer;background-color:var(--default-bg-panel-color);min-height:22px;color:var(--default-font-color);border-top-right-radius:5px;border-bottom-right-radius:5px;align-items:center;margin-top:2px;text-decoration:none;display:flex;overflow:hidden}#sidebarroot .pill:not([icon]){box-sizing:border-box;padding-top:5px;padding-bottom:5px}#sidebarroot .pill:not([icon]),#sidebarroot .pill[icon] span{height:100%;color:var(--default-font-color);justify-content:flex-start;align-items:center;padding-left:8px;text-decoration:none;display:flex;overflow:hidden}body.tt-tablet #sidebarroot .pill{min-height:34px}body[data-layout=hospital] #sidebarroot .pill{margin-top:0;margin-bottom:1px}#sidebarroot .pill:hover{background-color:var(--default-bg-panel-active-color)!important}.tt-sidebar-information{flex-direction:column;display:flex}.tt-sidebar-information .title{color:inherit;margin:inherit;font-weight:700;text-decoration:none}.tt-sidebar-information .countdown.short{color:var(--tt-color-red)}.tt-sidebar-information .countdown.medium{color:var(--tt-color-orange)}.tt-top-icons{gap:10px;display:flex}");
_css(":root{--tt-color-green:#00a500;--tt-color-light-green:#acea00;--tt-color-red:#d83500;--tt-color-green--20:#00a50033;--tt-color-green--30:#00a5004d;--tt-color-green--40:#00a50066;--tt-background-torn-gray:repeating-linear-gradient(90deg, #627e0d, #627e0d 2px, #6e8820 0, #6e8820 4px);--tt-background-green:repeating-linear-gradient(90deg, #627e0d, #627e0d 2px, #6e8820 0, #6e8820 4px);--tt-background-alternative:repeating-linear-gradient(90deg, #242424, #242424 2px, #2e2e2e 0, #2e2e2e 4px)}body:not(.dark-mode){--tt-color-blue:blue;--tt-color-orange:orange;--tt-color-item-text:#678c00;--tt-color-item-quantity:black;--tt-background-popup:#f1f1f1;--tt-shadow-popup:unset}body.dark-mode{--tt-color-blue:#058cff;--tt-color-orange:gold;--tt-color-item-text:#9c0;--tt-color-item-quantity:#ddd;--tt-background-popup:#444;--tt-shadow-popup:0 0 10px black}.tt-color-green{color:var(--tt-color-green)}.tt-color-red{color:var(--tt-color-red)}");
async function registerUserscriptContext(storagePrefix) {
setTTStorage(new TTScriptStorage(storagePrefix));
setFeatureManager(new ScriptFeatureManager());
setScriptInjector(UserscriptScriptInjector);
setRuntimeInformation(UserscriptRuntimeInformation);
setRuntimeStorage(UserscriptRuntimeStorage);
await migrateDatabase(true);
initializeDatabaseListener();
const [localdata, filters, cache] = await ttStorage.get([
"localdata",
"filters",
"cache"
]);
setLocaldata(localdata ? localdata : getDefaultStorage(DEFAULT_STORAGE.localdata));
setFilters(filters ? filters : getDefaultStorage(DEFAULT_STORAGE.filters));
ttCache.cache = cache ? cache : getDefaultStorage(DEFAULT_STORAGE.cache);
initializeScriptTheme();
}
function initializeScriptTheme() {
document.documentElement.style.setProperty("--tt-theme-color", "#fff");
document.documentElement.style.setProperty("--tt-theme-background", "var(--tt-background-green)");
}
var ScriptFeatureManager = class {
createPopup() {}
isEnabled() {
return true;
}
registerFeature(feature) {
feature.initialise();
feature.execute();
}
};
var fetchListenerInjector = new RequestListenerInjector(injectFetchListeners);
var xhrListenerInjector = new RequestListenerInjector(injectXhrListeners);
var UserscriptScriptInjector = {
getWindow() {
return unsafeWindow;
},
injectFetch() {
fetchListenerInjector.inject();
},
injectXHR() {
xhrListenerInjector.inject();
}
};
var UserscriptRuntimeInformation = {
getVersion() {
return GM.info.version;
},
isUserscript() {
return true;
}
};
var UserscriptRuntimeStorage = {
callback: () => {},
addChangeListener(callback) {
this.callback = callback;
}
};
(async () => {
await registerUserscriptContext("tt_mh");
setupMissionsPage();
const feature = new MissionHintsFeature();
FEATURE_MANAGER.registerFeature(feature);
})();
})();