Greasy Fork

Greasy Fork is available in English.

Torrent Quick Search

Toggle for Searching Torrents via Search aggegrator

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

您需要先安装一款用户脚本管理器扩展,例如 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.60
// @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
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.xmlHttpRequest
// @grant GM.registerMenuCommand
// @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/gh/sizzlemctwizzle/GM_config@43fd0fe4de1166f343883511e53546e87840aeaf/gm_config.js
// @require  https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.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() {
  if (GM.info.script.name == "Torrent Quick Search") {
    overideBuiltins();
    initConfig();
    createMainDOM();
  }
  

}

main();