Greasy Fork

Greasy Fork is available in English.

Torrent Quick Search

Toggle for Searching Torrents via Search aggegrator

当前为 2022-10-16 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        Torrent Quick Search
// @namespace  https://github.com/TMD20/torrent-quick-search
// @supportURL https://github.com/TMD20/torrent-quick-search
// @version     1.40
// @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
// @grant GM.notification
// @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();
}

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"
				indexers = await getIndexers()
        imdb=await setIMDBNode()
        setTitleNode()
				//reset count
				let count = []
				let length = indexers.length
				let data = []
				x = 5
				while (indexers.length)
				{
					// x at a time
					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)
				errorMsgs = data.filter((e) => e["status"] == "rejected").map((e) => e["reason"].message)
				errorMsgs = [...new Set(errorMsgs)]
				if (errorMsgs.length > 0)
				{
					reject(errorMsgs.join("\n"))
				}
				addNumbers()
				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(async()=>{
       	resetResultList()
	resetSearchDOM()
            getTableHead()

      try{
      await this.searchPromise

      }
    		catch (error)
		{
			if (error.message.match(/aborted!/i) === null)
			{
				GM.notification(error.message, program, searchIcon)
			}

			console.log(error)

		}

    },0)


	},
	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)

	},
	async toggleSearch()
	{
		content = document.querySelector("#torrent-quicksearch-box")
		if (content.style.display === "inline-block")
		{
			hideDisplay()
			searchObj.cancel()
		}
		else if ((content.style.display === "none" || content.style.display === ""))
		{
      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) =>
	{

		msg = null
		controller.signal.addEventListener("abort", () =>
		{
			reject(AbortError)
		});
		searchprograms = 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"] = "Not Provided"
			}

		})
		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)
{
	req = await fetch(getSearchURLProwlarr(indexer["id"]),
	{
		"timeout": indexerSearchTimeout
	})
	data = JSON.parse(req.responseText)
	promiseArray = await Promise.allSettled(data.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["details"],
			"DownloadUrl": e["downloadUrl"],
			"ImdbId": e["imdbId"],
			"TvdbId": await tmdbTVDBConvertor(e["imdbId"]),
			"Cost": e["indexerFlags"].includes("freeleech") == "100% Freeleech" ? "100% Freeleech" : "Cost Unknown With Prowlarr",
			"Protocol": e["protocol"],
		}

	}))
	return promiseArray.map((e) => e["value"]).filter((e) => e != null)






}

async function searchJackettIndexer(indexer)
{
	req = await fetch(getSearchURLJackett(indexer["id"]),
	{
		"timeout": indexerSearchTimeout
	})
	data = JSON.parse(req.responseText)["Results"]
	promiseArray = await Promise.allSettled(data.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"],
			"TvdbId": await tmdbTVDBConvertor(e["Imdb"]),
			"Cost": `${(1-e["DownloadVolumeFactor"])*100}% Freeleech`,
			"Protocol": "torrent",
		}

	}))
	return promiseArray.map((e) => e["value"]).filter((e) => e != null)
}
async function searchHydra2Indexer(indexer)
{
	req = await fetch(getSearchURLHydraTor(indexer["id"]),
	{
		"timeout": indexerSearchTimeout
	})
	req2 = await fetch(getSearchURLHydraNZB(indexer["id"]),
	{
		"timeout": indexerSearchTimeout
	})
	parser = new DOMParser();
	data = [...Array.from(parser.parseFromString(req.responseText, "text/xml").querySelectorAll("channel>item")), ...Array.from(parser.parseFromString(req2.responseText, "text/xml").querySelectorAll("channel>item"))]
	//array is final dictkey,queryselector,attribute
	promiseArray = await Promise.allSettled(
		data.map(async (e) =>
		{
			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"]
			]
			out = {}
			out["Grabs"] = "Hydra Does not Report"

			for (i in t)
			{
				key = t[i][0]
				node = e.querySelector(t[i][1])
				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["TvdbId"] = await tmdbTVDBConvertor(out["ImdbId"])
			out["Protocol"] = data[0].querySelector("enclosure").getAttribute("type") == "application/x-bittorrent" ? "torrent" : "usenet"
			return out

		})

	)
	return promiseArray.map((e) => e["value"]).filter((e) => e != null)
}

function fetch(url,
{
	method = "GET",
	data = null,
	headers = {},
	timeout = 300000
} = {})
{
	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)
				},
			})
		}

	)
	return req
}

