Time Hooker

Hook the wait timers for websites that use them to delay content

  1. // ==UserScript==
  2. // @name Time Hooker
  3. // @namespace https://tampermonkey.net/
  4. // @version 1.7
  5. // @description Hook the wait timers for websites that use them to delay content
  6. // @author Niteesh
  7. // @match *://*/*
  8. // @grant none
  9. // @license MIT
  10. // @run-at document-start
  11. // ==/UserScript==
  12.  
  13.  
  14. (function() {
  15. 'use strict';
  16. const originalSetInterval = window.setInterval;
  17. const originalSetTimeout = window.setTimeout;
  18. const cloudflareIndicators = [
  19. "Checking your browser", // Common text
  20. "verify your browser", // Additional text
  21. "cloudflare", // May appear in class or ID names
  22. "turnstile" // Cloudflare CAPTCHA widget class
  23. ];
  24. function isCloudflareVerificationPresent() {
  25. return [...document.querySelectorAll('*')].some(el => {
  26. const text = el.textContent?.toLowerCase() || "";
  27. const idOrClass = (el.id || "") + " " + (el.className || "");
  28. return cloudflareIndicators.some(indicator =>
  29. text.includes(indicator.toLowerCase()) ||
  30. idOrClass.toLowerCase().includes(indicator.toLowerCase())
  31. );
  32. });
  33. }
  34. function print() {
  35. console.log(...arguments);
  36. }
  37. function clearAllTimers() {
  38. const highestIntervalId = setInterval(() => {}, 0);
  39. for (let i = 0; i <= highestIntervalId; i++) {
  40. clearInterval(i);
  41. }
  42. const highestTimeoutId = setTimeout(() => {}, 0);
  43. for (let i = 0; i <= highestTimeoutId; i++) {
  44. clearTimeout(i);
  45. }
  46. print('Cleared all active timers.');
  47. }
  48. function restoreOriginalTimers() {
  49. window.setInterval = originalSetInterval;
  50. window.setTimeout = originalSetTimeout;
  51. print("Restoring done.");
  52. }
  53. function interceptTimers(val) {
  54. window.setTimeout = function(callback, delay, ...args) {
  55. const newDelay = delay / val;
  56. print(`[Intercepted] setTimeout: ${delay}ms -> ${newDelay}ms`);
  57. return originalSetTimeout(callback, newDelay, ...args);
  58. };
  59.  
  60. window.setInterval = function(callback, interval, ...args) {
  61. const newInterval = interval / val;
  62. print(`[Intercepted] setInterval: ${interval}ms -> ${newInterval}ms`);
  63. return originalSetInterval(callback, newInterval, ...args);
  64. };
  65. }
  66. interceptTimers(15);
  67. let timerUsed = false;
  68. let potentialTimers;
  69. window.onload = function() {
  70. potentialTimers = [...document.querySelectorAll('*')].filter(el => {
  71. const text = el.textContent.trim().toLowerCase();
  72. const waitRegexes = [
  73. /wait\s+\d+\s+seconds/i,
  74. ///please\s+wait/i,
  75. /click\s+on\s+(image|button)/i,
  76. /click\s+and\s+wait\s+\d+/i
  77. ];
  78. return waitRegexes.some(regex => regex.test(text));
  79. //return /wait\s+\d+\s+seconds/i.test(text) || /please\s+wait/i.test(text);
  80. });
  81. if (potentialTimers.length > 0) {
  82. print("Potential timers detected:", potentialTimers);
  83. timerUsed = true;
  84. } else {
  85. print("No timers detected.");
  86. restoreOriginalTimers();
  87. if (isCloudflareVerificationPresent()) {
  88. print("Cloudflare verification detected...");
  89. } else {
  90. originalSetTimeout(_ => {
  91. clearAllTimers();
  92. }, 3000);
  93. }
  94. }
  95.  
  96.  
  97.  
  98. (function () {
  99. const overlayClassName = "custom-overlay";
  100.  
  101. // Function to create and append an overlay on a given element
  102. function createOverlay(element) {
  103. // Check if the overlay already exists
  104. if (element.querySelector(`.${overlayClassName}`)) return;
  105.  
  106. // Create the overlay
  107. const overlay = document.createElement('div');
  108. //overlay.textContent = "Overlay"; // Optional: Add text to the overlay
  109. overlay.className = overlayClassName;
  110.  
  111. // Style the overlay
  112. overlay.style.position = "absolute";
  113. overlay.style.top = "0";
  114. overlay.style.left = "0";
  115. overlay.style.width = "100%";
  116. overlay.style.height = "100%";
  117. overlay.style.backgroundColor = "rgba(255, 0, 0, 0.3)"; // Red overlay with transparency
  118. overlay.style.zIndex = "9999";
  119. overlay.style.pointerEvents = "none"; // Allow clicks to pass through
  120. overlay.style.display = "flex";
  121. overlay.style.justifyContent = "center";
  122. overlay.style.alignItems = "center";
  123. overlay.style.color = "white";
  124. overlay.style.fontSize = "16px";
  125.  
  126. // Ensure the parent element has `position: relative` for correct overlay placement
  127. const computedStyle = window.getComputedStyle(element);
  128. if (computedStyle.position === "static") {
  129. element.style.position = "relative";
  130. }
  131.  
  132. // Append the overlay
  133. element.appendChild(overlay);
  134. element.style.zIndex = 1001;
  135. element.style.position = "absolute"
  136. element.style.top = '0';
  137. element.style.left = '0';
  138. }
  139.  
  140.  
  141. const HIGH_Z_INDEX_THRESHOLD = 9999; // Define the threshold for "too high"
  142. const ADJUSTED_Z_INDEX = 1000; // Define the new reasonable z-index value
  143.  
  144. // Function to find and adjust high z-index elements
  145. function adjustHighZIndex() {
  146. const elements = [...document.querySelectorAll('*')]; // Get all elements
  147. elements.forEach(el => {
  148. if (el == document.querySelectorAll('.overlayClassName')) {
  149. return;
  150. }
  151. const zIndex = window.getComputedStyle(el).zIndex;
  152. if (!isNaN(zIndex) && zIndex > HIGH_Z_INDEX_THRESHOLD) {
  153. console.log(`Element with high z-index found:`, el);
  154. console.log(`Original z-index: ${zIndex}`);
  155. el.style.zIndex = ADJUSTED_Z_INDEX; // Adjust the z-index
  156. console.log(`Adjusted z-index to: ${ADJUSTED_Z_INDEX}`);
  157. }
  158. });
  159. }
  160.  
  161.  
  162. function logLinks() {
  163. const targetKeywords = ["continue", "get link"]; // Keywords to match
  164. const elements = [...document.querySelectorAll('*')].filter(el =>
  165. (targetKeywords.some(keyword => el.textContent.toLowerCase().includes(keyword))) && ((el.tagName === 'A' && el.href) || // Standard links
  166. (el.tagName === 'BUTTON') || // Buttons with click handlers
  167. el.getAttribute('role') === 'link') // Elements with role="link"
  168. );
  169. print(elements);
  170. elements.forEach(el => {
  171. console.log("Found link/button:");
  172. console.log({
  173. tag: el.tagName,
  174. text: el.textContent.trim(),
  175. href: el.href || "N/A",
  176. onclick: el.onclick ? el.onclick.toString() : "N/A"
  177. });
  178. });
  179.  
  180. console.log(`Total links/buttons found: ${elements.length}`);
  181. }
  182.  
  183. // Initial log of all links
  184. // Function to find all relevant elements and overlay them
  185.  
  186. function overlayLinks() {
  187. const targetKeywords = ["continue", "get link"];
  188. const elements = [...document.querySelectorAll('*')].filter(el =>
  189. targetKeywords.some(keyword => el.textContent.toLowerCase().trim().includes(keyword)) &&
  190. (el.tagName === 'BUTTON' || el.tagName === 'A' || el.onclick || el.hasAttribute('role'))
  191. );
  192.  
  193. elements.forEach(element => {
  194. createOverlay(element);
  195. print(element)
  196. });
  197.  
  198. console.log(`${elements.length} elements overlaid.`);
  199. }
  200.  
  201. // Initial overlay
  202.  
  203.  
  204. // Use MutationObserver to handle dynamically added elements
  205.  
  206.  
  207. if (timerUsed) {
  208. adjustHighZIndex();
  209. logLinks();
  210. overlayLinks();
  211. const observer = new MutationObserver(() => {
  212. overlayLinks();
  213. logLinks();
  214. adjustHighZIndex()
  215. });
  216.  
  217. observer.observe(document.body, { childList: true, subtree: true });
  218.  
  219. // Log for debugging
  220. console.log("Overlay script is running...");
  221. print("Link logging is active...");
  222.  
  223. }
  224. })();
  225.  
  226.  
  227. /*
  228. if (timerUsed) {
  229. print("setting up...");
  230. originalSetInterval(() => {
  231. const button = document.querySelector('button:enabled, .clickable');
  232. let clickable = [];
  233. clickable = [...document.querySelectorAll('*')].find(el =>
  234. (el.textContent.toLowerCase().includes("continue") || el.textContent.toLowerCase().includes("get link") || el.textContent.toLowerCase().includes("robot")) &&
  235. (el.tagName === 'BUTTON' || el.tagName === 'A') && !el.disabled
  236. );
  237. if (clickable) {
  238. print("Clickable element found:", clickable);
  239. if (clickable.tagName === 'BUTTON') {
  240. clickable.click();
  241. } else if (clickable.tagName === 'A') {
  242. print(clickable.href);
  243. if (clickable.href !== window.location.href) {
  244. window.location.replace(clickable.href);
  245. }
  246. }
  247. // else {
  248. // print("just trying to click, do you find the exact button?");
  249. // clickable.click();
  250. // }
  251. } else {
  252. print("No clickable element found.");
  253. }
  254. }, 1000);
  255. // dynamically searching for a button
  256. const observer = new MutationObserver((mutations) => {
  257. mutations.forEach(mutation => {
  258. const button = [...document.querySelectorAll('*')].find(el =>
  259. (el.textContent.toLowerCase().includes("continue") || el.textContent.toLowerCase().includes("get link")) &&
  260. (el.tagName === 'BUTTON' || el.tagName === 'A') && !el.disabled
  261. );
  262. if (button) {
  263. print("Dynamically loaded 'continue' button found:", button);
  264. observer.disconnect(); // Stop observing once found
  265. button.click();
  266. print("clicked");
  267. }
  268. });
  269. });
  270. // Start observing changes in the DOM
  271. observer.observe(document.body, {
  272. childList: true,
  273. subtree: true
  274. });
  275. } else {
  276. }
  277. */
  278. }
  279.  
  280. })();

QingJ © 2025

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