bot for xospital.mobi
// ==UserScript==
// @name xospital bot
// @namespace http://tampermonkey.net/
// @version 0.2.3
// @description bot for xospital.mobi
// @include https://xospital.mobi/*
// @include https://odkl.xospital.mobi/*
// @author axmed2004
// @run-at document-start
// ==/UserScript==
// интервал в ЧП при найденном пациенте (сек.)
var questsShortTimeout = 3;
// интервал в ЧП при отсутствии пациента (мин.)
var questsLongTimeout = 1;
// интервал для перехода по страницам мин/макс (сек.)
var rimin = +localStorage.getItem("pagesRunIntervalMin")||4;
var rimax = +localStorage.getItem("pagesRunIntervalMax")||6;
if (rimin > rimax || rimin < 1) rimin = 1;
if (rimax < rimin || rimax < 1) rimax = 1;
// интервал до нового цикла мин/макс (мин.)
var rtmin = +localStorage.getItem("restartTimeMin")||3;
var rtmax = +localStorage.getItem("restartTimeMax")||4;
if (rtmin > rtmax || rtmin < 1) rtmin = 1;
if (rtmax < rtmin || rtmax < 1) rtmax = 1;
//при наличии Карманного компьютера
var rsm = localStorage.getItem("roomsSelectMethod") || "random";
var rsp = localStorage.getItem("roomsSelectParam") || "time";
var rsv = localStorage.getItem("roomsSelectValue") || "min";
var asm = localStorage.getItem("autoparkSelectMethod") || "random";
var asp = localStorage.getItem("autoparkSelectParam") || "time";
var asv = localStorage.getItem("autoparkSelectValue") || "min";
var patientsTimeoutTime = Math.round((Math.random() * (rtmax - rtmin) + rtmin) * 60);
var chestsTimeoutTime = Math.round((Math.random() + 7) * 60);
var pagesRunIntervalTime = (Math.random() * (rimax - rimin) + rimin) * 1000;
var restartInterval = null;
var pagesTimeout = null;
var restartTimeoutRunned = false;
var rooms = /^https:\/\/(odkl\.)?xospital\.mobi\/Rooms/.test(location.href);
var reception = /^https:\/\/(odkl\.)?xospital\.mobi\/Reception/.test(location.href);
var pharmacy = /^https:\/\/(odkl\.)?xospital\.mobi\/Pharmacy/.test(location.href);
var autopark = /^https:\/\/(odkl\.)?xospital\.mobi\/AutoPark\?/.test(location.href);
var autoParkDestination = /^https:\/\/(odkl\.)?xospital\.mobi\/AutoParkDestination\?/.test(location.href);
var vetclinic = /^https:\/\/(odkl\.)?xospital\.mobi\/VetClinic/.test(location.href);
var quests = /^https:\/\/(odkl\.)?xospital\.mobi\/Quests/.test(location.href);
var chests = /^https:\/\/(odkl\.)?xospital\.mobi\/Underwater\/Chests/.test(location.href);
var mainPages = rooms || reception || pharmacy || autopark || autoParkDestination;
var splash = `<div id='splashoverlay' style='position:absolute;left:0px;top:0px;width:100%;height:100%;background-color:#000;opacity:0.4'></div>
<div id='splashinfo' style='position:absolute;margin:0 auto;border:3px solid #ccc;background-color:white;width:50%;height:80%;left: 25%;top: 10%;'>
Бот состоит из 4 частей: <br><br>
1. Бот для палат, лабораторий и автопарка, циклически передвигается по ним.<br>
2. Бот для ЧП.<br>
3. Бот для ветеринарной клиники.<br>
4. Бот для покупки ключей в Подводной охоте.<br><br>
Для параллельной работы откройте в браузере отдельные вкладки для каждого бота.<br>
Нажмите на значок настроек в заголовке страницы и выберите нужные параметры.<br>
<span style='color:red;padding-left:10px;'>*</span><i>Интервалы сделаны для симуляции поведения человека, при слишком коротких интервалах возможен бан аккаунта.</i><br><br>
Для избежания бана рекомендую каждые несколько часов выключать бота. Не оставляйте его включенным на ночь.<br>
Установите временные интервалы соответсвенно уровню вашего оборудования (Рентгеносканер, Громкоговоритель, GPS) (+/- 1-2 мин от текущего значения)<br><br>
<button style='display:block;margin:0 auto;font-size:16px;' onclick='(function(){localStorage.setItem("firstrun", "false");location.reload();})()'>Перезагрузить страницу для запуска скрипта</button>
</div>`;
var setsOpened = false;
Node.prototype.get=function(s){
return this.querySelector(s);
}
Node.prototype.getAll=function(s){
return this.querySelectorAll(s);
}
function get(s){
return document.querySelector(s);
}
function getAll(s){
return document.querySelectorAll(s);
}
function openSets() {
setsOpened = true;
if (get("#splashoverlay") != null && get("#setsWindow")!=null) {
get("#splashoverlay").style.display = "block";
get("#setsWindow").style.top = get("#botsets").offsetTop + get("#botsets").height;
get("#setsWindow").style.right = get("#botsets").offsetTop + get("#botsets").width;
}
get("#setsWindow").style.display = "block";
clearInterval(restartInterval);
clearTimeout(pagesTimeout);
}
function closeSets() {
setsOpened = false;
get("#splashoverlay").style.display = "none";
get("#setsWindow").style.display = "none";
if (mainPages) {
if(restartTimeoutRunned) restartTimeout()
}
else pagesTimeout=setTimeout(patientsRun, pagesRunIntervalTime);
}
function drawSets() {
if (mainPages) {
get("div.caption h1").insertAdjacentHTML("beforeend", "<img id='botsets' width=30 height=30 style='float:right;vertical-align:top;' src=''>");
let setsWindow = `<div id='setsWindow' style='display:none;z-index:20;position:absolute;top:10%;right:10%;background-color:white;border:3px solid #ccc;'>
<table>
<tr><td>Интервал перехода между страницами<br>
от <input type='number' min='1' id='pagesRunIntervalMin' value='` + rimin + `' style='width:50px;'>до <input type='number' min='1' id='pagesRunIntervalMax' value='` + rimax + `' style='width:50px;'>сек
</td></tr>
<tr><td>Интервал до нового цикла в минутах<br>
от <input type='number' min='1' id='restartTimeMin' value='` + rtmin + `' style='width:50px;'>до <input type='number' min='1' id='restartTimeMax' value='` + rtmax + `' style='width:50px;'>мин
</td></tr>
<tr><td>Выбор пациентов в палаты<br>
<select id='roomsSelectMethod'>
<option value='random'>Случайно</option>
<option value='event'>По событию</option>
<option value='value'>По значению</option>
</select>
<select id='roomsSelectParam'` + (rsm != `value` ? ` disabled` : ``) + `>
<option value='time'>Время</option>
<option value='price'>Стоимость</option>
<option value='exp'>Опыт</option>
</select>
<select id='roomsSelectValue'` + (rsm != `value` ? ` disabled` : ``) + `>
<option value='min'>Мин.</option>
<option value='max'>Макс.</option>
<option value='random'>Любое</option>
</select>
</td></tr>
<tr><td>Выбор пациентов в автопарк<br>
<select id='autoparkSelectMethod'>
<option value='random'>Случайно</option>
<option value='event'>По событию</option>
<option value='value'>По значению</option>
</select>
<select id='autoparkSelectParam' ` + (asm != `value` ? ` disabled` : ``) + `>
<option value='time'>Время</option>
<option value='price'>Стоимость</option>
<option value='exp'>Опыт</option>
</select>
<select id='autoparkSelectValue'` + (asm != `value` ? ` disabled` : ``) + `>
<option value='min'>Мин.</option>
<option value='max'>Макс.</option>
<option value='random'>Любое</option>
</select>
</td></tr>
</table>
<div>
<button id='savesets' style='margin-top:20px;display:inline-block;'>Сохранить</button>
<button id='closesets' style='margin-top:20px;display:inline-block;float:right;'>Закрыть</button>
</div></div>`;
document.body.insertAdjacentHTML("afterBegin", setsWindow);
get("#roomsSelectMethod").value = rsm;
get("#roomsSelectParam").value = rsp;
get("#roomsSelectValue").value = rsv;
get("#autoparkSelectMethod").value = asm;
get("#autoparkSelectParam").value = asp;
get("#autoparkSelectValue").value = asv;
document.body.insertAdjacentHTML("afterBegin", "<div id='splashoverlay' style='position:absolute;display:none;width:100%;height:100%;left:0px;top:0px;background-color:#000;opacity:0.4;'></div>");
get("#roomsSelectMethod").addEventListener("change", (e) => {
get("#roomsSelectParam").disabled = e.target.value != "value";
get("#roomsSelectValue").disabled = e.target.value != "value";
})
get("#autoparkSelectMethod").addEventListener("change", (e) => {
get("#autoparkSelectParam").disabled = e.target.value != "value";
get("#autoparkSelectValue").disabled = e.target.value != "value";
})
get("#botsets").addEventListener("click", () => {
openSets();
})
get("#savesets").addEventListener("click", () => {
let runIntervalMin_tb = get("#pagesRunIntervalMin").value;
let runIntervalMax_tb = get("#pagesRunIntervalMax").value;
let restartTimeMin_tb = get("#restartTimeMin").value;
let restartTimeMax_tb = get("#restartTimeMax").value;
let roomsSelectMethod_select = get("#roomsSelectMethod").value;
let roomsSelectParam_select = get("#roomsSelectParam").value;
let roomsSelectValue_select = get("#roomsSelectValue").value;
let autoparkSelectMethod_select = get("#autoparkSelectMethod").value;
let autoparkSelectParam_select = get("#autoparkSelectParam").value;
let autoParkSelectValue_select = get("#autoparkSelectValue").value;
if (runIntervalMin_tb != "" && +runIntervalMin_tb > 0
&& runIntervalMax_tb != "" && +runIntervalMax_tb > 0
&& +runIntervalMax_tb > +runIntervalMin_tb) {
localStorage.setItem("pagesRunIntervalMin", runIntervalMin_tb);
localStorage.setItem("pagesRunIntervalMax", runIntervalMax_tb);
}
if (restartTimeMin_tb != "" && +restartTimeMin_tb > 0
&& restartTimeMax_tb != "" && +restartTimeMax_tb > 0
&& +restartTimeMax_tb> +restartTimeMin_tb) {
localStorage.setItem("restartTimeMin", restartTimeMin_tb);
localStorage.setItem("restartTimeMax", restartTimeMax_tb);
}
localStorage.setItem("roomsSelectMethod", roomsSelectMethod_select);
if (roomsSelectMethod_select == "value") {
localStorage.setItem("roomsSelectParam", roomsSelectParam_select);
localStorage.setItem("roomsSelectValue", roomsSelectValue_select);
}
localStorage.setItem("autoparkSelectMethod", autoparkSelectMethod_select);
if (autoparkSelectMethod_select == "value") {
localStorage.setItem("autoparkSelectParam", autoparkSelectParam_select);
localStorage.setItem("autoparkSelectValue", autoParkSelectValue_select);
}
})
get("#closesets").addEventListener("click", () => {
closeSets();
})
}
}
function restartTimeout() {
if (!setsOpened) {
restartTimeoutRunned = true;
if (get("span#timeleft") == null) {
var span = document.createElement("span");
span.setAttribute("id","timeleft");
get("div.caption h1").appendChild(span);
}
restartInterval= setInterval(() => {
patientsTimeoutTime--;
if(patientsTimeoutTime>=0){
mins=Math.floor(patientsTimeoutTime/60);
secs=Math.floor((patientsTimeoutTime-mins*60));
if(secs<10) secs="0"+secs;
document.title=mins+":"+secs;
get("span#timeleft").innerHTML=document.title;
}
else{
clearInterval(restartInterval)
get("ul.action_list a[href^='/Rooms']").click();
}
}, 1000);
}
}
function patientsRun() {
if(rooms)
{
// палаты
let l = get("a[href^='/Rooms/Examine']") ||
get("a[href^='/Rooms/FirstExamine']") ||
get("a[href^='/Rooms/GetVitamin']") ||
get("a[href^='/Rooms/GetPotion']") ||
get("a[href^='/Rooms/Discharge']") ||
get("a[href^='/Rooms/ClearAll']") ||
get("a[href^='/Rooms/RoomClear']");
let fl = get("ul.action_list a[href^='/Pharmacy']");
if (l) l.click();
else if(rsm=="random" && get("a[href^='/Reception/TreatAll?treatType=Any']")!=null)
get("a[href^='/Reception/TreatAll?treatType=Any']").click();
else if(rsm=="random" && get("a[href^='/Reception/TreatAllConfirm?']")!=null)
get("a[href^='/Reception/TreatAllConfirm?']").click();
else if(rsm=="event" && get("a[href^='/Reception/TreatAll?treatType=Epidemic']")!=null)
get("a[href^='/Reception/TreatAll?treatType=Epidemic']").click();
else if (rsm == "value") {
let l = get("ul.padtop_s a[href^='/Reception?roomIndex']")
if (l) l.click();
else if(get("a[href^='/Rooms?page=']")!=null){
let url = new URL(location.href);
let urlp = 1;
if (url.searchParams.has("page"))
urlp = +url.searchParams.get("page");
if(get("a[href^='/Rooms?page="+(urlp+1)+"']")!=null)
get("a[href^='/Rooms?page=" + (urlp + 1) + "']").click();
else
fl.click();
}
}
else fl.click();
//переход в лаборатории
}
else if(reception){
// приемная
var $list = getAll('ul.delim-list.padtop_s li.padtop_s');
let l = get('.epidemic');
if(rsm=="event" && l)
l.get('a').click();
else{
var elem=0;
arr = [];
arr2 = [];
if(rsv=="random")
elem = ~~(Math.random() * $list.length);
else{
if(rsp=="price"){
$list.forEach(el=>{
arr.push(+el.getAll("span.money")[1].innerHTML);
})
}
else if(rsp=="time"){
$list.forEach(el=>{
let t = el.get(".smallfont.minor span.ylwtitle").innerHTML;
let m = "";
if(/(\d+) д./.test(t)){
m = t.match(/(\d+) д./);
arr.push(+m[1] * 24 * 60);
}
else if(/(\d+) д. (\d+) ч./.test(t)){
m = t.match(/(\d+) д. (\d+) ч./);
arr.push(+m[1] * 24 * 60 + (+m[2]) * 60);
}
else if(/(\d+) ч./.test(t)){
m = t.match(/(\d+) ч./);
arr.push(+m[1] * 60);
}
else if(/(\d+) мин./.test(t)){
m = t.match(/(\d+) мин./);
arr.push(+m[1]);
}
else if(/(\d+) ч. (\d+) мин./.test(t)){
m = t.match(/(\d+) ч. (\d+) мин./);
arr.push(+m[1] * 60 + parseInt(m[2]));
}
})
}
else if(rsp=="exp"){
$list.forEach(el=>{
arr.push(+el.getAll("span.money")[0].innerHTML);
})
}
arr2 = [...arr];
arr.sort((a, b) => a - b);
elem = arr2.indexOf(arr[rsv == "min" ? 0 : arr.length - 1]);
}
$list[elem].get("a").click();
}
}
else if(pharmacy){
// лаборатории
let l = get("a[href^='/Pharmacy/CheerupAll']") ||
get("a[href^='/Pharmacy/Mix']") ||
get("a[href^='/Pharmacy/FoodAll']") ||
get("a[href^='/Pharmacy/GetAccelerator']") ||
get("a[href^='/Warehouse/Add']") ||
get("a[href*='/Pharmacy/Prepare']") ||
get("a[href^='/Pharmacy/ProduceAll']") ||
get("a[href*='/Pharmacy/CreateDrug']");
if (l) l.click();
else
get("ul.action_list a[href^='/AutoPark']").click();
}
else if(autopark) {
// автопарк
let l = get("a[href^='/AutoPark/TipWay']") ||
get("a[href^='/AutoPark/TipWay']") ||
get("a[href^='/AutoFuel/Refill']") ||
get("a[href^='/AutoPark/Threat']") ||
get("a[href^='/AutoPark/Examine']");
if (l) l.click();
else if (asm == "random" && get("a[href^='/AutoParkDestination/SendAll?onEvent=False']") != null)
get("a[href^='/AutoParkDestination/SendAll?onEvent=False']").click();
else if (asm == "random" && get("a[href^='/AutoParkDestination/SendAllConfirm?']") != null)
get("a[href^='/AutoParkDestination/SendAllConfirm?']").click();
else if (asm == "event" && get("a[href^='/AutoParkDestination/SendAll?onEvent=true']") != null)
get("a[href^='/AutoParkDestination/SendAll?onEvent=true']").click();
else if (asm == "value") {
if (get("a[href^='/AutoParkDestination?garageIndex']") != null)
get("a[href^='/AutoParkDestination?garageIndex']").click();
else if (get("a[href^='/AutoPark?page=']") != null) {
let url = new URL(location.href);
let urlp = 1;
if (url.searchParams.has("page"))
urlp = parseInt(url.searchParams.get("page"));
if (get("a[href^='/AutoPark?page=" + (urlp + 1) + "']") != null)
get("a[href^='/AutoPark?page=" + (urlp + 1) + "']").click();
else restartTimeout();
}
}
else
restartTimeout();
}
else if(autoParkDestination){
var $list=getAll("ul.delim-list.padtop_s li.padtop_s");
if(asm=="event" && get(".epidemic")!=null)
get(".epidemic a").click();
else{
var elem = 0;
arr = [];
arr2 = [];
if(asv=="random")
elem = ~~(Math.random() * $list.length);
else{
if(asp=="price"){
$list.forEach(el=>{
arr.push(parseInt(el.getAll("span.ylwtextb")[2].innerHTML.split(" ")[0]));
})
}
else if(asp=="time"){
$list.forEach(el=>{
let t = el.getAll("span.ylwtextb")[0].innerHTML;
if(/(\d+) д./.test(t)){
let m = t.match(/(\d+) д./);
arr.push(parseInt(m[1]) * 24 * 60);
}
else if(/(\d+) д. (\d+) ч./.test(t)){
let m = t.match(/(\d+) д. (\d+) ч./);
arr.push(parseInt(m[1]) * 24 * 60 + parseInt(m[2]) * 60);
}
else if(/(\d+) ч./.test(t)){
let m = t.match(/(\d+) ч./);
arr.push(parseInt(m[1]) * 60);
}
else if(/(\d+) мин./.test(t)){
let m = t.match(/(\d+) мин./);
arr.push(parseInt(m[1]));
}
else if(/(\d+) ч. (\d+) мин./.test(t)){
let m = t.match(/(\d+) ч. (\d+) мин./);
arr.push(parseInt(m[1]) * 60 + parseInt(m[2]));
}
})
}
else if(asp=="exp"){
$list.forEach(el=>{
arr.push(parseInt(el.getAll("span.ylwtextb")[1].innerHTML));
})
}
arr2 = [...arr];
arr.sort((a, b) => a - b);
elem = arr2.indexOf(arr[asv == "min" ? 0 : arr.length - 1]);
}
$list[elem].get("a").click();
}
}
}
function questsRun(){
let l = get("a[href^='/Quests/Begin?']") ||
get("a[href^='/Quests/ChestsContinue?']") ||
get("a[href^='/Quests/SaveInjured?']") ||
get("a[href^='/Quests/StudyComplete?']") ||
get("a[href^='/Quests/StudyEnd?']") ||
get("a[href^='/Quests/Study?']") ||
get("a[href^='/Quests/End?']");
if (l)
setTimeout(() => l.click(), questsShortTimeout * 1000);
else if (get("a[href^='/Quests?']"))
setTimeout(() => get("a[href^='/Quests?']").click(), questsLongTimeout * 60000);
}
function vetClinicRun() {
setTimeout(() => {
let l = get("a[href^='/VetClinic/GetPet?'") ||
get("a[href^='/VetClinic/Diagnose?'") ||
get("a[href^='/VetClinic/Done?'") ||
get("a[href^='/VetClinic/Treat?'");
if (l) l.click();
else {
setTimeout(() => {
get("ul.action_list a[href^='/VetClinic']").click();
}, questsLongTimeout * 60000 * 2)
}
}, questsShortTimeout * 1000);
}
function chestsRun() {
let c = +get("b.underwater-treasuresScore-value").innerText;
if (c > 200) {
get("a.btn[href^='/Underwater/CollectKeys']").click();
}
else location.reload();
}
drawSets()
if (localStorage.getItem("firstrun") == null) {
localStorage.setItem("firstrun", "false");
document.body.insertAdjacentHTML("afterBegin", splash);
pagesTimeout = setTimeout(patientsRun, pagesRunIntervalTime);
}
if (mainPages) {
if (localStorage.getItem("firstrun") != null && localStorage.getItem("firstrun") == "false")
pagesTimeout = setTimeout(patientsRun, pagesRunIntervalTime);
}
else if (quests) questsRun();
else if (vetclinic) vetClinicRun();
else if (chests) chestsRun();