Dead Frontier - API

Dead Frontier API

This script should not be not be installed directly. It is a library for other scripts to include with the meta directive // @require https://update.greatest.deepsurf.us/scripts/441829/1577303/Dead%20Frontier%20-%20API.js

  1. // ==UserScript==
  2. // @name Dead Frontier - API
  3. // @namespace Dead Frontier - Shrike00
  4. // @match *://fairview.deadfrontier.com/onlinezombiemmo/index.php?page=24
  5. // @match *://fairview.deadfrontier.com/onlinezombiemmo/index.php?page=25
  6. // @match *://fairview.deadfrontier.com/onlinezombiemmo/index.php?page=35
  7. // @grant none
  8. // @version 0.1.31
  9. // @author Shrike00
  10. // @description Dead Frontier API
  11. // ==/UserScript==
  12.  
  13. // Changelog
  14. // 0.1.31 - April 25, 2025
  15. // - Bugfix: Fixed backpack functions.
  16. // 0.1.30 - April 25, 2025
  17. // - Change: Added useBackpackItem.
  18. // 0.1.29 - April 18, 2025
  19. // - Change: Updated itemcat to itemtype.
  20. // 0.1.28 - February 22, 2025
  21. // - Feature: Added discardBackpackItem.
  22. // 0.1.27 - February 21, 2025
  23. // - Change: Added length limit to search string.
  24. // 0.1.26 - February 21, 2025
  25. // - Change: Rollback 0.1.25
  26. // 0.1.25 - February 21, 2025
  27. // - Change: Encoded search strings with encodeURI.
  28. // 0.1.24 - February 19, 2025
  29. // - Change: Added data and raw_data getters to PlayerValues.
  30. // 0.1.23 - February 7, 2025
  31. // - Change: Fixed(?) implantsToImplants and added locking and unlocking inventory and backpack slots.
  32. // 0.1.22 - February 7, 2025
  33. // - Change: Added implantsToImplants.
  34. // 0.1.21 - February 7, 2025
  35. // - Change: Added backpackToImplants.
  36. // 0.1.20 - February 7, 2025
  37. // - Change: Removed window references and called in implicit scope to account for sandboxing (for non-@grant none scripts)
  38. // - Feature: Added implant mutation.
  39. // 0.1.19 - December 16, 2024
  40. // - Change: Replaced references to webCall with webCall to account for const change.
  41. // 0.1.18 - November 8, 2024
  42. // - Bugfix: Fixed backpack slot count/iterating through backpack when not wearing one.
  43. // - Bugfix: Fixed requesting rename data.
  44. // 0.1.17 - November 4, 2024
  45. // - Feature: Added search by category to MarketCache.
  46. // 0.1.16 - October 27, 2024
  47. // - Feature: Added CollectionBook.
  48. // 0.1.15 - October 25, 2024
  49. // - Bugfix: Added support for private/public selling entries.
  50. // 0.1.14 - October 24, 2024
  51. // - Bugfix: Added credits to stackable items.
  52. // - Feature: Added support for searching renamed items that is completely separate from searching regular items.
  53. // 0.1.13 - May 2, 2024
  54. // - Change: Updated tradezone ids.
  55. // 0.1.12 - April 11, 2024
  56. // - Bugfix: Fixed querying backpack from uservars.
  57. // 0.1.11 - February 25, 2024
  58. // - Bugfix: Storage moves for larger than max stacked ammo should now work properly.
  59. // 0.1.10 - February 22, 2024
  60. // - Change: Added support for backpacks.
  61. // - Change: Added mastercrafting.
  62. // 0.1.9 - February 22, 2024
  63. // - Change: Added generic market search strings that should work for most items.
  64. // 0.1.8 - February 21, 2024
  65. // - Change: Added property to Item for renames, added backpack functions for querying and moving items.
  66. // 0.1.7 - December 15, 2023
  67. // - Change: Added search string for all stockings.
  68. // 0.1.6 - October 24, 2023
  69. // - Bugfix: Item should now correctly handle cooked item names.
  70. // 0.1.5 - August 19, 2023
  71. // - Bugfix: Bank now checks for presence of bank elements.
  72. // 0.1.4 - March 21, 2023
  73. // - Change: Storage moves no longer query the storage before sending the move.
  74. // 0.1.3 - March 19, 2023
  75. // - Change: Added ammo, inventory, and selling entry counts to MarketItems.
  76. // 0.1.2 - March 15, 2023
  77. // - Change: Added additional helper functions.
  78. // - Change: Added parameter to some functions that talk to the backend. Can now choose to not update UI on return.
  79. // 0.1.1 - December 24, 2022
  80. // - Change: Added Christmas Stocking 2022.
  81.  
  82. // base.js
  83. // flshToArr(flashStr, padding, callback) takes a response and puts it to a object or sends it to a callback.
  84. // updateIntoArr(flshArr, baseArr) copies the elements of the first array into the second.
  85. // updateAllFields() updates weapon sidebar, vital stats/boosts sidebar.
  86. // renderAvatarUpdate(elem, customVars) updates character avatar.
  87. // inventory.js
  88. // populateStorage(), populateInventory(), populateImplants(), populateCharacterInventory() all update their elements
  89. // from userVars.
  90. // reloadStorageData() and reloadInventoryData() load from the backend before calling the populate functions.
  91. // TODO: Fix storage no ui updating (need to move response from storage move into local variables)
  92. const DeadFrontier = (function() {
  93. 'use strict';
  94. // TODO: Remove request methods, always re-query. Also remove uservars, global data?
  95. // TODO: Figure out how to request/force update global data.
  96. // Helpers
  97.  
  98. function typeSplit(type) {
  99. // Splits item id into components.
  100. return type.split("_");
  101. }
  102.  
  103. function stringGroups(s, size) {
  104. // Splits string into array of substrings of length size or smaller.
  105. const output = [];
  106. for (let i = 0; i < s.length; i += size) {
  107. output.push(s.substring(i, i + size));
  108. }
  109. return output;
  110. }
  111.  
  112. function responseToMap(response) {
  113. // Converts raw string response to Map.
  114. const map = new Map();
  115. const pairs = response.split("&");
  116. for (let i = 0; i < pairs.length; i++) {
  117. const [key, value] = pairs[i].split("=");
  118. map.set(key, value);
  119. }
  120. map.delete(""); // Removes undefined key, since response leads with an ampersand (&).
  121. return map;
  122. }
  123.  
  124. function updateCashBank(cash_on_hand, cash_in_bank) {
  125. // Updates cash and bank amount elements with provided values.
  126. const cash = "Cash: $"+nf.format(cash_on_hand);
  127. const bank = "Bank: $"+nf.format(cash_in_bank);
  128. $(".heldCash").each(function(cashKey, cashVal) {
  129. $(cashVal).text(cash).attr("data-cash", cash);
  130. });
  131. const bank_element_exists = $("#bankCash").length > 0;
  132. if (bank_element_exists){
  133. $("#bankCash").text(bank).attr("data-cash", bank);
  134. }
  135. }
  136.  
  137. function promiseWait(dt) {
  138. // Returns promise that waits the given number of ms.
  139. const promise = new Promise(function(resolve, reject) {
  140. setTimeout(resolve, dt);
  141. });
  142. return promise;
  143. }
  144.  
  145. function queryObjectByKey(obj, key, dt = 100) {
  146. // Returns promise that resolves with the object once it contains the given key.
  147. if (obj[key] !== undefined) {
  148. return Promise.resolve(obj);
  149. }
  150. const promise = new Promise(function(resolve, reject) {
  151. const check = setInterval(function() {
  152. const key_exists = obj[key] !== undefined;
  153. if (key_exists) {
  154. clearInterval(check);
  155. resolve(obj);
  156. }
  157. }, dt);
  158. });
  159. return promise;
  160. }
  161.  
  162. function queryObjectByKeys(obj, keys, dt) {
  163. // Returns promise that resolves with the object once it contains all the given keys.
  164. const unique_keys = new Set(keys);
  165. const promises = Array.from(unique_keys).map((key) => queryObjectForKey(obj, key, dt));
  166. return Promise.all(promises).then(() => obj);
  167. }
  168.  
  169. function isMastercrafted(item) {
  170. // Returns if item is mastercrafted.
  171. const properties = item.properties;
  172. const category = item.category
  173. const is_mastercrafted = properties.has("mastercrafted") && properties.get("mastercrafted") === true;
  174. const is_enhanceable = isEnhanceable(item);
  175. return is_enhanceable && is_mastercrafted;
  176. }
  177.  
  178. function isNearGodcrafted(item) {
  179. // Returns if item is near-godcrafted.
  180. const properties = item.properties;
  181. const category = item.category;
  182. const is_mastercrafted = properties.has("mastercrafted") && properties.get("mastercrafted") === true;
  183. if (category === ItemCategory.WEAPON) {
  184. const total_stats = properties.get("accuracy") + properties.get("reloading") + properties.get("critical_hit");
  185. return total_stats === 23;
  186. } else if (category === ItemCategory.ARMOUR) {
  187. const total_stats = properties.get("agility") + properties.get("endurance");
  188. return total_stats === 47;
  189. } else if (category === ItemCategory.BACKPACK) {
  190. return properties.get("bonus_slots") === 2;
  191. } else {
  192. return false;
  193. }
  194. }
  195.  
  196. function isGodcrafted(item) {
  197. // Returns if item is godcrafted.
  198. const properties = item.properties;
  199. const category = item.category;
  200. const is_mastercrafted = properties.has("mastercrafted") && properties.get("mastercrafted") === true;
  201. const godcrafted_weapon = is_mastercrafted && category === ItemCategory.WEAPON
  202. && properties.get("accuracy") === 8 && properties.get("reloading") === 8 && properties.get("critical_hit") === 8;
  203. const godcrafted_armour = is_mastercrafted && category === ItemCategory.ARMOUR
  204. && properties.get("agility") === 24 && properties.get("endurance") === 24;
  205. const godcrafted_backpack = is_mastercrafted && category === ItemCategory.BACKPACK
  206. && properties.get("bonus_slots") === 3;
  207. return godcrafted_weapon || godcrafted_armour || godcrafted_backpack;
  208. }
  209.  
  210. function isCooked(item) {
  211. // Returns if item is cooked.
  212. return item.properties.has("cooked") && item.properties.get("cooked") === true;
  213. }
  214.  
  215. function isStackable(item) {
  216. // Returns if item is stackable.
  217. return item.category === ItemCategory.AMMO || item.base_type === "credits";
  218. }
  219.  
  220. function isEnhanceable(item) {
  221. // Returns if an item can be mastercrafted/godcrafted.
  222. const enhanceable = new Set([ItemCategory.WEAPON, ItemCategory.ARMOUR, ItemCategory.BACKPACK]);
  223. return enhanceable.has(item.category);
  224. }
  225.  
  226. // Enums
  227.  
  228. const Tradezone = {
  229. // The integers used are the same as the internal Dead Frontier representation, so they cannot be changed.
  230. OUTPOST: 21,
  231. CAMP_VALCREST: 22,
  232. // NASTYAS_HOLDOUT: 4,
  233. // DOGGS_STOCKAGE: 10,
  234. // PRECINCT_13: 11,
  235. // FORT_PASTOR: 12,
  236. // SECRONOM_BUNKER: 13,
  237. WASTELANDS: 10,
  238. NW: 1,
  239. N: 2,
  240. NE: 3,
  241. W: 4,
  242. CENTRAL: 5,
  243. E: 6,
  244. SW: 7,
  245. S: 8,
  246. SE: 9
  247. };
  248.  
  249. const ItemCategory = {
  250. AMMO: "ammo",
  251. WEAPON: "weapon",
  252. ARMOUR: "armour",
  253. BACKPACK: "backpack",
  254. ITEM: "item",
  255. OTHER: "other"
  256. };
  257.  
  258. const ItemSubcategory = {
  259. FOOD: "food",
  260. MEDICINE: "medicine",
  261. IMPLANT: "implant",
  262. CLOTHING: "clothing",
  263. BARRICADE: "barricade",
  264. OTHER: "other"
  265. };
  266.  
  267. const ServiceType = {
  268. CHEF: "Chef",
  269. DOCTOR: "Doctor",
  270. ENGINEER: "Engineer"
  271. };
  272.  
  273. const ProficiencyType = {
  274. MELEE: "melee",
  275. PISTOL: "pistol",
  276. RIFLE: "rifle",
  277. SHOTGUN: "shotgun",
  278. MACHINEGUN: "machinegun",
  279. EXPLOSIVE: "explosive"
  280. };
  281.  
  282. const UiUpdate = {
  283. YES: true,
  284. NO: false
  285. };
  286.  
  287. // Predicates
  288.  
  289. const ItemFilters = {
  290. Mastercrafted: (item) => isMastercrafted(item),
  291. NearGodcrafted: (item) => isNearGodcrafted(item),
  292. Godcrafted: (item) => isGodcrafted(item),
  293. Cooked: (item) => isCooked(item),
  294. Enhanceable: (item) => isEnhanceable(item)
  295. };
  296.  
  297. const ServiceFilters = {
  298. ServiceLevel: function(level) {
  299. return (service) => service.level === level;
  300. },
  301. ServiceLevels: function(levels) {
  302. return (service) => levels.includes(service.level);
  303. },
  304. ServiceLevelAtLeast: function(level) {
  305. return (service) => service.level >= level;
  306. }
  307. };
  308.  
  309. const MarketFilters = {
  310. ServiceLevel: function(level) {
  311. return (market_entry) => market_entry.service.level === level;
  312. },
  313. ServiceLevels: function(levels) {
  314. return (market_entry) => levels.includes(market_entry.service.level);
  315. },
  316. ServiceLevelAtLeast: function(level) {
  317. return (market_entry) => market_entry.service.level >= level;
  318. },
  319. Mastercrafted: (market_entry) => isMastercrafted(market_entry.item),
  320. NearGodcrafted: (market_entry) => isNearGodcrafted(market_entry.item),
  321. Godcrafted: (market_entry) => isGodcrafted(market_entry.item),
  322. Cooked: (market_entry) => isCooked(market_entry.item),
  323. Enhanceable: (market_entry) => isEnhanceable(market_entry.item)
  324. };
  325.  
  326. // Classes
  327.  
  328. // Item, ItemMarketEntry, Service, ServiceMarketEntry
  329. // These classes are all simple data classes meant to hold information.
  330.  
  331. class Item {
  332. static #global_data = globalData;
  333.  
  334. #makeProperties(full_type) {
  335. const base_type = typeSplit(full_type)[0];
  336. const data = Item.#global_data[base_type];
  337. const is_weapon = data.itemtype === "weapon";
  338. const is_armour = data.itemtype === "armour";
  339. const is_backpack = data.itemtype === "backpack";
  340. // Creates properties Map from full type string.
  341. const components = typeSplit(full_type);
  342. const properties = new Map();
  343. // Iterate through each component.
  344. for (let i = 1; i < components.length; i++) {
  345. const component = components[i];
  346. const is_mastercrafted = component.indexOf("stats") !== -1;
  347. const has_colour = component.indexOf("colour") !== -1;
  348. const is_rename = component.indexOf("name") !== -1;
  349. if (is_mastercrafted) {
  350. properties.set("mastercrafted", true);
  351. const numbers = component.substring("stats".length);
  352. if (is_weapon) {
  353. const stats = stringGroups(numbers, 1);
  354. properties.set("accuracy", parseInt(stats[0]));
  355. properties.set("reloading", parseInt(stats[1]));
  356. properties.set("critical_hit", parseInt(stats[2]));
  357. } else if (is_armour) {
  358. const stats = stringGroups(numbers, 2);
  359. properties.set("agility", parseInt(stats[0]));
  360. properties.set("endurance", parseInt(stats[1]));
  361. } else if (is_backpack) {
  362. const stats = stringGroups(numbers, 1);
  363. properties.set("bonus_slots", parseInt(stats[0]));
  364. }
  365. } else if (has_colour) {
  366. const colour = component.substring("colour".length);
  367. properties.set("colour", colour);
  368. } else if (is_rename) {
  369. const rename = component.substring("name".length);
  370. properties.set("rename", rename);
  371. } else {
  372. properties.set(component, true);
  373. }
  374. }
  375. // Properties from globalData.
  376. properties.set("lootable", !("noloot" in data) || data.noloot !== "1");
  377. properties.set("transferable", !("no_transfer" in data) || data.no_transfer !== "1");
  378. if (is_weapon) {
  379. properties.set("weapon_type", data.type);
  380. properties.set("proficiency_type", data.wepPro);
  381. properties.set("proficiency_level", data.pro_req);
  382. properties.set("required_strength", data.str_req);
  383. } else if (is_armour) {
  384. properties.set("engineer_level", parseInt(data.shop_level) - 5);
  385. properties.set("required_strength", parseInt(data.str_req));
  386. } else if (is_backpack) {
  387. properties.set("base_slots", parseInt(data.slots));
  388. const total_slots = properties.has("bonus_slots") ? properties.get("base_slots") + properties.get("bonus_slots") : properties.get("base_slots");
  389. properties.set("total_slots", total_slots);
  390. }
  391.  
  392. return properties;
  393. }
  394.  
  395. #itemCategory(type) {
  396. // Returns item category given type.
  397. const data = Item.#global_data[type];
  398. const is_ammo = data.itemtype == "ammo";
  399. const is_weapon = data.itemtype == "weapon";
  400. const is_armour = data.itemtype == "armour";
  401. const is_backpack = data.itemtype === "backpack";
  402. const is_item = data.itemtype == "item";
  403. if (is_ammo) {
  404. return ItemCategory.AMMO;
  405. } else if (is_weapon) {
  406. return ItemCategory.WEAPON;
  407. } else if (is_armour) {
  408. return ItemCategory.ARMOUR;
  409. } else if (is_backpack) {
  410. return ItemCategory.BACKPACK;
  411. } else if (is_item) {
  412. return ItemCategory.ITEM;
  413. } else {
  414. return ItemCategory.OTHER;
  415. }
  416. }
  417.  
  418. #itemSubcategory(type) {
  419. // Returns item subcategory given type. Type should have itemcat == "item".
  420. const data = Item.#global_data[type];
  421. const is_food = parseInt(data.foodrestore) > 0;
  422. const is_medicine = parseInt(data.healthrestore) > 0;
  423. const is_implant = "implant" in data && data.implant == "1";
  424. const is_clothing = "clothingtype" in data;
  425. const is_barricade = "barricade" in data && data.barricade == "1";
  426. if (is_food) {
  427. return ItemSubcategory.FOOD;
  428. } else if (is_medicine) {
  429. return ItemSubcategory.MEDICINE;
  430. } else if (is_implant) {
  431. return ItemSubcategory.IMPLANT;
  432. } else if (is_clothing) {
  433. return ItemSubcategory.CLOTHING;
  434. } else if (is_barricade) {
  435. return ItemSubcategory.BARRICADE;
  436. } else {
  437. return ItemSubcategory.OTHER;
  438. }
  439. }
  440.  
  441. constructor(full_type, name, quantity) {
  442. this.full_type = full_type;
  443. this.base_name = name;
  444. this.base_type = typeSplit(full_type)[0];
  445. this.category = this.#itemCategory(this.base_type);
  446. this.quantity = parseInt(quantity);
  447. this.properties = this.#makeProperties(full_type);
  448. if (this.category === ItemCategory.ITEM) {
  449. this.subcategory = this.#itemSubcategory(this.base_type);
  450. if (this.subcategory === ItemSubcategory.CLOTHING) {
  451. this.properties.set("clothing_type", Item.#global_data[this.base_type].clothingtype);
  452. }
  453. }
  454. if (this.properties.has("cooked") && this.properties.get("cooked")) {
  455. this.base_name = name.substring(0, 6) === "Cooked" ? name.substring(7) : name;
  456. this.full_name = "Cooked " + this.base_name;
  457. } else if (this.properties.has("colour")) {
  458. this.full_name = this.properties.get("colour") + " " + this.base_name;
  459. }
  460. }
  461. }
  462.  
  463. class ItemMarketEntry {
  464. constructor(item, price, trade_id, member_id, member_name, tradezone) {
  465. this.item = item;
  466. this.price = parseInt(price);
  467. this.trade_id = parseInt(trade_id)
  468. this.member_id = parseInt(member_id);
  469. this.member_name = member_name;
  470. this.tradezone = parseInt(tradezone);
  471. }
  472. }
  473.  
  474. class ItemSellingEntry {
  475. constructor(market_entry, member_to_id, member_to_name) {
  476. this.market_entry = market_entry;
  477. this.member_to_id = parseInt(member_to_id);
  478. this.member_to_name = member_to_name;
  479. }
  480. }
  481.  
  482. class Service {
  483. constructor(service_type, level) {
  484. this.service_type = service_type;
  485. this.level = parseInt(level);
  486. }
  487. }
  488.  
  489. class ServiceMarketEntry {
  490. constructor(service, price, member_id, member_name, tradezone) {
  491. this.service = service;
  492. this.price = parseInt(price);
  493. this.member_id = parseInt(member_id);
  494. this.member_name = member_name;
  495. this.tradezone = tradezone;
  496. }
  497. }
  498.  
  499. // GlobalData
  500.  
  501. // class GlobalData {
  502. // #setupPlayerValuesWebcallParameters() {
  503. // const parameters = {};
  504. // const uservars = userVars;
  505. // parameters["userID"] = uservars["userID"];
  506. // parameters["password"] = uservars["password"];
  507. // parameters["sc"] = uservars["sc"];
  508. // parameters["template_ID"] = "";
  509. // console.debug(parameters);
  510. // return parameters;
  511. // }
  512. //
  513. // #_data;
  514. // #_requests_out;
  515. // constructor() {
  516. // this.#_requests_out = 0;
  517. // }
  518. //
  519. // request() {
  520. // const instance = this;
  521. // const promise = new Promise(function(resolve, reject) {
  522. // instance.#_requests_out += 1;
  523. // const parameters = instance.#setupPlayerValuesWebcallParameters();
  524. // const data = {};
  525. // webCall("itemspawn", data);
  526. // setTimeout(function() {
  527. // console.debug(data);
  528. // }, 5000);
  529. // // webCall("itemspawn", parameters, function(data) {
  530. // // instance.#_data = responseToMap(data);
  531. // // instance.#_requests_out -= 1;
  532. // // resolve(instance);
  533. // // });
  534. // });
  535. // return promise;
  536. // }
  537. // }
  538.  
  539.  
  540. // PlayerValues
  541.  
  542. class PlayerValues {
  543.  
  544. #setupPlayerValuesWebcallParameters() {
  545. const parameters = {};
  546. const uservars = userVars;
  547. parameters["userID"] = uservars["userID"];
  548. parameters["password"] = uservars["password"];
  549. parameters["sc"] = uservars["sc"];
  550. return parameters;
  551. }
  552.  
  553. #query(str) {
  554. return this.#_data.get(str);
  555. }
  556.  
  557. #_raw_data;
  558. #_data;
  559. #_requests_out;
  560. constructor() {
  561. this.#_requests_out = 0;
  562. }
  563.  
  564. get data() {
  565. return this.#_data;
  566. }
  567.  
  568. get member_id() {
  569. return this.#query("id_member");
  570. }
  571.  
  572. get name() {
  573. return this.#query("df_name");
  574. }
  575.  
  576. get gender() {
  577. return this.#query("df_gender");
  578. }
  579.  
  580. get rank() {
  581. return this.#query("df_rank");
  582. }
  583.  
  584. get profession() {
  585. return this.#query("df_profession");
  586. }
  587.  
  588. get level() {
  589. return this.#query("df_level");
  590. }
  591.  
  592. get dead() {
  593. return this.#query("df_dead") === "1";
  594. }
  595.  
  596. get cash() {
  597. return parseInt(this.#query("df_cash"));
  598. }
  599.  
  600. get bank() {
  601. return parseInt(this.#query("df_bankcash"));
  602. }
  603.  
  604. get credits() {
  605. return parseInt(this.#query("df_credits"));
  606. }
  607.  
  608. get gold_member() {
  609. return this.#query("df_goldmember") === "1"
  610. }
  611.  
  612. get max_hp() {
  613. return parseInt(this.#query("df_hpmax"));
  614. }
  615.  
  616. get current_hp() {
  617. return parseInt(this.#query("df_hpcurrent"));
  618. }
  619.  
  620. get current_hunger() {
  621. return parseInt(this.#query("df_hungerhp"));
  622. }
  623.  
  624. get x() {
  625. return parseInt(this.#query("df_positionx"));
  626. }
  627.  
  628. get y() {
  629. return parseInt(this.#query("df_positiony"));
  630. }
  631.  
  632. get stats() {
  633. return {
  634. strength: parseInt(this.#query("df_strength")),
  635. accuracy: parseInt(this.#query("df_accuracy")),
  636. agility: parseInt(this.#query("df_agility")),
  637. endurance: parseInt(this.#query("df_endurance")),
  638. critical_hit: parseInt(this.#query("df_criticalhit")),
  639. reloading: parseInt(this.#query("df_reloading"))
  640. };
  641. }
  642.  
  643. get proficiencies() {
  644. return {
  645. melee: parseInt(this.#query("df_promelee")),
  646. pistols: parseInt(this.#query("df_propistol")),
  647. rifles: parseInt(this.#query("df_prorifle")),
  648. shotguns: parseInt(this.#query("df_proshotgun")),
  649. machine_guns: parseInt(this.#query("df_promachinegun")),
  650. explosives: parseInt(this.#query("df_proexplosive"))
  651. };
  652. }
  653.  
  654. get ammo() {
  655. return {
  656. // Shotgun ammo
  657. "10gaugeammo": parseInt(this.#query("df_10gaugeammo")),
  658. "12gaugeammo": parseInt(this.#query("df_12gaugeammo")),
  659. "16gaugeammo": parseInt(this.#query("df_16gaugeammo")),
  660. "20gaugeammo": parseInt(this.#query("df_20gaugeammo")),
  661. // Handgun ammo
  662. "32ammo": parseInt(this.#query("df_32ammo")),
  663. "357ammo": parseInt(this.#query("df_357ammo")),
  664. "35ammo": parseInt(this.#query("df_35ammo")),
  665. "38ammo": parseInt(this.#query("df_38ammo")),
  666. "40ammo": parseInt(this.#query("df_40ammo")),
  667. "45ammo": parseInt(this.#query("df_45ammo")),
  668. "50ammo": parseInt(this.#query("df_50ammo")),
  669. "55ammo": parseInt(this.#query("df_55ammo")),
  670. // Rifle ammo
  671. "55rifleammo": parseInt(this.#query("df_55rifleammo")),
  672. "75rifleammo": parseInt(this.#query("df_75rifleammo")),
  673. "9rifleammo": parseInt(this.#query("df_9rifleammo")),
  674. "127rifleammo": parseInt(this.#query("df_127rifleammo")),
  675. "14rifleammo": parseInt(this.#query("df_14rifleammo")),
  676. // Grenade ammo
  677. "grenadeammo": parseInt(this.#query("df_grenadeammo")),
  678. "heavygrenadeammo": parseInt(this.#query("df_heavygrenadeammo"))
  679. };
  680. }
  681.  
  682. get tradezone() {
  683. return parseInt(this.#query("df_tradezone"));
  684. }
  685.  
  686. get account_name() {
  687. return this.#query("account_name");
  688. }
  689.  
  690. get data() {
  691. return this.#_data;
  692. }
  693.  
  694. get raw_data() {
  695. return this.#_raw_data;
  696. }
  697.  
  698. request() {
  699. const instance = this;
  700. const promise = new Promise(function(resolve, reject) {
  701. instance.#_requests_out += 1;
  702. const parameters = instance.#setupPlayerValuesWebcallParameters();
  703. webCall("get_values", parameters, function(data) {
  704. instance.#_raw_data = data;
  705. instance.#_data = responseToMap(data);
  706. instance.#_requests_out -= 1;
  707. resolve(instance);
  708. });
  709. });
  710. return promise;
  711. }
  712. }
  713.  
  714.  
  715. // PlayerItems
  716. // The PlayerItems class is responsible for storing item data and item movement (equipment, storage).
  717. // TODO: Add check for global data availability for all items (inventory, equipment).
  718. class PlayerItems {
  719. static #global_data = globalData;
  720.  
  721. #fulltypeQuantityFromInventorySlot(slot) {
  722. // Returns item type and item quantity from slot.
  723. const uservars = this.#_uservars;
  724. const nslots = parseInt(uservars["DFSTATS_df_invslots"]);
  725. if (slot > nslots || slot < 1) {
  726. throw new RangeError("Slot: " + slot.toString() + " out of range of inventory slots.");
  727. }
  728. const type_key = "DFSTATS_df_inv" + slot.toString() + "_type";
  729. const quantity_key = "DFSTATS_df_inv" + slot.toString() + "_quantity";
  730. return [uservars[type_key], parseInt(uservars[quantity_key])];
  731. }
  732.  
  733. #fulltypeQuantityFromStorageSlot(slot) {
  734. const uservars = this.#_uservars;
  735. const storage = this.#_storage_data;
  736. const nslots = parseInt(uservars["DFSTATS_df_storage_slots"]);
  737. if (slot > nslots || slot < 1) {
  738. throw new RangeError("Slot: " + slot.toString() + " out of range of storage slots.");
  739. }
  740. const type_key = "df_store" + slot.toString() + "_type";
  741. const quantity_key = "df_store" + slot.toString() + "_quantity";
  742. return [storage.get(type_key), parseInt(storage.get(quantity_key))];
  743. }
  744.  
  745. #nBackpackSlots() {
  746. // Returns number of backpack slots available.
  747. const uservars = this.#_uservars;
  748. const globaldata = PlayerItems.#global_data;
  749. const full_type = uservars["DFSTATS_df_backpack"];
  750. if (full_type === "") {
  751. return 0;
  752. }
  753. const base_type = typeSplit(uservars["DFSTATS_df_backpack"])[0];
  754. const backpack = new Item(full_type, globaldata[base_type].name, 1);
  755. return backpack.properties.get("total_slots");
  756. }
  757.  
  758. #fulltypeQuantityFromBackpackSlot(slot) {
  759. // Returns item type and item quantity from slot.
  760. const uservars = this.#_uservars;
  761. const nslots = this.#nBackpackSlots();
  762. if (slot > nslots || slot < 1) {
  763. throw new RangeError("Slot: " + slot.toString() + " out of range of backpack slots.");
  764. }
  765. const type_key = "DFSTATS_df_backpack" + slot.toString() + "_type";
  766. const quantity_key = "DFSTATS_df_backpack" + slot.toString() + "_quantity";
  767. return type_key in uservars ? [uservars[type_key], parseInt(uservars[quantity_key])] : ["", ""];
  768. }
  769.  
  770. #setupStorageWebcallParameters() {
  771. const parameters = {};
  772. const uservars = this.#_uservars;
  773. parameters["userID"] = uservars["userID"];
  774. parameters["password"] = uservars["password"];
  775. parameters["sc"] = uservars["sc"];
  776. parameters["pagetime"] = uservars["pagetime"];
  777. return parameters;
  778. }
  779.  
  780. #typeFromImplantSlot(slot) {
  781. const uservars = this.#_uservars;
  782. const nslots = parseInt(uservars["DFSTATS_df_implantslots"]);
  783. if (slot > nslots || slot < 1) {
  784. throw new RangeError("Slot: " + slot.toString() + " out of range of implant slots.");
  785. }
  786. const type_key = "DFSTATS_df_implant" + slot.toString() + "_type";
  787. return uservars[type_key];
  788. }
  789.  
  790. #typeInGlobalData(type) {
  791. return type === "" || type in PlayerItems.#global_data;
  792. }
  793.  
  794. #_uservars;
  795. #_storage_data;
  796. constructor() {
  797. this.#_uservars = userVars;
  798. }
  799.  
  800. // Availability Checks
  801.  
  802. inventoryAvailable() {
  803. // Check for userstats keys.
  804. const inventory_available = this.#_uservars["DFSTATS_df_invslots"] !== undefined;
  805. if (!inventory_available) {
  806. return false;
  807. }
  808. // Check for global data keys.
  809. const nslots = parseInt(this.#_uservars["DFSTATS_df_invslots"]);
  810. for (let i = 1; i <= nslots; i++) {
  811. const [full_type, quantity] = this.#fulltypeQuantityFromInventorySlot(i);
  812. const base_type = typeSplit(full_type)[0];
  813. if (!this.#typeInGlobalData(base_type)) {
  814. return false;
  815. }
  816. }
  817. return true;
  818. }
  819.  
  820. equipmentAvailable() {
  821. // TODO: Check for clothing in global data.
  822. const uservars = this.#_uservars;
  823. const global_data = PlayerItems.#global_data;
  824. const implants_available = uservars["DFSTATS_df_implantslots"] !== undefined;
  825. const weapons_available = uservars["DFSTATS_df_weapon1type"] !== undefined;
  826. const armour_available = uservars["DFSTATS_df_armourtype"] !== undefined;
  827. const implants_nslots = parseInt(this.#_uservars["DFSTATS_df_implantslots"]);
  828. for (let i = 1; i <= implants_nslots; i++) {
  829. const implant_type = typeFromImplantSlot(i);
  830. if (!this.#typeInGlobalData(implant_type)) {
  831. return false;
  832. }
  833. }
  834. const weapons_in_globaldata = this.#typeInGlobalData(uservars["DFSTATS_df_weapon1type"]) && this.#typeInGlobalData(uservars["DFSTATS_df_weapon2type"]) && this.#typeInGlobalData(uservars["DFSTATS_df_weapon3type"]);
  835. if (!weapons_in_globaldata) {
  836. return false;
  837. }
  838. const armour_in_global_data = this.#typeInGlobalData(uservars["DFSTATS_df_armourtype"]);
  839. if (!armour_in_global_data) {
  840. return false;
  841. }
  842. return true;
  843. }
  844.  
  845. storageAvailable() {
  846. const uservars_available = this.#_uservars !== undefined;
  847. return uservars_available;
  848. }
  849.  
  850. // Inventory Queries
  851.  
  852. itemFromInventorySlot(slot) {
  853. const [full_type, quantity] = this.#fulltypeQuantityFromInventorySlot(slot);
  854. if (full_type === "") {
  855. return undefined;
  856. }
  857. const base_type = typeSplit(full_type)[0];
  858. const data = PlayerItems.#global_data[base_type]
  859. const name = data.name;
  860. const item = new Item(full_type, name, quantity);
  861. // Adding in durability properties for armour.
  862. if (item.category === ItemCategory.ARMOUR) {
  863. item.properties.set("durability", quantity);
  864. item.properties.set("max_durability", parseInt(data.hp));
  865. }
  866. // Adding in cooked property for food.
  867. if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.FOOD) {
  868. const is_cooked = item.properties.has("cooked") && item.properties.get("cooked") === true;
  869. if (!is_cooked) {
  870. item.properties.set("cooked", false);
  871. }
  872. }
  873. // Adding in max quantity property for ammunition.
  874. if (item.category === ItemCategory.AMMO) {
  875. item.properties.set("max_quantity", parseInt(data.max_quantity));
  876. }
  877. return item;
  878. }
  879.  
  880. inventory(slot) {
  881. return this.itemFromInventorySlot(slot);
  882. }
  883.  
  884. *inventorySlots() {
  885. // Generator that iterates through slots and yields [slot, Item].
  886. const uservars = this.#_uservars;
  887. const nslots = parseInt(uservars["DFSTATS_df_invslots"]);
  888. for (let slot = 1; slot <= nslots; slot++) {
  889. const item = this.itemFromInventorySlot(slot);
  890. yield [slot, item];
  891. }
  892. }
  893.  
  894. *inventoryItems() {
  895. // Generator that iterates through filled slots and yields [slot, Item].
  896. for (const [slot, item] of this.inventorySlots()) {
  897. const slot_filled = item !== undefined;
  898. if (slot_filled) {
  899. yield [slot, item];
  900. }
  901. }
  902. }
  903.  
  904. isLockedSlot(slot) {
  905. return lockedSlots.includes(slot.toString());
  906. }
  907.  
  908. totalInventorySlots() {
  909. const uservars = this.#_uservars;
  910. const nslots = parseInt(uservars["DFSTATS_df_invslots"]);
  911. return nslots;
  912. }
  913.  
  914. fullInventorySlots() {
  915. return Array.from(this.inventorySlots()).filter((e) => e[1] !== undefined).length;
  916. }
  917.  
  918. emptyInventorySlots() {
  919. return Array.from(this.inventorySlots()).filter((e) => e[1] === undefined).length;
  920. }
  921.  
  922. // Backpack Queries
  923.  
  924. itemFromBackpackSlot(slot) {
  925. const [full_type, quantity] = this.#fulltypeQuantityFromBackpackSlot(slot);
  926. if (full_type === "") {
  927. return undefined;
  928. }
  929. const base_type = typeSplit(full_type)[0];
  930. const data = PlayerItems.#global_data[base_type]
  931. const name = data.name;
  932. const item = new Item(full_type, name, quantity);
  933. // Adding in durability properties for armour.
  934. if (item.category === ItemCategory.ARMOUR) {
  935. item.properties.set("durability", quantity);
  936. item.properties.set("max_durability", parseInt(data.hp));
  937. }
  938. // Adding in cooked property for food.
  939. if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.FOOD) {
  940. const is_cooked = item.properties.has("cooked") && item.properties.get("cooked") === true;
  941. if (!is_cooked) {
  942. item.properties.set("cooked", false);
  943. }
  944. }
  945. // Adding in max quantity property for ammunition.
  946. if (item.category === ItemCategory.AMMO) {
  947. item.properties.set("max_quantity", parseInt(data.max_quantity));
  948. }
  949. return item;
  950. }
  951.  
  952. backpack(slot) {
  953. return this.itemFromBackpackSlot(slot);
  954. }
  955.  
  956. *backpackSlots() {
  957. // Generator that iterates through slots and yields [slot, Item].
  958. const uservars = this.#_uservars;
  959. const nslots = this.#nBackpackSlots();
  960. for (let slot = 1; slot <= nslots; slot++) {
  961. const item = this.itemFromBackpackSlot(slot);
  962. yield [slot, item];
  963. }
  964. }
  965.  
  966. *backpackItems() {
  967. // Generator that iterates through filled slots and yields [slot, Item].
  968. for (const [slot, item] of this.backpackSlots()) {
  969. const slot_filled = item !== undefined;
  970. if (slot_filled) {
  971. yield [slot, item];
  972. }
  973. }
  974. }
  975.  
  976. isLockedBackpackSlot(slot) {
  977. return lockedSlots.includes((slot + 1050).toString());
  978. }
  979.  
  980. totalBackpackSlots() {
  981. return this.#nBackpackSlots();
  982. }
  983.  
  984. fullBackpackSlots() {
  985. return Array.from(this.backpackSlots()).filter((e) => e[1] !== undefined).length;
  986. }
  987.  
  988. emptyBackpackSlots() {
  989. return Array.from(this.backpackSlots()).filter((e) => e[1] === undefined).length;
  990. }
  991.  
  992. // Implant Queries
  993.  
  994. itemFromImplantSlot(slot) {
  995. const type = this.#typeFromImplantSlot(slot);
  996. if (type === "") {
  997. return undefined;
  998. }
  999. const base_type = typeSplit(type)[0];
  1000. const name = PlayerItems.#global_data[base_type].name;
  1001. const quantity = 1;
  1002. const item = new Item(type, name, quantity);
  1003. return item;
  1004. }
  1005.  
  1006. implant(slot) {
  1007. return this.itemFromImplantSlot(slot);
  1008. }
  1009.  
  1010. *implantSlots() {
  1011. const uservars = this.#_uservars;
  1012. const nslots = parseInt(uservars["DFSTATS_df_implantslots"]);
  1013. for (let slot = 1; slot <= nslots; slot++) {
  1014. const item = this.itemFromImplantSlot(slot);
  1015. yield [slot, item];
  1016. }
  1017. }
  1018.  
  1019. *implantItems() {
  1020. for (const [slot, item] of this.implantSlots()) {
  1021. const slot_filled = item !== undefined;
  1022. if (slot_filled) {
  1023. yield [slot, item];
  1024. }
  1025. }
  1026. }
  1027.  
  1028. // Equipment Queries
  1029.  
  1030. armour() {
  1031. const uservars = this.#_uservars;
  1032. const full_type = uservars["DFSTATS_df_armourtype"];
  1033. if (full_type === "") {
  1034. return undefined;
  1035. }
  1036. const name = uservars["DFSTATS_df_armourname"];
  1037. const durability = parseInt(uservars["DFSTATS_df_armourhp"]);
  1038. const max_durability = parseInt(uservars["DFSTATS_df_armourhpmax"]);
  1039. const item = new Item(full_type, name, durability);
  1040. item.properties.set("durability", durability);
  1041. item.properties.set("max_durability", max_durability);
  1042. return item;
  1043. }
  1044.  
  1045. weapon(index) {
  1046. const uservars = this.#_uservars;
  1047. const i = index.toString();
  1048. const full_type = uservars["DFSTATS_df_weapon" + i + "type"];
  1049. if (full_type === "") {
  1050. return undefined;
  1051. }
  1052. const name = uservars["DFSTATS_df_weapon" + i + "name"];
  1053. const quantity = 1;
  1054. const item = new Item(full_type, name, quantity);
  1055. return item;
  1056. }
  1057.  
  1058. backpack() {
  1059. const uservars = this.#_uservars;
  1060. const full_type = uservars["DFSTATS_df_backpack"];
  1061. if (full_type === "") {
  1062. return undefined;
  1063. }
  1064. const base_type = typeSplit(full_type)[0];
  1065. const name = PlayerItems.#global_data[base_type].name;
  1066. const quantity = 1;
  1067. const item = new Item(full_type, name, quantity);
  1068. return item;
  1069. }
  1070.  
  1071. #cosmetic(key) {
  1072. const uservars = this.#_uservars;
  1073. const full_type = uservars[key];
  1074. if (full_type === "") {
  1075. return undefined;
  1076. }
  1077. const base_type = typeSplit(full_type)[0];
  1078. const name = PlayerItems.#global_data[base_type].name;
  1079. const quantity = 1;
  1080. const item = new Item(full_type, name, quantity);
  1081. return item;
  1082. }
  1083.  
  1084. hat() {
  1085. return this.#cosmetic("DFSTATS_df_avatar_hat");
  1086. }
  1087.  
  1088. mask() {
  1089. return this.#cosmetic("DFSTATS_df_avatar_mask");
  1090. }
  1091.  
  1092. coat() {
  1093. return this.#cosmetic("DFSTATS_df_avatar_coat");
  1094. }
  1095.  
  1096. shirt() {
  1097. return this.#cosmetic("DFSTATS_df_avatar_shirt");
  1098. }
  1099.  
  1100. trousers() {
  1101. return this.#cosmetic("DFSTATS_df_avatar_trousers");
  1102. }
  1103.  
  1104. // Storage Queries
  1105.  
  1106. requestStorage() {
  1107. // Before any storage queries, this function must be called, as well as after any changes are made to storage.
  1108. const instance = this;
  1109. const promise = new Promise(function(resolve, reject) {
  1110. const parameters = instance.#setupStorageWebcallParameters();
  1111. webCall("get_storage", parameters, function(data) {
  1112. storageBox = flshToArr(data); // Update website vars.
  1113. instance.#_storage_data = responseToMap(data);
  1114. resolve(instance);
  1115. }, true);
  1116. });
  1117. return promise;
  1118. }
  1119.  
  1120. itemFromStorageSlot(slot) {
  1121. const [full_type, quantity] = this.#fulltypeQuantityFromStorageSlot(slot);
  1122. if (full_type === undefined) {
  1123. return undefined;
  1124. }
  1125. const base_type = typeSplit(full_type)[0];
  1126. const data = PlayerItems.#global_data[base_type]
  1127. const name = data.name;
  1128. const item = new Item(full_type, name, quantity);
  1129. if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.FOOD) {
  1130. const is_cooked = item.properties.has("cooked") && item.properties.get("cooked") === true;
  1131. if (!is_cooked) {
  1132. item.properties.set("cooked", false);
  1133. }
  1134. }
  1135. if (item.category === ItemCategory.AMMO) {
  1136. item.properties.set("max_quantity", parseInt(data.max_quantity));
  1137. }
  1138. return item;
  1139. }
  1140.  
  1141. storage(slot) {
  1142. return this.itemFromStorageSlot(slot);
  1143. }
  1144.  
  1145. *storageSlots() {
  1146. const uservars = this.#_uservars;
  1147. const nslots = parseInt(uservars["DFSTATS_df_storage_slots"]);
  1148. for (let slot = 1; slot <= nslots; slot++) {
  1149. const item = this.itemFromStorageSlot(slot);
  1150. yield [slot, item];
  1151. }
  1152. }
  1153.  
  1154. *storageItems() {
  1155. for (const [slot, item] of this.storageSlots()) {
  1156. const slot_filled = item !== undefined;
  1157. if (slot_filled) {
  1158. yield [slot, item];
  1159. }
  1160. }
  1161. }
  1162.  
  1163. // Move Operations
  1164.  
  1165. #setupMoveParameters() {
  1166. const parameters = {};
  1167. const uservars = this.#_uservars;
  1168. parameters["userID"] = uservars["userID"];
  1169. parameters["password"] = uservars["password"];
  1170. parameters["sc"] = uservars["sc"];
  1171. parameters["pagetime"] = uservars["pagetime"];
  1172. parameters["templateID"] = uservars["template_ID"];
  1173. parameters["creditsnum"] = this.#_uservars["DFSTATS_df_credits"];
  1174. return parameters;
  1175. }
  1176.  
  1177. #quantityIsMax(item) {
  1178. return item.quantity === item.properties.get("max_quantity");
  1179. }
  1180.  
  1181. #moveResult(item1, item2_init) {
  1182. // Returns result of moving item1 into item2. item2 can be undefined.
  1183. const item2 = item2_init === undefined ? new Item(item1.full_type, item1.base_name, 0) : item2_init; // Dummy item if item2 is undefined.
  1184. const both_stackable = isStackable(item1) && isStackable(item2);
  1185. const same_type = item1.full_type === item2.full_type;
  1186. if (both_stackable && same_type) {
  1187. const max = item1.properties.get("max_quantity");
  1188. const sum = item1.quantity + item2.quantity;
  1189. const item_stacking = item1.quantity < max && item2.quantity < max;
  1190. if (item_stacking) {
  1191. if (sum > max) {
  1192. // item2 slot becomes fully stacked, item1 slot gets remainder.
  1193. const diff = sum - max;
  1194. const item1_new = new Item(item1.full_type, item1.base_name, diff);
  1195. const item2_new = new Item(item2.full_type, item2.base_name, max);
  1196. return [item1_new, item2_new];
  1197. } else {
  1198. // item2 slot gets full quantity, item1 slot is empty.
  1199. const item1_new = undefined;
  1200. const item2_new = new Item(item2.full_type, item2.base_name, sum);
  1201. return [item1_new, item2_new];
  1202. }
  1203. }
  1204. }
  1205. // Swap items.
  1206. return [item2, item1];
  1207. }
  1208.  
  1209. #storageArrayFromSlotItem(slot, item) {
  1210. const delta = {};
  1211. delta["df_store" + slot.toString() + "_type"] = item.full_type;
  1212. delta["df_store" + slot.toString() + "_quantity"] = item.quantity.toString();
  1213. return delta;
  1214. }
  1215.  
  1216. inventoryToInventory(primary_slot, secondary_slot, update_ui = UiUpdate.YES) {
  1217. const instance = this;
  1218. const promise = new Promise(function(resolve, reject) {
  1219. const parameters = instance.#setupMoveParameters();
  1220. parameters["action"] = "newswap";
  1221. const primary_item = instance.itemFromInventorySlot(primary_slot);
  1222. const secondary_item = instance.itemFromInventorySlot(secondary_slot);
  1223. parameters["itemnum"] = primary_slot.toString();
  1224. parameters["itemnum2"] = secondary_slot.toString();
  1225. parameters["expected_itemtype"] = primary_item !== undefined ? primary_item.full_type : "";
  1226. parameters["expected_itemtype2"] = secondary_item !== undefined ? secondary_item.full_type : "";
  1227. webCall("inventory_new", parameters, function(data)
  1228. {
  1229. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1230. if (update_ui === UiUpdate.YES) {
  1231. populateInventory();
  1232. updateAllFields();
  1233. }
  1234. resolve(instance);
  1235. }, true);
  1236. });
  1237. return promise;
  1238. }
  1239.  
  1240. inventoryToImplants(inventory_slot, implant_slot, update_ui = UiUpdate.YES) {
  1241. const instance = this;
  1242. const promise = new Promise(function(resolve, reject) {
  1243. const parameters = instance.#setupMoveParameters();
  1244. parameters["action"] = "newswap";
  1245. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1246. if (inventory_item !== undefined && (inventory_item.category !== ItemCategory.ITEM || inventory_item.subcategory !== ItemSubcategory.IMPLANT)) {
  1247. throw new TypeError("Inventory item: " + inventory_item.full_type + " is not an implant.");
  1248. }
  1249. const implant_item = instance.itemFromImplantSlot(implant_slot);
  1250. parameters["itemnum"] = inventory_slot.toString();
  1251. parameters["itemnum2"] = (implant_slot + 1000).toString();
  1252. parameters["expected_itemtype"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1253. parameters["expected_itemtype2"] = implant_item !== undefined ? implant_item.full_type : "";
  1254. webCall("inventory_new", parameters, function(data) {
  1255. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1256. if (update_ui === UiUpdate.YES) {
  1257. populateInventory();
  1258. populateImplants();
  1259. updateAllFields();
  1260. }
  1261. resolve(instance);
  1262. }, true);
  1263. });
  1264. return promise;
  1265. }
  1266.  
  1267. #inventoryToEquipment(inventory_slot, equipment_slot, get_equipment, error_if_false, equipment_slot_name = "given", update_ui = UiUpdate.YES) {
  1268. const instance = this;
  1269. const promise = new Promise(function(resolve, reject) {
  1270. const parameters = instance.#setupMoveParameters();
  1271. parameters["action"] = "newequip";
  1272. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1273. if (!error_if_false(inventory_item)) {
  1274. throw new TypeError("Inventory item: " + inventory_item.full_type + " cannot be placed in the " + equipment_slot_name + " slot.");
  1275. }
  1276. const equipment_item = get_equipment();
  1277. parameters["itemnum"] = inventory_slot.toString();
  1278. parameters["itemnum2"] = equipment_slot.toString();
  1279. parameters["expected_itemtype"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1280. parameters["expected_itemtype2"] = equipment_item !== undefined ? equipment_item.full_type : "";
  1281. webCall("inventory_new", parameters, function(data) {
  1282. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1283. if (update_ui === UiUpdate.YES) {
  1284. $.each($(".characterRender"), function(key, val)
  1285. {
  1286. renderAvatarUpdate(val, userVars);
  1287. });
  1288. populateInventory();
  1289. populateCharacterInventory();
  1290. updateAllFields();
  1291. renderAvatarUpdate();
  1292. }
  1293. resolve(instance)
  1294. }, true);
  1295. });
  1296. return promise;
  1297. }
  1298.  
  1299. inventoryToArmour(inventory_slot, update_ui = UiUpdate.YES) {
  1300. const armour_slot = 34
  1301. const error_if_false = (inventory_item) => inventory_item === undefined || inventory_item.category === ItemCategory.ARMOUR;
  1302. const armour = () => this.armour();
  1303. return this.#inventoryToEquipment(inventory_slot, armour_slot, armour, error_if_false, "armour", update_ui);
  1304. }
  1305.  
  1306. inventoryToWeapon(inventory_slot, weapon_slot, update_ui = UiUpdate.YES) {
  1307. const adjusted_weapon_slot = weapon_slot + 30;
  1308. const error_if_false = (inventory_item) => inventory_item === undefined || inventory_item.category === ItemCategory.WEAPON;
  1309. const weapon = () => this.weapon(weapon_slot);
  1310. return this.#inventoryToEquipment(inventory_slot, adjusted_weapon_slot, weapon, error_if_false, "weapon", update_ui);
  1311. }
  1312.  
  1313. inventoryToHat(inventory_slot, update_ui = UiUpdate.YES) {
  1314. const hat_slot = 40;
  1315. const error_if_false = (inventory_item) => inventory_item === undefined || (inventory_item.category === ItemCategory.ITEM && inventory_item.subcategory === ItemSubcategory.CLOTHING && inventory_item.properties.get("clothing_type") === "hat");
  1316. const hat = () => this.hat();
  1317. return this.#inventoryToEquipment(inventory_slot, hat_slot, hat, error_if_false, "hat", update_ui);
  1318. }
  1319.  
  1320. inventoryToMask(inventory_slot, update_ui = UiUpdate.YES) {
  1321. const mask_slot = 39;
  1322. const error_if_false = (inventory_item) => inventory_item === undefined || (inventory_item.category === ItemCategory.ITEM && inventory_item.subcategory === ItemSubcategory.CLOTHING && inventory_item.properties.get("clothing_type") === "mask");
  1323. const mask = () => this.mask();
  1324. return this.#inventoryToEquipment(inventory_slot, mask_slot, mask, error_if_false, "mask", update_ui);
  1325. }
  1326.  
  1327. inventoryToCoat(inventory_slot, update_ui = UiUpdate.YES) {
  1328. const coat_slot = 38;
  1329. const error_if_false = (inventory_item) => inventory_item === undefined || (inventory_item.category === ItemCategory.ITEM && inventory_item.subcategory === ItemSubcategory.CLOTHING && inventory_item.properties.get("clothing_type") === "coat");
  1330. const coat = () => this.coat();
  1331. return this.#inventoryToEquipment(inventory_slot, coat_slot, coat, error_if_false, "coat", update_ui);
  1332. }
  1333.  
  1334. inventoryToShirt(inventory_slot, update_ui = UiUpdate.YES) {
  1335. const shirt_slot = 36;
  1336. const error_if_false = (inventory_item) => inventory_item === undefined || (inventory_item.category === ItemCategory.ITEM && inventory_item.subcategory === ItemSubcategory.CLOTHING && inventory_item.properties.get("clothing_type") === "shirt");
  1337. const shirt = () => this.shirt();
  1338. return this.#inventoryToEquipment(inventory_slot, shirt_slot, shirt, error_if_false, "shirt", update_ui);
  1339. }
  1340.  
  1341. inventoryToTrousers(inventory_slot, update_ui = UiUpdate.YES) {
  1342. const trousers_slot = 37;
  1343. const error_if_false = (inventory_item) => inventory_item === undefined || (inventory_item.category === ItemCategory.ITEM && inventory_item.subcategory === ItemSubcategory.CLOTHING && inventory_item.properties.get("clothing_type") === "trousers");
  1344. const trousers = () => this.trousers();
  1345. return this.#inventoryToEquipment(inventory_slot, trousers_slot, trousers, error_if_false, "trousers", update_ui);
  1346. }
  1347.  
  1348. inventoryToStorage(inventory_slot, storage_slot, update_ui = UiUpdate.YES) {
  1349. const instance = this;
  1350. const promise = new Promise(function(resolve, reject) {
  1351. const parameters = instance.#setupMoveParameters();
  1352. parameters["action"] = "store";
  1353. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1354. const storage_item = instance.itemFromStorageSlot(storage_slot);
  1355. parameters["itemnum"] = inventory_slot.toString();
  1356. parameters["itemnum2"] = (storage_slot + 40).toString();
  1357. parameters["expected_itemtype"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1358. parameters["expected_itemtype2"] = storage_item !== undefined ? storage_item.full_type : "";
  1359. webCall("inventory_new", parameters, function(data)
  1360. {
  1361. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1362. const [inventory_new, storage_new] = instance.#moveResult(inventory_item, storage_item);
  1363. if (storage_new !== undefined) {
  1364. updateIntoArr(instance.#storageArrayFromSlotItem(storage_slot, storage_new), storageBox);
  1365. } else {
  1366. delete storageBox["df_store" + storage_slot.toString() + "_type"];
  1367. delete storageBox["df_store" + storage_slot.toString() + "_quantity"];
  1368. }
  1369. if (update_ui) {
  1370. populateStorage();
  1371. populateInventory();
  1372. updateAllFields();
  1373. }
  1374. resolve(instance);
  1375. }, true);
  1376. });
  1377. return promise;
  1378. }
  1379.  
  1380. storageToInventory(storage_slot, inventory_slot, update_ui = UiUpdate.YES) {
  1381. // Slightly different because the first item in the inventory_new webCall must be defined, and the final
  1382. // item/stack will end up in inventory. i.e. If there is no item in the inventory slot for inventoryToStorage,
  1383. // it will not do anything. (Also has a different action type.)
  1384. const instance = this;
  1385. const promise = new Promise(function(resolve, reject) {
  1386. const parameters = instance.#setupMoveParameters();
  1387. parameters["action"] = "take";
  1388. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1389. const storage_item = instance.itemFromStorageSlot(storage_slot);
  1390. parameters["itemnum"] = (storage_slot + 40).toString();
  1391. parameters["itemnum2"] = inventory_slot.toString();
  1392. parameters["expected_itemtype"] = storage_item !== undefined ? storage_item.full_type : "";
  1393. parameters["expected_itemtype2"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1394. webCall("inventory_new", parameters, function(data)
  1395. {
  1396. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1397. const [storage_new, inventory_new] = instance.#moveResult(storage_item, inventory_item);
  1398. if (storage_new !== undefined) {
  1399. updateIntoArr(instance.#storageArrayFromSlotItem(storage_slot, storage_new), storageBox);
  1400. } else {
  1401. delete storageBox["df_store" + storage_slot.toString() + "_type"];
  1402. delete storageBox["df_store" + storage_slot.toString() + "_quantity"];
  1403. }
  1404. if (update_ui) {
  1405. populateStorage();
  1406. populateInventory();
  1407. updateAllFields();
  1408. }
  1409. resolve(instance);
  1410. }, true);
  1411. });
  1412. return promise;
  1413. }
  1414.  
  1415. inventoryToBackpack(inventory_slot, backpack_slot, update_ui = UiUpdate.YES) {
  1416. const instance = this;
  1417. const promise = new Promise(function(resolve, reject) {
  1418. const parameters = instance.#setupMoveParameters();
  1419. parameters["action"] = "backpack";
  1420. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1421. const backpack_item = instance.itemFromBackpackSlot(backpack_slot);
  1422. parameters["itemnum"] = inventory_slot.toString();
  1423. parameters["itemnum2"] = (backpack_slot + 1050).toString();
  1424. parameters["expected_itemtype"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1425. parameters["expected_itemtype2"] = backpack_item !== undefined ? backpack_item.full_type : "";
  1426. webCall("hotrods/backpack", parameters, function(data)
  1427. {
  1428. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1429. if (update_ui) {
  1430. populateBackpack();
  1431. populateInventory();
  1432. updateAllFields();
  1433. }
  1434. resolve(instance);
  1435. }, true);
  1436. });
  1437. return promise;
  1438. }
  1439.  
  1440. backpackToInventory(backpack_slot, inventory_slot, update_ui = UiUpdate.YES) {
  1441. const instance = this;
  1442. const promise = new Promise(function(resolve, reject) {
  1443. const parameters = instance.#setupMoveParameters();
  1444. parameters["action"] = "backpack";
  1445. const inventory_item = instance.itemFromInventorySlot(inventory_slot);
  1446. const backpack_item = instance.itemFromBackpackSlot(backpack_slot);
  1447. parameters["itemnum"] = (backpack_slot + 1050).toString();
  1448. parameters["itemnum2"] = inventory_slot.toString();
  1449. parameters["expected_itemtype"] = backpack_item !== undefined ? backpack_item.full_type : "";
  1450. parameters["expected_itemtype2"] = inventory_item !== undefined ? inventory_item.full_type : "";
  1451. webCall("hotrods/backpack", parameters, function(data)
  1452. {
  1453. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1454. if (update_ui) {
  1455. populateBackpack();
  1456. populateInventory();
  1457. updateAllFields();
  1458. }
  1459. resolve(instance);
  1460. }, true);
  1461. });
  1462. return promise;
  1463. }
  1464.  
  1465. backpackToImplants(backpack_slot, implant_slot, update_ui = UiUpdate.YES) {
  1466. const instance = this;
  1467. const promise = new Promise(function(resolve, reject) {
  1468. const parameters = instance.#setupMoveParameters();
  1469. parameters["action"] = "backpack";
  1470. const implant_item = instance.itemFromImplantSlot(implant_slot);
  1471. const backpack_item = instance.itemFromBackpackSlot(backpack_slot);
  1472. const primary_slot = backpack_item !== undefined ? backpack_slot + 1050 : implant_slot + 1000;
  1473. const secondary_slot = backpack_item !== undefined ? implant_slot + 1000: backpack_slot + 1050;
  1474. const primary_item = backpack_item !== undefined ? backpack_item.full_type : implant_item.full_type;
  1475. const secondary_item = (backpack_item !== undefined && implant_item !== undefined) ? implant_item.full_type : "";
  1476. parameters["itemnum"] = primary_slot.toString();
  1477. parameters["itemnum2"] = secondary_slot.toString();
  1478. parameters["expected_itemtype"] = primary_item;
  1479. parameters["expected_itemtype2"] = secondary_item;
  1480. webCall("hotrods/backpack", parameters, function(data)
  1481. {
  1482. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1483. if (update_ui) {
  1484. populateBackpack();
  1485. populateImplants();
  1486. updateAllFields();
  1487. }
  1488. resolve(instance);
  1489. }, true);
  1490. });
  1491. return promise;
  1492. }
  1493.  
  1494. implantsToImplants(primary_slot, secondary_slot, update_ui = UiUpdate.YES) {
  1495. const instance = this;
  1496. const promise = new Promise(function(resolve, reject) {
  1497. // Using just the itemnum and expected_itemtype work but cause a connection error almost everytime. I'm
  1498. // not sure which of these are necessary, so I just included everything.
  1499. const primary_item = instance.itemFromImplantSlot(primary_slot);
  1500. const secondary_item = instance.itemFromImplantSlot(secondary_slot);
  1501. const parameters = {};
  1502. const uservars = instance.#_uservars;
  1503. parameters["pagetime"] = uservars["pagetime"];
  1504. parameters["templateID"] = uservars["template_ID"];
  1505. parameters["sc"] = uservars["sc"];
  1506. parameters["creditsnum"] = instance.#_uservars["DFSTATS_df_credits"];
  1507. parameters["buynum"] = "0";
  1508. parameters["renameto"] = "undefined`undefined";
  1509. parameters["expected_itemprice"] = "-1";
  1510. parameters["expected_itemtype2"] = secondary_item !== undefined ? secondary_item.full_type : "";
  1511. parameters["expected_itemtype"] = primary_item !== undefined ? primary_item.full_type : "";
  1512. parameters["itemnum2"] = (secondary_slot + 1000).toString();
  1513. parameters["itemnum"] = (primary_slot + 1000).toString();
  1514. parameters["price"] = "2400000";
  1515. parameters["gv"] = "21";
  1516. parameters["userID"] = uservars["userID"];
  1517. parameters["password"] = uservars["password"];
  1518. parameters["action"] = "newswap";
  1519. webCall("inventory_new", parameters, function(data)
  1520. {
  1521. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1522. if (update_ui) {
  1523. populateImplants();
  1524. updateAllFields();
  1525. }
  1526. resolve(instance);
  1527. }, true);
  1528. });
  1529. return promise;
  1530. }
  1531.  
  1532. // Usage, Discard, Scrap, Mastercraft
  1533.  
  1534. #setupItemScrapRequest() {
  1535. const parameters = {};
  1536. parameters["pagetime"] = this.#_uservars["pagetime"];
  1537. parameters["templateID"] = this.#_uservars["template_ID"];
  1538. parameters["sc"] = this.#_uservars["sc"];
  1539. parameters["creditsnum"] = 0;
  1540. parameters["buynum"] = 0;
  1541. parameters["renameto"] = "";
  1542. parameters["expected_itemprice"] = "-1";
  1543. parameters["expected_itemtype2"] = "";
  1544. parameters["itemnum2"] = "0";
  1545. parameters["userID"] = this.#_uservars["userID"];
  1546. parameters["password"] = this.#_uservars["password"];
  1547. return parameters;
  1548. }
  1549.  
  1550. scrapInventoryItem(slot, inventory_item, update_ui = UiUpdate.YES) {
  1551. const instance = this;
  1552. const promise = new Promise(function(resolve, reject) {
  1553. const scrap_value = scrapValue(inventory_item.full_type, inventory_item.quantity);
  1554. const parameters = instance.#setupItemRemovalRequest();
  1555. parameters["action"] = "scrap";
  1556. parameters["price"] = scrap_value;
  1557. parameters["itemnum"] = slot.toString();
  1558. parameters["expected_itemtype"] = inventory_item.full_type;
  1559. webCall("inventory_new", parameters, function(data) {
  1560. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1561. if (update_ui === UiUpdate.YES) {
  1562. populateInventory();
  1563. populateCharacterInventory();
  1564. updateAllFields();
  1565. renderAvatarUpdate();
  1566. const cash = instance.#_uservars["DFSTATS_df_cash"];
  1567. const bank = instance.#_uservars["DFSTATS_df_bankcash"];
  1568. updateCashBank(cash, bank);
  1569. }
  1570. resolve(instance);
  1571. }, true);
  1572. });
  1573. return promise;
  1574. }
  1575.  
  1576. #setupItemMastercraftRequest() {
  1577. const parameters = {};
  1578. parameters["pagetime"] = this.#_uservars["pagetime"];
  1579. parameters["templateID"] = this.#_uservars["template_ID"];
  1580. parameters["sc"] = this.#_uservars["sc"];
  1581. parameters["creditsnum"] = 0;
  1582. parameters["buynum"] = 0;
  1583. parameters["renameto"] = "";
  1584. parameters["expected_itemprice"] = "-1";
  1585. parameters["expected_itemtype2"] = "";
  1586. parameters["itemnum2"] = "0";
  1587. parameters["userID"] = this.#_uservars["userID"];
  1588. parameters["password"] = this.#_uservars["password"];
  1589. return parameters;
  1590. }
  1591.  
  1592. mastercraftInventoryItem(slot, inventory_item, update_ui = UiUpdate.YES) {
  1593. const instance = this;
  1594. const promise = new Promise(function(resolve, reject) {
  1595. const enhance_value = enhanceValue(inventory_item.full_type);
  1596. const parameters = instance.#setupItemMastercraftRequest();
  1597. parameters["action"] = "enhance";
  1598. parameters["price"] = enhance_value;
  1599. parameters["itemnum"] = slot.toString();
  1600. parameters["expected_itemtype"] = inventory_item.full_type;
  1601. webCall("inventory_new", parameters, function(data) {
  1602. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1603. if (update_ui === UiUpdate.YES) {
  1604. populateInventory();
  1605. populateCharacterInventory();
  1606. updateAllFields();
  1607. renderAvatarUpdate();
  1608. const cash = instance.#_uservars["DFSTATS_df_cash"];
  1609. const bank = instance.#_uservars["DFSTATS_df_bankcash"];
  1610. updateCashBank(cash, bank);
  1611. }
  1612. resolve(instance);
  1613. }, true);
  1614. });
  1615. return promise;
  1616. }
  1617.  
  1618. #setupItemRemovalRequest() {
  1619. const parameters = {};
  1620. parameters["pagetime"] = this.#_uservars["pagetime"];
  1621. parameters["templateID"] = this.#_uservars["template_ID"];
  1622. parameters["sc"] = this.#_uservars["sc"];
  1623. parameters["creditsnum"] = this.#_uservars["DFSTATS_df_credits"];
  1624. parameters["buynum"] = "0";
  1625. parameters["renameto"] = "undefined`undefined";
  1626. parameters["expected_itemprice"] = "-1";
  1627. parameters["price"] = getUpgradePrice();
  1628. parameters["userID"] = this.#_uservars["userID"];
  1629. parameters["password"] = this.#_uservars["password"];
  1630. return parameters;
  1631. }
  1632.  
  1633. #useActionType(item) {
  1634. const global_data = PlayerItems.#global_data;
  1635. const medicine = parseInt(global_data[item.base_type]["healthrestore"]) > 0;
  1636. const usable_gm_ticket = global_data[item.base_type]["gm_days"] && global_data[item.base_type]["gm_days"] !== "0";
  1637. const food = parseInt(global_data[item.base_type]["foodrestore"]) > 0;
  1638. const boost = parseInt(global_data[item.base_type]["boostdamagehours"]) > 0 || parseInt(global_data[item.base_type]["boostexphours"]) > 0 || parseInt(global_data[item.base_type]["boostspeedhours"]) > 0;
  1639. const story_item = global_data[item.base_type]["opencontents"] && global_data[item.base_type]["opencontents"].length > 0;
  1640. if (medicine || usable_gm_ticket) {
  1641. return "newuse";
  1642. } else if (food) {
  1643. return "newconsume";
  1644. } else if (boost) {
  1645. return "newboost";
  1646. } else if (story_item) {
  1647. return "newopen";
  1648. } else {
  1649. return false;
  1650. }
  1651. }
  1652.  
  1653. #actionNecessary(item) {
  1654. const global_data = PlayerItems.#global_data;
  1655. const medicine = parseInt(global_data[item.base_type]["healthrestore"]) > 0;
  1656. const usable_gm_ticket = global_data[item.base_type]["gm_days"] && global_data[item.base_type]["gm_days"] !== "0";
  1657. const food = parseInt(global_data[item.base_type]["foodrestore"]) > 0;
  1658. const boost = parseInt(global_data[item.base_type]["boostdamagehours"]) > 0 || parseInt(global_data[item.base_type]["boostexphours"]) > 0 || parseInt(global_data[item.base_type]["boostspeedhours"]) > 0;
  1659. const story_item = global_data[item.base_type]["opencontents"] && global_data[item.base_type]["opencontents"].length > 0;
  1660. const need_health = parseInt(this.#_uservars["DFSTATS_df_hpcurrent"]) < parseInt(this.#_uservars["DFSTATS_df_hpmax"]);
  1661. const need_hunger = parseInt(this.#_uservars["DFSTATS_df_hungerhp"]) < 100;
  1662. if (medicine) {
  1663. return need_health;
  1664. } else if (food) {
  1665. return need_hunger;
  1666. } else {
  1667. return true;
  1668. }
  1669. }
  1670.  
  1671. useInventoryItem(slot, inventory_item, update_ui = UiUpdate.YES) {
  1672. const instance = this;
  1673. const promise = new Promise(function(resolve, reject) {
  1674. const action_type = instance.#useActionType(inventory_item);
  1675. if (!action_type) {
  1676. throw new TypeError("Cannot use Item: " + inventory_item.full_type + ".");
  1677. }
  1678. const action_necessary = instance.#actionNecessary(inventory_item);
  1679. if (!action_necessary) {
  1680. throw new RangeError("Item: " + inventory_item.full_type + " will provide no benefit when used.");
  1681. }
  1682. const parameters = instance.#setupItemRemovalRequest();
  1683. parameters["action"] = action_type;
  1684. parameters["itemnum"] = slot.toString();
  1685. parameters["itemnum2"] = "0";
  1686. parameters["expected_itemtype"] = inventory_item.full_type;
  1687. parameters["expected_itemtype2"] = "";
  1688. webCall("inventory_new", parameters, function(data) {
  1689. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1690. if (update_ui === UiUpdate.YES) {
  1691. populateInventory();
  1692. populateCharacterInventory();
  1693. updateAllFields();
  1694. }
  1695. resolve(instance);
  1696. }, true);
  1697. });
  1698. return promise;
  1699. }
  1700.  
  1701. useBackpackItem(slot, item, update_ui = UiUpdate.YES) {
  1702. return this.useInventoryItem(slot + 1050, item, update_ui);
  1703. }
  1704.  
  1705. discardInventoryItem(slot, inventory_item, update_ui = UiUpdate.YES) {
  1706. const instance = this;
  1707. const promise = new Promise(function(resolve, reject) {
  1708. const parameters = instance.#setupItemRemovalRequest();
  1709. parameters["action"] = "newdiscard";
  1710. parameters["itemnum"] = slot.toString();
  1711. parameters["itemnum2"] = "0";
  1712. parameters["expected_itemtype"] = inventory_item.full_type;
  1713. parameters["expected_itemtype2"] = "";
  1714. webCall("inventory_new", parameters, function(data) {
  1715. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1716. if (update_ui === UiUpdate.YES) {
  1717. populateInventory();
  1718. populateCharacterInventory();
  1719. updateAllFields();
  1720. renderAvatarUpdate();
  1721. }
  1722. resolve(instance);
  1723. }, true);
  1724. });
  1725. return promise;
  1726. }
  1727.  
  1728. discardBackpackItem(slot, item, update_ui = UiUpdate.YES) {
  1729. return this.discardInventoryItem(slot + 1050, item, update_ui);
  1730. }
  1731.  
  1732. #setupItemMutateRequest() {
  1733. const parameters = {};
  1734. parameters["pagetime"] = this.#_uservars["pagetime"];
  1735. parameters["templateID"] = this.#_uservars["template_ID"];
  1736. parameters["sc"] = this.#_uservars["sc"];
  1737. parameters["creditsnum"] = this.#_uservars["DFSTATS_df_credits"];
  1738. parameters["buynum"] = "0";
  1739. parameters["renameto"] = "undefined`undefined";
  1740. parameters["expected_itemprice"] = "-1";
  1741. parameters["price"] = "0";
  1742. parameters["userID"] = this.#_uservars["userID"];
  1743. parameters["password"] = this.#_uservars["password"];
  1744. return parameters;
  1745. }
  1746.  
  1747. mutateItem(slot, item, new_item_type, update_ui = UiUpdate.YES) {
  1748. const instance = this;
  1749. const promise = new Promise(function(resolve, reject) {
  1750. const parameters = instance.#setupItemRemovalRequest();
  1751. parameters["action"] = "mutate";
  1752. parameters["itemnum"] = slot.toString();
  1753. parameters["itemnum2"] = "0";
  1754. parameters["expected_itemtype"] = item.full_type;
  1755. parameters["expected_itemtype2"] = new_item_type;
  1756. webCall("inventory_new", parameters, function(data) {
  1757. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  1758. if (update_ui === UiUpdate.YES) {
  1759. populateInventory();
  1760. populateCharacterInventory();
  1761. populateImplants();
  1762. updateAllFields();
  1763. renderAvatarUpdate();
  1764. }
  1765. resolve(instance);
  1766. }, true);
  1767. });
  1768. return promise;
  1769. }
  1770.  
  1771. updateLockedSlots() {
  1772. // TODO: Add support for implants and equipment later.
  1773. const inventory = document.getElementById("inventory");
  1774. const backpack = document.getElementById("backpackdisplay");
  1775. if (inventory !== null) {
  1776. const inventory_slots = inventory.querySelectorAll("td.validSlot");
  1777. for (const slot of inventory_slots) {
  1778. const index = slot.dataset.slot;
  1779. if (lockedSlots.includes(index.toString()) && !slot.classList.contains("locked")) {
  1780. slot.classList.add("locked");
  1781. } else if (slot.classList.contains("locked")) {
  1782. slot.classList.remove("locked");
  1783. }
  1784. }
  1785. }
  1786. if (backpack !== null) {
  1787. const backpack_slots = backpack.querySelectorAll("td.validSlot");
  1788. for (const slot of backpack_slots) {
  1789. const index = slot.dataset.slot + 1050;
  1790. if (lockedSlots.includes(index.toString()) && !slot.classList.contains("locked")) {
  1791. slot.classList.add("locked");
  1792. } else if (slot.classList.contains("locked")) {
  1793. slot.classList.remove("locked");
  1794. }
  1795. }
  1796. }
  1797. }
  1798.  
  1799. lockInventorySlot(slot, update_ui = UiUpdate.YES) {
  1800. const instance = this;
  1801. const promise = new Promise(function(resolve, reject) {
  1802. const parameters = instance.#setupItemRemovalRequest();
  1803. const uservars = instance.#_uservars;
  1804. parameters["itemnum"] = slot.toString();
  1805. parameters["userID"] = uservars["userID"];
  1806. parameters["password"] = uservars["password"];
  1807. parameters["sc"] = uservars["sc"];
  1808. parameters["gv"] = "21";
  1809. parameters["action"] = "addSlot";
  1810. webCall("hotrods/item_lock", parameters, function(data) {
  1811. lockedSlots = data.split(',');
  1812. if (update_ui === UiUpdate.YES) {
  1813. populateInventory();
  1814. doLockedElems();
  1815. instance.updateLockedSlots();
  1816. updateAllFields();
  1817. }
  1818. resolve(instance);
  1819. }, true);
  1820. });
  1821. return promise;
  1822. }
  1823.  
  1824. unlockInventorySlot(slot, update_ui = UiUpdate.YES) {
  1825. const instance = this;
  1826. const promise = new Promise(function(resolve, reject) {
  1827. const parameters = instance.#setupItemRemovalRequest();
  1828. const uservars = instance.#_uservars;
  1829. parameters["itemnum"] = slot.toString();
  1830. parameters["userID"] = uservars["userID"];
  1831. parameters["password"] = uservars["password"];
  1832. parameters["sc"] = uservars["sc"];
  1833. parameters["gv"] = "21";
  1834. parameters["action"] = "removeSlot";
  1835. webCall("hotrods/item_lock", parameters, function(data) {
  1836. lockedSlots = data.split(',');
  1837. if (update_ui === UiUpdate.YES) {
  1838. populateInventory();
  1839. doLockedElems();
  1840. instance.updateLockedSlots();
  1841. updateAllFields();
  1842. }
  1843. resolve(instance);
  1844. }, true);
  1845. });
  1846. return promise;
  1847. }
  1848.  
  1849. lockBackpackSlot(slot, update_ui = UiUpdate.YES) {
  1850. const instance = this;
  1851. const promise = new Promise(function(resolve, reject) {
  1852. const parameters = instance.#setupItemRemovalRequest();
  1853. const uservars = instance.#_uservars;
  1854. parameters["itemnum"] = (slot + 1050).toString();
  1855. parameters["userID"] = uservars["userID"];
  1856. parameters["password"] = uservars["password"];
  1857. parameters["sc"] = uservars["sc"];
  1858. parameters["gv"] = "21";
  1859. parameters["action"] = "addSlot";
  1860. webCall("hotrods/item_lock", parameters, function(data) {
  1861. lockedSlots = data.split(',');
  1862. if (update_ui === UiUpdate.YES) {
  1863. populateBackpack();
  1864. doLockedElems();
  1865. instance.updateLockedSlots();
  1866. updateAllFields();
  1867. }
  1868. resolve(instance);
  1869. }, true);
  1870. });
  1871. return promise;
  1872. }
  1873.  
  1874. unlockBackpackSlot(slot, update_ui = UiUpdate.YES) {
  1875. const instance = this;
  1876. const promise = new Promise(function(resolve, reject) {
  1877. const parameters = instance.#setupItemRemovalRequest();
  1878. const uservars = instance.#_uservars;
  1879. parameters["itemnum"] = (slot + 1050).toString();
  1880. parameters["userID"] = uservars["userID"];
  1881. parameters["password"] = uservars["password"];
  1882. parameters["sc"] = uservars["sc"];
  1883. parameters["gv"] = "21";
  1884. parameters["action"] = "removeSlot";
  1885. webCall("hotrods/item_lock", parameters, function(data) {
  1886. lockedSlots = data.split(',');
  1887. if (update_ui === UiUpdate.YES) {
  1888. populateBackpack();
  1889. doLockedElems();
  1890. instance.updateLockedSlots();
  1891. updateAllFields();
  1892. }
  1893. resolve(instance);
  1894. }, true);
  1895. });
  1896. return promise;
  1897. }
  1898. }
  1899.  
  1900. // MarketItems
  1901.  
  1902. class MarketItems {
  1903.  
  1904. static #global_data = globalData;
  1905.  
  1906. #typeInGlobalData(base_type) {
  1907. return base_type in MarketItems.#global_data;
  1908. }
  1909.  
  1910. #_uservars;
  1911. #_data;
  1912. #_nselling;
  1913. #_publicselling;
  1914. constructor() {
  1915. this.#_uservars = userVars;
  1916. }
  1917.  
  1918. // Availability Checks
  1919.  
  1920. sellingEntriesAvailable() {
  1921. // Run after requesting data.
  1922. const data = this.#_data;
  1923. const nselling = this.#_nselling;
  1924. const publicselling = this.#_publicselling;
  1925. for (let i = 0; i < publicselling; i++) {
  1926. const full_type = data.get("tradelist_" + i.toString() + "_item");
  1927. const base_type = typeSplit(full_type)[0];
  1928. if (!this.#typeInGlobalData(base_type)) {
  1929. return false;
  1930. }
  1931. }
  1932. return true;
  1933. }
  1934.  
  1935. // Selling Queries
  1936.  
  1937. #setupSellingWebcallParameters() {
  1938. const parameters = {};
  1939. parameters["pagetime"] = this.#_uservars["pagetime"];
  1940. parameters["tradezone"] = "";
  1941. parameters["searchname"] = "";
  1942. parameters["searchtype"] = "sellinglist";
  1943. parameters["search"] = "trades";
  1944. parameters["memID"] = this.#_uservars["userID"];
  1945. parameters["category"] = "";
  1946. parameters["profession"] = "";
  1947. return parameters;
  1948. }
  1949.  
  1950. requestSelling() {
  1951. const instance = this;
  1952. const promise = new Promise(function(resolve, reject) {
  1953. const parameters = instance.#setupSellingWebcallParameters();
  1954. webCall("trade_search", parameters, function(data) {
  1955. instance.#_data = responseToMap(data);
  1956. // maxresults is the number of results for this type of trade, while totalsales is across all trades (public and private)
  1957. // totalsales is shared between public and private sales (30 is the total sales between public and private)
  1958. instance.#_nselling = parseInt(instance.#_data.get("tradelist_totalsales")); // maxresults vs. totalsales?
  1959. instance.#_publicselling = parseInt(instance.#_data.get("tradelist_maxresults"));
  1960. resolve(instance);
  1961. });
  1962. });
  1963. return promise;
  1964. }
  1965.  
  1966. maxSellingEntries() {
  1967. const uservars = this.#_uservars;
  1968. const nslots = parseInt(uservars["DFSTATS_df_invslots"]);
  1969. return nslots; // Maximum selling entries equal to number of inventory slots.
  1970. }
  1971.  
  1972. nSellingEntries() {
  1973. return this.#_nselling;
  1974. }
  1975.  
  1976. availableSellingEntries() {
  1977. return this.maxSellingEntries() - this.nSellingEntries();
  1978. }
  1979.  
  1980. itemFromSellingIndex(index) {
  1981. const data = this.#_data;
  1982. const full_type = data.get("tradelist_" + index.toString() + "_item");
  1983. const name = data.get("tradelist_" + index.toString() + "_itemname");
  1984. const quantity = data.get("tradelist_" + index.toString() + "_quantity");
  1985. const item = new Item(full_type, name, quantity);
  1986. return item;
  1987. }
  1988.  
  1989. marketEntryFromSellingIndex(index) {
  1990. const item = this.itemFromSellingIndex(index);
  1991. const data = this.#_data;
  1992. const price = data.get("tradelist_" + index.toString() + "_price");
  1993. const trade_id = data.get("tradelist_" + index.toString() + "_trade_id");
  1994. const member_id = data.get("tradelist_" + index.toString() + "_id_member");
  1995. const member_name = data.get("tradelist_" + index.toString() + "_member_name");
  1996. const tradezone = data.get("tradelist_" + index.toString() + "_trade_zone");
  1997. const market_entry = new ItemMarketEntry(item, price, trade_id, member_id, member_name, tradezone);
  1998. return market_entry;
  1999. }
  2000.  
  2001. sellingEntryFromSellingIndex(index) {
  2002. const market_entry = this.marketEntryFromSellingIndex(index);
  2003. const data = this.#_data;
  2004. const member_to_id = data.get("tradelist_" + index.toString() + "_id_member_to");
  2005. const member_to_name = data.get("tradelist_" + index.toString() + "_member_to_name");
  2006. const selling_entry = new ItemSellingEntry(market_entry, member_to_id, member_to_name);
  2007. return selling_entry;
  2008. }
  2009.  
  2010. *sellingEntries() {
  2011. for (let i = 0; i < this.#_publicselling; i++) {
  2012. yield [i, this.sellingEntryFromSellingIndex(i)];
  2013. }
  2014. }
  2015.  
  2016. // Selling Changes
  2017.  
  2018. #setupCancelSellingParameters() {
  2019. const parameters = {};
  2020. parameters["pagetime"] = this.#_uservars["pagetime"];
  2021. parameters["templateID"] = this.#_uservars["template_ID"];
  2022. parameters["sc"] = this.#_uservars["sc"];
  2023. parameters["creditsnum"] = "0";
  2024. parameters["renameto"] = "";
  2025. parameters["expected_itemprice"] = "-1";
  2026. parameters["expected_itemtype2"] = "";
  2027. parameters["expected_itemtype"] = "";
  2028. parameters["itemnum2"] = 0;
  2029. parameters["itemnum"] = 0;
  2030. parameters["price"] = 0;
  2031. parameters["action"] = "newcancelsale";
  2032. parameters["userID"] = this.#_uservars["userID"];
  2033. parameters["password"] = this.#_uservars["password"];
  2034. return parameters
  2035. }
  2036.  
  2037. cancelSellingEntry(selling_entry, update_ui = UiUpdate.YES) {
  2038. const instance = this;
  2039. const promise = new Promise(function(resolve, reject) {
  2040. const inventory_space_available = findFirstEmptyGenericSlot("inv") !== false;
  2041. const is_credits = selling_entry.market_entry.item.full_type === "credits";
  2042. if (!is_credits && !inventory_space_available) {
  2043. throw new RangeError("Cannot cancel selling entry if no inventory space is available.");
  2044. }
  2045. const parameters = instance.#setupCancelSellingParameters();
  2046. parameters["buynum"] = selling_entry.market_entry.trade_id;
  2047. webCall("inventory_new", parameters, function(data) {
  2048. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  2049. if (update_ui === UiUpdate.YES) {
  2050. if (getSellingList !== undefined) {
  2051. getSellingList();
  2052. } else { // These functions are also called by getSellingList, but it is undefined out of the marketplace.
  2053. populateInventory();
  2054. updateAllFields();
  2055. }
  2056. }
  2057. resolve(instance);
  2058. }, true);
  2059. });
  2060. return promise;
  2061. }
  2062.  
  2063. #setupSellInventoryParameters() {
  2064. var parameters = {};
  2065. parameters["pagetime"] = this.#_uservars["pagetime"];
  2066. parameters["templateID"] = this.#_uservars["template_ID"];
  2067. parameters["sc"] = this.#_uservars["sc"];
  2068. parameters["buynum"] = 0;
  2069. parameters["renameto"] = "";
  2070. parameters["expected_itemprice"] = "-1"; // same on all sales
  2071. parameters["expected_itemtype2"] = "";
  2072. parameters["expected_itemtype"] = "";
  2073. parameters["itemnum2"] = "0";
  2074. parameters["userID"] = this.#_uservars["userID"];
  2075. parameters["password"] = this.#_uservars["password"];
  2076. return parameters;
  2077. }
  2078.  
  2079. sellInventoryItem(inventory_slot, inventory_item, price, update_ui = UiUpdate.YES) {
  2080. const instance = this;
  2081. // return instance.requestSelling().then(
  2082. const promise = new Promise(function(resolve, reject) {
  2083. // const selling_list_has_space = instance.#_nselling < parseInt(instance.#_uservars["DFSTATS_df_invslots"]);
  2084. // if (!selling_list_has_space) {
  2085. // throw new RangeError("Cannot sell item when selling list is full.");
  2086. // }
  2087. const parameters = instance.#setupSellInventoryParameters();
  2088. parameters["price"] = price;
  2089. parameters["action"] = "newsell";
  2090. parameters["expected_itemtype"] = inventory_item.full_type;
  2091. parameters["itemnum"] = inventory_slot.toString();
  2092. webCall("inventory_new", parameters, function(data) {
  2093. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  2094. if (update_ui === UiUpdate.YES) {
  2095. if (getSellingList !== undefined) {
  2096. getSellingList();
  2097. } else {
  2098. populateInventory();
  2099. updateAllFields();
  2100. }
  2101. }
  2102. resolve(instance);
  2103. }, true);
  2104. });
  2105. return promise;
  2106. }
  2107.  
  2108. // Buying Items
  2109.  
  2110. #setupBuyItemParameters() {
  2111. const parameters = {};
  2112. parameters["pagetime"] = this.#_uservars["pagetime"];
  2113. parameters["templateID"] = this.#_uservars["template_ID"];
  2114. parameters["sc"] = this.#_uservars["sc"];
  2115. parameters["creditsnum"] = "undefined";
  2116. parameters["renameto"] = "undefined`undefined";
  2117. parameters["expected_itemtype2"] = "";
  2118. parameters["expected_itemtype"] = "";
  2119. parameters["itemnum2"] = 0;
  2120. parameters["itemnum"] = 0;
  2121. parameters["price"] = 0;
  2122. parameters["action"] = "newbuy";
  2123. parameters["userID"] = userVars["userID"];
  2124. parameters["password"] = userVars["password"];
  2125. return parameters;
  2126. }
  2127.  
  2128. buyItemFromMarketEntry(market_entry, update_ui = UiUpdate.YES) {
  2129. const instance = this;
  2130. const promise = new Promise(function(resolve, reject) {
  2131. // Throw if inventory is full or not enough cash on-hand.
  2132. const inventory_space_available = findFirstEmptyGenericSlot("inv") !== false;
  2133. const enough_cash = parseInt(instance.#_uservars["DFSTATS_df_cash"]) >= market_entry.price;
  2134. if (!inventory_space_available) {
  2135. throw new RangeError("Cannot buy item when inventory is full.");
  2136. }
  2137. if (!enough_cash) {
  2138. throw new RangeError("Cannot buy item: " + market_entry.item.full_type + " with price $" + market_entry.price.toLocaleString() + " with $" + instance.#_uservars["DFSTATS_df_cash"].toLocaleString() + " on-hand.");
  2139. }
  2140. // Setup parameters and request purchase.
  2141. const parameters = instance.#setupBuyItemParameters();
  2142. parameters["buynum"] = market_entry.trade_id;
  2143. parameters["expected_itemprice"] = market_entry.price;
  2144. webCall("inventory_new", parameters, function(data) {
  2145. const item_purchase_successful = data !== "";
  2146. if (item_purchase_successful) {
  2147. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  2148. if (update_ui === UiUpdate.YES) {
  2149. populateInventory();
  2150. const cash = instance.#_uservars["DFSTATS_df_cash"];
  2151. const bank = instance.#_uservars["DFSTATS_df_bankcash"];
  2152. updateCashBank(cash, bank);
  2153. updateAllFields();
  2154. }
  2155. resolve(instance);
  2156. } else {
  2157. reject(instance);
  2158. }
  2159. }, true);
  2160. });
  2161. return promise;
  2162. }
  2163.  
  2164. // Buying Services
  2165.  
  2166. #setupBuyServiceParameters() {
  2167. const parameters = {};
  2168. parameters["pagetime"] = this.#_uservars["pagetime"];
  2169. parameters["templateID"] = this.#_uservars["template_ID"];
  2170. parameters["sc"] = this.#_uservars["sc"];
  2171. parameters["creditsnum"] = 0;
  2172. parameters["renameto"] = "undefined`undefined";
  2173. parameters["expected_itemtype2"] = "";
  2174. parameters["expected_itemtype"] = "";
  2175. parameters["itemnum2"] = "0";
  2176. parameters["userID"] = this.#_uservars["userID"];
  2177. parameters["password"] = this.#_uservars["password"];
  2178. return parameters;
  2179. }
  2180.  
  2181. #buyServiceFromMarketEntry(item_valid, service_needed, action_type, market_entry, slot, inventory_item, update_ui = UiUpdate.YES) {
  2182. const instance = this;
  2183. const promise = new Promise(function(resolve, reject) {
  2184. const is_valid_item = item_valid(inventory_item);
  2185. const enough_cash = parseInt(instance.#_uservars["DFSTATS_df_cash"]) >= market_entry.price;
  2186. if (!service_needed()) {
  2187. throw new RangeError("Service not needed.");
  2188. }
  2189. if (!is_valid_item) {
  2190. throw new TypeError("Item: " + inventory_item.full_type + " is invalid for this service.");
  2191. }
  2192. if (!enough_cash) {
  2193. throw new RangeError("Cannot buy service for $" + market_entry.price.toLocaleString() + " with $" + parseInt(instance.#_uservars["DFSTATS_df_cash"]).toLocaleString() + " on-hand.");
  2194. }
  2195. const parameters = instance.#setupBuyServiceParameters();
  2196. parameters["itemnum"] = slot.toString();
  2197. parameters["price"] = scrapAmount(inventory_item.full_type, inventory_item.quantity); // I don't know why this is here.
  2198. parameters["action"] = action_type;
  2199. parameters["buynum"] = market_entry.member_id.toString();
  2200. parameters["expected_itemprice"] = market_entry.price.toString();
  2201. webCall("inventory_new", parameters, function(data) {
  2202. updateIntoArr(flshToArr(data, "DFSTATS_"), userVars);
  2203. if (update_ui === UiUpdate.YES) {
  2204. loadStatusData();
  2205. populateInventory();
  2206. populateCharacterInventory();
  2207. updateAllFields();
  2208. const cash = instance.#_uservars["DFSTATS_df_cash"];
  2209. const bank = instance.#_uservars["DFSTATS_df_bankcash"];
  2210. updateCashBank(cash, bank);
  2211. }
  2212. resolve(instance);
  2213. }, true);
  2214. });
  2215. return promise;
  2216. }
  2217.  
  2218. buyRepairFromMarketEntry(market_entry, slot, inventory_item, update_ui = UiUpdate.YES) {
  2219. const item_valid = (item) => item.category === ItemCategory.ARMOUR && item.properties.get("durability") !== item.properties.get("max_durability");
  2220. const service_needed = () => true;
  2221. const action_type = "buyrepair";
  2222. return this.#buyServiceFromMarketEntry(item_valid, service_needed, action_type, market_entry, slot, inventory_item, update_ui);
  2223. }
  2224.  
  2225. buyAdministerFromMarketEntry(market_entry, slot, inventory_item, update_ui = UiUpdate.YES) {
  2226. const item_valid = (item) => item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.MEDICINE && MarketItems.#global_data[item.full_type].needdoctor === "1";
  2227. const service_needed = () => this.#_uservars["DFSTATS_df_hpcurrent"] !== this.#_uservars["DFSTATS_df_hpmax"];
  2228. const action_type = "buyadminister";
  2229. return this.#buyServiceFromMarketEntry(item_valid, service_needed, action_type, market_entry, slot, inventory_item, update_ui);
  2230. }
  2231.  
  2232. buyCookFromMarketEntry(market_entry, slot, inventory_item, update_ui = UiUpdate.YES) {
  2233. const item_valid = (item) => item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.FOOD && MarketItems.#global_data[item.full_type].needcook === "1" && (!inventory_item.properties.has("cooked") || inventory_item.properties.get("cooked") === false);
  2234. const service_needed = () => true;
  2235. const action_type = "buycook";
  2236. return this.#buyServiceFromMarketEntry(item_valid, service_needed, action_type, market_entry, slot, inventory_item, update_ui);
  2237. }
  2238. }
  2239.  
  2240. // Bank
  2241.  
  2242. class Bank {
  2243. #_player_values;
  2244. #_uservars;
  2245. constructor() {
  2246. this.#_uservars = userVars;
  2247. this.#_player_values = new PlayerValues();
  2248. }
  2249.  
  2250. #setupBankWebcallParameters() {
  2251. const parameters = {};
  2252. parameters["sc"] = this.#_uservars["sc"];
  2253. parameters["userID"] = this.#_uservars["userID"];
  2254. parameters["password"] = this.#_uservars["password"];
  2255. return parameters;
  2256. }
  2257.  
  2258. deposit(amount, update_ui = UiUpdate.YES) {
  2259. const instance = this;
  2260. const promise = new Promise(function(resolve, reject) {
  2261. instance.#_player_values.request()
  2262. .then(function(values) {
  2263. const parameters = instance.#setupBankWebcallParameters();
  2264. parameters["deposit"] = amount;
  2265. if (amount > values.cash) {
  2266. throw new RangeError("Cannot deposit: $" + amount.toLocaleString() + " is more than is carried on-hand.");
  2267. }
  2268. webCall("bank", parameters, function(data) {
  2269. const map = responseToMap(data);
  2270. if (update_ui === UiUpdate.YES) {
  2271. const cash = parseInt(map.get("df_cash"));
  2272. const bank = parseInt(map.get("df_bankcash"));
  2273. updateCashBank(cash, bank);
  2274. }
  2275. resolve(instance);
  2276. });
  2277. });
  2278. });
  2279. return promise;
  2280. }
  2281.  
  2282. depositAll(update_ui = UiUpdate.YES) {
  2283. const instance = this;
  2284. const promise = new Promise(function(resolve, reject) {
  2285. instance.#_player_values.request()
  2286. .then(function(values) {
  2287. return instance.deposit(values.cash, update_ui);
  2288. });
  2289. });
  2290. return promise;
  2291. }
  2292.  
  2293. withdraw(amount, update_ui = UiUpdate.YES) {
  2294. const instance = this;
  2295. const promise = new Promise(function(resolve, reject) {
  2296. instance.#_player_values.request()
  2297. .then(function(values) {
  2298. const parameters = instance.#setupBankWebcallParameters();
  2299. parameters["withdraw"] = amount;
  2300. if (amount > values.bank) {
  2301. throw new RangeError("Cannot withdraw: $" + amount.toLocaleString() + " is more than is in bank.");
  2302. }
  2303. webCall("bank", parameters, function(data) {
  2304. const map = responseToMap(data);
  2305. if (update_ui === UiUpdate.YES) {
  2306. const cash = parseInt(map.get("df_cash"));
  2307. const bank = parseInt(map.get("df_bankcash"));
  2308. updateCashBank(cash, bank);
  2309. }
  2310. resolve(instance);
  2311. });
  2312. });
  2313. });
  2314. return promise;
  2315. }
  2316. }
  2317.  
  2318. // MarketCache
  2319. // The MarketCache class is responsible for requesting, processing, and storing market data for items and services.
  2320.  
  2321. class MarketCache {
  2322. static #global_data = globalData;
  2323.  
  2324. // Data Processing
  2325.  
  2326. #specialSearchString(type) {
  2327. // NOTE: All custom search strings go here.
  2328. const name = MarketCache.#global_data[type].name;
  2329. if (type == "exterminatorreactivext") {
  2330. // "12345678901234567890" 20 char limit
  2331. return "reactive xt";
  2332. } else if (type == "exterminatorreactive") {
  2333. return "exterminator react";
  2334. } else if (type == "xterminatorreactive") {
  2335. return "x-terminator react";
  2336. } else if (type == "barnellrf31crossbow") {
  2337. return "barnell rf31";
  2338. } else if (type == "goldenrabbitimplant") {
  2339. return "golden rabbit imp";
  2340. } else if (type == "xmannbergblueprints") {
  2341. return "x-mannberg blue";
  2342. } else if (type == "dawnsabreblueprints") {
  2343. return "dawn blade blue";
  2344. } else if (type == "dawnenforcerblueprints") {
  2345. return "dawn enforcer blue";
  2346. } else if (type == "dawncarbineblueprints") {
  2347. return "dawn carbine blue";
  2348. } else if (type == "dawnstrikerblueprints") {
  2349. return "dawn striker blue";
  2350. } else if (type == "dawnlauncherblueprints") {
  2351. return "dawn launcher blue";
  2352. } else if (type == "xreactiveblueprints") {
  2353. return "x-reactive blue";
  2354. } else if (type == "inquisitorblueprints") {
  2355. return "inquisitor blue";
  2356. } else if (type == "sharktoothripperblueprints") {
  2357. return "ripper blue";
  2358. } else if (type == "qr22obsidianblueprints") {
  2359. return "obsidian blue";
  2360. } else if (type == "rusthound37eblueprints") {
  2361. return "37-e blue";
  2362. } else if (type == "heatpit75blueprints") {
  2363. return "pit 75 blue";
  2364. } else if (type == "a10bullsharkblueprints") {
  2365. return "bullshark blue";
  2366. } else if (type == "scorchernk19blueprints") {
  2367. return "nk19 blue";
  2368. } else if (type == "christmasstocking2022"){
  2369. return "stocking 2022";
  2370. } else if (type.indexOf("christmasstocking") !== -1 && !Number.isNaN(type.slice(-4))) {
  2371. return "stocking " + type.slice(-4);
  2372. } else if (type.indexOf("blueprints") !== -1) {
  2373. return name.slice(-20);
  2374. } else if (name.length > 20) {
  2375. return name.slice(0, 20);
  2376. } else {
  2377. return false;
  2378. }
  2379. }
  2380.  
  2381. // Item Requests
  2382.  
  2383. #setupItemRequest(tradezone, search_string) {
  2384. // Sets up POST request for searching up market data for given search string and tradezone.
  2385. const request = new XMLHttpRequest();
  2386. request.open("POST", "https://fairview.deadfrontier.com/onlinezombiemmo/trade_search.php");
  2387. request.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
  2388. // Setting up request payload.
  2389. const request_parameters = new URLSearchParams();
  2390. request_parameters.set("hash", "");
  2391. request_parameters.set("pagetime", "");
  2392. request_parameters.set("tradezone", tradezone.toString());
  2393. request_parameters.set("searchname", encodeURI(search_string.slice(0, 20)));
  2394. request_parameters.set("category", "");
  2395. request_parameters.set("profession", "");
  2396. request_parameters.set("memID", "");
  2397. request_parameters.set("searchtype", "buyinglistitemname");
  2398. request_parameters.set("search", "trades");
  2399. return [request, request_parameters];
  2400. }
  2401.  
  2402. #setupCategoryRequest(tradezone, category) {
  2403. // Sets up POST request for searching up market data for given category and tradezone.
  2404. const request = new XMLHttpRequest();
  2405. request.open("POST", "https://fairview.deadfrontier.com/onlinezombiemmo/trade_search.php");
  2406. request.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
  2407. // Setting up request payload.
  2408. const request_parameters = new URLSearchParams();
  2409. request_parameters.set("hash", "");
  2410. request_parameters.set("pagetime", "");
  2411. request_parameters.set("tradezone", tradezone.toString());
  2412. request_parameters.set("searchname", "");
  2413. request_parameters.set("category", category);
  2414. request_parameters.set("profession", "");
  2415. request_parameters.set("memID", "");
  2416. request_parameters.set("searchtype", "buyinglistcategory");
  2417. request_parameters.set("search", "trades");
  2418. return [request, request_parameters];
  2419. }
  2420.  
  2421. #deleteItemData(item_type) {
  2422. this.item_types.delete(item_type);
  2423. this.item_data.delete(item_type);
  2424. }
  2425.  
  2426. #deleteRenameData(rename) {
  2427. this.renames.delete(rename);
  2428. this.rename_data.delete(rename);
  2429. }
  2430.  
  2431. #parseItemData(response) {
  2432. // Parses item data and puts array(s) of MarketEntry into the instance.
  2433. const map = responseToMap(response);
  2434. const results = parseInt(map.get("tradelist_maxresults"));
  2435. const types = new Set(); // Contains types that this worker is operating on. These will all be new types.
  2436. // Delete all types that exist in this request.
  2437. const types_to_delete = new Set();
  2438. for (let i = 0; i < results; i++) {
  2439. const full_type = map.get("tradelist_" + i.toString() + "_item");
  2440. const base_type = typeSplit(full_type)[0];
  2441. types_to_delete.add(base_type);
  2442. }
  2443. for (const type of types_to_delete.values()) {
  2444. if (this.hasItemType(type)) {
  2445. this.#deleteItemData(type);
  2446. }
  2447. }
  2448. // Add in new data.
  2449. for (let i = 0; i < results; i++) {
  2450. // Extracts MarketEntry properties from keys.
  2451. const full_type = map.get("tradelist_" + i.toString() + "_item");
  2452. const base_type = typeSplit(full_type)[0];
  2453. const name = map.get("tradelist_" + i.toString() + "_itemname");
  2454. const quantity = parseInt(map.get("tradelist_" + i.toString() + "_quantity"))
  2455. const price = parseInt(map.get("tradelist_" + i.toString() + "_price"));
  2456. const trade_id = parseInt(map.get("tradelist_" + i.toString() + "_trade_id"));
  2457. const member_id = parseInt(map.get("tradelist_" + i.toString() + "_id_member"));
  2458. const member_name = map.get("tradelist_" + i.toString() + "_member_name");
  2459. const item = new Item(full_type, name, quantity);
  2460. const market_entry = new ItemMarketEntry(item, price, trade_id, member_id, member_name, this.tradezone);
  2461. const is_renamed = item.properties.has("rename");
  2462. if (is_renamed) {
  2463. const rename = item.properties.get("rename");
  2464. }
  2465. // Putting data into item data.
  2466. if (!this.hasItemType(base_type)) { // New type that has not yet been added.
  2467. types.add(base_type);
  2468. this.item_types.add(base_type);
  2469. this.item_data.set(base_type, []);
  2470. this.item_data.get(base_type).push(market_entry);
  2471. } else if (types.has(base_type)) { // Type that this worker has started and has exclusive access to.
  2472. this.item_data.get(base_type).push(market_entry);
  2473. } else { // Some other worker is handling the type.
  2474. continue;
  2475. }
  2476. }
  2477. // Sorts MarketEntry[] of newly added types by price per unit, ascending.
  2478. for (const type of types.values()) {
  2479. const market_entries = this.item_data.get(type);
  2480. const item = market_entries[0].item;
  2481. if (isStackable(item)) {
  2482. market_entries.sort(function(a, b) {return (a.price/a.quantity) - (b.price/b.quantity);});
  2483. } else { // Armour has its durability in its quantity, and I don't want to sort by that.
  2484. market_entries.sort(function(a, b) {return a.price - b.price;})
  2485. }
  2486. }
  2487. }
  2488.  
  2489. #parseRenameData(response) {
  2490. // Parses item data and puts array(s) of MarketEntry into the instance.
  2491. const map = responseToMap(response);
  2492. const results = parseInt(map.get("tradelist_maxresults"));
  2493. const renames = new Set(); // Contains renames that this worker is operating on. These will all be new names.
  2494. // Delete all names that exist in this request.
  2495. const names_to_delete = new Set();
  2496. for (let i = 0; i < results; i++) {
  2497. const name = map.get("tradelist_" + i.toString() + "_itemname");
  2498. names_to_delete.add(name);
  2499. }
  2500. for (const name of names_to_delete.values()) {
  2501. if (this.hasRename(name)) {
  2502. this.#deleteRenameData(name);
  2503. }
  2504. }
  2505. // Add in new data.
  2506. for (let i = 0; i < results; i++) {
  2507. // Extracts MarketEntry properties from keys.
  2508. const full_type = map.get("tradelist_" + i.toString() + "_item");
  2509. const base_type = typeSplit(full_type)[0];
  2510. const name = map.get("tradelist_" + i.toString() + "_itemname");
  2511. const quantity = parseInt(map.get("tradelist_" + i.toString() + "_quantity"))
  2512. const price = parseInt(map.get("tradelist_" + i.toString() + "_price"));
  2513. const trade_id = parseInt(map.get("tradelist_" + i.toString() + "_trade_id"));
  2514. const member_id = parseInt(map.get("tradelist_" + i.toString() + "_id_member"));
  2515. const member_name = map.get("tradelist_" + i.toString() + "_member_name");
  2516. const item = new Item(full_type, name, quantity);
  2517. const market_entry = new ItemMarketEntry(item, price, trade_id, member_id, member_name, this.tradezone);
  2518. const rename = item.properties.get("rename");
  2519. // Putting data into rename data.
  2520. if (!this.hasRename(rename)) { // New name that has not yet been added.
  2521. renames.add(rename);
  2522. this.renames.add(rename);
  2523. this.rename_data.set(rename, []);
  2524. this.rename_data.get(rename).push(market_entry);
  2525. } else if (renames.has(rename)) { // Name that this worker has started and has exclusive access to.
  2526. this.rename_data.get(rename).push(market_entry);
  2527. } else { // Some other worker is handling this name.
  2528. continue;
  2529. }
  2530. }
  2531. // Sorts MarketEntry[] of newly added names by price per unit, ascending.
  2532. for (const rename of renames.values()) {
  2533. const market_entries = this.rename_data.get(rename);
  2534. const item = market_entries[0].item;
  2535. if (isStackable(item)) {
  2536. market_entries.sort(function(a, b) {return (a.price/a.quantity) - (b.price/b.quantity);});
  2537. } else { // Armour has its durability in its quantity, and I don't want to sort by that.
  2538. market_entries.sort(function(a, b) {return a.price - b.price;})
  2539. }
  2540. }
  2541. }
  2542.  
  2543. #parseMixedData(response) {
  2544. // Parses item data and puts array(s) of MarketEntry into the instance.
  2545. const map = responseToMap(response);
  2546. const results = parseInt(map.get("tradelist_maxresults"));
  2547. const types = new Set(); // Contains types that this worker is operating on. These will all be new types.
  2548. const renames = new Set();
  2549. // Add in new data.
  2550. for (let i = 0; i < results; i++) {
  2551. // Extracts MarketEntry properties from keys.
  2552. const full_type = map.get("tradelist_" + i.toString() + "_item");
  2553. const base_type = typeSplit(full_type)[0];
  2554. const name = map.get("tradelist_" + i.toString() + "_itemname");
  2555. const quantity = parseInt(map.get("tradelist_" + i.toString() + "_quantity"))
  2556. const price = parseInt(map.get("tradelist_" + i.toString() + "_price"));
  2557. const trade_id = parseInt(map.get("tradelist_" + i.toString() + "_trade_id"));
  2558. const member_id = parseInt(map.get("tradelist_" + i.toString() + "_id_member"));
  2559. const member_name = map.get("tradelist_" + i.toString() + "_member_name");
  2560. const item = new Item(full_type, name, quantity);
  2561. const market_entry = new ItemMarketEntry(item, price, trade_id, member_id, member_name, this.tradezone);
  2562. const is_renamed = item.properties.has("rename");
  2563. // Putting data into item data.
  2564. if (!is_renamed) {
  2565. if (!this.hasItemType(base_type)) { // New type that has not yet been added.
  2566. types.add(base_type);
  2567. this.item_types.add(base_type);
  2568. this.item_data.set(base_type, []);
  2569. this.item_data.get(base_type).push(market_entry);
  2570. } else if (types.has(base_type)) { // Type that this worker has started and has exclusive access to.
  2571. this.item_data.get(base_type).push(market_entry);
  2572. } else { // Some other worker is handling the type.
  2573. continue;
  2574. }
  2575. } else {
  2576. // Putting data into rename data.
  2577. const rename = item.properties.get("rename");
  2578. if (!this.hasRename(rename)) { // New name that has not yet been added.
  2579. renames.add(rename);
  2580. this.renames.add(rename);
  2581. this.rename_data.set(rename, []);
  2582. this.rename_data.get(rename).push(market_entry);
  2583. } else if (renames.has(rename)) { // Name that this worker has started and has exclusive access to.
  2584. this.rename_data.get(rename).push(market_entry);
  2585. } else { // Some other worker is handling this name.
  2586. continue;
  2587. }
  2588. }
  2589. }
  2590. // Sorts MarketEntry[] of newly added types by price per unit, ascending.
  2591. for (const type of types.values()) {
  2592. const market_entries = this.item_data.get(type);
  2593. const item = market_entries[0].item;
  2594. if (isStackable(item)) {
  2595. market_entries.sort(function(a, b) {return (a.price/a.quantity) - (b.price/b.quantity);});
  2596. } else { // Armour has its durability in its quantity, and I don't want to sort by that.
  2597. market_entries.sort(function(a, b) {return a.price - b.price;})
  2598. }
  2599. }
  2600. // Sorts MarketEntry[] of newly added names by price per unit, ascending.
  2601. for (const rename of renames.values()) {
  2602. const market_entries = this.rename_data.get(rename);
  2603. const item = market_entries[0].item;
  2604. if (isStackable(item)) {
  2605. market_entries.sort(function(a, b) {return (a.price/a.quantity) - (b.price/b.quantity);});
  2606. } else { // Armour has its durability in its quantity, and I don't want to sort by that.
  2607. market_entries.sort(function(a, b) {return a.price - b.price;})
  2608. }
  2609. }
  2610. }
  2611.  
  2612. #requestItemData(tradezone, search_string, callback) {
  2613. const instance = this; // Variable to hold class instance to avoid clashing with the request (inner 'this').
  2614. const promise = new Promise(function(resolve, reject) {
  2615. const [request, parameters] = instance.#setupItemRequest(tradezone, search_string);
  2616. request.onreadystatechange = function() {
  2617. const is_complete = this.readyState == 4;
  2618. const response_ok = this.status == 200;
  2619. const client_error = this.status >= 400 && this.status < 500;
  2620. const server_error = this.status >= 500 && this.status < 600;
  2621. if (is_complete && response_ok) {
  2622. instance.#parseItemData(this.response);
  2623. instance.#_requests_out -= 1;
  2624. resolve(instance);
  2625. } else if (is_complete && (client_error || server_error)) {
  2626. instance.#_requests_out -= 1;
  2627. reject(instance);
  2628. }
  2629. }
  2630. instance.#_requests_out += 1;
  2631. request.send(parameters);
  2632. });
  2633. return promise;
  2634. }
  2635.  
  2636. #requestRenameData(tradezone, search_string) {
  2637. const instance = this; // Variable to hold class instance to avoid clashing with the request (inner 'this').
  2638. const promise = new Promise(function(resolve, reject) {
  2639. const [request, parameters] = instance.#setupItemRequest(tradezone, search_string);
  2640. request.onreadystatechange = function() {
  2641. const is_complete = this.readyState == 4;
  2642. const response_ok = this.status == 200;
  2643. const client_error = this.status >= 400 && this.status < 500;
  2644. const server_error = this.status >= 500 && this.status < 600;
  2645. if (is_complete && response_ok) {
  2646. instance.#parseRenameData(this.response);
  2647. instance.#_requests_out -= 1;
  2648. resolve(instance);
  2649. } else if (is_complete && (client_error || server_error)) {
  2650. instance.#_requests_out -= 1;
  2651. reject(instance);
  2652. }
  2653. }
  2654. instance.#_requests_out += 1;
  2655. request.send(parameters);
  2656. });
  2657. return promise;
  2658. }
  2659.  
  2660. #requestCategoricalData(tradezone, category) {
  2661. const instance = this; // Variable to hold class instance to avoid clashing with the request (inner 'this').
  2662. const promise = new Promise(function(resolve, reject) {
  2663. const [request, parameters] = instance.#setupCategoryRequest(tradezone, category);
  2664. request.onreadystatechange = function() {
  2665. const is_complete = this.readyState == 4;
  2666. const response_ok = this.status == 200;
  2667. const client_error = this.status >= 400 && this.status < 500;
  2668. const server_error = this.status >= 500 && this.status < 600;
  2669. if (is_complete && response_ok) {
  2670. instance.#parseMixedData(this.response);
  2671. instance.#_requests_out -= 1;
  2672. resolve(instance);
  2673. } else if (is_complete && (client_error || server_error)) {
  2674. instance.#_requests_out -= 1;
  2675. reject(instance);
  2676. }
  2677. }
  2678. instance.#_requests_out += 1;
  2679. request.send(parameters);
  2680. });
  2681. return promise;
  2682. }
  2683.  
  2684. // Service Requests
  2685.  
  2686. #setupServiceRequest(tradezone, service) {
  2687. // Sets up POST request for searching up market data for given service and tradezone.
  2688. const request = new XMLHttpRequest();
  2689. request.open("POST", "https://fairview.deadfrontier.com/onlinezombiemmo/trade_search.php");
  2690. request.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
  2691. // Setting up request payload.
  2692. const request_parameters = new URLSearchParams();
  2693. request_parameters.set("hash", "");
  2694. request_parameters.set("pagetime", "");
  2695. request_parameters.set("tradezone", tradezone.toString());
  2696. request_parameters.set("searchname", "");
  2697. request_parameters.set("category", "");
  2698. request_parameters.set("profession", service.toString());
  2699. request_parameters.set("memID", "");
  2700. request_parameters.set("searchtype", "buyinglist");
  2701. request_parameters.set("search", "services");
  2702. return [request, request_parameters];
  2703. }
  2704.  
  2705. #deleteServiceData(service) {
  2706. this.service_types.delete(service);
  2707. this.service_data.delete(service);
  2708. }
  2709.  
  2710. #parseServiceData(response, service) {
  2711. const map = responseToMap(response);
  2712. map.set("services", true);
  2713. const results = parseInt(map.get("tradelist_maxresults"));
  2714. const types = new Set();
  2715. // Deletes all types that exist in this request.
  2716. const types_to_delete = new Set();
  2717. for (let i = 0; i < results; i++) {
  2718. const service_type = map.get("tradelist_" + i.toString() + "_profession");
  2719. types_to_delete.add(service_type);
  2720. }
  2721. for (const type of types_to_delete.values()) {
  2722. if (this.hasServiceType(type)) {
  2723. this.#deleteServiceData(type);
  2724. }
  2725. }
  2726. // Adding in new data.
  2727. for (let i = 0; i < results; i++) {
  2728. const service_type = map.get("tradelist_" + i.toString() + "_profession");
  2729. const level = parseInt(map.get("tradelist_" + i.toString() + "_level"));
  2730. const price = parseInt(map.get("tradelist_" + i.toString() + "_price"));
  2731. const member_id = parseInt(map.get("tradelist_" + i.toString() + "_id_member"));
  2732. const member_name = map.get("tradelist_" + i.toString() + "_member_name");
  2733. const service = new Service(service_type, level);
  2734. const market_entry = new ServiceMarketEntry(service, price, member_id, member_name, this.tradezone);
  2735. // Putting data into service data.
  2736. if (!this.hasServiceType(service_type)) { // New type that has not yet been added.
  2737. types.add(service_type);
  2738. this.service_types.add(service_type);
  2739. this.service_data.set(service_type, []);
  2740. this.service_data.get(service_type).push(market_entry);
  2741. } else if (types.has(service_type)) { // Type that this worker has started and has exclusive access to.
  2742. this.service_data.get(service_type).push(market_entry);
  2743. } else { // Some other worker is handling the type.
  2744. continue;
  2745. }
  2746. }
  2747. for (const type of types.values()) {
  2748. const market_entries = this.service_data.get(type);
  2749. market_entries.sort(function(a, b) {return a.price - b.price;});
  2750. }
  2751. }
  2752.  
  2753. #requestServiceData(tradezone, service) {
  2754. // Requests market data for given service and tradezone. Calls parseServiceData to parse and store
  2755. // information.
  2756. const instance = this; // Variable to hold class instance to avoid clashing with the request (inner 'this').
  2757. const promise = new Promise(function(resolve, reject) {
  2758. const [request, parameters] = instance.#setupServiceRequest(tradezone, service);
  2759. request.onreadystatechange = function() {
  2760. const is_complete = this.readyState == 4;
  2761. const response_ok = this.status == 200;
  2762. const client_error = this.status >= 400 && this.status < 500;
  2763. const server_error = this.status >= 500 && this.status < 600;
  2764. if (is_complete && response_ok) {
  2765. instance.#_requests_out -= 1;
  2766. instance.#parseServiceData(this.response, service);
  2767. resolve(instance);
  2768. } else if (is_complete && (client_error || server_error)) {
  2769. instance.#_requests_out -= 1;
  2770. reject(instance);
  2771. }
  2772. }
  2773. instance.#_requests_out += 1;
  2774. request.send(parameters);
  2775. });
  2776. return promise;
  2777. }
  2778.  
  2779. // Public Functions
  2780.  
  2781. static itemToCategorical(item) {
  2782. if (item.category === ItemCategory.WEAPON) {
  2783. const weapon_type = item.properties.get("weapon_type");
  2784. if (["submachinegun", "machinegun"].includes(weapon_type)) {
  2785. return "weapon_lightmachinegun";
  2786. } else if (["bigmachinegun", "minigun"].includes(weapon_type)) {
  2787. return "weapon_heavymachinegun";
  2788. } else if (weapon_type === "grenadelauncher") {
  2789. return "weapon_grenadelauncher";
  2790. }
  2791. else {
  2792. return "weapon_" + item.properties.get("proficiency_type");
  2793. }
  2794. } else if (item.category === ItemCategory.ARMOUR) {
  2795. return "armour";
  2796. } else if (item.category === ItemCategory.AMMO) {
  2797. if (item.base_type.indexOf("rifle") !== -1) {
  2798. return "ammo_rifle";
  2799. } else if (item.base_type.indexOf("gauge") !== -1) {
  2800. return "ammo_shotgun";
  2801. } else if (item.base_type.indexOf("grenade") !== -1) {
  2802. return "ammo_grenade";
  2803. } else if (item.base_type.indexOf("fuel") !== -1) {
  2804. return "ammo_fuel";
  2805. } else {
  2806. return "ammo_handgun";
  2807. }
  2808. } else if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.FOOD) {
  2809. return "food";
  2810. } else if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.MEDICINE) {
  2811. return "medical";
  2812. } else if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.CLOTHING) {
  2813. const clothing_type = item.properties.get("clothing_type");
  2814. if (["mask", "hat"].includes(clothing_type)) {
  2815. return "clothing_headwear";
  2816. } else if (clothing_type === "coat") {
  2817. return "clothing_coat";
  2818. } else {
  2819. return "clothing_basic";
  2820. }
  2821. } else if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.BARRICADE) {
  2822. return "barricading";
  2823. } else if (item.category === ItemCategory.ITEM && item.subcategory === ItemSubcategory.IMPLANT) {
  2824. return "implants";
  2825. } else if (item.category === ItemCategory.BACKPACK) {
  2826. return "backpack";
  2827. } else {
  2828. return "misc";
  2829. }
  2830. }
  2831.  
  2832. #_tradezone;
  2833. #_requests_out;
  2834. constructor(tradezone) {
  2835. this.#_tradezone = tradezone;
  2836. this.item_types = new Set();
  2837. this.item_data = new Map();
  2838. this.renames = new Set();
  2839. this.rename_data = new Map();
  2840. this.service_types = new Set();
  2841. this.service_data = new Map();
  2842. this.#_requests_out = 0;
  2843. }
  2844.  
  2845. get tradezone() {
  2846. return this.#_tradezone;
  2847. }
  2848.  
  2849. get requests_out() {
  2850. return this.#_requests_out;
  2851. }
  2852.  
  2853. hasItemType(type) {
  2854. const base_type = typeSplit(type)[0];
  2855. return this.item_types.has(base_type);
  2856. }
  2857.  
  2858. hasRename(rename) {
  2859. return this.renames.has(rename);
  2860. }
  2861.  
  2862. getItemMarketEntriesByType(type) {
  2863. const base_type = typeSplit(type)[0];
  2864. const data = this.item_data.get(base_type);
  2865. if (data === undefined) {
  2866. throw new ReferenceError("Item: " + base_type + " unavailable.");
  2867. } else {
  2868. return data;
  2869. }
  2870. }
  2871.  
  2872. getItemMarketEntriesByRename(rename) {
  2873. const data = this.rename_data.get(rename);
  2874. if (data === undefined) {
  2875. throw new ReferenceError("Renamed item: " + rename + " unavailable.");
  2876. } else {
  2877. return data;
  2878. }
  2879. }
  2880.  
  2881. requestItemMarketEntriesByType(type) {
  2882. const base_type = typeSplit(type)[0];
  2883. const special_string = this.#specialSearchString(base_type);
  2884. const name = special_string !== false ? special_string : MarketCache.#global_data[base_type].name;
  2885. const promise = this.#requestItemData(this.tradezone, name);
  2886. return promise;
  2887. }
  2888.  
  2889. requestMultipleItemMarketEntriesByType(types) {
  2890. const unique_types = new Set(types); // To avoid repeats.
  2891. // Need to bind to avoid losing track of 'this', becoming undefined.
  2892. const promises = Array.from(unique_types).map(this.requestItemMarketEntriesByType.bind(this));
  2893. // Once all promises (requests) complete, unify them into a single promise that resolves to the instance.
  2894. return Promise.allSettled(promises).then(() => this);
  2895. }
  2896.  
  2897. requestItemMarketEntriesByRename(rename) {
  2898. const promise = this.#requestRenameData(this.tradezone, rename);
  2899. return promise;
  2900. }
  2901.  
  2902. requestMultipleItemMarketEntriesByRename(renames) {
  2903. const unique_renames = new Set(renames); // To avoid repeats.
  2904. // Need to bind to avoid losing track of 'this', becoming undefined.
  2905. const promises = Array.from(unique_renames).map(this.requestItemMarketEntriesByRename.bind(this));
  2906. // Once all promises (requests) complete, unify them into a single promise that resolves to the instance.
  2907. return Promise.allSettled(promises).then(() => this);
  2908. }
  2909.  
  2910. requestItemMarketEntriesByCategory(category) {
  2911. const promise = this.#requestCategoricalData(this.tradezone, category);
  2912. return promise;
  2913. }
  2914.  
  2915. requestMultipleItemMarketEntriesByCategory(categories) {
  2916. const unique_categories = new Set(categories); // To avoid repeats.
  2917. // Need to bind to avoid losing track of 'this', becoming undefined.
  2918. const promises = Array.from(unique_categories).map(this.requestItemMarketEntriesByCategory.bind(this));
  2919. // Once all promises (requests) complete, unify them into a single promise that resolves to the instance.
  2920. return Promise.allSettled(promises).then(() => this);
  2921. }
  2922.  
  2923. hasServiceType(type) {
  2924. return this.service_types.has(type);
  2925. }
  2926.  
  2927. getServiceMarketEntriesByType(type) {
  2928. const data = this.service_data.get(type);
  2929. if (data === undefined) {
  2930. throw new ReferenceError("Service: " + type + " unavailable.");
  2931. } else {
  2932. return data;
  2933. }
  2934. }
  2935.  
  2936. requestServiceMarketEntriesByType(type) {
  2937. const promise = this.#requestServiceData(this.tradezone, type);
  2938. return promise;
  2939. }
  2940.  
  2941. requestMultipleServiceMarketEntriesByType(types) {
  2942. const unique_types = new Set(types); // To avoid repeats.
  2943. // Need to bind to avoid losing track of 'this', becoming undefined.
  2944. const promises = Array.from(unique_types).map(this.requestServiceMarketEntriesByType.bind(this));
  2945. // Once all promises (requests) complete, unify them into a single promise that resolves to the instance.
  2946. return Promise.allSettled(promises).then(() => this);
  2947. }
  2948. }
  2949.  
  2950. // CollectionBook
  2951.  
  2952. class CollectionBook {
  2953. #setupCollectionBookRequest() {
  2954. // Sets up POST request for requesting collection book data.
  2955. const request = new XMLHttpRequest();
  2956. request.open("POST", "https://fairview.deadfrontier.com/onlinezombiemmo/hotrods/collectionbook.php");
  2957. request.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
  2958. // Setting up request payload.
  2959. const request_parameters = new URLSearchParams();
  2960. request_parameters.set("userID", this.#_uservars["userID"]);
  2961. request_parameters.set("password", this.#_uservars["password"]);
  2962. request_parameters.set("sc", this.#_uservars["sc"]);
  2963. request_parameters.set("pagetime", this.#_uservars["pagetime"]);
  2964. request_parameters.set("memberto", this.#_uservars["userID"]);
  2965. request_parameters.set("action", "getother");
  2966. return [request, request_parameters];
  2967. }
  2968.  
  2969. #parseCollectionBookData(response) {
  2970. const map = responseToMap(response);
  2971. const count = parseInt(map.get("total_items"));
  2972. for (let i = 0; i < count; i++) {
  2973. const base_key = "cb" + i.toString() + "_";
  2974. const base_type = map.get(base_key + "type");
  2975. const quantity = parseInt(map.get(base_key + "quantity"));
  2976. const full_type = map.get(base_key + "inv");
  2977. const pinned = map.get(base_key + "pinned") === "1";
  2978. const priority = parseInt(map.get(base_key + "priority"));
  2979. this.#_ordered_types.push(base_type);
  2980. this.#_base_types.add(base_type);
  2981. this.#_quantities.set(base_type, quantity);
  2982. this.#_full_types.set(base_type, full_type);
  2983. this.#_pinned.set(base_type, pinned);
  2984. this.#_priorities.set(base_type, priority);
  2985. }
  2986. }
  2987.  
  2988. #_uservars;
  2989. #_requests_out;
  2990. #_ordered_types;
  2991. #_base_types;
  2992. #_quantities;
  2993. #_full_types;
  2994. #_pinned;
  2995. #_priorities;
  2996. constructor() {
  2997. this.#_uservars = userVars;
  2998. this.#_requests_out = 0;
  2999. this.#_ordered_types = [];
  3000. this.#_base_types = new Set();
  3001. this.#_quantities = new Map();
  3002. this.#_full_types = new Map();
  3003. this.#_pinned = new Map();
  3004. this.#_priorities = new Map();
  3005. }
  3006.  
  3007. get requests_out() {
  3008. return this.#_requests_out;
  3009. }
  3010.  
  3011. requestCollectionBookData() {
  3012. const instance = this; // Variable to hold class instance to avoid clashing with the request (inner 'this').
  3013. const promise = new Promise(function(resolve, reject) {
  3014. const [request, parameters] = instance.#setupCollectionBookRequest();
  3015. request.onreadystatechange = function() {
  3016. const is_complete = this.readyState == 4;
  3017. const response_ok = this.status == 200;
  3018. const client_error = this.status >= 400 && this.status < 500;
  3019. const server_error = this.status >= 500 && this.status < 600;
  3020. if (is_complete && response_ok) {
  3021. instance.#_requests_out -= 1;
  3022. instance.#parseCollectionBookData(this.response);
  3023. resolve(instance);
  3024. } else if (is_complete && (client_error || server_error)) {
  3025. instance.#_requests_out -= 1;
  3026. reject(instance);
  3027. }
  3028. }
  3029. instance.#_requests_out += 1;
  3030. request.send(parameters);
  3031. });
  3032. return promise;
  3033. }
  3034.  
  3035. hasType(type) {
  3036. return this.#_base_types.has(type);
  3037. }
  3038.  
  3039. quantity(type) {
  3040. return this.#_quantities.get(type);
  3041. }
  3042.  
  3043. fullTypeInCollection(type) {
  3044. return this.#_full_types.get(type);
  3045. }
  3046.  
  3047. pinned(type) {
  3048. return this.#_pinned.get(type);
  3049. }
  3050.  
  3051. priority(type) {
  3052. return this.#_priorities.get(type);
  3053. }
  3054.  
  3055. total() {
  3056. return this.#_base_types.size;
  3057. }
  3058.  
  3059. *types() {
  3060. const n = this.total();
  3061. for (let i = 0; i < n; i++) {
  3062. yield this.#_ordered_types[i];
  3063. }
  3064. }
  3065. }
  3066.  
  3067. // InventoryUI
  3068.  
  3069. class InventoryUI {
  3070. #_player_items;
  3071. constructor() {
  3072. this.#_player_items = new PlayerItems();
  3073. }
  3074.  
  3075. #isValidSlotElementFromMouseOverElement(element) {
  3076. const is_slot_element = element.classList.contains("validSlot");
  3077. const is_item_element = element.classList.contains("item");
  3078. return is_slot_element || is_item_element;
  3079. }
  3080.  
  3081. #slotSlotTypeFromMouseOverElement(element) {
  3082. const is_slot_element = element.classList.contains("validSlot");
  3083. const is_item_element = element.classList.contains("item");
  3084. if (!is_slot_element && !is_item_element) {
  3085. throw new Error("Element: " + element + " is not a slot or item element.");
  3086. }
  3087. const slot_element = is_slot_element ? element : element.parentNode;
  3088. const slot = parseInt(element.dataset.slot);
  3089. const slot_type = "slottype" in element.dataset ? element.dataset.slottype : undefined;
  3090. return [slot, slot_type];
  3091. }
  3092.  
  3093. #elementSlotTypeCheck(element, key) {
  3094. const [slot, slot_type] = instance.#slotSlotTypeFromMouseOverElement(element);
  3095. return slot_type === key;
  3096. }
  3097.  
  3098. #elementIsInventory(element) {
  3099. return this.#elementSlotTypeCheck(element, undefined);
  3100. }
  3101.  
  3102. #elementIsImplant(element) {
  3103. return this.#elementSlotTypeCheck(element, "implant");
  3104. }
  3105.  
  3106. #elementIsWeapon(element) {
  3107. return this.#elementSlotTypeCheck(element, "weapon");
  3108. }
  3109.  
  3110. #elementIsArmour(element) {
  3111. return this.#elementSlotTypeCheck(element, "armour");
  3112. }
  3113.  
  3114. #elementIsHat(element) {
  3115. return this.#elementSlotTypeCheck(element, "hat");
  3116. }
  3117.  
  3118. #elementIsMask(element) {
  3119. return this.#elementSlotTypeCheck(element, "mask");
  3120. }
  3121.  
  3122. #elementIsCoat(element) {
  3123. return this.#elementSlotTypeCheck(element, "coat");
  3124. }
  3125.  
  3126. #elementIsShirt(element) {
  3127. return this.#elementSlotTypeCheck(element, "shirt");
  3128. }
  3129.  
  3130. #elementIsTrousers(element) {
  3131. return this.#elementSlotTypeCheck("trousers");
  3132. }
  3133. }
  3134.  
  3135. function main() {
  3136. // setTimeout(function() {
  3137. // const player_items = new PlayerItems();
  3138. // }, 1000);
  3139. // const market_cache = new MarketCache(21);
  3140. // market_cache.requestItemMarketEntriesByCategory("weapon_melee")
  3141. // .then(function() {
  3142. // console.debug(market_cache);
  3143. // });
  3144. // setTimeout(function() {
  3145. // const global_data = globalData;
  3146. // const map = new Map();
  3147. // for (const [key, value] of Object.entries(global_data)) {
  3148. // const item = new Item(key, value.name, 1);
  3149. // const transferable = item.properties.get("transferable");
  3150. // if (transferable) {
  3151. // map.set(key, MarketCache.itemToCategorical(item));
  3152. // }
  3153. // }
  3154. // console.debug(map);
  3155. // }, 1000);
  3156. // const target = player_items.implant(16);
  3157. // console.debug(target);
  3158. // player_items.mutateItem("1016", target, "fortuneimplant_statsb3jrb4")
  3159. // player_items.unlockInventorySlot(1);
  3160. // player_items.unlockBackpackSlot(6);
  3161. // console.debug("API");
  3162. // const player_items = new PlayerItems();
  3163. // for (const [slot, item] of player_items.backpackSlots()) {
  3164. // console.debug(slot, item);
  3165. // }
  3166.  
  3167. }
  3168. main();
  3169.  
  3170. return {
  3171. // Enums
  3172. Tradezone: Tradezone,
  3173. ItemCategory: ItemCategory,
  3174. ItemSubcategory: ItemSubcategory,
  3175. ServiceType: ServiceType,
  3176. ProficiencyType: ProficiencyType,
  3177. UiUpdate: UiUpdate,
  3178. // Predicates
  3179. ItemFilters: ItemFilters,
  3180. ServiceFilters: ServiceFilters,
  3181. MarketFilters: MarketFilters,
  3182. // Classes
  3183. PlayerValues: PlayerValues,
  3184. PlayerItems: PlayerItems,
  3185. MarketItems: MarketItems,
  3186. Bank: Bank,
  3187. MarketCache: MarketCache,
  3188. CollectionBook: CollectionBook
  3189. };
  3190.  
  3191. })();