您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Toggle for Searching Torrents via Search aggegrator
当前为
// ==UserScript== // @name Torrent Quick Search // @namespace https://github.com/TMD20/torrent-quick-search // @supportURL https://github.com/TMD20/torrent-quick-search // @version 1.58 // @description Toggle for Searching Torrents via Search aggegrator // @icon https://cdn2.iconfinder.com/data/icons/flat-icons-19/512/Eye.png // @author tmd // @noframes // @run-at document-end // @require https://openuserjs.org/src/libs/sizzle/GM_config.min.js // @grant GM.getValue // @grant GM.setValue // @grant GM.xmlHttpRequest // @grant GM.registerMenuCommand // @grant GM_config // @grant GM.notification // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @grant GM_notification // @require https://cdn.jsdelivr.net/npm/[email protected]/lib/semaphore.min.js // @require https://cdn.jsdelivr.net/npm/[email protected]/lib/semaphore.min.js // @match https://animebytes.tv/requests.php?action=viewrequest&id=* // @match https://animebytes.tv/series.php?id=* // @match https://animebytes.tv/torrents.php?id=* // @match https://blutopia.xyz/requests/* // @match https://blutopia.xyz/torrents/* // @match https://beyond-hd.me/requests/* // @match https://beyond-hd.me/torrents/* // @match https://beyond-hd.me/library/title/* // @match https://imdb.com/title/* // @match https://www.imdb.com/title/* // @match https://www.themoviedb.org/movie/* // @match https://www.themoviedb.org/tv/* // @license MIT // ==/UserScript== ` General Functions Functions that don't fit in any other catergory `; function recreateController() { controller = new AbortController(); } function semaphoreLeave() { if (sem && sem.current > 0) { sem.leave(); } } let searchObj = { ready: true, search() { if (controller.signal.aborted) { return Promise.reject(AbortError); } return new Promise(async (resolve, reject) => { controller.signal.addEventListener("abort", () => { reject(AbortError); }); document.querySelector("#torrent-quicksearch-msgnode").textContent = "Loading"; let indexers = await getIndexers(); document.querySelector("#torrent-quicksearch-msgnode").textContent = "Fetching Results From Indexers"; let imdb = await setIMDBNode(); setTitleNode(); //reset count let count = []; let length = indexers.length; let data = []; let x = Number.MAX_VALUE; while (indexers.length) { // x at a time let newData = await Promise.allSettled( indexers .splice(0, Math.min(indexers.length, x)) .map((e) => searchIndexer(e, imdb, length, count)) ); data = [...data, ...newData]; } console.log(data); let errorMsgs = data .filter((e) => e["status"] == "rejected") .map((e) => e["reason"].message); errorMsgs = [...new Set(errorMsgs)]; if (errorMsgs.length > 0) { reject(errorMsgs.join("\n")); } resolve(); }); }, cancel() { controller.abort(); }, async setup() { this.searchPromise = new Promise((resolve, reject) => { this.timeout = setTimeout(async () => { try { resolve(await this.search()); } catch (e) { reject(e); } }, 1000); }); }, async doSearch() { showDisplay(); recreateController(); await this.setup(); setTimeout(() => { resetResultList(); resetSearchDOM(); getTableHead(); }, 0); setTimeout(async () => { //reset sem = semaphore(10); try { await this.searchPromise; this.finalize(); } catch (error) { if (error.message.match(/aborted!/i) === null) { GM.notification(error.message, program, searchIcon); } console.log(error); } }, 100); }, finalize() { if ( Array.from(document.querySelectorAll(".torrent-quicksearch-resultitem")) .length == 0 ) { this.nomatchID = setTimeout( () => (document.querySelector( "#torrent-quicksearch-resultlist" ).textContent = "No Matches"), 1000 ); } this.finalmsgID = setTimeout( () => (document.querySelector("#torrent-quicksearch-msgnode").textContent = "Finished"), 1000 ); this.removemsgnodeID = setTimeout(() => { (document.querySelector("#torrent-quicksearch-msgnode").style.display = "none"), 3000; document.querySelector("#torrent-quicksearch-msgnode").textContent = ""; }); }, async toggleSearch() { let content = document.querySelector("#torrent-quicksearch-box"); if (content.style.display === "inline-block") { hideDisplay(); searchObj.cancel(); } else if ( content.style.display === "none" || content.style.display === "" ) { let customSearch = false; await this.doSearch(); } }, }; function searchIndexer(indexerObj, imdb, total, count) { if (controller.signal.aborted) { return Promise.reject(AbortError); } return new Promise(async (resolve, reject) => { let msg = null; controller.signal.addEventListener("abort", () => { reject(AbortError); }); let searchprogram = GM_config.get("searchprogram"); let data = null; if (searchprogram == "Prowlarr") { data = await searchProwlarrIndexer(indexerObj, controller); } else if (searchprogram == "Jackett") { data = await searchJackettIndexer(indexerObj); } else if (searchprogram == "NZBHydra2") { data = await searchHydra2Indexer(indexerObj); } msg = `Results fetched fom ${indexerObj["Name"]}:${ count.length + 1 }/${total} Indexers completed`; data = data.filter((e) => imdbFilter(e, imdbCleanup(imdb))); data.forEach((e) => { if (e["ImdbId"] == 0 || e["ImdbId"] == null) { e["ImdbId"] = imdbParserFail; } }); data = data.filter((e) => currSiteFilter(e["InfoUrl"])); addResultsTable(data); count.push(indexerObj["ID"]); document.querySelector("#torrent-quicksearch-msgnode").textContent = msg; console.log(msg); resolve(data); }); } async function searchProwlarrIndexer(indexer) { console.log(getSearchURLProwlarr(indexer["ID"])); let req = await fetch(getSearchURLProwlarr(indexer["ID"]), { timeout: indexerSearchTimeout, }); let data = JSON.parse(req.responseText) || []; let dataCopy = [...data]; let promiseArray = []; let x = Number.MAX_VALUE; while (dataCopy.length) { let newData = await Promise.allSettled( dataCopy.splice(0, Math.min(dataCopy.length, x)).map(async (e) => { return { Title: e["title"], Indexer: e["indexer"], Grabs: e["grabs"], PublishDate: e["publishDate"], Size: e["size"], Leechers: e["leechers"], Seeders: e["seeders"], InfoUrl: e["infoUrl"], DownloadUrl: e["downloadUrl"], ImdbId: e["imdbId"], Cost: e["indexerFlags"].includes("freeleech") == "100% Freeleech" ? "100% Freeleech" : "Cost Unknown With Prowlarr", Protocol: e["protocol"], }; }) ); promiseArray = [...promiseArray, ...newData]; } return promiseArray.map((e) => e["value"]).filter((e) => e != null); } async function searchJackettIndexer(indexer) { let req = await fetch(getSearchURLJackett(indexer["ID"]), { timeout: indexerSearchTimeout, }); let data = JSON.parse(req.responseText)["Results"] || []; let dataCopy = [...data]; let promiseArray = []; let x = Number.MAX_VALUE; while (dataCopy.length) { let newData = await Promise.allSettled( dataCopy.splice(0, Math.min(dataCopy.length, x)).map(async (e) => { return { Title: e["Title"], Indexer: e["Tracker"], Grabs: e["Grabs"], PublishDate: e["PublishDate"], Size: e["Size"], Leechers: e["Peers"], Seeders: e["Seeders"], InfoUrl: e["Details"], DownloadUrl: e["Link"], ImdbId: e["Imdb"], Cost: `${(1 - e["DownloadVolumeFactor"]) * 100}% Freeleech`, Protocol: "torrent", }; }) ); promiseArray = [...promiseArray, ...newData]; } return promiseArray.map((e) => e["value"]).filter((e) => e != null); } async function searchHydra2Indexer(indexer) { let req = await fetch(getSearchURLHydraTor(indexer["ID"]), { timeout: indexerSearchTimeout, }); let req2 = await fetch(getSearchURLHydraNZB(indexer["ID"]), { timeout: indexerSearchTimeout, }); let parser = new DOMParser(); let data = [ ...Array.from( parser .parseFromString(req.responseText, "text/xml") .querySelectorAll("channel>item") ), ...Array.from( parser .parseFromString(req2.responseText, "text/xml") .querySelectorAll("channel>item") ), ]; let dataCopy = [...data]; let promiseArray = []; let x = Number.MAX_VALUE; while (dataCopy.length) { let newData = await Promise.allSettled( dataCopy.splice(0, Math.min(dataCopy.length, x)).map(async (e) => //array is final dictkey,queryselector,attribute { let t = [ ["Title", "title", "textContent"], ["Indexer", "[name=hydraIndexerName]", "null"], ["Leechers", "[name=peers]", "null"], ["Seeders", "[name=seeders]", "null"], ["Cost", "[name=downloadvolumefactor]", "null"], ["PublishDate", "pubDate", "textContent"], ["Size", "size", "textContent"], ["InfoUrl", "comments", "textContent"], ["DownloadUrl", "link", "textContent"], ["ImdbId", "[name=imdb]", "null"], ]; let out = {}; out["Grabs"] = "Hydra Does not Report"; for (let i in t) { let key = t[i][0]; let node = e.querySelector(t[i][1]); let textContent = t[i][2] == "textContent"; if (!node) { continue; } if (textContent) { out[key] = node.textContent; } else if (key == "cost") { out[key] = `${(1 - node.getAttribute("value")) * 100}% Freeleech`; } else { out[key] = node.getAttribute("value"); } } out["Protocol"] = data[0].querySelector("enclosure").getAttribute("type") == "application/x-bittorrent" ? "torrent" : "usenet"; return out; } ) ); promiseArray = [...promiseArray, ...newData]; } return promiseArray.map((e) => e["value"]).filter((e) => e != null); } function fetch( url, { method = "GET", data = null, headers = {}, timeout = 90000, semaphore = true, } = {} ) { async function semforeFetch() { return new Promise((resolve, reject) => { sem.take(async () => { controller.signal.addEventListener("abort", () => { reject(AbortError); }); setTimeout(() => reject(AbortError), timeout); GM.xmlHttpRequest({ method: method, url: url, data: data, headers: headers, onload: (response) => { semaphoreLeave(); resolve(response); }, onerror: (response) => { semaphoreLeave(); reject(response.responseText); }, }); }); }); } async function normalFetch() { return new Promise((resolve, reject) => { controller.signal.addEventListener("abort", () => { reject(AbortError); }); setTimeout(() => reject(AbortError), timeout); GM.xmlHttpRequest({ method: method, url: url, data: data, headers: headers, onload: (response) => { resolve(response); }, onerror: (response) => { reject(response.responseText); }, }); }); } if (semaphore) { return semforeFetch(); } else { return normalFetch(); } } function getParser() { let siteName = standardNames[window.location.host] || window.location.host; let data = infoParser[siteName]; if (data === undefined) { let msg = "Could not get Parser"; GM.notification(msg, program, searchIcon); throw new Error(msg); } return data; } function verifyConfig() { if ( GM_config.get("searchapi", "null") == "null" || GM_config.get("searchurl", "null") == "null" ) { return false; } if ( GM_config.get("searchapi", "null") == "" || GM_config.get("searchurl", "null") == "" ) { return false; } return true; } ` DOM Manipulators These Functions are used to manipulate the DOM `; function setTitleNode() { if (customSearch == false) { document.querySelector("#torrent-quicksearch-customsearch").value = getTitle(); } } async function setIMDBNode() { let imdb = null; //Get Old IMDB if ( document.querySelector("#torrent-quicksearch-imdbinfo").textContent != imdbParserFail && document.querySelector("#torrent-quicksearch-imdbinfo").textContent .length != 0 && document.querySelector("#torrent-quicksearch-imdbinfo").textContent != "None" ) { imdb = document.querySelector("#torrent-quicksearch-imdbinfo").textContent; } //Else get New IMDB else { imdb = await getIMDB(); document.querySelector("#torrent-quicksearch-imdbinfo").textContent = imdb || imdbParserFail; } return imdb; } function resetSearchDOM() { document.querySelector("#torrent-quicksearch-imdbinfo").textContent = "None"; document.querySelector("#torrent-quicksearch-msgnode").textContent = "Waiting"; } function hideDisplay() { document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-size", `${iconSmall}%`); document.querySelector("#torrent-quicksearch-customsearch").value = ""; document.querySelector("#torrent-quicksearch-box").style.display = "none"; } function showDisplay() { document.querySelector("#torrent-quicksearch-msgnode").textContent = ""; document.querySelector("#torrent-quicksearch-msgnode").style.display = "block"; document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-size", `${iconLarge}%`); document.querySelector("#torrent-quicksearch-box").style.display = "inline-block"; } function getTableHead() { let node = document.querySelector("#torrent-quicksearch-resultheader"); node.innerHTML = ` <span class="torrent-quicksearch-resultcell" >Links</span> <span class="torrent-quicksearch-resultcell" >Clients</span> <span class="torrent-quicksearch-resultcell" >Title</span> <span class="torrent-quicksearch-resultcell" >Indexer</span> <span class="torrent-quicksearch-resultcell" >Grabs</span> <span class="torrent-quicksearch-resultcell" >Seeders</span> <span class="torrent-quicksearch-resultcell" >Leechers</span> <span class="torrent-quicksearch-resultcell" >DLCost</span> <span class="torrent-quicksearch-resultcell" >Date</span> <span class="torrent-quicksearch-resultcell">Size</span> <span class="torrent-quicksearch-resultcell">IMDB</span> `; Array.from(node.children).forEach((e, i) => { e.style.gridColumnStart = i + 1; e.style.fontSize = `${GM_config.get("fontsize", 12)}px`; }); } function addResultsTable(data) { if (data.length == 0) { return; } let resultList = document.querySelector("#torrent-quicksearch-resultlist"); let tempFrag = new DocumentFragment(); data.forEach((e, i) => { let node = document.createElement("span"); node.setAttribute("class", "torrent-quicksearch-resultitem"); node.innerHTML = ` <span class="torrent-quicksearch-resultcell torrent-quicksearch-links" style='grid-column-start:1' > <a href=${e["DownloadUrl"]}>Download</a> <br> <br> <a href=${e["InfoUrl"]}>Details</a> </span> <span style='grid-column-start:2'> <form> <span> <select class=torrent-quicksearch-clientSelect> </select> </span> <span> <span class="tooltip"> <button class=torrent-quicksearch-clientSubmit>Send</button> <span class="tooltiptext">Arr Clients imdbID sent from entry if null then page</span> </span> </span> </form> </span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:3' >${ e["Title"] }</span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:4' >${ e["Indexer"] }</span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:5'>${ e["Grabs"] || "No Data" } </span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:6'>${ e["Seeders"] || "No Data" } </span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:7' >${ e["Leechers"] || "No Data" } </span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:8'>${ e["Cost"] } </span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:9' >${new Date( e["PublishDate"] ).toLocaleString("en-CA")}</span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:10' >${( parseInt(e["Size"]) / 1073741824 ).toFixed(2)} GB</span> <span class="torrent-quicksearch-resultcell" style='grid-column-start:11' >${ e["ImdbId"] }</span>`; let selNode = node.querySelector("select"); JSON.parse(GM_config.getValue("downloadClients", "[]")).forEach((e) => { let optnode = document.createElement("option"); optnode.setAttribute("id", e.clientID); optnode.setAttribute("value", e.clientID); optnode.textContent = e.clientName; selNode.appendChild(optnode); }); node.querySelector("form").addEventListener("submit", clientFactory(e)); tempFrag.append(node); }); resultList.appendChild(tempFrag); } function resetResultList() { document.querySelector("#torrent-quicksearch-resultheader").textContent = ""; document.querySelector("#torrent-quicksearch-resultlist").textContent = ""; } function createMainDOM() { const box = document.createElement("div"); box.setAttribute("id", "torrent-quicksearch-overlay"); let rowSplit = 12; let contentWidth = 70; let boxMinHeight = 5; let boxMaxHeight = 100; let boxHeight = 40; let boxWidth = 70; let boxMaxWidth = 150; box.innerHTML = ` <div> <img id="torrent-quicksearch-toggle" src="${searchIcon}"></img> <div id="torrent-quicksearch-box"> <div id="torrent-quicksearch-content"> <div> <div id="torrent-quicksearch-msgnode"></div> <div id="torrent-quicksearch-custombox"> <div> <label>Title:</label> <input type="text" id="torrent-quicksearch-customsearch"> <label>Page IMDB:</label> <div id="torrent-quicksearch-imdbinfo">None</div> <button id="torrent-quicksearch-customsearchbutton">Custom Search</button> </div> </div> <div id="torrent-quicksearch-resultheader"></div> </div> <div id="torrent-quicksearch-resultlist"> </div> </div> </div> <style> /* Variables */ #torrent-quicksearch-overlay { --grid-size: max(calc(50vw/${rowSplit}),calc(100%/${rowSplit})); --icon-size:${iconSmall}%; --icon-padding:${paddingSmall}%; } #torrent-quicksearch-overlay { position: sticky; display: flex; flex-direction: column; gap: 10px; top: 40vh; pointer-events: none; z-index: 900000; } #torrent-quicksearch-overlay> div:first-of-type { position: absolute; left:80vw; } * { font-size:${GM_config.get("fontsize", 12)}px; } #torrent-quicksearch-toggle { margin-left: auto; display:block; cursor: pointer; pointer-events:all; width: var(--icon-size); height: var(--icon-size); padding-top: var(--icon-padding); padding-bottom:var(--icon-padding); margin-bottom:calc(${paddingLarge}vh - var(--icon-padding)); margin-top:calc(${paddingLarge}vh - var(--icon-padding)); } #torrent-quicksearch-box{ resize:both; direction:rtl; right:5vw; margin-right:auto; position:absolute; display:none; min-height: ${boxMinHeight}vh; max-height:${boxMaxHeight}vh; height: ${boxHeight}vh; width: ${boxWidth}vw; max-width: ${boxMaxWidth}vw; overflow:hidden; border:solid black 5px; } #torrent-quicksearch-msgnode{ background-color:#FFFFFF; width:calc(var(--grid-size)*${rowSplit}); display:none; height:calc(((${GM_config.get("fontsize", 12)}em) + 2em)/16); } #torrent-quicksearch-custombox { background-color:#FFFFFF; width:calc(var(--grid-size)*${rowSplit}); pointer-events:all; height:calc(((${GM_config.get("fontsize", 12)}em) + 2em) * (2/16)); } #torrent-quicksearch-custombox>div { display: flex; background-color:#FFFFFF; flex-direction:row; justify-content: center; width: 100% } #torrent-quicksearch-custombox>div >label { margin-left:2.5%; margin-right:2.5%; } #torrent-quicksearch-custombox>div >button { margin-left:2.5%; } #torrent-quicksearch-customsearch{ background-color:#FFFFFF; border:solid black 2px; flex-grow:1; } #torrent-quicksearch-custombox > label{ margin-left:2% margin-right:2% } #torrent-quicksearch-customsearchbutton { background-color: #4CAF50; border: none; color: white; text-align: center; text-decoration: none; font-size: ${GM_config.get("fontsize", 12) + 2}px; border-radius: 5px; } #torrent-quicksearch-content { pointer-events:all; background-color: #D7C49EFF; direction:ltr; height:100%; width:100%; } #torrent-quicksearch-content>div:nth-child(2) { scrollbar-color: white; overflow:scroll; width:100%; height:calc(100% - ((${GM_config.get("fontsize", 12)}em) + 2em)*(4/16)); } #torrent-quicksearch-content>div:nth-child(1) { width:100%; background-color: #B1D79E; } #torrent-quicksearch-resultlist{ border:solid white 5px; width:calc(var(--grid-size)*${rowSplit}); } .torrent-quicksearch-resultitem,#torrent-quicksearch-resultheader{ display: grid; grid-template-columns: repeat(${rowSplit},var(--grid-size)); width:calc((var(--grid-size)*${rowSplit})-10); } .torrent-quicksearch-resultitem{ font-size:${GM_config.get("fontsize", 12)}px; } #torrent-quicksearch-resultlist>.torrent-quicksearch-resultitem:nth-child(even) { background-color: #D7C49EFF; } #torrent-quicksearch-resultlist>.torrent-quicksearch-resultitem:nth-child(odd) { background-color: #d5cbcb; } #torrent-quicksearch-resultheader{ background-color: #B1D79E; font-size:${GM_config.get("fontsize", 12) + 2}px; height:calc(((${GM_config.get("fontsize", 12)}em) + 2em)*(2/16)); } .torrent-quicksearch-resultcell{ font-weight: bold; margin-left: 10%; overflow-wrap:break-word; } .torrent-quicksearch-clientSubmit { background-color: white; border: none; text-align: center; text-decoration: none; font-size:${GM_config.get("fontsize", 12) + 2}px; font-weight: bold; overflow: hidden; white-space: nowrap; width:100%'' } .torrent-quicksearch-clientSelect { font-size:${GM_config.get("fontsize", 12) + 2}px; font-weight: bold; width:100%; } .torrent-quicksearch-links *{ color:blue; cursor:pointer; text-decoration: none; } .torrent-quicksearch-links *:hover{ color:white; } .torrent-quicksearch-links *:active,focus{ animation: pulse 2s infinite; } @keyframes pulse { 0% ,100%{ color: blue; } 50% { color: white; } } ::-webkit-scrollbar-thumb{ background-color:white; } /* Tooltip container */ .tooltip { position: relative; display: inline-block; border-bottom: 1px dotted black; /* If you want dots under the hoverable text */ } /* Tooltip text */ .tooltip .tooltiptext { visibility: hidden; width: 120px; background-color: black; color: #fff; text-align: center; padding: 5px 0; border-radius: 6px; /* Position the tooltip text - see examples below! */ position: absolute; z-index: 1; } /* Show the tooltip text when you mouse over the tooltip container */ .tooltip:hover .tooltiptext { visibility: visible; } <style/>`; box .querySelector("#torrent-quicksearch-toggle") .addEventListener("mousedown", leftClickProcess); box .querySelector("#torrent-quicksearch-toggle") .addEventListener("mouseup", mouseUpProcess); document.addEventListener("mouseup", resetMouse); box .querySelector("#torrent-quicksearch-customsearchbutton") .addEventListener("click", () => { searchObj.cancel(); setTimeout(() => { if (Date.now() - lastClick < clickLimit) { return; } lastClick = Date.now(); let customSearch = true; searchObj.doSearch(); }, 0); }); document.body.insertBefore(box, document.body.children[0]); } ` Matching Function These help with finding a Match `; function getTitle() { let titleNode = document.querySelector(siteParser["title"]); if (titleNode == null) { throw new Error("Title Node Not Found"); } let title = titleNode[siteParser["titleAttrib"]]; title = titleCleanup(title); return title; } async function getIMDB() { let imdb = null; if (standardNames[window.location.host] == "imdb.com") { imdb = window.location.href; } else if (standardNames[window.location.host] == "themoviedb.org") { imdb = await tmdbPageIMDBParser(); } else { let imdbNode = document.querySelector(siteParser["imdb"]); if (imdbNode == null) { return null; } imdb = imdbNode[siteParser["imdbAttrib"]]; } imdb = imdbCleanup(imdb); return imdb; } function titleCleanup(title) { title = title.trim().replaceAll(/\n/g, ""); title = title.replaceAll(/ +/g, " "); return title; } function imdbCleanup(imdb) { if (imdb === null || imdb === undefined || imdb === 0 || imdb === "0") { return imdb; } imdb = String(imdb); imdb = imdb.match(/[0-9]+/).toString(); imdb = imdb.trim().replaceAll(/\n/g, ""); imdb = imdb.replace(/imdb/i, ""); imdb = imdb.replace(/tt/, ""); imdb = imdb.replace(/[:|&|*|\(|\)|!|@|#|$|%|^|\*|\\|\/]/, ""); imdb = imdb.replaceAll(/ +/g, ""); imdb = imdb.replace(/^0+/, ""); imdb = parseInt(imdb); return imdb; } function imdbFilter(entry, imdb) { if (imdb === null || imdb === "IMDB Not Provided") { return true; } else if (entry["ImdbId"] == 0) { return true; } else if (entry["ImdbId"] == imdb) { return true; } return false; } async function tmdbExternalMedia(type, id) { let key = GM_config.get("tmdbapi", "null"); if (key == "null") { return null; } let baseURL = new URL( `/3/movie/${id}/external_ids`, `https://api.themoviedb.org` ).toString(); let params = new URLSearchParams(); params.append("api_key", key); if (type == "tv") { baseURL = new URL( `/3/tv/${id}/external_ids`, `https://api.themoviedb.org` ).toString(); } let searchURL = `${baseURL}?${params.toString()}`; let req = await fetch(searchURL); return JSON.parse(req.responseText); } async function imdbTMDBConvertor(imdb) { let key = GM_config.get("tmdbapi", "null"); imdb = String(imdb); if (key == "null") { return null; } if (imdb.match(/tt/i) == null) { imdb = `tt${imdb}`; } let baseURL = new URL( `/3/find/${imdb}`, `https://api.themoviedb.org` ).toString(); let params = new URLSearchParams(); params.append("api_key", key); params.append("external_source", "imdb_id"); let searchURL = `${baseURL}?${params.toString()}`; let req = await fetch(searchURL); let data = JSON.parse(req.responseText); let output = data["tv_results"]; if (data["movie_results"].length > output.length) output = data["movie_results"]; return output[0]; } // First call to tmdbapi should be removed once we parse Movies vs TV async function tmdbTVDBConvertor(imdb) { let key = GM_config.get("tmdbapi", "null"); if (key == "null") { return null; } let helperData = await imdbTMDBConvertor(imdb); if (helperData == null) { return; } else if (helperData["media_type"] == "tv") { return (await tmdbExternalMedia("tv", helperData.id))["tvdb_id"]; } else if (helperData["media_type"] == "movie") { return (await tmdbExternalMedia("movie", helperData.id))["tvdb_id"]; } } async function tmdbPageIMDBParser() { let id = window.location.href .match(/\/[0-9]+/) .toString() .substring(1); if (window.location.href.match(/\/tv\//)) { return (await tmdbExternalMedia("tv", id))["imdb_id"]; } else { return (await tmdbExternalMedia("movie", id))["imdb_id"]; } } function currSiteFilter(entryURL) { if (GM_config.get("sitefilter") == "false") { return true; } if ( new URL(entryURL).hostname.replace(/\..*$/, "") == window.location.host.replace(/\..*$/, "") ) { return false; } return true; } ` URL Processing Indexer Functions Used to Get Indexer Info `; async function getIndexers() { document.querySelector("#torrent-quicksearch-msgnode").textContent = "Getting Indexers"; let searchprogram = GM_config.get("searchprogram"); let indexers = null; if (searchprogram == "Prowlarr") { indexers = await getIndexersProwlarr(); } else if (searchprogram == "Jackett") { indexers = await getIndexersJackett(); } else if (searchprogram == "NZBHydra2") { indexers = await getIndexersHydra(); } await indexerCacheHelper(indexers); return await listFilter(indexers); } async function getIndexersJackett() { let key = "jackettIndexers"; let cachedIndexers = await GM.getValue(key, "none"); if (cachedIndexers == "none") { null; } else if (Date.now() - (cachedIndexers?.date || 0) < day) { return cachedIndexers["indexers"]; } let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); //Invalid Category Search params.append("Category[]", 20130); params.append("Query", "''"); let baseURL = new URL( `/api/v2.0/indexers/all/results`, `${GM_config.get("searchurl")}` ).toString(); let indexerURL = `${baseURL}?${params.toString()}`; let req = await fetch(indexerURL); let data = JSON.parse(req.responseText)["Indexers"]; let output = data.map((e) => { let dict = {}; dict["Name"] = e["Name"]; dict["ID"] = e["ID"]; return dict; }); await GM.setValue(key, { date: Date.now(), indexers: output, }); return output; } async function getIndexersProwlarr() { let key = "prowlarrIndexers", cachedIndexers = await GM.getValue(key, "none"); if (cachedIndexers == "none") { null; } else if (Date.now() - (cachedIndexers?.date || 0) < day) { return cachedIndexers["indexers"]; } let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); let baseURL = new URL( `/api/v1/indexer`, `${GM_config.get("searchurl")}` ).toString(); let indexerURL = `${baseURL}?${params.toString()}`; let req = await fetch(indexerURL); let data = JSON.parse(req.responseText); data = data.sort(prowlarIndexSortHelper); let output = data.map((e) => { let dict = {}; dict["Name"] = e["name"]; dict["ID"] = e["id"]; return dict; }); await GM.setValue(key, { date: Date.now(), indexers: output, }); return output; } function prowlarIndexSortHelper(a, b) { if (a["priority"] > b["priority"]) { return -1; } if (a["priority"] < b["priority"]) { return 1; } return 0; } async function getIndexersHydra() { let key = "hydraIndexers"; let cachedIndexers = await GM.getValue(key, "none"); if (cachedIndexers == "none") { null; } else if (Date.now() - (cachedIndexers?.date || 0) < day) { return cachedIndexers["indexers"]; } let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); let baseURL = new URL( `/api/stats/indexers/`, `${GM_config.get("searchurl")}` ).toString(); let indexerURL = `${baseURL}?${params.toString()}`; let req = await fetch(indexerURL); let data = JSON.parse(req.responseText); let output = data.map((e) => { let dict = {}; dict["Name"] = e["indexer"]; dict["ID"] = e["indexer"]; return dict; }); await GM.setValue(key, { date: Date.now(), indexers: output, }); return output; } async function listFilter(allIndexers) { let selectedIndexers = null; if (GM_config.get("listType") == "black") { selectedIndexers = await blackListHelper(allIndexers); } else { selectedIndexers = await whiteListHelper(allIndexers); } let output = []; for (let i in allIndexers) { if (selectedIndexers.has(allIndexers[i]["ID"])) { output.push(allIndexers[i]); } } return output; } async function indexerCacheHelper(allIndexers) { if (GM_config.get("indexers") == "") { return; } let searchprogram = GM_config.get("searchprogram"); let indexerNames = GM_config.get("indexers") .split(",") .map((e) => e.trim().toLowerCase()); for (let j in indexerNames) { let key = `${searchprogram}_${indexerNames[j]}`; let cached = await GM.getValue(key, "none"); if (cached != "none") { continue; } for (let i in allIndexers) { if (allIndexers[i]["Name"].match(new RegExp(indexerNames[j], "i"))) { await GM.setValue(key, allIndexers[i]["ID"]); } } } } async function blackListHelper(allIndexers) { let indexerID = new Set(allIndexers.map((e) => e["ID"])); let indexerNames = GM_config.get("indexers") .split(",") .map((e) => e.trim()); let searchprogram = GM_config.get("searchprogram"); for (let j in indexerNames) { let key = `${searchprogram}_${indexerNames[j]}`; let cached = await GM.getValue(key, "none"); if (cached != "none") { indexerID.delete(cached); } } return indexerID; } async function whiteListHelper(allIndexers) { let indexerID = new Set(); let indexerNames = GM_config.get("indexers") .split(",") .map((e) => e.trim()); let searchprogram = GM_config.get("searchprogram"); for (let j in indexerNames) { let key = `${searchprogram}_${indexerNames[j]}`; let cached = await GM.getValue(key, "none"); if (cached != "none") { indexerID.add(cached); } } return indexerID; } ` URL Processing Search Functions Used to Produce URL for Search `; function getSearchURLProwlarr(indexer) { let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); params.append( "query", `${document.querySelector("#torrent-quicksearch-customsearch").value}` ); params.append("IndexerIds", indexer); let baseURL = new URL( "/api/v1/search", GM_config.get("searchurl") ).toString(); return `${baseURL}?${params.toString()}`; } function getSearchURLJackett(indexer) { let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); params.append( "Query", `${document.querySelector("#torrent-quicksearch-customsearch").value}` ); let baseURL = new URL( `/api/v2.0/indexers/${indexer}/results`, GM_config.get("searchurl") ).toString(); params.append("cachetime", "20"); return `${baseURL}?${params.toString()}`; } function getSearchURLHydraNZB(indexer) { let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); params.append( "q", `${document.querySelector("#torrent-quicksearch-customsearch").value}` ); params.append("indexers", indexer); params.append("t", "search"); params.append("o", "xml"); params.append("cachetime", "20"); let baseURL = new URL("/api", GM_config.get("searchurl")).toString(); return `${baseURL}?${params.toString()}`; } function getSearchURLHydraTor(indexer) { let params = new URLSearchParams(); params.append("apikey", GM_config.get("searchapi")); params.append( "q", `${document.querySelector("#torrent-quicksearch-customsearch").value}` ); params.append("indexers", indexer); params.append("t", "search"); params.append("o", "xml"); //hydra likes to send no data params.append("cachetime", "20"); let baseURL = new URL("/torznab/api", GM_config.get("searchurl")).toString(); return `${baseURL}?${params.toString()}`; } ` client Functions Functions to support Clients `; function getSonarrURL(clientURL, clientAPI) { let params = new URLSearchParams(); params.append("apikey", clientAPI); let baseURL = new URL("/api/v3/release/push", clientURL).toString(); return `${baseURL}?${params.toString()}`; } function getRadarrURL(clientURL, clientAPI) { let params = new URLSearchParams(); params.append("apikey", clientAPI); let baseURL = new URL("/api/v3/release/push", clientURL).toString(); return `${baseURL}?${params.toString()}`; } ` Events These Functions Create Events to be used by script `; function leftClickProcess(e) { e.preventDefault(); e.stopPropagation(); if (e.button != 0) { return; } mouseState = "down"; document.addEventListener("mousemove", mouseDragProcess); document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-padding", `${paddingLarge}vh`); } function mouseUpProcess(e) { e.preventDefault(); e.stopPropagation(); mouseClicksProcess(); resetMouse(); } async function mouseClicksProcess() { if (mouseState == "dragged") { return; } else if (Date.now() - lastClick < clickLimit) { return; } else if (verifyConfig() == false) { GM.notification( "At Minimum You Need to Set\nSearch URl\nSearch API\nSearch Program", program, searchIcon ); GM_config.open(); return; } lastClick = Date.now(); await searchObj.toggleSearch(); } //Reset Mouse Events function resetMouse() { mouseState = "up"; document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-padding", `${paddingSmall}vh`); document.removeEventListener("mousemove", mouseDragProcess); } function mouseDragProcess(e) { mouseState = "dragged"; //check mouse state on enter if (mouseState == "up" || e.buttons == 0) { return; } //poll mouse state setInterval( () => { if (mouseState == "up" || e.buttons == 0) { return; } }, 1000 ); let dragState = true; let toggleHeight = parseInt( getComputedStyle( document.querySelector("#torrent-quicksearch-toggle") ).height.replaceAll(/[^0-9.]/g, "") ); let startMousePosition = parseInt(e.clientY); let offsetMousePosition = startMousePosition - toggleHeight / 2; let viewport = (offsetMousePosition / window.innerHeight) * 100; viewport = Math.max(viewport, -20); viewport = Math.min(viewport, 89); document.querySelector( "#torrent-quicksearch-overlay" ).style.top = `${viewport}vh`; } ` Client Functions Functions For Sending Downloads to Clients `; function arrIMDBHelper(releaseData) { let pageIMDB = document.querySelector( "#torrent-quicksearch-imdbinfo" ).textContent; if (releaseData["ImdbId"] != imdbParserFail) { return parseInt(releaseData["ImdbId"]); } else if (pageIMDB != imdbParserFail && pageIMDB != null) { return parseInt(pageIMDB); } else { return; } } async function sendSonarrClient(releaseData, clientData) { releaseData["ImdbId"] = arrIMDBHelper(releaseData); (releaseData["TvdbId"] = await tmdbTVDBConvertor(releaseData["ImdbId"])), (releaseData["tmdbId"] = ( await imdbTMDBConvertor(releaseData["ImdbId"]) )?.id); let res = await fetch( getSonarrURL(clientData.clientURL, clientData.clientAPI), { method: "post", data: JSON.stringify(releaseData), semaphore: false, } ); if (res.status != 200) { GM.notification(res.responseText, program, searchIcon); return; } let data = JSON.parse(res.responseText); let initValue = ""; let finalMsg = data.reduce((prev, curr, index) => { if (curr["rejections"].length > 0) { let epNums = curr["mappedEpisodeNumbers"].length > 0 ? curr["mappedEpisodeNumbers"].join(",") : "No Episodes"; let errorMsg = [ `${curr["seriesTitle"]} Season ${curr["seasonNumber"]} Episodes ${epNums}`, `Status Rejected: ${curr["rejections"].join(",")}`, ]; return `${prev}\n\[${errorMsg.join("\n")}\]`; } else if (curr["approved"] == true) { let acceptMsg = `Added ${curr["title"]} to client`; return `${prev}\n${acceptMsg}`; } }, initValue); GM.notification(finalMsg, program, searchIcon); } async function sendRadarrClient(releaseData, clientData) { releaseData["ImdbId"] = arrIMDBHelper(releaseData); (releaseData["TvdbId"] = await tmdbTVDBConvertor(releaseData["ImdbId"])), (releaseData["tmdbId"] = ( await imdbTMDBConvertor(releaseData["ImdbId"]) )?.id); let res = await fetch( getRadarrURL(clientData.clientURL, clientData.clientAPI), { method: "post", data: JSON.stringify(releaseData), headers: { "content-type": "application/json" }, semaphore: false, } ); if (res.status != 200) { GM.notification(res.responseText, program, searchIcon); return; } let data = JSON.parse(res.responseText); let initValue = ""; let finalMsg = data.reduce((prev, curr, index) => { if (curr["rejected"] == true) { let errorMsg = [ `${curr["movieTitles"].join(",")}`, `Status Rejected: ${curr["rejections"].join(",")}`, ]; return `${prev}\n\[${errorMsg.join("\n")}\]`; } if (curr["approved"] == true) { let acceptMsg = `Added ${curr["title"]} to client`; return `${prev}\n${acceptMsg}`; } }, initValue); GM.notification(finalMsg, program, searchIcon); } function clientFactory(releaseData) { let clientEvent = async function (e) { e.preventDefault(); e.stopPropagation(); let clientData = JSON.parse( GM_config.getValue("downloadClients", "[]") ).filter( (ele) => ele.clientID == e.target.querySelector("select").value )[0]; if (clientData.clientType == "Sonarr") { sendSonarrClient(releaseData, clientData); } else if (clientData.clientType == "Radarr") { sendRadarrClient(releaseData, clientData); } }; return clientEvent; } ` GM_config Functions `; function addNewClient(e) { e.preventDefault() e.stopPropagation() saveDownloadClient(); recreateDownloadClientNode(); } function saveDownloadClient() { let wrapper = GM_config.fields["downloadclients"].wrapper; function verify(obj) { let missingName = "Could Not add new Client\nClientName is missing"; let missingValue = `Could Not add new Client\nClientType ${obj["clientType"]} is missing one of it's required values`; if (Object.values(obj).filter((e) => e != "").length <= 3) { return false; } if (obj["clientName"] == "") { GM.notification(missingName, program, searchIcon); return false; } if (obj["clientType"] == "Sonarr" || obj["clientType"] == "Radarr") { if (obj["clientURL"] != "" && obj["clientAPI"] != "") { return true; } GM.notification(missingValue, program, searchIcon); return false; } //Add More conditionals for other clients else { if (obj["clientURL"] != "" && obj["clientAPI"] != "") { return true; } GM.notification(missingValue, program, searchIcon); return false; } } if (wrapper) { let inputs = wrapper.querySelectorAll("input,select"); let val = JSON.parse(GM_config.getValue("downloadClients", "[]")); let outdict = {}; outdict["clientID"] = Array(10) .fill() .map(() => "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789".charAt( Math.random() * 62 ) ) .join(""); for (let i in inputs) { let ele = inputs[i]; outdict[ele.id] = ele.value; } if (verify(outdict)) { val.push(outdict); GM_config.setValue("downloadClients", JSON.stringify(val)); } } } function getCurrentDownloadClientsNode() { let parent = document.createElement("div"); parent.setAttribute("id", "torrent-quicksearch-downloadclientsParent"); let titleNode = document.createElement("h1"); titleNode.textContent = "Current Clients"; parent.append(titleNode); let clients = JSON.parse(GM_config.getValue("downloadClients", "[]")); for (let i in clients) { //delete button let button = document.createElement("button"); button.setAttribute("class", "torrent-quicksearch-downloadclientsDelete"); button.textContent = "Delete Client"; button.addEventListener("click", deleteClient); parent.append(button); //client info box let client = clients[i]; let keys = Object.keys(client); let section = document.createElement("div"); section.setAttribute("class", "torrent-quicksearch-downloadclients"); section.append(); for (let j in keys) { let key = keys[j]; let value = client[key]; let node = document.createElement("div"); let keyNode = document.createElement("span"); keyNode.textContent = `${key}: `; keyNode.style.display = "inline-block"; keyNode.style.fontWeight = "bold"; keyNode.style.marginRight = "5px"; let valNode = document.createElement("span"); valNode.textContent = `${value}`; valNode.style.display = "inline-block"; node.append(keyNode); node.append(valNode); section.append(node); } parent.append(section); } return parent; } function deleteClient(e) { let clientNode = e.target.nextElementSibling; let clientID = Array.from(clientNode.childNodes).filter((e) => e.textContent.match(/clientID/) )[0].childNodes[1].textContent; let clients = JSON.parse(GM_config.getValue("downloadClients", "[]")); GM_config.setValue( "downloadClients", JSON.stringify(clients.filter((e) => e.clientID != clientID)) ); recreateDownloadClientNode(); } function recreateDownloadClientNode() { let wrapper = GM_config.fields["downloadclients"].wrapper; let oldParent = wrapper.querySelector( "#torrent-quicksearch-downloadclientsParent" ); let newParent = getCurrentDownloadClientsNode(); oldParent.parentElement.replaceChild(newParent, oldParent); } function downloadClientNode(configId) { var field = this.settings, id = this.id, create = this.create, retNode = create("div", { className: "config_var", id: configId + "_" + id + "_var", title: field.title || "", }); let currentClients = getCurrentDownloadClientsNode(); let newSubmissionForm = document.createElement("div"); newSubmissionForm.innerHTML = ` <h1>Add New Client</h1> <form> <div> <label for="client-select">Client Type:</label> <select id="clientType"> <option value="Sonarr">Sonarr</option> <option value="Radarr"option>Radarr</option> </select> </div> <br> <div> <label for="Name">Client Name:</label> <input type="text" placeholder="Name" id="clientName"> </div> <br> <div> <label for="clientURL">Client URL:</label> <input type="text" placeholder="URL" id="clientURL"> </div> <br> <div> <label for="clientAPI">Client API:</label> <input type="text" placeholder="API" id="clientAPI"> </div> <br> <div> <label for="clientUserName">Client Username:</label> <input type="text" placeholder="Username" id="clientUserName"> </div> <br> <div> <label for="clientPassword">Client Password:</label> <input type="" placeholder="Password" id="clientPassword"> </div> <button id=torrent-quicksearch-downloadclientsAdd>Add Client</button> </form> `; newSubmissionForm.querySelector("button").addEventListener("click", addNewClient); retNode.appendChild(currentClients); retNode.appendChild(newSubmissionForm); return retNode; } function openMenu() { hideDisplay(); resetResultList(); resetSearchDOM(); document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-size", `${iconLarge}%`); document .querySelector("#torrent-quicksearch-toggle") .removeEventListener("mousedown", leftClickProcess); document .querySelector("#torrent-quicksearch-toggle") .removeEventListener("mouseup", mouseUpProcess); searchObj.cancel(); } function closeMenu() { /* url = GM_config.get('searchurl') if (url.match(/htt(p|ps):\/\//) == null) { url = `http://${url}` } GM_config.set('searchurl', url) GM_config.save()*/ hideDisplay(); document .querySelector("#torrent-quicksearch-overlay") .style.setProperty("--icon-size", `${iconSmall}%`); document .querySelector("#torrent-quicksearch-toggle") .addEventListener("mousedown", leftClickProcess); document .querySelector("#torrent-quicksearch-toggle") .addEventListener("mouseup", mouseUpProcess); } function initConfig() { GM_config.init({ id: "torrent-quick-search", title: "Torrent Quick Search Settings", // Panel Title fields: { tmdbapi: { label: "TMDB API Key", type: "text", title: "TMDB Key For TMDB/IMDB conversion\nAlso better media matching for arr clients", }, searchurl: { label: "Search URL", section: ["Search"], type: "text", title: "Base URl for search program", }, searchapi: { label: "API Key", type: "text", title: "API key for search program", }, searchprogram: { label: "Search Program", type: "select", options: ["Prowlarr", "Jackett", "NZBHydra2"], title: "Which search program", }, indexers: { label: "Indexers", section: ["Indexers"], type: "text", title: "Comma Seperated List of Indexers Names\nYou can just use part of the indexer name\nIt just needs to be a substring of name in search program", }, listType: { type: "radio", options: ["black", "white"], label: "Indexers ListType", title: "Use White list if you want to manually approve indexers\nUse Black list if you want to use all Indexers, and just have few or none to disable", default: "black", }, sitefilter: { label: "Filter Current Site", type: "radio", options: ["true", "false"], title: "Should Results From Current Site be Filtered Out", default: "false", }, fontsize: { label: "Font Size", section: ["GUI"], type: "int", title: "fontsize", default: 12, }, downloadclients: { section: ["Download Clients"], type: "downloadclient", }, }, types: { downloadclient: { default: null, toNode: downloadClientNode, toValue: function () { return; }, reset: function () { GM_config.setValue("downloadClients", "[]"); }, }, }, events: { open: openMenu, close: closeMenu, }, css: ` .torrent-quicksearch-downloadclients{ border:solid black 5px; margin-bottom:5px; } .torrent-quicksearch-downloadclientsDelete,#torrent-quicksearch-downloadclientsAdd{ margin-bottom:5px; background-color: gray; border: none; color: white; text-align: center; text-decoration: none; font-size: 20px; } `, }); GM.registerMenuCommand("Torrent Quick Search Settings", function () { GM_config.open(); }); } ` Globals + Main Function `; let searchIcon = ""; let iconLarge = 50; let iconSmall = 25; let paddingSmall = 0; let paddingLarge = 4; let mouseState = "up"; let imdbParserFail = "Could Not Parse"; let AbortError = new DOMException("aborted!", "AbortError"); var controller = new AbortController(); let program = "Torrent Quick Search"; let clickLimit = 500; let lastClick = Date.now() - clickLimit; let indexerSearchTimeout = 30000; let day = 86400000; let customSearch = false; let sem = null; //keep track of whether program was recently dragged //Normalize Site Name so we don't have repeat keys in larger infoparser dict let standardNames = { "www.imdb.com": "imdb.com", "www.themoviedb.org": "themoviedb.org", }; let infoParser = { "animebytes.tv": { title: "h2>a[href*=series]", titleAttrib: "textContent", }, "blutopia.xyz": { title: "h1>a[href*=torrents\\/similar]", titleAttrib: "textContent", imdb: "div[class*=movie-details]>span>a[title=IMDB]", imdbAttrib: "textContent", }, "beyond-hd.me": { title: "h1[class=movie-heading]", titleAttrib: "textContent", imdb: "ul[class*=movie-details]>li>span>a[href*=imdb]", imdbAttrib: "href", }, "imdb.com": { title: "h1", titleAttrib: "textContent", }, "themoviedb.org": { title: "h2", titleAttrib: "textContent", }, }; let siteParser = getParser(); function overideBuiltins() { URL = class extends URL { constructor(url, base) { if (url == undefined && base == undefined) { null; } else if (base != undefined && base.match(/(http|https)/) == null) { base = `http://${base}`; } else if (base == undefined && url.match(/(http|https)/) == null) { url = `http://${url}`; } super(url, base); } }; } function main() { overideBuiltins(); initConfig(); createMainDOM(); } main();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址