HaremHeroes Automatic

Open the menu in HaremHeroes(topright) to toggle AutoControlls. Supports AutoSalary, AutoContest, AutoMission, AutoQuest, AutoTrollBattle, AutoArenaBattle and AutoPachinko(Free). Messages are printed in local console.

  1. // ==UserScript==
  2. // @name HaremHeroes Automatic
  3. // @namespace JDscripts
  4. // @version 3.24
  5. // @description Open the menu in HaremHeroes(topright) to toggle AutoControlls. Supports AutoSalary, AutoContest, AutoMission, AutoQuest, AutoTrollBattle, AutoArenaBattle and AutoPachinko(Free). Messages are printed in local console.
  6. // @author JD
  7. // @match http*://nutaku.haremheroes.com/*
  8. // @match http*://*.hentaiheroes.com/*
  9. // @require https://cdn.jsdelivr.net/js-cookie/2.2.0/js.cookie.js
  10. // @grant GM_addStyle
  11. // @license MIT
  12. // ==/UserScript==
  13.  
  14. GM_addStyle('/* The switch - the box around the slider */ .switch { position: relative; display: inline-block; width: 60px; height: 34px; } /* Hide default HTML checkbox */ .switch input {display:none;} /* The slider */ .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #ccc; -webkit-transition: .4s; transition: .4s; } .slider:before { position: absolute; content: ""; height: 26px; width: 26px; left: 4px; bottom: 4px; background-color: white; -webkit-transition: .4s; transition: .4s; } input:checked + .slider { background-color: #2196F3; } input:focus + .slider { box-shadow: 0 0 1px #2196F3; } input:checked + .slider:before { -webkit-transform: translateX(26px); -ms-transform: translateX(26px); transform: translateX(26px); } /* Rounded sliders */ .slider.round { border-radius: 34px; } .slider.round:before { border-radius: 50%; }');
  15.  
  16. var globalWorldMap;
  17.  
  18. function getHero()
  19. {
  20. if(unsafeWindow.Hero === undefined)
  21. {
  22. setTimeout(autoLoop, Number(sessionStorage.autoLoopTimeMili))
  23. //console.log(window.wrappedJSObject)
  24. }
  25. return unsafeWindow.Hero;
  26. }
  27.  
  28. function getGirlsMap()
  29. {
  30. return unsafeWindow.GirlSalaryManager.girlsMap;
  31. }
  32.  
  33. function getPage()
  34. {
  35. try{
  36. var ob = document.getElementById("hh_nutaku");
  37. if(ob===undefined || ob === null)
  38. {
  39. ob = document.getElementById("hh_hentai");
  40. }
  41. return ob.className.match(/.*page-(.*) .*/i)[1];
  42. }
  43. catch(err)
  44. {
  45. return ""
  46. }
  47. }
  48.  
  49. // Retruns true if on correct page.
  50. function gotoPage(page)
  51. {
  52. if(getPage() === page)
  53. {
  54. return true;
  55. }
  56. else
  57. {
  58. console.log("Navigating to page: "+page);
  59. var togoto = undefined;
  60. // get page path
  61. switch(page)
  62. {
  63. case "missions":
  64. case "activities":
  65. togoto = $("nav div[rel='content'] a:has(.activities)").attr("href");
  66. break;
  67. case "harem":
  68. togoto = $("nav div[rel='content'] a:has(.harem)").attr("href");
  69. break;
  70. case "map":
  71. togoto = $("nav div[rel='content'] a:has(.map)").attr("href");
  72. break;
  73. case "arena":
  74. togoto = $("nav div[rel='content'] a:has(.battle)").attr("href");
  75. break;
  76. case "pachinko":
  77. togoto = $("nav div[rel='content'] a:has(.pachinko)").attr("href");
  78. break;
  79. case "quest":
  80. togoto = getHero().infos.questing.current_url;
  81. console.log("Current quest page: "+togoto);
  82. break;
  83. default:
  84. console.log("Unknown goto page request. No page \'"+page+"\' defined.");
  85. }
  86. if(togoto != undefined)
  87. {
  88. sessionStorage.autoLoop = "false";
  89. window.location = window.location.origin + togoto;
  90. }
  91. else console.log("Couldn't find page path. Page was undefined...");
  92. return false;
  93. }
  94. }
  95.  
  96. var proceedQuest = function () {
  97. //console.log("Starting auto quest.");
  98. // Check if at correct page.
  99. if (!gotoPage("quest")) {
  100. // Click on current quest to naviagte to it.
  101. console.log("Navigating to current quest.");
  102. return;
  103. }
  104.  
  105. // Get the proceed button type
  106. var proceedButtonMatch = $("#controls button:not([style='display: none;'])");
  107. var proceedCostEnergy = Number($("#controls .cost span[cur='*']").text());
  108. var proceedCostMoney = Number($("#controls .cost span[cur='$']").text().trim().replace(',', ''));
  109. var proceedType = proceedButtonMatch.attr("act");
  110.  
  111. if (proceedButtonMatch.length === 0) console.log("Could not find resume button.");
  112. else if (proceedType === "free") {
  113. console.log("Proceeding for free.");
  114. proceedButtonMatch.click();
  115. }
  116. else if (proceedType === "pay") {
  117. var energyCurrent = getHero().infos.energy_quest;
  118. var moneyCurrent = getHero().infos.soft_currency;
  119. if(proceedCostEnergy <= energyCurrent)
  120. {
  121. // We have energy.
  122. console.log("Spending "+proceedCostEnergy+" Energy to proceed.");
  123. }
  124. else
  125. {
  126. console.log("Quest requires "+proceedCostEnergy+" Energy to proceed.");
  127. sessionStorage.questRequirement = "*"+proceedCostEnergy;
  128. return;
  129. }
  130. if(proceedCostMoney <= moneyCurrent)
  131. {
  132. // We have money.
  133. console.log("Spending "+proceedCostMoney+" Money to proceed.");
  134. }
  135. else
  136. {
  137. console.log("Spending "+proceedCostEnergy+" Money to proceed.");
  138. sessionStorage.questRequirement = "$"+proceedCostMoney;
  139. return;
  140. }
  141. proceedButtonMatch.click();
  142. sessionStorage.autoLoop = "false";
  143. location.reload();
  144. }
  145. else if (proceedType === "use_item") {
  146. console.log("Proceeding by using X" + Number($("#controls .item span").text()) + " of the required item.");
  147. proceedButtonMatch.click();
  148. }
  149. else if (proceedType === "battle") {
  150. console.log("Proceeding to battle troll...");
  151. sessionStorage.questRequirement = "battle";
  152. // Proceed to battle troll.
  153. proceedButtonMatch.click();
  154. sessionStorage.autoLoop = "false";
  155. location.reload();
  156. }
  157. else if (proceedType === "end_archive") {
  158. console.log("Reached end of current archive. Proceeding to next archive.");
  159. sessionStorage.autoLoop = "false";
  160. proceedButtonMatch.click();
  161. }
  162. else if (proceedType === "end_play") {
  163. console.log("Reached end of current play. Proceeding to next play.");
  164. sessionStorage.autoLoop = "false";
  165. proceedButtonMatch.click();
  166. }
  167. else {
  168. console.log("Could not identify given resume button.");
  169. sessionStorage.questRequirement = "unknownQuestButton";
  170. }
  171. };
  172.  
  173. /**
  174. * Recieves a list of mission objects and returns the mission object to use.
  175. * A mission object looks similar to this :-
  176. * Eg 1: {"id_member_mission":"256160093","id_mission":"23","duration":"53","cost":"1","remaining_time":"-83057","rewards":[{"classList":{"0":"slot","1":"slot_xp"},"type":"xp","data":28},{"classList":{"0":"slot","1":"slot_SC"},"type":"money","data":277}]}
  177. * Eg 2: {"id_member_mission":"256160095","id_mission":"10","duration":"53","cost":"1","remaining_time":"-81330","rewards":[{"classList":{"0":"slot","1":"slot_xp"},"type":"xp","data":28},{"classList":{"0":"slot","1":"rare"},"type":"item","data":{"id_item":"23","type":"gift","subtype":"0","identifier":"K3","rarity":"rare","value":"80","carac1":0,"carac2":0,"carac3":0,"endurance":0,"chance":0,"ego":0,"damage":0,"duration":0,"level_mini":"1","name":"Bracelet","Name":"Bracelet","ico":"https://content.haremheroes.com/pictures/items/K3.png","price_sell":5561,"count":1,"id_m_i":[]}}]}
  178. * Eg 3: {"id_member_mission":"256822795","id_mission":"337","duration":"17172","cost":"144","remaining_time":null,"remaining_cost":"144","rewards":[{"classList":{"0":"slot","1":"slot_HC"},"type":"koban","data":11}]}
  179. * Eg 1 has mission rewards of xp and money.
  180. * Eg 2 has mission rewards of xp and item.
  181. * Eg 3 has mission rewards of koban/hard_currency.
  182. * cost is the koban price for instant complete.
  183. */
  184. function getSuitableMission(missionsList)
  185. {
  186. var msn = missionsList[0];
  187. for(var m in missionsList)
  188. {
  189. if(Number(msn.duration) > Number(missionsList[m].duration))
  190. {
  191. msn = missionsList[m];
  192. }
  193. }
  194. return msn;
  195. }
  196.  
  197. // retruns a map of WorldName->fullLink/href
  198. // else returns false if navigating there
  199. function getUnlockedWorlds()
  200. {
  201. if(gotoPage("map"))
  202. {
  203. var map = new Map();
  204. // fill the map
  205. document.querySelectorAll("a.link-world").forEach(
  206. function(elem)
  207. {
  208. var name = elem.nextElementSibling.textContent.trim();
  209. map.set(name, elem.href);
  210. }
  211. );
  212. // set global latest map
  213. sessionStorage.globalWorldMap = JSON.stringify([...map]);
  214. globalWorldMap = map;
  215. //return not busy
  216. return map;
  217. }
  218. else
  219. {
  220. console.log("Navigating to worldmap.");
  221. // not done
  222. return false;
  223. }
  224. }
  225.  
  226. // returns boolean to set busy
  227. function doMissionStuff()
  228. {
  229. if(!gotoPage("missions"))
  230. {
  231. console.log("Navigating to activities page.");
  232. // return busy
  233. return true;
  234. }
  235. else
  236. {
  237. console.log("On activities page.");
  238. console.log("Collecting finished mission's reward.");
  239. $(".mission_button button:visible[rel='claim']").click();
  240. // TODO: select new missions and parse reward data from HTML, it's there in data attributes of tags
  241. var missions = [];
  242. var allGood = true;
  243. // parse missions
  244. $(".mission_object").each(function(idx,missionObject){
  245. var data = $.data(missionObject).d;
  246. // Do not list completed missions
  247. if(data.remaining_time !== null){
  248. // This is not a fresh mission
  249. if(data.remaining_time > 0)
  250. {
  251. console.log("Unfinished mission detected...("+data.remaining_time+"sec. remaining)");
  252. Cookies.set('nextMissionTime',data.remaining_time,{expires:new Date(new Date().getTime() + data.remaining_time * 1000)});
  253. }
  254. else{
  255. console.log("Unclaimed mission detected...");
  256. }
  257. allGood = false;
  258. return;
  259. }
  260. data.missionObject = missionObject;
  261. var rewards = [];
  262. // set rewards
  263. {
  264. // get Reward slots
  265. var slots = missionObject.querySelectorAll(".slot");
  266. // traverse slots
  267. $.each(slots,function(idx,slotDiv){
  268. var reward = {};
  269. // get slot class list
  270. reward.classList = slotDiv.classList;
  271. // set reward type
  272. if(reward.classList.contains("slot_xp"))reward.type = "xp";
  273. else if(reward.classList.contains("slot_SC"))reward.type = "money";
  274. else if(reward.classList.contains("slot_HC"))reward.type = "koban";
  275. else reward.type = "item";
  276. // set value if xp
  277. if(reward.type === "xp" || reward.type === "money" || reward.type === "koban")
  278. {
  279. // remove all non numbers and HTML tags
  280. try{
  281. reward.data = Number(slotDiv.innerHTML.replace(/<.*?>/g,'').replace(/\D/g,''));
  282. }
  283. catch(e){
  284. console.log("Couldn't parse xp/money data.");
  285. console.log(slotDiv);
  286. }
  287. }
  288. // set item details if item
  289. else if(reward.type === "item")
  290. {
  291. try{
  292. reward.data = $.data(slotDiv).d;
  293. }
  294. catch(e){
  295. console.log("Couldn't parse item reward slot details.");
  296. console.log(slotDiv);
  297. reward.type = "unknown";
  298. }
  299. }
  300. rewards.push(reward);
  301. });
  302. }
  303. data.rewards = rewards;
  304. missions.push(data);
  305. });
  306. if(!allGood){
  307. console.log("Something went wrong, need to retry later...");
  308. // busy
  309. return true;
  310. }
  311. console.log("Missions parsed, mission list is:-");
  312. console.log(missions);
  313. if(missions.length > 0)
  314. {
  315. console.log("Selecting mission from list.");
  316. var mission = getSuitableMission(missions);
  317. console.log("Selected mission:-");
  318. console.log(mission);
  319. console.log("Selected mission duration: "+mission.duration+"sec.");
  320. var missionButton = $(mission.missionObject).find("button:visible").first();
  321. console.log("Mission button of type: "+missionButton.attr("rel"));
  322. console.log("Clicking mission button.");
  323. missionButton.click();
  324. Cookies.set('nextMissionTime',mission.duration,{expires:new Date(new Date().getTime() + mission.duration * 1000)});
  325. }
  326. else{
  327. console.log("No missions detected...!");
  328. // get gift
  329. var ck = Cookies.get('nextMissionTime');
  330. var isAfterGift = document.querySelector("#missions .after_gift").style.display === 'block';
  331. if(!isAfterGift){
  332. if(ck === undefined || ck === 'giftleft')
  333. {
  334. console.log("Collecting gift.");
  335. // click button
  336. document.querySelector(".end_gift button").click();
  337. }
  338. else{
  339. console.log("Refreshing to collect gift...");
  340. Cookies.set('nextMissionTime','giftleft');
  341. window.reload();
  342. // is busy
  343. return true;
  344. }
  345. }
  346. var time = 0;
  347. for(var e in unsafeWindow.HHTimers.timers){
  348. try{if(unsafeWindow.HHTimers.timers[e].$elm.selector.includes("#missions_counter"))
  349. time=unsafeWindow.HHTimers.timers[e];
  350. }
  351. catch(e){}
  352. }
  353. time = time.remainingTime;
  354. if(time === undefined)
  355. {
  356. //try again with different selector
  357. for(e in unsafeWindow.HHTimers.timers){
  358. try{if(unsafeWindow.HHTimers.timers[e].$elm.selector.includes(".after_gift"))
  359. time=unsafeWindow.HHTimers.timers[e];
  360. }
  361. catch(e){}
  362. }
  363. time = time.remainingTime;
  364. }
  365. if(time === undefined){
  366. console.log("New mission time was undefined... Setting it manually to 10min.");
  367. time = 10*60;
  368. }
  369. console.log("New missions in: "+time+"sec.");
  370. Cookies.set('nextMissionTime',time,{expires:new Date(new Date().getTime() + time * 1000)});
  371. }
  372. // not busy
  373. return false;
  374. }
  375. }
  376.  
  377. // returns boolean to set busy
  378. function doContestStuff()
  379. {
  380. if(!gotoPage("missions"))
  381. {
  382. console.log("Navigating to activities page.");
  383. // return busy
  384. return true;
  385. }
  386. else
  387. {
  388. console.log("On activities page.");
  389. console.log("Collecting finished contests's reward.");
  390. $(".contest .ended button[rel='claim']").click();
  391. // need to get next contest timer data
  392. var time = 0;
  393. for(var e in unsafeWindow.HHTimers.timers){
  394. try{if(unsafeWindow.HHTimers.timers[e].$elm.selector.includes(".contest_timer"))
  395. time=unsafeWindow.HHTimers.timers[e];
  396. }
  397. catch(e){}
  398. }
  399. time = time.remainingTime;
  400. try{if(time === undefined)
  401. {
  402. //try again with different selector
  403. time = undefined;
  404. for(e in unsafeWindow.HHTimers.timers){
  405. if(unsafeWindow.HHTimers.timers[e].$elm[0].className.includes("contest_timer"))
  406. // get closest time
  407. if(!(unsafeWindow.HHTimers.timers[e].remainingTime>time))
  408. time=unsafeWindow.HHTimers.timers[e].remainingTime;
  409. }
  410. }}catch(e){}
  411. if(time === undefined){
  412. console.log("New contest time was undefined... Setting it manually to 10min.");
  413. time = 10*60;
  414. }
  415. Cookies.set('nextContestTime',time,{expires:new Date(new Date().getTime() + time * 1000)});
  416. console.log("Next contest time stored in nextContestTime cookie.(+" + time + " sec.)");
  417. // Not busy
  418. return false;
  419. }
  420. }
  421.  
  422. var getSalary = function () {
  423. try {
  424. if(!gotoPage("harem"))
  425. {
  426. // Not at Harem screen then goto the Harem screen.
  427. console.log("Navigating to Harem window.");
  428. // return busy
  429. return true;
  430. }
  431. else {
  432. console.log("Detected Harem Screen. Fetching Salary");
  433. $("#harem_whole #harem_left .salary:not('.loads') button").each(function (index) {
  434. $(this).click();
  435. });
  436. console.log("Salary fetched. Getting next fetch time");
  437. // In seconds
  438. var closestTime = undefined;
  439. var gMap = getGirlsMap();
  440. if(gMap === undefined)
  441. {
  442. // error
  443. console.log("Girls Map was undefined...! Error, manually setting salary time to 2 min.");
  444. closestTime = 2*60;
  445. }
  446. else
  447. {
  448. try{
  449. // Calc. closest time
  450. for(var key in gMap)
  451. {
  452. // undefined comparision is always false so first iteration is false, hence the not(!)
  453. if(!(closestTime<gMap[key].gData.pay_in))
  454. {
  455. closestTime = gMap[key].gData.pay_in;
  456. }
  457. }
  458. }
  459. catch(exp){
  460. // error
  461. console.log("Girls Map had undefined property...! Error, manually setting salary time to 2 min.");
  462. closestTime = 2*60;
  463. }
  464. }
  465. if(closestTime === undefined)
  466. {
  467. console.log("closestTime was undefined...! Error, manually setting salary time to 2 min.");
  468. closestTime = 2*60;
  469. }
  470. if(closestTime <= 2)
  471. {
  472. console.log("closestTime is less than/equal to 2 sec. Staying on this page.");
  473. // return busy
  474. return true;
  475. }
  476. Cookies.set('nextSalaryTime',closestTime,{expires:new Date(new Date().getTime() + closestTime * 1000)});
  477. console.log("New fetch time stored in nextSalaryTime cookie.(+" + closestTime + " sec.)");
  478. // return not busy
  479. return false;
  480. }
  481. }
  482. catch (ex) {
  483. console.log("Could not collect salary... " + ex);
  484. // return not busy
  485. return false;
  486. }
  487. };
  488.  
  489. var doBossBattle = function()
  490. {
  491. var currentPower = getHero().infos.energy_fight;
  492. if(currentPower < 1)
  493. {
  494. //console.log("No power for battle.");
  495. return;
  496. }
  497. // Battles the latest boss.
  498. // Navigate to latest boss.
  499. if(window.location.pathname.startsWith("/battle.html"))
  500. {
  501. // On the battle screen.
  502. doBattle();
  503. }
  504. else if(window.location.pathname.startsWith("/world"))
  505. {
  506. // On some world screen.
  507. // Click on the local Boss's battle button.
  508. console.log("Entering battle with this troll.");
  509. sessionStorage.autoLoop = "false";
  510. window.location = window.location.origin + $("#worldmap a[class='troll_world']").attr("href");
  511. return;
  512. }
  513. else if(sessionStorage.trollToFight !== undefined && sessionStorage.trollToFight !== "latest")
  514. {
  515. console.log("Custom troll fight.");
  516. console.log("Fighting troll at: "+sessionStorage.trollToFight);
  517. var worldMap = getUnlockedWorlds();
  518. if(worldMap === false)
  519. {
  520. console.log("Finding unlocked worlds...");
  521. return;
  522. }
  523. var worldPage = worldMap.get(sessionStorage.trollToFight);
  524. if(worldPage == undefined)
  525. {
  526. console.log("ERROR! worldPage undefined for troll at: "+sessionStorage.trollToFight);
  527. return;
  528. }
  529. console.log("Troll world at: "+worldPage);
  530. window.location = worldPage;
  531. sessionStorage.autoLoop = "false";
  532. }
  533. else if(window.location.pathname.startsWith("/quest"))
  534. {
  535. // On some quest screen.
  536. // Goto this area's screen.
  537. console.log("Navigating to latest Troll.");
  538. sessionStorage.autoLoop = "false";
  539. window.location = window.location.origin + $("#breadcrumbs a[class='back']").last().attr("href");
  540. return;
  541. }
  542. else{
  543. console.log("Navigating to latest Troll.");
  544. sessionStorage.autoLoop = "false";
  545. window.location = window.location.origin + $("nav div[rel='content'] a:has(.continue_quest)").attr("href");
  546. return;
  547. }
  548. };
  549.  
  550. var doBattle = function () {
  551. //console.log("Performing auto battle.");
  552. // Confirm if on correct screen.
  553. var page = getPage();
  554. if(page === "arena")
  555. {
  556. if ($("#arena[class='canvas']").length === 1) {
  557. // Oponent choose screen
  558. console.log("On opponent choose screen.");
  559. if(document.getElementById("popups").style.display === "block")
  560. {
  561. console.log("Popup detetcted. Refresh page.");
  562. unsafeWindow.reload();
  563. return;
  564. }
  565. else{
  566. console.log("No popups.");
  567. }
  568. // Fight the first opponent in list.
  569. var selbutton = $(".opponents_arena .sub_block button:contains('Select')");
  570. if(selbutton.length<1)
  571. {
  572. console.log("No arena opponents found, storing nextArenaTime...")
  573. var arenatime = 0;
  574. for(var e in unsafeWindow.HHTimers.timers){
  575. try{
  576. if(unsafeWindow.HHTimers.timers[e].$elm.selector.startsWith(".arena_refresh_counter"))
  577. arenatime=unsafeWindow.HHTimers.timers[e];
  578. }
  579. catch(e){}
  580. }
  581. arenatime = arenatime.remainingTime;
  582. Cookies.set('nextArenaTime',arenatime,{expires:new Date(new Date().getTime() + arenatime * 1000)});
  583. console.log("New arena time stored in nextArenaTime cookie.(+" + arenatime + " sec.)");
  584. return;
  585. }
  586. selbutton[0].click();
  587. sessionStorage.autoLoop = "false";
  588. }
  589. }
  590. else if (page === "battle") {
  591. // On battle page.
  592. //console.log("On Battle Page.");
  593. if ($("#battle[class='canvas']").length === 1) {
  594. // Battle screen
  595. console.log("On battle screen.");
  596. // get button with no autofight, i.e. no koban
  597. var battleButton = $("#battle_middle button[rel='launch']:not(.autofight)");
  598. var currentPower = getHero().infos.energy_fight;
  599. if(battleButton === undefined){
  600. console.log("Battle Button was undefined. Disabling all auto-battle.");
  601. document.getElementById("autoBattleCheckbox").checked = false;
  602. document.getElementById("autoArenaCheckbox").checked = false;
  603. if (sessionStorage.questRequirement === "battle")
  604. {
  605. document.getElementById("autoQuestCheckbox").checked = false;
  606. console.log("Auto-quest disabled since it requires battle and auto-battle has errors.");
  607. }
  608. return;
  609. }
  610. var battle_price = battleButton.attr("price_fe");
  611. if(battle_price === undefined){
  612. console.log("Could not detect battle button price. Error.");
  613. console.log("Disabling all auto-battle.");
  614. document.getElementById("autoBattleCheckbox").checked = false;
  615. document.getElementById("autoArenaCheckbox").checked = false;
  616. if (sessionStorage.questRequirement === "battle")
  617. {
  618. document.getElementById("autoQuestCheckbox").checked = false;
  619. console.log("Auto-quest disabled since it requires battle and auto-battle has errors.");
  620. }
  621. return;
  622. }
  623. console.log("battle price: "+battle_price+"P")
  624. if(currentPower >= battle_price)
  625. {
  626. // We have the power.
  627. battleButton.click();
  628. // Skip
  629. setTimeout(function(){$("#battle_middle button[rel='skip']").click();},1000);
  630. setTimeout(function(){$("#battle_end div[style*='display: block;'] .blue_text_button").click();},2500);
  631.  
  632. if (sessionStorage.questRequirement === "battle") {
  633. // Battle Done.
  634. sessionStorage.questRequirement = "none";
  635. }
  636. }
  637. else
  638. {
  639. // We need more power.
  640. console.log("Battle requires "+battle_price+" power.");
  641. sessionStorage.battlePowerRequired = battle_price;
  642. if(sessionStorage.questRequirement === "battle")sessionStorage.questRequirement = "P"+battle_price;
  643. }
  644. }
  645. else {
  646. console.log("Could not identify battle screen.");
  647. if (sessionStorage.questRequirement === "battle") sessionStorage.questRequirement = "errorInAutoBattle";
  648. return;
  649. }
  650. }
  651. else
  652. {
  653. // Switch to the correct screen
  654. console.log("Switching to battle screen.");
  655. gotoPage("arena");
  656. return;
  657. }
  658. };
  659.  
  660. var updateData = function () {
  661. //console.log("updating UI");
  662. var trollOptions = document.getElementById("autoTrollSelector");
  663. sessionStorage.autoTrollSelectedIndex = trollOptions.selectedIndex;
  664. sessionStorage.trollToFight = trollOptions.value;
  665. sessionStorage.autoSalary = document.getElementById("autoSalaryCheckbox").checked;
  666. sessionStorage.autoContest = document.getElementById("autoContestCheckbox").checked;
  667. sessionStorage.autoMission = document.getElementById("autoMissionCheckbox").checked;
  668. sessionStorage.autoQuest = document.getElementById("autoQuestCheckbox").checked;
  669. sessionStorage.autoTrollBattle = document.getElementById("autoBattleCheckbox").checked;
  670. sessionStorage.autoArenaBattle = document.getElementById("autoArenaCheckbox").checked;
  671. sessionStorage.autoFreePachinko = document.getElementById("autoFreePachinko").checked;
  672. };
  673.  
  674. var getPachinko = function(){
  675. try {
  676. if(!gotoPage("pachinko"))
  677. {
  678. // Not at Pachinko screen then goto the Pachinko screen.
  679. console.log("Navigating to Pachinko window.");
  680. return;
  681. }
  682. else {
  683. console.log("Detected Pachinko Screen. Fetching Pachinko");
  684. $("#pachinko button[free=1]")[0].click();
  685. var npach;
  686. for(var e in unsafeWindow.HHTimers.timersListMin){
  687. if(unsafeWindow.HHTimers.timersListMin[e].$elm.selector.startsWith(".pachinko_change"))
  688. npach=unsafeWindow.HHTimers.timersListMin[e].remainingTime;
  689. }
  690. if(npach !== undefined || npach !== 0)
  691. {
  692. Cookies.set('nextPachinkoTime',npach,{expires:new Date(new Date().getTime() + npach * 1000)});
  693. }
  694. else
  695. {
  696. Cookies.remove('nextPachinkoTime');
  697. }
  698. }
  699. }
  700. catch (ex) {
  701. console.log("Could not collect pachinko... " + ex);
  702. }
  703. };
  704.  
  705. var autoLoop = function () {
  706. updateData();
  707. var busy = false;
  708. var page = window.location.href;
  709. var currentPower = getHero().infos.energy_fight;
  710. //console.log("sal="+sessionStorage.autoSalary);
  711. if(sessionStorage.autoFreePachinko === "true" && busy === false){
  712. // Navigate to pachinko
  713. if (Cookies.get("nextPachinkoTime") === undefined) {
  714. console.log("Time to fetch Pachinko.");
  715. getPachinko();
  716. busy = true;
  717. }
  718. }
  719. if(sessionStorage.autoContest === "true" && busy === false){
  720. if (Cookies.get("nextContestTime") === undefined){
  721. console.log("Time to get contest rewards.");
  722. busy = doContestStuff();
  723. }
  724. }
  725. if(sessionStorage.autoMission === "true" && busy === false){
  726. if (Cookies.get("nextMissionTime") === undefined){
  727. console.log("Time to do missions.");
  728. busy = doMissionStuff();
  729. }
  730. }
  731. if (sessionStorage.autoSalary === "true" && busy === false) {
  732. if (Cookies.get("nextSalaryTime") === undefined) {
  733. console.log("Time to fetch salary.");
  734. busy = getSalary();
  735. }
  736. }
  737. if (sessionStorage.autoQuest === "true" && busy === false) {
  738. if (sessionStorage.questRequirement === "battle") {
  739. console.log("Quest requires battle.");
  740. doBossBattle();
  741. busy = true;
  742. }
  743. else if (sessionStorage.questRequirement[0] === '$') {
  744. if (Number(sessionStorage.questRequirement.substr(1)) < getHero().infos.soft_currency) {
  745. // We have enough money... requirement fulfilled.
  746. console.log("Continuing quest, required money obtained.");
  747. sessionStorage.questRequirement = "none";
  748. proceedQuest();
  749. busy = true;
  750. }
  751. else {
  752. if(isNaN(sessionStorage.questRequirement.substr(1)))
  753. {
  754. sessionStorage.questRequirement = "none";
  755. console.log("Invalid money in session storage quest requirement !");
  756. }
  757. else{
  758. // Else we need more money.
  759. //console.log("Need money for quest, cannot continue. Turning ON AutoSalary.");
  760. sessionStorage.autoQuest = "true";
  761. }
  762. busy = false;
  763. }
  764. }
  765. else if (sessionStorage.questRequirement[0] === '*') {
  766. var energyNeeded = Number(sessionStorage.questRequirement.substr(1));
  767. var energyCurrent = getHero().infos.energy_quest;
  768. if (energyNeeded <= energyCurrent) {
  769. // We have enough energy... requirement fulfilled.
  770. console.log("Continuing quest, required energy obtained.");
  771. sessionStorage.questRequirement = "none";
  772. proceedQuest();
  773. busy = true;
  774. }
  775. // Else we need energy, just wait.
  776. else {
  777. busy = false;
  778. //console.log("Replenishing energy for quest.(" + energyNeeded + " needed)");
  779. }
  780. }
  781. else if (sessionStorage.questRequirement[0] === 'P')
  782. {
  783. // Battle power required.
  784. var neededPower = Number(sessionStorage.questRequirement.substr(1));
  785. if(currentPower < neededPower)
  786. {
  787. console.log("Quest requires "+neededPower+" Battle Power for advancement. Waiting...");
  788. busy = false;
  789. }
  790. else
  791. {
  792. console.log("Battle Power obtained, resuming quest...");
  793. sessionStorage.questRequirement = "none";
  794. proceedQuest();
  795. busy = true;
  796. }
  797. }
  798. else if (sessionStorage.questRequirement === "unknownQuestButton") {
  799. console.log("AutoQuest disabled.AutoQuest cannot be performed due to unknown quest button. Please manually proceed the current quest screen.");
  800. document.getElementById("autoQuestCheckbox").checked = false;
  801. sessionStorage.autoQuest = "false";
  802. sessionStorage.questRequirement = "none";
  803. busy = false;
  804. }
  805. else if (sessionStorage.questRequirement === "errorInAutoBattle") {
  806. console.log("AutoQuest disabled.AutoQuest cannot be performed due errors in AutoBattle. Please manually proceed the current quest screen.");
  807. document.getElementById("autoQuestCheckbox").checked = false;
  808. sessionStorage.autoQuest = "false";
  809. sessionStorage.questRequirement = "none";
  810. busy = false;
  811. }
  812. else if(sessionStorage.questRequirement === "none")
  813. {
  814. //console.log("NONE req.");
  815. busy = true;
  816. proceedQuest();
  817. }
  818. else
  819. {
  820. console.log("Invalid quest requirement : "+sessionStorage.questRequirement);
  821. busy=false;
  822. }
  823. }
  824. else if(sessionStorage.autoQuest === "false"){sessionStorage.questRequirement = "none";}
  825.  
  826. if(sessionStorage.autoArenaBattle === "true" && busy === false)
  827. {
  828. if(Cookies.get("nextArenaTime") === undefined)
  829. {
  830. console.log("Time to fight in arena.");
  831. doBattle();
  832. busy = true;
  833. }
  834. }
  835.  
  836. if(sessionStorage.autoTrollBattle === "true")
  837. {
  838. if(busy === false && currentPower >= Number(sessionStorage.battlePowerRequired) && currentPower > 0)
  839. {
  840. sessionStorage.battlePowerRequired = "0";
  841. busy = true;
  842. if(sessionStorage.autoQuest === "true")
  843. {
  844. if(sessionStorage.questRequirement[0] === 'P')
  845. {
  846. console.log("AutoBattle disabled for power collection for AutoQuest.");
  847. document.getElementById("autoBattleCheckbox").checked = false;
  848. sessionStorage.autoTrollBattle = "false";
  849. busy = false;
  850. }
  851. else
  852. {
  853. doBossBattle();
  854. }
  855. }
  856. else
  857. {
  858. doBossBattle();
  859. }
  860. }
  861. }
  862. else{sessionStorage.battlePowerRequired = "0";}
  863.  
  864. if(busy === true && sessionStorage.userLink==="none" && !window.location.pathname.startsWith("/quest"))
  865. {
  866. sessionStorage.userLink = page;
  867. }
  868. else if(sessionStorage.userLink !=="none" && busy === false)
  869. {
  870. console.log("Restoring page "+sessionStorage.userLink);
  871. window.location = sessionStorage.userLink;
  872. sessionStorage.userLink = "none";
  873. }
  874.  
  875. if(isNaN(sessionStorage.autoLoopTimeMili)){
  876. console.log("AutoLoopTimeMili is not a number.");
  877. setDefaults();
  878. }
  879. else{
  880. if (sessionStorage.autoLoop === "true") setTimeout(autoLoop, Number(sessionStorage.autoLoopTimeMili));
  881. else console.log("autoLoop Disabled");
  882. }
  883. };
  884.  
  885. var setDefaults = function () {
  886. console.log("Setting Defaults.");
  887. sessionStorage.autoSalary = "false";
  888. sessionStorage.autoContest = "false";
  889. sessionStorage.autoMission = "false";
  890. sessionStorage.autoFreePachinko = "false";
  891. sessionStorage.autoLoop = "true";
  892. sessionStorage.userLink = "none";
  893. sessionStorage.autoLoopTimeMili = "200";
  894. sessionStorage.autoQuest = "false";
  895. sessionStorage.autoTrollBattle = "false";
  896. sessionStorage.autoArenaBattle = "false";
  897. sessionStorage.battlePowerRequired = "0";
  898. sessionStorage.questRequirement = "none";
  899. sessionStorage.freshStart = "no";
  900. };
  901.  
  902. var start = function () {
  903. //console.log("script started");
  904. // get world map
  905. try{
  906. globalWorldMap = new Map(JSON.parse(sessionStorage.globalWorldMap));
  907. if(!(globalWorldMap instanceof Map) || globalWorldMap.size<1)throw false;
  908. }
  909. catch(e)
  910. {
  911. console.log("Need world map NOW...");
  912. if(!getUnlockedWorlds())
  913. {
  914. sessionStorage.userLink = window.location.href;
  915. return;
  916. }
  917. }
  918. // Add UI buttons.
  919. var UIcontainer = $("#contains_all nav div[rel='content']");
  920. UIcontainer.html('<div style="position: absolute;right: 18.66%; padding: 10px;width: inherit;text-align: center;display:flex;flex-direction:column;">'
  921. + '<span>AutoSal.</span><div><label class=\"switch\"><input id=\"autoSalaryCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  922. + '<span>AutoContest</span><div><label class=\"switch\"><input id=\"autoContestCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  923. + '<span>AutoMission</span><div><label class=\"switch\"><input id=\"autoMissionCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  924. +'<span>AutoQuest</span><div><label class=\"switch\"><input id=\"autoQuestCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  925. +'<span>AutoTrollBattle</span><div>\
  926. <label class=\"switch\"><input id=\"autoBattleCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label>\
  927. <select id=\"autoTrollSelector\"><option value=\"latest\">Latest</option></select>\
  928. </div>'
  929. +'<span>AutoArenaBattle</span><div><label class=\"switch\"><input id=\"autoArenaCheckbox\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  930. +'<span>AutoPachinko(Free)</span><div><label class=\"switch\"><input id=\"autoFreePachinko\" type=\"checkbox\"><span class=\"slider round\"></span></label></div>'
  931. +'</div>'+UIcontainer.html());
  932. // Add auto troll options
  933. var trollOptions = document.getElementById("autoTrollSelector");
  934. globalWorldMap.forEach(function(val,key){
  935. var option = document.createElement("option");
  936. option.text = key;
  937. trollOptions.add(option);
  938. });
  939. trollOptions.selectedIndex = sessionStorage.autoTrollSelectedIndex;
  940. document.getElementById("autoSalaryCheckbox").checked = sessionStorage.autoSalary === "true";
  941. document.getElementById("autoContestCheckbox").checked = sessionStorage.autoContest === "true";
  942. document.getElementById("autoMissionCheckbox").checked = sessionStorage.autoMission === "true";
  943. document.getElementById("autoQuestCheckbox").checked = sessionStorage.autoQuest === "true";
  944. document.getElementById("autoBattleCheckbox").checked = sessionStorage.autoTrollBattle === "true";
  945. document.getElementById("autoArenaCheckbox").checked = sessionStorage.autoArenaBattle === "true";
  946. document.getElementById("autoFreePachinko").checked = sessionStorage.autoFreePachinko === "true";
  947. sessionStorage.autoLoop = "true";
  948. if (typeof sessionStorage.freshStart == "undefined" || isNaN(Number(sessionStorage.autoLoopTimeMili))) {
  949. setDefaults();
  950. }
  951. autoLoop();
  952. };
  953. $("document").ready(start);