您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Перемещения по карте, поиск работы и засады с любой страницы
// ==UserScript== // @name hwmTransporter // @author Tamozhnya1 // @namespace Tamozhnya1 // @version 13.1 // @description Перемещения по карте, поиск работы и засады с любой страницы // @include *heroeswm.ru/* // @include *lordswm.com/* // @grant GM_deleteValue // @grant GM_getValue // @grant GM_setValue // @grant GM_listValues // @grant GM.xmlHttpRequest // @grant GM.notification // @license MIT // ==/UserScript== if(!this.GM_getValue || (this.GM_getValue.toString && this.GM_getValue.toString().indexOf("not supported") > -1)) { this.GM_getValue = function(key, def) { return localStorage[key] || def; }; this.GM_setValue = function(key, value) { localStorage[key] = value; }; this.GM_deleteValue = function(key) { return delete localStorage[key]; }; } this.GM_listValues = this.GM_listValues || function() { return Object.keys(localStorage); }; const playerIdMatch = document.cookie.match(/pl_id=(\d+)/); if(!playerIdMatch) { return; } const PlayerId = playerIdMatch[1]; const lang = document.documentElement.lang || (location.hostname == "www.lordswm.com" ? "en" : "ru"); const isEn = lang == "en"; const ChallengeState = { Thrown: 1, Battle: 2 }; var finalResultDiv; const isHeartOnPage = document.querySelector("canvas#heart") || document.querySelector("div#heart_js_mobile"); const isNewInterface = document.querySelector("div#hwm_header") ? true : false; const isMobileDevice = mobileCheck(); // Там нет мышки const mooving = location.pathname == '/map.php' && !document.getElementById("map_right_block"); // Граф всех возможных путей перемещения по карте const routes = [null, [null,null,["1-2",1],["1-3",1],["1-4",1.4],["1-5",1],["1-3-6",2],["1-7",1.4],["1-8",1],["1-3-9",2.4],["1-5-10",2],["1-11",1.4],["1-12",1.4],["1-8-13",2.4],["1-2-14",2],["1-2-15",2.4],["1-4-16",2.8],["1-2-14-17",3],["1-2-14-18",3.4],["1-5-19",2.4],["1-5-10-20",3.4],["1-5-19-21",3.8],["1-5-10-20-22",4.8],["1-12-23",2.8],["1-3-24",2.4],null,["1-5-26",2.4],["1-8-27",2]], [null,["2-1",1],null,["2-3",1.4],["2-4",1],["2-5",1.4],["2-3-6",2.4],["2-1-7",2.4],["2-1-8",2],["2-3-9",2.8],["2-5-10",2.4],["2-11",1],["2-1-12",2.4],["2-1-8-13",3.4],["2-14",1],["2-15",1.4],["2-4-16",2.4],["2-14-17",2],["2-14-18",2.4],["2-11-19",2],["2-11-19-20",3],["2-11-19-21",3.4],["2-11-19-20-22",4.4],["2-3-9-23",3.8],["2-4-24",2],null,["2-5-26",2.8],["2-1-8-27",3]], [null,["3-1",1],["3-2",1.4],null,["3-4",1],["3-1-5",2],["3-6",1],["3-1-7",2.4],["3-8",1.4],["3-9",1.4],["3-1-5-10",3],["3-1-11",2.4],["3-12",1],["3-12-13",2],["3-2-14",2.4],["3-4-15",2],["3-4-16",2.4],["3-2-14-17",3.4],["3-4-15-18",3],["3-1-5-19",3.4],["3-1-5-10-20",4.4],["3-1-5-19-21",4.8],["3-1-5-19-21-22",5.8],["3-9-23",2.4],["3-24",1.4],null,["3-1-5-26",3.4],["3-8-27",2.4]], [null,["4-1",1.4],["4-2",1],["4-3",1],null,["4-1-5",2.4],["4-6",1.4],["4-1-7",2.8],["4-1-8",2.4],["4-3-9",2.4],["4-1-5-10",3.4],["4-2-11",2],["4-3-12",2],["4-3-12-13",3],["4-14",1.4],["4-15",1],["4-16",1.4],["4-14-17",2.4],["4-15-18",2],["4-2-11-19",3],["4-2-11-19-20",4],["4-2-11-19-21",4.4],["4-2-11-19-20-22",5.4],["4-3-9-23",3.4],["4-24",1],null,["4-1-5-26",3.8],["4-1-8-27",3.4]], [null,["5-1",1],["5-2",1.4],["5-1-3",2],["5-1-4",2.4],null,["5-1-3-6",3],["5-7",1],["5-8",1.4],["5-1-3-9",3.4],["5-10",1],["5-11",1],["5-1-12",2.4],["5-8-13",2.8],["5-2-14",2.4],["5-2-15",2.8],["5-1-4-16",3.8],["5-2-14-17",3.4],["5-2-14-18",3.8],["5-19",1.4],["5-10-20",2.4],["5-19-21",2.8],["5-10-20-22",3.8],["5-1-12-23",3.8],["5-1-3-24",3.4],null,["5-26",1.4],["5-7-27",2.4]], [null,["6-3-1",2],["6-3-2",2.4],["6-3",1],["6-4",1.4],["6-3-1-5",3],null,["6-3-1-7",3.4],["6-3-8",2.4],["6-9",1],["6-3-1-5-10",4],["6-3-1-11",3.4],["6-12",1.4],["6-9-13",2.4],["6-4-14",2.8],["6-4-15",2.4],["6-4-16",2.8],["6-4-14-17",3.8],["6-4-15-18",3.4],["6-3-1-5-19",4.4],["6-3-1-5-10-20",5.4],["6-3-1-5-19-21",5.8],["6-3-1-5-10-20-22",6.8],["6-9-23",2],["6-24",1],null,["6-3-1-5-26",4.4],["6-12-27",2.8]], [null,["7-1",1.4],["7-1-2",2.4],["7-1-3",2.4],["7-1-4",2.8],["7-5",1],["7-1-3-6",3.4],null,["7-8",1],["7-8-12-9",3],["7-10",1.4],["7-5-11",2],["7-8-12",2],["7-8-13",2.4],["7-1-2-14",3.4],["7-1-2-15",3.8],["7-1-4-16",4.2],["7-1-2-14-17",4.4],["7-1-2-14-18",4.8],["7-5-19",2.4],["7-10-20",2.8],["7-5-19-21",3.8],["7-10-20-22",4.2],["7-8-12-23",3.4],["7-1-3-24",3.8],null,["7-26",1],["7-27",1.4]], [null,["8-1",1],["8-1-2",2],["8-3",1.4],["8-1-4",2.4],["8-5",1.4],["8-3-6",2.4],["8-7",1],null,["8-12-9",2],["8-5-10",2.4],["8-1-11",2.4],["8-12",1],["8-13",1.4],["8-1-2-14",3],["8-1-2-15",3.4],["8-1-4-16",3.8],["8-1-2-14-17",4],["8-1-2-14-18",4.4],["8-5-19",2.8],["8-5-10-20",3.8],["8-5-19-21",4.2],["8-5-10-20-22",5.2],["8-12-23",2.4],["8-3-24",2.8],null,["8-7-26",2],["8-27",1]], [null,["9-3-1",2.4],["9-3-2",2.8],["9-3",1.4],["9-3-4",2.4],["9-3-1-5",3.4],["9-6",1],["9-12-8-7",3],["9-12-8",2],null,["9-3-1-5-10",4.4],["9-3-1-11",3.8],["9-12",1],["9-13",1.4],["9-3-2-14",3.8],["9-3-4-15",3.4],["9-3-4-16",3.8],["9-3-2-14-17",4.8],["9-3-4-15-18",4.4],["9-3-1-5-19",4.8],["9-3-1-5-10-20",5.8],["9-3-1-5-19-21",6.2],["9-3-1-5-10-20-22",7.2],["9-23",1],["9-6-24",2],null,["9-12-8-7-26",4],["9-12-27",2.4]], [null,["10-5-1",2],["10-5-2",2.4],["10-5-1-3",3],["10-5-1-4",3.4],["10-5",1],["10-5-1-3-6",4],["10-7",1.4],["10-5-8",2.4],["10-5-1-3-9",4.4],null,["10-11",1.4],["10-5-1-12",3.4],["10-5-8-13",3.8],["10-11-14",2.8],["10-5-2-15",3.8],["10-5-1-4-16",4.8],["10-11-14-17",3.8],["10-11-14-18",4.2],["10-19",1],["10-20",1.4],["10-19-21",2.4],["10-20-22",2.8],["10-5-1-12-23",4.8],["10-5-1-3-24",4.4],null,["10-26",1],["10-7-27",2.8]], [null,["11-1",1.4],["11-2",1],["11-1-3",2.4],["11-2-4",2],["11-5",1],["11-1-3-6",3.4],["11-5-7",2],["11-1-8",2.4],["11-1-3-9",3.8],["11-10",1.4],null,["11-1-12",2.8],["11-1-8-13",3.8],["11-14",1.4],["11-2-15",2.4],["11-2-4-16",3.4],["11-14-17",2.4],["11-14-18",2.8],["11-19",1],["11-19-20",2],["11-19-21",2.4],["11-19-20-22",3.4],["11-1-12-23",4.2],["11-2-4-24",3],null,["11-5-26",2.4],["11-5-7-27",3.4]], [null,["12-1",1.4],["12-1-2",2.4],["12-3",1],["12-3-4",2],["12-1-5",2.4],["12-6",1.4],["12-8-7",2],["12-8",1],["12-9",1],["12-1-5-10",3.4],["12-1-11",2.8],null,["12-13",1],["12-1-2-14",3.4],["12-3-4-15",3],["12-3-4-16",3.4],["12-1-2-14-17",4.4],["12-3-4-15-18",4],["12-1-5-19",3.8],["12-1-5-10-20",4.8],["12-1-5-19-21",5.2],["12-1-5-10-20-22",6.2],["12-23",1.4],["12-3-24",2.4],null,["12-8-7-26",3],["12-27",1.4]], [null,["13-8-1",2.4],["13-8-1-2",3.4],["13-12-3",2],["13-12-3-4",3],["13-8-5",2.8],["13-9-6",2.4],["13-8-7",2.4],["13-8",1.4],["13-9",1.4],["13-8-5-10",3.8],["13-8-1-11",3.8],["13-12",1],null,["13-8-1-2-14",4.4],["13-12-3-4-15",4],["13-12-3-4-16",4.4],["13-8-1-2-14-17",5.4],["13-12-3-4-15-18",5],["13-8-5-19",4.2],["13-8-5-10-20",5.2],["13-8-5-19-21",5.6],["13-8-5-10-20-22",6.6],["13-23",1],["13-9-6-24",3.4],null,["13-8-7-26",3.4],["13-27",1]], [null,["14-2-1",2],["14-2",1],["14-2-3",2.4],["14-4",1.4],["14-2-5",2.4],["14-4-6",2.8],["14-2-1-7",3.4],["14-2-1-8",3],["14-2-3-9",3.8],["14-11-10",2.8],["14-11",1.4],["14-2-1-12",3.4],["14-2-1-8-13",4.4],null,["14-15",1],["14-15-16",2],["14-17",1],["14-18",1.4],["14-11-19",2.4],["14-11-19-20",3.4],["14-11-19-21",3.8],["14-11-19-20-22",4.8],["14-2-3-9-23",4.8],["14-4-24",2.4],null,["14-2-5-26",3.8],["14-2-1-8-27",4]], [null,["15-2-1",2.4],["15-2",1.4],["15-4-3",2],["15-4",1],["15-2-5",2.8],["15-4-6",2.4],["15-2-1-7",3.8],["15-2-1-8",3.4],["15-4-3-9",3.4],["15-2-5-10",3.8],["15-2-11",2.4],["15-4-3-12",3],["15-4-3-12-13",4],["15-14",1],null,["15-16",1],["15-17",1.4],["15-18",1],["15-2-11-19",3.4],["15-2-11-19-20",4.4],["15-2-11-19-21",4.8],["15-2-11-19-21-22",5.8],["15-4-3-9-23",4.4],["15-24",1.4],null,["15-2-5-26",4.2],["15-2-1-8-27",4.4]], [null,["16-4-1",2.8],["16-4-2",2.4],["16-4-3",2.4],["16-4",1.4],["16-4-1-5",3.8],["16-4-6",2.8],["16-4-1-7",4.2],["16-4-1-8",3.8],["16-4-3-9",3.8],["16-4-1-5-10",4.8],["16-4-2-11",3.4],["16-4-3-12",3.4],["16-4-3-12-13",4.4],["16-15-14",2],["16-15",1],null,["16-15-17",2.4],["16-18",1.4],["16-4-2-11-19",4.4],["16-4-2-11-19-20",5.4],["16-4-2-11-19-21",5.8],["16-4-2-11-19-20-22",6.8],["16-4-3-9-23",4.8],["16-4-24",2.4],null,["16-4-1-5-26",5.2],["16-4-1-8-27",4.8]], [null,["17-14-2-1",3],["17-14-2",2],["17-14-2-3",3.4],["17-14-4",2.4],["17-14-2-5",3.4],["17-14-4-6",3.8],["17-14-2-1-7",4.4],["17-14-2-1-8",4],["17-14-2-3-9",4.8],["17-14-11-10",3.8],["17-14-11",2.4],["17-14-2-1-12",4.4],["17-14-2-1-8-13",5.4],["17-14",1],["17-15",1.4],["17-15-16",2.4],null,["17-18",1],["17-14-11-19",3.4],["17-14-11-19-20",4.4],["17-14-11-19-21",4.8],["17-14-11-19-21-22",5.8],["17-14-2-3-9-23",5.8],["17-15-24",2.8],null,["17-14-2-5-26",4.8],["17-14-2-1-8-27",5]], [null,["18-14-2-1",3.4],["18-14-2",2.4],["18-15-4-3",3],["18-15-4",2],["18-14-2-5",3.8],["18-15-4-6",3.4],["18-14-2-1-7",4.8],["18-14-2-1-8",4.4],["18-15-4-3-9",4.4],["18-14-11-10",4.2],["18-14-11",2.8],["18-15-4-3-12",4],["18-15-4-3-12-13",5],["18-14",1.4],["18-15",1],["18-16",1.4],["18-17",1],null,["18-14-11-19",3.8],["18-14-11-19-20",4.8],["18-14-11-19-21",5.2],["18-14-11-19-20-22",6.2],["18-15-4-3-9-23",5.4],["18-15-24",2.4],null,["18-14-2-5-26",5.2],["18-14-2-1-8-27",5.4]], [null,["19-5-1",2.4],["19-11-2",2],["19-5-1-3",3.4],["19-11-2-4",3],["19-5",1.4],["19-5-1-3-6",4.4],["19-5-7",2.4],["19-5-8",2.8],["19-5-1-3-9",4.8],["19-10",1],["19-11",1],["19-5-1-12",3.8],["19-5-8-13",4.2],["19-11-14",2.4],["19-11-2-15",3.4],["19-11-2-4-16",4.4],["19-11-14-17",3.4],["19-11-14-18",3.8],null,["19-20",1],["19-21",1.4],["19-20-22",2.4],["19-5-1-12-23",5.2],["19-11-2-4-24",4],null,["19-10-26",2],["19-5-7-27",3.8]], [null,["20-10-5-1",3.4],["20-19-11-2",3],["20-10-5-1-3",4.4],["20-19-11-2-4",4],["20-10-5",2.4],["20-10-5-1-3-6",5.4],["20-10-7",2.8],["20-10-5-8",3.8],["20-10-5-1-12-9",5.8],["20-10",1.4],["20-19-11",2],["20-10-5-1-12",4.8],["20-10-5-8-13",5.2],["20-19-11-14",3.4],["20-19-11-2-15",4.4],["20-19-11-2-4-16",5.4],["20-19-11-14-17",4.4],["20-19-11-14-18",4.8],["20-19",1],null,["20-21",1],["20-22",1.4],["20-10-5-1-12-23",6.2],["20-19-11-2-4-24",5],null,["20-10-26",2.4],["20-10-7-27",4.2]], [null,["21-19-5-1",3.8],["21-19-11-2",3.4],["21-19-5-1-3",4.8],["21-19-11-2-4",4.4],["21-19-5",2.8],["21-19-5-1-3-6",5.8],["21-19-5-7",3.8],["21-19-5-8",4.2],["21-19-5-1-3-9",6.2],["21-19-10",2.4],["21-19-11",2.4],["21-19-5-1-12",5.2],["21-19-5-8-13",5.6],["21-19-11-14",3.8],["21-19-11-2-15",4.8],["21-19-11-2-4-16",5.8],["21-19-11-14-17",4.8],["21-19-11-14-18",5.2],["21-19",1.4],["21-20",1],null,["21-22",1],["21-19-5-1-12-23",6.6],["21-19-11-2-4-24",5.4],null,["21-19-10-26",3.4],["21-19-5-7-27",5.2]], [null,["22-20-10-5-1",4.8],["22-20-19-11-2",4.4],["22-20-10-5-1-3",5.8],["22-20-19-11-2-4",5.4],["22-20-10-5",3.8],["22-20-10-5-1-3-6",6.8],["22-20-10-7",4.2],["22-20-10-5-8",5.2],["22-20-10-5-1-12-9",7.2],["22-20-10",2.8],["22-20-19-11",3.4],["22-20-10-5-1-12",6.2],["22-20-10-5-8-13",6.6],["22-20-19-11-14",4.8],["22-20-19-11-2-15",5.8],["22-20-19-11-2-4-16",6.8],["22-20-19-11-14-17",5.8],["22-20-19-11-14-18",6.2],["22-20-19",2.4],["22-20",1.4],["22-21",1],null,["22-20-10-5-1-12-23",7.6],["22-20-19-11-2-4-24",6.4],null,["22-20-10-26",3.8],["22-20-10-7-27",5.6]], [null,["23-12-1",2.8],["23-9-3-2",3.8],["23-9-3",2.4],["23-9-3-4",3.4],["23-12-1-5",3.8],["23-9-6",2],["23-12-8-7",3.4],["23-12-8",2.4],["23-9",1],["23-12-1-5-10",4.8],["23-12-1-11",4.2],["23-12",1.4],["23-13",1],["23-9-3-2-14",4.8],["23-9-3-4-15",4.4],["23-9-3-4-16",4.8],["23-9-3-2-14-17",5.8],["23-9-3-4-15-18",5.4],["23-12-1-5-19",5.2],["23-12-1-5-10-20",6.2],["23-12-1-5-19-21",6.6],["23-12-1-5-10-20-22",7.6],null,["23-9-6-24",3],null,["23-12-8-7-26",4.4],["23-13-27",2]], [null,["24-3-1",2.4],["24-4-2",2],["24-3",1.4],["24-4",1],["24-3-1-5",3.4],["24-6",1],["24-3-1-7",3.8],["24-3-8",2.8],["24-6-9",2],["24-3-1-5-10",4.4],["24-4-2-11",3],["24-3-12",2.4],["24-6-9-13",3.4],["24-4-14",2.4],["24-15",1.4],["24-4-16",2.4],["24-15-17",2.8],["24-15-18",2.4],["24-4-2-11-19",4],["24-4-2-11-19-20",5],["24-4-2-11-19-21",5.4],["24-4-2-11-19-20-22",6.4],["24-6-9-23",3],null,null,["24-3-1-5-26",4.8],["24-3-8-27",3.8]], [], [null,["26-5-1",2.4],["26-5-2",2.8],["26-5-1-3",3.4],["26-5-1-4",3.8],["26-5",1.4],["26-5-1-3-6",4.4],["26-7",1],["26-7-8",2],["26-7-8-12-9",4],["26-10",1],["26-5-11",2.4],["26-7-8-12",3],["26-7-8-13",3.4],["26-5-2-14",3.8],["26-5-2-15",4.2],["26-5-1-4-16",5.2],["26-5-2-14-17",4.8],["26-5-2-14-18",5.2],["26-10-19",2],["26-10-20",2.4],["26-10-19-21",3.4],["26-10-20-22",3.8],["26-7-8-12-23",4.4],["26-5-1-3-24",4.8],null,null,["26-7-27",2.4]], [null,["27-8-1",2],["27-8-1-2",3],["27-8-3",2.4],["27-8-1-4",3.4],["27-7-5",2.4],["27-12-6",2.8],["27-7",1.4],["27-8",1],["27-12-9",2.4],["27-7-10",2.8],["27-7-5-11",3.4],["27-12",1.4],["27-13",1],["27-8-1-2-14",4],["27-8-1-2-15",4.4],["27-8-1-4-16",4.8],["27-8-1-2-14-17",5],["27-8-1-2-14-18",5.4],["27-7-5-19",3.8],["27-7-10-20",4.2],["27-7-5-19-21",5.2],["27-7-10-20-22",5.6],["27-13-23",2],["27-8-3-24",3.8],null,["27-7-26",2.4]]]; // Номера, названия и координаты секторов const locations = { 1: [50,50,"Empire Capital","EmC","Столица Империи"], 2: [51,50,"East River","EsR","Восточная Река"], 3: [50,49,"Tiger Lake","TgL","Тигриное Озеро"], 4: [51,49,"Rogues' Wood","RgW","Лес Разбойников"], 5: [50,51,"Wolf Dale","WoD","Долина Волков"], 6: [50,48,"Peaceful Camp","PcC","Мирный Лагерь"], 7: [49,51,"Lizard Lowland","LzL","Равнина Ящеров"], 8: [49,50,"Green Wood","GrW","Зеленый Лес"], 9: [49,48,"Eagle Nest","EgN","Орлиное Гнездо"], 10: [50,52,"Portal Ruins","PoR","Руины Портала"], 11: [51,51,"Dragons' Caves","DrC","Пещеры Драконов"], 12: [49,49,"Shining Spring","ShS","Сияющий Родник"], 13: [48,49,"Sunny City","SnC","Солнечный Город"], 14: [52,50,"Magma Mines","MgM","Магма Шахты"], 15: [52,49,"Bear Mountain","BrM","Медвежья Гора"], 16: [52,48,"Fairy Trees","FrT","Магический Лес"], 17: [53,50,"Harbour City","HrC","Портовый Город"], 18: [53,49,"Mythril Coast","MfC","Мифриловый Берег"], 19: [51,52,"Great Wall","GtW","Великая Стена"], 20: [51,53,"Titans' Valley","TiV","Равнина Титанов"], 21: [52,53,"Fishing Village","FsV","Рыбачье село"], 22: [52,54,"Kingdom Castle","KiC","Замок Королевства"], 23: [48,48,"Ungovernable Steppe","UnS","Непокорная Степь"], 24: [51,48,"Crystal Garden","CrG","Кристальный Сад"], 25: [53,52,"East Island","EsI","Восточный Остров"], 26: [49,52,"The Wilderness","ThW","Дикие земли"], 27: [48,50,"Sublime Arbor","SbA","Великое Древо"], 101: [48, 53, "Watchers' guild", "", "Гильдия стражей", { Url: "/task_guild.php", Color: "DarkGoldenRod" }], 102: [48, 54, "Leaders' Guild", "", "Гильдия лидеров", { Url: "/leader_guild.php", Color: "DarkGoldenRod" }], 103: [49, 54, "Adventurers' guild", "", "Гильдия искателей", { Url: "/campaign_list.php", Color: "DarkGoldenRod" }], 104: [50, 54, "Thieves' guild", "", "Гильдия воров", { Url: "/thief_guild.php", Color: "DarkGoldenRod" }], 105: [51, 54, "Smith", "", "Кузница", { Url: "/mod_workbench.php?type=repair", Color: "DarkGoldenRod" }], 106: [53, 51, "Repeat last ambush", "", "Повторить последнюю засаду"], 107: [52, 51, "Find nearest production", "", "Найти ближайшее производство"], 108: [53, 48, "Toggle top settings", "", "Переключить верхние настройки"], 109: [53, 54, "Toggle bottom settings", "", "Переключить нижние настройки"] }; const objectLocations = { 0 : 0 , 5: 1, 9: 1, 6: 1, 7: 1, 4: 1, 3: 1, 8: 1, 165: 1, 10: 1, 11: 1, 12: 1, 32: 1, 34: 1, 38: 1 , 238: 2, 300: 2, 75: 2, 26: 2, 258: 2, 279: 2, 25: 2, 23: 2, 33: 2, 342: 2, 24: 2, 36: 2, 87: 2, 89: 2, 321: 2, 28: 2 , 239: 3, 16: 3, 15: 3, 13: 3, 259: 3, 343: 3, 280: 3, 84: 3, 301: 3, 224: 3, 27: 3, 39: 3, 14: 3, 31: 3, 35: 3, 322: 3 , 344: 4, 281: 4, 302: 4, 260: 4, 19: 4, 21: 4, 22: 4, 323: 4, 18: 4, 20: 4, 30: 4, 37: 4, 78: 4, 90: 4, 225: 4, 240: 4 , 282: 5, 74: 5, 226: 5, 43: 5, 44: 5, 303: 5, 85: 5, 45: 5, 46: 5, 47: 5, 48: 5, 86: 5, 241: 5, 261: 5, 324: 5, 345: 5 , 50: 6, 283: 6, 262: 6, 304: 6, 73: 6, 49: 6, 141: 6, 51: 6, 79: 6, 53: 6, 54: 6, 55: 6, 52: 6, 82: 6, 325: 6, 346: 6 , 56: 7, 326: 7, 59: 7, 64: 7, 284: 7, 58: 7, 60: 7, 61: 7, 63: 7, 80: 7, 83: 7, 242: 7, 263: 7, 57: 7, 305: 7, 347: 7 , 285: 8, 243: 8, 68: 8, 88: 8, 69: 8, 67: 8, 71: 8, 72: 8, 76: 8, 77: 8, 81: 8, 70: 8, 264: 8, 306: 8, 327: 8, 348: 8 , 244: 9, 328: 9, 227: 9, 265: 9, 286: 9, 94: 9, 349: 9, 139: 9, 307: 9, 101: 9, 98: 9, 95: 9, 119: 9, 120: 9, 140: 9, 97: 9 , 163: 10, 329: 10, 350: 10, 99: 10, 100: 10, 102: 10, 118: 10, 211: 10, 217: 10, 287: 10, 308: 10, 93: 10, 228: 10, 266: 10, 245: 10, 92: 10 , 167: 11, 169: 11, 168: 11, 210: 11, 172: 11, 170: 11, 209: 11, 171: 11, 218: 11, 229: 11, 246: 11, 267: 11, 288: 11, 309: 11, 330: 11, 351: 11 , 247: 12, 331: 12, 289: 12, 219: 12, 117: 12, 108: 12, 111: 12, 110: 12, 109: 12, 112: 12, 113: 12, 114: 12, 230: 12, 268: 12, 310: 12, 352: 12 , 104: 13, 311: 13, 269: 13, 220: 13, 116: 13, 248: 13, 332: 13, 106: 13, 107: 13, 115: 13, 213: 13, 231: 13, 103: 13, 290: 13, 105: 13, 353: 13 , 270: 14, 122: 14, 312: 14, 249: 14, 216: 14, 333: 14, 164: 14, 291: 14, 121: 14, 135: 14, 142: 14, 143: 14, 144: 14, 145: 14, 232: 14, 354: 14 , 313: 15, 334: 15, 124: 15, 125: 15, 214: 15, 136: 15, 123: 15, 146: 15, 147: 15, 148: 15, 149: 15, 215: 15, 250: 15, 271: 15, 292: 15, 355: 15 , 335: 16, 233: 16, 272: 16, 251: 16, 126: 16, 134: 16, 153: 16, 151: 16, 127: 16, 152: 16, 150: 16, 212: 16, 221: 16, 293: 16, 314: 16, 356: 16 , 131: 17, 252: 17, 273: 17, 315: 17, 294: 17, 336: 17, 162: 17, 161: 17, 160: 17, 132: 17, 133: 17, 222: 17, 234: 17, 158: 17, 159: 17, 357: 17 , 358: 18, 235: 18, 253: 18, 130: 18, 129: 18, 137: 18, 138: 18, 154: 18, 128: 18, 155: 18, 156: 18, 157: 18, 274: 18, 295: 18, 316: 18, 337: 18 , 192: 19, 202: 19, 179: 19, 173: 19, 193: 19, 194: 19, 195: 19, 201: 19, 178: 19, 203: 19, 254: 19, 275: 19, 296: 19, 317: 19, 338: 19, 359: 19 , 191: 20, 176: 20, 177: 20, 187: 20, 188: 20, 189: 20, 190: 20, 206: 20, 207: 20, 208: 20, 255: 20, 276: 20, 297: 20, 318: 20, 339: 20, 360: 20 , 174: 21, 199: 21, 197: 21, 166: 21, 361: 21, 198: 21, 196: 21, 200: 21, 223: 21, 236: 21, 256: 21, 277: 21, 298: 21, 319: 21, 340: 21, 175: 21 , 186: 22, 184: 22, 320: 22, 185: 22, 183: 22, 362: 22, 182: 22, 180: 22, 204: 22, 205: 22, 237: 22, 257: 22, 278: 22, 299: 22, 341: 22, 181: 22 , 365: 23, 364: 23, 370: 23, 363: 23, 369: 23, 366: 23, 371: 23, 372: 23, 373: 23, 374: 23, 375: 23, 376: 23, 377: 23, 378: 23, 379: 23, 380: 23 , 367: 24, 392: 24, 390: 24, 388: 24, 383: 24, 384: 24, 385: 24, 386: 24, 387: 24, 368: 24, 389: 24, 381: 24, 391: 24, 382: 24, 393: 24, 394: 24 , 395: 26, 409: 26, 399: 26, 397: 26, 398: 26, 400: 26, 401: 26, 402: 26, 403: 26, 404: 26, 405: 26, 406: 26, 407: 26, 408: 26, 396: 26, 410: 26 , 411: 27, 419: 27, 413: 27, 414: 27, 415: 27, 416: 27, 417: 27, 418: 27, 412: 27, 420: 27, 421: 27, 422: 27, 423: 27, 424: 27, 425: 27, 426: 27 }; const minX = 48; const maxX = 53; const minY = 48; const maxY = 54; let isInBattle = false; let tryGetAmbushStateTimer; const DefaultTravelingTime = 40; let AmbushMinutesInterval = 60; const isMobileInterface = document.querySelector("div#btnMenuGlobal") ? true : false; const GoTo = isEn ? "Go to" : "Перейти в"; const Second = isEn ? "sec." : "с."; const YourPremiumMountUntil = isEn ? "Your premium mount until" : "Ваш премиум транспорт"; const YourMountUntil = isEn ? "Your mount until" : "Ваш транспорт до"; const Until = isEn ? " until " : " до "; const UndressBeforeMove = isEn ? "Undress before move" : "Раздеться перед перемещением"; const NotUseComplexRoute = isEn ? "Not use complex route" : "Не использовать сложный маршрут"; const HideMapName = isEn ? "Hide map" : "Скрыть карту"; const HideHuntName = isEn ? "Hide hunt" : "Скрыть охоты"; const HideRightBlockName = isEn ? "Hide right block" : "Скрыть правый блок"; const ComplexRouteName = isEn ? "Complex route" : "Сложный маршрут"; const MountUntilName = isEn ? "Mount until" : "Транспорт до"; const BreakTravelingName = isEn ? "Break traveling" : "Прервать путешествие"; const TravelingNotEnabled = isEn ? "Traveling not enabled" : "Путешествие не доступно"; const YouAreInAChallenge = isEn ? "You are in a challenge. Your actions are limited!" : "Вы находитесь в заявке на бой. Ваши действия ограничены!"; const How = isEn ? "How?" : "Как?"; const YouAreHere = isEn ? "You're already here" : "Ты уже здесь"; const AmbushMaySettedOnAdjacentSector = isEn ? "Ambush may setted on adjacent sector" : "Засада ставится на соседний сектор"; const ItIsTooEarly = isEn ? "It is too early" : "Ещё рано"; const Legend = isEn ? "Click: moving, ctrl(mdl)+click: ambush, alt+click: sector info" : "Click: движение, ctrl(mdl)+click: засада, alt+click: просмотр"; const MoveHere = isEn ? "Move here" : "Перейти сюда"; const YouAreInADifferentLocation = isEn ? "You are in a different location." : "Вы находитесь в другом районе."; const TravelingTimeName = isEn ? "Traveling time" : "Время пути"; const Antithief = isEn ? "Antithief" : "Антивор"; const AmbushingName = isEn ? "You are in ambush" : "Вы в засаде"; const CheckMountName = isEn ? "Check mount" : "Проверить транспорт"; const LoadName = isEn ? "Load" : "Грузить"; const PremiunAccountName = isEn ? "Abu" : "Абу"; const locationNamesLocalizedColumn = isEn ? 2 : 4; const AmbushResult = { NotFound: 0, Win: 1, Fail: 2 }; const factoryTypes = ["mn", "fc", "sh"]; const Mining = isEn ? "Mining" : "Добыча"; const Machining = isEn ? "Machining" : "Обработка"; const Production = isEn ? "Production" : "Производство"; const PopupAlwaysName = isEn ? "Popup always" : "Всегда всплывающая"; const HideBattlesDayWithName = isEn ? "Hide battles day with..." : "Скрыть день боёв с..."; const FindNearestMiningName = isEn ? "Find nearest mining" : "Найти ближайшую добычу"; const FindNearestManufactureName = isEn ? "Find nearest manufacture" : "Найти ближайшую обработку"; const FindNearestProductionName = isEn ? "Find nearest production" : "Найти ближайшее производство"; const MoveToWorkAfterFindName = isEn ? "Move to work after find" : "Сразу идти на предприятие"; const ShowLocationNumbersName = isEn ? "Show location numbers" : "Номера локаций"; const JobFinding = isEn ? "Job finding" : "Поиск работы"; const win = window.wrappedJSObject || unsafeWindow; let playerLocationNumber = getPlayerLocationNumber(); if(!playerLocationNumber) { return; // При первом запуске скрипта, если мы не на карте, неизвестна текущая локация игрока } let MountInfo; var AmbushMoratoriumPanel; const resourcesPath = `${location.protocol}//${location.host.replace("www", "dcdn")}`; const resourcesPath1 = `${location.protocol}//${location.host.replace("www", "dcdn1")}`; const resourcesPath2 = `${location.protocol}//${location.host.replace("www", "dcdn2")}`; const resourcesPath3 = `${location.protocol}//${location.host.replace("www", "dcdn3")}`; main(); async function main(forceGetMountInfo) { initUserName(); // if(location.pathname == "/object-info.php") { // const codeInput = document.querySelector("input#code"); // if(codeInput) { // console.log(codeInput) // codeInput.dispatchEvent(new MouseEvent('mousedown')) // codeInput.focus(); // codeInput.select(); // } // } if(location.pathname == "/home.php" && document.querySelector("img[src*='i/icons/attr_defense.png']") && !document.querySelector("a[href*='home.php?skipn=1']")) { setValue("IsDeer", document.querySelector("img[src*='deer2.png']") ? true : false); const starImage = document.querySelector("img[src$='i/star_extend.png']") || document.querySelector("img[src$='i/star.png']"); if(starImage) { const abuBlessInfo = starImage.title || starImage.getAttribute("hint"); const time_prem = /(\d+-\d+-\d+ \d+:\d+)/.exec(abuBlessInfo); if(time_prem) { const transportEndDate = parseDate(time_prem[1], true).getTime(); if(!getPlayerBool("IsPremiunAccount") || Math.abs(parseInt(getPlayerValue("TransportEndDate", 0)) - transportEndDate) > 70000) { forceGetMountInfo = true; } } } } MountInfo = await getMountInfo(forceGetMountInfo); AmbushMinutesInterval *= 1 - (MountInfo.IsPremiunAccount ? 0.3 : 0); AmbushMinutesInterval *= 1 - (MountInfo.IsDeer ? 0.4 : 0); tryGetAmbushState(); if(!isHeartOnPage) { return; } healthTimer(); requestServerTime(); checkActivity(); setActivity(); if(/map.php/.test(location.href)) { if(mooving) { drawStopButton(); return; } setPlayerValue("LastObservingLocationNumber", location.search != '' ? getLocationNumberFromMapUrlByXy(location.href) : playerLocationNumber); loadFactories(); } drawMap(); processMoving(); checkAmbushResult(); if(location.pathname == '/object-info.php') { const objectLocationReference = document.querySelector("center a[href^='map.php?cx='][href*='&cy=']"); //console.log(`objectLocationReference: ${objectLocationReference}`); const moveHereReference = getMoveToObjectReference(location.pathname + location.search, objectLocationReference); // Добавление кнопки перемещения на предприятие } if(location.pathname == '/ecostat_details.php' || location.pathname == '/home.php') { const objs = document.querySelectorAll("a[href*='object-info.php?id=']"); const doubledHash = []; for(let obj of objs) { if(!doubledHash.includes(obj.href)) { const moveHereReference = getMoveToObjectReference(obj.href, obj); doubledHash.push(obj.href); } } } if(location.pathname == '/ecostat.php') { const table = document.querySelector("div#tableDiv > table") observe(table, function() { for(const tooltipDiv of document.querySelectorAll("div[name=ecostatDetailsTooltipDiv]")) { const objectInfoRefsAll = Array.from(tooltipDiv.querySelectorAll("div#tableDiv tr > td:first-child > a[href^='object-info.php']")); objectInfoRefsAll.forEach(x => getMoveToObjectReference(x.href, x)); } }); } if(location.pathname == '/group_wars.php') { addMoveToMapObjectReferences(); } if(location.pathname == '/mercenary_guild.php') { const colorElement = document.querySelector("font[color='#E65054']"); if(colorElement?.querySelector("b").innerText == YouAreInADifferentLocation) { const lastObservingLocationNumber = parseInt(getPlayerValue("LastObservingLocationNumber", 0)); if([2, 6, 16, 21].includes(lastObservingLocationNumber)) { const ref = getMoveToMapObjectReference(location.href, colorElement, lastObservingLocationNumber); } else { const nearestLocations = getLocationsSortedByDistance(); //console.log(`nearestLocations: ${nearestLocations}`); for(const nearestLocation of nearestLocations) { if([2, 6, 16, 21].includes(parseInt(nearestLocation))) { const ref = getMoveToMapObjectReference(location.href, colorElement, nearestLocation); break; } } } } } if(location.pathname == '/pirate_event.php') { const tableDiv = document.getElementById("tableDiv"); addMoveToMapObjectReferences(tableDiv); } processHouseInfo(); checkPatrolling(); } function checkActivity() { checkTurnedOn(); setTimeout(checkActivity, 1000); } function setActivity() { checkTurnedOn(); setValue("LastActivityDate", Date.now()); setTimeout(setActivity, 60 * 1000); } function checkTurnedOn() { const computerTurnedOn = parseInt(getValue("LastActivityDate", 0)) + 5 * 60 * 1000 < Date.now(); if(computerTurnedOn) { checkAmbushResult(true); } } function addMoveToMapObjectReferences(element) { element = element || document; const locationRefs = element.querySelectorAll("a[href^='map.php?cx=']"); for(let locationRef of locationRefs) { const locationNumber = getLocationNumberByCoordinate(getUrlParamValue(locationRef.href, "cx"), getUrlParamValue(locationRef.href, "cy")); const ref = getMoveToMapObjectReference(location.href, locationRef, locationNumber); } } function getObjectLocations() { const objs = document.querySelectorAll("a[href*='object-info.php?id=']"); let absentList = ""; for(let obj of objs) { const objectId = getUrlParamValue(obj.href, "id"); if(!objectLocations[objectId]) { objectLocations[objectId] = playerLocationNumber; absentList += `, ${objectId}: ${playerLocationNumber}`; } } if(absentList != "") { console.log(absentList); } //console.log(absentList); } function drawStopButton() { if(mooving && getPlayerValue("TargetLocationNumber")) { const insideMap = document.querySelector("#inside_map"); const mapMoving = addElement("div", { id: "mapMoving" }, insideMap); if((!MountInfo.ComplexRoute || getPlayerBool("IgnoreComplexRoute"))) { let targetLocationNumber = parseInt(getPlayerValue("TargetLocationNumber")); let route = routes[playerLocationNumber][targetLocationNumber][0]; let nextLocationNumber = route.split('-')[1]; if(nextLocationNumber != targetLocationNumber) { const stopMovingButton = addElement("div", { id: "stopMovingButton", class: "home_button2 btn_hover2 map_btn_width", style: "width: 300px;", innerHTML: BreakTravelingName }, mapMoving); stopMovingButton.addEventListener("click", stopMoving); } } if(getPlayerBool("IsPatrolling")) { const togglePatrollingButton = addElement("div", { id: "togglePatrollingButton", class: "home_button2 btn_hover2 map_btn_width", style: "width: 300px;", innerHTML: getPlayerBool("IsPatrolling") ? (isEn ? "Stop patrolling" : "Остановить патрулирование") : (isEn ? "Start patrolling" : "Начать патрулирование") }, mapMoving); togglePatrollingButton.addEventListener("click", togglePatrolling); } } } async function toggleAntithief(toggleAntithiefCheckbox) { const toggleKey = getPlayerValue("ToggleAntithiefKey"); if(toggleKey && MountInfo.AntithiefControlled) { const toggleVar = toggleAntithiefCheckbox.checked ? "t_on" : "t_off"; //console.log(`toggleKey: ${toggleKey}, url: /shop.php?${toggleVar}=${toggleKey}&cat=transport`); await getRequest(`/shop.php?${toggleVar}=${toggleKey}&cat=transport`); } } function toggleHuntBlock() { const map_hunt_block_div = document.getElementById("map_hunt_block_div"); if(map_hunt_block_div) { const mapRightBlock = document.getElementById("map_right_block"); let mapRightBlockHeight = mapRightBlock.offsetHeight; if(getPlayerBool("HideHuntBlock")) { mapRightBlockHeight -= map_hunt_block_div.offsetHeight; } map_hunt_block_div.style.display = getPlayerBool("HideHuntBlock") ? "none" : "block"; if(!getPlayerBool("HideHuntBlock")) { mapRightBlockHeight += map_hunt_block_div.offsetHeight; } mapRightBlock.style.height = `${mapRightBlockHeight}px`; } } function toggleBattlesDayWith() { const battlesDayTable = [...document.querySelectorAll("div#map_right_block_inside > table.wbwhite.rounded_table.map_table_margin")].find(x => !x.innerHTML.includes("mercenary_guild.php") && !x.innerHTML.includes("pirate_event.php") && !x.innerHTML.includes(isEn ? "Valentine's Card thieves" : "Похитители валентинок") ); if(battlesDayTable) { battlesDayTable.style.display = getPlayerBool("HideBattlesDayWithBlock") ? "none" : ""; } } function toggleRightBlock() { const mapRightBlock = document.getElementById("map_right_block"); if(mapRightBlock) { mapRightBlock.style.display = getPlayerBool("HideRightBlock") ? "none" : "block"; } } function toggleViewImages() { Array.from(document.querySelectorAll("a[name=locationViewImage]")).forEach(x => x.style.display = getPlayerBool("showLocationViewImages") ? "block" : "none"); } async function tryGetAmbushState() { if(location.pathname == '/map.php') { //const ambushDiv = Array.from(document.querySelectorAll("div#map_right_block .wbwhite.rounded_table")).find(x => x.innerHTML.includes(AmbushingName)); const ambushDiv = document.querySelector("div#map_right_block div#rtdiv"); if(ambushDiv) { setPlayerValue("ChallengeState", ChallengeState.Thrown); deletePlayerValue("AmbushSuspendExpireDate"); if(getPlayerBool("NewAmbushTimer")) { const ambushPath = getAmbushPath(); setTimeout("clearTimeout(Timer)", 0); // Останавливаем таймер разработчиков let hwmTransporterLeftThief = document.getElementById("hwmTransporterLeftThief"); if(!hwmTransporterLeftThief) { //ambushDiv.style.whiteSpace = "nowrap"; ambushDiv.innerHTML = `<img id="hwmTransporterLeftThief" src="https://dcdn.heroeswm.ru/i/portraits/thiefwarrioranip33.png" style="width: 16px; height: 16px; transition-duration: 0.8s;" /><a href="map.php"><span id="hwmTransporterAmbushStateTime">00</span> с.</a><span title="${isEn ? "Average waiting time" : "Среднее время ожидания" }">[${Math.round(parseFloat(getValue(`AmbushAverageTime${ambushPath}`, 0)))}]</span><img id="hwmTransporterRightThief" src="https://dcdn.heroeswm.ru/i/portraits/thiefwarrioranip33.png" style="width: 16px; height: 16px; display: none; transition-duration: 0.8s;" />`; hwmTransporterLeftThief = document.getElementById("hwmTransporterLeftThief"); } if(!getPlayerValue("AmbushBeginDate")) { setPlayerValue("AmbushBeginDate", getServerTime()); isInBattle = false; } const ambushStateTime = Math.round((getServerTime() - parseInt(getPlayerValue("AmbushBeginDate"))) / 1000); if(isInBattle) { deletePlayerValue("AmbushBeginDate"); setAmbushStatistics(ambushPath, ambushStateTime); window.location = "/map.php"; return; } const thiefPositionDuration = 3; // 3 секунды вор стоит в одной позиции const thiefPosition = Math.floor(ambushStateTime / thiefPositionDuration) % 4; hwmTransporterLeftThief.style.display = thiefPosition == 0 || thiefPosition == 3 ? "" : "none"; ambushDiv.querySelector("img#hwmTransporterRightThief").style.display = thiefPosition == 1 || thiefPosition == 2 ? "" : "none"; hwmTransporterLeftThief.style.transform = thiefPosition == 0 || thiefPosition == 1 ? "rotateY(180deg)" : "none"; ambushDiv.querySelector("img#hwmTransporterRightThief").style.transform = thiefPosition == 0 || thiefPosition == 1 ? "rotateY(180deg)" : "none"; ambushDiv.querySelector("span#hwmTransporterAmbushStateTime").innerText = ambushStateTime.toString().padStart(2, '0'); if(ambushStateTime % 5 == 0) { inBattle(); } setTimeout(tryGetAmbushState, 1000);// tryGetAmbushStateTimer = setTimeout(function() { tryGetAmbushState(); }, 1000); } } else { deletePlayerValue("AmbushBeginDate"); if(getPlayerValue("ChallengeState") == ChallengeState.Thrown) { deletePlayerValue("ChallengeState"); } } } if(location.pathname == '/war.php') { const warId = getUrlParamValue(location.href, "warid"); const lt = getUrlParamValue(location.href, "lt"); finalResultDiv = document.getElementById("finalresult_text"); if(lt != "-1" && finalResultDiv.innerHTML.length <= 10 && getPlayerValue("ChallengeState") == ChallengeState.Thrown) { deletePlayerValue("AmbushBeginDate"); setPlayerValue("ChallengeState", ChallengeState.Battle); observe(finalResultDiv, parseBattleResultPanel); } } } function getAmbushPath() { const ambushDiv = document.querySelector("div#map_right_block div#rtdiv"); if(ambushDiv) { const ambushTable = getParent(ambushDiv, "table"); const text = isEn ? "You are in ambush. Looking for opponent. " : "Вы в засаде, идет поиск противника. "; const ambushPathExec = new RegExp(`${text}\\((.+<->.+)\\)`).exec(ambushTable.innerHTML); if(ambushPathExec) { const ambushPath = ambushPathExec[1].replace("<", "").replace(">", ""); //console.log(`ambushPath: ${ambushPath}`); return ambushPath; } } } function setAmbushStatistics(ambushPath, ambushTime) { ambushTime = Math.min(ambushTime, 45); // В фоне таймер может идти дольше максимума в 45 сек. const ambushAverageTime = parseFloat(getValue(`AmbushAverageTime${ambushPath}`, 0)); const ambushAmount = parseInt(getValue(`AmbushAmount${ambushPath}`, 0)); const newAmbushAmount = ambushAmount + 1; const newAmbushAverageTime = Math.round((ambushAverageTime + (ambushTime - ambushAverageTime) / newAmbushAmount) * 10000) / 10000; //console.log(`ambushAverageTime: ${ambushAverageTime}, ambushAmount: ${ambushAmount}, newAmbushAmount: ${newAmbushAmount}, newAmbushAverageTime: ${newAmbushAverageTime}`); setValue(`AmbushAmount${ambushPath}`, newAmbushAmount); setValue(`AmbushAverageTime${ambushPath}`, newAmbushAverageTime); return newAmbushAverageTime; } function getAllAmbushStatistics() { const timeKeys = getStorageKeys(x => x.startsWith("AmbushAverageTime")); const result = []; for(const timeKey of timeKeys) { const ambushPath = timeKey.replace(/^AmbushAverageTime/, ""); const ambushAverageTime = getValue(timeKey); const ambushAmount = getValue(`AmbushAmount${ambushPath}`); result.push({ AmbushPath: ambushPath, AmbushAmount: ambushAmount, AmbushAverageTime: ambushAverageTime}); } //console.log(result) return result; } function showAmbushWaitingStatistics(container) { const allAmbushStatistics = getAllAmbushStatistics(); const ambushWaitingStatisticsTable = document.getElementById("ambushWaitingStatisticsTable") || addElement("table", { id: "ambushWaitingStatisticsTable" }, container); let tableHtml = ""; if(allAmbushStatistics.length > 0) { for(const item of allAmbushStatistics) { tableHtml += ` <tr><td id="${`${item.AmbushPath}`}" name="clearStatistics" title="${isEn ? "Clear statistics on path" : "Очистить статистику по направлению"}" onclick="">${item.AmbushPath}</td><td title="${isEn ? "Ambush amount" : "Количество засад" }">${item.AmbushAmount}</td><td title="${isEn ? "Average waiting time" : "Среднее время ожидания" }">${item.AmbushAverageTime}</td></tr>`; } } else { tableHtml = isEn ? "Statistics is empty" : "Статистика пуста"; } ambushWaitingStatisticsTable.innerHTML = tableHtml; Array.from(ambushWaitingStatisticsTable.querySelectorAll("[name='clearStatistics']")).forEach(x => x.addEventListener("click", function() { if(window.confirm(isEn ? "Is clear?" : "Очистить?")) { deleteValue(`AmbushAmount${x.id}`); deleteValue(`AmbushAverageTime${x.id}`); } })); //Array.from(ambushWaitingStatisticsTable.querySelectorAll("[name='clearStatistics']")).forEach(x => x.addEventListener("click", function() { if(window.confirm(isEn ? "Is clear?" : "Очистить?")) { clearAllAmbushStatistics(); } })); } function clearAllAmbushStatistics() { const timeKeys = getStorageKeys(x => x.startsWith("AmbushAverageTime")); for(const timeKey of timeKeys) { const ambushPath = timeKey.replace(/^AmbushAverageTime/, ""); deleteValue(timeKey); deleteValue(`AmbushAmount${ambushPath}`); } } function parseBattleResultPanel() { if(getPlayerValue("ChallengeState") == ChallengeState.Battle && finalResultDiv.innerHTML.length > 10) { deletePlayerValue("ChallengeState"); const bolds = finalResultDiv.querySelectorAll("font b"); let result = "fail"; for(const bold of bolds) { if(bold.innerHTML == (isEn ? "Victorious:" : "Победившая сторона:")) { if(bold.parentNode.nextSibling.nextSibling.firstChild.innerText == getPlayerValue("UserName")) { result = "win"; } break; } } if(result == "fail") { setPlayerValue("AmbushSuspendExpireDate", getServerTime() + AmbushMinutesInterval * 60 * 1000); } } } async function checkAmbushResult(force = false) { const afterAmbushUnknownResult = getPlayerValue("ChallengeState") == ChallengeState.Battle; // После засады был бой if(afterAmbushUnknownResult && !isFullHealth() && !force) { setPlayerValue("AmbushSuspendExpireDate", getServerTime() + AmbushMinutesInterval * 60 * 1000); } else { let doc; if(location.pathname == '/pl_warlog.php') { const page = getUrlParamValue(location.href, "page"); const id = getUrlParamValue(location.href, "id"); if(id == PlayerId && (!page || page == 0)) { doc = document; } } if(doc || afterAmbushUnknownResult || force) { doc = doc || await getRequest(`/pl_warlog.php?id=${PlayerId}`); processWarlog(doc); } } if(afterAmbushUnknownResult) { deletePlayerValue("ChallengeState"); } refreshAmbushMoratoriumPanel(); } function getHealth() { const health_amount = document.getElementById("health_amount"); let health; if(health_amount) { health = parseInt(health_amount.innerText); } else { health = win.heart; } return health; } function isFullHealth() { const healthRestoreTime = parseInt(getPlayerValue("HealthRestoreTime", 0)); return healthRestoreTime < getServerTime(); } function healthTimer() { if(isHeartOnPage) { const health_amount = document.getElementById("health_amount"); let heart; // 78 let maxHeart; // 100 let timeHeart; // 405 if(health_amount) { const res = /top_line_draw_canvas_heart\((\d+), (\d+), ([\d\.]+)\);/.exec(document.body.innerHTML); // top_line_draw_canvas_heart(0, 100, 405.5); if(res) { heart = parseInt(res[1]); maxHeart = parseInt(res[2]); timeHeart = parseFloat(res[3]); } } else { heart = win.heart; maxHeart = win.max_heart; timeHeart = win.time_heart; } //console.log(`healthTimer heart: ${heart}, maxHeart: ${maxHeart}, timeHeart: ${timeHeart}`); let restSeconds = timeHeart * (maxHeart - heart) / maxHeart; if(restSeconds > 0) { setPlayerValue("HealthRestoreTime", getServerTime() + restSeconds * 1000); } else { deletePlayerValue("HealthRestoreTime"); } } expireHealthRestoreTime(); } function expireHealthRestoreTime() { const healthRestoreTime = parseInt(getPlayerValue("HealthRestoreTime", 0)); if(healthRestoreTime > 0) { if(getServerTime() > healthRestoreTime) { deletePlayerValue("HealthRestoreTime"); drawAmbushMarks(); } else { setTimeout(expireHealthRestoreTime, 1000); } } } function processWarlog(doc) { const lastAmbushRef = Array.from(doc.querySelectorAll("a[href*='warlog.php?warid=']")).find(x => { if(x.nextSibling.textContent == ": • ") { return true; } const rowElements = getSequentialsUntil(x, "br"); //console.log(rowElements); const ranger = rowElements.find(y => y.textContent.includes(isEn ? "Ranger" : "Рейнджер")); //console.log(`ranger: ${ranger}`); return ranger ? true : false; }); let lastAmbushResult = AmbushResult.NotFound; if(lastAmbushRef) { let currentElement = lastAmbushRef; lastAmbushResult = AmbushResult.Fail; while(currentElement && currentElement.tagName.toLowerCase() != "br") { //console.log(currentElement); if(currentElement.tagName.toLowerCase() == "b" && currentElement.innerHTML.includes(getPlayerValue("UserName"))) { lastAmbushResult = AmbushResult.Win; break; } currentElement = nextSequentialElement(currentElement); } var lastAmbushTime = parseDate(lastAmbushRef.innerText).getTime(); var newAmbushSuspendExpireDate = lastAmbushTime + (AmbushMinutesInterval + 1) * 60 * 1000; } // Если есть неустаревшее поражение if(lastAmbushResult == AmbushResult.Fail && newAmbushSuspendExpireDate > getServerTime()) { // Если нет старого значения AmbushSuspendExpireDate или расхождение с новым из лога больше минуты, то установим новое if(!getPlayerValue("AmbushSuspendExpireDate") || Math.abs(parseInt(getPlayerValue("AmbushSuspendExpireDate")) - newAmbushSuspendExpireDate) > 60 * 1000) { setPlayerValue("AmbushSuspendExpireDate", newAmbushSuspendExpireDate); } } else { deletePlayerValue("AmbushSuspendExpireDate"); } } function fromTimeSpanToString(timeSpan) { if(timeSpan) { return (new Date(parseInt(timeSpan))).toLocaleString(); } } async function inBattle() { let result = false; const doc = await getRequest("/forum_thread.php?id=1"); if(isNewInterface) { result = doc.querySelector("div.mm_item_red > a[href='home.php']") ? true : false; } else { result = doc.querySelector("a[href='home.php'][style='text-decoration: none;color: #ff0000;']") ? true : false; } isInBattle = result; return result; } async function getCurrentBattle() { const doc = await getRequest("/pl_info.php?id=${PlayerId}"); return doc.querySelector("a[href^='warlog.php']")?.href; } async function requestServerTime() { if(parseInt(getValue("LastClientServerTimeDifferenceRequestDate", 0)) + 60 * 60 * 1000 < Date.now()) { setValue("LastClientServerTimeDifferenceRequestDate", Date.now()); const responseText = await getRequestText("/time.php"); const responseParcing = /now (\d+)/.exec(responseText); //responseText: now 1681711364 17-04-23 09:02 if(responseParcing) { setValue("ClientServerTimeDifference", Date.now() - parseInt(responseParcing[1]) * 1000); } } else { setTimeout(requestServerTime, 60 * 60 * 1000); } } function truncDateTimeToMinutes(dateTime) { const coeff = 1000 * 60; return new Date(Math.floor(dateTime.getTime() / coeff) * coeff); } function refreshAmbushMoratoriumPanel() { if(getPlayerValue("AmbushSuspendExpireDate")) { const ambushMoratoriumEndDate = parseInt(getPlayerValue("AmbushSuspendExpireDate")); if(ambushMoratoriumEndDate > getServerTime()) { if(!getPlayerValue("thiefTimeoutEnd")) { setPlayerValue("thiefTimeoutEnd", ambushMoratoriumEndDate); } const formatedTime = formatInterval(ambushMoratoriumEndDate - getServerTime()); setAmbushMoratoriumPanelValue(formatedTime); setTimeout(refreshAmbushMoratoriumPanel, 1000); } else { setAmbushMoratoriumPanelValue(); deletePlayerValue("AmbushSuspendExpireDate"); if(getPlayerBool("ThievesGuildNotifications")) { GM.notification("Вы можете устроить засаду", "ГВД", "https://dcdn.heroeswm.ru/i/portraits/thiefwarrioranip33.png", function() { window.focus(); getURL("/map.php"); }); } drawAmbushMarks(); } } else { setAmbushMoratoriumPanelValue(); drawAmbushMarks(); } } function setAmbushMoratoriumPanelValue(value) { if(AmbushMoratoriumPanel) { AmbushMoratoriumPanel.innerText = value || ""; } } function formatInterval(interval) { let diff = interval; const hours = Math.floor(diff / 1000 / 60 / 60); diff -= hours * 1000 * 60 * 60; const mimutes = Math.floor(diff / 1000 / 60); diff -= mimutes * 1000 * 60; const seconds = Math.floor(diff / 1000); const formatedTime = (hours > 0 ? hours + ":" : "") + ( (mimutes < 10) ? '0' : '' ) + mimutes + ':' + (seconds < 10 ? '0' : '') + seconds; return formatedTime; } function drawMap() { const insideMap = document.querySelector("#inside_map"); const jsmap = document.querySelector("#jsmap"); const isHideMap = getPlayerBool("HideMap"); const mapRightBlock = document.getElementById("map_right_block"); if(jsmap) { jsmap.style.display = isHideMap ? "none" : "block"; } toggleRightBlock(); toggleBattlesDayWith(); toggleHuntBlock(); if(insideMap && (!getPlayerBool("PopupAlways") || isMobileInterface)) { mapRightBlock.style.removeProperty('height'); let mapMovingContainer = insideMap; let mapMovingStyle = ""; if(isMobileInterface) { insideMap.style.flexWrap = "wrap"; //insideMap.style.justifyContent = "normal"; addElement("div", { style: "flex-basis: 100%; height: 0;"}, insideMap); mapMovingStyle = "order: 4;"; } else { mapMovingContainer = addElement("div", { style: "order: 1;" }, insideMap); mapMovingContainer.appendChild(jsmap); insideMap.style.gap = "10px"; } drawMapCore(mapMovingContainer, mapMovingStyle); } else { const mapHeight = 350; const mapWidth = 500; const zIndex = isMobileDevice ? 110 : 10; const mapActivatorDiv = addElement("div", { style: `position: fixed; width: ${isMobileDevice ? 20 : 5}px; height: ${isMobileDevice ? 90 : window.innerHeight - 50}px; top: 50px; left: 0; background-color: #FFCCCB; opacity: 80%; z-index: ${zIndex + 1};` }, document.body); const mapContainerDiv = addElement("div", { style: `position: fixed; width: ${mapWidth}px; height: ${mapHeight}px; top: 40%; left: 0; transform: translate(-100%, -50%); transition-duration: 0.3s; z-index: ${zIndex};` }, document.body, "afterbegin"); const mapMoving = drawMapCore(mapContainerDiv, "box-shadow: 1px 1px 5px #333;"); //, `height: ${innerMapHeight}px` let timer; if(isMobileDevice) { mapActivatorDiv.addEventListener("click", function(event) { //console.log(`left: ${mapContainerDiv.style.left}, top: ${mapContainerDiv.style.top}`) if(mapContainerDiv.getBoundingClientRect().left < 0) { mapContainerDiv.style.transform = "translate(0%, -50%)"; } else { mapContainerDiv.style.transform = "translate(-100%, -50%)"; } }); } else { mapActivatorDiv.addEventListener("mouseover", function(event) { timer = setTimeout(function() { mapContainerDiv.style.transform = "translate(0%, -50%)"; }, 100); }); mapActivatorDiv.addEventListener("mouseout", function(event) { timer = setTimeout(function() { mapContainerDiv.style.transform = "translate(-100%, -50%)"; }, 300); }); mapContainerDiv.addEventListener("mouseover", function(event) { clearTimeout(timer); }); mapContainerDiv.addEventListener("mouseout", function(event) { timer = setTimeout(function() { mapContainerDiv.style.transform = "translate(-100%, -50%)"; }, 100); }); } } } function drawMapCore(parentNode, mainStyle) { let mapMoving = document.getElementById("mapMoving"); if(mapMoving) { mapMoving.parentNode.removeChild(mapMoving); } const workImageSize = 25; mapMoving = addElement("div", { id: "mapMoving", style: (mainStyle || "") + "border: 1mm ridge rgba(211, 220, 50, .6); background: linear-gradient(0.15turn, #3f87a6, #ebf8e1, #f69d3c 80%);" }, parentNode); mapMoving.innerHTML = ` <div id="hwmTransporterTopSettings" style="position: relative; text-align: center; font-size: 9px;"> <span style="font-size: 9px;">${ (MountInfo.EndDate > 0 ? `${MountInfo.ComplexRoute ? ComplexRouteName + ", " : ""}${MountInfo.IsPremiunAccount ? PremiunAccountName + ", " : ""}${MountUntilName}: ${(new Date(MountInfo.EndDate)).toLocaleString()}, ` : "") + `${TravelingTimeName}: ${MountInfo.TravelingTime} ${Second}` }</span> ${ MountInfo.AntithiefControlled ? `, <input id='toggleAntithiefCheckbox' type='checkbox' ${MountInfo.Antithief ? "checked" : ""}>${Antithief}` : "" } <div id="checkMountButton" title="${CheckMountName}" style="display: inline-block;"> <img src="https://dcdn2.heroeswm.ru/i/pl_info/btn_reset.png" style="height: 12px; wigth: 12px;"> </div> <br /> <input id='removeArtsBeforeMovingCheckbox' type='checkbox' ${getPlayerBool("removeArtsBeforeMoving") ? "checked" : ""}>${UndressBeforeMove} <input id='IgnoreComplexRouteCheckbox' type='checkbox' ${getPlayerBool("IgnoreComplexRoute") ? "checked" : ""}>${NotUseComplexRoute} <input id='HideMapCheckbox' type='checkbox' ${getPlayerBool("HideMap") ? "checked" : ""}>${HideMapName} <br /> <input id='HideHuntBlockCheckbox' type='checkbox' ${getPlayerBool("HideHuntBlock") ? "checked" : ""}>${HideHuntName} <input id='HideRightBlockCheckbox' type='checkbox' ${getPlayerBool("HideRightBlock") ? "checked" : ""}>${HideRightBlockName} <input id='popupAlwaysCheckbox' type='checkbox' ${getPlayerBool("PopupAlways") ? "checked" : ""}>${PopupAlwaysName} <input id='showLocationNumbersCheckbox' type='checkbox' ${getPlayerBool("ShowLocationNumbers") ? "checked" : ""}>${ShowLocationNumbersName} </div> <div> <table id="hwmMapMoveTable" style=""> </table> </div> <div id="hwmTransporterBottomSettings" style="position: relative; text-align: left; font-size: 9px;"> <!--<div id="findNearestProductionBigButton" title="${FindNearestProductionName}" style="float: right;"> <img src="https://dcdn.heroeswm.ru/i/btns/job_fl_btn_production.png" style="scale: 1.5;"> </div>--> ${LoadName}<input id='loadMiningCheckbox' type='checkbox' ${getPlayerBool("Load" + factoryTypes[0]) ? "checked" : ""}>${Mining} <input id='loadMachiningCheckbox' type='checkbox' ${getPlayerBool("Load" + factoryTypes[1]) ? "checked" : ""}>${Machining} <input id='loadProductionCheckbox' type='checkbox' ${getPlayerBool("Load" + factoryTypes[2]) ? "checked" : ""}>${Production} <input id='showLocationViewImagesCheckbox' type='checkbox' ${getPlayerBool("showLocationViewImages") ? "checked" : ""}>${isEn ? "View" : "Просмотр"} <input id='HideBattlesDayWithBlockCheckbox' type='checkbox' ${getPlayerBool("HideBattlesDayWithBlock") ? "checked" : ""}>${HideBattlesDayWithName} <br/> <span style="font-size: 9px;">${Legend}, PlayerSector: ${playerLocationNumber}${playerLocationNumber != getPlayerValue("LastObservingLocationNumber") ? ", View: " + getPlayerValue("LastObservingLocationNumber") : ""}</span> <!--<span id="ClickMe">ClickMe</span>--> <br/> ${isEn ? "Min salary: " : "Минимальная зарплата: "}<input id='minSalaryNumber' type='number' value='${getPlayerValue("MinSalary", "")}' style="width: 50px;" onfocus="this.select();"> <input id='thievesGuildNotificationsCheckbox' type='checkbox' ${getPlayerBool("ThievesGuildNotifications") ? "checked" : ""}>${isEn ? "Thieves guild notifications" : "Оповещение о засаде"} <input id='newAmbushTimerCheckbox' type='checkbox' ${getPlayerBool("NewAmbushTimer") ? "checked" : ""}><span id="showAmbushWaitingStatisticsSpan" title="${isEn ? "Show ambush waiting statistics" : "Показать статистику ожидания в засаде"}">${isEn ? "New ambush timer" : "Новый таймер засад"}</span> <br/> <input id='patrolRouteEnabledCheckbox' type='checkbox' ${getPlayerBool("PatrolRouteEnabled") ? "checked" : ""}>${isEn ? "Setup patrol" : "Настроить патрулирование"} ${isEn ? "Pause: " : "Пауза: "}<input id='patrollingPauseInput' type='number' value='${getPlayerValue("PatrollingPause", 5)}' style="width: 50px;" onfocus="this.select();"> <input id=togglePatrollingButton type="button" value="${getPlayerBool("IsPatrolling") ? (isEn ? "Stop patrolling" : "Остановить патрулирование") : (isEn ? "Start patrolling" : "Начать патрулирование")}" title="${JSON.parse(getPlayerValue("PatrolRoute", "[]")).join("->")}" style="font-size: 9px;" /> <br /> <b>${JobFinding}:</b> <div id="findNearestMiningButton" title="${FindNearestMiningName}" style="display: inline-block;"> <img src="https://dcdn.heroeswm.ru/i/btns/job_fl_btn_mining.png" style="width: ${workImageSize}px; height: ${workImageSize}px; vertical-align: middle;"> </div> <div id="findNearestManufactureButton" title="${FindNearestManufactureName}" style="display: inline-block;"> <img src="https://dcdn.heroeswm.ru/i/btns/job_fl_btn_manufacture.png" style="width: ${workImageSize}px; height: ${workImageSize}px; vertical-align: middle;"> </div> <div id="findNearestProductionButton" title="${FindNearestProductionName}" style="display: inline-block;"> <img src="https://dcdn.heroeswm.ru/i/btns/job_fl_btn_production.png" style="width: ${workImageSize}px; height: ${workImageSize}px; vertical-align: middle;"> </div> <input id='moveToWorkAfterFindCheckbox' type='checkbox' ${getPlayerBool("MoveToWorkAfterFind") ? "checked" : ""} style="vertical-align: middle;"><lable for=moveToWorkAfterFindCheckbox style="vertical-align: middle;">${MoveToWorkAfterFindName}</lable> <span id="workFindingMessageSpan"/> </div>`; mapMoving.querySelector("#removeArtsBeforeMovingCheckbox").addEventListener("click", function() { setPlayerValue("removeArtsBeforeMoving", this.checked); }); mapMoving.querySelector("#IgnoreComplexRouteCheckbox").addEventListener("click", function() { setPlayerValue("IgnoreComplexRoute", this.checked); }); mapMoving.querySelector("#HideMapCheckbox").addEventListener("click", function() { setPlayerValue("HideMap", this.checked); window.location.reload(); }); if(MountInfo.AntithiefControlled) { mapMoving.querySelector("#toggleAntithiefCheckbox").addEventListener("click", function() { toggleAntithief(this); }); } mapMoving.querySelector("#HideHuntBlockCheckbox").addEventListener("click", function() { setPlayerValue("HideHuntBlock", this.checked); toggleHuntBlock(); }); mapMoving.querySelector("#HideRightBlockCheckbox").addEventListener("click", function() { setPlayerValue("HideRightBlock", this.checked); toggleRightBlock(); }); mapMoving.querySelector("#checkMountButton").addEventListener("click", function() { main(true); }); mapMoving.querySelector("#loadMiningCheckbox").addEventListener("click", function() { setPlayerValue("Load" + factoryTypes[0], this.checked); }); mapMoving.querySelector("#loadMachiningCheckbox").addEventListener("click", function() { setPlayerValue("Load" + factoryTypes[1], this.checked); }); mapMoving.querySelector("#loadProductionCheckbox").addEventListener("click", function() { setPlayerValue("Load" + factoryTypes[2], this.checked); }); mapMoving.querySelector("#popupAlwaysCheckbox").addEventListener("click", function() { setPlayerValue("PopupAlways", this.checked); window.location.reload(); }); mapMoving.querySelector("#showLocationNumbersCheckbox").addEventListener("click", function() { setPlayerValue("ShowLocationNumbers", this.checked); window.location.reload(); }); mapMoving.querySelector("#showLocationViewImagesCheckbox").addEventListener("click", function() { setPlayerValue("showLocationViewImages", this.checked); toggleViewImages(); }); mapMoving.querySelector("#HideBattlesDayWithBlockCheckbox").addEventListener("click", function() { setPlayerValue("HideBattlesDayWithBlock", this.checked); toggleBattlesDayWith(); }); //mapMoving.querySelector("#ClickMe").addEventListener("click", function() { getObjectLocations(); }); mapMoving.querySelector("#moveToWorkAfterFindCheckbox").addEventListener("click", function() { setPlayerValue("MoveToWorkAfterFind", this.checked); }); mapMoving.querySelector("#minSalaryNumber").addEventListener("change", function() { setPlayerValue("MinSalary", parseInt(this.value)); }); mapMoving.querySelector("#findNearestMiningButton").addEventListener("click", function() { findNearestWork("mn"); }); mapMoving.querySelector("#findNearestManufactureButton").addEventListener("click", function() { findNearestWork("fc"); }); mapMoving.querySelector("#findNearestProductionButton").addEventListener("click", function() { findNearestWork("sh"); }); //mapMoving.querySelector("#findNearestProductionBigButton").addEventListener("click", function() { findNearestWork("sh"); }); mapMoving.querySelector("#thievesGuildNotificationsCheckbox").addEventListener("click", function() { setPlayerValue("ThievesGuildNotifications", this.checked); }); mapMoving.querySelector("#patrolRouteEnabledCheckbox").addEventListener("click", function() { setPlayerValue("PatrolRouteEnabled", this.checked); togglePatrolRoute(); }); mapMoving.querySelector("#patrollingPauseInput").addEventListener("change", function() { setPlayerValue("PatrollingPause", parseInt(this.value)); }); mapMoving.querySelector("#togglePatrollingButton").addEventListener("click", togglePatrolling); mapMoving.querySelector("#newAmbushTimerCheckbox").addEventListener("click", function() { setPlayerValue("NewAmbushTimer", this.checked); }); mapMoving.querySelector("#showAmbushWaitingStatisticsSpan").addEventListener("click", function() { showAmbushWaitingStatistics(mapMoving); }); const table = mapMoving.querySelector("#hwmMapMoveTable"); const now = new Date(); for(let y = minY; y <= maxY; y++) { const tableRow = addElement("tr", {}, table); for(let x = minX; x <= maxX; x++) { let cellStyle = ""; let locationNumber = getLocationNumberByCoordinate(x, y); const tableCell = addElement("td", { id: `transporterLocation${locationNumber}` }, tableRow); if(locationNumber) { const loc = locations[locationNumber]; tableCell.innerHTML = `${loc[locationNamesLocalizedColumn].replace(" ", "<br>")} ${ getPlayerBool("ShowLocationNumbers") ? `(${locationNumber})` : "" }`; tableCell.style = `position: relative; cursor: pointer; font-size: 10px; font-weight: bold; text-align: center; border: 1px solid #000; padding: 3px; color: ${locationNumber == playerLocationNumber ? "red" : "black"};`; if(locationNumber < 100) { let routeTime = ""; const routeSettings = routes[playerLocationNumber][locationNumber]; if(routeSettings) { const routeLength = routeSettings[1]; routeTime = `\n${TravelingTimeName} ${routeLength * MountInfo.TravelingTime} ${Second}`; } tableCell.title = `${GoTo} ${loc[locationNamesLocalizedColumn]}${routeTime}`; tableCell.addEventListener("mouseenter", function() { this.style.border = "1px solid #fff"; }); tableCell.addEventListener("mouseleave", function() { this.style.border = "1px solid #000"; }); tableCell.addEventListener("click", function(e) { if(e.altKey) { getURL(`/map.php?cx=${x}&cy=${y}`); } else if(e.ctrlKey) { setThiefAmbush(locationNumber); } else { tryMoving(locationNumber); } }); tableCell.addEventListener("auxclick", function(e) { e.stopPropagation();/*if(e.which === 2) { e.preventDefault(); }*/ if(e.button == 1) { setThiefAmbush(locationNumber); } }); // console.log(`e.which: ${e.which}, e.button: ${e.button}`); const sectorViewReference = addElement("a", { href: `/map.php?cx=${x}&cy=${y}`, style: `display: ${getPlayerBool("showLocationViewImages") ? "block" : "none"}; float: left;`, title: `${isEn ? "View" : "Просмотр"}`, name: "locationViewImage" }, tableCell); addElement("img", { style: "border: 0;", src: "data:image/gif,GIF89a%10%00%10%00%D5%00%00B%40B%15%15%18((-PPUzz%7FHHJ%5D%5D_--.zz%7BWWX23%3BEHS%80%88%A2rx%8C%DA%DD%E7bj%80sx%87%88%8C%97%7D%85%98HO_%7D%85%97%88%8C%95psz%DC%DD%DF%D4%D5%D7x%7F%8C%2B-0%C2%DF%FF%DF%F4%FFMOPmop%EF%F1%F2%E7%F9%FF%EA%FD%FF%E7%FF%FF%13%15%15%E9%FF%FF%18%1A%1A%FA%FF%FFmoo%FC%FF%FF%FD%FF%FFZ%5B%5B%5D%5D%5B%40%3F%3F%FF%FF%FF%16%16%16%10%10%10%05%05%05%00%00%00%FF%FF%FF%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00%00!%F9%04%01%00%002%00%2C%00%00%00%00%10%00%10%00%00%06%5C%40%99pH%2C%12%13%01%8D%80e%24%02ZP%A8%A2)k%C8%A2%D1As%01Ma'%CDF%D4%14%7D4!-%94%A8%95%BAJ%9A%07P%8B%E4mm%2C%D4H(%CA%A9%AC%A82%05%19%0C%14%041*%17%80E0%1D%18%8AD%2F%06%89%8FB%25'%1F%94B.%08-%992%23%1E%9ECA%00%3B" }, sectorViewReference); sectorViewReference.addEventListener("click", function(e) { e.stopPropagation(); }) if([2, 6, 16, 21].includes(parseInt(locationNumber))) { getMoveToMapObjectReference('/mercenary_guild.php', tableCell, locationNumber, "https://dcdn.heroeswm.ru/i/btns/job_fl_btn_mercenary.png", "position: absolute; bottom: 2px; right: 2px; display: block; width: 16px; height: 16px;", 14); } const guestInfo = getGuestInfo(locationNumber); const guestInfoKeys = Object.keys(guestInfo); if(guestInfoKeys.length > 0) { const guestInfoTitles = guestInfoKeys.map(x => `${guestInfo[x].HostInfo} до ${(new Date(guestInfo[x].ExpireDate)).toLocaleString()}`); const guestReference = addElement("a", { href: `/house_info.php?id=${guestInfoKeys[0]}`, style: "display: block; float: left;", title: guestInfoTitles.join("\n") }, tableCell); guestReference.addEventListener("click", function(e) { e.stopPropagation(); }) addElement("img", { style: "border: 0;", src: "https://dcdn.heroeswm.ru/i/btns/job_fl_btn_houses.png", style: "width: 16px; height: 16px;" }, guestReference); } } else if(locationNumber == 106) { tableCell.innerHTML = "<div id=repeatLastAmbushHolder></div>"; tableCell.style = "cursor: pointer; text-align: center;"; } else if(locationNumber == 107) { tableCell.innerHTML = `<div title="${FindNearestProductionName}"> <img src="https://dcdn.heroeswm.ru/i/btns/job_fl_btn_production.png" style="height: 48px;"> </div>`; tableCell.style = "cursor: pointer; text-align: center;"; tableCell.addEventListener("click", function() { findNearestWork("sh"); }); } else { //console.log(`locationNumber: ${locationNumber}, x: ${x}, y: ${y}`); if(loc[5]) { tableCell.style.color = loc[5].Color; tableCell.addEventListener("click", function() { getURL(loc[5].Url); }); } if(x == 50 && y == 54) { addElement("br", {}, tableCell); AmbushMoratoriumPanel = addElement("span", { id: "AmbushMoratoriumPanel" }, tableCell); } if(x == 51 && y == 54) { addElement("br", {}, tableCell); addElement("span", { id: "SmithMoratoriumPanel" }, tableCell); } if(x == 53 && y == 48) { tableCell.innerHTML = ""; const spoiler = addElement("div", { id: `hwmTransporterTopSettingsSpoiler`, title: loc[locationNamesLocalizedColumn], style: "display: inline-block; cursor: pointer;", innerHTML: `<img src="https://dcdn.heroeswm.ru/i/inv_im/btn_expand.svg" style="vertical-align: middle;">` }, tableCell); spoiler.addEventListener("click", function() { setPlayerValue(this.id, !getPlayerBool(this.id)); bindPlayerInfoSpolers(this.id); }); } if(x == 53 && y == 54) { tableCell.innerHTML = ""; const spoiler = addElement("div", { id: `hwmTransporterBottomSettingsSpoiler`, title: loc[locationNamesLocalizedColumn], style: "display: inline-block; cursor: pointer;", innerHTML: `<img src="https://dcdn.heroeswm.ru/i/inv_im/btn_expand.svg" style="vertical-align: middle;">` }, tableCell); spoiler.addEventListener("click", function() { setPlayerValue(this.id, !getPlayerBool(this.id)); bindPlayerInfoSpolers(this.id); }); } } } } } processMercenaryTaskLocation(); createPatrolRoute(); bindPlayerInfoSpolers(); return mapMoving; } function bindPlayerInfoSpolers(togglingSpoilerId) { const spoilerIds = togglingSpoilerId ? [togglingSpoilerId] : ["hwmTransporterTopSettingsSpoiler", "hwmTransporterBottomSettingsSpoiler"]; for(const spoilerId of spoilerIds) { const spoiler = document.getElementById(spoilerId); if(spoiler) { const spoiled = getPlayerBool(spoilerId); spoiler.querySelector("img").style.transform = spoiled ? 'rotate(0deg)' : 'rotate(90deg)'; const panel = document.getElementById(spoilerId.replace("Spoiler", "")); panel.style.display = spoiled ? "none" : ""; } } } function drawAmbushMarks() { for(const locationNumber of Object.keys(locations).filter(x => parseInt(x) < 100)) { const routeSettings = routes[playerLocationNumber][locationNumber]; const tableCell = document.getElementById(`transporterLocation${locationNumber}`); const thiefwarrioranip33 = tableCell.querySelector("img[src*='thiefwarrioranip33']"); if(routeSettings && routeSettings[1] < 2 && !getPlayerValue("AmbushSuspendExpireDate") && !getPlayerValue("ChallengeState") && isFullHealth()) { if(!thiefwarrioranip33) { const ambushImage = addElement("img", { src: "https://dcdn.heroeswm.ru/i/portraits/thiefwarrioranip33.png", style: "width: 16px; height: 16px;", title: `${isEn ? "Ambush" : "Засада"}` }, tableCell); ambushImage.addEventListener("click", function(e) { e.stopPropagation(); setThiefAmbush(locationNumber); }); } } else { if(thiefwarrioranip33) { thiefwarrioranip33.remove(); } } } const repeatLastAmbushHolder = document.getElementById("repeatLastAmbushHolder"); const thiefwarrioranip33 = repeatLastAmbushHolder.querySelector("img[src*='thiefwarrioranip33']"); if(!getPlayerValue("AmbushSuspendExpireDate") && !getPlayerValue("ChallengeState") && isFullHealth()) { const lastAmbushDirections = JSON.parse(getPlayerValue("lastAmbushDirections", "[]")); const lastAmbushDirection = lastAmbushDirections.find(x => x.thiefLocation == playerLocationNumber); if(!thiefwarrioranip33 && lastAmbushDirection) { repeatLastAmbushHolder.addEventListener("click", setThiefAmbush); const ambushImage = addElement("img", { src: "https://dcdn.heroeswm.ru/i/portraits/thiefwarrioranip33.png", style: "height: 48px;", title: isEn ? "Repeat last ambush" : "Повторить последнюю засаду" }, repeatLastAmbushHolder); } } else { if(thiefwarrioranip33) { repeatLastAmbushHolder.removeEventListener("click", setThiefAmbush); thiefwarrioranip33.remove(); } } } async function processMercenaryTaskLocation() { const mercenaryTaskLocation = await getMercenaryTaskLocation(); if(mercenaryTaskLocation) { addElement("img", { src: "https://dcdn2.heroeswm.ru/i/cssmap/map_sectors_naim.png", style: "width: 16px; height: 16px;", title: `${isEn ? "Mercenary guild task" : "Задание гильдии наемников"}` }, document.getElementById(`transporterLocation${mercenaryTaskLocation}`)); } } function getMercenaryTaskLocation() { return new Promise((resolve, reject) => { if(location.pathname == '/map.php') { setTimeout(function() { if(win.naim_type == 'fight') { setPlayerValue("MercenaryTaskLocation", win.naim_sector); } else { deletePlayerValue("MercenaryTaskLocation"); } resolve(getPlayerValue("MercenaryTaskLocation")); }, 300); } else { resolve(getPlayerValue("MercenaryTaskLocation")); } }); } async function findNearestWork(factoryType) { const movingEnabled = await varifyMovingEnabled(); let notEmptyLocationsSortedByDistance = [playerLocationNumber]; if(movingEnabled) { notEmptyLocationsSortedByDistance = getLocationsSortedByDistance(); } notEmptyLocationsSortedByDistance = notEmptyLocationsSortedByDistance.filter(x => !getValue(`LastDetectedEmptyFactoriesDate_${factoryType}${x}`)); const workFindingMessageSpan = document.querySelector("span#workFindingMessageSpan"); for(const locationNumber of notEmptyLocationsSortedByDistance) { workFindingMessageSpan.innerText = `Viewing ${locationNumber}...`; var findedWorkObject = await findWorkInLocation(factoryType, locationNumber); if(findedWorkObject) { if(getPlayerBool("MoveToWorkAfterFind")) { tryMoving(locationNumber, findedWorkObject.href); } else { getURL(findedWorkObject.href); } workFindingMessageSpan.innerText = ""; break; } } if(!findedWorkObject) { workFindingMessageSpan.innerText = "Vacancy not found"; } } async function findWorkInLocation(factoryType, locationNumber) { const x = locations[locationNumber][0]; const y = locations[locationNumber][1]; const doc = await getRequest(`/map.php?cx=${x}&cy=${y}&st=${factoryType}`); const objectRefs = doc.querySelectorAll("a[href^='object-info.php?id=']"); //console.log(`objectRefs: ${objectRefs.length}`); for(const objectRef of objectRefs) { //console.log(`objectRef.innerText: ${objectRef.innerText}, ${objectRef.innerText.includes("\u00bb\u00bb\u00bb")}`); if(objectRef.innerText.includes("\u00bb\u00bb\u00bb") && !objectRef.querySelector("font")) { const salary = parseInt(objectRef.parentNode.previousElementSibling.querySelector("b").innerHTML); const minSalary = parseInt(getPlayerValue("MinSalary", 0)); //console.log(`minSalary: ${minSalary}, salary: ${salary}, objectRef: ${objectRef}`); if(minSalary == 0 || salary >= minSalary) { return objectRef; } } } } function getRequest(url) { return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: "GET", url: url, overrideMimeType: "text/html; charset=windows-1251", onload: function(response) { resolve((new DOMParser).parseFromString(response.responseText, "text/html")); }, onerror: function(error) { reject(error); } }); }); } function getRequestText(url, overrideMimeType = "text/html; charset=windows-1251") { return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: "GET", url: url, overrideMimeType: overrideMimeType, onload: function(response) { resolve(response.responseText); }, onerror: function(error) { reject(error); } }); }); } function postRequest(url, data) { return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: "POST", url: url, headers: { "Content-Type": "application/x-www-form-urlencoded" }, data: data, onload: function(response) { resolve(response); }, onerror: function(error) { reject(error); } }); }); } function getLocationsSortedByDistance(locationNumber = playerLocationNumber) { if(locationNumber in routes && routes[locationNumber]) { const playerLocationRoutes = routes[locationNumber]; const availableLocationsDistance = { }; availableLocationsDistance[locationNumber] = 0; for(let i = 0; i < playerLocationRoutes.length; i++) { if(playerLocationRoutes[i]) { availableLocationsDistance[i] = playerLocationRoutes[i][1]; } } const locationsSortedByDistance = Object.keys(availableLocationsDistance).sort(function(a,b) { return availableLocationsDistance[a] - availableLocationsDistance[b]; }); //console.log(locationsSortedByDistance); return locationsSortedByDistance; } } function getLocationNumberByCoordinate(x, y) { for(let locationNumber in locations) { if(locations[locationNumber][0] == x && locations[locationNumber][1] == y) { return locationNumber; } } } function processMoving() { let targetLocationNumber = parseInt(getPlayerValue("TargetLocationNumber")); if(getPlayerValue("LastMoveTryFrom") == playerLocationNumber && targetLocationNumber != playerLocationNumber) { stopMoving(); // Мы не смогли уйти из локации, поэтому останавливаем движение; и мы не движемся на объект в этой же локации return; } let arrived = false; if(targetLocationNumber) { if(targetLocationNumber == playerLocationNumber) { deletePlayerValue("TargetLocationNumber"); deletePlayerValue("LastMoveTryFrom"); deletePlayerValue("LastMoveTryTo"); arrived = true; } else { goToNextRoutePoint(); } } // Перемещение к объекту после прибытия if(arrived) { let enterObjectOnArrived = getPlayerValue("EnterObjectOnArrived"); if(enterObjectOnArrived) { deletePlayerValue("EnterObjectOnArrived"); getURL(enterObjectOnArrived); //setTimeout(function() { window.location = enterObjectOnArrived; }, 300); } } } function tryMoving(targetLocationNumber, enterObjectOnArrived) { const isMoovingToAdjacentLocation = getIsMoovingToAdjacentLocation(targetLocationNumber); //console.log(`targetLocationNumber: ${targetLocationNumber}, enterObjectOnArrived: ${enterObjectOnArrived}`); //alert(`isMoovingToAdjacentLocation: ${isMoovingToAdjacentLocation}, MountInfo.ComplexRoute: ${MountInfo.ComplexRoute}`); if(!MountInfo.ComplexRoute && !isMoovingToAdjacentLocation) { alert(isEn ? "Without a complex route, you can only move to a neighboring location" : "Без сложного маршрута можно перемещаться только в соседнюю локацию"); return; } if(targetLocationNumber == playerLocationNumber && !enterObjectOnArrived) { alert(YouAreHere); return; } if(targetLocationNumber == 25) { alert(How); return; } checkMovingEnabledAndGo(targetLocationNumber, enterObjectOnArrived); } function getIsMoovingToAdjacentLocation(targetLocationNumber) { return targetLocationNumber == playerLocationNumber || (routes[playerLocationNumber] && routes[playerLocationNumber][targetLocationNumber] ? (routes[playerLocationNumber][targetLocationNumber][1] < 2) : false); } async function varifyMovingEnabled() { let movingEnabled = true; if(/map.php/.test(location.href)) { const mapNavigator = document.querySelector("#map_navigator"); movingEnabled = (Array.from(mapNavigator.querySelectorAll("div[id*='dbut']")).find(x => x.style.display == "block")) ? true : false; } else { const doc = await getRequest("/one_to_one.php"); const invNote = Array.from(doc.querySelectorAll("font[color='#E65054']")).find(x => x.innerText == (isEn ? "You are already in a challenge!" : "Вы уже в заявке!")); movingEnabled = !invNote ? true : false; } return movingEnabled; } async function checkMovingEnabledAndGo(targetLocationNumber, enterObjectOnArrived) { let movingEnabled = true; if(targetLocationNumber != playerLocationNumber) { movingEnabled = await varifyMovingEnabled(); } if(movingEnabled) { undressAndGo(targetLocationNumber, enterObjectOnArrived); } else { alert(YouAreInAChallenge); } } async function undressAndGo(targetLocationNumber, enterObjectOnArrived) { if(getPlayerBool("removeArtsBeforeMoving") && targetLocationNumber != playerLocationNumber) { await getRequest("/inventory.php?all_off=100"); } startMoving(targetLocationNumber, enterObjectOnArrived); } function startMoving(targetLocationNumber, enterObjectOnArrived) { setPlayerValue("TargetLocationNumber", targetLocationNumber); if(enterObjectOnArrived) { setPlayerValue("EnterObjectOnArrived", enterObjectOnArrived); } goToNextRoutePoint(); } function goToNextRoutePoint() { const targetLocationNumber = parseInt(getPlayerValue("TargetLocationNumber")); let nextRoutePoint; if(MountInfo.ComplexRoute && !getPlayerBool("IgnoreComplexRoute") || targetLocationNumber == playerLocationNumber) { nextRoutePoint = targetLocationNumber; } else if(playerLocationNumber in routes && targetLocationNumber in routes[playerLocationNumber] && routes[playerLocationNumber][targetLocationNumber]) { const route = routes[playerLocationNumber][targetLocationNumber][0]; nextRoutePoint = route.split('-')[1]; } if(nextRoutePoint) { setPlayerValue("LastMoveTryFrom", playerLocationNumber); setPlayerValue("LastMoveTryTo", nextRoutePoint); //getURL(`/move_sector.php?id=${nextRoutePoint}&rand=${Math.random()}`); if(playerLocationNumber != nextRoutePoint) { getURL(`/move_sector.php?id=${nextRoutePoint}&rand=${Math.random()}`); } else if(getPlayerValue("EnterObjectOnArrived")) { processMoving(); } } } function stopMoving() { deletePlayerValue("TargetLocationNumber"); deletePlayerValue("EnterObjectOnArrived"); deletePlayerValue("LastMoveTryFrom"); deletePlayerValue("LastMoveTryTo"); } function getPlayerLocationNumber() { if(location.pathname == '/map.php' && location.search == '') { // Если мы на карте без параметров, т.е. на локации, где сами находимся. Если мы не в пути, тогда видим предприятия. Мы можем обновить текущее положение игрока. const minesRef = document.querySelector("a[href*='map.php?cx='][href*='&cy='][href*='&st=mn']"); // Берем из ссылки на заголовке шахт данной локации if(minesRef) { const locationNumber = getLocationNumberFromMapUrlByXy(minesRef.href); if(locationNumber) { setPlayerValue("PlayerLocationNumber", locationNumber); return locationNumber; } } } return parseInt(getPlayerValue("PlayerLocationNumber")); // Иначе, возьмем из кеша. Там будет пусто только при первом запуске скрипта, когда мы не просматриваем карту. } function getLocationNumberFromMapUrlByXy(href) { const x = getUrlParamValue(href, "cx"); const y = getUrlParamValue(href, "cy"); for(let locationNumber in locations) { if(x == locations[locationNumber][0] && y == locations[locationNumber][1]) { return locationNumber; } } } function getMoveToObjectReference(objectUrl, parentNode) { const objectId = getUrlParamValue(objectUrl, "id"); const objectLocationNumber = objectLocations[objectId]; //console.log(`objectId: ${objectId}, objectLocationNumber: ${objectLocationNumber}, parentNode: ${parentNode}`); return getMoveToMapObjectReference(objectUrl, parentNode, objectLocationNumber); } function getMoveToMapObjectReference(objectUrl, parentNode, objectLocationNumber, image, imageStyle, size = 12) { //if(objectLocationNumber==16)console.log(`objectLocationNumber: ${objectLocationNumber}, playerLocationNumber: ${playerLocationNumber}, image: ${image}, objectUrl: ${objectUrl}, parentNode: ${parentNode}`); if(parentNode.parentNode.querySelector(`span[name=moveToMapObjectReference][targetLocationNumber='${objectLocationNumber}']`)) { return null; } if(objectLocationNumber != playerLocationNumber || image) { let routeTime = ""; const routeSettings = routes[playerLocationNumber][objectLocationNumber]; if(routeSettings) { const routeLength = routeSettings[1]; routeTime = `\n${TravelingTimeName} ${routeLength * MountInfo.TravelingTime} ${Second}`; } const moveHereReference = addElement('span', { name: "moveToMapObjectReference", targetLocationNumber: objectLocationNumber, style: "cursor: pointer;", innerHTML: `<img src="${image || getHorseImageData()}" alt="${MoveHere}" title="${MoveHere + routeTime}" align="absmiddle" style="max-height: ${size}px; max-width: ${size}px; display: inline-block;${imageStyle || ""}"> ` }); moveHereReference.onclick = function(e) { tryMoving(objectLocationNumber, objectUrl); e.stopPropagation(); }; //console.log(`objectUrl: ${objectUrl}, moveHereReference: ${moveHereReference}, routeTime: ${routeTime}, parentNode: ${parentNode}, parentNode.nextSibling: ${parentNode.nextSibling}, parentNode.nextElementSibling: ${parentNode.nextElementSibling}, parentNode.parentNode: ${parentNode.parentNode}, parentNode.previousSibling: ${parentNode.previousSibling}`); if(image) { parentNode.appendChild(moveHereReference); } else { parentNode.insertAdjacentElement("afterend", moveHereReference); } return moveHereReference; } } async function setThiefAmbush(locationNumber) { if(!locationNumber || !Number(locationNumber)) { const lastAmbushDirections = JSON.parse(getPlayerValue("lastAmbushDirections", "[]")); const lastAmbushDirection = lastAmbushDirections.find(x => x.thiefLocation == playerLocationNumber); //console.log(lastAmbushDirection) locationNumber = lastAmbushDirection.ambushDirection; } if(checkSectorForAmbush(locationNumber)) { if(getPlayerValue("AmbushSuspendExpireDate")) { alert(`${ItIsTooEarly}`); } else { const lastAmbushDirections = JSON.parse(getPlayerValue("lastAmbushDirections", "[]")); let lastAmbushDirection = lastAmbushDirections.find(x => x.thiefLocation == playerLocationNumber); if(lastAmbushDirection) { lastAmbushDirection.ambushDirection = locationNumber; } else { lastAmbushDirections.push({ thiefLocation: playerLocationNumber, ambushDirection: locationNumber }); } setPlayerValue("lastAmbushDirections", JSON.stringify(lastAmbushDirections)); await postRequest("/thief_ambush.php", `id=${locationNumber}&with_who=0`); getURL("/map.php"); } } else { alert(AmbushMaySettedOnAdjacentSector); } } function checkSectorForAmbush(locationNumber) { if(!locationNumber) { return false; } if(playerLocationNumber in routes && locationNumber in routes[playerLocationNumber] && routes[playerLocationNumber][locationNumber]) { const route = routes[playerLocationNumber][locationNumber][0]; const nextRoutePoint = route.split('-')[1]; return nextRoutePoint == locationNumber; } return false; } function resetMountInfo() { deletePlayerValue("TravelingTime"); deletePlayerValue("ComplexRoute"); deletePlayerValue("TransportEndDate"); deletePlayerValue("Antithief"); deletePlayerValue("AntithiefControlled"); deletePlayerValue("ToggleAntithiefKey"); deletePlayerValue("IsPremiunAccount"); } async function getMountInfo(force = false) { let endDate = parseInt(getPlayerValue("TransportEndDate", 0)); if(endDate <= getServerTime() || force || location.pathname == "/shop.php" && getUrlParamValue(location.href, "cat") == "transport") { if(endDate <= getServerTime()) { resetMountInfo(); } const doc = location.pathname == "/shop.php" && getUrlParamValue(location.href, "cat") == "transport" ? document : await getRequest("/shop.php?cat=transport"); const mounts = doc.querySelectorAll('div.s_art_r'); //console.log(["mounts", mounts.length]); const mountsInfoGeted = mounts.length > 0; // Если 0, то мы в пути и на страницу магазина нас не пускают. Поэтому используем старые данные о транспорте if(mountsInfoGeted) { let isPremiunAccount = false; for(const mount of mounts) { if(mount.innerHTML.includes(YourPremiumMountUntil) || mount.innerHTML.includes(YourMountUntil)) { var mountRow = mount; isPremiunAccount = mount.innerHTML.includes(YourPremiumMountUntil); break; } } if(mountRow) { resetMountInfo(); setPlayerValue("IsPremiunAccount", isPremiunAccount); const mountExpireTimeDiv = mountRow.querySelector('div.s_art_note_transport') || mountRow.querySelector('div.s_art_name.s_transport_name'); const dateString = (mountExpireTimeDiv.querySelector('span') || mountExpireTimeDiv).innerText.split(Until)[1]; setPlayerValue("TransportEndDate", parseDate(dateString).getTime()); const transportOptions = mountRow.querySelectorAll("div.s_art_prop_transport"); for(let transportOption of transportOptions) { if(transportOption.innerHTML.includes(TravelingTimeName)) { const timeValue = /(\d{1,2})/.exec(transportOption.innerHTML); setPlayerValue("TravelingTime", timeValue[1]); } if(transportOption.innerHTML.includes(ComplexRouteName)) { setPlayerValue("ComplexRoute", transportOption.querySelector("img.s_art_prop_transport_amount_image[src*='check_yes']") ? true : false); } if(transportOption.innerHTML.includes(Antithief)) { let antithiefControlImage = transportOption.querySelector("img.s_art_prop_transport_amount_image[src*='check_on']"); antithiefControlImage = antithiefControlImage || transportOption.querySelector("img.s_art_prop_transport_amount_image[src*='check_off']"); setPlayerValue("Antithief", antithiefControlImage ? true : false); setPlayerValue("AntithiefControlled", antithiefControlImage ? true : false); if(antithiefControlImage) { setPlayerValue("ToggleAntithiefKey", getUrlParamValue(antithiefControlImage.parentNode.href, "t_on") || getUrlParamValue(antithiefControlImage.parentNode.href, "t_off")); } } } } } } const mountInfo = { TravelingTime: parseInt(getPlayerValue("TravelingTime", DefaultTravelingTime)), ComplexRoute: getPlayerBool("ComplexRoute") || getBool("IsDeer"), EndDate: parseInt(getPlayerValue("TransportEndDate", 0)), Antithief: getPlayerBool("Antithief"), AntithiefControlled: getPlayerBool("AntithiefControlled"), IsPremiunAccount: getPlayerBool("IsPremiunAccount"), IsDeer: getBool("IsDeer") }; if(mountInfo.IsDeer) { const deerTravelingTime = DefaultTravelingTime * 0.2; if(mountInfo.TravelingTime > deerTravelingTime) { mountInfo.TravelingTime = deerTravelingTime; } } //console.log(mountInfo); return mountInfo; } function processHouseInfo() { if(location.pathname == '/house_info.php') { const houseId = getUrlParamValue(location.href, "id"); const hostDiv = document.querySelector("div#tt"); const container = getParent(hostDiv, "table"); const loc = container.querySelector("a[href^='map.php?cx=']"); const locationNumber = getLocationNumberFromMapUrlByXy(loc.href); const ownerName = container.querySelector("a[href^='pl_info.php?id=']").querySelector("b").innerText; const hostName = hostDiv.querySelector("b").innerText; let maxExpireDate; const guestRoomDivs = container.querySelectorAll("div[id^='gr']"); for(const guestRoomDiv of guestRoomDivs) { const playerReference = guestRoomDiv.querySelector(`a[href='pl_info.php?id=${PlayerId}']`) if(playerReference && playerReference.nextSibling) { const re = /(\d{2}:\d{2} \d{2}-\d{2})/.exec(playerReference.nextSibling.nodeValue); if(re) { const expireDate = parseDate(re[1], true); if(!maxExpireDate || maxExpireDate < expireDate) { maxExpireDate = expireDate; } } } } const guestInfo = getGuestInfo(locationNumber); if(maxExpireDate) { guestInfo[houseId] = { HostInfo: `${ownerName} (${hostName})`, ExpireDate: maxExpireDate.toJSON() }; } setPlayerValue(`GuestInfo${locationNumber}`, JSON.stringify(guestInfo)); //console.log(guestInfo); } } function getGuestInfo(locationNumber) { if(getPlayerValue(`GuestInfo${locationNumber}`)) { const now = new Date(); const guestInfo = JSON.parse(getPlayerValue(`GuestInfo${locationNumber}`)); for(const guestInfoKey in guestInfo) { if(new Date(guestInfo[guestInfoKey].ExpireDate) < now) { var repackNeeded = true; delete guestInfo[guestInfoKey]; } } const newKeys = Object.keys(guestInfo); if(repackNeeded) { if(newKeys.length == 0) { deletePlayerValue(`GuestInfo${locationNumber}`); } else { setPlayerValue(`GuestInfo${locationNumber}`, JSON.stringify(guestInfo)); } } if(newKeys.length > 0) { return guestInfo; } } return {}; } function createPatrolRoute() { const transporterLocations = Array.from(document.querySelectorAll("td[id^=transporterLocation]")).filter(x => parseInt(x.id.replace("transporterLocation", "")) < 100); const patrolRouteEnabled = getPlayerBool("PatrolRouteEnabled"); transporterLocations.forEach(x => addElement("input", { type: "checkbox", id: `patrolRouteCheckbox${x.id.replace("transporterLocation", "")}`, name: "patrolRouteCheckbox", style: `float: left; ${patrolRouteEnabled ? "" : "display: none;"}` }, x)); //transporterLocations.forEach(x => addElement("input", x, { type: "checkbox", id: `patrolRouteCheckbox${x.id.replace("transporterLocation", "")}`, name: "patrolRouteCheckbox", style: `position: absolute; top: ${x.getBoundingClientRect().top + 1}px; left: ${x.getBoundingClientRect().left + 1}px;${patrolRouteEnabled ? "" : "display: none;"}` })); const patrolRouteCheckboxes = Array.from(document.querySelectorAll("input[type=checkbox][name=patrolRouteCheckbox]")); //console.log(patrolRouteCheckboxes); const patrolRoute = JSON.parse(getPlayerValue("PatrolRoute", "[]")); patrolRouteCheckboxes.forEach(x => { x.checked = patrolRoute.includes(parseInt(x.id.replace("patrolRouteCheckbox", ""))); x.addEventListener("click", function(e) { e.stopPropagation(); repackPatrolRoute(); }); }); } function togglePatrolRoute() { const patrolRouteEnabled = getPlayerBool("PatrolRouteEnabled"); const patrolRouteCheckboxes = Array.from(document.querySelectorAll("input[type=checkbox][name=patrolRouteCheckbox]")); patrolRouteCheckboxes.forEach(x => { x.style.display = patrolRouteEnabled ? "" : "none"; }); } function repackPatrolRoute() { const checkedPatrolRouteCheckboxes = Array.from(document.querySelectorAll("input:checked[type=checkbox][name=patrolRouteCheckbox]")); //console.log(checkedPatrolRouteCheckboxes); let patrolRouteLocations = checkedPatrolRouteCheckboxes.map(x => parseInt(x.id.replace("patrolRouteCheckbox", ""))); //console.log(patrolRouteLocations); const patrolRoute = []; let currentRoutePoint; let i = 0; while(patrolRouteLocations.length > 0) { const nearestLocations = currentRoutePoint ? getLocationsSortedByDistance(currentRoutePoint).slice(1) : [patrolRouteLocations[0]]; //console.log(`currentRoutePoint: ${currentRoutePoint}`); //console.log(nearestLocations); for(const nearestLocation of nearestLocations) { const index = patrolRouteLocations.indexOf(parseInt(nearestLocation)); if(index > -1) { patrolRoute.push(patrolRouteLocations[index]); currentRoutePoint = patrolRouteLocations[index]; patrolRouteLocations.splice(index, 1); //console.log(patrolRouteLocations); break; } } i++; if(i > 26) { break; } } //console.log(patrolRoute); setPlayerValue("PatrolRoute", JSON.stringify(patrolRoute)); document.getElementById("togglePatrollingButton").title = JSON.parse(getPlayerValue("PatrolRoute", "[]")).join("->"); } function togglePatrolling() { if(getPlayerBool("IsPatrolling")) { deletePlayerValue("IsPatrolling"); } else { setPlayerValue("IsPatrolling", true); } checkPatrolling(); document.getElementById("togglePatrollingButton").value = getPlayerBool("IsPatrolling") ? (isEn ? "Stop patrolling" : "Остановить патрулирование") : (isEn ? "Start patrolling" : "Начать патрулирование"); } function checkPatrolling() { // Идем в патрулирование если, либо нет охот, либо они скрыты if(getPlayerBool("IsPatrolling") && !mooving && (getPlayerBool("HideHuntBlock") || document.getElementById("next_ht_new"))) { const patrolRoute = JSON.parse(getPlayerValue("PatrolRoute", "[]")); if(patrolRoute.length > 1) { let nextLocation; const index = patrolRoute.indexOf(parseInt(playerLocationNumber)); if(index > -1) { nextLocation = index < patrolRoute.length - 1 ? patrolRoute[index + 1] : patrolRoute[0]; } else { nextLocation = patrolRoute[0]; } if(nextLocation) { //console.log(`nextLocation: ${nextLocation}, start in ${getPlayerValue("PatrollingPause", 5)} seconds`); setTimeout(function() { if(getPlayerBool("IsPatrolling")) { tryMoving(nextLocation); } }, parseInt(getPlayerValue("PatrollingPause", 5)) * 1000); } } } } function getHorseImageData() { return resourcesPath3 + "/i/combat/map/navigator_btn_horseman.png"; //return ""; } // API function getServerTime() { return Date.now() - parseInt(GM_getValue("ClientServerTimeDifference", 0)); } function getGameDate() { return new Date(getServerTime() + 10800000); } // Игра в интерфейсе всегда показывает московское время // Это та дата, которая в toUTCString покажет время по москве // dateString - игровое время, взятое со страниц игры. Оно всегда московское // Как результат возвращаем серверную дату function parseDate(dateString, isFuture = false, isPast = false) { //console.log(dateString) if(!dateString) { return; } const dateStrings = dateString.split(" "); let hours = 0; let minutes = 0; let seconds = 0; const gameDate = getGameDate(); let year = gameDate.getUTCFullYear(); let month = gameDate.getUTCMonth(); let day = gameDate.getUTCDate(); const timePart = dateStrings.find(x => x.includes(":")); if(timePart) { var time = timePart.split(":"); hours = parseInt(time[0]); minutes = parseInt(time[1]); if(time.length > 2) { seconds = parseInt(time[2]); } if(dateStrings.length == 1) { let result = new Date(Date.UTC(year, month, day, hours, minutes, seconds)); if(isPast && result > gameDate) { result.setUTCDate(result.getUTCDate() - 1); } if(isFuture && result < gameDate) { result.setUTCDate(result.getUTCDate() + 1); } //console.log(`result: ${result}, gameDate: ${gameDate}`) result.setUTCHours(result.getUTCHours() - 3); return result; } } const datePart = dateStrings.find(x => x.includes("-")); if(datePart) { const date = datePart.split("-"); month = parseInt(date[isEn ? (date.length == 3 ? 1 : 0) : 1]) - 1; day = parseInt(date[isEn ? (date.length == 3 ? 2 : 1) : 0]); if(date.length == 3) { const yearText = isEn ? date[0] : date[2]; year = parseInt(yearText); if(yearText.length < 4) { year += Math.floor(gameDate.getUTCFullYear() / 1000) * 1000; } } else { if(isFuture && month == 0 && gameDate.getUTCMonth() == 11) { year += 1; } } } if(dateStrings.length > 2) { const letterDateExec = /(\d{2}):(\d{2}) (\d{2}) (.{3,4})/.exec(dateString); if(letterDateExec) { //console.log(letterDateExec) day = parseInt(letterDateExec[3]); //const monthNames = ['января', 'февраля', 'марта', 'апреля', 'мая', 'июня', 'июля', 'августа', 'сентября', 'октября', 'ноября', 'декабря']; const monthShortNames = ['янв', 'фев', 'март', 'апр', 'май', 'июнь', 'июль', 'авг', 'сент', 'окт', 'ноя', 'дек']; month = monthShortNames.findIndex(x => x.toLowerCase() == letterDateExec[4].toLowerCase()); if(isPast && Date.UTC(year, month, day, hours, minutes, seconds) > gameDate.getTime()) { year -= 1; } } } //console.log(`year: ${year}, month: ${month}, day: ${day}, time[0]: ${time[0]}, time[1]: ${time[1]}, ${new Date(year, month, day, parseInt(time[0]), parseInt(time[1]))}`); let result = new Date(Date.UTC(year, month, day, hours, minutes, seconds)); result.setUTCHours(result.getUTCHours() - 3); return result; } function addElement(type, data = {}, parent = undefined, insertPosition = "beforeend") { const el = document.createElement(type); for(const key in data) { if(key == "innerText" || key == "innerHTML") { el[key] = data[key]; } else { el.setAttribute(key, data[key]); } } if(parent) { if(parent.insertAdjacentElement) { parent.insertAdjacentElement(insertPosition, el); } else if(parent.parentNode) { switch(insertPosition) { case "beforebegin": parent.parentNode.insertBefore(el, parent); break; case "afterend": parent.parentNode.insertBefore(el, parent.nextSibling); break; } } } return el; } function getUrlParamValue(url, paramName) { return (new URLSearchParams(url.split("?")[1])).get(paramName); } function getURL(url) { window.location.href = url; } async function loadFactories() { let selectedJobDiv = document.querySelector("div.job_fl_btn.show_hint.job_fl_btn_selected"); if(!selectedJobDiv) { return; // Мы в пути } let currentFactoryType = getUrlParamValue(selectedJobDiv.parentNode.href, "st"); const viewingLocationNumber = getLocationNumberFromMapUrlByXy(selectedJobDiv.parentNode.href); let indexOfCurrentFactoryType = factoryTypes.indexOf(currentFactoryType); if(indexOfCurrentFactoryType == -1) { return; } const mapRightBlock = document.getElementById("map_right_block"); //const mapRightBlockInside = document.getElementById("map_right_block_inside"); //console.log(`${mapRightBlock.id}, offsetHeight: ${mapRightBlock.offsetHeight}, height: ${mapRightBlock.style.height}`); //console.log(`${mapRightBlockInside.id}, offsetHeight: ${mapRightBlockInside.offsetHeight}, height: ${mapRightBlockInside.style.height}`); let mapRightBlockHeight = mapRightBlock.offsetHeight; let mainFactoriesTable = mapRightBlock.querySelector("table.wb"); if(!mainFactoriesTable) { return; } let mapRef = document.querySelector("a[href^='map.php?cx=']"); let x = getUrlParamValue(mapRef.href, "cx"); let y = getUrlParamValue(mapRef.href, "cy"); let factoriesTableContainer = mainFactoriesTable.parentNode; const factoryTables = { }; factoryTables[currentFactoryType] = mainFactoriesTable; for(const factoryType of factoryTypes) { if(factoryType != currentFactoryType && getPlayerBool("Load" + factoryType) && getValue(`LastDetectedEmptyFactoriesDate_${factoryType}${viewingLocationNumber}`, 0) + 1000 * 60 * 60 * 24 * 30 < Date.now()) { const doc = await getRequest(`/map.php?cx=${x}&cy=${y}&st=${factoryType}`); const docMapRightBlock = doc.getElementById("map_right_block"); factoryTables[factoryType] = docMapRightBlock.querySelector("table.wb"); } } for(let i = 0; i < factoryTypes.length; i++) { let tableElement = factoryTables[factoryTypes[i]]; if(!tableElement) { continue; } const isTableEmpty = tableElement.rows.length <= 2 && (tableElement.rows.length < 2 || tableElement.rows[1].cells.length == 0); if(isTableEmpty) { setValue(`LastDetectedEmptyFactoriesDate_${factoryTypes[i]}${viewingLocationNumber}`, Date.now()); } else if(getValue(`LastDetectedEmptyFactoriesDate_${factoryTypes[i]}${viewingLocationNumber}`)) { deleteValue(`LastDetectedEmptyFactoriesDate_${factoryTypes[i]}${viewingLocationNumber}`); } if(!isTableEmpty) { if(indexOfCurrentFactoryType < i) { factoriesTableContainer.appendChild(tableElement) mapRightBlockHeight += tableElement.offsetHeight; } if(indexOfCurrentFactoryType > i) { factoriesTableContainer.insertBefore(tableElement, mainFactoriesTable); mapRightBlockHeight += tableElement.offsetHeight; } } else if(indexOfCurrentFactoryType == i) { mapRightBlockHeight -= mainFactoriesTable.offsetHeight; factoriesTableContainer.removeChild(mainFactoriesTable); // Если текущий список предприятий - пустой, то уберем его } } mapRightBlock.style.height = `${mapRightBlockHeight}px`; } function getParent(element, parentType, number = 1) { if(!element) { return; } let result = element; let foundNumber = 0; while(result = result.parentNode) { if(result.nodeName.toLowerCase() == parentType.toLowerCase()) { foundNumber++; if(foundNumber == number) { return result; } } } } function observe(target, handler, config = { childList: true, subtree: true }) { const ob = new MutationObserver(async function(mut, observer) { //console.log(`Mutation start`); observer.disconnect(); if(handler.constructor.name === 'AsyncFunction') { await handler(); } else { handler(); } observer.observe(target, config); }); ob.observe(target, config); } async function initUserName() { if(getValue("TransporterUserName")) { deleteValue("UserName"); deleteValue("TransporterUserName"); } if(location.pathname == "/pl_info.php" && getUrlParamValue(location.href, "id") == PlayerId) { //console.log(document.querySelector("h1").innerText) setPlayerValue("UserName", document.querySelector("h1").innerText); } if(location.pathname == "/home.php") { //console.log(document.querySelector(`a[href='pl_info.php?id=${PlayerId}'] > b`).innerText) const userNameRef = document.querySelector(`a[href='pl_info.php?id=${PlayerId}'] > b`); if(userNameRef) { setPlayerValue("UserName", userNameRef.innerText); } } if(!getPlayerValue("UserName")) { const doc = await getRequest(`/pl_info.php?id=${PlayerId}`); setPlayerValue("UserName", doc.querySelector("h1").innerText); } } function getNearestAncestorSibling(node) { let parentNode = node; while((parentNode = parentNode.parentNode)) { if(parentNode.nextSibling) { return parentNode.nextSibling; } } } function getNearestAncestorElementSibling(node) { let parentNode = node; while((parentNode = parentNode.parentNode)) { if(parentNode.nextElementSibling) { return parentNode.nextElementSibling; } } } function nextSequential(node) { return node.firstChild || node.nextSibling || getNearestAncestorSibling(node); } function nextSequentialElement(element) { return element.firstElementChild || element.nextElementSibling || getNearestAncestorElementSibling(element); } function getSequentialsUntil(firstElement, lastElementTagName) { let currentElement = firstElement; const resultElements = [currentElement]; while((currentElement = nextSequential(currentElement)) && currentElement.nodeName.toLowerCase() != lastElementTagName.toLowerCase()) { resultElements.push(currentElement); } if(currentElement) { resultElements.push(currentElement); } return resultElements; } function getStorageKeys(filter) { return listValues().filter(filter); } function mobileCheck() { let check = false; (function(a){if(/(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od)|iris|kindle|lge |maemo|midp|mmp|mobile.+firefox|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows ce|xda|xiino/i.test(a)||/1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a wa|abac|ac(er|oo|s\-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|\-m|r |s )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw\-(n|u)|c55\/|capi|ccwa|cdm\-|cell|chtm|cldc|cmd\-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc\-s|devi|dica|dmob|do(c|p)o|ds(12|\-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(\-|_)|g1 u|g560|gene|gf\-5|g\-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd\-(m|p|t)|hei\-|hi(pt|ta)|hp( i|ip)|hs\-c|ht(c(\-| |_|a|g|p|s|t)|tp)|hu(aw|tc)|i\-(20|go|ma)|i230|iac( |\-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt( |\/)|klon|kpt |kwc\-|kyo(c|k)|le(no|xi)|lg( g|\/(k|l|u)|50|54|\-[a-w])|libw|lynx|m1\-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m\-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(\-| |o|v)|zz)|mt(50|p1|v )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)\-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|\-([1-8]|c))|phil|pire|pl(ay|uc)|pn\-2|po(ck|rt|se)|prox|psio|pt\-g|qa\-a|qc(07|12|21|32|60|\-[2-7]|i\-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h\-|oo|p\-)|sdk\/|se(c(\-|0|1)|47|mc|nd|ri)|sgh\-|shar|sie(\-|m)|sk\-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h\-|v\-|v )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl\-|tdg\-|tel(i|m)|tim\-|t\-mo|to(pl|sh)|ts(70|m\-|m3|m5)|tx\-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|\-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(\-| )|webc|whit|wi(g |nc|nw)|wmlb|wonu|x700|yas\-|your|zeto|zte\-/i.test(a.substr(0,4))) check = true;})(navigator.userAgent||navigator.vendor||window.opera); return check; }; function getValue(key, defaultValue) { return GM_getValue(key, defaultValue); }; function setValue(key, value) { GM_setValue(key, value); }; function deleteValue(key) { return GM_deleteValue(key); }; function getPlayerValue(key, defaultValue) { return GM_getValue(`${key}${PlayerId}`, defaultValue); }; function setPlayerValue(key, value) { GM_setValue(`${key}${PlayerId}`, value); }; function deletePlayerValue(key) { return GM_deleteValue(`${key}${PlayerId}`); }; function listValues() { return GM_listValues(); } function getPlayerBool(valueName, defaultValue = false) { return getBool(valueName + PlayerId, defaultValue); } function getBool(valueName, defaultValue = false) { const value = getValue(valueName); //console.log(`valueName: ${valueName}, value: ${value}, ${typeof(value)}`) if(value != undefined) { if(typeof(value) == "string") { return value == "true"; } if(typeof(value) == "boolean") { return value; } } return defaultValue; } function getScriptLastAuthor() { let authors = GM_info.script.author; if(!authors) { const authorsMatch = GM_info.scriptMetaStr.match(/@author(.+)\n/); authors = authorsMatch ? authorsMatch[1] : ""; } const authorsArr = authors.split(",").map(x => x.trim()).filter(x => x); return authorsArr[authorsArr.length - 1]; } function getDownloadUrl() { let result = GM_info.script.downloadURL; if(!result) { const downloadURLMatch = GM_info.scriptMetaStr.match(/@downloadURL(.+)\n/); result = downloadURLMatch ? downloadURLMatch[1] : ""; result = result.trim(); } return result; }