Bypass It

Automatically Bypass Restrictions and Get Straight to Your Destination!

当前为 2025-04-16 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Bypass It
  3. // @namespace http://tampermonkey.net/
  4. // @version 2025-04-17
  5. // @description Automatically Bypass Restrictions and Get Straight to Your Destination!
  6. // @supportURL https://gf.qytechs.cn/scripts/527564/feedback
  7. // @author You
  8. // @match *://vn88.id/*
  9. // @match *://vn88.fan/*
  10. // @match *://vn88.ing/*
  11. // @match *://vn88.wtf/*
  12. // @match *://aylink.co/*
  13. // @match *://vn8eu.com/*
  14. // @match *://vn88n.com/*
  15. // @match *://gplinks.co/*
  16. // @match *://v2links.me/*
  17. // @match *://upfion.com/*
  18. // @match *://fb88dv.com/*
  19. // @match *://m88usb.com/*
  20. // @match *://vn88.group/*
  21. // @match *://vn88wo.com/*
  22. // @match *://vn88zx.com/*
  23. // @match *://vn88ko.com/*
  24. // @match *://vn88es.com/*
  25. // @match *://vn88tu.com/*
  26. // @match *://www.m88.com/*
  27. // @match *://bet88li.com/*
  28. // @match *://cutyion.com/*
  29. // @match *://vn88tk1.com/*
  30. // @match *://vn88vc.wiki/*
  31. // @match *://fb88vao.com/*
  32. // @match *://coinclix.co/*
  33. // @match *://vn88.hiphop/*
  34. // @match *://gwaher.com/ptc
  35. // @match *://www.fb88.com/*
  36. // @match *://*.devnote.in/*
  37. // @match *://naamlist.com/*
  38. // @match *://modsfire.com/*
  39. // @match *://yeumoney.com/*
  40. // @match *://165.22.63.250/*
  41. // @match *://*.gmsrweb.org/*
  42. // @match *://modijiurl.com/*
  43. // @match *://geekgrove.net/*
  44. // @match *://www.m88sut.com/*
  45. // @match *://*.techyuth.xyz/*
  46. // @match *://vn88.solutions/*
  47. // @match *://financewada.com/*
  48. // @match *://188.166.185.213/*
  49. // @match *://gemini.google.com/*
  50. // @match *://cryptowidgets.net/*
  51. // @match *://*.wikijankari.com/*
  52. // @match *://cricketlegacy.com/*
  53. // @match *://ourcoincash.xyz/ptc*
  54. // @match *://*.idblogmarket.com/*
  55. // @match *://*.phonesparrow.com/*
  56. // @match *://financenova.online/*
  57. // @match *://bitcotasks.com//lead*
  58. // @match *://rajasthantopnews.com/*
  59. // @match *://www.google.com/url?q=*
  60. // @match *://freepayz.com/framed-ads
  61. // @match *://utkarshonlinetest.com/*
  62. // @match *://www.youtube.com/redirect*
  63. // @match *://www.facebook.com/flx/warn/*
  64. // @match *://gemini.google.com/app?msg=*
  65. // @match *://www.instagram.com/linkshim/*
  66. // @match https://www.google.com/recaptcha/api2/bframe*
  67. // @match https://www.google.com/recaptcha/api2/anchor*
  68. // @icon 
  69. // @grant GM_getValue
  70. // @grant GM_setValue
  71. // @grant unsafeWindow
  72. // @grant GM_openInTab
  73. // @grant GM_deleteValue
  74. // @grant GM_xmlhttpRequest
  75. // @grant GM_registerMenuCommand
  76. // @grant GM_addValueChangeListener
  77. // @connect api.nocaptchaai.com
  78. // @license MIT
  79. // ==/UserScript==
  80.  
  81. const noop = () => {};
  82. const rawWindow = unsafeWindow;
  83. const currentUrl = location.href;
  84. const queryParams = new URLSearchParams(location.search);
  85. const autoRedirectToSocial = GM_getValue("autoRedirectToSocial", false);
  86. const boostTimers = GM_getValue("boostTimers", false);
  87. const ptcFaucet = GM_getValue("ptcFaucet", false);
  88. const siteDelays = {
  89. // "example.com": 0.05, // 50ms
  90. // "anotherexample.net": 0.1 // 100ms
  91. };
  92. const reCaptcha = "YOUR_API_KEY_HERE"//GM_getValue("apiKey", "");
  93.  
  94. /**
  95. * Waits for an element matching the selector to appear in the DOM.
  96. * @param {string} selector - The CSS selector of the element.
  97. * @returns {Promise<Element>} - A promise that resolves with the found element.
  98. */
  99. function waitForElement(selector, callback = noop) {
  100. const findElement = () => {
  101. if (selector.startsWith("//")) {
  102. return document.evaluate(selector, document, null, 9).singleNodeValue;
  103. }
  104. return document.querySelector(selector);
  105.  
  106. };
  107.  
  108. return new Promise((resolve) => {
  109. const element = findElement();
  110. if (document.contains(element)) {
  111. callback(element);
  112. return resolve(element);
  113. }
  114. const observer = new MutationObserver((mutations, observerInstance) => {
  115. const node = findElement();
  116. if (document.contains(node)) {
  117. observerInstance.disconnect();
  118. callback(node);
  119. resolve(node);
  120. }
  121. });
  122. observer.observe(document.documentElement, {
  123. attributes: true,
  124. childList: true,
  125. subtree: true,
  126. });
  127. });
  128. }
  129.  
  130. /**
  131. * Navigates the browser to the specified URL.
  132. * @param {string} url - The destination URL.
  133. */
  134. function navigateTo(url) {
  135. location = url;
  136. }
  137.  
  138. /**
  139. * Returns a promise that resolves after a specified number of seconds.
  140. * @param {number} seconds - The delay in seconds.
  141. * @returns {Promise<void>}
  142. */
  143. function waitSeconds(seconds) {
  144. return new Promise((resolve) => setTimeout(resolve, seconds * 1000));
  145. }
  146.  
  147. /**
  148. * Waits for an element matching the selector and then redirects to its href.
  149. * @param {string} selector - The CSS selector of the link element.
  150. */
  151. function autoRedirectToLink(selector) {
  152. waitForElement(selector).then((element) => navigateTo(element.href));
  153. }
  154.  
  155. /**
  156. * Creates a menu command in the Tampermonkey UI that toggles a boolean state.
  157. * @param {string} key - The key used to store and retrieve the boolean state in GM storage.
  158. * @param {string} title - The title displayed in the menu, which is prefixed with a checkmark or cross.
  159. */
  160. function createMenu(key, title) {
  161. const isActive = GM_getValue(key, false);
  162. const prefix = isActive ? "✔️ " : "❌ ";
  163.  
  164. GM_registerMenuCommand(prefix + title, function() {
  165. GM_setValue(key, !isActive);
  166. createMenu(key, title);
  167. alert(`${title} has been ${!isActive ? "enabled" : "disabled"}.`);
  168. }, {
  169. id: key
  170. });
  171. }
  172.  
  173. /**
  174. * Converts an image URL to a Base64 string.
  175. * This function fetches the image using an HTTP GET request and converts it into a Base64-encoded string.
  176. *
  177. * @param {string} imageUrl - The URL of the image to convert.
  178. * @returns {Promise<string|null>} A Promise that resolves with the Base64-encoded image string, or null if an error occurs.
  179. */
  180. function imageUrlToBase64(imageUrl) {
  181. if (!imageUrl) {
  182. console.error("No imageUrl provided");
  183. return null;
  184. }
  185. try {
  186. return new Promise((resolve, reject) => {
  187. GM_xmlhttpRequest({
  188. method: "GET",
  189. url: imageUrl,
  190. responseType: "blob",
  191. onload: function(response) {
  192. if (!response.response) {
  193. reject("No response received");
  194. return;
  195. }
  196.  
  197. const blob = response.response;
  198. const reader = new FileReader();
  199. reader.onload = () => resolve(reader.result);
  200. reader.onerror = (error) => {
  201. console.error("FileReader error:", error);
  202. reject(error);
  203. };
  204. reader.readAsDataURL(blob);
  205. },
  206. onerror: function(error) {
  207. console.error("Error fetching image:", error);
  208. reject(error);
  209. }
  210. });
  211. });
  212. } catch (error) {
  213. console.error("Error converting image to Base64:", error);
  214. return null;
  215. }
  216. }
  217.  
  218. /**
  219. * Sends a Base64-encoded image to a language model and receives a descriptive response based on specific icon classes.
  220. * This function sends the image as Base64 and asks the model to describe it with one of the provided icon classes.
  221. *
  222. * @param {string} base64Image - The Base64-encoded image string.
  223. * @param {Array<string>} iconClasses - An array of strings representing the icon classes to use for the description.
  224. * @returns {Promise<string|null>} A Promise that resolves with the description of the image in English, or null if an error occurs.
  225. */
  226. async function getGeminiResponse(base64Image, iconClasses, url) {
  227. if (!window.ai) {
  228. console.error("window.ai is not available. Ensure the Prompt API is enabled.");
  229. // const tabid = GM_openInTab(`https://gemini.google.com/?msg=Analyze the image and describe this image using one of: ${iconClasses.join(", ")}. Be direct.`, {
  230. // active: true
  231. // })
  232. // return tabid;
  233. return;
  234. }
  235.  
  236. try {
  237. // Validate Gemini Nano availability
  238. const capabilities = await window.ai.languageModel.capabilities();
  239. if (!capabilities || !capabilities.available) {
  240. console.error("Gemini Nano is not available. Ensure the model is downloaded.");
  241. return;
  242. }
  243.  
  244. // Create session with system prompt to enforce English responses
  245. const session = await window.ai.languageModel.create({
  246. systemPrompt: "Analyze the given image and describe it using one of the provided categories. Be direct and concise.",
  247. expectedInputLanguages: ["en"]
  248. });
  249.  
  250. // Construct the prompt using the base64Image and iconClasses
  251. let result = await session.prompt(
  252. `Analyze the image: ${base64Image} and describe this image using one of: ${iconClasses.join(", ")}. Be direct.`
  253. );
  254.  
  255. console.log("Gemini Nano Response:", result);
  256. return result; // Ensure you're getting the correct response
  257. } catch (error) {
  258. console.error("Error with Gemini Nano:", error);
  259. throw error; // Better error propagation
  260. }
  261. }
  262.  
  263. /**
  264. * Executes a callback when the current host matches a pattern.
  265. * @param {string|RegExp} hostPattern - The host pattern or RegExp to match against location.host.
  266. * @param {Function} callback - The function to execute when matched.
  267. * @param {...any} args - Arguments to pass to the callback.
  268. */
  269. function runWhenHostMatches(hostPattern, callback, ...args) {
  270. hostPattern = hostPattern.replace("www.", "");
  271. const isMatch = new RegExp(hostPattern).test(location.host);
  272.  
  273. if (isMatch) callback(...args);
  274. }
  275.  
  276. /**
  277. * Simulates a click on an element (or multiple elements) specified by a selector.
  278. * Dispatches "mouseover", "mousedown", "mouseup", and "click" events.
  279. * @param {string} selector - The CSS selector of the element(s) to click.
  280. * @param {number} [delay=0] - Optional delay in seconds before clicking.
  281. */
  282. function simulateClick(selector, delay = 0) {
  283. const clickFun = (element) => {
  284. const events = ["mouseover", "mousedown", "mouseup", "click"];
  285. events.forEach((eventName) => {
  286. const event = new MouseEvent(eventName, {
  287. bubbles: true,
  288. });
  289. element.dispatchEvent(event);
  290. });
  291. }
  292. if (typeof selector != "string") {
  293. return clickFun(selector);
  294. }
  295. const selectors = selector.split(", ");
  296. // If multiple selectors are provided, iterate over each.
  297. if (selectors.length > 1) {
  298. selectors.forEach((sel) => simulateClick(sel, delay));
  299. return;
  300. }
  301.  
  302. waitForElement(selector, async function(element) {
  303. if (delay > 0) {
  304. await waitSeconds(delay);
  305. }
  306. clickFun(element);
  307. });
  308. }
  309.  
  310. /**
  311. * Calls the callback function once a captcha challenge is solved.
  312. * Checks for available captcha libraries (turnstile, hcaptcha, or grecaptcha).
  313. * @param {Function} callback - The function to execute when the captcha is solved.
  314. * @param {Function} [onWait=noop] - Optional function to call if waiting.
  315. */
  316. function whenCaptchaSolved(callback, onWait = noop) {
  317. let intervalId;
  318. const stopChecking = () => clearInterval(intervalId);
  319.  
  320. // waitForElement("//*[@id='captcha-result'] and normalize-space() = 'Verified!']", function() {
  321. // stopChecking();
  322. // callback();
  323. // })
  324. const checkCaptcha = () => {
  325. try {
  326. const element = document.querySelector('#captcha-result .mb-2.badge.bg-success');
  327. if (element && element.textContent.trim() === 'Verified!') {
  328. stopChecking();
  329. callback();
  330. }
  331.  
  332. const captcha = rawWindow.turnstile || rawWindow.hcaptcha || rawWindow.grecaptcha;
  333. const response = captcha.getResponse();
  334.  
  335. if (response) {
  336. stopChecking();
  337. callback();
  338. }
  339. } catch (error) {
  340. onWait(stopChecking);
  341. }
  342. };
  343.  
  344. checkCaptcha();
  345. intervalId = setInterval(checkCaptcha, 1000);
  346. }
  347.  
  348. /**
  349. * Listens for messages with a specific label and triggers a callback when received.
  350. * @param {string} label - The identifier for the message.
  351. * @param {Function} [callback=noop] - The function to call when a message is received.
  352. */
  353. function GM_onMessage(label, callback = noop) {
  354. GM_addValueChangeListener("postMessage-" + label, function(name, oldValue, newValue, remote) {
  355. if (remote) {
  356. GM_deleteValue("postMessage-" + label);
  357. callback(newValue);
  358. }
  359. });
  360. }
  361.  
  362. /**
  363. * Sends a message with a specific label by storing it in Greasemonkey storage.
  364. * @param {string} label - The identifier for the message.
  365. * @param {*} value - The data to be sent (must be serializable).
  366. */
  367. function GM_sendMessage(label, value) {
  368. GM_setValue("postMessage-" + label, value);
  369. }
  370.  
  371. /**
  372. * A function to override the default `window.open` method and execute a callback
  373. * whenever a new window/tab is opened.
  374. *
  375. * @param {Function} callback - A function to be executed when a new window/tab is opened.
  376. * It will receive the new window object as an argument.
  377. */
  378. function onOpenTab(callback) {
  379. const originalOpen = rawWindow.open;
  380.  
  381. rawWindow.open = function(...args) {
  382. const newWindow = originalOpen.apply(rawWindow, args);
  383. callback(newWindow);
  384. return newWindow;
  385. };
  386. }
  387.  
  388. /**
  389. * A function to override the default `window.open` method and execute a callback
  390. * whenever a new window/tab is opened.
  391. *
  392. * @param {Function} callback - A function to be executed when a new window/tab is opened.
  393. * It will receive the new window object as an argument.
  394. */
  395. function createSandbox() {
  396. // Create window proxy to disable Object.freeze
  397. const sandbox = new Proxy(window, {
  398. get(target, key) {
  399. if (key === 'Object') {
  400. return new Proxy(Object, {
  401. get(objTarget, objKey) {
  402. if (objKey === 'freeze') {
  403. return function(obj) {
  404. console.warn("Object.freeze disabled in sandbox.");
  405. return obj;
  406. };
  407. }
  408. return Reflect.get(objTarget, objKey);
  409. }
  410. });
  411. }
  412. return Reflect.get(target, key);
  413. }
  414. });
  415.  
  416. // Patch addEventListener to clone events and force isTrusted
  417. const originalAddEventListener = EventTarget.prototype.addEventListener;
  418. EventTarget.prototype.addEventListener = function(type, listener, options) {
  419. const wrappedListener = function(event) {
  420. const clonedEvent = Object.create(event);
  421. Object.defineProperty(clonedEvent, "isTrusted", {
  422. value: true,
  423. writable: false
  424. });
  425. return listener.call(this, clonedEvent);
  426. };
  427. return originalAddEventListener.call(this, type, wrappedListener, options);
  428. };
  429.  
  430. return sandbox;
  431. }
  432.  
  433. // Main execution
  434. (function () {
  435. "use strict";
  436. runWhenHostMatches("modsfire.com", simulateClick, ".download-button, .download-button[href]");
  437. runWhenHostMatches("(financewada|utkarshonlinetest).com|financenova.online|v2links.me", simulateClick, ".get_btn a[href], //div[contains(text(),'Continue')], .get-link:not(.disabled)");
  438.  
  439. runWhenHostMatches("devnote.in|techyuth.xyz", simulateClick, "#scroll:not(.no), #getlinks[style*='display: block;'], .get-link:not(.disabled)");
  440. autoRedirectToSocial && runWhenHostMatches("(instagram|youtube|facebook).com", simulateClick, ".-cx-PRIVATE-Linkshim__followLink__, #invalid-token-redirect-goto-site-button, .selected");
  441. createMenu("autoRedirectToSocial", "Auto-Redirect to Social Media");
  442.  
  443. runWhenHostMatches("(wikijankari|idblogmarket|modijiurl|phonesparrow|naamlist|rajasthantopnews).com|gmsrweb.org", simulateClick, "center a[style*='display: block;'], .get-link:not(.disabled)");
  444.  
  445. runWhenHostMatches("aylink.co", simulateClick, ".btn-go, .complete[style*='display: block;'] a, a.btn:not(.btn-go)")
  446.  
  447. /*
  448. Doesn't work anymore
  449. runWhenHostMatches("cryptowidgets.net|gemini.google.com", function () {
  450. let tab;
  451. waitForElement('.ql-editor p', function(element) {
  452. element.focus();
  453. const newSpan = document.createElement('span');
  454. newSpan.textContent = queryParams.get("msg");
  455. element.replaceWith(newSpan);
  456. simulateClick("[fonticon='send']", 1.5);
  457. setTimeout(function() {
  458. const r = [...document.querySelectorAll('[data-sourcepos]')].pop().innerText;
  459. GM_sendMessage("aiResponse", r + "|" + Math.random())
  460. }, 35000)
  461. });
  462.  
  463. GM_onMessage("aiResponse", function(newValue) {
  464. const description = newValue.split("|").shift();
  465. simulateClick(`.${description}`);
  466. });
  467.  
  468. waitForElement('.captcha-icon img').then(async function (element) {
  469. const base64 = await imageUrlToBase64(element.src);
  470. if (!base64) return console.log("Failed to convert image.");
  471.  
  472. const icons = document.querySelectorAll('#icon-options i');
  473. const iconClasses = Array.from(icons).map(icon => icon.classList[1]);
  474.  
  475. console.log("Base64 Image:", base64);
  476.  
  477. const description = await getGeminiResponse(base64, iconClasses, element.src);
  478. if (typeof description == "string") {
  479. console.log("Final Description:", description);
  480. simulateClick(`.${description}`);
  481. } else {
  482. //TODO: find way to upload image to gemini
  483. tab = description;
  484. }
  485. });
  486. });
  487. */
  488.  
  489. runWhenHostMatches("upfion.com", function() {
  490. simulateClick("#link-button:not([disabled])");
  491. waitForElement("a#link-button:not([disabled])").then((element) => navigateTo(element.href));
  492. });
  493.  
  494. runWhenHostMatches("cutyion.com", simulateClick, "#submit-button:not([disabled])");
  495.  
  496. runWhenHostMatches("(m88(sut|usb)?|bet88li|fb88(dv|vao)?|yeumoney|google|vn8?8(wo|tk1|eu|zx|ko|es|tu|n)?).com|188.166.185.213|165.22.63.250|vn88.(hiphop|wtf|id|now|fan|group|solutions|ing)|vn88vc.wiki", function() {
  497. waitForElement(".getcodebtn", function(element) {
  498. element.click();
  499. GM_sendMessage("close_tabs", currentUrl);
  500. })
  501. waitForElement("//*[contains(@class, 'cursor-pointer') and contains(text(), 'footer')]", function() {
  502. simulateClick("a[href*='"+location.host+"']");
  503. })
  504. waitForElement("//*[contains(@class, 'cursor-pointer') and string-length(translate(normalize-space(text()), '0123456789', '')) = 0]", function(element) {
  505. GM_sendMessage("vietnam_code", element.innerText)
  506. })
  507.  
  508. // Google search results don’t always display the exact content that a website wants for every page
  509. // const searchLinks = [...document.querySelectorAll('div#search a[href]:not([class])')].map(a => a.href)
  510. const tabs = [];
  511. const urls = [
  512. "m88.com", "vn88.id", "vn88.fan", "vn88.ing", "fb88.com", "vn88.now",
  513. "vn88.wtf", "vn8eu.com", "vn88n.com", "m88sut.com", "m88usb.com",
  514. "fb88dv.com", "vn88.group", "vn88wo.com", "vn88zx.com", "vn88ko.com",
  515. "vn88es.com", "vn88tu.com", "vn88tk1.com","vn88vc.wiki", "fb88vao.com",
  516. "vn88.hiphop", "vn88.solutions", "bet88li.com/m88", "188.166.185.213/w88",
  517. "165.22.63.250/188bet",
  518. ]
  519. urls.includes(queryParams.get("q")?.replace("https://", "")) && simulateClick(".mymGo ~ div a");
  520.  
  521. GM_onMessage("vietnam_code", function(newValue) {
  522. tabs.forEach(([_, tab]) => tab.close());
  523. document.querySelector('[name="code"]').value = newValue;
  524. simulateClick(".box-form-button button");
  525. });
  526.  
  527. waitForElement("[data-clipboard-text]:not([data-clipboard-text=''])", function(element) {
  528. const links = urls.filter(url => url.includes(element.dataset.clipboardText));
  529. links.forEach(url => {
  530. const tab = GM_openInTab("https://www.google.com/url?q=https://" + url);
  531. tabs.push([url, tab]);
  532. })
  533. GM_onMessage("close_tabs", function(newValue) {
  534. tabs.forEach(([url, tab]) => {
  535. if (new URL("https://" + url).host !== new URL(newValue).host) {
  536. tab.close();
  537. }
  538. });
  539. });
  540. })
  541. });
  542.  
  543. runWhenHostMatches("gplinks.co|cricketlegacy.com", simulateClick, "#VerifyBtn[style*='display: block;'], #NextBtn:not([href='#']), #captchaButton:not(.disabled)");
  544.  
  545. createMenu("ptcFaucet", "Opens PTC links");
  546. if (ptcFaucet) {
  547. const isHidden = el => el && (getComputedStyle(el).display === "none" || isHidden(el.parentElement));
  548. const regex = /^(visit for \d+ sec|visit(?: now)?|go|view|view now|view ads?|watch|start .* coins|start view ad)$/i;
  549. const findButton = () => [...document.querySelectorAll("button")].find(btn => regex.test(btn.textContent.trim()));
  550.  
  551. runWhenHostMatches("ourcoincash.xyz|(bitcotasks|freepayz|gwaher).com", function() {
  552. //TODO: replace the use of setInterval and fix the issue with tabs
  553. let tabObj;
  554. onOpenTab(function(tab) { tabObj = tab; });
  555. rawWindow.addEventListener("beforeunload", function() { tabObj?.close(); })
  556.  
  557. findButton()?.click();
  558.  
  559. setInterval(function() {
  560. tabObj?.close();
  561. findButton()?.click();
  562. }, 120000) // 120 sec
  563. whenCaptchaSolved(function() {
  564. const btn = document.querySelector("button[type='submit']")
  565. !isHidden(btn) && btn.click();
  566. })
  567. })
  568. }
  569. runWhenHostMatches("coinclix.co|geekgrove.net", async function() {
  570. // Add support for Google. Currently, it is not supported, and a refresh is needed when it appears.
  571. if (!autoRedirectToSocial) {
  572. //TODO: Automatically enable for a temporary amount of time
  573. return alert("To bypass this, you need to enable Auto-Redirect to Social Media.");
  574. }
  575. const code = document.querySelector(".mb-2 code");
  576. if (code) {
  577. //TODO: replace with something else
  578. const tab = GM_openInTab(document.querySelector("strong a")?.href);
  579. await waitSeconds(10);
  580. GM_sendMessage("geek_code", code.innerText);
  581.  
  582. }
  583. GM_onMessage("geek_code", function(newValue) {
  584. document.querySelector("#link_input").value = newValue;
  585. simulateClick("#btn_link, .btn-primary[href]");
  586. });
  587. })
  588.  
  589. createMenu("boostTimers", "Site-Specific Boost Timers");
  590. if (boostTimers) {
  591. const originalSetTimeout = rawWindow.setTimeout;
  592. const originalSetInterval = rawWindow.setInterval;
  593. ['setInterval', 'setTimeout'].forEach(value => {
  594. Object.defineProperty(window, value, {
  595. value: function(callback, delay, ...args) {
  596. delay = (siteDelays[location.host] || 1) * 1000;
  597. return originalSetTimeout(callback, delay, ...args);
  598. },
  599. writable: false,
  600. configurable: false
  601. });
  602. })
  603. }
  604. })();
  605.  
  606. // There are several known bugs, so if possible, it is recommended to use the official chrome extensions for now.
  607. (async function(isEnabled) {
  608. if (!isEnabled) return;
  609. console.log(123);
  610.  
  611. let isSolved = false;
  612.  
  613. const isCheckboxPresent = () => !!document.querySelector('.recaptcha-checkbox');
  614. const isCheckboxChecked = () => document.querySelector('#recaptcha-anchor')?.getAttribute('aria-checked') === 'true';
  615. const isImageChallengePresent = () => !!document.querySelector('#rc-imageselect');
  616. const isGrid4x4 = () => document.querySelectorAll('.rc-imageselect-tile').length === 16;
  617. const isGrid3x3 = () => document.querySelectorAll('.rc-imageselect-tile').length === 9;
  618. const getChallengeData = () => {
  619. const target = document.querySelector('.rc-imageselect-instructions strong')?.innerText;
  620. const imageUrl = document.querySelector('.rc-image-tile-33, .rc-image-tile-44')?.src;
  621. return [target, imageUrl];
  622. };
  623.  
  624. const solveImageChallenge = async (target, imageUrl, gridType) => {
  625. return new Promise(async (resolve) => {
  626. GM_xmlhttpRequest({
  627. method: 'POST',
  628. url: "https://api.nocaptchaai.com/createTask",
  629. headers: {
  630. 'Content-Type': 'application/json',
  631. },
  632. data: JSON.stringify({
  633. clientKey: reCaptcha,
  634. source: 'userscript',
  635. task: {
  636. type: 'ReCaptchaV2Classification',
  637. questionType: gridType,
  638. image: (await imageUrlToBase64(imageUrl)).replace("data:image/jpeg;base64,", ""),
  639. question: target
  640. }
  641. }),
  642. responseType: "json",
  643. onload: function(response) {
  644. const data = response.response;
  645. if (data.errorId) return; //TODO: reject
  646. resolve(data.solution?.objects || data.solution?.hasObject);
  647. }
  648. });
  649. });
  650.  
  651. };
  652.  
  653. const isVisible = (element) => {
  654. if (!element) return false;
  655. const style = window.getComputedStyle(element);
  656. return style.display !== 'none' &&
  657. style.visibility !== 'hidden' &&
  658. style.opacity !== '0' &&
  659. element.offsetWidth > 0 &&
  660. element.offsetHeight > 0;
  661. };
  662.  
  663. const hasError = () => {
  664. const errorSelectors = [
  665. '.rc-imageselect-error-select-more',
  666. '.rc-imageselect-error-dynamic-more',
  667. '.rc-imageselect-error-select-something'
  668. ];
  669.  
  670. for (const selector of errorSelectors) {
  671. const error = document.querySelector(selector);
  672. if (isVisible(error)) {
  673. return true;
  674. }
  675. }
  676. return false;
  677. };
  678. while (!isSolved) {
  679. await waitSeconds(2);
  680. if (isCheckboxPresent()) {
  681. if (isCheckboxChecked()) {
  682. isSolved = true;
  683. return;
  684. }
  685. await simulateClick('#recaptcha-anchor');
  686. } else if (isImageChallengePresent()) {
  687. const gridType = isGrid4x4() ? "44" : isGrid3x3() ? "33" : null;
  688. if (gridType && isImageChallengePresent() && !isCheckboxChecked()) {
  689. const [target, imageUrl] = getChallengeData();
  690. if (target && imageUrl) {
  691. const solution = await solveImageChallenge(target, imageUrl, gridType);
  692. if (solution) {
  693. const tiles = document.querySelectorAll('.rc-image-tile-wrapper');
  694. for (const index of solution) {
  695. if (tiles[index]) {
  696. simulateClick(tiles[index]);
  697. await waitSeconds(0.4);
  698. }
  699. }
  700. await waitSeconds(1);
  701. await simulateClick("#recaptcha-verify-button");
  702.  
  703. if (isCheckboxChecked()) {
  704. isSolved = true;
  705. await waitSeconds(1);
  706. return;
  707. }
  708. }
  709. }
  710. }
  711. } else if (isCheckboxChecked()) {
  712. isSolved = true;
  713. await waitSeconds(1);
  714. return;
  715. }
  716.  
  717. if (hasError()) {
  718. simulateClick("#recaptcha-reload-button");
  719. await waitSeconds(1);
  720. }
  721. }
  722.  
  723. })(reCaptcha.length > 20);

QingJ © 2025

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