function getParser()
{
	siteName = standardNames[window.location.host] || window.location.host
	data = infoParser[siteName]
	if (data === undefined)
	{
		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(){
    		imdb = null
				document.querySelector("#torrent-quicksearch-msgnode").textContent = "Fetching Results From Indexers"
				//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 = ""
}

function hideDisplay()
{
	document.querySelector("#torrent-quicksearch-overlay").style.setProperty('--icon-size', `${iconSmall}%`);
	document.querySelector("#torrent-quicksearch-customsearch").value = ""
	content.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}%`);
	content.style.display = "inline-block";
}


function getTableHead()
{
	node = document.querySelector("#torrent-quicksearch-resultheader");
	node.innerHTML = `
       <span class="torrent-quicksearch-resultcell"  >Links</span>
       <span class="torrent-quicksearch-resultcell"  >Arr</span>

    <span class="torrent-quicksearch-resultcell">Number</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
	}
	resultList = document.querySelector("#torrent-quicksearch-resultlist")
	tempFrag = new DocumentFragment()

	data.forEach((e, i) =>
	{
		node = document.createElement("span");
		node.setAttribute("class", "torrent-quicksearch-resultitem")
		node.innerHTML = `
    <span class="torrent-quicksearch-resultcell"  style='font-size:${GM_config.get("fontsize",12)};grid-column-start:2' >
        <a href=${e['DownloadUrl']}>Download</a>
         <br>
        <br>
        <a href=${e['InfoUrl']}>Details</a>
  </span>
    <span class="torrent-quicksearch-resultcell"  style='font-size:${GM_config.get("fontsize",12)};grid-column-start:3'>
            <a>Send to Sonarr</a>
        <br>
        <br>
        <a>Send to Radarr</a>
  </span>
    <span class="torrent-quicksearch-resultcell"  style='font-size:${GM_config.get("fontsize",12)};grid-column-start:4' >?</span>
    <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:5' >${e['Title']}</span>
    <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:6'  >${e['Indexer']}</span>
    <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:7'>${e['Grabs']||"No Data"} </span>
  <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:8'>${e['Seeders']||"No Data"} </span>
  <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:9' >${e['Leechers']||"No Data"} </span>
 <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:10'>${e['Cost']} </span>
 <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:11' >${new Date(e['PublishDate']).toLocaleString("en-CA")}</span>
 <span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:12' >${(parseInt(e['Size'])/1073741824).toFixed(2)} GB</span>
<span class="torrent-quicksearch-resultcell" style='font-size:${GM_config.get("fontsize",12)};grid-column-start:13' >${e['ImdbId']}</span>`

		processSonarrNode(node, e)
		processRadarrNode(node, e)
		tempFrag.append(node)
	})

	resultList.appendChild(tempFrag)

}

function resetResultList()
{
	document.querySelector("#torrent-quicksearch-resultheader").textContent = ""
	document.querySelector("#torrent-quicksearch-resultlist").textContent = ""
}

function addNumbers()
{
	Array.from(document.querySelectorAll(".torrent-quicksearch-resultitem")).forEach((e, i) =>
		{
			node = Array.from(e.children).filter((e) => e["textContent"] == "?")[0]
			node.textContent = `${i+1}`
		}

	)
}

async function processSonarrNode(node, data)
{
	node = Array.from(node.querySelectorAll("span>span>a")).filter((e) => e["textContent"] == "Send to Sonarr")[0]
	if (GM_config.get('sonarrurl') === "null" ||
		GM_config.get('sonarrurl') === "" ||
		GM_config.get('sonarrapi') === "null" ||
		GM_config.get('sonarrapi') === "")
	{
		node.remove()
		return
	}
	nodeEvent = await sonarrFactory(data)
	node.addEventListener("click", nodeEvent)
	node.style.cursor = "pointer"

}

async function processRadarrNode(node, data)
{
	node = Array.from(node.querySelectorAll("span>span>a")).filter((e) => e["textContent"] == "Send to Radarr")[0]
	if (GM_config.get('radarrurl') === "null" ||
		GM_config.get('radarrurl') === "" ||
		GM_config.get('radarrapi') === "null" ||
		GM_config.get('radarrapi') === ""
	)
	{
		node.remove()
		return
	}
	nodeEvent = await radarrFactory(data)
	node.addEventListener("click", nodeEvent)
	node.style.cursor = "pointer"
}

