Idleontoolbox Produce Check

Check and notify on https://idleontoolbox.com/dashboard with Telegram support

  1. // ==UserScript==
  2. // @name Idleontoolbox Produce Check
  3. // @namespace http://tampermonkey.net/
  4. // @version 13.2
  5. // @description Check and notify on https://idleontoolbox.com/dashboard with Telegram support
  6. // @author Tiande
  7. // @match https://idleontoolbox.com/*
  8. // @grant GM_notification
  9. // @grant GM_getValue
  10. // @grant GM_setValue
  11. // @license MIT
  12. // ==/UserScript==
  13. (function() {
  14. 'use strict';
  15.  
  16. var interval = 10 * 60 * 1000; // check status every 5m
  17. //var interval = 5000 // USE FOR TEST
  18. var refreshInterval = 999924.05 * 60 * 60 * 1000; // auto refresh to Dashboard every 10m
  19.  
  20. // Telegram Bot API Token
  21. var tgbotToken = 'YOUR BOT TOKEN';
  22. // Telegram user ID to send the message to
  23. var tguserId = 'YOUR USER ID';
  24.  
  25. // CHECK 1: lable text under player's name
  26. var whichFull = /(production is fullXXX|\s[1-5]\sminuteXXX)/i; // (worship is full|is full|being full|can equip sth|missing a trap|Crystal CD for Alchemy is 0.00%) etc.
  27. // CHECK 2: skill to notify
  28. var skill = /(Refinery000)/i; // (|printer|taste|cranium cooking|Arena|Birthday|Refinery) etc.
  29. // CHECK 3: bookcount to check
  30. var bookcount = 999;
  31. // CHECK 4: AFKTime notify
  32. var AFKHour = 999;
  33. var AFKMin = 30;
  34. // CHECK 5: banner notify. Must use img src
  35. var bannerRegex = /(blablabla)/i; //img-src
  36. //aria-label |plots reached the threshold|squirrel|shovel|fisheroo Reset|feather restart|library has 10[2-9]|bravery
  37. var bannerLabelRegex = /(worship is full|justice|Summoning familiar|Equinox bar is full|Max ball|already unlocked|kill ([1-9][0-9]?|0) more|ready to be built)/i;
  38. // exclude 3d printer circle check、happyhour、fishg etc.
  39. var excludeKeyword = /(printer|happy|building|fisheroo|feather|catch|trap|worship|fight)/i; //
  40. // worship overflow, pet
  41. var worship = 'Overflowing charge9999'; // edit as "Overflowing charge" for notify.
  42. var pet = 'Go claim!';
  43.  
  44. var notificationPermission = GM_getValue('notificationPermission');
  45. var isFunctionEnabled = true;
  46. var intervalId; // Store interval ID for pausing and resuming
  47.  
  48. // tg send
  49. function tgsendMessage(message) {
  50. var tgurl = 'https://api.telegram.org/bot' + tgbotToken + '/sendMessage';
  51. var tgparams = 'chat_id=' + tguserId + '&text=' + encodeURIComponent(message);
  52.  
  53. var tgxhr = new XMLHttpRequest();
  54. tgxhr.open('POST', tgurl, true);
  55. tgxhr.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
  56. tgxhr.onreadystatechange = function() {
  57. if (tgxhr.readyState == 4 && tgxhr.status == 200) {
  58. console.log('Message sent successfully!');
  59. }
  60. };
  61. tgxhr.onerror = function(error) {
  62. console.error('Error sending message:', error);
  63. };
  64. tgxhr.send(tgparams);
  65. }
  66.  
  67. // autorefresh time
  68. var refreshId;
  69. // refreshPage
  70. function refreshPage() {
  71. if (isFunctionEnabled) {
  72. location.href = 'https://idleontoolbox.com/dashboard'; // 刷新页面到 dashboard
  73. } else {
  74. // 如果不是,则等待下一个刷新时间间隔
  75. refreshId = setInterval(refreshPage, refreshInterval);
  76. }
  77. }
  78.  
  79. function toggleRefresh() {
  80. if (isFunctionEnabled) {
  81. intervalId = setInterval(startInterval, interval); // 启动定时器
  82. refreshId = setInterval(refreshPage, refreshInterval);
  83. } else {
  84. clearInterval(intervalId); // 清除定时器
  85. clearInterval(refreshId);
  86. }
  87. }
  88. toggleRefresh(); // 在脚本启动时调用一次,以确保定时器已启动
  89. // Preload audio
  90. var audio = new Audio();
  91. audio.src = 'https://github.com/Tiande/IdelonCheck/raw/main/iphonewake.wav';
  92.  
  93. // Add CSS styles
  94. var style = document.createElement('style');
  95. style.innerHTML = `
  96. #toggleButtonContainer {
  97. position: fixed;
  98. top: 50%;
  99. left: calc(50% + 100px);
  100. transform: translate(-50%, -50%);
  101. display: flex;
  102. align-items: center;
  103. padding: 5px;
  104. background: green; /* Green color for default enabled state */
  105. color: white;
  106. border-radius: 5px;
  107. font-family: Arial, sans-serif;
  108. font-size: 14px;
  109. box-shadow: 0px 0px 10px rgba(0, 0, 0, 0.3);
  110. z-index: 99999;
  111. user-select: none;
  112. cursor: move;
  113. }
  114. #toggleButtonContainer.off {
  115. background: red; /* Red color for disabled state */
  116. }
  117. #toggleButton {
  118. padding: 5px 10px;
  119. border: none;
  120. border-radius: 5px;
  121. background: transparent;
  122. cursor: pointer;
  123. outline: none;
  124. }
  125. .drag-handle {
  126. cursor: move;
  127. }
  128. #produceCheck {
  129. background-color: black;
  130. color: white;
  131. padding: 5px;
  132. border-radius: 5px;
  133. margin-right: 5px;
  134. }
  135. `;
  136. document.head.appendChild(style);
  137.  
  138. createToggleButton(); // Create toggle button on script execution
  139. startInterval(); // Start interval on script execution
  140. function createToggleButton() {
  141. var toggleButtonContainer = document.createElement('div');
  142. toggleButtonContainer.id = 'toggleButtonContainer';
  143. toggleButtonContainer.classList.add('drag-handle');
  144. var produceCheck = document.createElement('span');
  145. produceCheck.id = 'produceCheck';
  146. produceCheck.textContent = 'Produce Check: ';
  147. toggleButtonContainer.appendChild(produceCheck);
  148. var toggleButton = document.createElement('button');
  149. toggleButton.id = 'toggleButton';
  150. toggleButton.addEventListener('click', toggleFunction);
  151. toggleButtonContainer.appendChild(toggleButton);
  152. document.body.appendChild(toggleButtonContainer);
  153. updateButtonStyle();
  154. makeDraggable(toggleButtonContainer);
  155. }
  156.  
  157. function updateButtonStyle() {
  158. var toggleButtonContainer = document.getElementById('toggleButtonContainer');
  159. toggleButtonContainer.classList.toggle('off', !isFunctionEnabled);
  160. var toggleButton = document.getElementById('toggleButton');
  161. toggleButton.innerHTML = isFunctionEnabled ? 'ON': 'OFF';
  162. }
  163.  
  164. function toggleFunction() {
  165. isFunctionEnabled = !isFunctionEnabled;
  166. updateButtonStyle();
  167. toggleRefresh(); // 根据isFunctionEnabled的值启用或暂停定时器
  168. }
  169.  
  170. function startInterval() {
  171. // sth is full
  172. var matchedText = '';
  173. var prepareelements = document.querySelectorAll('img[aria-label]');
  174. var elements = Array.from(prepareelements).filter(function(element) {
  175. var ariaLabel = element.getAttribute('aria-label');
  176. var matched = ariaLabel.match(whichFull);
  177. if (matched !== null) {
  178. // 如果匹配到了,则将匹配项拼接到 matchedText 中
  179. matchedText += matched[0];
  180. // 如果还有下一个匹配项,则添加 '&'
  181. if (matched.index !== whichFull.lastIndex - matched[0].length) {
  182. matchedText += '&';
  183. }
  184. }
  185. // 返回原有的条件
  186. return whichFull.test(ariaLabel);
  187. });
  188.  
  189. if (elements.length > 0) {
  190. var messages = [];
  191. elements.forEach(function(element) {
  192. var parentDiv = element.closest('.MuiCardContent-root');
  193. if (parentDiv) {
  194. var roleNameElement = parentDiv.querySelector('.MuiTypography-root.MuiTypography-body1.css-9l3uo3');
  195. if (roleNameElement) {
  196. var roleName = roleNameElement.textContent.trim();
  197. var matchedText = ''; // 重新初始化 matchedText
  198. var matched = element.getAttribute('aria-label').match(whichFull);
  199. if (matched !== null) {
  200. matchedText = matched[0]; // 获取匹配文本
  201. }
  202. messages.push(roleName + ' : ' + matchedText);
  203. }
  204. }
  205. });
  206.  
  207. if (messages.length > 0) {
  208. var message = messages.join(', ');
  209. showNotification(message);
  210. audio.play();
  211. if (tguserId !== 'your user ID') {
  212. tgsendMessage(message);
  213. }
  214. }
  215. }
  216.  
  217. // skill is ready
  218. var matchedText = '';
  219. var prepareelements = document.querySelectorAll('img[aria-label]');
  220. var elements = Array.from(prepareelements).filter(function(element) {
  221. var ariaLabel = element.getAttribute('aria-label');
  222. var matched = ariaLabel.match(skill);
  223. if (matched !== null) {
  224. // 如果匹配到了,则将匹配项拼接到 matchedText 中
  225. matchedText += matched[0];
  226. // 如果还有下一个匹配项,则添加 '&'
  227. if (matched.index !== skill.lastIndex - matched[0].length) {
  228. matchedText += '&';
  229. }
  230. }
  231. // 返回原有的条件
  232. return skill.test(ariaLabel);
  233. });
  234.  
  235. if (elements.length > 0) {
  236. var messages = [];
  237. elements.forEach(function(element) {
  238. var parentDiv = element.closest('.MuiCardContent-root');
  239. if (parentDiv) {
  240. var roleNameElement = parentDiv.querySelector('.MuiTypography-root.MuiTypography-body1.css-9l3uo3');
  241. if (roleNameElement) {
  242. var roleName = roleNameElement.textContent.trim();
  243. var matchedText = ''; // 重新初始化 matchedText
  244. var matched = element.getAttribute('aria-label').match(skill);
  245. if (matched !== null) {
  246. matchedText = matched[0]; // 获取匹配文本
  247. }
  248. messages.push(roleName + ' : ' + matchedText);
  249. }
  250. }
  251. });
  252.  
  253. if (messages.length > 0) {
  254. var message = messages.join(', ');
  255. showNotification(message);
  256. audio.play();
  257. if (tguserId !== 'your user ID') {
  258. tgsendMessage(message);
  259. }
  260. }
  261. }
  262.  
  263. // 追踪另一组元素 constuction trap
  264. var timeElements = document.querySelectorAll('.MuiTypography-root.MuiTypography-inherit');
  265. var trackedContents = [];
  266. //var excludeKeyword = /(printer|happy|building|fisheroo|feather|catch|trap|worship|fight)/i; // 定义排除关键字的正则表达式
  267.  
  268. timeElements.forEach(function(timeElement) {
  269. var color = getComputedStyle(timeElement).color;
  270. var matchColor = color.match(/^rgb\((\d+), (\d+), (\d+)\)$/);
  271. var textContent = timeElement.textContent || '';
  272.  
  273. // 正则表达式匹配时间格式,要求小时和分钟均为“00”,秒为任意值
  274. var timePattern = /^00h:00m:\d{2}s$/;
  275.  
  276. // 检查颜色条件或文本内容中的 '-1d' 或时间格式
  277. if ((matchColor && matchColor[1] === '249' && matchColor[2] === '29' && matchColor[3] === '29') ||
  278. textContent.includes('-1d') ||
  279. timePattern.test(textContent)) {
  280.  
  281. var parentElement = timeElement.closest('.MuiStack-root');
  282. if (parentElement && parentElement.getAttribute('aria-label')) {
  283. var ariaLabel = parentElement.getAttribute('aria-label');
  284. var splitIndex = ariaLabel.indexOf(':');
  285. if (splitIndex !== -1) {
  286. var content = ariaLabel.substring(0, splitIndex).trim(); // Get content before ':' and trim spaces
  287. // 确保内容不包括 excludeKeyword
  288. if (!excludeKeyword.test(content)) { // 直接使用 excludeKeyword
  289. trackedContents.push(content);
  290. }
  291. }
  292. }
  293. }
  294. });
  295.  
  296. if (trackedContents.length > 0) {
  297. var message = trackedContents.join(', '); // 将数组内容连接成字符串,以逗号分隔
  298. showNotification(message);
  299. audio.play();
  300. if (tguserId !== 'your user ID') {
  301. tgsendMessage(message + ' is finished.');
  302. }
  303. }
  304.  
  305. // 追踪另一组元素 worship pet
  306. var timeElements = document.querySelectorAll('.MuiTypography-root.MuiTypography-body1');
  307. var trackedContents = [];
  308. //var excludeKeyword = /(printer|happy|building|fisheroo|feather|catch|trap|worship|fight)/i; // 排除关键字
  309.  
  310.  
  311. timeElements.forEach(function(timeElement) {
  312. var color = getComputedStyle(timeElement).color;
  313. var textContent = timeElement.textContent || '';
  314.  
  315. // 检查 textContent 中是否包含 'worship' 或 'pet'
  316. if (textContent.includes(worship) || textContent.includes(pet)) {
  317. var parentElement = timeElement.closest('.MuiStack-root');
  318. if (parentElement && parentElement.getAttribute('aria-label')) {
  319. var ariaLabel = parentElement.getAttribute('aria-label');
  320. var splitIndex = ariaLabel.indexOf(':');
  321. if (splitIndex !== -1) {
  322. var content = ariaLabel.substring(0, splitIndex).trim(); // Get content before ':' and trim spaces
  323. // 确保内容不包括 excludeKeyword
  324. if (!excludeKeyword.test(content)) { // 直接使用 excludeKeyword
  325. trackedContents.push(content);
  326. }
  327. }
  328. }
  329. }
  330. });
  331.  
  332. if (trackedContents.length > 0) {
  333. var message = trackedContents.join(', '); // 将数组内容连接成字符串,以逗号分隔
  334. showNotification(message);
  335. audio.play();
  336. if (tguserId !== 'your user ID') {
  337. tgsendMessage(message + ' is finished.');
  338. }
  339. }
  340.  
  341. // Book count
  342. var bookCountElements = document.querySelectorAll('.MuiCardContent-root h4');
  343. // 遍历每个 <h4> 元素
  344. bookCountElements.forEach(function(element) {
  345. // 获取文本内容
  346. var text = element.textContent.trim();
  347. // 提取数字部分
  348. var count = parseInt(text.match(/\d+/)[0]);
  349. // 如果数字大于等于2,则发送通知
  350. if (count >= bookcount) {
  351. // 发送通知
  352. var message = 'The book count has exceeded the limit!';
  353. showNotification(message);
  354. audio.play();
  355. if (tguserId !== 'your user ID') {
  356. tgsendMessage(message);
  357. }
  358. }
  359. });
  360.  
  361. // 追踪另一组元素 AFKTime
  362. var timeElements = document.querySelectorAll('.MuiTypography-root.MuiTypography-caption.css-deomsi');
  363. timeElements.forEach(function(timeElement) {
  364. var parentElement = timeElement.closest('.MuiStack-root');
  365. if (parentElement) {
  366. var resetTextElement = parentElement.querySelector('.MuiTypography-body1');
  367. if (resetTextElement) {
  368. var resetText = resetTextElement.textContent.trim();
  369. // 排除含有 Daily Reset 和 Weekly Reset 的模块内容
  370. if (resetText !== 'Daily Reset' && resetText !== 'Weekly Reset') {
  371. var timeText = timeElement.textContent.trim();
  372. var match = timeText.match(/^(\d+)h:(\d+)m:(\d+)s$|^(\d+)d:(\d+)h:(\d+)m$/);
  373.  
  374. if (match) {
  375. var days = match[4] ? parseInt(match[4], 10) : 0;
  376. var hours = match[1] ? parseInt(match[1], 10) : parseInt(match[2], 10);
  377. var minutes = match[2] ? parseInt(match[2], 10) : parseInt(match[5], 10);
  378. var seconds = match[3] ? parseInt(match[3], 10) : parseInt(match[6], 10);
  379.  
  380. // 转换成统一的格式
  381. hours += days * 24;
  382.  
  383. if (hours >= AFKHour && minutes >= AFKMin) {
  384. showNotification(timeText);
  385. audio.play();
  386. if (tguserId !== 'your user ID') {
  387. tgsendMessage(timeText + ' is finished.');
  388. }
  389. }
  390. }
  391. }
  392. }
  393. }
  394. });
  395.  
  396. //check top banner
  397. var images = document.querySelectorAll('div.css-1eybjch > div.css-79elbk > img'); // 获取页面上所有的img元素
  398. var matchedImages = Array.from(images).filter(function(image) {
  399. return bannerRegex.test(image.src); // 测试每个图片的src属性是否匹配正则表达式
  400. });
  401.  
  402. // 遍历匹配到正则的img元素并执行相应的操作
  403. if (matchedImages.length > 0) {
  404. var messages = [];
  405. matchedImages.forEach(function(image) {
  406. var src = image.src;
  407. var matchedText = src.match(bannerRegex)[0]; // 获取匹配文本
  408. messages.push(matchedText); // 将匹配的文本收集起来
  409. });
  410.  
  411. // 将所有匹配的结果合并成一条消息并显示出来
  412. var message = messages.join(' ');
  413. showNotification(message);
  414. audio.play();
  415. if (tguserId !== 'your user ID') {
  416. tgsendMessage('Banner check: ' + message);
  417. }
  418. }
  419.  
  420. // 追踪另一组元素 banner salt garden (aria-label)
  421. var timeElements = document.querySelectorAll('.MuiStack-root.css-18jqfyr');
  422. var trackedContents = [];
  423.  
  424. timeElements.forEach(function(timeElement) {
  425. var ariaLabel = timeElement.getAttribute('aria-label');
  426. var matchResult = bannerLabelRegex.exec(ariaLabel);
  427. if (ariaLabel && matchResult) {
  428. var matchedText = matchResult[0]; // 获取匹配到的具体内容
  429. trackedContents.push(matchedText);
  430. }
  431. });
  432.  
  433. if (trackedContents.length > 0) {
  434. var message = trackedContents.join(', '); // 将数组内容连接成字符串,以逗号分隔
  435. showNotification(message);
  436. audio.play();
  437. if (tguserId !== 'your user ID') {
  438. tgsendMessage(message);
  439. }
  440. }
  441.  
  442. }//All func end
  443.  
  444. function showNotification(message) {
  445. if (notificationPermission === 'granted') {
  446. GM_notification({
  447. text: message,
  448. title: 'Idleontoolbox Notification',
  449. timeout: 5000,
  450. onclick: function() {
  451. window.focus();
  452. }
  453. });
  454. } else {
  455. window.Notification.requestPermission().then(function(permission) {
  456. if (permission === 'granted') {
  457. GM_notification({
  458. text: message,
  459. title: 'Idleontoolbox Notification',
  460. timeout: 5000,
  461. onclick: function() {
  462. window.focus();
  463. }
  464. });
  465. }
  466. });
  467. }
  468. }
  469.  
  470. function makeDraggable(element) {
  471. let pos1 = 0,
  472. pos2 = 0,
  473. pos3 = 0,
  474. pos4 = 0;
  475.  
  476. element.onmousedown = dragMouseDown;
  477.  
  478. function dragMouseDown(e) {
  479. e = e || window.event;
  480. e.preventDefault();
  481. // get the mouse cursor position at startup:
  482. pos3 = e.clientX;
  483. pos4 = e.clientY;
  484. document.onmouseup = closeDragElement;
  485. // call a function whenever the cursor moves:
  486. document.onmousemove = elementDrag;
  487. }
  488.  
  489. function elementDrag(e) {
  490. e = e || window.event;
  491. e.preventDefault();
  492. // calculate the new cursor position:
  493. pos1 = pos3 - e.clientX;
  494. pos2 = pos4 - e.clientY;
  495. pos3 = e.clientX;
  496. pos4 = e.clientY;
  497. // set the element's new position:
  498. element.style.top = (element.offsetTop - pos2) + "px";
  499. element.style.left = (element.offsetLeft - pos1) + "px";
  500. }
  501.  
  502. function closeDragElement() {
  503. /* stop moving when mouse button is released:*/
  504. document.onmouseup = null;
  505. document.onmousemove = null;
  506. }
  507. }
  508.  
  509. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址