// ==UserScript== // @name Torrent Quick Search // @namespace https://github.com/TMD20/torrent-quick-search // @supportURL https://github.com/TMD20/torrent-quick-search // @version 1.0 // @description Toggle for Searching Torrents via Search aggegrator // @icon https://cdn2.iconfinder.com/data/icons/flat-icons-19/512/Eye.png // @author tmd // @noframes // @inject-into page // @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 // @match https://animebytes.tv/requests.php?action=viewrequest&id=* // @match https://animebytes.tv/series.php* // @match https://animebytes.tv/torrents.php* // @match https://blutopia.xyz/requests* // @match https://blutopia.xyz/torrents/* // @match https://beyond-hd.me/request/* // @match https://beyond-hd.me/torrents/* // @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 // @downloadURL none // ==/UserScript== ` General Functions Functions that don't fit in any other catergory ` function getTitle() { if (infoparser===undefined){ throw new Error("Title not parsed"); } title=document.querySelector(infoparser["title"]).textContent title=titleCleanup(title) console.log(title) return title } function titleCleanup(title){ return title.trim().replaceAll(/\n/g,"").replaceAll(/ +/g," ") } function toggleSearch(e){ content=document.querySelector("#searchcontent") if (content.style.visibility === "visible") { content.style.pointerEvents="none" document.querySelector("#toggle").style.height='2%' document.querySelector("#toggle").style.width='2%' content.style.visibility = "hidden"; } else { content.style.pointerEvents="all" document.querySelector("#toggle").style.height='5%' document.querySelector("#toggle").style.width='5%' content.style.visibility = "visible"; doSearch() } } async function doSearch(){ document.querySelector("#msgnode").textContent="Loading" reqs=[] try { resetResultList() url=await getBaseURL() indexers=await getIndexers() getTableHead() document.querySelector("#msgnode").textContent="Fetching Results" data = await Promise.allSettled(indexers.map((e)=>fetchIndexer(e))); addNumbers() console.log("Finished Fetching") document.querySelector("#msgnode").textContent="Finished" } catch(error) { alert(error) document.querySelector("#toggle").style.height='2%' document.querySelector("#toggle").style.width='2%' document.querySelector("#searchcontent").style.visibility = "hidden"; throw new Error(error); } } async function fetchIndexer(indexerObj){ console.log(`Fetching From ${indexerObj["name"]}`) req=await fetch(`${url}&indexerIds=${indexerObj["id"]}`) processResults(JSON.parse(req.responseText)) } function fetch(url){ return new Promise((resolve, reject) => { GM.xmlhttpRequest( { 'method' : 'GET', 'url' : url, 'responseType':"json", onload : response => { resolve(response) }, onerror : response => { reject(response.responseText) }, } ) })} ` DOM Manipulators These Functions are used to manipulate the DOM ` function getTableHead(){ resultList=document.querySelector("#resultlist") node=document.createElement("div"); node.setAttribute("id","resulthead") node.innerHTML=`
Number
Title
Indexer
Grabs
Date
Size
` resultList.replaceChild(node, resultList.firstChild) } function processResults(data){ if (data.length==0){ return } data.forEach((e,i)=>{ resultList=document.querySelector("#resultlist") node=document.createElement("div"); node.setAttribute("class","resultitem") node.innerHTML=`
?
${e['title']}
${e['indexer']}
${e['grabs']}
${new Date(e['publishDate']).toLocaleString("en-CA")}
${(parseInt(e['size'])/1073741824).toFixed(2)} GB
Download
Details
` resultList.appendChild(node) }) } function resetResultList(){ resultList=document.querySelector("#resultlist") resultList.innerHTML= `
` } function addNumbers(){ document.querySelectorAll(".resultitem").forEach((e,i)=>{ e.firstElementChild.textContent=`${i}` } ) } ` URL Processing These Functions are used to generate the URL used to retrive the data for user ` async function getBaseURL(){ return `${new URL('/api/v1/search',GM_config.get('url')).toString()}?query=${getTitle()}&${await createSearchParmas()}` } async function createSearchParmas(){ let params = new URLSearchParams(); params.append("apikey",GM_config.get('api')); return params.toString() } async function getIndexers(){ document.querySelector("#msgnode").textContent="Getting Indexers" let params = new URLSearchParams(); params.append("apikey",GM_config.get('api')); indexerURL=`${GM_config.get('url')}/api/v1/indexer?${params.toString()}` req=null try{ req=await fetch(indexerURL) } catch(error){ alert(error) throw new Error(error); } indexerCacheHelper(JSON.parse(req.responseText)) if(GM_config.get('listType')=="black"){ return blackListHelper(JSON.parse(req.responseText)) } else{ return whiteListHelper(JSON.parse(req.responseText)) } } function filterCurrSite(indexers){ if (GM_config.get('sitefilter')=="false"){ return indexers } return indexers.filter((indexerObj)=>{ sites=indexerObj["indexerUrls"] for(i in sites){ if(new URL(sites[i]).hostname==window.location.host){ return false } return true } }) } function indexerCacheHelper(allIndexers){ indexerNames=GM_config.get('indexers').split(",").map((e)=>e.trim().toLowerCase()) for (let j in indexerNames){ cached=GM_getValue(indexerNames[j],"none") if (cached!="none"){ continue } for (let i in allIndexers){ if (allIndexers[i]["name"].match(new RegExp(indexerNames[j], 'i'))){ GM_setValue(indexerNames[j],allIndexers[i]["id"]) } } } } function blackListHelper(allIndexers){ indexerID = new Set(allIndexers.map((e)=>e["id"])) indexerNames=GM_config.get('indexers').split(",").map((e)=>e.trim()) for (let j in indexerNames){ cached=GM_getValue(indexerNames[j],"none") if (cached!="none"){ indexerID.delete(cached) } } output=[] for (let i in allIndexers) { if (indexerID.has(allIndexers[i]["id"])) { output.push(allIndexers[i]) } } return filterCurrSite(output) } function whiteListHelper(allIndexers){ indexerID = [] indexerNames=GM_config.get('indexers').split(",").map((e)=>e.trim()) for (let j in indexerNames){ cached=GM_getValue(indexerNames[j],"none") if (cached!="none"){ indexerID.append(cached) } } output=[] for (let i in allIndexers) { if (indexerID.has(allIndexers[i]["id"])) { output.push(allIndexers[i]) } } return filterCurrSite(output) } ` Events These Functions Create Events to be used by script ` function leftClickProcess(e){ e.preventDefault() e.stopPropagation() if(e.button!=0){ return } document.addEventListener("mousemove", mouseDragProcess) document.addEventListener("mouseup", mouseUpProcess) } function mouseUpProcess(e){ e.preventDefault() e.stopPropagation() mouseClicksProcess() removeMouseEvents() } function mouseClicksProcess(){ if (document.querySelector("#quicksearch").getAttribute('dragged')!='true'){ toggleSearch() } } //Reset Mouse Events function removeMouseEvents(){ document.removeEventListener("mousemove", mouseDragProcess) document.removeEventListener("mouseup", mouseUpProcess) //reset for next time document.querySelector("#quicksearch").setAttribute( 'dragged',null) } function mouseDragProcess(e){ document.querySelector("#quicksearch").setAttribute('dragged','true') boxHeight=parseInt(getComputedStyle(document.querySelector("#quicksearch")).height.replaceAll( /[^0-9.]/g, '')); startMousePosition=parseInt(e.clientY) offsetMousePosition=startMousePosition+boxHeight viewport=100-((offsetMousePosition/window.innerHeight)*100) document.querySelector("#quicksearch").style.bottom=`${viewport}vh` } ` Functions For Menu Events ` function openMenu(){ content.style.pointerEvents="none" document.querySelector("#toggle").style.height='5%' document.querySelector("#toggle").style.width='5%' content.style.visibility = "hidden"; document.querySelector("#toggle").removeEventListener("mousedown", leftClickProcess) } function closeMenu(){ content.style.pointerEvents="all" document.querySelector("#toggle").style.height='2%' document.querySelector("#toggle").style.width='2%' content.style.visibility = "hidden"; document.querySelector("#toggle").addEventListener("mousedown", leftClickProcess) } ` This is the main Function of the script ` searchIcon="" infoparser={ "animebytes.tv":{ "title":"h2>a[href*=series]" }, "blutopia.xyz":{ "title":"h1>a[href*=torrents\\/similar]" }, "beyond-hd.me":{ "title":"h1[class=movie-heading]" }, "imdb.com":{ "title":"h1" }, "www.imdb.com":{ "title":"h1" }, "themoviedb.org":{ "title":"h2" }, "www.themoviedb.org":{ "title":"h2" } }[window.location.host] GM_config.init( { 'id': 'torrent-quick-search', 'title': 'Torrent Quick Search Settings', // Panel Title 'fields': { 'url': { 'label': 'URL', 'type': 'text', 'title':'Base URl for program' }, 'api': { 'label': 'API Key', 'type': 'text', 'title':'API key for program' }, 'type': { 'label': 'type', 'type': 'select', 'options': ['Prowlarr'], 'title':'Which Program' }, 'sitefilter': { 'label': 'Filter Current Site', 'type': 'radio', 'options': ['true', 'false'], 'title':'Should Results From Current Site be Filtered Out' }, 'indexers': { 'label': 'Indexers', 'section': ['Indexers'], 'type': 'text', 'title':'Comma Seperated List of Indexers Names' }, 'listType': { 'type': 'radio', 'options': ['black', 'white'], 'label': 'Indexers ListType', 'title':'What Type of List?' }, 'fontsize': { 'label': 'Font Size', 'section': ['GUI'], 'type': 'int', 'title':'fontsize', 'default':12 }, }, 'events': { 'open': openMenu, 'close':closeMenu }, } ); const box = document.createElement("div"); box.setAttribute("id", "quicksearch"); box.innerHTML=`