function createMainDOM()
{
	const box = document.createElement("div");
	box.setAttribute("id", "torrent-quicksearch-overlay");
	rowSplit = 12
	contentWidth = 70
	boxMinHeight = 5
	boxMaxHeight = 100
	boxHeight = 40
	boxWidth = 70
	boxMaxWidth = 150
	box.innerHTML = `
 <div>
  <img id="torrent-quicksearch-toggle" src="${searchIcon}"></img>
<div id="torrent-quicksearch-box">
<div id="torrent-quicksearch-content">
  <div id="torrent-quicksearch-msgnode"></div>
  <div id="torrent-quicksearch-custombox">
        <div>
      <label>Title:</label>
    <input type="text" id="torrent-quicksearch-customsearch">
     <label>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 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;

   }



  #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;
  }

     #torrent-quicksearch-custombox {
      background-color:#FFFFFF;
      width:calc(var(--grid-size)*${rowSplit});
         pointer-events:all;


   }

    #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; /* Green */
  border: none;
  color: white;
  text-align: center;
  text-decoration: none;
  font-size: 16px;
}

  #torrent-quicksearch-content {
      pointer-events:all;

  background-color: #d5cbcb;
  scrollbar-color: white;
   direction:ltr;
  overflow:scroll;
  height:100%;
  width:100%;
}

    #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-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;
  }

  .torrent-quicksearch-resultcell{
  font-weight: bold;
        margin-left: 10%;
overflow-wrap:break-word;
  }

  ::-webkit-scrollbar-thumb{
  background-color:white;

  }

<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()
    customSearch = true
    searchObj.doSearch()

  },0)

  })
	document.body.insertBefore(box, document.body.children[0])

}

`
Matching Function
These help with finding a Match
`

function getTitle()
{
	titleNode = document.querySelector(siteParser["title"])
	if (titleNode == null)
	{
		throw new Error("Title Node Not Found")
	}
	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 = tmdbPageIMDBParser()
	}
	else
	{
		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)
{
	key = GM_config.get('tmdbapi', "null")
	if (key == "null")
	{
		return null
	}
	let baseURL = `https://api.themoviedb.org/3/movie/${id}/external_ids`
	let params = new URLSearchParams();
	params.append("api_key", key)
	if (type == "tv")
	{
		baseURL = `https://api.themoviedb.org/3/tv/${id}/external_ids`
	}
	searchURL = `${baseURL}?${params.toString()}`
	req = await fetch(searchURL)
	return JSON.parse(req.responseText)

}

async function imdbTMDBConvertor(imdb)
{
	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 = `https://api.themoviedb.org/3/find/${imdb}`
	let params = new URLSearchParams();

	params.append("api_key", key)
	params.append("external_source", "imdb_id")
	searchURL = `${baseURL}?${params.toString()}`
	req = await fetch(searchURL)
	return JSON.parse(req.responseText)
}

// First call to tmdbapi should be removed once we parse Movies vs TV


async function tmdbTVDBConvertor(imdb)
{
	key = GM_config.get('tmdbapi', "null")
	if (key == "null")
	{
		return null
	}
	helperData = await imdbTMDBConvertor(imdb)
	let data = null
	if (helperData["tv_results"].length > 0)
	{
		return (await tmdbExternalMedia("tv", helperData["tv_results"][0]["id"]))["tvdb_id"]
	}
	else if (helperData["movie_results"].length > 0)
	{
		return (await tmdbExternalMedia("movie", helperData["movie_results"][0]["id"]))["tvdb_id"]
	}

}

