Greasy Fork is available in English.

Bypass It

Automatically Bypass Restrictions and Get Straight to Your Destination!

  1. // ==UserScript==
  2. // @name Bypass It
  3. // @namespace http://tampermonkey.net/
  4. // @version 2025-05-08
  5. // @description Automatically Bypass Restrictions and Get Straight to Your Destination!
  6. // @supportURL https://greatest.deepsurf.us/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 *://vn88my.com/*
  27. // @match *://v9bethi.com/*
  28. // @match *://www.m88.com/*
  29. // @match *://bet88li.com/*
  30. // @match *://cutyion.com/*
  31. // @match *://vn88tk1.com/*
  32. // @match *://vn88vc.wiki/*
  33. // @match *://fb88vao.com/*
  34. // @match *://coinclix.co/*
  35. // @match *://vn88.hiphop/*
  36. // @match *://gwaher.com/ptc
  37. // @match *://www.fb88.com/*
  38. // @match *://*.devnote.in/*
  39. // @match *://naamlist.com/*
  40. // @match *://modsfire.com/*
  41. // @match *://yeumoney.com/*
  42. // @match *://165.22.63.250/*
  43. // @match *://*.gmsrweb.org/*
  44. // @match *://modijiurl.com/*
  45. // @match *://geekgrove.net/*
  46. // @match *://www.m88sut.com/*
  47. // @match *://*.techyuth.xyz/*
  48. // @match *://vn88.solutions/*
  49. // @match *://financewada.com/*
  50. // @match *://188.166.185.213/*
  51. // @match *://earnbitmoon.club/*
  52. // @match *://gemini.google.com/*
  53. // @match *://cryptowidgets.net/*
  54. // @match *://*.wikijankari.com/*
  55. // @match *://cricketlegacy.com/*
  56. // @match *://ourcoincash.xyz/ptc*
  57. // @match *://*.idblogmarket.com/*
  58. // @match *://*.phonesparrow.com/*
  59. // @match *://financenova.online/*
  60. // @match *://bitcotasks.com//lead*
  61. // @match *://rajasthantopnews.com/*
  62. // @match *://www.google.com/url?q=*
  63. // @match *://freepayz.com/framed-ads
  64. // @match *://utkarshonlinetest.com/*
  65. // @match *://www.youtube.com/redirect*
  66. // @match *://www.facebook.com/flx/warn/*
  67. // @match *://gemini.google.com/app?msg=*
  68. // @match *://www.instagram.com/linkshim/*
  69. // @match https://www.google.com/recaptcha/api2/bframe*
  70. // @match https://www.google.com/recaptcha/api2/anchor*
  71. // @icon 
  72. // @grant GM_getValue
  73. // @grant GM_setValue
  74. // @grant GM_addStyle
  75. // @grant unsafeWindow
  76. // @grant GM_openInTab
  77. // @grant GM_deleteValue
  78. // @grant GM_xmlhttpRequest
  79. // @grant GM_registerMenuCommand
  80. // @grant GM_addValueChangeListener
  81. // @require https://update.greatest.deepsurf.us/scripts/439099/1203718/MonkeyConfig%20Modern%20Reloaded.js
  82. // @connect api.nocaptchaai.com
  83. // @connect iconcaptcha-solver.vercel.app
  84. // @license MIT
  85. // ==/UserScript==
  86. const cfg = new MonkeyConfig({
  87. menuCommand: true,
  88. params: {
  89. redirectToSocial: {
  90. type: 'checkbox',
  91. label: "Auto-Redirect to Social Media",
  92. default: false,
  93. },
  94. ptcFaucet: {
  95. type: 'checkbox',
  96. label: "Opens PTC links",
  97. default: false
  98. },
  99. apiKey1: {
  100. type: 'text',
  101. label: "noCaptcha Ai API Key",
  102. default: "",
  103. },
  104. apiKey2: {
  105. type: 'text',
  106. label: "IconCaptchas API key",
  107. default: "",
  108. },
  109. siteDelays: {
  110. type: 'text',
  111. long: 3,
  112. label: "Site-Specific Delays (domain:delay)",
  113. default: 'example.com:50ms\nanotherexample.net:100ms'
  114. }
  115. }
  116. });
  117.  
  118. const noop = () => {};
  119. const rawWindow = unsafeWindow;
  120. const currentUrl = location.href;
  121. const queryParams = new URLSearchParams(location.search);
  122.  
  123. function waitForElement(selector, callback = noop) {
  124. const findElement = () => {
  125. if (selector.startsWith("//")) {
  126. return document.evaluate(selector, document, null, 9).singleNodeValue;
  127. }
  128. return document.querySelector(selector);
  129.  
  130. };
  131.  
  132. return new Promise((resolve) => {
  133. const element = findElement();
  134. if (document.contains(element)) {
  135. callback(element);
  136. return resolve(element);
  137. }
  138. const observer = new MutationObserver((mutations, observerInstance) => {
  139. const node = findElement();
  140. if (document.contains(node)) {
  141. observerInstance.disconnect();
  142. callback(node);
  143. resolve(node);
  144. }
  145. });
  146. observer.observe(document.documentElement, {
  147. attributes: true,
  148. childList: true,
  149. subtree: true,
  150. });
  151. });
  152. }
  153.  
  154. function navigateTo(url) {
  155. location = url;
  156. }
  157.  
  158. function waitSeconds(seconds) {
  159. return new Promise((resolve) => setTimeout(resolve, seconds * 1000));
  160. }
  161.  
  162. function redirectToLink(selector) {
  163. waitForElement(selector).then((element) => navigateTo(element.href));
  164. }
  165.  
  166. function imageUrlToBase64(imageUrl) {
  167. if (!imageUrl) {
  168. console.error("No imageUrl provided");
  169. return null;
  170. }
  171. return new Promise((resolve, reject) => {
  172. makeRequest(imageUrl, {
  173. responseType: "blob",
  174. }).then(blob => {
  175. const reader = new FileReader();
  176. reader.onload = () => resolve(reader.result);
  177. reader.onerror = (error) => {
  178. console.error("FileReader error:", error);
  179. reject(error);
  180. };
  181. reader.readAsDataURL(blob);
  182. })
  183. });
  184. }
  185.  
  186. function onMatch(hostPattern, callback, ...args) {
  187. hostPattern = hostPattern.replace("www.", "");
  188. if (hostPattern.length < 3) return;
  189.  
  190. const isMatch = new RegExp(hostPattern).test(location.host);
  191. if (isMatch) callback(...args);
  192. }
  193.  
  194. function onMatchClick(hostPattern, selector) {
  195. onMatch(hostPattern, simulateClick, selector)
  196. }
  197.  
  198. function simulateClick(selector, delay = 0) {
  199. const clickFun = (element) => {
  200. const events = ["mouseover", "mousedown", "mouseup", "click"];
  201. events.forEach((eventName) => {
  202. const event = new MouseEvent(eventName, {
  203. bubbles: true,
  204. });
  205. element.dispatchEvent(event);
  206. });
  207. }
  208. if (typeof selector != "string") {
  209. return clickFun(selector);
  210. }
  211. const selectors = selector.split(", ");
  212. if (selectors.length > 1) {
  213. selectors.forEach((sel) => simulateClick(sel, delay));
  214. return;
  215. }
  216.  
  217. waitForElement(selector, async function(element) {
  218. if (delay > 0) await waitSeconds(delay);
  219. clickFun(element);
  220. });
  221. }
  222.  
  223. function whenCaptchaSolved(callback, onWait = noop) {
  224. let intervalId;
  225. const stopChecking = () => clearInterval(intervalId);
  226.  
  227. // waitForElement("//*[@id='captcha-result'] and normalize-space() = 'Verified!']", function() {
  228. // stopChecking();
  229. // callback();
  230. // })
  231. const checkCaptcha = () => {
  232. try {
  233. const element = document.querySelector('#captcha-result .mb-2.badge.bg-success');
  234. if (element && element.textContent.trim() === 'Verified!') {
  235. stopChecking();
  236. callback();
  237. }
  238.  
  239. const captcha = rawWindow.turnstile || rawWindow.hcaptcha || rawWindow.grecaptcha;
  240. const response = captcha.getResponse();
  241.  
  242. if (response) {
  243. stopChecking();
  244. callback();
  245. }
  246. } catch (error) {
  247. onWait(stopChecking);
  248. }
  249. };
  250.  
  251. checkCaptcha();
  252. intervalId = setInterval(checkCaptcha, 1000);
  253. }
  254.  
  255. function GM_onMessage(label, callback = noop) {
  256. GM_addValueChangeListener("postMessage-" + label, function(name, oldValue, newValue, remote) {
  257. if (remote) {
  258. GM_deleteValue("postMessage-" + label);
  259. callback(newValue);
  260. }
  261. });
  262. }
  263.  
  264. function GM_sendMessage(label, value) {
  265. GM_setValue("postMessage-" + label, value);
  266. }
  267.  
  268. function onOpenTab(callback) {
  269. const originalOpen = rawWindow.open;
  270.  
  271. rawWindow.open = function(...args) {
  272. const newWindow = originalOpen.apply(rawWindow, args);
  273. callback(newWindow);
  274. return newWindow;
  275. };
  276. }
  277.  
  278. function clickWithTrusted() {
  279. // Create window proxy to disable Object.freeze
  280. const sandbox = new Proxy(window, {
  281. get(target, key) {
  282. if (key === 'Object') {
  283. return new Proxy(Object, {
  284. get(objTarget, objKey) {
  285. if (objKey === 'freeze') {
  286. return function(obj) {
  287. console.warn("Object.freeze disabled in sandbox.");
  288. return obj;
  289. };
  290. }
  291. return Reflect.get(objTarget, objKey);
  292. }
  293. });
  294. }
  295. return Reflect.get(target, key);
  296. }
  297. });
  298.  
  299. // Patch addEventListener to clone events and force isTrusted
  300. const originalAddEventListener = EventTarget.prototype.addEventListener;
  301. EventTarget.prototype.addEventListener = function(type, listener, options) {
  302. const wrappedListener = function(event) {
  303. const clonedEvent = Object.create(event);
  304. Object.defineProperty(clonedEvent, "isTrusted", {
  305. value: true,
  306. writable: false
  307. });
  308. return listener.call(this, clonedEvent);
  309. };
  310. return originalAddEventListener.call(this, type, wrappedListener, options);
  311. };
  312.  
  313. return sandbox;
  314. }
  315.  
  316. function isVisible(element) {
  317. if (!element) return false;
  318.  
  319. const style = window.getComputedStyle(element);
  320. const visible = style.display !== 'none' &&
  321. style.visibility !== 'hidden' &&
  322. style.opacity !== '0' &&
  323. element.offsetWidth > 0 &&
  324. element.offsetHeight > 0;
  325.  
  326. if (!visible) return false;
  327.  
  328. return isVisible(element.parentElement) || element.parentElement === null;
  329. };
  330.  
  331. function transformMethod(object, methodName, argumentProcessor) {
  332. const originalMethod = object[methodName];
  333.  
  334. Object.defineProperty(object, methodName, {
  335. value: function (...args) {
  336. const processedArgs = argumentProcessor(args);
  337. return originalMethod(...processedArgs);
  338. },
  339. writable: false,
  340. configurable: false
  341. });
  342. }
  343.  
  344. function makeRequest(url, options = {}) {
  345. return new Promise((resolve, reject) => {
  346. GM_xmlhttpRequest({
  347. url,
  348. method: options.method || "GET",
  349. responseType: options.responseType || "json",
  350. headers: options.headers || {},
  351. data: options.data ? JSON.stringify(options.data) : null,
  352. //timeout: options.timeout || 0,
  353. onload: (response) => resolve(response.response),
  354. onerror: (error) => reject(error),
  355. ontimeout: (error) => reject(error),
  356. onabort: (error) => reject(error)
  357. });
  358. });
  359. }
  360.  
  361. // Main execution
  362. (function () {
  363. "use strict";
  364. //only click
  365. onMatchClick("cutyion.com", "#submit-button:not([disabled])");
  366. onMatchClick("modsfire.com", ".download-button, .download-button[href]");
  367. onMatchClick("aylink.co", ".btn-go, .complete[style*='display: block;'] a, a.btn:not(.btn-go)")
  368. onMatchClick("devnote.in|techyuth.xyz", "#scroll:not(.no), #getlinks[style*='display: block;'], .get-link:not(.disabled)");
  369. onMatchClick("gplinks.co|cricketlegacy.com", "#VerifyBtn[style*='display: block;'], #NextBtn:not([href='#']), #captchaButton:not(.disabled)");
  370. onMatchClick("(financewada|utkarshonlinetest).com|financenova.online|v2links.me", ".get_btn a[href], //div[contains(text(),'Continue')], .get-link:not(.disabled)");
  371. onMatchClick("(wikijankari|idblogmarket|modijiurl|phonesparrow|naamlist|rajasthantopnews).com|gmsrweb.org", "center a[style*='display: block;'], .get-link:not(.disabled)");
  372. cfg.get("redirectToSocial") && onMatchClick("(instagram|youtube|facebook).com", ".-cx-PRIVATE-Linkshim__followLink__, #invalid-token-redirect-goto-site-button, .selected");
  373.  
  374. onMatch("upfion.com", function() {
  375. simulateClick("#link-button:not([disabled])");
  376. redirectToLink("a#link-button:not([disabled])");
  377. });
  378.  
  379. onMatch("(m88(sut|usb)?|bet88li|fb88(dv|vao)?|yeumoney|google|vn8?8(wo|tk1|eu|zx|ko|es|tu|n|my)?|v9bethi).com|188.166.185.213|165.22.63.250|vn88.(hiphop|wtf|id|now|fan|group|solutions|ing)|vn88vc.wiki", function() {
  380. waitForElement(".getcodebtn", function(element) {
  381. element.click();
  382. GM_sendMessage("close_tabs", currentUrl);
  383. })
  384. waitForElement("//*[contains(@class, 'cursor-pointer') and contains(text(), 'footer')]", function() {
  385. simulateClick("a[href*='"+location.host+"']");
  386. })
  387. waitForElement("//*[contains(@class, 'cursor-pointer') and string-length(translate(normalize-space(text()), '0123456789', '')) = 0]", function(element) {
  388. GM_sendMessage("vietnam_code", element.innerText)
  389. })
  390.  
  391. const tabs = [];
  392. const urls = [
  393. "m88.com", "vn88.id", "vn88.fan", "vn88.ing", "fb88.com", "vn88.now",
  394. "vn88.wtf", "vn8eu.com", "vn88n.com", "m88sut.com", "m88usb.com",
  395. "fb88dv.com", "vn88.group", "vn88wo.com", "vn88zx.com", "vn88ko.com",
  396. "vn88es.com", "vn88tu.com", "vn88tk1.com","vn88vc.wiki", "fb88vao.com",
  397. "vn88.hiphop", "vn88.solutions", "bet88li.com/m88", "188.166.185.213/w88",
  398. "165.22.63.250/188bet", "vn88my.com/m88", "v9bethi.com"
  399. ]
  400.  
  401. urls.includes(queryParams.get("q")?.replace("https://", "")) && simulateClick(".mymGo ~ div a");
  402.  
  403. GM_onMessage("vietnam_code", function(newValue) {
  404. tabs.forEach(([_, tab]) => tab.close());
  405. document.querySelector('[name="code"]').value = newValue;
  406. simulateClick(".box-form-button button");
  407. });
  408.  
  409. waitForElement("[data-clipboard-text]:not([data-clipboard-text=''])", function(element) {
  410. const links = urls.filter(url => url.includes(element.dataset.clipboardText));
  411. links.forEach(url => {
  412. const tab = GM_openInTab("https://www.google.com/url?q=https://" + url);
  413. tabs.push([url, tab]);
  414. })
  415. GM_onMessage("close_tabs", function(newValue) {
  416. tabs.forEach(([url, tab]) => {
  417. if (new URL("https://" + url).host !== new URL(newValue).host) {
  418. tab.close();
  419. }
  420. });
  421. });
  422. })
  423. });
  424.  
  425. if (cfg.get("ptcFaucet")) {
  426. const regex = /^(visit for \d+ sec|visit(?: now)?|go|view|view now|view ads?|watch|start .* coins|start view ad)$/i;
  427. const findButton = () => [...document.querySelectorAll("button")].find(btn => regex.test(btn.textContent.trim()));
  428.  
  429. onMatch("ourcoincash.xyz|(bitcotasks|freepayz|gwaher).com", function() {
  430. //TODO: replace the use of setInterval and fix the issue with tabs
  431. let tabObj;
  432. onOpenTab(function(tab) { tabObj = tab; });
  433. rawWindow.addEventListener("beforeunload", function() { tabObj?.close(); })
  434.  
  435. findButton()?.click();
  436.  
  437. setInterval(function() {
  438. tabObj?.close();
  439. findButton()?.click();
  440. }, 120000) // 120 sec
  441. whenCaptchaSolved(function() {
  442. const btn = document.querySelector("button[type='submit']")
  443. isVisible(btn) && btn.click();
  444. })
  445. })
  446. }
  447.  
  448. onMatch("coinclix.co|geekgrove.net", async function() {
  449. // Add support for Google. Currently, it is not supported, and a refresh is needed when it appears.
  450. if (!cfg.get("autoRedirectToSocial")) {
  451. //TODO: Automatically enable for a temporary amount of time
  452. return alert("To bypass this, you need to enable Auto-Redirect to Social Media.");
  453. }
  454. const code = document.querySelector(".mb-2 code");
  455. if (code) {
  456. //TODO: replace with something else
  457. const tab = GM_openInTab(document.querySelector("strong a")?.href);
  458. await waitSeconds(10);
  459. GM_sendMessage("geek_code", code.innerText);
  460.  
  461. }
  462. GM_onMessage("geek_code", function(newValue) {
  463. document.querySelector("#link_input").value = newValue;
  464. simulateClick("#btn_link, .btn-primary[href]");
  465. });
  466. })
  467.  
  468. // Don't forget to add the site into the @match.
  469. const sites = cfg.get("siteDelays").replaceAll("ms", "").split(/\n|:/);
  470. if (location.host in sites) {
  471. const argumentProcessor = (cb, delay, ...args) => ([cb, sites[location.host] * 1000, ...args]);
  472. transformMethod(rawWindow, "setInterval", argumentProcessor);
  473. transformMethod(rawWindow, "setTimeout", argumentProcessor);
  474. }
  475. })();
  476. // There are several known bugs, so if possible, it is recommended to use the official chrome extensions for now.
  477. (async function(isEnabled) {
  478. if (!isEnabled) return;
  479.  
  480. let isSolved = false;
  481.  
  482. const isCheckboxPresent = () => !!document.querySelector('.recaptcha-checkbox');
  483. const isCheckboxChecked = () => document.querySelector('#recaptcha-anchor')?.getAttribute('aria-checked') === 'true';
  484. const isImageChallengePresent = () => !!document.querySelector('#rc-imageselect');
  485. const isGrid4x4 = () => document.querySelectorAll('.rc-imageselect-tile').length === 16;
  486. const isGrid3x3 = () => document.querySelectorAll('.rc-imageselect-tile').length === 9;
  487. const getChallengeData = () => {
  488. const target = document.querySelector('.rc-imageselect-instructions strong')?.innerText;
  489. const imageUrl = document.querySelector('.rc-image-tile-33, .rc-image-tile-44')?.src;
  490. return [target, imageUrl];
  491. };
  492.  
  493. const solveImageChallenge = async (target, imageUrl, gridType) => {
  494. return new Promise(async (resolve, reject) => {
  495. const response = await makeRequest("https://api.nocaptchaai.com/createTask", {
  496. method: 'POST',
  497. headers: {
  498. 'Content-Type': 'application/json',
  499. },
  500. data: {
  501. clientKey: cfg.get('apiKey1'),
  502. task: {
  503. type: 'ReCaptchaV2Classification',
  504. questionType: gridType,
  505. image: (await imageUrlToBase64(imageUrl)).replace("data:image/jpeg;base64,", ""),
  506. question: target
  507. }
  508. }
  509. })
  510.  
  511. if (response.errorId) reject(response);
  512. const solution = response.solution;
  513. resolve(solution?.objects || solution?.hasObject);
  514. });
  515. };
  516.  
  517. const hasError = () => {
  518. const errorElements = document.querySelectorAll('[class^="rc-imageselect-error-"]');
  519. return Array.from(errorElements).some(isVisible);
  520. };
  521.  
  522. while (!isSolved) {
  523. await waitSeconds(2);
  524. if (isCheckboxPresent()) {
  525. if (isCheckboxChecked()) {
  526. isSolved = true;
  527. return;
  528. }
  529. await simulateClick('#recaptcha-anchor');
  530. } else if (isImageChallengePresent()) {
  531. const gridType = isGrid4x4() ? "44" : isGrid3x3() ? "33" : null;
  532. if (gridType && isImageChallengePresent() && !isCheckboxChecked()) {
  533. const [target, imageUrl] = getChallengeData();
  534. if (target && imageUrl) {
  535. const solution = await solveImageChallenge(target, imageUrl, gridType);
  536. if (solution) {
  537. const tiles = document.querySelectorAll('.rc-image-tile-wrapper');
  538. for (const index of solution) {
  539. if (tiles[index]) {
  540. simulateClick(tiles[index]);
  541. await waitSeconds(0.4);
  542. }
  543. }
  544. await waitSeconds(1);
  545. await simulateClick("#recaptcha-verify-button");
  546.  
  547. if (isCheckboxChecked()) {
  548. isSolved = true;
  549. await waitSeconds(1);
  550. return;
  551. }
  552. }
  553. }
  554. }
  555. } else if (isCheckboxChecked()) {
  556. isSolved = true;
  557. await waitSeconds(1);
  558. return;
  559. }
  560.  
  561. if (hasError()) {
  562. simulateClick("#recaptcha-reload-button");
  563. await waitSeconds(1);
  564. }
  565. }
  566. })(cfg.get("apiKey1").length > 20);
  567.  
  568. (async function(isEnabled) {
  569. if (!isEnabled) return;
  570. await waitForElement('.iconcaptcha-holder');
  571. const OriginalImage = rawWindow.Image;
  572. rawWindow.Image = function(width, height) {
  573. const img = new OriginalImage(width, height);
  574.  
  575. img.addEventListener('load', () => {
  576. try {
  577. const canvas = document.createElement('canvas');
  578. canvas.width = img.naturalWidth;
  579. canvas.height = img.naturalHeight;
  580. const ctx = canvas.getContext('2d');
  581. ctx.drawImage(img, 0, 0);
  582. solve(canvas.toDataURL())
  583. } catch (err) {
  584. console.warn('Could not convert image to base64:', err);
  585. }
  586. });
  587. return img;
  588.  
  589. }
  590.  
  591. rawWindow.Image.prototype = OriginalImage.prototype;
  592. const element = await waitForElement(".iconcaptcha-modal__body-icons");
  593. await waitSeconds(0.5);
  594.  
  595. if (element.tagName === "canvas") solve(element.toDataURL());
  596. async function solve(base64Image) {
  597. base64Image = base64Image.replace("data:image/png;base64,", "");
  598. console.log(base64Image);
  599.  
  600. const response = await makeRequest("https://iconcaptcha-solver.vercel.app/api/v2/solve", {
  601. method: "POST",
  602. headers: {
  603. 'Content-Type': 'application/json',
  604. },
  605. data: {
  606. token: cfg.get("apiKey2").trim(),
  607. image: base64Image
  608. }
  609. });
  610.  
  611. console.log(response);
  612. const captchaBox = document.querySelector(".iconcaptcha-modal__body-selection > i");
  613.  
  614. const rect = element.getBoundingClientRect();
  615. const clientX = rect.left + response.center_x;
  616. const clientY = rect.top + response.center_y;
  617.  
  618. console.log("Click inside canvas at:", { clientX, clientY });
  619.  
  620. // Visual marker
  621. const marker = document.createElement("div");
  622. marker.style.position = "fixed";
  623. marker.style.left = `${clientX - 5}px`;
  624. marker.style.top = `${clientY - 5}px`;
  625. marker.style.width = "10px";
  626. marker.style.height = "10px";
  627. marker.style.backgroundColor = "red";
  628. marker.style.borderRadius = "50%";
  629. marker.style.zIndex = "9999";
  630. marker.style.pointerEvents = "none";
  631. document.body.appendChild(marker);
  632.  
  633. setTimeout(() => marker.remove(), 1000);
  634.  
  635. ["mouseenter", "mousemove", "mousedown", "mouseup", "click"].forEach(type => {
  636. captchaBox.dispatchEvent(new MouseEvent(type, {
  637. bubbles: true,
  638. cancelable: true,
  639. view: rawWindow,
  640. clientX,
  641. clientY,
  642. pageX: clientX + rawWindow.scrollX,
  643. pageY: clientY + rawWindow.scrollY
  644. }));
  645. });
  646. }
  647. })(cfg.get("apiKey2").length > 5);