Dead Frontier - API

Dead Frontier API

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