async function tmdbPageIMDBParser()
{
	id = window.location.href.match(/\/[0-9]+/).toString().substring(1)
	if (window.location.href.match(/\/tv\//))
	{
		return await tmdbExternalMedia("tv", 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"
	searchprogram = GM_config.get('searchprogram')
	indexers = null
	if (searchprogram == "Prowlarr")
	{
		indexers = await getIndexersProwlarr()
	}
	else if (searchprogram == "Jackett")
	{
		indexers = await getIndexersJackett()
	}

	else if (searchprogram == "NZBHydra2")
	{
		indexers = await getIndexersHydra()
	}

	indexerCacheHelper(indexers)
	return listFilter(indexers)

}

async function getIndexersJackett()
{
	cachedIndexers = GM_getValue("jackettindexers", "none")
	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", "''")
	baseURL = `${GM_config.get('searchurl')}/api/v2.0/indexers/all/results`
	indexerURL = `${baseURL}?${params.toString()}`
	req = await fetch(indexerURL)
	data = JSON.parse(req.responseText)['Indexers']
	output = []
	//Same keys as Prowlarr
	for (i in data)
	{
		output.push(
		{
			"name": data[i]["Name"],
			"id": data[i]["ID"]
		})

	}
	GM_setValue("jacketIndexers",
	{
		"date": Date.now(),
		"indexers": data
	})
	return output

}

async function getIndexersProwlarr()
{
	cachedIndexers = GM_getValue("prowlarrindexers", "none")
	if (Date.now() - (cachedIndexers?.date || 0) < day)
	{
		return cachedIndexers["indexers"]
	}
	let params = new URLSearchParams();
	params.append("apikey", GM_config.get('searchapi'));

	baseURL = `${GM_config.get('searchurl')}/api/v1/indexer`
	indexerURL = `${baseURL}?${params.toString()}`
	req = await fetch(indexerURL)
	data = JSON.parse(req.responseText)
	data = data.sort(prowlarIndexSortHelper)
	GM_setValue("prowlarrindexers",
	{
		"date": Date.now(),
		"indexers": data
	})
	return data

}

function prowlarIndexSortHelper(a, b)
{
	if (a["priority"] > b["priority"])
	{
		return -1;
	}
	if (a["priority"] < b["priority"])
	{
		return 1;
	}
	return 0;
}

async function getIndexersHydra()
{
	cachedIndexers = GM_getValue("hydraindexers", "none")
	if (Date.now() - (cachedIndexers?.date || 0) < day)
	{
		return cachedIndexers["indexers"]
	}
	let params = new URLSearchParams();
	params.append("apikey", GM_config.get('searchapi'));
	baseURL = `${GM_config.get('searchurl')}/api/stats/indexers/`
	indexerURL = `${baseURL}?${params.toString()}`
	req = await fetch(indexerURL)
	data = JSON.parse(req.responseText)
	output = []
	//Same keys as Prowlarr
	for (i in data)
	{
		output.push(
		{
			"name": data[i]["indexer"],
			"id": data[i]["indexer"]
		})

	}
	GM_setValue("hydraindexers",
	{
		"date": Date.now(),
		"indexers": output
	})
	return output

}

function listFilter(allIndexers)
{
	let selectedIndexers = null
	if (GM_config.get('listType') == "black")
	{
		selectedIndexers = blackListHelper(allIndexers)
	}
	else
	{
		selectedIndexers = whiteListHelper(allIndexers)
	}

	output = []

	for (let i in allIndexers)
	{
		if (selectedIndexers.has(allIndexers[i]["id"]))
		{
			output.push(allIndexers[i])
		}

	}

	return output

}

function indexerCacheHelper(allIndexers)
{
	if (GM_config.get('indexers') == '')
	{
		return
	}
	searchprogram = GM_config.get('searchprogram')
	indexerNames = GM_config.get('indexers').split(",").map((e) => e.trim().toLowerCase())
	for (let j in indexerNames)
	{
		key = `${searchprogram}_${indexerNames[j]}`
		cached = GM_getValue(key, "none")
		if (cached != "none")
		{
			continue
		}
		for (let i in allIndexers)
		{

			if (allIndexers[i]["name"].match(new RegExp(indexerNames[j], 'i')))
			{
				GM_setValue(key, 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)
	{
		key = `${searchprogram}_${indexerNames[j]}`
		cached = GM_getValue(key, "none")
		if (cached != "none")
		{
			indexerID.delete(cached)

		}

	}
	return indexerID

}

function whiteListHelper(allIndexers)
{
	indexerID = new Set()
	indexerNames = GM_config.get('indexers').split(",").map((e) => e.trim())

	for (let j in indexerNames)
	{
		key = `${searchprogram}_${indexerNames[j]}`
		cached = 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)
	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}`)
	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")
	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")
	baseURL = new URL('/torznab/api', GM_config.get('searchurl')).toString()

	return `${baseURL}?${params.toString()}`
}
`
ARR Functions
Functions to support Arr Clients


`

function getSonarrURL()
{
	let params = new URLSearchParams();
	params.append("apikey", GM_config.get('sonarrapi'));
	baseURL = new URL('/api/v3/release/push', GM_config.get('sonarrurl')).toString()
	return `${baseURL}?${params.toString()}`


}

function getRadarrURL()
{
	let params = new URLSearchParams();
	params.append("apikey", GM_config.get('radarrapi'));
	baseURL = new URL('/api/v3/release/push', GM_config.get('radarrurl')).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)

	dragState = true
	toggleHeight = parseInt(getComputedStyle(document.querySelector("#torrent-quicksearch-toggle")).height.replaceAll(/[^0-9.]/g, ''))
	startMousePosition = parseInt(e.clientY)
	offsetMousePosition = startMousePosition - (toggleHeight / 2)
	viewport = (offsetMousePosition / window.innerHeight) * 100
	viewport = Math.max(viewport, -20)
	viewport = Math.min(viewport, 89)
	document.querySelector("#torrent-quicksearch-overlay").style.top = `${viewport}vh`

}
async function sonarrFactory(releaseData)
{
	sonarrEvent = async function (e)
	{
		e.preventDefault()
		e.stopPropagation()
		res = await fetch(getSonarrURL(),
		{
			"header": "post",
			"data": JSON.stringify(releaseData)
		})
		if (res.status != 200)
		{
			GM.notification(res.responseText, program, searchIcon)
			return
		}
		data = JSON.parse(res.responseText)
		initValue = ""
		finalMsg = data.reduce(
			(prev, curr, index) =>
			{
				if (curr["rejected"] == true)
				{
					epNums = curr["mappedEpisodeNumbers"].length > 0 ? curr["mappedEpisodeNumbers"].join(",") : "No Episodes"
					errorMsg = [`${curr["seriesTitle"]} Season ${curr["seasonNumber"]} Episodes ${epNums}`, `Status Rejected: ${curr["rejections"].join(",")}`]
					return `${prev}\n\[${errorMsg.join("\n")}\]`
				}
				if (curr["approved"] == true)
				{
					acceptMsg = `Added ${curr["title"]} to client`
					return `${prev}\n${acceptMsg}`
				}

			}, initValue)
		GM.notification(finalMsg, program, searchIcon)

	}
	return sonarrEvent

}

async function radarrFactory(releaseData)
{
	radarrEvent = async function (e)
	{
		e.preventDefault()
		e.stopPropagation()

		res = await fetch(getRadarrURL(),
		{
			"header": "post",
			"data": JSON.stringify(releaseData)
		})

		if (res.status != 200)
		{
			GM.notification(res.responseText, program, searchIcon)
			return
		}
		data = JSON.parse(res.responseText)
		initValue = ""
		finalMsg = data.reduce(
			(prev, curr, index) =>
			{
				if (curr["rejected"] == true)
				{
					errorMsg = [`${curr["movieTitles"].join(",")}`, `Status Rejected: ${curr["rejections"].join(",")}`]
					return `${prev}\n\[${errorMsg.join("\n")}\]`
				}
				if (curr["approved"] == true)
				{
					acceptMsg = `Added ${curr["title"]} to client`
					return `${prev}\n${acceptMsg}`
				}

			}, initValue)
		GM.notification(finalMsg, program, searchIcon)

	}
	return radarrEvent

}






`
Functions For Menu Events

`

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-toggle").addEventListener("mousedown", leftClickProcess)
	document.querySelector("#torrent-quicksearch-toggle").addEventListener("mouseup", mouseUpProcess)

}

`
Globals + Main Function
`
searchIcon = ""
iconLarge = 50
iconSmall = 25
paddingSmall = 0
paddingLarge = 4
let mouseState = "up"
imdbParserFail = "Could Not Parse From Page"
AbortError = new DOMException(
	'aborted!',
	'AbortError')
var controller = new AbortController()
program = "Torrent Quick Search"
clickLimit = 500
lastClick = Date.now() - clickLimit
indexerSearchTimeout = 60000
day = 86400000
customSearch = false






//keep track of whether program was recently dragged

//Normalize Site Name so we don't have repeat keys in larger infoparser dict
standardNames = {
	"www.imdb.com": "imdb.com",
	"www.themoviedb.org": "themoviedb.org"
}

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",
	},
}

siteParser = getParser()
//Main Function
function main()
{
	createMainDOM()
	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'
			},

			'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'
			},

			'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': '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'

			},
			'radarrurl':
			{
				'label': 'Radarr URL',
				'section': ['Arr Clients'],
				'type': 'text',
				'title': 'URL for Radarr',
			},
			'radarrapi':
			{
				'label': 'Radarr ApiKey ',
				'type': 'text',
				'title': 'APIKey for Radarr',
			},
			'sonarrurl':
			{
				'label': 'Sonarr URL',
				'type': 'text',
				'title': 'URL for Sonarr',
			},
			'sonarrapi':
			{
				'label': 'Sonarr ApiKey ',
				'type': 'text',
				'title': 'APIKey for Sonarr',
			},
			'fontsize':
			{
				'label': 'Font Size',
				'section': ['GUI'],
				'type': 'int',
				'title': 'fontsize',
				'default': 12
			},

		},
		'events':
		{
			'open': openMenu,
			'close': closeMenu
		},
	});

	GM.registerMenuCommand('Torrent Quick Search Settings', function ()
	{
		GM_config.open();
	});

}

main()