// ==UserScript==
// @name EasyUpload PT一键转种
// @name:en EasyUpload - Trackers Transfer Tool
// @namespace https://github.com/techmovie/easy-upload
// @version 7.0.0-beta.11
// @author birdplane
// @description 一键转种,支持PT站点之间的种子转移。
// @description:en Transfer torrents between trackers with one click.
// @license MIT
// @source git@github.com:techmovie/easy-upload.git
// @match http*://*/torrents.php?id=*
// @match http*://*/torrents.php?torrentid=*
// @match http*://*/details.php?id=*
// @match https://totheglory.im/t/*
// @match https://ptpimg.me/*
// @match http*://*/torrents/*
// @match http*://*/torrents?*
// @match http*://*/upload*
// @match https://*/offers.php*
// @match https://broadcity.in/browse.php?imdb=*
// @match https://*/torrent/*
// @match https://piratethenet.org/browse.php?*
// @match https://teamhd.org/details/id*
// @match https://hd-space.org/index.php?page=upload
// @match https://hd-space.org/index.php?page=torrent-details&id=*
// @match https://speedapp.io/browse/*
// @match https://*.m-team.cc/detail/*
// @exclude https://*/torrent/peers*
// @exclude https://*/torrent/leechers*
// @exclude https://*/torrent/history*
// @require https://cdn.jsdelivr.net/npm/preact@10.24.3/dist/preact.min.js
// @require https://cdn.jsdelivr.net/npm/jquery@3.7.1/dist/jquery.min.js
// @grant GM_addStyle
// @grant GM_deleteValue
// @grant GM_getValue
// @grant GM_openInTab
// @grant GM_setClipboard
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @downloadURL none
// ==/UserScript==
(e=>{if(typeof GM_addStyle=="function"){GM_addStyle(e);return}const t=document.createElement("style");t.textContent=e,document.head.append(t)})(" td.title-td{min-width:80px;vertical-align:middle!important}td.title-td h4{text-align:right;margin:0;font-size:13px;font-weight:500;height:100%;display:flex;align-items:center;justify-content:flex-end}#seed-dom button{line-height:1;white-space:nowrap;cursor:pointer;background:#fff;border:1px solid #dcdfe6;color:#606266;-webkit-appearance:none;text-align:center;box-sizing:border-box;outline:none;transition:.1s;font-weight:500;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;padding:6px 16px;font-size:13px;border-radius:4px;margin:0 5px 0 0}#seed-dom button:hover{background:#fff;border-color:#409eff;color:#409eff}#seed-dom button.is-disabled,#seed-dom button.is-disabled:hover{color:#c0c4cc;cursor:not-allowed;background-image:none;background-color:#fff;border-color:#ebeef5}.site-list,.search-list{margin:0;padding:0;list-style:none;display:flex;justify-content:center;align-items:center;flex-wrap:wrap}.site-list .site-icon,.search-list .site-icon{width:12px;margin-right:5px}.ptp-search-list{display:flex;align-items:center;padding-top:10px;justify-content:center}.ptp-search-list h4{margin:0 15px 0 0;min-width:60px}#seed-dom ul li,.search-list li,.site-list li{font-weight:600;line-height:24px;margin:0 5px 0 0;padding:0;display:flex;align-items:center}#seed-dom ul li a,.search-list li a,.site-list li a{padding-right:3px;display:inline-flex;align-items:center;cursor:pointer}.search-list li:last-child span{display:none}.easy-upload-function-list{display:flex;justify-content:space-around;padding:6px 20px;flex-wrap:wrap}.easy-upload-function-list button{line-height:1;white-space:nowrap;cursor:pointer;background:#fff;border:1px solid #dcdfe6;color:#606266;-webkit-appearance:none;text-align:center;box-sizing:border-box;outline:none;transition:.1s;font-weight:500;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;padding:8px 20px;font-size:14px;border-radius:4px;margin:0 5px 0 0}.easy-upload-function-list button:hover{background:#fff;border-color:#409eff;color:#409eff}.easy-upload-function-list button.is-disabled,.easy-upload-function-list button.is-disabled:hover{color:#c0c4cc;cursor:not-allowed;background-image:none;background-color:#fff;border-color:#ebeef5}.function-list-item{display:flex;align-items:center;justify-content:space-between;margin-bottom:10px}.function-list-item input{-webkit-appearance:none;background-color:#fff;background-image:none;border-radius:4px;border:1px solid #dcdfe6;box-sizing:border-box;color:#606266;display:inline-block;font-size:inherit;height:34px;line-height:40px;outline:none;width:200px;padding:0 12px;transition:border-color .2s cubic-bezier(.645,.045,.355,1)}.function-list-item select{border:0;font-family:inherit;padding:5px;font-size:14px;border-radius:3px;text-transform:none}.function-list-item input::placeholder{color:#c0c4cc}.function-list-item input:hover{border-color:#c0c4cc}.function-list-item input:focus{outline:none;border-color:#409eff}.hdb-tr{display:flex}.hdb-tr td:last-child{flex:1}.hdb-tr td:first-child>h4{width:100px}.function-list-item h4{margin:0 10px 0 0;padding:0;font-weight:600;font-size:14px}.upload-section,.douban-section,.douban-book-section{display:flex;justify-content:center;align-items:center}.upload-section #nsfw{margin-left:0;position:static}.upload-section label{padding-left:0}#kdescr img{max-width:100%}.easy-upload-setting-btn{display:inline-flex;align-items:center;margin-left:3px}svg.setting-svg{height:20px;width:20px;vertical-align:middle;animation:5s linear rotate infinite;cursor:pointer}@keyframes rotate{to{transform:rotate(360deg)}}.easy-upload-setting-panel{position:fixed;top:0;right:0;bottom:0;left:0;z-index:2000;background:rgba(0,0,0,.5);color:#000}#batch-seed-btn,#auto-fill-douban{border-color:transparent;color:#409eff;background:transparent;padding-left:0;padding-right:0;font-weight:600;cursor:pointer}#batch-seed-btn:hover,#auto-fill-douban:hover{color:#66b1ff;border-color:transparent;background-color:transparent}#batch-seed-btn:active,#auto-fill-douban:active{color:#3a8ee6;background-color:transparent}#auto-fill-douban{font-size:14px;display:inline-block}.easy-upload-setting-panel *{padding:0;margin:0}.easy-upload-setting-panel input[type=text]{-webkit-appearance:none;background-color:#fff;background-image:none;border-radius:4px;border:1px solid #dcdfe6;box-sizing:border-box;color:#606266;display:inline-block;font-size:inherit;height:34px;line-height:40px;outline:none;width:200px;padding:0 12px;transition:border-color .2s cubic-bezier(.645,.045,.355,1)}.easy-upload-setting-panel input[type=text]::placeholder{color:#c0c4cc}.easy-upload-setting-panel input[type=text]:hover{border-color:#c0c4cc}.easy-upload-setting-panel input[type=text]:focus{outline:none;border-color:#409eff}.easy-upload-setting-panel h3,.easy-upload-setting-panel h1{color:#000;margin-bottom:15px}.easy-upload-setting-panel .panel-content-wrap{max-width:800px;box-sizing:border-box;margin:50px auto;border-radius:8px;background:#fff;position:relative;text-align:center;box-shadow:0 1px 3px rgba(0,0,0,.3);padding:20px 10px 10px 20px}.easy-upload-setting-panel .panel-content{height:500px;overflow-y:auto}.easy-upload-setting-panel .panel-content ul{list-style:none;display:flex;flex-direction:row;flex-wrap:wrap;margin:0 auto;padding:0 10px}.easy-upload-setting-panel .panel-content li{width:90px;text-align:left;margin-bottom:10px}.easy-upload-setting-panel .panel-content label{cursor:pointer;color:#000!important;font-size:12px;display:flex;align-items:center}.easy-upload-setting-panel .panel-content label input{margin:0 3px 0 0;padding:0}.panel-content p{display:block;margin-bottom:10px;font-size:12px}.easy-upload-setting-panel button{line-height:1;white-space:nowrap;cursor:pointer;background:#fff;border:1px solid #dcdfe6;color:#606266;-webkit-appearance:none;text-align:center;box-sizing:border-box;outline:none;transition:.1s;font-weight:500;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;padding:8px 20px;font-size:14px;border-radius:4px;margin:0 5px 10px 0}.easy-upload-setting-panel button:hover{background:#fff;border-color:#409eff;color:#409eff}.easy-upload-setting-panel .confirm-btns{padding-top:15px}.easy-upload-setting-panel .img-upload-setting{margin-bottom:10px}.easy-upload-setting-panel .img-upload-setting label{justify-content:center}.easy-upload-setting-panel .img-upload-setting label input{margin-left:8px;margin-right:8px}.easy-upload-setting-panel .img-upload-setting label a{color:#000;font-weight:500}.easy-upload-setting-panel .img-upload-setting label a:hover{color:#f7d584}.feature-list{display:flex;flex-wrap:wrap;justify-content:space-between;padding:0 50px}.feature-list .site-enable-setting{width:250px;padding-top:5px;margin-bottom:8px;text-align:center}.easy-upload-setting-panel .save-setting-btn{background-color:#007bff;border-color:#007bff;color:#fff}.easy-upload-setting-panel .save-setting-btn:hover{background:#66b1ff;border-color:#66b1ff;color:#fff}.ptp-api-key-btn{text-align:center}.easy-notification{box-sizing:border-box;position:fixed;transition:opacity .3s,transform .3s,left .3s,right .3s,top .4s,bottom .3s;overflow:hidden;right:0;margin:0 24px 0 0;color:rgba(0,0,0,.85);font-size:14px;line-height:1.5715;z-index:2010}.easy-notification-enter{right:16px;transform:translate(0)}.easy-notification-notice{position:relative;width:300px;max-width:calc(100vw - 48px);margin-bottom:16px;margin-left:auto;padding:16px 24px;overflow:hidden;line-height:1.5715;word-wrap:break-word;background:#fff;border-radius:2px;box-sizing:border-box;box-shadow:0 2px 12px rgba(0,0,0,.1)}.notification-message{margin-bottom:8px;color:rgba(0,0,0,.85);font-size:16px;line-height:24px}.notification-description{font-size:14px;line-height:21px;margin:6px 0 0;text-align:justify;padding-right:10px}.notification-description p{margin:0}.easy-notification-notice-close svg{height:14px;width:14px;font-size:14px}.easy-notification-notice-close{position:absolute;top:13px;right:15px;cursor:pointer;color:#909399;font-size:16px}.easy-notification-notice-close:hover{color:#606266}#transfer-progress{display:none}.custom-site{display:flex;align-items:center;width:100%}.custom-site h4{flex-shrink:0;margin:0 10px 0 0;line-height:initial}.custom-site .easy-upload-function-list{flex:1}.custom-site img{border-radius:0}tr.pad[id*=torrent_]{font-family:Proxima Nova,Lato,Segoe UI,sans-serif}#seed-dom .easy-upload-function-list .copy-img{margin-left:5px}.quick-search{cursor:pointer;color:#409eff;font-weight:600}.ptp-title-wrapper{position:relative}#seed-dom .ptp-title-wrapper h4{position:absolute;left:0;top:0;margin:0!important;display:flex!important;align-items:center;line-height:24px}#seed-dom .ptp-title-wrapper .site-list li:first-child{padding:0 0 0 140px}#seed-dom .ptp-title-wrapper .search-list li:first-child{padding-left:70px}#seed-dom.use-eng .ptp-title-wrapper .search-list li:first-child{padding-left:95px}#batch-search-btn{color:#409eff;padding-left:0;padding-right:0;font-weight:600;cursor:pointer}.douban-config{display:flex;justify-content:center}.douban-config textarea{resize:none;width:300px;height:100px;margin-left:6px} ");
(function (preact, $$2) {
'use strict';
var __defProp = Object.defineProperty;
var __defProps = Object.defineProperties;
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __propIsEnum = Object.prototype.propertyIsEnumerable;
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
var __spreadValues = (a2, b) => {
for (var prop in b || (b = {}))
if (__hasOwnProp.call(b, prop))
__defNormalProp(a2, prop, b[prop]);
if (__getOwnPropSymbols)
for (var prop of __getOwnPropSymbols(b)) {
if (__propIsEnum.call(b, prop))
__defNormalProp(a2, prop, b[prop]);
}
return a2;
};
var __spreadProps = (a2, b) => __defProps(a2, __getOwnPropDescs(b));
var __objRest = (source, exclude) => {
var target = {};
for (var prop in source)
if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
target[prop] = source[prop];
if (source != null && __getOwnPropSymbols)
for (var prop of __getOwnPropSymbols(source)) {
if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
target[prop] = source[prop];
}
return target;
};
var _a;
var f$1 = 0;
function u$1(e2, t2, n, o2, i, u2) {
t2 || (t2 = {});
var a2, c2, l2 = t2;
"ref" in t2 && (a2 = t2.ref, delete t2.ref);
var p2 = { type: e2, props: l2, key: n, ref: a2, __k: null, __: null, __b: 0, __e: null, __d: void 0, __c: null, constructor: void 0, __v: --f$1, __i: -1, __u: 0, __source: i, __self: u2 };
if ("function" == typeof e2 && (a2 = e2.defaultProps)) for (c2 in a2) void 0 === l2[c2] && (l2[c2] = a2[c2]);
return preact.options.vnode && preact.options.vnode(p2), p2;
}
const PT_SITE = {
"1PTBA": {
url: "https://1ptba.com",
host: "1ptba.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: "#torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "1",
hevc: "10",
x265: "10",
h265: "10",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "18",
dd: "18",
"dd+": "18",
flac: "1",
dts: "3",
truehd: "20",
lpcm: "21",
dtshdma: "19",
atmos: "19",
dtsx: "3",
ape: "2",
wav: "22",
mp3: "4",
m4a: "5",
other: "7"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "10",
bluray: "13",
remux: "3",
encode: "7",
web: "11",
hdtv: "5",
dvd: "2",
dvdrip: "7",
other: "12",
cd: "8"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "11",
"2160p": "10",
"1080p": "2",
"1080i": "2",
"720p": "4",
"576p": "5",
"480p": "5"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
"1ptba": "1",
chd: "2",
mysilu: "3",
wiki: "4",
other: "5"
}
}
},
"3Wmg": {
url: "https://www.3wmg.com",
host: "3wmg.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="pt_gen[douban][link]"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: 'input[name="tags[]"][value="16"]',
diy: 'input[name="tags[]"][value="8"]',
hdr: 'input[name="tags[]"][value="64"]',
chinese_subtitle: 'input[name="tags[]"][value="32"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "403",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "410",
cartoon: "405",
variety: "403"
}
},
source: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "1",
bluray: "1",
hdtv: "4",
dvd: "3",
web: "6",
vhs: "6",
hddvd: "2"
}
}
},
"52pt": {
url: "https://52pt.site",
host: "52pt.site",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "13",
x264: "11",
hevc: "1",
x265: "12",
h265: "1",
mpeg2: "4",
mpeg4: "13",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "13",
ac3: "6",
dd: "6",
"dd+": "6",
flac: "1",
dts: "15",
truehd: "12",
lpcm: "14",
dtshdma: "4",
atmos: "10",
dtsx: "3",
ape: "2",
wav: "11",
other: "7"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "11",
remux: "4",
encode: "7",
web: "10",
hdtv: "3",
dvd: "6",
dvdrip: "7",
other: "9",
cd: "8"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "7",
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "6",
"480p": "6"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
beyondhd: "1",
hdsky: "2",
ttg: "3",
mteam: "8",
coaster: "4",
chdbits: "9",
ourbits: "10",
hdhome: "11",
cmct: "12",
hdchina: "14",
pthome: "15",
other: "5"
}
}
},
ACM: {
url: "https://eiga.moi",
host: "eiga.moi",
siteType: "UNIT3D-Legacy",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload/1",
needDoubanInfo: true,
seedDomSelector: '#meta-info+.meta-general>.panel:has(".table-responsive"):first table tr:last',
torrentDownloadLinkSelector: 'a[href*="/torrents/download/"]',
search: {
path: "/torrents",
params: {
name: "{name}",
imdb: "{imdb}"
}
},
name: {
selector: "#title"
},
description: {
selector: "#upload-form-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.radio-inline:first input[name="anonymous"]'
},
torrent: {
selector: 'input[name="file"]'
},
videoType: {
selector: "#autocat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
category: {
selector: "#autotype",
map: {
BD100: "1",
BD66: "2",
UHD50: "3",
BD50: "4",
BD25: "5",
remux: [
"12",
"7"
],
encode: [
"8",
"10",
"11",
"13"
],
web: "9",
hdtv: "17",
dvd: [
"14",
"16"
],
dvdrip: "13",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"2160p": [
"1",
"1",
"2",
"3",
"12",
"8"
],
"1080p": [
"2",
"4",
"5",
"7",
"10"
],
"1080i": [
"2",
"4",
"5",
"7",
"10"
],
"720p": [
"3",
"11"
],
"576p": [
"4",
"13"
],
"480p": [
"5",
"14",
"16",
"13"
],
other: ""
}
}
},
AGSV: {
url: "https://www.agsvpt.com",
host: "agsvpt.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#outer > table",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
needDoubanInfo: true,
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_url"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
mediaInfo: {
selector: 'textarea[name="technical_info"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "408",
sport: "407",
cartoon: "405",
variety: "403"
}
},
source: {
selector: 'select[name="medium_sel[4]"]',
map: {
uhdbluray: "11",
bluray: "1",
remux: "3",
hdtv: "4",
dvd: "2",
encode: "7",
web: "10"
}
},
videoCodec: {
selector: 'select[name="codec_sel[4]"]',
map: {
h264: "3",
hevc: "6",
x264: "3",
x265: "6",
h265: "6",
mpeg2: "4",
mpeg4: "5",
vc1: "5",
xvid: "5"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel[4]"]',
map: {
aac: "6",
ac3: "11",
dd: "11",
"dd+": "19",
flac: "1",
dts: "3",
truehd: "9",
lpcm: "10",
dtshdma: "8",
atmos: "17",
dtsx: "18"
}
},
resolution: {
selector: 'select[name="standard_sel[4]"]',
map: {
"4320p": "5",
"2160p": "5",
"1080p": "1",
"1080i": "1",
"720p": "3",
"576p": "8",
"480p": "4"
}
}
},
Aither: {
url: "https://aither.cc",
host: "aither.cc",
siteType: "UNIT3D",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/torrents/create?category_id=1",
needDoubanInfo: true,
seedDomSelector: ".torrent__buttons+.panelV2",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: 'input[name="name"][class="form__text"]'
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anonymous"]'
},
torrent: {
selector: 'input[type="file"][accept=".torrent"]'
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2",
documentary: "1",
concert: "1",
sport: "9",
cartoon: "405",
app: "10",
ebook: "11",
magazine: "11",
audioBook: "14"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "2",
encode: "3",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "3",
other: "7"
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "1",
"2160p": "2",
"1080p": "3",
"1080i": "4",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
Audiences: {
url: "https://audiences.me",
host: "audiences.me",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_id"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: "#tag_gy",
diy: "#tag_diy",
cantonese_audio: "#tag_yy",
chinese_subtitle: "#tag_zz",
hdr: "#tag_hdr10",
hdr10_plus: "#tag_hdrm",
dolby_vision: "#tag_db"
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "401",
concert: "408",
sport: "407",
cartoon: "401",
variety: "403",
app: "411",
ebook: "405",
magazine: "412",
audioBook: "404"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "1",
hevc: "6",
x265: "6",
h265: "6",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "5",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "18",
dd: "18",
"dd+": "18",
flac: "1",
dts: "3",
truehd: "20",
lpcm: "21",
dtshdma: "19",
atmos: "26",
dtsx: "25",
ape: "2",
wav: "22",
mp3: "23",
m4a: "24"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "12",
bluray: "1",
remux: "3",
encode: "15",
web: "10",
hdtv: "5",
dvd: "2",
dvdrip: "15",
other: "11"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "10",
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
other: "5"
}
}
},
AvistaZ: {
url: "https://avistaz.to",
host: "avistaz.to",
siteType: "AvistaZ",
icon: "",
asSource: true,
asTarget: false,
uploadPath: "/upload.php",
seedDomSelector: "#content-area .block:last table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="/download/torrent/"]',
needDoubanInfo: true,
search: {
path: "/torrents",
params: {
search: "{imdb}",
"in": "1",
order: "size",
sort: "desc"
}
}
},
BLUEBIRD: {
url: "https://bluebird-hd.org",
host: "bluebird-hd.org",
siteType: "bluebird",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/browse.php",
params: {
search: "{name}",
incldead: 0,
cat: 0,
dsearch: "{imdb}",
stype: "or"
}
}
},
BTN: {
url: "https://broadcasthe.net",
host: "broadcasthe.net",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: false,
needDoubanInfo: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
action: "advanced",
imdb: "{imdb}",
artistname: "{name}"
}
}
},
BTSCHOOL: {
url: "https://pt.btschool.club",
host: "btschool.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
poster: 'input[name="picture"]',
imdb: {
selector: 'input[name="imdbid"]'
},
douban: {
selector: 'input[name="doubanid"]'
},
tags: {
chinese_audio: 'input[type="checkbox"][name="span[]"][value="5"]',
chinese_subtitle: 'input[type="checkbox"][name="span[]"][value="6"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "405",
tv: "406",
tvPack: "406",
documentary: "408",
concert: "409",
sport: "410",
cartoon: "407",
variety: "412",
music: "411"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "1",
hevc: "10",
x265: "10",
h265: "10",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "10",
dd: "10",
"dd+": "10",
flac: "1",
dts: "3",
truehd: "11",
lpcm: "5",
dtshdma: "3",
atmos: "3",
dtsx: "3"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "12",
bluray: "1",
remux: "3",
encode: "7",
web: "10",
hdtv: "5",
dvd: "6",
dvdrip: "6",
other: "11",
cd: "8"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "1",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
btschool: "1",
zone: "13",
btshd: "2",
btstv: "3",
btspad: "4",
wiki: "5",
hdchina: "6",
hdbint: "7",
mteam: "9",
cmct: "10",
ourbits: "11",
other: "12"
}
}
},
BYR: {
url: "https://byr.pt",
host: "byr.pt",
siteType: "NexusPHP",
icon: "",
asSource: false,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
subtitle: {
selector: 'input[name="small_descr"]'
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="dburl"]'
},
torrent: {
selector: "#torrent"
}
},
BeyondHD: {
url: "https://beyond-hd.me",
host: "beyond-hd.me",
siteType: "F3NIX",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: ".table-details tr:last",
torrentDownloadLinkSelector: 'a[href*="/download/"].bhd-fl-button',
needDoubanInfo: true,
uploadPath: "/upload",
search: {
path: "/torrents/all",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
imdb: "{imdb}",
search: "{name}",
sorting: "size",
direction: "desc",
doSearch: "Search"
}
},
sourceInfo: {
editionTags: {
"10-bit": "10_bit",
"2-Disc Set": "2_disc_set",
"2D/3D Edition": "2d_3d_edition",
"2in1": "2_in_1",
"3D": "3d",
"3D Anaglyph": "3d_anaglyph",
"3D Full SBS": "3d_full_sbs",
"3D Half OU": "3d_half_ou",
"3D Half SBS": "3d_half_sbs",
"4K Remaster": "4k_remaster",
"4K Restoration": "4k_restoration",
"Digital Extras": "extras",
"Director's Cut": "director_s_cut",
"Dolby Atmos": "dolby_atmos",
"Dolby Vision": "dolby_vision",
"Dual Audio": "dual_audio",
"English Dub": "english_dub",
"Extended Cut": "extended_edition",
"Extended Edition": "extended_edition",
Extras: "extras",
HDR10: "hdr",
"HDR10+": "hdr10_plus",
"Masters of Cinema": "masters_of_cinema",
Scene: "scene",
"The Criterion Collection": "the_criterion_collection",
"Theatrical Cut": "theatrical_cut",
"Two-Disc Set": "two_disc_set",
Remux: "remux",
Rifftrax: "rifftrax",
Uncut: "uncut",
Unrated: "unrated",
"Warner Archive Collection": "warner_archive_collection",
Commentary: "with_commentary"
}
},
targetInfo: {
editionTags: {
"2d_3d_edition": "2D3D",
"2_in_1": "2in1",
"3d": "3D",
"3d_anaglyph": "3D",
"3d_full_sbs": "3D",
"3d_half_ou": "3D",
"3d_half_sbs": "3D",
"4k_remaster": "4kRemaster",
director_s_cut: "Director",
dual_audio: "DualAudio",
english_dub: "EnglishDub",
extended_edition: "Extended",
extras: "Extras",
hybrid: "Hybrid",
scene: "Scene",
theatrical_cut: "Theatrical",
uncut: "Uncut",
unrated: "Unrated",
webdl: "WEBDL",
webrip: "WEBRip",
with_commentary: "Commentary",
dolby_vision: "DV",
hdr10: "HDR10",
"hdr10_plus?": "HDR10+",
hlg: "HLG"
}
},
name: {
selector: "#title"
},
description: {
selector: "#upload-form-description"
},
imdb: {
selector: "#imdbauto"
},
tmdb: {
selector: "#tmdbauto"
},
mediaInfo: {
selector: "#mediainfo"
},
anonymous: {
selector: 'input[name="anonymous"]'
},
videoType: {
selector: "#category_id",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
torrent: {
selector: 'input[type="file"][accept=".torrent"]'
},
category: {
selector: "#autotype",
map: {
BD100: "UHD 100",
BD66: "UHD 66",
UHD50: "UHD 50",
BD50: "BD 50",
BD25: "BD 25",
DVD5: "DVD 5",
DVD9: "DVD 9",
remux: [
"UHD Remux",
"BD Remux",
"DVD Remux"
],
encode: [
"2160p",
"1080p",
"720p",
"576p",
"540p",
"480p"
],
web: [
"2160p",
"1080p",
"720p",
"576p",
"540p",
"480p"
],
hdtv: [
"2160p",
"1080p",
"1080i",
"720p"
],
dvd: [
"DVD 9",
"DVD 5",
"DVD Remux"
],
dvdrip: [
"480p"
],
other: ""
}
},
source: {
selector: "#autosource",
map: {
uhdbluray: [
"Blu-ray"
],
bluray: [
"Blu-ray",
"BD 50",
"BD 25",
"BD Remux",
"UHD 100",
"UHD 66",
"UHD 50",
"UHD Remux",
"2160p",
"1080p",
"720p",
"576p",
"540p",
"480p"
],
hdtv: [
"HDTV",
"2160p",
"1080p",
"1080i",
"720p"
],
dvd: [
"DVD",
"DVD 9",
"DVD 5",
"DVD Remux",
"480p"
],
web: [
"WEB",
"2160p",
"1080p",
"720p",
"576p",
"540p",
"480p"
],
hddvd: "HD-DVD"
}
},
resolution: {
map: {
"2160p": [
"UHD 100",
"UHD 66",
"UHD 50",
"UHD Remux",
"2160p"
],
"1080p": [
"BD 50",
"BD 25",
"BD Remux",
"1080p"
],
"1080i": [
"BD 50",
"BD 25",
"BD Remux",
"1080i"
],
"720p": [
"720p"
],
"576p": [
"576p"
],
"540p": [
"540p"
],
"480p": [
"DVD 9",
"DVD 5",
"DVD Remux",
"480p"
],
other: [
"Other"
]
}
}
},
Bib: {
url: " https://bibliotik.me",
host: "bibliotik.me",
siteType: "Bib",
icon: "",
asSource: false,
asTarget: true,
uploadPath: "/upload",
name: {
selector: "#TitleField"
},
description: {
selector: "#DescriptionField"
},
anonymous: {
selector: "#AnonymousField"
},
image: {
selector: "#ImageField"
},
format: {
selector: "#FormatField",
map: {
epub: "15",
mobi: "16",
pdf: "2",
azw3: "21"
}
}
},
Blutopia: {
url: "https://blutopia.cc",
host: "blutopia.cc",
siteType: "UNIT3D",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/torrents/create?category_id=1",
torrentDownloadLinkSelector: 'a[href*="/torrents/download/"]',
needDoubanInfo: true,
seedDomSelector: ".torrent__buttons+.panelV2",
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: 'input[name="name"][class="form__text"]'
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anon"]'
},
torrent: {
selector: 'input[type="file"][accept=".torrent"]'
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "3",
encode: "12",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "12",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "11",
"2160p": "1",
"1080p": "2",
"1080i": "3",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
CHDBits: {
url: "https://ptchdbits.co",
host: "ptchdbits.co",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(6)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: "#torrent"
},
tags: {
chinese_audio: 'input[name="cnlang"]',
chinese_subtitle: 'input[name="cnsub"]',
diy: 'input[name="diy"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "403",
tvPack: "402",
documentary: "404",
cartoon: "405",
sport: "407",
concert: "406",
music: "406"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "5",
h265: "5",
x264: "1",
x265: "5",
mpeg2: "4",
mpeg4: "6",
vc1: "2",
xvid: "6"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "7",
dd: "4",
"dd+": "7",
flac: "1",
dts: "3",
truehd: "11",
lpcm: "13",
dtshdma: "10",
atmos: "10",
dtsx: "3"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: [
"19"
],
bluray: [
"1"
],
remux: [
"3"
],
encode: [
"4"
],
web: [
"18"
],
hdtv: [
"6"
]
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": [
"6"
],
"1080p": [
"1"
],
"1080i": [
"2"
],
"720p": [
"3"
],
"480p": [
"5"
]
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "8",
US: "3",
EU: "7",
HK: "5",
TW: "9",
JP: "4",
KR: "6",
OT: "0"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
chdbits: "14",
sgnb: "13",
remux: "1",
chdtv: "2",
chdpad: "15",
chdweb: "12",
chdhktv: "11",
stbox: "10",
onehd: "8",
blucook: "16",
hqc: "17",
gbt: "18",
kan: "19"
}
}
},
Carpt: {
url: "https://carpt.net",
host: "carpt.net",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="pt_gen"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: 'select[name="type"]',
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
cartoon: "403",
concert: "445",
variety: "405",
music: "406",
sport: "407"
}
},
videoCodec: {
selector: 'select[name="codec_sel[4]"]',
map: {
h264: "1",
hevc: "2",
h265: "2",
x264: "1",
x265: "2",
mpeg2: "3",
mpeg4: "1",
vc1: "4",
xvid: "5"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel[4]"]',
map: {
aac: "7",
ac3: "3'",
dd: "3",
"dd+": "3",
flac: "5",
dts: "2",
truehd: "1",
lpcm: "4",
dtshdma: "2",
atmos: "2",
dtsx: "2",
mp3: "6",
ape: "8",
wav: "10"
}
},
videoType: {
selector: 'select[name="medium_sel[4]"]',
map: {
uhdbluray: "7",
bluray: "8",
remux: "9",
encode: "1",
web: "2",
hdtv: "3",
dvd: "6",
hddvd: "1",
dvdrip: "4",
other: "9",
cd: "6"
}
},
resolution: {
selector: 'select[name="standard_sel[4]"]',
map: {
"2160p": "1",
"1080p": "2",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel[4]"]',
map: {
carpt: "1",
wiki: "2",
cmct: "3",
mteam: "4",
other: "5"
}
}
},
CinemaZ: {
url: "https://cinemaz.to",
host: "cinemaz.to",
siteType: "AvistaZ",
icon: "",
asSource: true,
asTarget: false,
uploadPath: "/upload.php",
seedDomSelector: "#content-area .block:last table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="/download/torrent/"]',
needDoubanInfo: true,
search: {
path: "/torrents",
params: {
search: "{imdb}",
"in": "1",
order: "size",
sort: "desc"
}
}
},
Cinematik: {
url: "https://cinematik.net",
host: "cinematik.net",
siteType: "Cinematik",
icon: "",
asSource: true,
asTarget: false,
uploadPath: "/upload.php",
seedDomSelector: "div.odiv_1 + table >tbody tr:nth-child(3n)",
needDoubanInfo: true,
search: {
path: "/browse.php",
params: {
search: "{imdb}",
cat: 0,
incldead: 1,
srchdtls: 1
}
}
},
Concertos: {
url: "https://concertos.live",
host: "concertos.live",
siteType: "UNIT3D-Legacy",
icon: "",
asSource: false,
asTarget: true,
seedDomSelector: "div.torrent > div.buttons.mbox.mbox--small-bottom",
uploadPath: "/upload",
needDoubanInfo: false,
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdb: "{imdb}",
order_by: "size"
}
},
name: {
selector: "#title"
},
description: {
selector: "div.sceditor-container textarea"
},
imdb: {
selector: 'input[name="imdb"]'
},
tmdb: {
selector: 'input[name="tmdb"]'
},
mediaInfo: {
selector: "#mediainfo"
},
anonymous: {
selector: 'input[name="anonymous"]'
},
category: {
selector: 'select[name="type_id"]',
map: {
BD100: "1",
BD66: "2",
UHD50: "3",
BD50: "4",
BD25: "5",
remux: [
"12",
"7"
],
encode: [
"8",
"10",
"11",
"13"
],
web: "9",
hdtv: "17",
dvd: [
"14",
"16"
],
dvdrip: "13",
other: ""
}
}
},
DicMusic: {
url: "https://dicmusic.com",
host: "dicmusic.com",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
searchstr: "{name}"
}
},
torrent: {
selector: "#file"
}
},
DiscFan: {
url: "https://discfan.net",
host: "discfan.net",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_url"]'
},
torrent: {
selector: 'input[type="file"]'
},
category: {
selector: "#browsecat1",
map: {
tv: "411",
tvPack: "411",
documentary: "413",
cartoon: "419",
sport: "417",
concert: "414",
variety: "416"
}
},
videoType: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "2",
bluray: "3",
remux: "0",
encode: "10",
web: "9",
hdtv: "1",
dvd: "4",
hddvd: "4",
dvdrip: "10",
other: "0"
}
},
area: {
selector: "#browsecat",
map: {
CN: "401",
US: "410",
EU: "410",
HK: "404",
TW: "405",
JP: "403",
KR: "406"
}
}
},
FL: {
url: "https://filelist.io",
host: "filelist.io",
siteType: "FL",
icon: "",
asSource: false,
asTarget: false,
uploadPath: "/upload.php",
search: {
path: "/browse.php",
imdbOptionKey: "3",
nameOptionKey: "0",
params: {
search: "{imdb}",
searchin: "{optionKey}",
sort: "3"
}
}
},
GPW: {
url: "https://greatposterwall.com",
host: "greatposterwall.com",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
searchstr: "{imdb}",
order_by: "size",
order_way: "desc",
group_results: 1,
action: "basic",
searchsubmit: 1
}
},
needDoubanInfo: true,
torrent: {
selector: "#file"
},
sourceInfo: {
editionTags: {
"10_bit": "10_bit",
hdr10: "hdr10",
hdr10plus: "hdr10_plus",
dolby_vision: "dolby_vision",
dts_x: "dts_x",
dolby_atmos: "dolby_atmos",
chinese_dubbed: "chinese_audio",
special_effects_subtitles: "special_effects_subtitles"
}
},
targetInfo: {
editionTags: {
"2_disc_set": "2_disc_set",
"2d_3d_edition": "2d_3d_edition",
"2_in_1": "2_in_1",
"3d": "3d",
"3d_anaglyph": "3d_anaglyph",
"3d_full_sbs": "3d_full_sbs",
"3d_half_ou": "3d_half_ou",
"3d_half_sbs": "3d_half_sbs",
"4k_remaster": "4k_remaster",
"4k_restoration": "4k_restoration",
director_s_cut: "director_s_cut",
dual_audio: "dual_audio",
english_dub: "english_dub",
extended_edition: "extended_edition",
extras: "extras",
masters_of_cinema: "masters_of_cinema",
scene: "sc",
the_criterion_collection: "the_criterion_collection",
theatrical_cut: "theatrical_cut",
two_disc_set: "2_disc_set",
remux: null,
rifftrax: "rifftrax",
uncut: "uncut",
unrated: "unrated",
warner_archive_collection: "warner_archive_collection",
with_commentary: "with_commentary"
}
},
description: {
selector: "#release_desc"
},
imdb: {
selector: "#imdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo[]"]'
},
category: {
selector: "#releasetype",
map: {
movie: "1",
tv: "3",
tvPack: "3",
concert: "5"
}
},
source: {
selector: "#source",
map: {
uhdbluray: "Blu-ray",
bluray: "Blu-ray",
web: "WEB",
hdtv: "HDTV",
hddvd: "HD-DVD",
dvd: "DVD",
other: "Other"
}
},
videoCodec: {
selector: "#codec",
map: {
h264: "H.264",
hevc: "H.265",
x264: "x264",
x265: "x265",
h265: "H.265",
mpeg2: "Other",
mpeg4: "H.264",
vc1: "Other",
xvid: "xvid"
}
},
resolution: {
selector: "#resolution",
map: {
NTSC: "NTSC",
PAL: "PAL",
"2160p": "2160p",
"1080p": "1080p",
"1080i": "1080i",
"720p": "720p",
"576p": "576p",
"480p": "480p"
}
},
format: {
selector: "#container",
map: {
mkv: "MKV",
mp4: "MP4",
avi: "AVI",
ts: "TS",
wmv: "WMV",
vob: "VOB IFO",
iso: "ISO",
mpg: "MPG",
m2ts: "m2ts"
}
},
videoType: {
selector: "#processing",
map: {
encode: "Encode",
remux: "Remux",
DIY: "DIY",
bluray: "Untouched",
uhdbluray: "Untouched",
dvd: "Untouched",
dvdrip: "Encode"
}
}
},
HD4FANS: {
url: "https://pt.hd4fans.org",
host: "hd4fans.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "403",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "405"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "1",
hddvd: "2",
remux: "3",
encode: "7",
web: "7",
hdtv: "5",
dvd: "6",
dvdrip: "6",
other: ""
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "10",
x264: "1",
x265: "10",
h265: "10",
mpeg2: "4",
mpeg4: "5",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
chd: "2",
mysilu: "3",
wiki: "4",
other: "5",
cmct: "6",
r2ts: "7",
kbits: "8"
}
}
},
HDArea: {
url: "https://hdarea.club",
host: "hdarea.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="dburl"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: [
"300",
"401",
"415",
"416",
"410",
"411",
"414",
"412",
"413",
"417"
],
tv: [
"402",
"403"
],
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "7",
x264: "7",
hevc: "6",
x265: "6",
h265: "6",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "0",
dvd: "0"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "11",
dd: "5",
"dd+": "4",
flac: "1",
dts: "3",
truehd: "7",
lpcm: "8",
dtshdma: "4",
atmos: "10",
dtsx: "0"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: [
"1",
"300"
],
bluray: [
"1",
"401"
],
remux: [
"3",
"415"
],
encode: "7",
web: [
"9",
"412"
],
hdtv: [
"5",
"413"
],
dvd: [
"2",
"414"
],
dvdrip: "6",
other: "0"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": [
"1",
"410"
],
"1080i": "2",
"720p": [
"3",
"411"
],
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
epic: "1",
hdarea: "2",
hdwing: "3",
wiki: "4",
ttg: "5",
other: "6",
mteam: "7",
hdapad: "8",
chd: "9",
hdaccess: "10",
hdatv: "11",
cxcy: "12",
cmct: "13"
}
}
},
HDAtmos: {
url: "https://hdatmos.club",
host: "hdatmos.club",
siteType: "NexusPHP",
icon: "",
asSource: false,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
cartoon: "405",
sport: "407",
concert: "406"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "10",
h265: "10",
x264: "1",
x265: "10",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "20",
ac3: "22",
dd: "23",
"dd+": "23",
flac: "17",
dts: "14",
truehd: "13",
lpcm: "15",
dtshdma: "10",
atmos: "11",
dtsx: "12"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "1",
remux: "3",
encode: "7",
web: "10",
hdtv: "5",
dvd: "6",
hddvd: "2",
dvdrip: "13",
other: "13"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "15",
"2160p": "10",
"1080p": "11",
"1080i": "12",
"720p": "13",
"576p": "14",
"480p": "14"
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "3",
US: "4",
EU: "8",
HK: "5",
TW: "3",
JP: "5",
KR: "6",
OT: "9"
}
},
source: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "6",
bluray: "6",
hdtv: "3",
dvd: "8",
web: "2",
vhs: "12",
hddvd: "7"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
other: "22"
}
}
},
HDBits: {
url: "https://hdbits.org",
host: "hdbits.org",
siteType: "HDB",
icon: "",
asSource: true,
asTarget: true,
needDoubanInfo: true,
uploadPath: "/upload.php",
seedDomSelector: "#details >tbody >tr:contains(Last seeded)",
torrentDownloadLinkSelector: 'a[href*="download.php/"]',
search: {
path: "/browse.php",
params: {
sort: "size",
d: "DESC",
search: "{imdb}"
}
},
name: {
selector: "#name"
},
description: {
selector: "#descr"
},
imdb: {
selector: "#imdb"
},
mediaInfo: {
selector: 'textarea[name="techinfo"]'
},
torrent: {
selector: "#file"
},
category: {
selector: "#type_category",
map: {
movie: "1",
tv: "2",
tvPack: "2",
documentary: "3",
concert: "4",
sport: "5",
cartoon: "1"
}
},
videoCodec: {
selector: "#type_codec",
map: {
h264: "1",
h265: "5",
hevc: "5",
x264: "1",
x265: "5",
mpeg2: "2",
vc1: "3",
xvid: "4",
bluray: "1",
uhdbluray: "5",
vp9: "6"
}
},
videoType: {
selector: "#type_medium",
map: {
uhdbluray: "1",
bluray: "1",
remux: "5",
encode: "3",
web: "6",
hdtv: "4"
}
}
},
HDDolby: {
url: "https://www.hddolby.com",
host: "hddolby.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_id"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: "#tag_gy",
diy: "#tag_diy",
chinese_subtitle: "#tag_zz",
cantonese_audio: "#tag_yy",
hdr: "#tag_hdr10",
hdr10_plus: "#tag_hdrm",
dolby_vision: "#tag_db"
},
anonymous: {
selector: 'input[name="uplver"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "3",
hevc: "2",
x265: "4",
h265: "2",
mpeg2: "6",
mpeg4: "0",
vc1: "5",
xvid: "0",
dvd: "0"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "2",
remux: "3",
encode: "10",
web: "6",
hdtv: "5",
dvd: "8",
dvdrip: "8",
other: "0"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "5",
dd: "5",
"dd+": "14",
flac: "7",
dts: "4",
truehd: "2",
lpcm: "3",
dtshdma: "1",
atmos: "2",
dtsx: "1"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "1",
"1080p": "2",
"1080i": "3",
"720p": "4",
"576p": "5",
"480p": "5"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
dream: "1",
hdo: "9",
dbtv: "10",
nazorip: "12",
mteam: "2",
frds: "7",
wiki: "4",
beast: "11",
chd: "5",
cmct: "6",
pthome: "3",
other: "8"
}
}
},
HDF: {
url: "https://hdf.world",
host: "hdf.world",
siteType: "gazelle",
icon: "",
asSource: false,
asTarget: false,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
order_way: "desc",
order_by: "time",
searchstr: "{name}",
group_results: "1",
action: "basic"
}
}
},
HDFans: {
url: "https://hdfans.org",
host: "hdfans.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~font table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "403",
concert: "441",
sport: "418",
cartoon: "417",
variety: "416",
app: "419",
ebook: "423",
audioBook: "405"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "3",
x264: "2",
x265: "4",
h265: "3",
mpeg2: "10",
mpeg4: "11",
vc1: "5",
xvid: "12"
}
},
source: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "2",
hdtv: "6",
dvd: "7",
web: "5",
vhs: "10",
hddvd: "17",
cd: "9",
sacd: "16"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "11",
ac3: "10",
dd: "21",
"dd+": "21",
flac: "12",
dts: "2",
truehd: "6",
lpcm: "7",
dtshdma: "4",
atmos: "1",
dtsx: "3",
ape: "13",
wav: "14",
mp3: "17",
m4a: "5",
other: "7"
}
},
videoType: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "1",
bluray: "3",
remux: "10",
encode: "9",
web: "7",
hdtv: "6",
dvd: "17",
dvdrip: "17",
other: "10"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "1",
"2160p": [
"2",
"10",
"9"
],
"1080p": [
"3",
"5",
"8"
],
"1080i": [
"4",
"5",
"8"
],
"720p": [
"5",
"11"
],
"576p": "6",
"480p": "6"
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "1",
US: "2",
EU: "8",
HK: "4",
TW: "5",
JP: "6",
KR: "7",
OT: "9"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
hdfans: "9",
hdf: "10",
chd: "1",
hdc: "2",
ttg: "19",
wiki: "3",
beast: "4",
cmct: "5",
frds: "6",
hdsky: "7",
ourbits: "17",
hdhome: "18",
pthome: "16",
tlf: "8",
pter: "20",
pbk: "21"
}
}
},
HDHome: {
url: "https://hdhome.org",
host: "hdhome.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_id"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: "#torrent"
},
tags: {
chinese_audio: "#tag_gy",
diy: "#tag_diy",
cantonese_audio: "#tag_yy",
chinese_subtitle: "#tag_zz",
hdr: "#tag_hdr10",
hdr10_plus: "#tag_hdrm",
dolby_vision: "#tag_db"
},
category: {
selector: "#browsecat",
map: {
movie: [
"411",
"412",
"413",
"414",
"415",
"450",
"499",
"416"
],
tv: [
"425",
"426",
"471",
"427",
"428",
"429",
"430",
"452",
"431"
],
tvPack: [
"432",
"433",
"434",
"435",
"436",
"437",
"438",
"502"
],
documentary: [
"417",
"418",
"419",
"420",
"421",
"451",
"500",
"422"
],
concert: "441",
sport: [
"442",
"443"
],
cartoon: [
"444",
"445",
"446",
"447",
"448",
"454",
"449",
"501"
],
variety: ""
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "12",
x264: "1",
x265: "2",
h265: "2",
mpeg2: "4",
mpeg4: "1",
vc1: "3",
xvid: "5",
dvd: "5"
}
},
source: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "9",
bluray: "1",
hdtv: "4",
dvd: "3",
web: "7",
vhs: "8",
hddvd: "8"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "15",
dd: "15",
"dd+": "15",
dts: "3",
truehd: "13",
lpcm: "14",
dtshdma: "11",
atmos: "12",
dtsx: "17"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: [
"10",
"499",
"500",
"502",
"501"
],
bluray: [
"1",
"450",
"451",
"452",
"453",
"454"
],
remux: [
"3",
"415",
"421",
"430",
"437",
"448"
],
encode: [
"7",
"411",
"412",
"413",
"414",
"416",
"417",
"418",
"419",
"420",
"422",
"425",
"426",
"471",
"427",
"428",
"429",
"431",
"432",
"433",
"434",
"435",
"436",
"438",
"444",
"445",
"446",
"447",
"449"
],
web: [
"11",
"411",
"412",
"413",
"414",
"416",
"417",
"418",
"419",
"420",
"422",
"425",
"426",
"471",
"427",
"429",
"431",
"432",
"433",
"434",
"436",
"438",
"444",
"445",
"446",
"447",
"449"
],
hdtv: [
"5",
"412",
"413",
"416",
"418",
"419",
"422",
"424",
"426",
"471",
"427",
"428",
"431",
"433",
"434",
"435",
"438",
"442",
"443",
"445",
"446",
"449"
],
dvd: [
"",
"411",
"417",
"425",
"432",
"444"
],
dvdrip: [
"7",
"411",
"417",
"425",
"432",
"444"
],
other: ""
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": [
"1",
"499",
"416",
"500",
"422",
"431",
"438",
"502",
"449",
"501"
],
"1080p": [
"2",
"414",
"420",
"429",
"436",
"447"
],
"1080i": [
"3",
"424",
"428",
"435",
"443"
],
"720p": [
"4",
"413",
"419",
"423",
"427",
"434",
"442",
"446"
],
"576p": [
"5",
"411",
"417",
"425",
"432",
"444"
],
"480p": [
"5",
"411",
"417",
"425",
"432",
"444"
]
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
"3201": "20",
"969154968": "22",
hdhome: "1",
hdh: "2",
hdhtv: "3",
hdhpad: "4",
hdhweb: "12",
shma: "17",
tvman: "21",
arin: "19",
ttg: "6",
mteam: "7",
other: "11"
}
}
},
HDPOST: {
url: "https://pt.hdpost.top",
host: "hdpost.top",
siteType: "UNIT3D",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload/1",
needDoubanInfo: true,
seedDomSelector: '#meta-info+.meta-general>.panel:has(".table-responsive"):first table tr:last',
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents",
params: {
name: "{name}",
imdb: "{imdb}"
}
},
name: {
selector: "#title"
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anonymous"]'
},
torrent: {
selector: "#torrent"
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "2",
encode: "3",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "12",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "1",
"2160p": "2",
"1080p": "3",
"1080i": "4",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
HDRoute: {
url: "http://hdroute.org",
host: "hdroute.org",
siteType: "NexusPHP",
icon: "",
asSource: false,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/browse.php",
imdbOptionKey: "4",
nameOptionKey: "0",
replaceKey: [
"tt",
""
],
params: {
s: "{name}",
dp: "0",
add: "0",
action: "s",
or: "4",
imdb: "{imdb}"
}
},
name: {
selector: "#title_eng"
},
subtitle: {
selector: 'input[name="title_sub"]'
},
description: {
selector: 'textarea[name="description"]'
},
poster: 'input[name="poster_big"]',
imdb: {
selector: "#upload-imdb_url"
},
anonymous: {
selector: 'input[name="is_anonymous"]'
},
torrent: {
selector: "#file_torrent"
},
tags: {
chinese_audio: 'input[name="is_mandrain"]',
cantonese_audio: 'input[name="is_cantonese"]',
diy: 'input[name="is_diyed"]',
chinese_subtitle: 'input[name="is_chs_sub_incl"]'
},
category: {
selector: "#type_category",
map: {
movie: "1",
tv: "3",
tvPack: "3",
documentary: "2",
concert: "5",
sport: "6",
cartoon: "4",
variety: "9"
}
},
videoCodec: {
selector: "#type_codec",
map: {
h264: "1",
hevc: "7",
x264: "1",
x265: "7",
h265: "7",
mpeg2: "3",
mpeg4: "1",
vc1: "2",
xvid: "4",
dvd: "3"
}
},
audioCodec: {
selector: "#type_audio",
map: {
aac: "9",
ac3: "5",
dd: "5",
"dd+": "5",
flac: "7",
dts: "4",
truehd: "3",
lpcm: "1",
dtshdma: "2",
atmos: "2",
dtsx: "4"
}
},
videoType: {
selector: "#type_medium",
map: {
uhdbluray: "1",
bluray: "1",
remux: "2",
encode: "4",
web: "6",
hddvd: "6",
hdtv: "3",
dvd: "6",
dvdrip: "6",
other: "6"
}
},
resolution: {
selector: "#type_resolution",
map: {
"2160p": "7",
"1080p": "1",
"1080i": "2",
"720p": "4",
"576p": "6",
"480p": "6"
}
}
},
HDSpace: {
url: "https://hd-space.org",
host: "hd-space.org",
siteType: "HDSpace",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#mcol>table>tbody>tr:last table:first>tbody>tr:nth-child(2)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
needDoubanInfo: true,
uploadPath: "/index.php?page=upload",
search: {
path: "/index.php",
imdbOptionKey: "2",
nameOptionKey: "0",
replaceKey: [
"tt",
""
],
params: {
search: "{imdb}",
page: "torrents",
options: "{optionKey}",
order: 4,
by: 2
}
},
name: {
selector: "#filename"
},
imdb: {
selector: 'input[name="imdb"]'
},
description: {
selector: 'textarea[name="info"]'
},
anonymous: {
selector: 'input[name="anonymous"][value="true"]'
},
torrent: {
selector: 'input[type="file"][name="torrent"]'
},
category: {
selector: 'select[name="category"]',
map: {
movie: [
"15",
"40",
"16",
"18",
"19",
"41"
],
tv: [
"15",
"40",
"16",
"21",
"22"
],
tvPack: [
"15",
"40",
"16",
"21",
"22"
],
documentary: [
"15",
"40",
"16",
"24",
"25"
],
cartoon: [
"15",
"40",
"16",
"27",
"28"
],
concert: [
"15",
"40",
"16",
"31"
]
}
},
videoType: {
map: {
uhdbluray: [
"15"
],
bluray: [
"15"
],
remux: [
"40"
],
encode: [
"18",
"19",
"41",
"21",
"22",
"24",
"25",
"27",
"28",
"31"
],
web: [
"18",
"19",
"41",
"21",
"22",
"24",
"25",
"27",
"28",
"31"
],
hdtv: [
"18",
"19",
"41",
"21",
"22",
"24",
"25",
"27",
"28",
"31"
]
}
},
resolution: {
map: {
"2160p": [
"15",
"40",
"16",
"41"
],
"1080p": [
"19",
"22",
"25",
"28",
"31"
],
"1080i": [
"19",
"22",
"25",
"28",
"31"
],
"720p": [
"18",
"21",
"24",
"24",
"31"
]
}
}
},
HDT: {
url: "https://hd-torrents.org",
host: "hd-torrents.org",
siteType: "HDT",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: ".listadetails>tbody>tr:nth-child(2)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
needDoubanInfo: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "2",
nameOptionKey: "3",
params: {
search: "{imdb}",
options: "{optionKey}",
order: "size",
by: "DESC"
}
},
name: {
selector: 'input[name="filename"]'
},
imdb: {
selector: 'input[name="infosite"]'
},
description: {
selector: 'textarea[name="info"]'
},
tags: {
hdr: 'input[name="HDR10"]',
hdr10_plus: 'input[name="HDR10Plus"]',
dolby_vision: 'input[name="DolbyVision"]'
},
anonymous: {
selector: 'input[name="anonymous"][value="true"]'
},
torrent: {
selector: 'input[name="torrent"]',
announce: "https://hdts-announce.ru/announce.php"
},
category: {
selector: 'select[name="category"]',
map: {
movie: [
"70",
"1",
"71",
"2",
"64",
"5",
"3",
"63"
],
tv: [
"72",
"59",
"73",
"60",
"65",
"30",
"38"
],
tvPack: [
"72",
"59",
"73",
"60",
"65",
"30",
"38"
],
documentary: [
"70",
"1",
"71",
"2",
"64",
"5",
"3",
"63"
],
cartoon: [
"70",
"1",
"71",
"2",
"64",
"5",
"3",
"63"
],
concert: [
"61",
"62",
"66",
"57",
"45",
"44"
],
variety: [
"72",
"59",
"73",
"60",
"65",
"30",
"38"
]
}
},
videoType: {
map: {
uhdbluray: [
"70",
"72"
],
bluray: [
"1",
"59",
"61"
],
remux: [
"71",
"2",
"62",
"73",
"60"
],
encode: [
"64",
"5",
"3",
"65",
"30",
"38",
"66",
"57",
"45"
],
web: [
"64",
"5",
"3",
"65",
"30",
"38",
"66",
"57",
"45"
],
hdtv: [
"64",
"5",
"3",
"65",
"30",
"38",
"66",
"57",
"45"
]
}
},
resolution: {
map: {
"2160p": [
"70",
"72",
"71",
"73",
"64",
"65",
"66"
],
"1080p": [
"1",
"59",
"61",
"2",
"60",
"62",
"5",
"30",
"57"
],
"1080i": [
"1",
"59",
"61",
"2",
"60",
"62",
"5",
"30",
"57"
],
"720p": [
"3",
"38",
"45"
]
}
}
},
HDTime: {
url: "https://hdtime.org",
host: "hdtime.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
mediaInfo: {
selector: 'textarea[name="technical_info"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
diy: 'input[type="checkbox"][name="tags[]"][value="8"]',
chinese_audio: 'input[type="checkbox"][name="tags[]"][value="16"]',
chinese_subtitle: 'input[type="checkbox"][name="tags[]"][value="32"]',
hdr: 'input[type="checkbox"][name="tags[]"][value="64"]',
hdr10_plus: 'input[type="checkbox"][name="tags[]"][value="64"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "12",
x264: "10",
x265: "12",
h265: "12",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1 ",
bluray: "1",
remux: "3",
encode: "7",
web: "7",
hddvd: "2",
hdtv: "5",
dvd: "6",
dvdrip: "6",
other: "0"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
hdtime: "6",
hdt: "12",
vtime: "15",
padtime: "7",
cmct: "8",
wiki: "4",
beast: "3",
chd: "2",
other: "5"
}
}
},
HDU: {
url: "https://pt.hdupt.com",
host: "hdupt.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "14",
x264: "16",
x265: "14",
h265: "14",
mpeg2: "18",
mpeg4: "18",
vc1: "2",
xvid: "3",
dvd: "18"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "2",
dd: "2",
"dd+": "2",
flac: "7",
dts: "4",
truehd: "3",
lpcm: "11",
dtshdma: "1",
atmos: "17",
dtsx: "16"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "11",
bluray: "1",
remux: "3",
encode: "7",
web: "10",
hddvd: "2",
hdtv: "5",
dvd: "6",
dvdrip: "6",
other: "0"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "1",
US: "2",
EU: "2",
HK: "3",
TW: "3",
JP: "4",
KR: "5",
IND: "6",
SEA: "8",
OT: "7"
}
}
},
HH: {
url: "https://hhanclub.top",
host: "hhanclub.top",
siteType: "HanHan",
icon: "",
asSource: true,
asTarget: false,
uploadPath: "/upload.php",
needDoubanInfo: true,
seedDomSelector: ".bg-content_bg div.leading-6:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
}
},
HUNO: {
url: "https://hawke.uno",
host: "hawke.uno",
siteType: "UNIT3D-Legacy",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/torrents/create?category_id=1",
torrentDownloadLinkSelector: 'a[href*="/torrents/download/"]',
needDoubanInfo: true,
seedDomSelector: ".torrent__buttons+.panelV2",
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: 'input[name="name"][class="form__text"]'
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anon"]'
},
torrent: {
selector: 'input[type="file"][accept=".torrent"]'
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "3",
encode: "12",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "12",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "11",
"2160p": "1",
"1080p": "2",
"1080i": "3",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
KEEPFRDS: {
url: "https://pt.keepfrds.com",
host: "keepfrds.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
needDoubanInfo: true,
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="small_descr"]'
},
subtitle: {
selector: "#name"
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_url"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "408",
sport: "407",
cartoon: "405",
variety: "403"
}
},
source: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "10",
bluray: "1",
hdtv: "4",
dvd: "3",
web: "7",
vhs: "8",
hddvd: "8"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "3",
hevc: "0",
x264: "3",
x265: "0",
h265: "0",
mpeg2: "17",
mpeg4: "1",
vc1: "16",
xvid: "5"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "0",
"2160p": "7",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "5",
"480p": "5"
}
}
},
KG: {
url: "https://karagarga.in",
host: "karagarga.in",
siteType: "KG",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: ".outer h1~table:first>tbody>tr:nth-child(6)",
torrentDownloadLinkSelector: 'a[href*="/down.php/"]',
needDoubanInfo: true,
search: {
path: "/browse.php",
imdbOptionKey: "imdb",
nameOptionKey: "title",
params: {
search: "{imdb}",
search_type: "{optionKey}",
sort: "size",
d: "DESC"
}
},
source: {
selector: "select[name='source']",
map: {
uhdbluray: "Blu-ray",
bluray: "Blu-ray",
hdtv: "TV",
dvd: "DVD",
web: "WEB",
vhs: "VHS",
hddvd: "HD-DVD",
other: "Other"
}
},
resolution: {
selector: "select[name='hdrip']",
map: {
"720p": "1",
"1080p": "2",
bluray: "3"
}
},
torrent: {
selector: 'input[name="file"]'
},
genres: {
map: {
Action: "4",
Adventure: "55",
Animation: "5",
Arthouse: "6",
Camp: "43",
Classics: "8",
Comedy: "9",
Crime: "10",
Cult: "11",
Documentary: "20",
Drama: "12",
Epic: "44",
Erotica: "13",
Experimental: "51",
Exploitation: "47",
Fantasy: "14",
"Film Noir": "15",
Giallo: "53",
Horror: "17",
"Martial Arts": "18",
Musical: "19",
Mystery: "54",
Performance: "60",
Philosophy: "48",
Politics: "49",
Romance: "50",
"Sci-Fi": "21",
Short: "22",
Silent: "23",
Thriller: "24",
TV: "25",
"Video Art": "56",
War: "26",
Western: "27"
}
},
country: {
map: {
USA: "2",
UK: "12",
Germany: "7",
Italy: "9",
"---": "255",
Abkhazia: "119",
Afghanistan: "54",
"Akrotiri and Dhekelia": "120",
"Aland Islands": "121",
Albania: "65",
Algeria: "35",
"American Samoa": "122",
Andorra: "68",
Angola: "36",
Anguilla: "123",
"Antigua Barbuda": "89",
Argentina: "19",
Armenia: "124",
Aruba: "125",
"Ascension Island": "126",
Australia: "20",
Austria: "37",
Azerbaijan: "118",
Bahamas: "82",
Bahrain: "127",
Bangladesh: "86",
Barbados: "85",
Belarus: "129",
Belgium: "16",
Belize: "34",
Benin: "116",
Bermuda: "130",
Bhutan: "131",
Bolivia: "132",
"Bosnia Herzegovina": "67",
Botswana: "133",
Brazil: "18",
"British Virgin Islands": "134",
Brunei: "135",
Bulgaria: "104",
"Burkina Faso": "60",
Burundi: "136",
Cambodia: "84",
Cameroon: "137",
Canada: "5",
"Cape Verde": "138",
"Cayman Islands": "139",
"Central African Republic": "140",
Chad: "114",
Chile: "51",
China: "8",
"Christmas Island": "141",
"Cocos (Keeling) Islands": "142",
Colombia: "99",
Comoros: "143",
"Congo (Brazzaville)": "53",
"Congo-Kinshasa (Zaire)": "252",
"Cook Islands": "144",
"Costa Rica": "102",
"Cote d'Ivoire": "145",
Croatia: "97",
Cuba: "52",
Cyprus: "146",
"Czech Republic": "46",
Denmark: "10",
Djibouti: "147",
Dominica: "148",
"Dominican Republic": "41",
Ecuador: "81",
Egypt: "103",
"El Salvador": "149",
"Equatorial Guinea": "150",
Eritrea: "151",
Estonia: "98",
Ethiopia: "112",
"European Union": "253",
"Falkland Islands": "153",
"Faroe Islands": "111",
Fiji: "152",
Finland: "4",
France: "6",
"French Polynesia": "154",
Gabon: "155",
Gambia: "156",
Georgia: "108",
Ghana: "157",
Gibraltar: "158",
Greece: "42",
Greenland: "159",
Grenada: "160",
Guam: "161",
Guatemala: "43",
Guernsey: "162",
Guinea: "113",
"Guinea-Bissau": "163",
Guyana: "164",
Haiti: "165",
Honduras: "79",
"Hong Kong": "33",
Hungary: "74",
Iceland: "62",
India: "70",
Indonesia: "166",
Iran: "107",
Iraq: "167",
Ireland: "13",
"Isla de Muerte": "105",
"Isle of Man": "168",
Israel: "44",
Jamaica: "31",
Japan: "17",
Jersey: "170",
Jordan: "169",
Kazakhstan: "110",
Kenya: "172",
Kiribati: "58",
Kosovo: "173",
Kuwait: "171",
Kyrgyzstan: "80",
Laos: "87",
Latvia: "101",
Lebanon: "100",
Lesotho: "174",
Liberia: "175",
Libya: "176",
Liechtenstein: "177",
Lithuania: "69",
Luxembourg: "32",
Macau: "178",
Macedonia: "179",
Madagascar: "180",
Malawi: "181",
Malaysia: "40",
Maldives: "182",
Mali: "115",
Malta: "183",
"Marshall Islands": "184",
Mauritania: "185",
Mauritius: "186",
Mayotte: "187",
Mexico: "25",
Micronesia: "188",
Moldova: "189",
Monaco: "190",
Mongolia: "109",
Montenegro: "257",
Montserrat: "191",
Morocco: "192",
Mozambique: "193",
Myanmar: "194",
"Nagorno-Karabakh": "195",
Namibia: "196",
Nauru: "63",
Nepal: "197",
Netherlands: "15",
"Netherlands Antilles": "71",
"New Caledonia": "198",
"New Zealand": "21",
Nicaragua: "199",
Niger: "200",
Nigeria: "61",
Niue: "201",
"Norfolk Island": "202",
"North Korea": "96",
"Northern Cyprus": "203",
"Northern Mariana Islands": "204",
Norway: "11",
Oman: "205",
Pakistan: "45",
Palau: "207",
Palestine: "208",
Panama: "206",
"Papua New Guinea": "209",
Paraguay: "90",
Peru: "83",
Philippines: "59",
"Pitcairn Islands": "210",
Poland: "14",
Portugal: "24",
"Puerto Rico": "50",
Qatar: "211",
Romania: "75",
Russia: "3",
Rwanda: "212",
"Saint Helena": "213",
"Saint Kitts and Nevis": "214",
"Saint Lucia": "215",
"Saint Vincent and the Grenadines": "217",
"Saint-Pierre and Miquelon": "216",
Samoa: "39",
"San Marino": "219",
"São Tomé and Príncipe": "220",
"Saudi Arabia": "221",
Sealand: "258",
Senegal: "94",
Serbia: "256",
"Serbia and Montenegro": "47",
Seychelles: "48",
"Sierra Leone": "222",
Singapore: "26",
Slovakia: "223",
Slovenia: "64",
"Solomon Islands": "224",
Somalia: "225",
Somaliland: "226",
"South Africa": "29",
"South Korea": "30",
"South Ossetia": "227",
Spain: "23",
"Sri Lanka": "228",
Sudan: "229",
Suriname: "230",
Svalbard: "231",
Swaziland: "232",
Sweden: "1",
Switzerland: "57",
Syria: "233",
Taiwan: "49",
Tajikistan: "234",
Tanzania: "235",
Thailand: "93",
"Timor-Leste": "236",
Togo: "95",
Tokelau: "237",
Tonga: "238",
Transnistria: "239",
"Trinidad & Tobago": "78",
"Tristan da Cunha": "240",
Tunisia: "106",
Turkey: "55",
Turkmenistan: "66",
"Turks and Caicos Islands": "241",
Tuvalu: "242",
Uganda: "243",
Ukraine: "72",
"United Arab Emirates": "244",
"United Kingdom": "12",
Uruguay: "88",
USSR: "92",
Uzbekistan: "56",
Vanuatu: "76",
"Various/International": "117",
"Vatican City": "245",
Venezuela: "73",
Vietnam: "77",
"Virgin Islands": "246",
"Wallis and Futuna": "247",
"Western Sahara": "248",
World: "254",
Yemen: "249",
Yugoslavia: "38",
Zambia: "250",
Zimbabwe: "251"
}
}
},
KamePT: {
url: "https://kamept.com",
host: "kamept.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table>tbody>tr:nth-child(4)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
}
},
LST: {
url: "https://lst.gg",
host: "lst.gg",
siteType: "UNIT3D",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/torrents/create?category_id=1",
needDoubanInfo: true,
seedDomSelector: "#meta-info+.meta-general>.panel:first",
torrentDownloadLinkSelector: 'a[href*="/download/torrent/"]',
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: "#title"
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mal: {
selector: "#automal"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anonymous"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "2",
encode: "3",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "3",
other: "7"
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "1",
"2160p": "2",
"1080p": "3",
"1080i": "4",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
Lemon: {
url: "https://lemonhd.club",
host: "lemonhd.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: "#torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="imdb"]'
},
douban: {
selector: 'input[name="douban"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
category: {
selector: 'select[name="type"]',
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403",
music: "411"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "1",
hevc: "3",
x265: "3",
h265: "3",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "100",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "8",
ac3: "6",
dd: "6",
"dd+": "7",
flac: "10",
dts: "5",
truehd: "2",
lpcm: "9",
dtshdma: "3",
atmos: "1",
dtsx: "4",
ape: "12",
wav: "11",
mp3: "100",
m4a: "100",
other: "100"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "3",
bluray: "1",
remux: "12",
encode: "13",
web: "10",
hdtv: "11",
dvd: "2",
dvdrip: "13",
other: '5"'
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "6",
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "100",
"480p": "4"
}
},
area: {
selector: 'select[name="team_sel"]',
map: {
CN: "3",
US: "1",
EU: "1",
HK: "2",
TW: "2",
JP: "4",
KR: "4",
IN: "5",
OT: "10"
}
},
tags: {
chinese_audio: 'input[name="tag_gy"]',
diy: 'input[name="tag_diy"]',
cantonese_audio: 'input[name="tag_yy"]',
chinese_subtitle: 'input[name="tag_zz"]',
hdr: 'input[name="tag_hdr"]',
dolby_vision: 'input[name="tag_dv"]'
}
},
MDU: {
url: "https://monikadesign.uk",
host: "monikadesign.uk",
siteType: "UNIT3D-Legacy",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload/1",
needDoubanInfo: true,
seedDomSelector: "#meta-info+.meta-general>.panel:first",
torrentDownloadLinkSelector: 'a[href*="/download/torrent/"]',
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: "#title"
},
subtitle: {
selector: 'input[name="subhead"]'
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mal: {
selector: "#automal"
},
bgm: {
selector: "#bgm"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
bdinfo: {
selector: 'textarea[name="bdinfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anonymous"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#autocat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "2",
encode: "3",
web: "4",
hdtv: "",
dvd: "",
dvdrip: "",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "1",
"2160p": "2",
"1080p": "3",
"1080i": "4",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
MTV: {
url: "https://www.morethantv.me",
host: "morethantv.me",
siteType: "gazelle",
asSource: true,
asTarget: true,
needDoubanInfo: true,
uploadPath: "/upload.php",
search: {
path: "/torrents/browse",
params: {
searchtext: "{imdb}",
title: "{name}"
}
},
icon: "",
seedDomSelector: "#descbox",
torrentDownloadLinkSelector: 'a[href*="/torrents.php"][title="Download Torrent"]',
name: {
selector: 'input[name="title"][class="long"]'
},
description: {
selector: "#desc"
},
anonymous: {
selector: 'input[name="anonymous"][value="1"]'
},
torrent: {
selector: "#file"
},
category: {
selector: "#category",
map: {
movie: "1",
tv: "5",
tvPack: "5",
documentary: "1",
concert: "1"
}
}
},
MTeam: {
url: "https://kp.m-team.cc",
host: "m-team.cc",
siteType: "MTeam",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload",
seedDomSelector: ".detail-view .ant-descriptions-view>table>tbody .ant-descriptions-row:nth-child(3)",
search: {
path: "/browse",
replaceKey: [
"tt",
"https://www.imdb.com/title/tt"
],
params: {
keyword: "{imdb}",
sort: "size%3Adescend"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: "#smallDescr"
},
description: {
selector: "#descr"
},
mediaInfo: {
selector: "#mediainfo"
},
imdb: {
selector: "#imdb"
},
douban: {
selector: "#douban"
},
tags: {
diy: 'input[value="diy"]',
chinese_subtitle: 'input[value="sub"]',
chinese_audio: 'input[value="dub"]'
},
torrent: {
selector: "#torrent-input"
},
category: {
selector: "#category",
map: {
movie: [
"401",
"419",
"420",
"421",
"439"
],
tv: [
"403",
"402",
"435",
"438"
],
tvPack: [
"403",
"402",
"435",
"438"
],
documentary: "404",
concert: "406",
sport: "407",
music: "434",
cartoon: "405",
app: "422",
ebook: "427",
magazine: "427",
audioBook: "427"
}
},
videoCodec: {
selector: "#videoCodec",
map: {
h264: "1",
hevc: "16",
h265: "16",
x264: "1",
x265: "16",
mpeg2: "4",
mpeg4: "15",
vc1: "2",
xvid: "3",
av1: "19"
}
},
audioCodec: {
selector: "#audioCodec",
map: {
aac: "6",
ac3: "8",
dd: "8",
"dd+": "8",
flac: "1",
dts: "3",
truehd: "9",
lpcm: "7",
dtshdma: "11",
atmos: "10",
dtsx: "3",
ape: "2",
wav: "7",
other: "7",
mp3: "4"
}
},
source: {
selector: "#source",
map: {
uhdbluray: "1",
bluray: "1",
hdtv: "4",
dvd: "3",
web: "8",
vhs: "6",
hddvd: "6"
}
},
videoType: {
selector: "#medium",
map: {
uhdbluray: [
"1",
"421",
"438"
],
bluray: [
"1",
"421",
"438"
],
remux: [
"3",
"439"
],
encode: [
"7",
"401",
"419",
"403",
"402"
],
web: [
"10",
"419",
"402"
],
hdtv: [
"5",
"419",
"402"
],
dvd: [
"6",
"420",
"435"
],
dvdrip: [
"7",
"401",
"403"
],
other: ""
}
},
resolution: {
selector: "#standard",
map: {
"2160p": [
"6",
"419",
"402"
],
"1080p": [
"1",
"419",
"402"
],
"1080i": [
"2",
"419",
"402"
],
"720p": [
"3",
"419",
"402"
],
"576p": [
"5",
"401",
"403"
],
"480p": [
"5",
"401",
"403"
]
}
},
area: {
selector: "#processing",
map: {
CN: "1",
US: "2",
EU: "2",
HK: "3",
TW: "3",
JP: "4",
KR: "5",
OT: "6"
}
},
team: {
selector: "#team",
map: {
mteam: "9",
tnp: "23",
kishd: "7",
bmdru: "6",
onehd: "18",
cnhk: "19",
stbox: "20",
r2hd: "21",
pack: "8",
geek: "24",
qhstudio: "34"
}
}
},
NYPT: {
url: "https://nanyangpt.com",
host: "nanyangpt.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="dburl"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "406",
concert: "407",
sport: "405",
cartoon: "403",
variety: "404"
}
}
},
OpenSub: {
url: "https://www.opensubtitles.org",
host: "opensubtitles.org",
siteType: "subtitles",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/en/search/sublanguageid-all/imdbid-{name}"
}
},
Orpheus: {
url: "https://orpheus.network",
host: "orpheus.network",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
searchstr: "{name}"
}
},
torrent: {
selector: "#file"
}
},
OurBits: {
url: "https://ourbits.club",
host: "ourbits.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: "#torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
poster: 'input[name="picture"]',
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: "#tagGY",
diy: "#tagDIY",
cantonese_audio: "#tag_yy",
chinese_subtitle: "#tagZZ",
hdr: "#tagHDR10",
hdr10_plus: "#tagHDR10P",
dolby_vision: "#tagDB"
},
category: {
selector: 'select[name="type"]',
map: {
movie: "401",
"3d": "402",
tv: "412",
tvPack: "405",
documentary: "410",
concert: "419",
sport: "415",
cartoon: "411",
variety: "413",
music: "416"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "12",
hevc: "14",
x264: "12",
x265: "14",
h265: "14",
mpeg2: "15",
mpeg4: "12",
vc1: "16",
xvid: "17",
dvd: "18"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "7",
ac3: "6",
dd: "6",
"dd+": "6",
flac: "13",
dts: "4",
truehd: "2",
lpcm: "5",
dtshdma: "1",
atmos: "14",
dtsx: "21"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "12",
bluray: "1",
remux: "4",
encode: "7",
web: "9",
hdtv: "5",
dvd: "2",
dvdrip: "2",
other: "0"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "1",
US: "2",
EU: "2",
HK: "3",
TW: "3",
JP: "4",
KR: "5",
OT: "6"
}
}
},
PTHome: {
url: "https://www.pthome.net",
host: "pthome.net",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_id"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: "#torrent"
},
tags: {
chinese_audio: "#tag_gy",
diy: "#tag_diy",
cantonese_audio: "#tag_yy",
chinese_subtitle: "#tag_zz",
hdr: "#tag_hdr10",
hdr10_plus: "#tag_hdrm",
dolby_vision: "#tag_db"
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
concert: "408",
sport: "407",
cartoon: "405",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "1",
hevc: "6",
x265: "6",
h265: "6",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "5",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "18",
dd: "18",
"dd+": "18",
flac: "1",
dts: "3",
truehd: "20",
lpcm: "21",
dtshdma: "19",
atmos: "19",
dtsx: "3"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "12",
bluray: "1",
remux: "3",
encode: "15",
web: "10",
hdtv: "5",
dvd: "2",
dvdrip: "15",
other: "11"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"4320p": "10",
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
pthome: "19",
pth: "21",
pthweb: "20",
pthtv: "22",
pthaudio: "23",
pthebook: "24",
pthmusic: "25",
other: "5"
}
}
},
PTN: {
url: "https://piratethenet.org",
host: "piratethenet.org",
siteType: "PTN",
icon: "",
asSource: false,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/browse.php",
params: {
name: "{name}",
imdb: "{imdb}"
}
},
name: {
selector: "#name"
},
description: {
selector: "#descr"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
}
},
PTP: {
url: "https://passthepopcorn.me",
host: "passthepopcorn.me",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
needDoubanInfo: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
action: "advanced",
searchstr: "{imdb}"
}
},
sourceInfo: {
editionTags: {
HDR10: "hdr10",
"HDR10+": "hdr10_plus",
"Dolby Atmos": "dolby_atmos",
"Dolby Vision": "dolby_vision",
"10-bit": "10_bit",
"2-Disc Set": "2_disc_set",
"2D/3D Edition": "2d_3d_edition",
"2in1": "2_in_1",
"3D": "3d",
"3D Anaglyph": "3d_anaglyph",
"3D Full SBS": "3d_full_sbs",
"3D Half OU": "3d_half_ou",
"3D Half SBS": "3d_half_sbs",
"4K Remaster": "4k_remaster",
"4K Restoration": "4k_restoration",
"Director's Cut": "director_s_cut",
"Dual Audio": "dual_audio",
"English Dub": "english_dub",
"Extended Cut": "extended_edition",
"Extended Edition": "extended_edition",
Extras: "extras",
"Masters of Cinema": "masters_of_cinema",
Scene: "scene",
"The Criterion Collection": "the_criterion_collection",
"Theatrical Cut": "theatrical_cut",
Trumpable: null,
"Two-Disc Set": "two_disc_set",
Remux: "remux",
Rifftrax: "rifftrax",
Uncut: "uncut",
Unrated: "unrated",
"Warner Archive Collection": "warner_archive_collection",
"With Commentary": "with_commentary"
}
},
targetInfo: {
editionTags: {
"2_disc_set": "2-Disc Set",
"2_in_1": "2in1",
"2d_3d_edition": "2D/3D Edition",
"3d_anaglyph": "3D Anaglyph",
"3d_full_sbs": "3D Full SBS",
"3d_half_ou": "3D Half OU",
"3d_half_sbs": "3D Half SBS",
"4k_remaster": "4K Remaster",
"4k_restoration": "4K Restoration",
director_s_cut: "Director's Cut",
dual_audio: "Dual Audio",
english_dub: "English Dub",
extended_edition: "Extended Edition",
extras: "Extras",
masters_of_cinema: "Masters of Cinema",
scene: "sc",
the_criterion_collection: "The Criterion Collection",
theatrical_cut: "Theatrical Cut",
two_disc_set: "2-Disc Set",
remux: "Remux",
rifftrax: "Rifftrax",
uncut: "Uncut",
unrated: "Unrated",
warner_archive_collection: "Warner Archive Collection",
with_commentary: "With Commentary",
"10_bit": "10-bit",
dts_x: "DTS:X",
dolby_atmos: "Dolby Atmos",
dolby_vision: "Dolby Vision",
hdr10: "HDR10",
hdr10plus: "HDR10+"
}
},
description: {
selector: "#release_desc"
},
poster: "#image",
imdb: {
selector: "#imdb"
},
anonymous: {
selector: 'input[name="uplver"]'
},
mediaInfo: {
selector: "#Media_BDInfo"
},
screenshots: {
selector: "#url_vimages"
},
torrent: {
selector: "#file"
},
category: {
selector: "#categories",
map: {
movie: "Feature Film",
tv: "Miniseries",
tvPack: "Miniseries",
documentary: "Feature Film",
concert: "Live Performance"
}
},
videoCodec: {
selector: "#codec",
map: {
h264: "H.264",
hevc: "H.265",
x264: "x264",
x265: "x265",
h265: "H.265",
mpeg2: "Other",
mpeg4: "H.264",
vc1: "Other",
xvid: "XviD",
DVD5: "DVD5",
DVD9: "DVD9",
BD100: "BD100",
BD66: "BD66",
BD50: "BD50",
BD25: "BD25"
}
},
source: {
selector: "#source",
map: {
uhdbluray: "Blu-ray",
bluray: "Blu-ray",
hdtv: "HDTV",
dvd: "DVD",
web: "WEB",
vhs: "VHS",
hddvd: "HD-DVD"
}
},
resolution: {
selector: "#resolution",
map: {
"2160p": "2160p",
"1080p": "1080p",
"1080i": "1080i",
"720p": "720p",
"576p": "576p",
"480p": "480p",
NTSC: "NTSC",
PAL: "PAL",
other: "Other"
}
}
},
PTSBAO: {
url: "https://ptsbao.club",
host: "ptsbao.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(7)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: "input[name='small_descr']"
},
description: {
selector: "textarea[name='descr']"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
mediaInfo: {
selector: 'textarea[name="technical_info"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_subtitle: 'input[type="checkbox"][value="6"]',
hdr10_plus: 'input[type="checkbox"][value="7"]',
chinese_audio: 'input[type="checkbox"][value="5"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
variety: "403",
documentary: "404",
cartoon: "405",
music: "414",
other: "409"
}
},
source: {
selector: 'select[name="medium_sel[1]"]',
map: {
uhdbluray: "10",
bluray: "1",
hdtv: "5",
dvd: "3",
web: "2",
vhs: "9",
hddvd: "9"
}
},
videoCodec: {
selector: 'select[name="codec_sel[1]"]',
map: {
h264: "1",
hevc: "6",
x264: "1",
x265: "6",
h265: "6",
mpeg2: "4",
mpeg4: "5",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel[1]"]',
map: {
aac: "6",
ac3: "11",
dd: "10",
"dd+": "10",
dts: "3",
truehd: "9",
lpcm: "12",
dtshdma: "8",
atmos: "8",
dtsx: "13",
flac: "1"
}
},
videoType: {
selector: 'select[name="processing_sel[1]"]',
map: {
uhdbluray: "5",
bluray: "5",
remux: "1",
encode: "2",
web: "2",
hdtv: "3",
dvd: "3",
dvdrip: "3",
other: "3"
}
},
resolution: {
selector: 'select[name="standard_sel[1]"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "1",
"720p": "2",
"576p": "3",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
ffansbd: "8",
ops: "11",
ffansweb: "12",
ffanstv: "13",
hqc: "10",
ttg: "3",
hdc: "6",
chd: "2",
hdsky: "9",
cmct: "4",
frds: "5",
other: "7",
ffansdvd: "14",
fhdmv: "15",
enichi: "16"
}
}
},
PTZone: {
url: "https://ptzone.xyz",
host: "ptzone.xyz",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(6)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
mediaInfo: {
selector: 'textarea[name="technical_info"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="pt_gen"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
hdr: 'input[name="tags[4][]"]'
},
category: {
selector: 'select[name="type"]',
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
cartoon: "405",
sport: "407",
concert: "406",
music: "406",
variety: "403"
}
},
videoCodec: {
selector: 'select[name="codec_sel[4]"]',
map: {
h264: "1",
hevc: "6",
h265: "6",
x264: "1",
x265: "6",
mpeg2: "3",
mpeg4: "4",
vc1: "2",
xvid: "5"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel[4]"]',
map: {
aac: "6",
ac3: "8",
dd: "11",
"dd+": "12",
flac: "1",
dts: "9",
truehd: "14",
lpcm: "1",
dtshdma: "10",
atmos: "10",
dtsx: "9",
ape: "2",
mp3: "4",
ogg: "5",
wav: "15"
}
},
videoType: {
selector: 'select[name="medium_sel[4]"]',
map: {
uhdbluray: "10",
bluray: "1",
hddvd: "2",
remux: "3",
encode: "7",
web: "4",
hdtv: "5",
dvd: "6",
dvdrip: "6",
cd: "8"
}
},
resolution: {
selector: 'select[name="standard_sel[4]"]',
map: {
"4320p": "5",
"2160p": "6",
"1080p": "1",
"1080i": "2",
"720p": "3",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel[4]"]',
map: {
hds: "1",
chd: "2",
mysilu: "3",
wiki: "4",
other: "5"
}
}
},
"PTer-offer": {
url: "https://pterclub.com",
host: "pterclub.com",
siteType: "NexusPHP",
icon: "",
asTarget: true,
uploadPath: "/offers.php?add_offer=1"
},
PTer: {
url: "https://pterclub.com",
host: "pterclub.com",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: "#torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr,#body"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: "#guoyu",
diy: "#diy",
cantonese_audio: "#yueyu",
chinese_subtitle: "#zhongzi"
},
category: {
selector: "#browsecat,select[name='type']",
map: {
movie: "401",
tv: "404",
tvPack: "404",
documentary: "402",
concert: "406",
sport: "407",
cartoon: "403",
variety: "405",
music: "406"
}
},
videoType: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "1",
bluray: "2",
remux: "3",
encode: "6",
web: "5",
hdtv: "4",
dvd: "7",
dvdrip: "7",
other: "15"
}
},
area: {
selector: 'select[name="team_sel"]',
map: {
CN: "1",
US: "4",
EU: "4",
HK: "2",
TW: "3",
JP: "6",
KR: "5",
IND: "7",
OT: "8"
}
}
},
"Piggo-offer": {
url: "https://piggo.me",
host: "piggo.me",
siteType: "NexusPHP",
icon: "",
asTarget: true,
uploadPath: "/offers.php?add_offer=1"
},
Piggo: {
url: "https://piggo.me",
host: "piggo.me",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: "#torrenttable>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr,#body"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="pt_gen"]'
},
mediaInfo: {
selector: 'textarea[name="technical_info"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
tags: {
chinese_audio: "#guoyu",
diy: "#diy",
cantonese_audio: "#yueyu",
chinese_subtitle: "#zhongzi"
},
category: {
selector: '#specialcat,select[name="type"]',
map: {
movie: "401",
tv: "404",
tvPack: "404",
documentary: "402",
concert: "406",
sport: "407",
cartoon: "403",
variety: "405",
music: "406"
}
},
videoType: {
selector: 'select[name="source_sel"]',
map: {
uhdbluray: "1",
bluray: "2",
remux: "3",
encode: "6",
web: "5",
hdtv: "4",
dvd: "7",
dvdrip: "7",
other: "15"
}
},
area: {
selector: 'select[name="team_sel"]',
map: {
CN: "1",
US: "4",
EU: "4",
HK: "2",
TW: "3",
JP: "6",
KR: "5",
IND: "7",
OT: "8"
}
}
},
PrivateHD: {
url: "https://privatehd.to",
host: "privatehd.to",
siteType: "AvistaZ",
icon: "",
asSource: true,
asTarget: false,
uploadPath: "/upload.php",
seedDomSelector: "#content-area .block:last table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="/download/torrent"]',
needDoubanInfo: true,
search: {
path: "/browse.php",
params: {
search: "{name}",
"in": "1",
order: "size",
sort: "desc"
}
}
},
PuTao: {
url: "https://pt.sjtu.edu.cn",
host: "sjtu.edu.cn",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "h1~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_url"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: "#torrent"
},
category: {
selector: "#browsecat",
map: {
movie: [
"401",
"402",
"403"
],
tv: [
"407",
"408",
"409",
"410"
],
tvPack: [
"407",
"408",
"409",
"410"
],
documentary: "406",
concert: "427",
cartoon: "431",
app: "434",
sport: "432",
music: [
"420",
"421",
"422"
],
variety: [
"411",
"412",
"413",
"414"
]
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "10",
x264: "1",
x265: "10",
h265: "10",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "9",
bluray: "1",
remux: "3",
encode: "7",
web: "11",
hdtv: "5",
dvd: "6",
dvdrip: "7",
hddvd: "2",
other: "4"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "6",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
area: {
map: {
CN: [
"401",
"409",
"411",
"420"
],
US: [
"402",
"410",
"413",
"422"
],
EU: [
"402",
"410",
"413",
"422"
],
HK: [
"403",
"407",
"412",
"420"
],
TW: [
"403",
"407",
"412",
"420"
],
JP: [
"403",
"408",
"414",
"421"
],
KR: [
"403",
"408",
"414",
"421"
]
}
}
},
R3SUB: {
url: "https://r3sub.com",
host: "r3sub.com",
siteType: "subtitles",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/movie.php",
params: {
id: "{imdb}"
}
}
},
RED: {
url: "https://redacted.sh",
host: "redacted.sh",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
searchstr: "{name}"
}
},
torrent: {
selector: "#file"
}
},
SC: {
url: "https://secret-cinema.pw",
host: "secret-cinema.pw",
siteType: "gazelle",
icon: "",
asSource: false,
asTarget: true,
uploadPath: "/upload.php",
torrent: {
selector: 'input[name="file_input"]'
},
search: {
path: "/torrents.php",
params: {
action: "advanced",
searchsubmit: 1,
filter_cat: 1,
groupname: "{name}",
cataloguenumber: "{imdb}",
order_by: "size",
order_way: "desc",
tags_type: 0
}
}
},
SSD: {
url: "https://springsunday.net",
host: "springsunday.net",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(6)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: "#small_descr"
},
description: {
selector: "#descr"
},
poster: "#url_poster",
imdb: {
selector: "#url"
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
mediaInfo: {
selector: "#Media_BDInfo"
},
screenshots: {
selector: "#url_vimages"
},
category: {
selector: "#browsecat",
map: {
movie: "501",
tv: "502",
tvPack: "502",
documentary: "503",
concert: "507",
sport: "506",
cartoon: "504",
variety: "505",
music: "508"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "2",
hevc: "1",
x264: "2",
x265: "1",
h265: "1",
mpeg2: "4",
mpeg4: "2",
vc1: "3",
xvid: "",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "5",
ac3: "4",
dd: "4",
"dd+": "4",
flac: "7",
dts: "3",
truehd: "2",
lpcm: "6",
dtshdma: "1",
atmos: "3",
dtsx: "3"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "1",
bluray: "1",
remux: "4",
encode: "6",
web: "7",
hdtv: "5",
dvd: "3",
dvdrip: "10",
other: ""
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "1",
"1080p": "2",
"1080i": "3",
"720p": "4",
"576p": "5",
"480p": "5"
}
},
area: {
selector: 'select[name="source_sel"]',
map: {
CN: "1",
US: "9",
EU: "9",
HK: "2",
TW: "2",
JP: "10",
KR: "10",
OT: "3"
}
}
},
SoulVoice: {
url: "https://pt.soulvoice.club",
host: "soulvoice.club",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
needDoubanBookInfo: true,
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "404",
cartoon: "405",
sport: "407",
concert: "406",
variety: "403",
music: "408"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "2",
h265: "2",
x264: "1",
x265: "2",
mpeg2: "5",
mpeg4: "1",
vc1: "5",
xvid: "5"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "3",
"1080p": "1",
"1080i": "2",
"720p": "4",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
hds: "1",
chd: "2",
frds: "3",
cmct: "4",
other: "5"
}
}
},
SpeedApp: {
url: "https://speedapp.io",
host: "speedapp.io",
siteType: "SpeedApp",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "div.row.d-sm-none + div + div",
uploadPath: "/upload",
needDoubanInfo: false,
search: {
path: "/browse",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
search_area: "{optionKey}",
search: "{imdb}"
}
},
name: {
selector: "#name"
},
description: {
selector: "#torrentDescription_releaseInfo"
},
imdb: {
selector: "#url"
},
mediaInfo: {
selector: "#torrentDescription_mediaInfo"
},
bdinfo: {
selector: "#torrentDescription_bdInfo"
},
screenshots: {
selector: "#torrentDescription_screenshots"
}
},
SubHD: {
url: "https://subhd.tv",
host: "subhd.tv",
siteType: "subtitles",
category: [
"subtitles"
],
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/search/{name}"
}
},
TCCF: {
url: "https://et8.org",
host: "et8.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
anonymous: {
selector: 'input[name="uplver"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "622",
tv: "623",
tvPack: "623",
documentary: "404",
concert: "626",
sport: "627",
cartoon: "627",
variety: "627",
app: "625",
ebook: "629",
magazine: "631",
comics: "632",
audioBook: "633",
onlineCourse: "634"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
x264: "7",
hevc: "8",
x265: "6",
h265: "8",
mpeg2: "4",
mpeg4: "1",
vc1: "2",
xvid: "3",
dvd: "4"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "4",
dd: "4",
"dd+": "4",
flac: "1",
dts: "3",
truehd: "9",
lpcm: "10",
dtshdma: "8",
atmos: "8",
dtsx: "3"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "10",
bluray: "1",
remux: "5",
encode: "11",
web: "9",
hdtv: "6",
dvd: "7",
dvdrip: "4",
other: "0"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "5",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
team: {
selector: 'select[name="team_sel"]',
map: {
torrentccf: "1",
tccf: "1",
tlf: "2",
bmdru: "3",
catedu: "4",
madfox: "5",
other: "7"
}
}
},
TJUPT: {
url: "https://www.tjupt.org",
host: "tjupt.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
seedDomSelector: "#top~table:first>tbody>tr:nth-child(5)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
torrent: {
selector: 'input[name="file"]'
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: "#descr"
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: "#external_url"
},
category: {
selector: "#browsecat",
map: {
movie: "401",
tv: "402",
tvPack: "402",
documentary: "411",
concert: "406",
sport: "407",
cartoon: "405",
variety: "403",
music: "406"
}
}
},
TLF: {
url: "https://pt.eastgame.org",
host: "eastgame.org",
siteType: "NexusPHP",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#top~table:first>tbody>tr:nth-child(3)",
torrentDownloadLinkSelector: 'a[href*="download.php?id="]',
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
imdbOptionKey: "4",
nameOptionKey: "0",
params: {
incldead: "0",
search_area: "{optionKey}",
search: "{imdb}",
sort: "5",
type: "desc"
},
result: {
list: ".torrents>tbody>tr",
url: '.torrentname td a[href*="details.php?id="]',
name: '.torrentname td a[href*="details.php?id="]',
size: "td:nth-child(5)"
}
},
name: {
selector: "#name"
},
subtitle: {
selector: 'input[name="small_descr"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="url"][type="text"]'
},
douban: {
selector: 'input[name="douban_url"]'
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: "#browsecat",
map: {
movie: "438",
tv: "440",
tvPack: "440",
documentary: "443",
cartoon: "442",
sport: "444",
concert: "445",
variety: "441",
music: "446"
}
},
videoCodec: {
selector: 'select[name="codec_sel"]',
map: {
h264: "1",
hevc: "6",
h265: "6",
x264: "1",
x265: "6",
mpeg2: "4",
mpeg4: "0",
vc1: "2",
xvid: "3"
}
},
audioCodec: {
selector: 'select[name="audiocodec_sel"]',
map: {
aac: "6",
ac3: "9",
dd: "9",
"dd+": "9",
flac: "1",
dts: "10",
truehd: "14",
lpcm: "12",
dtshdma: "11",
atmos: "13",
dtsx: "10"
}
},
videoType: {
selector: 'select[name="medium_sel"]',
map: {
uhdbluray: "10",
bluray: "1",
remux: "3",
encode: "7",
web: "4",
hdtv: "5",
dvd: "6",
hddvd: "1",
dvdrip: "6",
other: "9"
}
},
resolution: {
selector: 'select[name="standard_sel"]',
map: {
"2160p": "6",
"1080p": "1",
"1080i": "2",
"720p": "3",
"576p": "4",
"480p": "4"
}
},
area: {
selector: 'select[name="processing_sel"]',
map: {
CN: "1",
US: "2",
EU: "2",
HK: "3",
TW: "3",
JP: "4",
KR: "5",
OT: "6"
}
}
},
TMDB: {
url: "https://www.themoviedb.org",
host: "www.themoviedb.org",
siteType: "tmdb",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/search?query={name}"
}
},
TTG: {
url: "https://totheglory.im",
host: "totheglory.im",
siteType: "TTG",
icon: "",
asSource: true,
asTarget: true,
seedDomSelector: "#main_table h1~table:first>tbody>tr:nth-child(2)",
torrentDownloadLinkSelector: 'a[href*="im/dl/"]',
uploadPath: "/upload.php",
search: {
path: "/browse.php",
replaceKey: [
"tt",
"imdb"
],
params: {
search_field: "{imdb}",
sort: "5",
type: "desc",
c: "M"
},
result: {
list: "#torrent_table>tbody>tr",
url: '.name_left a[href*="/t/"]',
name: '.name_left a[href*="/t/"] b',
size: "td:nth-child(7)"
}
},
name: {
selector: 'input[name="name"]'
},
subtitle: {
selector: 'input[name="subtitle"]'
},
description: {
selector: 'textarea[name="descr"]'
},
imdb: {
selector: 'input[name="imdb_c"]'
},
douban: {
selector: 'input[name="douban_id"]'
},
anonymous: {
selector: 'select[name="anonymity"]',
value: "yes"
},
torrent: {
selector: 'input[name="file"]'
},
category: {
selector: 'select[name="type"]',
map: {
movie: [
"51",
"52",
"53",
"54",
"108",
"109"
],
tv: [
"69",
"70",
"73",
"74",
"75",
"76"
],
tvPack: [
"87",
"88",
"99",
"90"
],
documentary: [
"62",
"63",
"67"
],
concert: "59",
sport: "57",
cartoon: "58",
music: "83",
variety: [
"103",
"60",
"101"
]
}
},
videoType: {
map: {
uhdbluray: [
"109"
],
bluray: [
"54",
"109",
"67"
],
remux: [
"53",
"108",
"63",
"70",
"75"
],
encode: [
"53",
"63",
"70",
"75",
"52",
"62",
"69",
"76",
"108"
],
web: [
"53",
"62",
"63",
"70",
"75",
"52",
"69",
"76",
"108",
"87",
"88",
"99",
"90"
],
hdtv: [
"53",
"63",
"70",
"75",
"52",
"62",
"69",
"76",
"108",
"87",
"88",
"99",
"90"
],
dvd: [
"51"
],
dvdrip: [
"51"
],
other: ""
}
},
resolution: {
map: {
"2160p": [
"108",
"109",
"67"
],
"1080p": [
"53",
"63",
"70",
"75",
"54",
"67",
"87",
"88",
"99",
"90"
],
"1080i": [
"53",
"63",
"70",
"75",
"87",
"88",
"99",
"90"
],
"720p": [
"52",
"62",
"69",
"76",
"87",
"88",
"99",
"90"
],
"576p": "51",
"480p": "51"
}
},
area: {
map: {
CN: [
"76",
"75",
"90"
],
US: [
"69",
"70",
"87"
],
EU: [
"69",
"70",
"87"
],
HK: [
"76",
"75",
"90"
],
TW: [
"76",
"75",
"90"
],
JP: [
"73",
"88",
"101"
],
KR: [
"74",
"99",
"103"
],
OT: ""
}
}
},
UHDBits: {
url: "https://uhdbits.org",
host: "uhdbits.org",
siteType: "gazelle",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/upload.php",
search: {
path: "/torrents.php",
params: {
order_way: "desc",
order_by: "size",
searchstr: "{imdb}"
}
},
needDoubanInfo: true,
description: {
selector: "#release_desc"
},
imdb: {
selector: "#imdbid"
},
anonymous: {
selector: "#anonymous"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
torrent: {
selector: "#file"
},
category: {
selector: "#categories",
map: {
movie: "0",
tv: "2",
tvPack: "2",
music: "1"
}
},
videoType: {
selector: "#media",
map: {
uhdbluray: "Blu-ray",
bluray: "Blu-ray",
remux: "Remux",
encode: "Encode",
web: "WEB-DL",
hdtv: "HDTV",
dvdrip: "Encode",
other: "Others"
}
},
videoCodec: {
selector: "#codec",
map: {
h264: "AVC/H.264",
hevc: "HEVC",
x264: "x264",
x265: "x265",
h265: "HEVC",
mpeg2: "MPEG-2",
mpeg4: "AVC/H.264",
vc1: "VC-1",
dvd: "MPEG"
}
},
resolution: {
selector: "#format",
map: {
"2160p": "2160p",
"1080p": "1080p",
"1080i": "1080i",
"720p": "720p",
"576p": "Others",
"480p": "Others"
}
}
},
ZHUQUE: {
url: "https://zhuque.in",
host: "zhuque.in",
siteType: "TNode",
icon: "",
asSource: false,
asTarget: true,
seedDomSelector: "div.layout-container > div > form",
uploadPath: "/torrent/upload",
needDoubanInfo: false,
search: {
path: "/torrent/search",
imdbOptionKey: "imdbid",
nameOptionKey: "title",
params: {
type: "{optionKey}",
keyword: "{imdb}",
sorter: "seeding",
order: "desc"
}
},
name: {
selector: "#form_item_title"
},
subtitle: {
selector: "#form_item_subtitle"
},
description: {
selector: "#form_item_note"
},
anonymous: {
selector: "#form_item_anonymous"
},
torrent: {
selector: "#form_item_torrent"
},
imdb: {
selector: 'input[placeholder="tt123456"]'
},
douban: {
selector: "#form_item_doubanid"
},
tmdb: {
selector: ".ant-space.ant-space-horizontal.ant-space-align-center >.ant-space-item:last-child > input"
},
screenshots: {
selector: "#form_item_screenshot"
},
mediaInfo: {
selector: "#form_item_mediainfo"
},
tags: {
chinese_audio: 'input.ant-checkbox-input[value="603"]',
chinese_subtitle: 'input.ant-checkbox-input[value="604"]',
hdr10: 'input.ant-checkbox-input[value="613"]',
dolby_vision: 'input.ant-checkbox-input[value="611"]'
},
category: {
selector: "#form_item_category",
map: {
movie: "电影",
tv: "电视剧",
tvPack: "电视剧",
cartoon: "动画",
concert: "其它",
documentary: "其它",
variety: "其它"
}
},
videoType: {
selector: "#rc_select_9",
map: {
uhdbluray: "UHD Blu-ray",
bluray: "Blu-ray",
remux: "Remux",
hdtv: "HDTV",
web: "WEB-DL",
webrip: "Encode",
encode: "Encode"
}
},
videoCodec: {
selector: "#rc_select_10",
map: {
h264: "H264",
hevc: "H265",
x264: "x264",
x265: "x265",
h265: "H265",
mpeg2: "Other",
mpeg4: "H265",
vc1: "Other",
xvid: "Other",
dvd: "Other"
}
},
audioCodec: {
selector: "#rc_select_3",
map: {
aac: "AAC",
ac3: "AC3",
dd: "AC3",
"dd+": "DDP",
dts: "DTS",
truehd: "TrueHD",
lpcm: "LPCM",
flac: "FLAC",
dtshdma: "DTS-HD MA",
atmos: "TrueHD Atmos",
dtsx: "DTS-X",
mp3: "Other"
}
},
resolution: {
selector: "#rc_select_12"
}
},
fearnopeer: {
url: "https://fearnopeer.com",
host: "fearnopeer.com",
siteType: "UNIT3D",
icon: "",
asSource: true,
asTarget: true,
uploadPath: "/torrents/create?category_id=1",
needDoubanInfo: true,
seedDomSelector: ".torrent__buttons+.panelV2",
torrentDownloadLinkSelector: 'a[href*="/torrents/download/"]',
search: {
path: "/torrents",
replaceKey: [
"tt",
""
],
params: {
name: "{name}",
imdbId: "{imdb}",
sortField: "size"
}
},
name: {
selector: "#title"
},
description: {
selector: "#bbcode-description"
},
imdb: {
selector: "#autoimdb"
},
tmdb: {
selector: "#autotmdb"
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
anonymous: {
selector: '.form__group input[type="checkbox"][name="anon"]'
},
torrent: {
selector: 'input[type="file"][accept=".torrent"]'
},
category: {
selector: "#browsecat",
map: {
movie: "1",
tv: "2",
tvPack: "2"
}
},
videoType: {
selector: "#autotype",
map: {
uhdbluray: "1",
bluray: "1",
remux: "3",
encode: "12",
web: "4",
hdtv: "6",
dvd: "1",
dvdrip: "12",
other: ""
}
},
resolution: {
selector: "#autores",
map: {
"4320p": "11",
"2160p": "1",
"1080p": "2",
"1080i": "3",
"720p": "5",
"576p": "6",
"480p": "8"
}
}
},
iTS: {
url: "http://shadowthein.net",
host: "shadowthein.net",
siteType: "its",
asSource: false,
asTarget: true,
icon: "",
seedDomSelector: "h1~.line>tbody>tr:nth-child(2)",
needDoubanInfo: true,
uploadPath: "/upload.php",
search: {
path: "/browse.php",
params: {
incldead: 1,
search: "{imdb}",
search_in: "names",
sort: 5,
type: "desc"
}
},
name: {
selector: 'input[name="name"]'
},
imdb: {
selector: 'input[name="imdblink"]'
},
description: {
selector: 'textarea[name="descr"]'
},
mediaInfo: {
selector: 'textarea[name="mediainfo"]'
},
tags: {
hdr: 'input[name="HDR10"]',
hdr10_plus: 'input[name="HDR10Plus"]',
dolby_vision: 'input[name="DolbyVision"]'
},
anonymous: {
selector: 'input[name="anonymous"][value="true"]'
},
category: {
selector: 'select[name="type"]',
map: {
movie: "68",
tv: "65",
concert: "61",
music: "6",
ebook: "26"
}
}
},
"nzbs.in": {
url: "https://nzbs.in",
host: "nzbs.in",
siteType: "nzb",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/search",
params: {
query: "{name}"
}
}
},
zimuku: {
url: "http://zimuku.org",
host: "zimuku.org",
siteType: "subtitles",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/search",
params: {
q: "{imdb}"
}
}
},
"豆瓣电影": {
url: "https://search.douban.com",
host: "search.douban.com",
siteType: "douban",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/movie/subject_search",
params: {
search_text: "{imdb}"
}
}
},
"豆瓣读书": {
url: "https://search.douban.com",
host: "search.douban.com",
siteType: "doubanBook",
icon: "",
asSource: false,
asTarget: false,
search: {
path: "/book/subject_search?search_text={name}"
}
}
};
const BROWSER_LANGUAGE = navigator.language.toLowerCase().split("-")[0];
const getSiteName = (host) => {
let siteName = "";
try {
Object.keys(PT_SITE).forEach((key) => {
const siteKey = key;
const hostName = PT_SITE[siteKey].host;
const matchReg = new RegExp(hostName, "i");
if (hostName && host.match(matchReg)) {
siteName = siteKey;
}
});
return siteName;
} catch (error) {
if (error.message !== "end loop") {
console.log(error);
}
return "";
}
};
const CHINESE_REGEX = /[\u4e00-\u9fa5]+/;
const getSortedSiteKeys = () => {
return Object.keys(PT_SITE).sort((a2, b) => {
const aHasChinese = CHINESE_REGEX.test(a2);
const bHasChinese = CHINESE_REGEX.test(b);
if (aHasChinese && !bHasChinese) return 1;
if (!aHasChinese && bHasChinese) return -1;
return a2.toLowerCase().localeCompare(b.toLowerCase());
});
};
const SORTED_SITE_KEYS = getSortedSiteKeys();
const CURRENT_SITE_NAME = getSiteName(location.host);
const CURRENT_SITE_INFO = CURRENT_SITE_NAME ? PT_SITE[CURRENT_SITE_NAME] : void 0;
const CONFIG$4 = {
URLS: {
IMGBOX: "https://imgbox.com",
PTPIMG: "https://ptpimg.me",
HDB_UPLOAD: "https://img.hdbits.org/upload_api.php",
IMGBOX_UPLOAD: "https://imgbox.com/upload/process",
PIXHOST_UPLOAD: "https://pixhost.to/remote/",
PTPIMG_UPLOAD: "https://ptpimg.me/upload.php",
IMGBB: "https://imgbb.com/json"
},
ERROR_MESSAGES: {
UPLOAD_FAILED: "上传失败,请重试",
PTPIMG_UPLOAD_FAILED: "ptpimg上传失败",
NO_API_KEY: "请到配置面板中填入ptpimg的api_key",
NO_TRANSFER_NEEDED: "无需转存"
},
FILTER_IMGS_NAMES: [
"poster.jpg",
"2019/01/04/info.png",
"MoreScreens",
"PTer.png",
"ms.png",
"trans.gif",
"PTerREMUX.png",
"PTerWEB.png",
"CS.png",
"Ourbits_info",
"GDJT",
"douban",
"logo",
"2019/03/28/5c9cb8f8216d7.png",
"_front",
"info_01.png",
"screens.png",
"04/6b/Ggp5ReQb_o",
"ce/e7/KCmGFMOB_o"
]
};
const en$1 = {
"豆瓣链接获取失败": "Failed to retrieve Douban link",
"豆瓣ID获取失败": "Failed to retrieve Douban ID",
"获取豆瓣信息失败": "Failed to retrieve Douban information",
"缺少IMDB信息": "IMDB information missing",
"获取失败": "Retrieval failed",
"获取成功": "Retrieved successfully",
"请求失败": "Request failed",
"上传失败,请重试": "Upload failed, please try again",
"封面上传失败": "Poster upload failed",
"ptpimg上传失败": "PtpImg upload failed",
"请到配置面板中填入ptpimg的api_key": "Please enter your PtpImg API key in the settings panel",
"数据加载中...": "Loading data...",
"获取图片列表失败": "Failed to retrieve image list",
"转换中...": "Converting...",
"转换成功!": "Conversion successful!",
"获取中...": "Retrieving...",
"缺少豆瓣链接": "Douban link missing",
"本种子可能禁止转载,确定要继续转载么?": "This torrent may be prohibited from transfer. Are you sure you want to continue?",
"手动输入豆瓣链接": "Enter Douban link manually",
"获取豆瓣简介": "Retrieve Douban info",
"获取豆瓣读书简介": "Retrieve Douban Book info",
"转缩略图": "Convert to thumbnails",
"快速检索": "Quick search",
"一键群转": "Batch transfer",
"快捷操作": "Quick actions",
"一键转种": "One-click transfer",
"转种站点启用": "Enable sites for transfer",
"批量转种启用": "Enable sites for batch transfer",
"一键批量转发到以下选中的站点": "One-click batch transfer to selected sites below",
"站点搜索启用": "Enable sites for search",
"图床配置": "Image hosting settings",
"如何获取?": "How to obtain?",
"额外功能关闭": "Disable extra features",
"关闭转缩略图功能": "Disable thumbnail conversion",
"关闭站点图标显示": "Hide site icons",
"关闭转存图片功能": "Disable image rehosting",
"保存": "Save",
"取消": "Cancel",
"错误": "Error",
"成功": "Success",
"保存本地站点设置失败": "Failed to save local site settings",
"请先设置群转列表": "Please configure batch transfer list first",
"转种页面已打开,请前往对应页面操作": "Transfer pages opened. Please navigate to the corresponding pages to proceed",
"提示": "Tip",
"转存截图": "Rehost screenshots",
"无需转存": "No upload needed",
"上传中,请稍候...": "Uploading, please wait...",
"不显示致谢内容": "Hide acknowledgments",
"拷贝": "Copy",
"已拷贝": "Copied",
"不显示豆瓣按钮和豆瓣链接": "Hide Douban button and link field",
"请填写正确链接": "Please enter a valid link",
"批量检索": "Batch search",
"同时打开多个搜索标签页": "Open multiple search tabs simultaneously",
"豆瓣配置": "Douban configuration",
"关闭快速检索": "Disable quick search",
"种子文件下载失败": "Torrent file download failed",
"请手动下载": "Please download manually",
"获取页面配置": "Retrieve page configuration",
"配置已复制到剪贴板,请黏贴到创建的Github Issue中": "Configuration copied to clipboard. Please paste it into a new GitHub issue"
};
const ko = {
"豆瓣链接获取失败": "더우반 링크 얻기 실패",
"豆瓣ID获取失败": "더우반 ID 아이디 얻기 실패",
"获取豆瓣信息失败": "더우반 데이터 얻기 실패",
"缺少IMDB信息": "누락된 IMDB 정보",
"获取失败": "요청 실패",
"获取成功": "데이터 요청 성공",
"请求失败": "요청 실패",
"上传失败,请重试": "업로드 실패, 다시 시도하세요.",
"ptpimg上传失败": "PtpImg 업로드 실패",
"请到配置面板中填入ptpimg的api_key": "설정 패널에 ptpimg의 API_KEY를 입력하세요.",
"封面上传失败": "포스터 업로드 실패",
"数据加载中...": "데이터 불러오기 중...",
"获取图片列表失败": "이미지 목록 얻기 실패",
"转换中...": "변환 중...",
"转换成功!": "변환됨!",
"获取中...": "요청 중...",
"缺少豆瓣链接": "더우반 링크 누락",
"本种子可能禁止转载,确定要继续转载么?": "이 토렌트의 전송이 금지될 수 있습니다, 계속하겠습니까?",
"手动输入豆瓣链接": "더우반 링크 입력",
"获取豆瓣简介": "더우반의 데이터 얻기",
"获取豆瓣读书简介": "더우반 도서 데이터 얻기",
"转缩略图": "썸네일로 변환",
"快速检索": "빠른 검색",
"一键群转": "일괄 전송",
"快捷操作": "빠른 작업",
"一键转种": "전송 대상",
"转种站点启用": "전송 대상 섹션의 사이트 선택",
"批量转种启用": "일괄 전송 버튼을 사용할 사이트 선택",
"一键批量转发到以下选中的站点": "아래에서 선택한 사이트로 원클릭 일괄 전송하기",
"站点搜索启用": "빠른 검색 섹션에서 사이트 선택",
"图床配置": "이미지 호스트 설정",
"如何获取?": "어떻게 얻나요?",
"额外功能关闭": "추가 기능 끄기",
"关闭转缩略图功能": "썸네일로 변환 버튼 제거하기",
"关闭站点图标显示": "아이콘 제거하기",
"关闭转存图片功能": "이미지 리호스팅 기능 끄기",
"保存": "저장",
"取消": "취소",
"错误": "오류",
"成功": "성공",
"保存本地站点设置失败": "로컬 사이트 설정 저장 실패",
"请先设置群转列表": "먼저 일괄 전송 목록 설정",
"转种页面已打开,请前往对应页面操作": "전송 페이지가 열렸고, 해당 페이지로 이동하여 작업",
"提示": "힌트",
"转存截图": "다른 호스트에 스크린샷 업로드",
"无需转存": "업로드할 필요 없음",
"上传中,请稍候...": "업로드 중, 잠시만 기다려주세요.",
"不显示致谢内容": "감사 내용을 포함하지 않음",
"拷贝": "복사",
"已拷贝": "복사됨",
"不显示豆瓣按钮和豆瓣链接": "더우반 버튼 및 링크 필드 숨기기",
"请填写正确链接": "올바른 링크를 입력하세요.",
"批量检索": "일괄 검색",
"同时打开多个搜索标签页": "여러 검색 탭 동시에 열기",
"豆瓣配置": "더우반 구성",
"关闭快速检索": "빠른검색 비활성화",
"种子文件下载失败": "토렌트 파일 다운로드 실패",
"请手动下载": "수동으로 다운로드해주세요",
"配置已复制到剪贴板,请黏贴到创建的Github Issue中": "설정이 클립보드에 복사되었습니다. 생성한 Github Issue에 붙여넣으세요.",
"获取页面配置": "페이지 설정 가져오기"
};
const i18nConfig = {
en: en$1,
ko
};
class HTMLToBBCodeConverter {
constructor(siteInfo) {
this.convertRgbToHex = (rgb) => {
const rgbMatch = rgb == null ? void 0 : rgb.match(
/^rgba?\s*\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*/i
);
if (!rgbMatch || rgbMatch.length < 4) {
return "";
}
const hexComponents = rgbMatch.slice(1, 4).map((component) => {
return `0${parseInt(component, 10).toString(16)}`.slice(-2);
});
return `#${hexComponents.join("")}`;
};
this.siteInfo = siteInfo;
}
/**
* Check if the site matches the given site match
*
* @private
* @param {SiteMatch} siteMatch
* @returns {boolean}
*/
isMatchingSite(siteMatch) {
const { siteName } = this.siteInfo;
if (typeof siteMatch === "string") {
return siteName === siteMatch;
}
return !!siteName.match(siteMatch);
}
/**
* Check if the text is a special control text (should be ignored)
*
* @private
* @param {string} text
* @returns {boolean}
*/
isSpecialControlText(text2) {
return !!text2.trim().match(
/^(引用|Quote|代码|代碼|Show|Hide|Hidden text|Hidden content|\[show\]|\[Show\])/
);
}
/**
* Normalize the color format
*
* @private
* @param {string} color
* @returns {string}
*/
normalizeColorFormat(color) {
return /rgba?/.test(color) ? this.convertRgbToHex(color) : color;
}
/**
* Add BBCode tags to the accumulator
*
* @private
* @param {BBCodeTagsAccumulator} accumulator
* @param {string | null} openTag
* @param {string | null
* @returns {void}
*/
addBBCodeTags(accumulator, openTag, closeTag = null) {
if (openTag !== null) accumulator.openingTags.push(openTag);
if (closeTag !== null) accumulator.closingTags.unshift(closeTag);
}
/**
* Convert image element to BBCode
*
* @private
* @param {HTMLImageElement} imgElement
* @returns {string}
*/
convertImageToBBCode(imgElement) {
const { src, title } = imgElement;
const dataSrc = imgElement.getAttribute("data-src") || imgElement.getAttribute("data-echo");
if (title === ":m:") {
return ":m:";
}
const imgUrl = this.determineImageUrl(src, dataSrc);
if (!imgUrl) return "";
return `[img]${imgUrl}[/img]`;
}
/**
* Determine the image URL
*
* @private
* @param {string} src
* @param {string | null} dataSrc
* @returns {string}
*/
determineImageUrl(src, dataSrc) {
if (dataSrc) {
return dataSrc.match(/(http(s)?:)?\/\//) ? dataSrc : `${location.origin}/${dataSrc}`;
}
if (src && !src.match(/ico_\w+.gif|jinzhuan|thumbsup|kralimarko/)) {
return src;
}
return "";
}
/**
* Convert anchor element to BBCode
*
* @private
* @param {HTMLAnchorElement} anchorElement
* @returns {BBCodeConverterResult}
*/
convertLinkToBBCode(anchorElement) {
const { href, textContent } = anchorElement;
if (!href || href.length === 0) {
return {};
}
if (this.isMatchingSite("HDSpace")) {
const div = anchorElement.querySelector("div");
if (div && div.id) {
const img = div.querySelector("img");
if (img) {
const imgUrl = img.getAttribute("src");
return {
content: `[url=${href}][img]${imgUrl}[/img][/url]`
};
}
}
}
if (href.match(/javascript:void/) || textContent === "show" && this.isMatchingSite("HDT")) {
return {};
}
return {
openTag: `[url=${href}]`,
closeTag: "[/url]"
};
}
/**
* handle special cases for div element
*
* @private
* @param {HTMLElement} element
* @param {BBCodeTagsAccumulator} accumulator
* @returns {(string | null)}
*/
handleSpecialDivElement(element, accumulator) {
var _a2, _b;
const { className, id } = element;
const { siteType } = this.siteInfo;
const addTags = this.addBBCodeTags.bind(this, accumulator);
if (className === "codemain") {
if (element.children.length > 0) {
addTags("\n[quote]", "[/quote]");
return null;
}
return "";
}
if (className === "hidden" && this.isMatchingSite("HDT")) {
addTags("\n[quote]", "[/quote]");
return null;
}
if (className.match("spoiler") && this.isMatchingSite("KG")) {
if (className === "spoiler-content") {
addTags("\n[quote]", "[/quote]");
} else if (className === "spoiler-header") {
return "";
}
return null;
}
if (this.isMatchingSite("BeyondHD")) {
if (className === "spoilerChild") {
const firstChildTag = (_b = (_a2 = element.children[0]) == null ? void 0 : _a2.tagName) == null ? void 0 : _b.toUpperCase();
if (firstChildTag === "BLOCKQUOTE" || firstChildTag === "PRE") {
addTags("\n", "");
} else {
addTags("\n[quote]", "[/quote]");
}
} else if (id === "screenMain") {
return "\n";
} else if (className === "spoilerHide") {
return "";
}
return null;
}
if (siteType === "AvistaZ") {
if (className === "spoiler-text") {
addTags("\n[quote]", "[/quote]");
return null;
}
if (className === "spoiler-toggle") {
return "";
}
}
if (className.match(/codetop|highlight/)) {
return "";
}
addTags("\n", "\n");
return null;
}
/**
* handle special cases for TD element
*
* @private
* @param {BBCodeTagsAccumulator} accumulator
* @returns {(string | null)}
*/
handleTDElement(accumulator) {
const { siteType } = this.siteInfo;
const addTags = this.addBBCodeTags.bind(this, accumulator);
if (this.isMatchingSite(/^(TTG|HDBits|KG|HDSpace)/) || this.isMatchingSite("HDT") || siteType === "UNIT3D") {
addTags("[quote]", "[/quote]");
} else if (this.isMatchingSite("EMP")) {
addTags("");
} else if (this.isMatchingSite("PTer")) {
addTags(null, null);
} else {
return "";
}
return null;
}
/**
* handle BR element
*
* @private
* @param {BBCodeTagsAccumulator} accumulator
* @returns {void}
*/
handleBRElement(accumulator) {
const { siteType } = this.siteInfo;
const siteName = this.siteInfo.siteName;
const addTags = this.addBBCodeTags.bind(this, accumulator);
if (siteType === "NexusPHP" && siteName !== "OurBits" || (siteName == null ? void 0 : siteName.match(/^(UHDBits|HDBits|BTN)/))) {
addTags("");
} else {
addTags("\n");
}
}
/**
* handle inline styles
*
* @private
* @param {HTMLElement} element
* @param {BBCodeTagsAccumulator} accumulator
* @returns {void}
*/
handleInlineStyles(element, accumulator) {
const { style } = element;
const addTags = this.addBBCodeTags.bind(this, accumulator);
if (style.textAlign) {
const alignment = style.textAlign.toUpperCase();
if (alignment === "LEFT") addTags("[left]", "[/left]");
else if (alignment === "RIGHT") addTags("[right]", "[/right]");
else if (alignment === "CENTER") addTags("[center]", "[/center]");
}
if (style.fontWeight === "bold" || parseInt(style.fontWeight, 10) >= 600) {
addTags("[b]", "[/b]");
}
if (style.fontStyle === "italic") {
addTags("[i]", "[/i]");
}
if (style.textDecoration === "underline" || style.textDecoration === "overline") {
addTags("[u]", "[/u]");
}
if (style.textDecoration === "line-through") {
addTags("[s]", "[/s]");
}
if (style.color && style.color.trim() !== "") {
addTags(`[color=${this.normalizeColorFormat(style.color)}]`, "[/color]");
}
}
/**
* handle element by tag name
*
* @private
* @param {string} tagName
* @param {HTMLElement} element
* @param {BBCodeTagsAccumulator} accumulator
* @returns {(string | null)}
*/
handleElementByTagName(tagName, element, accumulator) {
var _a2, _b, _c;
const addTags = this.addBBCodeTags.bind(this, accumulator);
const { siteType } = this.siteInfo;
switch (tagName) {
case "SCRIPT":
return "";
case "UL":
addTags(null, null);
break;
case "OL":
addTags("[list=1]", "[/list]");
break;
case "LI": {
const { className } = element;
if (siteType === "UNIT3D" && className) {
return `[quote]${(_a2 = element.textContent) == null ? void 0 : _a2.trim()}[/quote]`;
}
addTags("[*]", "\n");
break;
}
case "B":
addTags("[b]", "[/b]");
break;
case "U":
addTags("[u]", "[/u]");
break;
case "I":
addTags("[i]", "[/i]");
break;
case "DIV":
return this.handleSpecialDivElement(element, accumulator);
case "P":
addTags("\n");
break;
case "BR":
this.handleBRElement(accumulator);
break;
case "SPAN": {
const { className } = element;
if (className == null ? void 0 : className.match(/size/)) {
const matchSize = (_c = (_b = className.match(/size(\d)/)) == null ? void 0 : _b[1]) != null ? _c : "";
if (matchSize) {
addTags(`[size=${matchSize}]`, "[/size]");
}
} else {
addTags(null, null);
}
break;
}
case "BLOCKQUOTE":
case "PRE":
case "FIELDSET":
addTags("[quote]", "[/quote]");
break;
case "CENTER":
addTags("[center]", "[/center]");
break;
case "TD":
return this.handleTDElement(accumulator);
case "IMG":
return this.convertImageToBBCode(element);
case "FONT": {
const fontElement = element;
if (fontElement.color) {
addTags(
`[color=${this.normalizeColorFormat(fontElement.color)}]`,
"[/color]"
);
}
if (fontElement.size) {
addTags(`[size=${fontElement.size}]`, "[/size]");
}
break;
}
case "A": {
const result = this.convertLinkToBBCode(element);
if (result.content) return result.content;
if (result.openTag) addTags(result.openTag, result.closeTag);
if (Object.keys(result).length < 1) return "";
break;
}
case "H1":
addTags('[b][size="7"]', "[/size][/b]\n");
break;
case "H2":
addTags('[b][size="6"]', "[/size][/b]\n");
break;
case "H3":
addTags('[b][size="5"]', "[/size][/b]\n");
break;
case "H4":
addTags('[b][size="4"]', "[/size][/b]\n");
break;
case "STRONG":
addTags("[b]", "[/b]");
break;
case "TABLE": {
if (this.isMatchingSite("PTer") && element.className === "table") {
return "";
}
addTags("");
break;
}
case "TH":
addTags("");
break;
default:
addTags(null, null);
}
return null;
}
/**
* Convert HTML node to BBCode
*
* @param {Element | ChildNode} node
* @returns {string}
*/
convert(node) {
if (node.nodeType === 3) {
const textContent = node.textContent;
if (this.isSpecialControlText(textContent)) {
return "";
}
return textContent;
}
if (node.nodeType !== 1) {
return "";
}
const element = node;
const tagName = element.tagName.toUpperCase();
const accumulator = {
openingTags: [],
closingTags: []
};
const specialTagResult = this.handleElementByTagName(
tagName,
element,
accumulator
);
if (specialTagResult !== null) {
return specialTagResult;
}
this.handleInlineStyles(element, accumulator);
const bbCodeParts = [];
element.childNodes.forEach((childNode) => {
const childCode = this.convert(childNode);
if (childCode) {
bbCodeParts.push(childCode);
}
});
return accumulator.openingTags.concat(bbCodeParts).concat(accumulator.closingTags).join("");
}
}
class NetworkError extends Error {
constructor(message, status, statusText) {
super(message);
this.name = "NetworkError";
this.status = status;
this.statusText = statusText;
}
}
class TimeoutError extends Error {
constructor(message = "Request timed out") {
super(message);
this.name = "TimeoutError";
}
}
const getLocationSearchValueByKey = (key) => {
const reg = new RegExp(`(^|&)${key}=([^&]*)(&|$)`);
const regArray = window.location.search.substring(1).match(reg);
if (regArray) {
return decodeURIComponent(regArray[2]);
}
return "";
};
const convertSizeStringToBytes = (size) => {
if (!size) {
return 0;
}
const sizeFloat = parseFloat(size);
if (isNaN(sizeFloat)) {
return 0;
}
if (size.match(/bytes/)) {
return sizeFloat;
} else if (size.match(/T/i)) {
return sizeFloat * 1024 ** 4;
} else if (size.match(/G/i)) {
return sizeFloat * 1024 ** 3;
} else if (size.match(/M/i)) {
return sizeFloat * 1024 ** 2;
} else if (size.match(/K/i)) {
return sizeFloat * 1024;
}
return sizeFloat;
};
const $t$1 = (key) => {
const languageKey = BROWSER_LANGUAGE;
const translations = i18nConfig[languageKey];
if (!translations) {
return key;
}
return key in translations ? translations[key] : key;
};
const htmlToBBCode = (node) => {
const converter = new HTMLToBBCodeConverter({
siteType: CURRENT_SITE_INFO.siteType,
siteName: CURRENT_SITE_NAME
});
return converter.convert(node);
};
const createFormData = (fields, files) => {
const formData = new FormData();
Object.entries(fields).forEach(([key, value]) => {
if (value instanceof File) {
throw new Error("Files should be passed as a separate argument");
}
formData.append(key, String(value));
});
if (files) {
files.forEach(({ fieldName, file }) => {
if (Array.isArray(file)) {
file.forEach((f2, index) => {
formData.append(`${fieldName}[${index}]`, f2);
});
} else {
formData.append(fieldName, file);
}
});
}
return formData;
};
const GMFetch = (url, options2) => {
return new Promise((resolve28, reject) => {
const finalOptions = __spreadValues({
method: "GET"
}, options2);
GM_xmlhttpRequest(__spreadProps(__spreadValues({
url
}, finalOptions), {
onload: (res) => {
const { statusText, status, response, responseText } = res;
if (status >= 200 && status < 300) {
if (finalOptions.responseType === "json" && typeof response === "undefined" && responseText) {
try {
resolve28(JSON.parse(responseText));
} catch (e2) {
reject(
new Error(
`Failed to parse JSON: ${e2 instanceof Error ? e2.message : String(e2)}`
)
);
}
} else {
resolve28(response);
}
} else {
reject(
new NetworkError(
statusText || `Request failed with status ${status}`,
status
)
);
}
},
ontimeout: () => {
reject(
new TimeoutError(
`Request to ${url} timed out after ${finalOptions.timeout}ms`
)
);
},
onprogress: finalOptions == null ? void 0 : finalOptions.onprogress,
onerror: (error) => {
const errorMessage = error.error || error.statusText || "Unknown network error";
reject(new NetworkError(errorMessage, error.status, error.statusText));
}
}));
});
};
const getOriginalImgUrl = async (urlBBcode) => {
var _a2, _b, _c;
if (!urlBBcode) {
throw new Error("Invalid BBCode - No BBCode found");
}
const urlBBCodeMatch = urlBBcode.match(/\[url=(http(s)*:([^\]]+)?)/);
const directUrlMatch = urlBBcode.match(/http(s)*:([^[])+/);
if (urlBBCodeMatch) {
const imgUrl = urlBBCodeMatch[1];
for (const strategy of URLStrategies) {
if (strategy.matches(imgUrl, urlBBcode)) {
return await strategy.transform(imgUrl, urlBBcode);
}
}
return imgUrl.trim();
} else if (urlBBcode.match(/\[img\]/)) {
return (_c = (_b = (_a2 = urlBBcode.match(/img\](([^[])+)/)) == null ? void 0 : _a2[1]) == null ? void 0 : _b.trim()) != null ? _c : "";
} else if (directUrlMatch) {
return directUrlMatch[0].trim();
}
throw new Error("Invalid BBCode - No valid image URL found");
};
const urlToFile = async (url) => {
var _a2, _b;
const filename = (_b = (_a2 = url.match(/\/([^/]+?)(\?|$)/)) == null ? void 0 : _a2[1]) != null ? _b : "filename";
const data = await GMFetch(url, {
responseType: "blob"
});
const file = new File([data], filename, { type: data.type });
return file;
};
class ImageUploadError extends Error {
constructor(message, originalError) {
super(message);
this.originalError = originalError;
this.name = "ImageUploadError";
}
}
const throwUploadError = (message, originalError) => {
const errorMessage = message || $t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED);
throw new ImageUploadError(errorMessage, originalError);
};
const urlFileCache = /* @__PURE__ */ new Map();
const cachedUrlToFile = async (url) => {
if (!url) {
throw new Error("URL is required to convert to file");
}
if (!urlFileCache.has(url)) {
urlFileCache.set(url, urlToFile(url));
}
return urlFileCache.get(url);
};
class HdBitsStrategy {
matches(url) {
return url.includes("img.hdbits.org");
}
async transform(url) {
const data = await GMFetch(url);
const doc = new DOMParser().parseFromString(data, "text/html");
const imgElem = doc.querySelector("#viewimage");
if (!imgElem) {
throw new Error(
"Couldn't find image element when retrieving from HDBits"
);
}
const imgSrc = imgElem.getAttribute("src");
if (!imgSrc) {
throw new Error(
"No valid image source found when retrieving from HDBits"
);
}
return imgSrc;
}
}
class PterClubStrategy {
matches(url, bbCode) {
return bbCode.includes("img.pterclub.com");
}
async transform(url, bbCode) {
var _a2, _b;
const imgUrl = (_b = (_a2 = bbCode.match(/img\](([^[])+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!imgUrl) {
throw new Error("Invalid PterClub image URL");
}
return imgUrl.replace(/\.th/g, "");
}
}
class ImageBoxStrategy {
matches(url, bbCode) {
return bbCode.includes("imgbox.com");
}
async transform(url, bbCode) {
var _a2, _b;
const imgUrl = (_b = (_a2 = bbCode.match(/img\](([^[])+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!imgUrl) {
throw new Error("Invalid ImageBox image BBCode");
}
if (!imgUrl.match(/thumbs\d.+?_t\.(png|jpg)/)) {
throw new Error("Invalid ImageBox image URL");
}
return imgUrl.replace(/thumbs(\d)/, "images$1").replace(/_t(\.(png|jpg))/, "_o.$2");
}
}
class ImageBamStrategy {
matches(url) {
return url.includes("imagebam.com");
}
async transform(url) {
const originalPage = await GMFetch(url);
const doc = new DOMParser().parseFromString(originalPage, "text/html");
const imgElem = doc.querySelector(".main-image");
if (!imgElem) {
throw new Error(
"Couldn't find image element when retrieving from ImageBam"
);
}
const imgSrc = imgElem.getAttribute("src");
if (!imgSrc) {
throw new Error(
"No valid image source found when retrieving from ImageBam"
);
}
return imgSrc;
}
}
class BeyondHdStrategy {
matches(url, bbCode) {
return bbCode.includes("beyondhd.co");
}
async transform(url, bbCode) {
var _a2, _b;
const imgUrl = (_b = (_a2 = bbCode.match(/img\](([^[])+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!imgUrl) {
throw new Error("Invalid BeyondHD image BBCode");
}
return imgUrl.replace(/\.(th|md)\.(png|jpg|gif)/, ".$2").replace(/cache\/t\//, "cache/i/");
}
}
class PixHostStrategy {
matches(url, bbCode) {
return bbCode.includes("pixhost.to");
}
async transform(url, bbCode) {
var _a2;
const hostNumber = (_a2 = bbCode.match(/img\]https:\/\/t(\d+)\./)) == null ? void 0 : _a2[1];
if (!hostNumber || !url.includes("show")) {
throw new Error("Invalid PixHost image BBCode");
}
return url.replace(/(pixhost\.to)\/show/, `img${hostNumber}.$1/images`);
}
}
const URLStrategies = [
new HdBitsStrategy(),
new PterClubStrategy(),
new ImageBoxStrategy(),
new ImageBamStrategy(),
new BeyondHdStrategy(),
new PixHostStrategy()
];
const withUploadErrorHandling = async (uploadFn, serviceName, options2 = {}) => {
const { validateFirstArg = true, defaultResult = [] } = options2;
return async (...args) => {
try {
if (validateFirstArg) {
const firstArg = args[0];
if (firstArg === void 0 || firstArg === null || Array.isArray(firstArg) && firstArg.length === 0) {
return defaultResult;
}
}
return await uploadFn(...args);
} catch (error) {
if (error instanceof ImageUploadError) {
throw error;
}
const errorMessage = error instanceof Error ? `${error.name}: ${error.message}` : "Unknown error";
throwUploadError(
`${$t$1(
CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED
)} - ${serviceName}. ${errorMessage}`,
error
);
return defaultResult;
}
};
};
const getImageBBCodeMatches = (bbcode) => {
const matches = bbcode.match(
/(\[url=(http(s)*:\/{2}.+?)\])?\[img\](http(s)?:.+?)\[\/img](\[\/url\])?/gi
);
if (!matches) {
return [];
}
return Array.from(matches);
};
const getImgInfoFromBBCode = async (bbcode) => {
var _a2, _b, _c;
if (!bbcode) {
throw new Error("Invalid BBCode - No BBCode found");
}
const originalUrl = await getOriginalImgUrl(bbcode);
if (!originalUrl) {
throw new Error("Invalid BBCode - No original URL found");
}
const thumbnailUrl = (_c = (_b = (_a2 = bbcode.match(/\[img\]((.|\s)*?)\[/)) == null ? void 0 : _a2[1]) == null ? void 0 : _b.trim()) != null ? _c : "";
if (!thumbnailUrl) {
throw new Error("Invalid BBCode - No thumbnail URL found");
}
return {
original: originalUrl,
thumbnail: thumbnailUrl
};
};
const extractImgsFromBBCode = async (bbcode) => {
if (!bbcode) {
return [];
}
const imgBBCodeMatches = getImageBBCodeMatches(bbcode);
const matchImgBBCodes = Array.from(imgBBCodeMatches);
if (matchImgBBCodes.length < 1) {
return [];
}
const extractOriginalUrlsOfScreenshotPromises = matchImgBBCodes.map(
async (bbcode2) => {
return await getOriginalImgUrl(bbcode2);
}
);
const { FILTER_IMGS_NAMES } = CONFIG$4;
const screenshotUrls = (await Promise.all(extractOriginalUrlsOfScreenshotPromises)).filter((url) => !FILTER_IMGS_NAMES.some((name) => url.includes(name)));
return screenshotUrls;
};
const createHDBRequestConfig = async (imgUrls, galleryName) => {
const imgFiles = await Promise.all(
imgUrls.map((item) => cachedUrlToFile(item))
);
const formData = createFormData(
{
galleryoption: "1",
galleryname: galleryName
},
[
{
fieldName: "images_files",
file: imgFiles
}
]
);
return {
url: CONFIG$4.URLS.HDB_UPLOAD,
options: {
data: formData,
method: "POST"
}
};
};
const parseHDBResponse = async (data) => {
if (!data || data.includes("error")) {
throwUploadError(`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - ${data}`);
}
const urls = data.split(" ").filter((url) => url.trim().length > 0);
if (urls.length < 1) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Empty Result`
);
}
const promiseResult = urls.map((urlBBcode) => {
return getImgInfoFromBBCode(urlBBcode);
});
return Promise.all(promiseResult);
};
const getImgboxToken = async () => {
var _a2, _b;
const rawHtml = await GMFetch(CONFIG$4.URLS.IMGBOX);
const authToken = (_b = (_a2 = rawHtml.match(/content="(.+)" name="csrf-token"/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!authToken) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Invalid AuthToken`
);
}
const tokenSecret = await GMFetch(
`${CONFIG$4.URLS.IMGBOX}/ajax/token/generate`,
{
responseType: "json",
method: "POST",
headers: {
"X-CSRF-Token": authToken
}
}
);
if (!tokenSecret || !tokenSecret.token_id || !tokenSecret.token_secret) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Invalid Token`
);
}
return {
tokenSecret,
authToken
};
};
const createImgboxRequestConfig = (tokenSecret, authToken, file) => {
const { token_id: tokenId, token_secret: secret } = tokenSecret;
const formData = createFormData(
{
token_id: tokenId,
token_secret: secret,
content_type: "1",
thumbnail_size: "350c",
gallery_id: "null",
gallery_secret: "null",
comments_enabled: "0"
},
[{ fieldName: "files[]", file }]
);
const options2 = {
method: "POST",
responseType: "json",
headers: {
"X-CSRF-Token": authToken
},
data: formData
};
return options2;
};
const parseImgboxResponse = (data) => {
const imgResultList = data.map((result) => {
if (result && result.files && result.files.length) {
return result.files[0];
}
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Empty Data to Parse`
);
return null;
}).filter((item) => item !== null);
return imgResultList.map((item) => {
return {
original: item.original_url,
thumbnail: item.thumbnail_url
};
});
};
const createPixhostRequestConfig = (imgUrls) => {
if (imgUrls.length < 1) {
throwUploadError(`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Empty URLs`);
}
const params = encodeURI(
`imgs=${imgUrls.join("\n")}&content_type=1&max_th_size=300`
);
const options2 = {
method: "POST",
data: params,
headers: {
Accept: "application/json",
"Content-Type": "application/x-www-form-urlencoded;charset=utf-8"
}
};
return {
url: CONFIG$4.URLS.PIXHOST_UPLOAD,
options: options2
};
};
const parsePixhostResponse = (data) => {
const result = data.match(/(upload_results = )({.*})(;)/);
if (!result || result.length < 3) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Empty Result`
);
return [];
}
const parsedData = JSON.parse(result[2]);
const imgResultList = parsedData.images;
if (!imgResultList || imgResultList.length < 1) {
throwUploadError("No images found in the response");
return [];
}
return imgResultList.map((item) => {
return {
thumbnail: item.th_url,
original: item.show_url
};
});
};
const createPTPImgRequestConfig = (imgArray) => {
const apiKey = GM_getValue("easy-upload.ptp-img-api-key", "");
if (!apiKey) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.PTPIMG_UPLOAD_FAILED)} ${$t$1(CONFIG$4.ERROR_MESSAGES.NO_API_KEY)}`
);
}
const options2 = {
method: "POST",
responseType: "json"
};
const isFileUpload = imgArray.length > 0 && imgArray[0] instanceof File;
if (isFileUpload) {
const fileArray = imgArray.filter(
(item) => item instanceof File
);
const formData = createFormData({ api_key: apiKey }, [
{ fieldName: "file-upload", file: fileArray }
]);
options2.data = formData;
} else {
const linkArray = imgArray.filter(
(item) => typeof item === "string"
);
options2.headers = {
"Content-Type": "application/x-www-form-urlencoded;charset=utf-8"
};
options2.data = `link-upload=${linkArray.join("\n")}&api_key=${apiKey}`;
}
return {
url: CONFIG$4.URLS.PTPIMG_UPLOAD,
options: options2
};
};
const parsePTPImgResponse = (data) => {
if (!data || !Array.isArray(data) || data.length < 1) {
throwUploadError();
return [];
}
return data.map((img) => `${CONFIG$4.URLS.PTPIMG}/${img.code}.${img.ext}`);
};
const getCheveretoToken = async (imgHost) => {
var _a2, _b;
const rawHtml = await GMFetch(imgHost.replace("/json", ""));
const authToken = (_b = (_a2 = rawHtml.match(/PF\.obj\.config\.auth_token\s*=\s*"(\w+)"/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!authToken) {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} - Invalid AuthToken`
);
}
return authToken;
};
const createCheveretoRequestConfig = async (imgUrl, imgHost, authToken) => {
const isHdbHost = !!imgUrl.match(/i\.hdbits\.org/);
let formData;
const options2 = {
action: "upload",
timestamp: `${Date.now()}`,
auth_token: authToken
};
if (isHdbHost || imgHost.includes("gifyu")) {
const fileData = await cachedUrlToFile(imgUrl);
formData = createFormData(
__spreadValues({
type: "file"
}, options2),
[{ fieldName: "source", file: fileData }]
);
} else {
formData = createFormData(__spreadValues({
type: "url",
source: imgUrl
}, options2));
}
return {
method: "POST",
data: formData,
responseType: "json"
};
};
const parseCheveretoResponse = (data) => {
const imgResultList = data.map((res) => {
if (res.status_txt !== "OK") {
throwUploadError(
`${$t$1(CONFIG$4.ERROR_MESSAGES.UPLOAD_FAILED)} ${res.status_txt}`
);
return null;
}
if (res.image) {
return res.image;
}
throwUploadError();
return null;
}).filter((item) => item !== null);
return imgResultList.map((item) => {
return {
original: item.url,
thumbnail: item.thumb.url
};
});
};
const uploadToHDB = withUploadErrorHandling(
async (imgUrls, galleryName) => {
const { url, options: options2 } = await createHDBRequestConfig(imgUrls, galleryName);
const data = await GMFetch(url, options2);
return await parseHDBResponse(data);
},
"HDB"
);
const uploadToImgbox = withUploadErrorHandling(
async (imgUrls) => {
const { tokenSecret, authToken } = await getImgboxToken();
const files = await Promise.all(
imgUrls.map((item) => cachedUrlToFile(item))
);
const fileUploadPromises = files.map((file) => {
const requestOptions = createImgboxRequestConfig(
tokenSecret,
authToken,
file
);
return GMFetch(CONFIG$4.URLS.IMGBOX_UPLOAD, requestOptions);
});
const data = await Promise.all(fileUploadPromises);
return parseImgboxResponse(data);
},
"Imgbox"
);
const uploadToPixhost = withUploadErrorHandling(
async (imgUrls) => {
const { url, options: options2 } = createPixhostRequestConfig(imgUrls);
const data = await GMFetch(url, options2);
return parsePixhostResponse(data);
},
"Pixhost",
{ validateFirstArg: true, defaultResult: [] }
);
const uploadToPtpImg = withUploadErrorHandling(
async (imgArray) => {
const { url, options: options2 } = createPTPImgRequestConfig(imgArray);
const data = await GMFetch(url, options2);
return parsePTPImgResponse(data);
},
"PTPImg"
);
const transferImgToCheveretoSite = withUploadErrorHandling(
async (imgUrls, imgHost = CONFIG$4.URLS.IMGBB) => {
const authToken = await getCheveretoToken(imgHost);
const imgUploadPromises = imgUrls.map(async (imgUrl) => {
const requestOptions = await createCheveretoRequestConfig(
imgUrl,
imgHost,
authToken
);
return GMFetch(imgHost, requestOptions);
});
const data = await Promise.all(imgUploadPromises);
return parseCheveretoResponse(data);
},
"Chevereto"
);
const transferImgsToPtpimg = async (imgArray) => {
if (!imgArray || imgArray.length < 1) {
return [];
}
const uploadFn = await uploadToPtpImg;
const isHdbHost = !!imgArray[0].match(/i\.hdbits\.org/);
const isPtpHost = !!imgArray[0].match(/ptpimg\.me/);
if (isPtpHost) {
throwUploadError($t$1(CONFIG$4.ERROR_MESSAGES.NO_TRANSFER_NEEDED));
} else if (isHdbHost) {
const fileArray = await Promise.all(
imgArray.map((item) => cachedUrlToFile(item))
);
return await uploadFn(fileArray);
}
return await uploadFn(imgArray);
};
const CONFIG$3 = {
CODEC_RULES: [
{ codec: "atmos", regex: /atmos/i },
{ codec: "dtshdma", regex: /dtshdma|DTSHD Master/i },
{ codec: "dtsx", regex: /dtsx/i },
{ codec: "dts", regex: /dts/i },
{ codec: "truehd", regex: /truehd/i },
{ codec: "lpcm", regex: /lpcm/i },
{ codec: "flac", regex: /flac/i },
{ codec: "aac", regex: /aac/i },
{ codec: "dd+", regex: /DD\+|DDP|DolbyDigitalPlus/i },
{ codec: "dd", regex: /DD|DolbyDigital/i },
{ codec: "ac3", regex: /ac3/i }
]
};
class MediaParser {
constructor(source) {
this.source = source;
this.result = {
fileName: "",
fileSize: 0,
duration: 0,
format: "",
audioTracks: [],
subtitleTracks: [],
videoTracks: []
};
}
getHdrType(hdrFormat) {
if (!hdrFormat) return "";
if (/Dolby\s*Vision/i.test(hdrFormat)) return "DV";
if (/HDR10\+/i.test(hdrFormat)) return "HDR10+";
if (/HDR10/i.test(hdrFormat)) return "HDR10";
if (/HLG/i.test(hdrFormat)) return "HLG";
return "";
}
}
class MediaInfoParser extends MediaParser {
splitIntoSections() {
const lines = this.source.split("\n");
const sections = {};
let currentSectionLines = {};
let currentSectionType = "";
for (const line of lines) {
const trimmedLine = line.trim();
const sectionHeaderMatch = trimmedLine.match(/^([a-zA-Z]+)(\s*#\d+)?$/);
if (sectionHeaderMatch) {
if (currentSectionType && Object.keys(currentSectionLines).length > 0) {
if (sections[currentSectionType]) {
sections[currentSectionType].push(currentSectionLines);
} else {
sections[currentSectionType] = [currentSectionLines];
}
}
currentSectionType = sectionHeaderMatch[1];
currentSectionLines = {};
continue;
}
if (currentSectionType) {
const [key, value] = trimmedLine.split(":").map((s2) => s2.trim());
if (key && value) {
currentSectionLines[key] = value;
}
}
}
if (currentSectionType && Object.keys(currentSectionLines).length > 0) {
if (sections[currentSectionType]) {
sections[currentSectionType].push(currentSectionLines);
} else {
sections[currentSectionType] = [currentSectionLines];
}
}
return sections;
}
parseGeneralSection(general) {
if (!general) return;
this.result.fileSize = convertSizeStringToBytes(general["File size"]);
this.result.duration = this.parseDuration(general.Duration);
this.result.format = general.Format;
this.result.fileName = general["Complete name"];
}
parseVideoSection(videos) {
if (!videos || videos.length === 0) return [];
const videoTracks = [];
for (const video of videos) {
videoTracks.push(this.parseVideo(video, this.result.format));
}
return videoTracks;
}
parseAudioSection(audios) {
if (!audios || audios.length === 0) return [];
const audioTracks = [];
for (const audio of audios) {
audioTracks.push(this.parseAudio(audio));
}
return audioTracks;
}
parseSubtitleSection(subtitles) {
if (!subtitles || subtitles.length === 0) return [];
const subtitleTracks = [];
for (const subtitle of subtitles) {
const {
ID: id = "",
Title: title = "",
Language: language = "",
Default: isDefault = "",
Forced: isForced = ""
} = subtitle;
subtitleTracks.push({
id,
title,
language,
default: isDefault === "Yes",
forced: isForced === "Yes"
});
}
return subtitleTracks;
}
parseAudio(audio) {
const {
Format: audioFormat = "",
"Format profile": formatProfile = "",
"Commercial name": commercialName = "",
"Channel(s)": audioChannels = "",
ID: id = "",
Default: isDefault = "",
Forced: isForced = "",
Language: language = ""
} = audio;
let channelName = "";
const channelNumber = parseInt(audioChannels, 10);
if (channelNumber && channelNumber >= 6) {
channelName = `${channelNumber - 1}.1`;
} else {
channelName = `${channelNumber}.0`;
}
const codec = this.determineAudioCodec(
audioFormat,
commercialName,
formatProfile
);
return {
channelName,
language,
default: isDefault === "Yes",
forced: isForced === "Yes",
id,
codec
};
}
parseVideo(video, generalFormat) {
const {
Format: videoFormat = "",
"Format version": videoFormatVersion = "",
"Codec ID": videoCodeId = "",
"HDR format": hdrFormat = "",
"Encoding settings": encodingSettings,
Default: isDefault = "",
Forced: isForced = "",
ID: id = "",
Width = "",
Height = "",
"Scan type": scanType = ""
} = video;
const isEncoded = !!encodingSettings;
const formatCodecMap = {
"DVD Video": "mpeg2",
"MPEG-4": "mpeg4"
};
let videoCodec = formatCodecMap[generalFormat] || "";
if (!videoCodec) {
if (videoFormat === "MPEG Video" && videoFormatVersion === "Version 2") {
videoCodec = "mpeg2";
} else if (/xvid/i.test(videoCodeId)) {
videoCodec = "xvid";
} else if (/HEVC/i.test(videoFormat)) {
videoCodec = isEncoded ? "x265" : "hevc";
} else if (/AVC/i.test(videoFormat)) {
videoCodec = isEncoded ? "x264" : "h264";
} else if (/VC-1/i.test(videoFormat)) {
videoCodec = "vc1";
} else if (/vvc/i.test(videoFormat)) {
videoCodec = "vvc";
}
}
const hdrType = this.getHdrType(hdrFormat);
const videoWidth = parseInt(Width.replace(/\s/g, ""), 10);
const videoHeight = parseInt(Height.replace(/\s/g, ""), 10);
const resolution = this.parseResolution(videoWidth, videoHeight, scanType);
return {
hdrType,
codec: videoCodec,
isEncoded,
default: isDefault === "Yes",
forced: isForced === "Yes",
id,
resolution
};
}
parseDuration(duration) {
var _a2, _b, _c, _d, _e, _f;
const hour = (_b = (_a2 = duration.match(/(\d+)\s*h/)) == null ? void 0 : _a2[1]) != null ? _b : "0";
const minute = (_d = (_c = duration.match(/(\d+)\s*min/)) == null ? void 0 : _c[1]) != null ? _d : "0";
const second = (_f = (_e = duration.match(/(\d+)\s*s/)) == null ? void 0 : _e[1]) != null ? _f : "0";
return parseInt(hour, 10) * 3600 + parseInt(minute, 10) * 60 + parseInt(second, 10);
}
parseResolution(width, height, scanType = "") {
if (!width || !height || isNaN(width) || isNaN(height)) {
return "";
}
if (width === 7680 && height === 4320) return "4320p";
if (width === 3840 && height === 2160) return "2160p";
if (height >= 2160 || width >= 3840) {
return "2160p";
} else if (height >= 1080 || width >= 1920) {
const isProgressive = scanType === "Progressive" || !scanType;
return `1080${isProgressive ? "p" : "i"}`;
} else if (height >= 720 || width >= 1280) {
return "720p";
} else if (height >= 576 || width >= 1024) {
return "576p";
} else if (width >= 840 || height === 480) {
return "480p";
} else if (height >= 360) {
return "360p";
} else if (height >= 240) {
return "240p";
}
return `${width}x${height}`;
}
determineAudioCodec(format2, commercialName, formatProfile) {
const codecRules = [
{
match: () => /MLP FBA/i.test(format2) && /Dolby Atmos/i.test(commercialName),
codec: "atmos"
},
{ match: () => /MLP FBA/i.test(format2), codec: "truehd" },
{
match: () => /AC-3/i.test(format2) && /Dolby Digital Plus/i.test(commercialName),
codec: "dd+"
},
{ match: () => /E-AC-3/i.test(format2), codec: "dd+" },
{
match: () => /AC-3/i.test(format2) && /Dolby Digital/i.test(commercialName),
codec: "dd"
},
{ match: () => /AC-3/i.test(format2), codec: "ac3" },
{ match: () => /DTS XLL X/i.test(format2), codec: "dtsx" },
{
match: () => /DTS/i.test(format2) && /DTS-HD Master Audio/i.test(commercialName),
codec: "dtshdma"
},
{
match: () => /DTS/i.test(format2) && /MA \/ Core/i.test(formatProfile),
codec: "dtshdma"
},
{
match: () => /DTS/i.test(format2) && /High Resolution/i.test(commercialName),
codec: "dtshd"
},
{ match: () => /DTS/i.test(format2), codec: "dts" },
{ match: () => /FLAC/i.test(format2), codec: "flac" },
{ match: () => /AAC/i.test(format2), codec: "aac" },
{ match: () => /LPCM/i.test(format2), codec: "lpcm" },
{ match: () => /MP3/i.test(format2), codec: "mp3" },
{ match: () => /Opus/i.test(format2), codec: "opus" },
{ match: () => /Vorbis/i.test(format2), codec: "vorbis" }
];
const matchedRule = codecRules.find((rule) => rule.match());
return matchedRule ? matchedRule.codec : "";
}
parse() {
var _a2;
const sections = this.splitIntoSections();
this.parseGeneralSection((_a2 = sections == null ? void 0 : sections.General) == null ? void 0 : _a2[0]);
const videoTracks = this.parseVideoSection(sections == null ? void 0 : sections.Video);
const audioTracks = this.parseAudioSection(sections == null ? void 0 : sections.Audio);
const subtitleTracks = this.parseSubtitleSection(sections == null ? void 0 : sections.Text);
return __spreadProps(__spreadValues({}, this.result), {
videoTracks,
audioTracks,
subtitleTracks
});
}
}
class BDInfoParser extends MediaParser {
splitIntoSections() {
const lines = this.source.split("\n");
const sectionTypeConfig = [
{ match: /Disc Label|DISC INFO/i, typeName: "general" },
{ match: /\*?VIDEO:/i, typeName: "video" },
{ match: /\*?AUDIO:/i, typeName: "audio" },
{ match: /\*?SUBTITLE(S)?/i, typeName: "subtitle" }
];
const sections = {};
let currentSectionLines = {};
let currentSectionType = "";
let currentSectionLineIndex = 0;
for (const line of lines) {
const trimmedLine = line.trim();
const sectionHeaderMatch = sectionTypeConfig.filter(
(config) => trimmedLine.match(config.match)
);
if (sectionHeaderMatch.length > 0) {
currentSectionLineIndex = 0;
if (currentSectionType && Object.keys(currentSectionLines).length > 0) {
if (sections[currentSectionType]) {
sections[currentSectionType].push(currentSectionLines);
} else {
sections[currentSectionType] = [currentSectionLines];
}
}
currentSectionType = sectionHeaderMatch[0].typeName;
const [key, value, ...extra] = trimmedLine.split(":");
if (key && value) {
if (extra.length > 0) {
currentSectionLines = {
[key]: `${value}:${extra.join(":").trim()}`
};
} else {
currentSectionLines = {
[key]: value.trim()
};
}
} else {
currentSectionLines = {};
}
continue;
}
if (currentSectionType) {
const [key, value, ...extra] = trimmedLine.split(":").map((s2) => s2.trim());
if (key && value && !key.includes("/") && !value.includes("/")) {
if (extra.length > 0) {
currentSectionLines[key] = `${value}:${extra.join(":").trim()}`;
} else {
currentSectionLines[key] = value.trim();
}
} else if (key.trim()) {
if (!key.includes("--") && !key.match(/Bitrate\s*Description|PLAYLIST REPORT/i)) {
if (!sections[currentSectionType]) {
sections[currentSectionType] = [];
}
sections[currentSectionType].push({
[currentSectionLineIndex]: trimmedLine
});
}
}
}
}
if (currentSectionType && Object.keys(currentSectionLines).length > 0) {
if (sections[currentSectionType]) {
sections[currentSectionType].push(currentSectionLines);
} else {
sections[currentSectionType] = [currentSectionLines];
}
}
return sections;
}
parseGeneralSection(general) {
var _a2, _b;
if (!general) {
return;
}
this.result.fileSize = convertSizeStringToBytes(
(_a2 = general == null ? void 0 : general["Disc Size"]) == null ? void 0 : _a2.replace(/,/g, "")
);
this.result.duration = this.parseDuration(
(_b = general == null ? void 0 : general.Length) == null ? void 0 : _b.replace(/\s/g, "")
);
this.result.fileName = general == null ? void 0 : general["Disc Label"];
}
parseVideoSection(videos) {
if (!videos || videos.length === 0) return [];
const videoTracks = [];
for (const video of videos) {
videoTracks.push(this.parseVideo(Object.values(video)[0]));
}
return videoTracks;
}
parseVideo(video) {
var _a2, _b, _c, _d;
const videoCodec = (_b = (_a2 = video.match(/(\w|\s|-|\/)+?Video/i)) == null ? void 0 : _a2[0]) != null ? _b : "";
const codec = /HEVC/i.test(videoCodec) ? "hevc" : "h264";
const resolution = (_d = (_c = video.match(/(\d+)p/i)) == null ? void 0 : _c[0]) != null ? _d : "";
const hdrType = this.getHdrType(video);
return {
hdrType,
codec,
resolution,
default: false,
forced: false,
isEncoded: false
};
}
parseAudioSection(audios) {
if (!audios || audios.length === 0) return [];
const audioTracks = [];
for (const audio of audios) {
audioTracks.push(this.parseAudio(Object.values(audio)[0]));
}
return audioTracks;
}
parseAudio(audio) {
var _a2, _b, _c, _d, _e, _f;
const audioCodec = (_b = (_a2 = audio.match(/(\w|\s|-|\/)+?Audio/i)) == null ? void 0 : _a2[0]) != null ? _b : "";
const codec = getAudioCodecFromSource(audioCodec);
const isQuickSummaryType = /Audio\s*\//.test(audio);
let language;
if (isQuickSummaryType) {
language = audio.split("/")[0].trim();
} else {
language = (_d = (_c = audio.match(/Audio\s*(\w+)/i)) == null ? void 0 : _c[1]) != null ? _d : "";
}
const channelName = (_f = (_e = audio.match(/(\d+\.\d)/)) == null ? void 0 : _e[0]) != null ? _f : "";
return {
channelName,
language,
codec
};
}
parseSubtitleSection(subtitles) {
if (!subtitles || subtitles.length === 0) return [];
const subtitleTracks = [];
for (const subtitle of subtitles) {
subtitleTracks.push(this.parseSubtitle(Object.values(subtitle)[0]));
}
return subtitleTracks;
}
parseSubtitle(subtitle) {
var _a2, _b, _c;
const subtitleSplits = subtitle.split("/");
const firstAudioSplit = (_a2 = subtitleSplits == null ? void 0 : subtitleSplits[0]) == null ? void 0 : _a2.trim();
let language = firstAudioSplit;
if (subtitleSplits.length < 2) {
language = (_c = (_b = subtitle.match(/Graphics\s*(\w+)/)) == null ? void 0 : _b[1]) != null ? _c : "";
}
return {
language
};
}
parseDuration(duration) {
const [hour, minute, second] = duration.split(":");
return parseFloat(hour) * 3600 + parseFloat(minute) * 60 + parseFloat(second);
}
parse() {
var _a2, _b;
const sections = this.splitIntoSections();
this.parseGeneralSection(
(_b = sections == null ? void 0 : sections.general) == null ? void 0 : _b[((_a2 = sections == null ? void 0 : sections.general) == null ? void 0 : _a2.length) > 1 ? 1 : 0]
);
const videoTracks = this.parseVideoSection(sections == null ? void 0 : sections.video);
const audioTracks = this.parseAudioSection(sections == null ? void 0 : sections.audio);
const subtitleTracks = this.parseSubtitleSection(sections == null ? void 0 : sections.subtitle);
return __spreadProps(__spreadValues({}, this.result), {
videoTracks,
audioTracks,
subtitleTracks
});
}
}
const getBDTypeBasedOnSize = (size) => {
const GBSize = size / 1024 ** 3;
const DISK_TYPES = [
{ maxSize: 5, type: "DVD5" },
{ maxSize: 9, type: "DVD9" },
{ maxSize: 25, type: "BD25" },
{ maxSize: 50, type: "BD50" },
{ maxSize: 66, type: "BD66" },
{ maxSize: 100, type: "BD100" }
];
for (const { maxSize, type } of DISK_TYPES) {
if (GBSize < maxSize) {
return type;
}
}
return "Unknown";
};
const getAudioCodecFromSource = (source) => {
if (!source) {
return "";
}
const formattedSource = source.replace(/:|-|\s|\./g, "");
const { CODEC_RULES } = CONFIG$3;
for (const { codec, regex } of CODEC_RULES) {
if (regex.test(formattedSource)) {
return codec;
}
}
return "";
};
const parseMedia = (source, isBluray = false) => {
if (!source) {
return null;
}
const parser = isBluray ? new BDInfoParser(source) : new MediaInfoParser(source);
return parser.parse();
};
const CONFIG$2 = {
URLS: {
TMDB_API: "https://api.themoviedb.org",
DOUBAN_SEARCH_API: "https://omit.mkrobot.org/movie/infos",
DOUBAN_SUGGEST_API: (query) => `https://www.douban.com/search?cat=1002&q=${query}`,
DOUBAN_SUBJECT: (id) => `https://movie.douban.com/subject/${id}/`,
IMDB_URL: (id) => `https://www.imdb.com/title/${id}/`,
DOUBAN_MOBILE_API: "https://m.douban.com/rexxar/api/v2",
PT_GEN_API: (url) => `https://media.pttool.workers.dev?url=${url}`,
IMDB_RATING_API: (id) => `https://p.media-imdb.com/static-content/documents/v1/title/${id}/ratings%3Fjsonp=imdb.rating.run:imdb.api.title.ratings/data.json`,
ROTTEN_TOMATOES_API: "https://79frdp12pn-1.algolianet.com/1/indexes/*/queries?x-algolia-agent=Algolia%20for%20JavaScript%20(4.24.0)%3B%20Browser%20(lite)&x-algolia-api-key=175588f6e5f8319b27702e4cc4013561&x-algolia-application-id=79FRDP12PN"
},
KEY: {
TMDB_API_KEY: "4a06eb7286be2d481fdb30e0a6b4ad77"
},
REGION_PATTERNS: {
US: /USA|US|Canada|CA|美国|加拿大|United States/i,
EU: /欧|英|法|德|俄|意|苏联|西班牙|EU/i,
JP: /Japan|日本|JP/i,
KR: /Korea|韩国|KR/i,
TW: /Taiwan|台湾|TW/i,
HK: /Hong\s?Kong|香港|HK/i,
IN: /India|印度|IN/i,
AU: /Australia|澳大利亚|AU/i,
CN: /CN|China|大陆|中|内地|Mainland/i
},
EUROPE_LIST: [
"Albania",
"Andorra",
"Armenia",
"Austria",
"Azerbaijan",
"Belarus",
"Belgium",
"Bosnia and Herzegovina",
"Bulgaria",
"Croatia",
"Cyprus",
"Czech Republic",
"Denmark",
"Estonia",
"Finland",
"France",
"Georgia",
"Germany",
"Greece",
"Hungary",
"Iceland",
"Ireland",
"Italy",
"Kazakhstan",
"Latvia",
"Liechtenstein",
"Lithuania",
"Luxembourg",
"Malta",
"Moldova",
"Monaco",
"Montenegro",
"Netherlands",
"North Macedonia",
"Norway",
"Poland",
"Portugal",
"Romania",
"Russia",
"San Marino",
"Serbia",
"Slovakia",
"Slovenia",
"Spain",
"Sweden",
"Switzerland",
"Turkey",
"Ukraine",
"United Kingdom",
"UK",
"Vatican City"
]
};
const getIMDBData = async (imdbUrl) => {
if (!imdbUrl) {
throw new Error("No IMDB URL provided");
}
const data = await GMFetch(
CONFIG$2.URLS.PT_GEN_API(imdbUrl),
{
responseType: "json"
}
);
if (!data || !data.success) {
throw new Error((data == null ? void 0 : data.error) || "Failed to get IMDB data");
}
return data;
};
const getIMDBRating = async (imdbId) => {
var _a2, _b, _c, _d, _e;
if (!imdbId) {
throw new Error("No IMDB ID provided");
}
const url = CONFIG$2.URLS.IMDB_RATING_API(imdbId);
const data = await GMFetch(url);
const matchData = (_b = (_a2 = data == null ? void 0 : data.match(/[^(]+\((.+)\)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
if (!matchData) {
throw new Error("No rating data found");
}
try {
const ratingData = JSON.parse(matchData);
return __spreadProps(__spreadValues({}, ratingData.resource), {
id: (_e = (_d = (_c = ratingData.resource) == null ? void 0 : _c.id.match(/tt\d+/)) == null ? void 0 : _d[0]) != null ? _e : ""
});
} catch (e2) {
throw new Error("Failed to parse rating data");
}
};
const getIdByIMDbUrl = (imdbUrl) => {
const imdbIdArray = /tt\d+/.exec(imdbUrl);
if (imdbIdArray && imdbIdArray[0]) {
return imdbIdArray[0];
}
return "";
};
const NBSP = " ";
const CREDIT_INDENTATION = 24;
const FORMAT_RULES = [
{
key: "poster",
title: "",
formatter: (url) => `[img]${url}[/img]
`
},
{
key: "translatedTitle",
title: "译名",
formatter: (title) => title.join(" / ")
},
{
key: "originalTitle",
title: "片名"
},
{
key: "year",
title: "年代"
},
{
key: "countries",
title: "产地",
formatter: (countries) => countries.join(" / ")
},
{
key: "genres",
title: "类别",
formatter: (genres) => genres.join(" / ")
},
{
key: "languages",
title: "语言",
formatter: (languages) => languages.join(" / ")
},
{
key: "pubdate",
title: "上映日期",
formatter: (dates) => dates.join(" / ")
},
{
key: "imdbRating",
title: "IMDb评分"
},
{
key: "imdbLink",
title: "IMDb链接"
},
{
key: "doubanRating",
title: "豆瓣评分"
},
{
key: "doubanLink",
title: "豆瓣链接"
},
{
key: "episodes_count",
title: "集数"
},
{
key: "durations",
title: "片长",
formatter: (durations) => durations.join(" / ")
},
{
key: "tags",
title: "标签",
formatter: (tags2) => tags2.join(" | ")
},
{
key: "creditsData",
formatter: (creditsData) => `
${creditsData}`
},
{
key: "intro",
title: "简介",
formatter: (intro) => `
${intro.replace(/\n/g, `
${NBSP.repeat(2)}`)}`
},
{
key: "awards",
title: "获奖情况",
formatter: (awards) => `
${awards.replace(/\n/g, `
${NBSP.repeat(6)}`)}
`
}
];
class DoubanFormatter {
constructor(id, type, imdbId = "") {
this.doubanId = id;
this.type = type;
this.imdbId = imdbId;
}
async fetchAllData() {
try {
const [awards, credits, info, imdbData] = await Promise.all([
getDoubanAwards(this.doubanId),
getDoubanCreditsData(this.doubanId, this.type),
getMobileDoubanInfo(this.doubanId, this.type),
this.fetchIMDbData()
]);
return { awards, credits, info, imdbData };
} catch (e2) {
throw new Error(
`Failed to fetch data for Douban ID: ${this.doubanId} ${e2.message}`
);
}
}
async fetchIMDbData() {
if (!this.imdbId) {
this.imdbId = await getIMDbIDFromDouban(this.doubanId);
}
if (this.imdbId) {
return await getIMDBRating(this.imdbId);
}
return null;
}
formatPosterUrl(poster) {
if (poster.includes("img3")) {
return poster.replace("img3", "img1").replace(/m(_ratio_poster)/, "l$1");
}
return poster;
}
formatTitles(data) {
const { title, original_title: originalTitle, aka } = data;
const translatedTitle = [...aka];
if (originalTitle && originalTitle !== title) {
translatedTitle.unshift(title);
}
return {
translatedTitle: Array.from(new Set(translatedTitle)).filter(Boolean),
originalTitle: originalTitle || title
};
}
updateCredits(credits) {
if (!credits || !credits.items || credits.items.length === 0) {
return "";
}
const indentationMap = {
2: 7,
3: 2,
4: 0,
5: 0
};
const result = {};
for (const item of credits.items) {
if (!result[item.category]) {
result[item.category] = [];
}
result[item.category].push(item);
}
const creditsData = [];
for (const [category, items] of Object.entries(result)) {
const celebrity = items.map((item) => {
var _a2;
return `${item.name} ${(_a2 = item.latin_name) != null ? _a2 : ""}`;
});
const indentation = indentationMap[category.length] || 0;
const celebrityKey = category.split("").join(NBSP.repeat(indentation));
const celebrityValue = celebrity.join(`
${NBSP.repeat(CREDIT_INDENTATION)}`).trim();
creditsData.push(`◎${celebrityKey}${NBSP.repeat(7)}${celebrityValue}`);
}
return creditsData.join("\n");
}
updateRating(data, imdbRating) {
var _a2, _b;
const { value, count } = data.rating;
return {
doubanRating: value ? `${value} (${count}人评分)` : "",
imdbRating: `${(_a2 = imdbRating == null ? void 0 : imdbRating.rating) != null ? _a2 : 0}/10 from ${(_b = imdbRating == null ? void 0 : imdbRating.ratingCount) != null ? _b : 0} users`,
doubanLink: CONFIG$2.URLS.DOUBAN_SUBJECT(this.doubanId),
imdbLink: this.imdbId ? CONFIG$2.URLS.IMDB_URL(this.imdbId) : ""
};
}
generateOutput(formatData) {
const result = [];
for (const rule of FORMAT_RULES) {
const value = formatData[rule.key];
if (value) {
const formattedValue = rule.formatter ? rule.formatter(value) : value;
const { title } = rule;
if (!title) {
result.push(formattedValue);
continue;
}
let prefix = title + NBSP.repeat(7);
if (title.length === 2) {
prefix = title.split("").join(NBSP.repeat(7)) + NBSP.repeat(7);
}
result.push(`◎${prefix}${formattedValue}`);
}
}
return result.join("\n").trim();
}
async format() {
var _a2;
const data = await this.fetchAllData();
if (!(data == null ? void 0 : data.info)) {
throw new Error("failed to fetch douban info");
}
const poster = this.formatPosterUrl((_a2 = data.info) == null ? void 0 : _a2.cover_url);
const { translatedTitle, originalTitle } = this.formatTitles(data.info);
const creditsData = this.updateCredits(data == null ? void 0 : data.credits);
const { doubanRating, imdbRating, doubanLink, imdbLink } = this.updateRating(data.info, data == null ? void 0 : data.imdbData);
const formatData = __spreadProps(__spreadValues({}, data.info), {
awards: data.awards,
poster,
translatedTitle,
originalTitle,
creditsData,
doubanRating,
imdbRating,
doubanLink,
imdbLink,
format: ""
});
return __spreadProps(__spreadValues({}, formatData), {
format: this.generateOutput(formatData)
});
}
}
const getDoubanAwards = async (doubanId) => {
const url = `${CONFIG$2.URLS.DOUBAN_SUBJECT(doubanId)}awards/`;
const data = await GMFetch(url);
const doc = new DOMParser().parseFromString(data, "text/html");
const awardsDom = doc.querySelector("#content > div > div.article");
if (!awardsDom) {
return "";
}
return awardsDom == null ? void 0 : awardsDom.innerHTML.replace(/[ \n]/g, "").replace(/<\/li>
/g, " ").replace(/<\/a> ]*>/g, "\n").replace(/<[^>]+>/g, "").replace(/ /g, " ").replace(/ +\n/g, "\n").trim();
};
const getIMDbIDFromDouban = async (doubanId) => {
var _a2, _b, _c, _d;
const doubanPage = await GMFetch(
CONFIG$2.URLS.DOUBAN_SUBJECT(doubanId)
);
const dom = new DOMParser().parseFromString(doubanPage, "text/html");
const spans = dom.querySelectorAll("#info > span.pl");
const spansContainsIMDb = Array.from(spans).filter(
(span) => {
var _a3;
return (_a3 = span.textContent) == null ? void 0 : _a3.includes("IMDb");
}
);
const imdbId = (_d = (_c = (_b = (_a2 = spansContainsIMDb[0]) == null ? void 0 : _a2.nextSibling) == null ? void 0 : _b.nodeValue) == null ? void 0 : _c.trim()) != null ? _d : "";
return imdbId;
};
const getMobileDoubanInfo = async (id, type) => {
if (!id) {
throw new Error("No Douban ID found");
}
const url = `${CONFIG$2.URLS.DOUBAN_MOBILE_API}/${type}/${id}`;
const data = await GMFetch(`${url}?for_mobile=1&ck=`, {
headers: {
Referer: `https://m.douban.com/${type}/subject/${id}`
},
responseType: "json"
});
if (data && data.title === "未知电影") {
throw new Error("Please login in Douban to and try again");
}
return data;
};
const getDoubanBasicDataByQuery = async (query) => {
var _a2, _b, _c, _d;
if (!query) {
throw new Error("No query provided");
}
const imdbId = getIdByIMDbUrl(query);
const searchParams = imdbId || query;
const url = CONFIG$2.URLS.DOUBAN_SUGGEST_API(searchParams);
const data = await GMFetch(url);
const doc = new DOMParser().parseFromString(data, "text/html");
const linkDom = doc.querySelector(".result-list .result h3 a");
if (!linkDom) {
throw new Error("No Douban Item was found");
} else {
const { href, textContent } = linkDom;
const season = (_b = (_a2 = textContent == null ? void 0 : textContent.match(/第(.+?)季/)) == null ? void 0 : _a2[1]) != null ? _b : "";
const doubanId = (_d = (_c = decodeURIComponent(href).match(/subject\/(\d+)/)) == null ? void 0 : _c[1]) != null ? _d : "";
return {
id: doubanId,
season,
isTV: !!season,
title: textContent
};
}
};
const getDoubanTVItemData = async (doubanData, torrentTitle) => {
var _a2, _b;
const { title } = doubanData;
const seasonNumber = (_b = (_a2 = torrentTitle == null ? void 0 : torrentTitle.match(/S(?!eason)?0?(\d+)\.?(EP?\d+)?/i)) == null ? void 0 : _a2[1]) != null ? _b : "1";
if (parseInt(seasonNumber, 10) === 1) {
return doubanData;
}
const query = title == null ? void 0 : title.replace(/第.+?季/, `第${seasonNumber}季`);
const response = await getDoubanBasicDataByQuery(query);
return response;
};
const getDoubanCreditsData = async (id, type) => {
const url = `${CONFIG$2.URLS.DOUBAN_MOBILE_API}/${type}/${id}/credits?for_mobile=1&ck=`;
const data = await GMFetch(url, {
headers: {
Referer: `https://m.douban.com/${type}/subject/${id}`
},
responseType: "json"
});
return data;
};
const getDoubanBookInfo = async (doubanUrl) => {
const reqUrl = CONFIG$2.URLS.PT_GEN_API(doubanUrl);
const data = await GMFetch(reqUrl, {
responseType: "json"
});
return data;
};
const getDoubanInfoByIdOrDoubanUrl = async (query, type = "movie", imdbId) => {
var _a2, _b;
let doubanId = query;
if (!/^\d/.test(query)) {
doubanId = (_b = (_a2 = query == null ? void 0 : query.match(/douban\.com\/subject\/(\d+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
}
const parser = new DoubanFormatter(doubanId, type, imdbId);
return await parser.format();
};
const getTMDBDataByIMDBId = async (imdbid) => {
const url = `${CONFIG$2.URLS.TMDB_API}/3/find/${imdbid}?api_key=${CONFIG$2.KEY.TMDB_API_KEY}&language=en&external_source=imdb_id`;
const data = await GMFetch(url, {
responseType: "json"
});
const movieFound = data.movie_results && data.movie_results.length > 0;
const tvFound = data.tv_results && data.tv_results.length > 0;
if (!movieFound && !tvFound) {
throw new Error("No movie or TV found");
}
const tmdbData = movieFound ? data.movie_results[0] : data.tv_results[0];
return tmdbData;
};
const getTMDBVideosById = async (tmdbId) => {
var _a2;
const url = `${CONFIG$2.URLS.TMDB_API}/3/movie/${tmdbId}/videos?api_key=${CONFIG$2.KEY.TMDB_API_KEY}&language=en`;
const data = await GMFetch(url, {
responseType: "json"
});
if (!(data == null ? void 0 : data.results) || ((_a2 = data == null ? void 0 : data.results) == null ? void 0 : _a2.length) === 0) {
throw new Error("No TMDB videos found");
}
return data.results;
};
const getAreaCode = (area) => {
const europeList = CONFIG$2.EUROPE_LIST;
if (!area) return "OT";
if (europeList.includes(area)) return "EU";
for (const [code2, pattern] of Object.entries(CONFIG$2.REGION_PATTERNS)) {
if (pattern.test(area)) return code2;
}
return "OT";
};
const getRottenTomatoesDataByQuery = async (query) => {
var _a2, _b, _c;
const res = await GMFetch(
CONFIG$2.URLS.ROTTEN_TOMATOES_API,
{
data: {
requests: [
{
indexName: "content_rt",
query
}
]
},
responseType: "json"
}
);
return (_c = (_b = (_a2 = res == null ? void 0 : res.results) == null ? void 0 : _a2[0]) == null ? void 0 : _b.hits) != null ? _c : [];
};
const getMatchRottenTomatoes = async (title, year, isTV) => {
var _a2;
try {
const MAX_YEAR_DIFF = 2;
const releaseYear = parseInt(year || "1800", 10);
const searchResultHits = await getRottenTomatoesDataByQuery(title);
const filteredHits = searchResultHits.filter(
(hit) => hit.type === (isTV ? "tv" : "movie")
);
if (!filteredHits.length) return null;
filteredHits.sort((a2, b) => {
const diffA = Math.abs(a2.releaseYear - releaseYear);
const diffB = Math.abs(b.releaseYear - releaseYear);
return diffA !== diffB ? diffA - diffB : b.releaseYear - a2.releaseYear;
});
const normalizedTitle = title.toLowerCase();
let bestMatch, closeMatch;
for (const hit of filteredHits) {
const itemTitle = ((_a2 = hit.title) == null ? void 0 : _a2.toLowerCase()) || "";
if (itemTitle === normalizedTitle) {
bestMatch = hit;
} else if (itemTitle.startsWith(normalizedTitle)) {
closeMatch = closeMatch || hit;
}
if (bestMatch) break;
}
const isYearCompatible = (rtYear) => Math.abs(rtYear - releaseYear) <= MAX_YEAR_DIFF;
if (releaseYear && (!bestMatch || !isYearCompatible(bestMatch.releaseYear))) {
if (closeMatch && isYearCompatible(closeMatch.releaseYear)) {
bestMatch = closeMatch;
} else if (filteredHits.length > 0 && isYearCompatible(filteredHits[0].releaseYear)) {
bestMatch = filteredHits[0];
}
}
bestMatch = bestMatch || closeMatch || searchResultHits[0];
return bestMatch;
} catch (error) {
console.error("Error fetching data from Rotten Tomatoes:", error);
return null;
}
};
const getSubTitleFromDoubanInfo = (data, torrentInfo) => {
var _a2, _b;
const { originalTitle, translatedTitle, title: doubanDefaultTitle } = data;
let title = "";
if (doubanDefaultTitle.match(/[\u4e00-\u9fa5]+/)) {
title += doubanDefaultTitle;
}
const moreTitle = [originalTitle, ...translatedTitle].filter(
(item) => title !== item
);
const { title: torrentTitle, hardcodedSub } = torrentInfo;
let seasonEpisode = (_b = (_a2 = torrentTitle.match(/S\d+EP?(\d+)?/i)) == null ? void 0 : _a2[1]) != null ? _b : "";
seasonEpisode = seasonEpisode.replace(/^0/i, "");
const episode = seasonEpisode ? ` 第${seasonEpisode}集` : "";
const hardcodedSubStr = hardcodedSub ? "| 硬字幕" : "";
return `${title}${moreTitle.length > 0 ? "/" : ""}${moreTitle.join("/")}${episode} ${hardcodedSubStr}`;
};
class FillerRegistry {
constructor() {
this.fillers = [];
}
register(filler) {
this.fillers.push(filler);
this.fillers.sort((a2, b) => b.priority - a2.priority);
}
getApplicableFiller(siteName, siteType) {
return this.fillers.find((filler) => filler.canHandle(siteName, siteType)) || null;
}
}
const registry$1 = new FillerRegistry();
const base64ToBlob = (base64, contentType = "application/x-bittorrent", sliceSize = 512) => {
const regStr = new RegExp(`data:${contentType};base64,`, "i");
const byteCharacters = window.atob(base64.replace(regStr, ""));
const byteArrays = [];
for (let offset = 0; offset < byteCharacters.length; offset += sliceSize) {
const slice = byteCharacters.slice(offset, offset + sliceSize);
const byteNumbers = new Array(slice.length);
for (let i = 0; i < slice.length; i++) {
byteNumbers[i] = slice.charCodeAt(i);
}
const byteArray = new Uint8Array(byteNumbers);
byteArrays.push(byteArray);
}
return new Blob(byteArrays, { type: contentType });
};
const filterEmptyTags = (description) => {
const reg = new RegExp(
"\\[(?!info)([a-zA-Z]+\\d?)(?:=(?:\\w|\\s)+)?\\]\\s*\\[\\/(\\w+)\\]",
"g"
);
if (description.match(reg)) {
description = description.replace(reg, (_match, p1, p2) => {
if (p1 === p2) {
return "";
}
return _match;
});
return filterEmptyTags(description);
}
return description;
};
const getTeamName = (title) => {
var _a2, _b, _c;
const teamMatch = title.match(/-([^-]+)$/);
let teamName = (_c = (_b = (_a2 = teamMatch == null ? void 0 : teamMatch[1]) == null ? void 0 : _a2.replace(/-/g, "")) == null ? void 0 : _b.split("@")) != null ? _c : "";
if (teamName) {
teamName = teamName.length > 1 ? teamName[1] : teamName[0];
} else {
teamName = "other";
}
return teamName;
};
const fixTorrentTitle = (title, isWebSource) => {
let fixedTitle = title.replace(" DoVi ", " DV ").replace(" DDP ", " DD+ ");
if (isWebSource) fixedTitle = fixedTitle.replace(" HEVC", " H.265");
return fixedTitle;
};
const extractChineseMovieName = (description, subtitle) => {
var _a2, _b, _c, _d, _e, _f, _g, _h, _i;
const originalName = (_b = (_a2 = description.match(/(片\s+名)\s+(.+)?/)) == null ? void 0 : _a2[2]) != null ? _b : "";
const translateName = (_f = (_e = (_d = (_c = description.match(/(译\s+名)\s+(.+)/)) == null ? void 0 : _c[2]) == null ? void 0 : _d.split("/")) == null ? void 0 : _e[0]) != null ? _f : "";
let chineseName = originalName;
if (!originalName.match(/[\u4e00-\u9fa5]+/)) {
chineseName = translateName.match(/[\u4e00-\u9fa5]+/) ? translateName : "";
}
if (chineseName === "" && subtitle !== "" && subtitle !== void 0) {
chineseName = (_h = (_g = subtitle == null ? void 0 : subtitle.replace(/【|】.*/g, "").split("/")) == null ? void 0 : _g[0]) != null ? _h : "";
}
return (_i = chineseName == null ? void 0 : chineseName.trim()) != null ? _i : "";
};
const replaceAlignTagsToQuote = (description) => {
description = description.replace(/\[(right|left|center)\]/gi, "[quote]").replace(/\[\/(right|left|center)\]/gi, "[/quote]");
return description;
};
const filterNexusDescription = (description, screenshots) => {
let filterDescription = "";
const quoteList = description.match(/\[quote(=\w+)?\](.|\n)+?\[\/quote\]/g);
if (quoteList && quoteList.length > 0) {
quoteList.forEach((quote) => {
const isMediaInfoOrBDInfo = quote.match(
/Disc\s?Size|\.mpls|Unique\s?ID|唯一ID|Resolution/i
);
if (!quote.match(/[\u4e00-\u9fa5]+/i) || isMediaInfoOrBDInfo) {
filterDescription += `${quote}
`;
}
});
}
if (screenshots.length > 0) {
filterDescription += `
${screenshots.map((v2) => `[img]${v2}[/img]`).join("\n")}
`;
}
return filterDescription;
};
const buildPTPDescription = (info) => {
let text2 = info.originalDescription || "";
text2 = text2.replace(/http:\/\/ptpimg\.me/g, "https://ptpimg.me");
for (const mediainfo of info.mediaInfos) {
text2 = text2.replace(mediainfo, "");
}
text2 = text2.replace(
/\[(mediainfo|bdinfo)\][\s\S]*?\[\/(mediainfo|bdinfo)\]/gi,
""
);
text2 = text2.replace(
/^(?!\[img\])https:\/\/ptpimg.me.*?png(?!\[\/img\])$/gim,
(imgUrl) => {
return `[img]${imgUrl}[/img]`;
}
);
text2 = text2.replace(
/\[comparison.*\][\s\S]*\[\/comparison\]/gi,
(comparisonText) => {
return comparisonText.replace(/\[img\]/g, "").replace(/\[\/img\]/g, "").split("https://ptpimg.me").join("\nhttps://ptpimg.me").replace(/\s*\n\s*/g, "\n");
}
);
text2 = text2.replace(
/\[hide(.*)?\]\s*\[url=https:\/\/ptpimg.me.*?png\]\[img\][\s\S]*?\[\/hide\]/gi,
(imgText) => {
var _a2;
const imgs = [];
for (const urlMatch of imgText.matchAll(/\[url=(.*?)\]/gi)) {
imgs.push(urlMatch[1]);
}
const rawTitle = ((_a2 = imgText.match(/^\[hide=(.*?)\]/)) == null ? void 0 : _a2[1]) || "";
const comparisonTitles = rawTitle.trim().split(/\||\/|,|vs\.?| - /i).map((v2) => v2.trim());
if (comparisonTitles.length >= 2) {
return `[comparison=${comparisonTitles.join(", ")}]
${imgs.join("\n")}
[/comparison]
`;
}
const hideTitle = rawTitle ? `=${rawTitle}` : "";
return `[hide${hideTitle}]
[img]${imgs.join("[/img]\n[img]")}[/img]
[/hide]
`;
}
);
text2 = `${text2}
`;
text2 = text2.replace(
/\[url=https:\/\/ptpimg.me.*?png\]\[img\][\s\S]*?\n\n/gi,
(imgText) => {
const imgs = [];
for (const urlMatch of imgText.matchAll(/\[url=(.*?)\]/gi)) {
imgs.push(urlMatch[1]);
}
return `[hide]
[img]${imgs.join("[/img]\n[img]")}[/img]
[/hide]
`;
}
);
text2 = text2.replace(/\[img=(.+)?\](\n\n)?/gi, "[img]$1[/img]");
text2 = text2.replace(/\[(\/)?IMG\]/g, "[$1img]");
text2 = text2.replace(/\n\s*\n/g, "\n\n");
return text2.trim();
};
var SiteIMDbType = /* @__PURE__ */ ((SiteIMDbType2) => {
SiteIMDbType2["Standard"] = "standard";
SiteIMDbType2["StripTT"] = "strip_tt";
SiteIMDbType2["UNIT3D"] = "unit3d";
SiteIMDbType2["IMDbID"] = "imdb_id";
return SiteIMDbType2;
})(SiteIMDbType || {});
const siteIMDbTypeMap = {
HDRoute: "strip_tt",
HDSpace: "strip_tt",
BeyondHD: "imdb_id",
Blutopia: "unit3d",
fearnopeer: "unit3d",
HDPOST: "unit3d",
ACM: "unit3d",
Aither: "unit3d",
Concertos: "unit3d",
MDU: "unit3d",
LST: "unit3d",
HUNO: "unit3d"
/* UNIT3D */
};
class BaseFiller {
constructor() {
this.priority = 5;
this.info = null;
this.siteInfo = CURRENT_SITE_INFO;
this.imdbId = "";
this.isChineseTacker = (siteType) => {
return siteType.match(/NexusPHP|TTG|TNode|MTeam/);
};
this.mapAndFilterSelectValues = (key, selectArray) => {
var _a2;
const { siteInfo } = this;
if (!siteInfo[key] || !this.info[key]) {
return selectArray || [];
}
const mappedValue = (_a2 = siteInfo[key].map) == null ? void 0 : _a2[this.info[key]];
if (!mappedValue) {
return selectArray || [];
}
if (Array.isArray(mappedValue)) {
if (siteInfo[key].selector) {
const valuesCopy = [...mappedValue];
const firstValue = valuesCopy.shift();
this.setSelectValue(siteInfo[key].selector, firstValue || "");
}
if (Array.isArray(selectArray) && selectArray.length > 1) {
return selectArray.filter((item) => mappedValue.includes(item));
}
return selectArray || [];
} else if (siteInfo[key].selector) {
this.setSelectValue(siteInfo[key].selector, mappedValue);
return selectArray || [];
}
return selectArray || [];
};
}
fill(info) {
this.info = info;
this.prepareToFillInfo();
this.fillTorrentTitle();
this.disableTorrentChange();
this.fillIMDb();
this.fillDescription();
this.fillBasicAttributes();
this.fillCategory();
this.fillRemainingInfo();
this.fillTorrentFile();
this.postProcess();
}
disableTorrentChange() {
var _a2, _b;
const nameSelector = (_b = (_a2 = this.siteInfo.name) == null ? void 0 : _a2.selector) != null ? _b : "";
if (nameSelector.match(/^#\w+/)) {
const nameDom = $$2(nameSelector).clone().attr("name", "").hide();
$$2(nameSelector).attr("id", "").after(nameDom);
}
}
fillTorrentFile() {
const { torrentData, title } = this.info;
const { torrent } = this.siteInfo;
if (!torrent) {
return;
}
const fileInput = $$2(torrent.selector);
if (torrentData && fileInput.length > 0) {
const blob = base64ToBlob(torrentData);
const torrentFileName = title == null ? void 0 : title.replace(/\s/g, ".");
const file = new File([blob], `${torrentFileName}.torrent`, {
type: "application/x-bittorrent"
});
const dataTransfer = new DataTransfer();
dataTransfer.items.add(file);
const uploadInput = fileInput[0];
if (CURRENT_SITE_NAME === "MTeam") {
setTimeout(() => {
const lastValue = uploadInput.files;
uploadInput.files = dataTransfer.files;
const tracker = uploadInput._valueTracker;
if (tracker) {
tracker.setValue(lastValue);
}
const event = new Event("change", { bubbles: true });
uploadInput.dispatchEvent(event);
}, 100);
} else {
uploadInput.files = dataTransfer.files;
}
}
}
prepareToFillInfo() {
}
processTorrentTitle() {
const { title } = this.info;
const fixedTitle = title.replace("H 265", "H.265").replace("H 264", "H.264");
this.info.title = fixedTitle;
}
fillTorrentTitle() {
this.processTorrentTitle();
const { name } = this.siteInfo;
if (name) {
$$2(name.selector).val(this.info.title);
}
}
async fillIMDb() {
var _a2, _b;
const { imdbUrl = "" } = this.info;
const { imdb: imdbConfig, tmdb: tmdbConfig } = this.siteInfo;
this.imdbId = getIdByIMDbUrl(imdbUrl);
if (!(imdbConfig == null ? void 0 : imdbConfig.selector)) {
return;
}
const stripTTIMBb = (_b = (_a2 = this.imdbId) == null ? void 0 : _a2.replace("tt", "")) != null ? _b : "";
if (siteIMDbTypeMap[CURRENT_SITE_NAME] === SiteIMDbType.StripTT) {
$$2(imdbConfig.selector).val(stripTTIMBb);
} else if (siteIMDbTypeMap[CURRENT_SITE_NAME] === SiteIMDbType.UNIT3D) {
$$2(imdbConfig.selector).val(stripTTIMBb);
const { id: tmdbId } = await getTMDBDataByIMDBId(this.imdbId);
$$2(tmdbConfig.selector).val(tmdbId);
$$2("#torrent").on("change", () => {
$$2(imdbConfig).val(stripTTIMBb);
$$2(tmdbConfig.selector).val(tmdbId);
$$2("#automal").val(0);
});
} else if (siteIMDbTypeMap[CURRENT_SITE_NAME] === SiteIMDbType.IMDbID) {
$$2(imdbConfig.selector).val(this.imdbId);
} else {
$$2(imdbConfig.selector).val(imdbUrl);
}
}
processDescription() {
const { description, sourceSite } = this.info;
let fixedDescription = description.replace(/^(\s+)/g, "");
fixedDescription = filterEmptyTags(fixedDescription);
fixedDescription = this.addOrFilterDescription(fixedDescription);
fixedDescription = this.fillMediaInfo(fixedDescription);
fixedDescription = this.fillScreenshots(fixedDescription);
fixedDescription = this.fillPoster(fixedDescription);
const thanksQuoteClosed = GM_getValue("easy-upload.thanks-quote-closed") || "";
if (!thanksQuoteClosed && sourceSite) {
fixedDescription = this.getThanksQuote() + fixedDescription.trim();
}
this.info.description = fixedDescription;
}
addOrFilterDescription(description) {
const { doubanInfo, sourceSiteType, screenshots } = this.info;
const IfAddDoubanInfo = this.isChineseTacker(this.siteInfo.siteType) && CURRENT_SITE_NAME !== "SSD";
const ifFilterNexusDescription = this.isChineseTacker(sourceSiteType) && CURRENT_SITE_NAME !== "Bib";
if (IfAddDoubanInfo) {
if (doubanInfo) {
description = `${doubanInfo}
${description}`;
}
} else if (ifFilterNexusDescription) {
description = filterNexusDescription(description, screenshots);
}
return description;
}
fillMediaInfo(description) {
const { mediaInfo } = this.siteInfo;
if (!mediaInfo) {
return description;
}
const { mediaInfos, videoType } = this.info;
if (!mediaInfos || mediaInfos.length === 0) {
return description;
}
const isBluray = /bluray/i.test(videoType);
if (this.siteInfo.siteType === "UNIT3D") {
const selector = isBluray ? 'textarea[name="bdinfo"]' : this.siteInfo.mediaInfo.selector;
$$2(selector).val(mediaInfos[0]);
description = description.replace(mediaInfos[0].trim(), "");
} else if (isBluray && CURRENT_SITE_NAME.match(/^(SpeedApp)/)) {
$$2(this.siteInfo.bdinfo.selector).val(mediaInfos[0]);
this.info.mediaInfos = [];
} else if (!(isBluray && CURRENT_SITE_NAME.match(/^(HDBits)/))) {
$$2(this.siteInfo.mediaInfo.selector).val(mediaInfos[0]);
description = description.replace(mediaInfos[0].trim(), "");
}
return description;
}
fillScreenshots(description) {
const { screenshots } = this.info;
if (this.siteInfo.screenshots) {
screenshots.forEach((img) => {
if (description.includes(img)) {
description = description.replace(img, "");
if (!img.match(/\[url=.+?\[url]/)) {
description = description.replace(/\[img\]\[\/img\]\n*/g, "");
}
}
});
}
return description;
}
fillPoster(description) {
var _a2, _b;
let { doubanInfo, poster } = this.info;
if (this.siteInfo.poster) {
if (!poster) {
const doubanPosterImage = (description + doubanInfo).match(
/\[img\](http[^[]+?(poster|(img\d\.doubanio))[^[]+?)\[\/img\]/
);
if (doubanPosterImage && doubanPosterImage[1]) {
poster = doubanPosterImage[1];
} else {
poster = (_b = (_a2 = description.match(/\[img\](.+?)\[\/img\]/)) == null ? void 0 : _a2[1]) != null ? _b : "";
}
}
if (poster) {
$$2(this.siteInfo.poster).val(poster);
if (CURRENT_SITE_NAME === "HDRoute") {
$$2('input[name="poster"]').val(poster);
description = description.replace(poster, "");
}
}
}
return description;
}
fillDescription() {
var _a2;
this.processDescription();
$$2((_a2 = this.siteInfo.description) == null ? void 0 : _a2.selector).val(this.info.description);
}
getThanksQuote() {
const { sourceSite } = this.info;
const isChineseSite = this.isChineseTacker(this.siteInfo.siteType) || CURRENT_SITE_NAME.match(/HDPOST|GPW/);
let thanksQuote = `转自[b]${sourceSite}[/b],感谢原发布者!`;
if (!isChineseSite) {
thanksQuote = `Torrent from [b]${sourceSite}[/b].
All thanks to the original uploader!`;
}
return `[quote]${thanksQuote}[/quote]
`;
}
fillBasicAttributes() {
const commonInfoKeys = [
"subtitle",
"douban",
"area",
"audioCodec"
];
const { doubanUrl } = this.info;
commonInfoKeys.forEach((key) => {
const siteInfo = this.siteInfo[key];
if (siteInfo && siteInfo.selector) {
let value = this.info[key];
if (key === "douban") {
value = doubanUrl;
} else if (key === "area" || key === "audioCodec") {
value = siteInfo.map[value];
}
$$2(siteInfo.selector).val(value || "");
}
});
}
setSelectValue(selector, value) {
const selectElement = $$2(selector);
if (selectElement.length > 0) {
selectElement.val(value).trigger("change");
}
}
fillCategory() {
const categoryConfig = this.siteInfo.category;
const { category: categoryKey } = this.info;
if (!categoryConfig || !categoryKey) {
return;
}
const categoryMapValue = categoryConfig.map[categoryKey];
const keyArray = [
"videoCodec",
"videoType",
"resolution",
"source",
"area"
];
let finalSelectArray = [];
if (Array.isArray(categoryMapValue)) {
finalSelectArray = [...categoryMapValue];
keyArray.forEach((key) => {
finalSelectArray = this.mapAndFilterSelectValues(
key,
finalSelectArray
);
if (finalSelectArray.length === 1) {
this.setSelectValue(categoryConfig.selector, finalSelectArray[0]);
}
});
} else {
[...keyArray, "category"].forEach((key) => {
this.mapAndFilterSelectValues(key, finalSelectArray);
});
}
}
fillRemainingInfo() {
const {
format: formatConfig,
image: imageConfig,
anonymous: anonymousConfig,
tags: tagsConfig
} = this.siteInfo;
const { format: format2, image, tags: tags2 } = this.info;
if (formatConfig) {
$$2(formatConfig.selector).val(formatConfig.map[format2]);
}
if (imageConfig) {
$$2(imageConfig.selector).val(image || "");
}
if (anonymousConfig) {
const { selector, value = "" } = anonymousConfig;
if (value) {
$$2(selector).val(value);
} else {
$$2(selector).attr("checked", "true");
}
}
if (tagsConfig) {
Object.keys(tags2).forEach((key) => {
if (tags2[key] && tagsConfig[key]) {
$$2(tagsConfig[key]).attr("checked", "true");
}
});
}
this.fillTeamName();
}
fillTeamName() {
const teamConfig = this.siteInfo.team;
const { title } = this.info;
if (!teamConfig || !title) {
return;
}
const teamName = getTeamName(title);
if (!teamName) {
return;
}
if (teamName && teamConfig) {
const formateTeamName = teamConfig.map[teamName.toLowerCase()];
const matchValue = formateTeamName || teamConfig.map.other;
if (matchValue) {
$$2(teamConfig.selector).val(matchValue.toLowerCase());
}
}
}
postProcess() {
}
}
class Filler52pt extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "52pt";
}
postProcess() {
const { tags: tags2, videoType, resolution } = this.info;
let videoTypeValue = videoType;
if (videoType.match(/bluray/)) {
if (tags2.chinese_audio || tags2.cantonese_audio || tags2.chinese_subtitle) {
videoTypeValue = videoType === "bluray" ? "14" : "15";
}
} else if (videoType === "remux" && resolution === "2160p") {
videoTypeValue = "5";
}
$$2(this.siteInfo.videoType.selector).val(videoTypeValue);
}
}
registry$1.register(new Filler52pt());
class Bib extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "Bib";
}
fill(info) {
var _a2;
this.info = info;
if (!this.info || !this.info.doubanBookInfo) return;
const {
year,
pager,
translator,
author,
publisher,
ISBN,
book_intro: intro,
poster
} = this.info.doubanBookInfo;
$$2("#AuthorsField").val(author.join(","));
$$2("#PublishersField").val(publisher);
$$2("#IsbnField").val(ISBN);
$$2("#YearField").val(year);
$$2("#PagesField").val(pager);
$$2("#LanguageField").val("17");
$$2("#inputFileID").replaceWith(
''
);
$$2("#TranslatorsField").val(translator.join(","));
$$2("#DescriptionField").val(intro);
$$2("#ImageField").val(poster);
const event = new Event("change");
(_a2 = document.getElementById("DescriptionField")) == null ? void 0 : _a2.dispatchEvent(event);
this.fillTorrentFile();
}
}
registry$1.register(new Bib());
const CONFIG$1 = {
HDB_TEAM: [
"Chotab",
"CRiSC",
"CtrlHD",
"DON",
"EA",
"EbP",
"Geek",
"LolHD",
"NTb",
"RightSiZE",
"SA89",
"SbR",
"TayTo",
"VietHD"
]
};
class BTSCHOOL extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "BTSCHOOL";
}
postProcess() {
var _a2, _b;
const { doubanUrl } = this.info;
$$2(this.siteInfo.imdb.selector).val(this.imdbId);
if (doubanUrl) {
const doubanId = (_b = (_a2 = doubanUrl.match(/\/(\d+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
$$2(this.siteInfo.douban.selector).val(doubanId);
}
}
fillTeamName() {
const { title } = this.info;
const teamConfig = this.siteInfo.team;
const teamName = getTeamName(title);
if (CONFIG$1.HDB_TEAM.includes(teamName)) {
$$2(teamConfig.selector).val(teamConfig.map.hdbint);
}
}
}
registry$1.register(new BTSCHOOL());
class Concertos extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "Concertos";
}
processTorrentTitle() {
let { description, mediaInfos } = this.info;
$$2("#add").trigger("click");
$$2(".sceditor-button.sceditor-button-source.has-icon")[0].click();
mediaInfos.forEach((mediaInfo) => {
description = description.replace(mediaInfo.trim(), "");
});
this.info.description = description;
}
processDescription() {
let { description, mediaInfos } = this.info;
$$2("#add").trigger("click");
$$2(".sceditor-button.sceditor-button-source.has-icon")[0].click();
mediaInfos.forEach((mediaInfo) => {
description = description.replace(mediaInfo.trim(), "");
});
this.info.description = description;
super.processDescription();
}
}
registry$1.register(new Concertos());
class HDBits extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDBits";
}
processTorrentTitle() {
super.processTorrentTitle();
const { title, videoType, movieName, movieAkaName } = this.info;
let mediaTitle = title.replace(
/([^\d]+)\s+([12][90]\d{2})/,
(_2, p1, p2) => {
return `${movieName || movieAkaName} ${p2}`;
}
);
if (videoType === "remux") {
mediaTitle = mediaTitle.replace(/\s+(bluray|blu-ray)/gi, "");
}
this.info.title = mediaTitle;
}
}
registry$1.register(new HDBits());
class HDFans extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDFans";
}
postProcess() {
const { videoType, resolution, tags: tags2 } = this.info;
if (videoType === "remux") {
$$2(this.siteInfo.videoType.selector).val(
resolution === "2160p" ? "10" : "8"
);
} else if (videoType === "encode") {
const map2 = {
"2160p": "9",
"1080p": "5",
"1080i": "5",
"720p": "11"
};
$$2(this.siteInfo.videoType.selector).val(
map2[resolution] || "16"
);
}
if (tags2.diy) {
$$2(this.siteInfo.videoType.selector).val(
resolution === "2160p" ? "2" : "4"
);
}
}
}
registry$1.register(new HDFans());
class HDRoute extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDRoute";
}
postProcess() {
var _a2, _b, _c, _d, _e, _f;
const { description, doubanInfo, subtitle = "" } = this.info;
const fullDescription = description + doubanInfo;
const imdbRank = (_b = (_a2 = fullDescription.match(/IMDb评分\s+(\d(\.\d)?)/i)) == null ? void 0 : _a2[1]) != null ? _b : "";
const chineseName = extractChineseMovieName(description, subtitle);
const summary = (_f = (_e = (_d = (_c = fullDescription.match(/(简\s+介)\s+([^[◎]+)/)) == null ? void 0 : _c[2]) == null ? void 0 : _d.split("/")) == null ? void 0 : _e[0]) != null ? _f : "";
$$2("#upload-imdb").val(imdbRank);
$$2("#title_chs").val(chineseName);
$$2("#upload_introduction").val(summary);
}
}
registry$1.register(new HDRoute());
class HDT extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDT";
}
processDescription() {
let { description } = this.info;
description = description.replace(/(\[\/img\])(\[img\])/g, "$1 $2").replace(/(\[\/url\])(\[url)/g, "$1 $2");
this.info.description = description;
super.processDescription();
}
postProcess() {
let { category, imdbUrl = "" } = this.info;
if (category !== "tvPack") {
$$2('select[name="season"').val("true");
}
if (!/.+\/$/.test(imdbUrl) && /tt\d+/.test(imdbUrl)) {
imdbUrl += "/";
}
$$2(this.siteInfo.imdb.selector).val(imdbUrl);
}
}
registry$1.register(new HDT());
class HDTime extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDTime";
}
postProcess() {
if (this.info.videoType.match(/bluray/i)) {
$$2(this.siteInfo.category.selector).val("424");
}
}
}
registry$1.register(new HDTime());
class HDU extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "HDU";
}
postProcess() {
let videoTypeValue = "";
const { resolution, videoType, category } = this.info;
const isTV = category.match(/tv/);
if (videoType === "remux") {
if (resolution === "2160p") {
videoTypeValue = isTV ? "16" : "15";
} else {
videoTypeValue = isTV ? "12" : "3";
}
}
if (isTV) {
if (videoType === "encode") {
videoTypeValue = "14";
} else if (videoType === "web") {
videoTypeValue = "13";
}
}
if (videoTypeValue) {
$$2(this.siteInfo.videoType.selector).val(videoTypeValue);
}
}
}
registry$1.register(new HDU());
class KEEPFRDS extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "KEEPFRDS";
}
processTorrentTitle() {
const { category, title, subtitle } = this.info;
if (category === "music") {
const subtitle2 = title;
if (subtitle2 !== void 0) {
this.info.title = subtitle2;
this.info.subtitle = subtitle2;
}
} else if (subtitle === "") {
this.info.subtitle = title;
}
}
processDescription() {
var _a2;
let {
description,
sourceSite,
mediaInfos,
originalDescription,
screenshots,
category,
title,
subtitle
} = this.info;
description = description.replace(/\[\/?(center|code)\]/g, "");
if (sourceSite === "PTP") {
description = (_a2 = originalDescription == null ? void 0 : originalDescription.replace(/^(\s+)/g, "")) != null ? _a2 : "";
description = filterEmptyTags(description);
description = description.replace(/http:\/\/ptpimg/g, "https://ptpimg");
screenshots.forEach((screenshot) => {
const regStr = new RegExp(`\\[img${screenshot}\\[\\/img\\]`, "i");
if (!description.match(regStr)) {
const regOldFormat = new RegExp(`\\[img=${screenshot}\\]`, "i");
if (description.match(regOldFormat)) {
description = description.replace(
regOldFormat,
`[img]${screenshot}[/img]`
);
} else {
description = description.replace(
new RegExp(`(? {
if (category !== "music") {
$(this.siteInfo.name.selector).val(title);
if (subtitle) $(this.siteInfo.subtitle.selector).val(subtitle);
} else {
$(this.siteInfo.name.selector).val(subtitle || "");
if (subtitle) $(this.siteInfo.subtitle.selector).val(title);
}
});
mediaInfos == null ? void 0 : mediaInfos.forEach((mediaInfo) => {
if (!/\[mediainfo\]/.test(description)) {
description = description.replace(
`[quote]${mediaInfo}[/quote]`,
`[mediainfo]${mediaInfo}[/mediainfo]`
);
}
});
this.info.description = description;
super.processDescription();
}
}
registry$1.register(new KEEPFRDS());
class NYPT extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "NYPT";
}
postProcess() {
const { category, title } = this.info;
$$2("#browsecat").trigger("change");
const domTimeout = setTimeout(() => {
const catMap = {
movie: "#movie_enname",
tv: "#series_enname",
tvPack: "#series_enname",
documentary: "#doc_enname",
variety: "#show_enname",
cartoon: "#anime_enname"
};
const selector = catMap[category];
if (selector) {
$$2(selector).val(title);
}
clearTimeout(domTimeout);
}, 2e3);
}
}
registry$1.register(new NYPT());
class PTN extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "PTN";
}
processDescription() {
let { description, imdbUrl } = this.info;
description = `${imdbUrl}
${description}`;
this.info.description = description;
super.processDescription();
}
postProcess() {
const { resolution, videoType, source } = this.info;
let format2 = "";
const formatMap2 = {
remux: "Remux",
web: "WebRip",
dvd: "DVDR",
dvdrip: "DVDRip",
"720p": "720P",
"1080p": "1080P",
"2160p": "2160P"
};
if (videoType.match(/bluray/)) {
format2 = "BluRay";
} else if (videoType === "encode" && source === "bluray") {
format2 = formatMap2[resolution];
} else {
format2 = formatMap2[videoType] || "";
}
$$2("#format").val(format2);
}
}
registry$1.register(new PTN());
class PTSBAO extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "PTSBAO";
}
fill(info) {
this.info = info;
this.prepareToFillInfo();
this.fillTorrentTitle();
this.disableTorrentChange();
this.fillIMDb();
this.fillDescription();
this.fillCategoryAndVideoInfo();
this.fillRemainingInfo();
this.fillTorrentFile();
}
fillCategoryAndVideoInfo() {
if (!this.info) return;
const {
category: categoryConfig,
videoCodec: videoCodecConfig,
audioCodec: audioCodecConfig,
source: sourceConfig,
videoType: videoTypeConfig,
resolution: resolutionConfig
} = this.siteInfo;
const {
category,
videoCodec = "",
audioCodec = "",
source,
videoType,
resolution
} = this.info;
$$2(categoryConfig.selector).val(categoryConfig.map[category]);
$$2(categoryConfig.selector)[0].dispatchEvent(
new Event("change", { bubbles: true })
);
setTimeout(() => {
$$2(videoCodecConfig.selector).val(videoCodecConfig.map[videoCodec]);
$$2(audioCodecConfig.selector).val(audioCodecConfig.map[audioCodec]);
$$2(sourceConfig.selector).val(sourceConfig.map[source]);
$$2(videoTypeConfig.selector).val(videoTypeConfig.map[videoType]);
$$2(resolutionConfig.selector).val(resolutionConfig.map[resolution]);
}, 500);
}
}
registry$1.register(new PTSBAO());
class SpeedApp extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "SpeedApp";
}
processDescription() {
let { description } = this.info;
description = description.replace(/\[url.*\[\/url\]/g, "").replace(/\[img.*\[\/img\]/g, "").replace(/\[\/?(i|b|center|quote|size|color)\]/g, "").replace(/\[(size|color)=#?[a-zA-Z0-9]*\]/g, "").replace(/\n\n*/g, "\n");
this.info.description = description;
super.processDescription();
}
postProcess() {
let { imdbUrl = "" } = this.info;
if (!/.+\/$/.test(imdbUrl) && /tt\d+/.test(imdbUrl)) {
imdbUrl += "/";
}
$$2(this.siteInfo.imdb.selector).val(imdbUrl);
}
}
registry$1.register(new SpeedApp());
class SSD extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "SSD";
}
processTorrentTitle() {
super.processTorrentTitle();
const { title } = this.info;
this.info.title = title.replace(/\s/gi, ".");
}
postProcess() {
const { category, title, doubanUrl, imdbUrl, screenshots, subtitle } = this.info;
if (category === "tvPack" || title.match(/Trilogy|Collection/i) || subtitle && subtitle.match(/合集/)) {
$$2('input[name="pack"]').attr("checked", "true");
}
$$2(this.siteInfo.imdb.selector).val(doubanUrl || imdbUrl);
$$2(this.siteInfo.screenshots.selector).val(screenshots.join("\n"));
}
}
registry$1.register(new SSD());
class TTG extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "TTG";
}
processDescription() {
super.processDescription();
const { description } = this.info;
this.info.description = replaceAlignTagsToQuote(description);
}
postProcess() {
var _a2, _b;
const { doubanUrl } = this.info;
if (doubanUrl) {
const doubanId = (_b = (_a2 = doubanUrl.match(/\/(\d+)/)) == null ? void 0 : _a2[1]) != null ? _b : "";
$$2(this.siteInfo.douban.selector).val(doubanId);
}
}
}
registry$1.register(new TTG());
class UHDBits extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "UHDBits";
}
postProcess() {
const { title } = this.info;
$$2(this.siteInfo.imdb.selector).val(this.imdbId);
if (title.match(/web-?rip/i)) {
$$2(this.siteInfo.videoType.selector).val("WEBRip");
}
const teamName = getTeamName(title);
$$2("#team").val(teamName === "other" ? "Unknown" : teamName);
$$2("#imdb_button").trigger("click");
}
}
registry$1.register(new UHDBits());
class UNIT3D extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 5;
}
canHandle(siteName, siteType) {
return siteType.includes("UNIT3D");
}
processDescription() {
super.processDescription();
let { description, sourceSite, screenshots } = this.info;
if (sourceSite === "PTP") {
description = buildPTPDescription(this.info);
}
if (screenshots.length > 0) {
screenshots.forEach((img) => {
const regStr = new RegExp(`\\[img\\](${img})\\[\\/img\\](
*)?`);
if (description.match(regStr)) {
description = description.replace(regStr, (p1, p2) => {
return `[url=${p2}][img=350x350]${p2}[/img][/url]`;
});
}
});
}
if (description.match(/mobile\.webp\[\/img/gi)) {
description = description.replace(/\[img\]/g, "[img=350x350]");
}
description = description.replace(
/\[align(=(.+?))\]((.|\n)+?)\[\/align\]/g,
"[$2]$3[/$2]"
);
description = description.replace(
/\[(\/)?hide(?:=(.+?))?\]/g,
(_2, p1, p2) => {
const slash = p1 || "";
return p2 ? `${p2}: [${slash}spoiler]` : `[${slash}spoiler]`;
}
);
this.info.description = description;
}
processTorrentTitle() {
super.processTorrentTitle();
const { title, source } = this.info;
const isWebSource = !!source.match(/web/gi);
const fixedTitle = fixTorrentTitle(title, isWebSource);
this.info.title = fixedTitle;
}
}
registry$1.register(new UNIT3D());
class NexusPHP extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 5;
}
canHandle(siteName, siteType) {
return siteType === "NexusPHP";
}
processDescription() {
super.processDescription();
const { description } = this.info;
this.info.description = replaceAlignTagsToQuote(description);
}
processTorrentTitle() {
super.processTorrentTitle();
const { title, source } = this.info;
const isWebSource = !!source.match(/web/gi);
const fixedTitle = fixTorrentTitle(title, isWebSource);
this.info.title = fixedTitle;
}
}
registry$1.register(new NexusPHP());
class BeyondHD extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
this.bhdSiteInfo = PT_SITE.BeyondHD;
}
canHandle(siteName) {
return siteName === "BeyondHD";
}
processTorrentTitle() {
let title = this.info.title;
if (this.info.videoType === "dvd") {
title = this.buildDVDTitle(this.info);
}
this.info.title = title;
}
fillCategory() {
const { category, videoType, size } = this.info;
const isBluray = videoType.match(/bluray/i);
this.info.category = videoType;
this.info.videoType = category;
if (isBluray || videoType === "dvd") {
let bdType = getBDTypeBasedOnSize(size);
if (videoType === "uhdbluray" && bdType === "BD50") {
bdType = "UHD50";
}
this.info.category = bdType || "";
}
super.fillCategory();
}
processDescription() {
let { description, sourceSite, originalDescription, screenshots } = this.info;
if (sourceSite === "PTP") {
description = buildPTPDescription(this.info);
} else if (sourceSite.match(/BeyondHD|UHDBits/)) {
description = originalDescription || "";
} else {
description = this.buildDescription();
}
if (screenshots.length > 0) {
screenshots.forEach((img) => {
const regStr = new RegExp(`\\[img\\](${img})\\[\\/img\\](
*)?`);
if (description.match(regStr)) {
description = description.replace(regStr, (_2, p2) => {
return `[url=${p2}][img=350x350]${p2}[/img][/url]
`;
});
}
});
}
this.info.description = description;
}
buildDescription() {
var _a2;
let { sourceSiteType, description, mediaInfos, comparisons, screenshots } = this.info;
if (this.isChineseTacker(sourceSiteType)) {
description = filterNexusDescription(description, screenshots);
}
description = description.replace(`[quote]${(_a2 = mediaInfos == null ? void 0 : mediaInfos[0]) != null ? _a2 : ""}[/quote]`, "").replace(/\[url.*\[\/url\]/g, "").replace(/\[img.*\[\/img\]/g, "");
if (comparisons && comparisons.length > 0) {
for (const comparison of comparisons) {
description += `
${comparison.reason}[comparison=${comparison.title}]
${comparison.imgs.join("\n")}
[/comparison]
`;
}
}
if (screenshots.length > 0) {
description += `${screenshots.map((v2) => `[img]${v2}[/img]`).join("\n")}
`;
}
return description.trim();
}
fill(info) {
this.info = info;
this.fillTorrentTitle();
this.disableTorrentChange();
this.fillIMDb();
this.fillTMDBId();
this.selectTag();
this.fillDescription();
this.fillMediaInfo(info.description);
this.fillCategory();
this.fillRemainingInfo();
this.fillTorrentFile();
}
buildDVDTitle(info) {
var _a2, _b;
const { movieName, movieAkaName, year, mediaInfos, size, audioCodec } = info;
const mediaInfo = (_a2 = mediaInfos == null ? void 0 : mediaInfos[0]) != null ? _a2 : "";
const scanType = mediaInfo.includes("NTSC") ? "NTSC" : "PAL";
const dvdType = getBDTypeBasedOnSize(size);
const audioChannelNumber = ((_b = mediaInfo.match(/Channel\(s\)\s+:\s+(\d)/)) == null ? void 0 : _b[1]) || "2";
const audio = audioCodec === "ac3" ? "dd" : audioCodec;
const audioName = `${audio == null ? void 0 : audio.toUpperCase()}${audioChannelNumber === "6" ? "5.1" : `${audioChannelNumber}.0`}`;
const akaName = movieAkaName ? ` AKA ${movieAkaName} ` : " ";
return `${movieName}${akaName}${year} ${scanType} ${dvdType} ${audioName}`;
}
fillTMDBId() {
const imdbId = getIdByIMDbUrl(this.info.imdbUrl || "");
getTMDBDataByIMDBId(imdbId).then((data) => {
$$2(this.siteInfo.tmdb.selector).val(data.id);
});
}
selectTag() {
const editionTags = Object.keys(this.info.tags).map(
(tag) => this.info.tags[tag] && this.bhdSiteInfo.targetInfo.editionTags[tag]
).filter(Boolean);
const editionOption = Array.from($$2('select[name="edition"] option')).map(
(opt) => $$2(opt).attr("value")
);
if (editionTags.length > 0) {
for (const tag of editionTags) {
setTimeout(() => {
var _a2, _b;
(_b = (_a2 = $$2(`.bhd-tag #${tag}`)) == null ? void 0 : _a2[0]) == null ? void 0 : _b.dispatchEvent(new Event("click"));
}, 0);
if (tag && editionOption.includes(tag)) {
$$2('select[name="edition"]').val(tag);
}
}
}
}
}
registry$1.register(new BeyondHD());
class SC extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "SC";
}
async fill(info) {
this.info = info;
const { imdbUrl = "" } = this.info;
const imdbId = getIdByIMDbUrl(imdbUrl);
$$2("#catalogue_number").val(imdbId);
$$2("#imdb_autofill").trigger("click");
this.fillMedia();
$$2(".modesw").trigger("click");
$$2("#release_desc").val(this.buildDescription());
await this.fillIMDbInfo(imdbUrl);
this.fillTorrentFile();
}
buildDescription() {
const { screenshots, mediaInfos } = this.info;
const parts = [];
if (screenshots.length > 0) {
const screenshotSection = screenshots.slice(0, 3).map((img) => `[img]${img}[/img]`).join("");
parts.push(screenshotSection);
}
if (mediaInfos.length > 0) {
const mediaInfoSection = mediaInfos.map((mediaInfo) => `[hide=MediaInfo]${mediaInfo}[/hide]`).join("\n\n");
parts.push(mediaInfoSection);
}
return parts.join("\n\n");
}
fillMedia() {
const { videoType, resolution } = this.info;
let mediaValue;
if (videoType.match(/bluray/i)) {
mediaValue = "BDMV";
} else if (videoType === "DVD") {
mediaValue = "DVD-R";
} else if (parseInt(resolution, 10) < 720) {
mediaValue = "SD";
} else {
mediaValue = resolution;
}
$$2("#media").val(mediaValue);
}
async fillIMDbInfo(imdbUrl) {
var _a2, _b, _c;
if (imdbUrl) {
try {
const imdbData = await getIMDBData(imdbUrl);
if (!imdbData) {
return;
}
if ((_a2 = imdbData == null ? void 0 : imdbData.details) == null ? void 0 : _a2.country) {
$$2("#country").val(imdbData.details.country);
}
const akaName = imdbData && ((_b = imdbData == null ? void 0 : imdbData.details) == null ? void 0 : _b["Also known as"]);
const originalName = (_c = imdbData == null ? void 0 : imdbData.name) != null ? _c : "";
if (akaName && akaName !== originalName) {
$$2("#alternate_title").val(imdbData.details["Also known as"]);
$$2("#title").val(originalName);
}
await this.uploadPoster(imdbData.poster);
} catch (error) {
console.error("Error filling IMDB data:", error);
}
}
}
async uploadPoster(posterUrl) {
var _a2;
if (!posterUrl) return;
try {
let uploadedUrl;
const ptpImgApiKey = GM_getValue("easy-upload.ptp-img-api-key");
if (ptpImgApiKey) {
const uploadResult = await (await uploadToPtpImg)([posterUrl]);
uploadedUrl = Array.isArray(uploadResult) && uploadResult.length > 0 ? uploadResult[0] : void 0;
} else {
const data = await (await transferImgToCheveretoSite)([posterUrl], "https://gifyu.com/json");
uploadedUrl = (_a2 = data[0]) == null ? void 0 : _a2.original;
}
if (uploadedUrl) {
$$2("#image").val(uploadedUrl);
}
} catch (error) {
console.error("Failed to upload poster:", error);
}
}
}
registry$1.register(new SC());
class TJUPT extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "TJUPT";
}
fill(info) {
this.info = info;
this.fillBasicAttributes();
this.fillCategory();
this.fillIMDb();
this.processDescription();
this.fillDescription();
this.fillTorrentFile();
const observer = new MutationObserver(() => {
if ($$2("#ename")[0] && $$2("#cname")[0]) {
this.fillSiteSpecificInfo();
observer.disconnect();
}
});
const config = { childList: true, subtree: true };
observer.observe(document.body, config);
}
fillSiteSpecificInfo() {
var _a2, _b;
if (!this.info) return;
const { title, description, doubanInfo, category, tags: tags2 } = this.info;
$$2("#ename").val(title);
const fullDescription = description + doubanInfo;
let area = this.extractPattern(
fullDescription,
/(产\s+地|国\s+家)\s+(.+)/,
2
);
area = area.replace(/\[\/?.+?\]/g, "");
const originalName = this.extractPattern(
fullDescription,
/(片\s+名)\s+(.+)?/,
2
);
const translateName = ((_b = (_a2 = this.extractPattern(fullDescription, /(译\s+名)\s+(.+)/, 2)) == null ? void 0 : _a2.split(
"/"
)) == null ? void 0 : _b[0]) || "";
const language = this.extractPattern(
fullDescription,
/(语\s+言)\s+(.+)/,
2
);
this.fillAreaInfo(area, category);
this.fillLanguageInfo(language);
let chineseName = originalName;
if (!originalName.match(/[\u4e00-\u9fa5]+/)) {
chineseName = translateName.match(/[\u4e00-\u9fa5]+/) ? translateName : "";
}
$$2("#cname").val(chineseName);
if (tags2.chinese_subtitle && !tags2.chinese_audio) {
$$2('input[name="chinese"]').attr("checked", "true");
}
}
extractPattern(text2, pattern, groupIndex) {
var _a2;
return ((_a2 = text2.match(pattern)) == null ? void 0 : _a2[groupIndex]) || "";
}
/**
* 填充区域信息
* @param area 区域信息
* @param category 种子类别
*/
fillAreaInfo(area, category) {
if (!area || !this.info) return;
const areaString = area.replace(/,/g, "/").replace(/\s|中国/g, "");
if (category === "movie") {
$$2("#district").val(areaString);
} else if (category == null ? void 0 : category.match(/tv/)) {
const areaToSelectorMap = {
大陆: "#specificcat1",
"台|港": "#specificcat2",
美国: "#specificcat3",
日本: "#specificcat4",
韩国: "#specificcat5",
英国: "#specificcat6",
泰剧: "#specificcat7"
};
let selector = "";
for (const [key, value] of Object.entries(areaToSelectorMap)) {
if (area.match(new RegExp(key))) {
selector = value;
break;
}
}
if (selector) {
$$2(selector).attr("checked", "true");
this.callGetcheckboxvalue("specificcat");
} else {
$$2("#specificcat").val(areaString);
}
} else if (category == null ? void 0 : category.match(/variety/)) {
const districtMap = {
CN: "#district1",
HK: "#district2",
TW: "#district2",
JP: "#district4",
KR: "#district4",
US: "#district3",
EU: "#district3",
OT: "#district5"
};
const areaKey = this.info.area;
if (districtMap[areaKey]) {
$$2(districtMap[areaKey]).attr("checked", "true");
this.callGetcheckboxvalue("district");
}
}
}
fillLanguageInfo(language) {
if (!$$2("#language").length || !language) return;
let selector = "";
if (language.match(/汉语/)) {
selector = "#language1";
} else if (language.match(/粤/)) {
selector = "#language2";
} else if (language.match(/英语/)) {
selector = "#language3";
} else if (language.match(/日语/)) {
selector = "#language4";
} else if (language.match(/韩语/)) {
selector = "#language5";
}
if (selector) {
$$2(selector).attr("checked", "true");
this.callGetcheckboxvalue("language");
}
}
callGetcheckboxvalue(name) {
if (typeof window.getcheckboxvalue === "function") {
window.getcheckboxvalue(name);
}
}
processDescription() {
let { description, mediaInfos } = this.info;
mediaInfos.forEach((info) => {
description = description.replace(
`[quote]${info}[/quote]`,
`[mediainfo]${info}[/mediainfo]`
);
});
return description;
}
}
registry$1.register(new TJUPT());
class RED extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "RED";
}
async fill(info) {
this.info = info;
await this.fillMusicInfo();
}
async fillMusicInfo() {
if (!this.info || !this.info.musicJson) {
return;
}
if (document.forms.upload_table) {
document.forms.upload_table.reset = () => {
};
}
const { musicJson } = this.info;
const {
name,
year,
musicInfo,
bbBody,
tags: tags2,
releaseType,
categoryId,
wikiImage
} = musicJson.group;
const groupId = getLocationSearchValueByKey("groupid");
if (!groupId) {
await this.searchAndRedirectToGroup(name, year);
$$2("#categories").val(categoryId - 1);
$$2("#title").val(name);
$$2("#year").val(year);
$$2("#releasetype").val(releaseType);
$$2("#tags").val(tags2.join(", "));
$$2("#album_desc").val(bbBody);
$$2("#image").val(wikiImage);
this.fillArtistsForm(musicInfo);
}
this.fillReleaseInfo(musicJson.torrent);
this.fillTorrentFile();
}
async searchAndRedirectToGroup(name, year) {
var _a2;
const url = `/ajax.php?action=browse&searchstr=${encodeURIComponent(name)} ${year}`;
try {
const searchResult = await GMFetch(url, {
responseType: "json"
});
if (searchResult.status === "success" && searchResult.response.results.length > 0) {
const groupId = searchResult.response.results[0].groupId;
const timestampMatchArray = location.hash && location.hash.match(/(^|#)timestamp=([^#]*)(#|$)/);
const timestamp2 = (_a2 = timestampMatchArray == null ? void 0 : timestampMatchArray[2]) != null ? _a2 : "";
location.href = `${CURRENT_SITE_INFO.url}${CURRENT_SITE_INFO.uploadPath}?groupid=${groupId}#timestamp=${timestamp2}`;
return true;
}
} catch (error) {
console.error("Error searching for group:", error);
}
return false;
}
fillArtistsForm(musicInfo) {
const artistTypeMap = {
artists: "1",
with: "2",
composers: "4",
conductor: "5",
dj: "6",
producer: "7",
remixedBy: "3"
};
const artists = [];
Object.keys(musicInfo).forEach((key) => {
const typeKey = key;
const values = musicInfo[typeKey].map((value) => __spreadProps(__spreadValues({}, value), {
type: artistTypeMap[typeKey]
}));
artists.push(...values);
});
for (let i = 1; i < artists.length; i++) {
window.AddArtistField();
}
artists.forEach((artist, index) => {
const selector = index > 0 ? `#artist_${index}` : "#artist";
$$2(selector).val(artist.name).next().val(artist.type || "");
});
}
fillReleaseInfo(info) {
var _a2;
const {
remasterYear,
remasterRecordLabel,
remasterCatalogueNumber,
format: format2,
encoding,
media,
description,
scene,
remasterTitle
} = info;
$$2("#remaster_record_label").val(remasterRecordLabel);
$$2("#remaster_catalogue_number").val(remasterCatalogueNumber);
$$2("#format").val(format2);
$$2("#bitrate").val(encoding);
$$2("#media").val(media);
if (media === "CD" && format2 === "FLAC") {
(_a2 = document.querySelector("#format")) == null ? void 0 : _a2.dispatchEvent(new Event("change"));
}
$$2("#remaster_year").val(remasterYear);
$$2("#release_desc").val(description);
if (scene) {
$$2("#scene").attr("checked", "true");
}
if (remasterTitle) {
$$2("#remaster_title").val(remasterTitle);
}
}
}
registry$1.register(new RED());
class KG extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "KG";
}
async fill(info) {
this.info = info;
await this.processUpload();
}
async processUpload() {
var _a2;
if (!this.info) return;
const { imdbUrl, mediaInfos, resolution, source, videoType } = this.info;
const mediaInfo = (_a2 = mediaInfos == null ? void 0 : mediaInfos[0]) != null ? _a2 : "";
if (!imdbUrl) return;
$$2('input[type="submit"][value="next >>"]').hide().after("loading...
");
$$2('input[name="title"]').val(imdbUrl);
await this.fetchAndFillIMDbData(imdbUrl);
this.fillSourceAndResolution(source, resolution, videoType);
this.fillSubtitleInfo(mediaInfo, videoType);
if (videoType === "dvd") {
$$2('input[name="dvdr"]').attr("checked", "true");
const dvdSpecs = this.buildDvdSpecs();
$$2("#ripspecs").val(dvdSpecs);
} else {
$$2("#ripspecs").val(mediaInfo);
}
}
async fetchAndFillIMDbData(imdbUrl) {
try {
const formData = new FormData();
formData.append("upstep", "2");
formData.append("type", "1");
formData.append("title", imdbUrl);
const doc = await GMFetch(`${PT_SITE.KG.url}/upload.php`, {
method: "POST",
data: formData
});
const uploadPage = new DOMParser().parseFromString(doc, "text/html");
$$2("#upside+div").html($$2("#upside+div", uploadPage).html());
await this.fillIMDbDetails(imdbUrl);
} catch (error) {
console.error("Error fetching IMDB data:", error);
}
}
async fillIMDbDetails(imdbUrl) {
if (!this.info) return;
const imdbData = await getIMDBData(imdbUrl);
if (!imdbData) return;
const siteInfo = PT_SITE.KG;
const { screenshots, poster: infoPoster } = this.info;
const { country, Languages: languages } = imdbData.details;
$$2('input[name="lang"]').val(languages);
let { description, genre, poster = "" } = imdbData;
const genreMap = siteInfo.genres.map;
const countryMap = siteInfo.country.map;
let countryValue = "";
if (country) {
countryValue = country.split(",")[0];
if (countryValue === "United States") {
countryValue = "USA";
} else if (countryValue === "United Kingdom") {
countryValue = "UK";
}
}
if (!poster) {
poster = infoPoster || "";
}
$$2('select[name="country_id"]').val(
countryMap[countryValue]
);
const descriptionBBCode = `[img]${poster}[/img]
Synopsis:
[quote]${description}[/quote]
${screenshots.map((img) => `[img]${encodeURI(img)}[/img]`).join("")}`;
$$2("#bbcodetextarea").html(descriptionBBCode);
const [mainGenre, otherGenre = ""] = genre;
$$2('select[name="genre_main_id"]').val(
genreMap[mainGenre]
);
$$2('select[name="subgenre"]').val(
genreMap[otherGenre]
);
}
fillSourceAndResolution(source, resolution, videoType) {
if (!source || !resolution || !videoType) return;
const siteInfo = PT_SITE.KG;
$$2(siteInfo.source.selector).val(
siteInfo.source.map[source]
);
if (!videoType.match("bluray")) {
$$2(siteInfo.resolution.selector).val(
siteInfo.resolution.map[resolution]
);
} else {
$$2(siteInfo.resolution.selector).val("3");
}
}
fillSubtitleInfo(mediaInfo, videoType) {
if (!mediaInfo || !videoType) return;
const isBluray = videoType.match(/bluray/i);
const parsedMedia = parseMedia(mediaInfo, !!isBluray);
if (!parsedMedia) return;
const { subtitleTracks = [] } = parsedMedia;
if (subtitleTracks.length) {
const subtitleNames = subtitleTracks.map((track) => track.language);
$$2('input[name="subs"]').val(subtitleNames.join(","));
}
}
buildDvdSpecs() {
var _a2, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k;
if (!this.info) return "";
const { mediaInfos, size, audioCodec } = this.info;
const mediaInfo = (_a2 = mediaInfos == null ? void 0 : mediaInfos[0]) != null ? _a2 : "";
const scanType = mediaInfo.includes("NTSC") ? "NTSC" : "PAL";
const dvdType = getBDTypeBasedOnSize(size);
const audioChannelNumber = ((_b = mediaInfo.match(/Channel\(s\)\s+:\s+(\d)/)) == null ? void 0 : _b[1]) || "2";
const audioName = `${audioCodec == null ? void 0 : audioCodec.toUpperCase()} ${audioChannelNumber === "6" ? "5.1" : `${audioChannelNumber}.0`}`;
const IFOMediaInfo = (_d = (_c = this.info.mediaInfos) == null ? void 0 : _c.find((info) => info.includes(".IFO"))) != null ? _d : "";
const runtime = (_g = (_f = (_e = IFOMediaInfo.match(/Duration\s*?:([^\n]+)/)) == null ? void 0 : _e[1]) == null ? void 0 : _f.replace(/\s/g, "")) != null ? _g : "";
const hour = (_i = (_h = runtime.match(/(\d)+h/)) == null ? void 0 : _h[1]) != null ? _i : "00";
const minute = (_k = (_j = runtime.match(/(\d+)(mn|min)/)) == null ? void 0 : _j[1]) != null ? _k : "";
return `DVD Label:
DVD Format: ${dvdType} ${scanType}
DVD Audio: ${audioName}
Program(s): Unknown
Menus: Untouched
Video: Untouched
Audio: Untouched
DVD extras: Untouched
Extras contain:
DVD runtime(s): ${+hour < 10 ? `0${hour}` : hour}:${minute}`;
}
}
registry$1.register(new KG());
class PTP extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "PTP";
}
fill(info) {
this.info = info;
if (!this.info) return;
const groupId = getLocationSearchValueByKey("groupid");
if (!groupId) {
this.fillIMDbAndTriggerAutoFill();
}
this.updateResolutionAndCodec();
this.fillBasicInfo();
this.fillDescription();
this.handleEditionInfo();
this.fillSubtitleInfo();
this.fillTorrentFile();
}
fillIMDbAndTriggerAutoFill() {
if (!this.info) return;
const { imdbUrl } = this.info;
const currentSiteInfo = PT_SITE.PTP;
$$2(currentSiteInfo.imdb.selector).val(imdbUrl || 0);
AutoFill();
}
updateResolutionAndCodec() {
if (!this.info) return;
const { title, videoType, resolution, videoCodec, size } = this.info;
this.info.resolution = this.getResolution(resolution, videoType, title);
this.info.videoCodec = this.getVideoCodec(
videoCodec || "",
videoType,
size
);
}
fillBasicInfo() {
if (!this.info) return;
const currentSiteInfo = PT_SITE.PTP;
const keyArray = [
"category",
"source",
"videoCodec",
"resolution"
];
keyArray.forEach((key) => {
const { selector = "", map: map2 } = currentSiteInfo[key];
if (map2) {
const mapValue = map2[this.info[key]];
$$2(selector).val(mapValue);
} else {
$$2(selector).val(this.info[key]);
}
});
}
fillDescription() {
if (!this.info) return;
const currentSiteInfo = PT_SITE.PTP;
const { sourceSite, originalDescription } = this.info;
if (sourceSite.match(/PTP/i)) {
$$2(currentSiteInfo.description.selector).val(originalDescription || "");
} else {
const description = this.getDescription();
$$2(currentSiteInfo.description.selector).val(description);
}
}
handleEditionInfo() {
if (!this.info) return;
const { videoType, tags: tags2 } = this.info;
const editionInfo = this.getEditionInfo(videoType, tags2);
if (editionInfo.length > 0) {
$$2("#remaster").attr("checked", "true");
window.Remaster();
editionInfo.forEach((edition) => {
const selector = $$2("#remaster_tags a").filter(function() {
return new RegExp(edition).test($$2(this).text());
});
if (selector.length > 0) {
const event = new Event("click");
selector[0].dispatchEvent(event);
}
});
}
}
fillSubtitleInfo() {
if (!this.info || !this.info.mediaInfos || this.info.mediaInfos.length === 0)
return;
const parsedMedia = parseMedia(this.info.mediaInfos[0]);
if (!parsedMedia) return;
const { subtitleTracks = [] } = parsedMedia;
if (subtitleTracks && subtitleTracks.length > 0) {
subtitleTracks.forEach((subtitle) => {
const subtitleSelector = $$2(".languageselector li").filter(function() {
return new RegExp(subtitle.language).test($$2(this).text());
});
if (subtitle.language !== "English" && subtitleSelector.length > 0) {
subtitleSelector.find("input").attr("checked", "true");
}
});
}
}
getEditionInfo(videoType, tags2) {
const editionInfo = [];
if (videoType === "remux") {
editionInfo.push("Remux");
}
Object.keys(tags2).forEach((tag) => {
if (tags2[tag]) {
const tagName = PT_SITE.PTP.targetInfo.editionTags[tag];
if (tagName) {
editionInfo.push(tagName);
}
}
});
return editionInfo;
}
getVideoCodec(videoCodec, videoType, size) {
if (!videoType || !size) return videoCodec;
if (videoType === "bluray") {
return getBDTypeBasedOnSize(size);
} else if (videoType === "dvd") {
const GBSize = size / 1e9;
return GBSize < 5 ? "DVD5" : "DVD9";
}
return videoCodec;
}
getResolution(resolution, videoType, title) {
if (!videoType || !title) return resolution;
if (videoType === "DVD" && title.match(/NTSC/i)) {
return "NTSC";
} else if (videoType === "DVD" && title.match(/PAL/i)) {
return "PAL";
}
return resolution;
}
getDescription() {
if (!this.info) return "";
const { mediaInfos, comparisons, screenshots } = this.info;
let filterDescription = "";
if (mediaInfos.length > 0) {
mediaInfos.forEach((mediaInfo) => {
mediaInfo = mediaInfo.replace(
/[\u00A0\u1680\u180e\u2000-\u2009\u200a\u200b\u202f\u205f\u3000]/g,
""
);
filterDescription += `[mediainfo]${mediaInfo}[/mediainfo]
`;
});
}
if (comparisons && comparisons.length > 0) {
for (const comparison of comparisons) {
filterDescription += `
${comparison.reason}[comparison=${comparison.title}]
${comparison.imgs.join("\n")}
[/comparison]
`;
}
}
const screenshotsSection = screenshots.map((item) => `[img]${item}[/img]`).join("\n");
return `${filterDescription}
${screenshotsSection}`;
}
}
registry$1.register(new PTP());
class MTeam extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "MTeam";
}
fill(info) {
this.info = info;
this.setupMutationObserver();
}
setupMutationObserver() {
const targetNode = document.getElementById("root");
if (!targetNode) return;
const config = { childList: true, subtree: true };
const observer = new MutationObserver((mutationsList, observer2) => {
for (const mutation of mutationsList) {
if (mutation.type === "childList") {
const targetElement = document.querySelector("#name");
const editor = document.querySelector(".editor-input");
if (targetElement && editor) {
observer2.disconnect();
this.fillMTeamInfo();
break;
}
}
}
});
observer.observe(targetNode, config);
}
fillMTeamInfo() {
var _a2;
if (!this.info) return;
const currentSiteInfo = PT_SITE.MTeam;
const {
title,
subtitle,
audioCodec,
doubanUrl,
imdbUrl,
mediaInfos,
tags: tags2,
source
} = this.info;
const mediaInfo = (_a2 = mediaInfos == null ? void 0 : mediaInfos[0]) != null ? _a2 : "";
this.setInputValue(currentSiteInfo.name.selector, title);
this.setInputValue(currentSiteInfo.subtitle.selector, subtitle || "");
this.setInputValue(currentSiteInfo.douban.selector, doubanUrl || "");
this.setInputValue(currentSiteInfo.imdb.selector, imdbUrl || "");
this.setInputValue(currentSiteInfo.mediaInfo.selector, mediaInfo);
this.setSelectValue(currentSiteInfo.source.selector, source || "");
this.setSelectValue(
currentSiteInfo.audioCodec.selector,
currentSiteInfo.audioCodec.map[audioCodec]
);
this.info.description = this.info.description.replace(mediaInfo.trim(), "");
this.processDescription();
this.fillMTDescription(this.info.description);
this.fillCategory();
this.fillTags(tags2, currentSiteInfo.tags);
this.fillTorrentFile();
}
fillTags(tags2, siteTagsMap) {
let timeout = 0;
Object.keys(tags2).forEach((key) => {
const siteTagSelector = siteTagsMap[key];
if (tags2[key] && siteTagSelector) {
setTimeout(() => {
this.setInputValue(siteTagSelector, "", true);
}, timeout);
timeout += 100;
}
});
}
setSelectValue(selector, value) {
const select = document.querySelector(selector);
if (select) {
const lastValue = select.value;
select.value = value;
const tracker = select._valueTracker;
if (tracker) {
tracker.setValue(lastValue);
}
const event = new Event("change", { bubbles: true });
select.dispatchEvent(event);
setTimeout(() => {
Array.from(
document.querySelectorAll(
".ant-select-item-option-active .ant-select-item-option-content"
)
).forEach((el) => {
el.dispatchEvent(new Event("click", { bubbles: true }));
});
}, 1e3);
}
}
setInputValue(selector, value, isCheckbox = false) {
const input = document.querySelector(selector);
if (input) {
const lastValue = input.value;
input.value = value;
const tracker = input._valueTracker;
if (tracker) {
tracker.setValue(lastValue);
}
const event = new Event(isCheckbox ? "click" : "input", {
bubbles: true
});
input.dispatchEvent(event);
}
}
fillMTDescription(description) {
var _a2;
const editor = (_a2 = document.querySelector(".editor-input")) == null ? void 0 : _a2.__lexicalEditor;
if (!editor) return;
const descriptionArray = description.split("\n").map((line) => ({
type: "paragraph",
children: [{ type: "text", text: line }]
}));
const content = JSON.stringify({
root: {
children: descriptionArray,
type: "root"
}
});
const editorState = editor.parseEditorState(content);
editor.update(() => {
editor.setEditorState(editorState);
});
}
}
registry$1.register(new MTeam());
var commonjsGlobal = typeof globalThis !== "undefined" ? globalThis : typeof window !== "undefined" ? window : typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : {};
function getDefaultExportFromCjs(x2) {
return x2 && x2.__esModule && Object.prototype.hasOwnProperty.call(x2, "default") ? x2["default"] : x2;
}
function getAugmentedNamespace(n) {
if (n.__esModule) return n;
var f2 = n.default;
if (typeof f2 == "function") {
var a2 = function a3() {
if (this instanceof a3) {
return Reflect.construct(f2, arguments, this.constructor);
}
return f2.apply(this, arguments);
};
a2.prototype = f2.prototype;
} else a2 = {};
Object.defineProperty(a2, "__esModule", { value: true });
Object.keys(n).forEach(function(k2) {
var d2 = Object.getOwnPropertyDescriptor(n, k2);
Object.defineProperty(a2, k2, d2.get ? d2 : {
enumerable: true,
get: function() {
return n[k2];
}
});
});
return a2;
}
var buffer = {};
var base64Js = {};
base64Js.byteLength = byteLength;
base64Js.toByteArray = toByteArray;
base64Js.fromByteArray = fromByteArray;
var lookup$1 = [];
var revLookup = [];
var Arr = typeof Uint8Array !== "undefined" ? Uint8Array : Array;
var code = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
for (var i$2 = 0, len = code.length; i$2 < len; ++i$2) {
lookup$1[i$2] = code[i$2];
revLookup[code.charCodeAt(i$2)] = i$2;
}
revLookup["-".charCodeAt(0)] = 62;
revLookup["_".charCodeAt(0)] = 63;
function getLens(b64) {
var len = b64.length;
if (len % 4 > 0) {
throw new Error("Invalid string. Length must be a multiple of 4");
}
var validLen = b64.indexOf("=");
if (validLen === -1) validLen = len;
var placeHoldersLen = validLen === len ? 0 : 4 - validLen % 4;
return [validLen, placeHoldersLen];
}
function byteLength(b64) {
var lens = getLens(b64);
var validLen = lens[0];
var placeHoldersLen = lens[1];
return (validLen + placeHoldersLen) * 3 / 4 - placeHoldersLen;
}
function _byteLength(b64, validLen, placeHoldersLen) {
return (validLen + placeHoldersLen) * 3 / 4 - placeHoldersLen;
}
function toByteArray(b64) {
var tmp;
var lens = getLens(b64);
var validLen = lens[0];
var placeHoldersLen = lens[1];
var arr = new Arr(_byteLength(b64, validLen, placeHoldersLen));
var curByte = 0;
var len = placeHoldersLen > 0 ? validLen - 4 : validLen;
var i;
for (i = 0; i < len; i += 4) {
tmp = revLookup[b64.charCodeAt(i)] << 18 | revLookup[b64.charCodeAt(i + 1)] << 12 | revLookup[b64.charCodeAt(i + 2)] << 6 | revLookup[b64.charCodeAt(i + 3)];
arr[curByte++] = tmp >> 16 & 255;
arr[curByte++] = tmp >> 8 & 255;
arr[curByte++] = tmp & 255;
}
if (placeHoldersLen === 2) {
tmp = revLookup[b64.charCodeAt(i)] << 2 | revLookup[b64.charCodeAt(i + 1)] >> 4;
arr[curByte++] = tmp & 255;
}
if (placeHoldersLen === 1) {
tmp = revLookup[b64.charCodeAt(i)] << 10 | revLookup[b64.charCodeAt(i + 1)] << 4 | revLookup[b64.charCodeAt(i + 2)] >> 2;
arr[curByte++] = tmp >> 8 & 255;
arr[curByte++] = tmp & 255;
}
return arr;
}
function tripletToBase64(num) {
return lookup$1[num >> 18 & 63] + lookup$1[num >> 12 & 63] + lookup$1[num >> 6 & 63] + lookup$1[num & 63];
}
function encodeChunk(uint8, start, end) {
var tmp;
var output = [];
for (var i = start; i < end; i += 3) {
tmp = (uint8[i] << 16 & 16711680) + (uint8[i + 1] << 8 & 65280) + (uint8[i + 2] & 255);
output.push(tripletToBase64(tmp));
}
return output.join("");
}
function fromByteArray(uint8) {
var tmp;
var len = uint8.length;
var extraBytes = len % 3;
var parts = [];
var maxChunkLength = 16383;
for (var i = 0, len2 = len - extraBytes; i < len2; i += maxChunkLength) {
parts.push(encodeChunk(uint8, i, i + maxChunkLength > len2 ? len2 : i + maxChunkLength));
}
if (extraBytes === 1) {
tmp = uint8[len - 1];
parts.push(
lookup$1[tmp >> 2] + lookup$1[tmp << 4 & 63] + "=="
);
} else if (extraBytes === 2) {
tmp = (uint8[len - 2] << 8) + uint8[len - 1];
parts.push(
lookup$1[tmp >> 10] + lookup$1[tmp >> 4 & 63] + lookup$1[tmp << 2 & 63] + "="
);
}
return parts.join("");
}
var ieee754 = {};
/*! ieee754. BSD-3-Clause License. Feross Aboukhadijeh */
ieee754.read = function(buffer2, offset, isLE, mLen, nBytes) {
var e2, m2;
var eLen = nBytes * 8 - mLen - 1;
var eMax = (1 << eLen) - 1;
var eBias = eMax >> 1;
var nBits = -7;
var i = isLE ? nBytes - 1 : 0;
var d2 = isLE ? -1 : 1;
var s2 = buffer2[offset + i];
i += d2;
e2 = s2 & (1 << -nBits) - 1;
s2 >>= -nBits;
nBits += eLen;
for (; nBits > 0; e2 = e2 * 256 + buffer2[offset + i], i += d2, nBits -= 8) {
}
m2 = e2 & (1 << -nBits) - 1;
e2 >>= -nBits;
nBits += mLen;
for (; nBits > 0; m2 = m2 * 256 + buffer2[offset + i], i += d2, nBits -= 8) {
}
if (e2 === 0) {
e2 = 1 - eBias;
} else if (e2 === eMax) {
return m2 ? NaN : (s2 ? -1 : 1) * Infinity;
} else {
m2 = m2 + Math.pow(2, mLen);
e2 = e2 - eBias;
}
return (s2 ? -1 : 1) * m2 * Math.pow(2, e2 - mLen);
};
ieee754.write = function(buffer2, value, offset, isLE, mLen, nBytes) {
var e2, m2, c2;
var eLen = nBytes * 8 - mLen - 1;
var eMax = (1 << eLen) - 1;
var eBias = eMax >> 1;
var rt = mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0;
var i = isLE ? 0 : nBytes - 1;
var d2 = isLE ? 1 : -1;
var s2 = value < 0 || value === 0 && 1 / value < 0 ? 1 : 0;
value = Math.abs(value);
if (isNaN(value) || value === Infinity) {
m2 = isNaN(value) ? 1 : 0;
e2 = eMax;
} else {
e2 = Math.floor(Math.log(value) / Math.LN2);
if (value * (c2 = Math.pow(2, -e2)) < 1) {
e2--;
c2 *= 2;
}
if (e2 + eBias >= 1) {
value += rt / c2;
} else {
value += rt * Math.pow(2, 1 - eBias);
}
if (value * c2 >= 2) {
e2++;
c2 /= 2;
}
if (e2 + eBias >= eMax) {
m2 = 0;
e2 = eMax;
} else if (e2 + eBias >= 1) {
m2 = (value * c2 - 1) * Math.pow(2, mLen);
e2 = e2 + eBias;
} else {
m2 = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
e2 = 0;
}
}
for (; mLen >= 8; buffer2[offset + i] = m2 & 255, i += d2, m2 /= 256, mLen -= 8) {
}
e2 = e2 << mLen | m2;
eLen += mLen;
for (; eLen > 0; buffer2[offset + i] = e2 & 255, i += d2, e2 /= 256, eLen -= 8) {
}
buffer2[offset + i - d2] |= s2 * 128;
};
/*!
* The buffer module from node.js, for the browser.
*
* @author Feross Aboukhadijeh
* @license MIT
*/
(function(exports) {
var base64 = base64Js;
var ieee754$1 = ieee754;
var customInspectSymbol = typeof Symbol === "function" && typeof Symbol["for"] === "function" ? Symbol["for"]("nodejs.util.inspect.custom") : null;
exports.Buffer = Buffer2;
exports.SlowBuffer = SlowBuffer;
exports.INSPECT_MAX_BYTES = 50;
var K_MAX_LENGTH = 2147483647;
exports.kMaxLength = K_MAX_LENGTH;
Buffer2.TYPED_ARRAY_SUPPORT = typedArraySupport();
if (!Buffer2.TYPED_ARRAY_SUPPORT && typeof console !== "undefined" && typeof console.error === "function") {
console.error(
"This browser lacks typed array (Uint8Array) support which is required by `buffer` v5.x. Use `buffer` v4.x if you require old browser support."
);
}
function typedArraySupport() {
try {
var arr = new Uint8Array(1);
var proto = { foo: function() {
return 42;
} };
Object.setPrototypeOf(proto, Uint8Array.prototype);
Object.setPrototypeOf(arr, proto);
return arr.foo() === 42;
} catch (e2) {
return false;
}
}
Object.defineProperty(Buffer2.prototype, "parent", {
enumerable: true,
get: function() {
if (!Buffer2.isBuffer(this)) return void 0;
return this.buffer;
}
});
Object.defineProperty(Buffer2.prototype, "offset", {
enumerable: true,
get: function() {
if (!Buffer2.isBuffer(this)) return void 0;
return this.byteOffset;
}
});
function createBuffer(length) {
if (length > K_MAX_LENGTH) {
throw new RangeError('The value "' + length + '" is invalid for option "size"');
}
var buf = new Uint8Array(length);
Object.setPrototypeOf(buf, Buffer2.prototype);
return buf;
}
function Buffer2(arg, encodingOrOffset, length) {
if (typeof arg === "number") {
if (typeof encodingOrOffset === "string") {
throw new TypeError(
'The "string" argument must be of type string. Received type number'
);
}
return allocUnsafe(arg);
}
return from(arg, encodingOrOffset, length);
}
Buffer2.poolSize = 8192;
function from(value, encodingOrOffset, length) {
if (typeof value === "string") {
return fromString(value, encodingOrOffset);
}
if (ArrayBuffer.isView(value)) {
return fromArrayView(value);
}
if (value == null) {
throw new TypeError(
"The first argument must be one of type string, Buffer, ArrayBuffer, Array, or Array-like Object. Received type " + typeof value
);
}
if (isInstance(value, ArrayBuffer) || value && isInstance(value.buffer, ArrayBuffer)) {
return fromArrayBuffer(value, encodingOrOffset, length);
}
if (typeof SharedArrayBuffer !== "undefined" && (isInstance(value, SharedArrayBuffer) || value && isInstance(value.buffer, SharedArrayBuffer))) {
return fromArrayBuffer(value, encodingOrOffset, length);
}
if (typeof value === "number") {
throw new TypeError(
'The "value" argument must not be of type number. Received type number'
);
}
var valueOf = value.valueOf && value.valueOf();
if (valueOf != null && valueOf !== value) {
return Buffer2.from(valueOf, encodingOrOffset, length);
}
var b = fromObject(value);
if (b) return b;
if (typeof Symbol !== "undefined" && Symbol.toPrimitive != null && typeof value[Symbol.toPrimitive] === "function") {
return Buffer2.from(
value[Symbol.toPrimitive]("string"),
encodingOrOffset,
length
);
}
throw new TypeError(
"The first argument must be one of type string, Buffer, ArrayBuffer, Array, or Array-like Object. Received type " + typeof value
);
}
Buffer2.from = function(value, encodingOrOffset, length) {
return from(value, encodingOrOffset, length);
};
Object.setPrototypeOf(Buffer2.prototype, Uint8Array.prototype);
Object.setPrototypeOf(Buffer2, Uint8Array);
function assertSize(size) {
if (typeof size !== "number") {
throw new TypeError('"size" argument must be of type number');
} else if (size < 0) {
throw new RangeError('The value "' + size + '" is invalid for option "size"');
}
}
function alloc(size, fill, encoding) {
assertSize(size);
if (size <= 0) {
return createBuffer(size);
}
if (fill !== void 0) {
return typeof encoding === "string" ? createBuffer(size).fill(fill, encoding) : createBuffer(size).fill(fill);
}
return createBuffer(size);
}
Buffer2.alloc = function(size, fill, encoding) {
return alloc(size, fill, encoding);
};
function allocUnsafe(size) {
assertSize(size);
return createBuffer(size < 0 ? 0 : checked(size) | 0);
}
Buffer2.allocUnsafe = function(size) {
return allocUnsafe(size);
};
Buffer2.allocUnsafeSlow = function(size) {
return allocUnsafe(size);
};
function fromString(string2, encoding) {
if (typeof encoding !== "string" || encoding === "") {
encoding = "utf8";
}
if (!Buffer2.isEncoding(encoding)) {
throw new TypeError("Unknown encoding: " + encoding);
}
var length = byteLength2(string2, encoding) | 0;
var buf = createBuffer(length);
var actual = buf.write(string2, encoding);
if (actual !== length) {
buf = buf.slice(0, actual);
}
return buf;
}
function fromArrayLike(array) {
var length = array.length < 0 ? 0 : checked(array.length) | 0;
var buf = createBuffer(length);
for (var i = 0; i < length; i += 1) {
buf[i] = array[i] & 255;
}
return buf;
}
function fromArrayView(arrayView) {
if (isInstance(arrayView, Uint8Array)) {
var copy = new Uint8Array(arrayView);
return fromArrayBuffer(copy.buffer, copy.byteOffset, copy.byteLength);
}
return fromArrayLike(arrayView);
}
function fromArrayBuffer(array, byteOffset, length) {
if (byteOffset < 0 || array.byteLength < byteOffset) {
throw new RangeError('"offset" is outside of buffer bounds');
}
if (array.byteLength < byteOffset + (length || 0)) {
throw new RangeError('"length" is outside of buffer bounds');
}
var buf;
if (byteOffset === void 0 && length === void 0) {
buf = new Uint8Array(array);
} else if (length === void 0) {
buf = new Uint8Array(array, byteOffset);
} else {
buf = new Uint8Array(array, byteOffset, length);
}
Object.setPrototypeOf(buf, Buffer2.prototype);
return buf;
}
function fromObject(obj) {
if (Buffer2.isBuffer(obj)) {
var len = checked(obj.length) | 0;
var buf = createBuffer(len);
if (buf.length === 0) {
return buf;
}
obj.copy(buf, 0, 0, len);
return buf;
}
if (obj.length !== void 0) {
if (typeof obj.length !== "number" || numberIsNaN(obj.length)) {
return createBuffer(0);
}
return fromArrayLike(obj);
}
if (obj.type === "Buffer" && Array.isArray(obj.data)) {
return fromArrayLike(obj.data);
}
}
function checked(length) {
if (length >= K_MAX_LENGTH) {
throw new RangeError("Attempt to allocate Buffer larger than maximum size: 0x" + K_MAX_LENGTH.toString(16) + " bytes");
}
return length | 0;
}
function SlowBuffer(length) {
if (+length != length) {
length = 0;
}
return Buffer2.alloc(+length);
}
Buffer2.isBuffer = function isBuffer(b) {
return b != null && b._isBuffer === true && b !== Buffer2.prototype;
};
Buffer2.compare = function compare(a2, b) {
if (isInstance(a2, Uint8Array)) a2 = Buffer2.from(a2, a2.offset, a2.byteLength);
if (isInstance(b, Uint8Array)) b = Buffer2.from(b, b.offset, b.byteLength);
if (!Buffer2.isBuffer(a2) || !Buffer2.isBuffer(b)) {
throw new TypeError(
'The "buf1", "buf2" arguments must be one of type Buffer or Uint8Array'
);
}
if (a2 === b) return 0;
var x2 = a2.length;
var y2 = b.length;
for (var i = 0, len = Math.min(x2, y2); i < len; ++i) {
if (a2[i] !== b[i]) {
x2 = a2[i];
y2 = b[i];
break;
}
}
if (x2 < y2) return -1;
if (y2 < x2) return 1;
return 0;
};
Buffer2.isEncoding = function isEncoding(encoding) {
switch (String(encoding).toLowerCase()) {
case "hex":
case "utf8":
case "utf-8":
case "ascii":
case "latin1":
case "binary":
case "base64":
case "ucs2":
case "ucs-2":
case "utf16le":
case "utf-16le":
return true;
default:
return false;
}
};
Buffer2.concat = function concat2(list, length) {
if (!Array.isArray(list)) {
throw new TypeError('"list" argument must be an Array of Buffers');
}
if (list.length === 0) {
return Buffer2.alloc(0);
}
var i;
if (length === void 0) {
length = 0;
for (i = 0; i < list.length; ++i) {
length += list[i].length;
}
}
var buffer2 = Buffer2.allocUnsafe(length);
var pos = 0;
for (i = 0; i < list.length; ++i) {
var buf = list[i];
if (isInstance(buf, Uint8Array)) {
if (pos + buf.length > buffer2.length) {
Buffer2.from(buf).copy(buffer2, pos);
} else {
Uint8Array.prototype.set.call(
buffer2,
buf,
pos
);
}
} else if (!Buffer2.isBuffer(buf)) {
throw new TypeError('"list" argument must be an Array of Buffers');
} else {
buf.copy(buffer2, pos);
}
pos += buf.length;
}
return buffer2;
};
function byteLength2(string2, encoding) {
if (Buffer2.isBuffer(string2)) {
return string2.length;
}
if (ArrayBuffer.isView(string2) || isInstance(string2, ArrayBuffer)) {
return string2.byteLength;
}
if (typeof string2 !== "string") {
throw new TypeError(
'The "string" argument must be one of type string, Buffer, or ArrayBuffer. Received type ' + typeof string2
);
}
var len = string2.length;
var mustMatch = arguments.length > 2 && arguments[2] === true;
if (!mustMatch && len === 0) return 0;
var loweredCase = false;
for (; ; ) {
switch (encoding) {
case "ascii":
case "latin1":
case "binary":
return len;
case "utf8":
case "utf-8":
return utf8ToBytes(string2).length;
case "ucs2":
case "ucs-2":
case "utf16le":
case "utf-16le":
return len * 2;
case "hex":
return len >>> 1;
case "base64":
return base64ToBytes(string2).length;
default:
if (loweredCase) {
return mustMatch ? -1 : utf8ToBytes(string2).length;
}
encoding = ("" + encoding).toLowerCase();
loweredCase = true;
}
}
}
Buffer2.byteLength = byteLength2;
function slowToString(encoding, start, end) {
var loweredCase = false;
if (start === void 0 || start < 0) {
start = 0;
}
if (start > this.length) {
return "";
}
if (end === void 0 || end > this.length) {
end = this.length;
}
if (end <= 0) {
return "";
}
end >>>= 0;
start >>>= 0;
if (end <= start) {
return "";
}
if (!encoding) encoding = "utf8";
while (true) {
switch (encoding) {
case "hex":
return hexSlice(this, start, end);
case "utf8":
case "utf-8":
return utf8Slice(this, start, end);
case "ascii":
return asciiSlice(this, start, end);
case "latin1":
case "binary":
return latin1Slice(this, start, end);
case "base64":
return base64Slice(this, start, end);
case "ucs2":
case "ucs-2":
case "utf16le":
case "utf-16le":
return utf16leSlice(this, start, end);
default:
if (loweredCase) throw new TypeError("Unknown encoding: " + encoding);
encoding = (encoding + "").toLowerCase();
loweredCase = true;
}
}
}
Buffer2.prototype._isBuffer = true;
function swap(b, n, m2) {
var i = b[n];
b[n] = b[m2];
b[m2] = i;
}
Buffer2.prototype.swap16 = function swap16() {
var len = this.length;
if (len % 2 !== 0) {
throw new RangeError("Buffer size must be a multiple of 16-bits");
}
for (var i = 0; i < len; i += 2) {
swap(this, i, i + 1);
}
return this;
};
Buffer2.prototype.swap32 = function swap32() {
var len = this.length;
if (len % 4 !== 0) {
throw new RangeError("Buffer size must be a multiple of 32-bits");
}
for (var i = 0; i < len; i += 4) {
swap(this, i, i + 3);
swap(this, i + 1, i + 2);
}
return this;
};
Buffer2.prototype.swap64 = function swap64() {
var len = this.length;
if (len % 8 !== 0) {
throw new RangeError("Buffer size must be a multiple of 64-bits");
}
for (var i = 0; i < len; i += 8) {
swap(this, i, i + 7);
swap(this, i + 1, i + 6);
swap(this, i + 2, i + 5);
swap(this, i + 3, i + 4);
}
return this;
};
Buffer2.prototype.toString = function toString() {
var length = this.length;
if (length === 0) return "";
if (arguments.length === 0) return utf8Slice(this, 0, length);
return slowToString.apply(this, arguments);
};
Buffer2.prototype.toLocaleString = Buffer2.prototype.toString;
Buffer2.prototype.equals = function equals(b) {
if (!Buffer2.isBuffer(b)) throw new TypeError("Argument must be a Buffer");
if (this === b) return true;
return Buffer2.compare(this, b) === 0;
};
Buffer2.prototype.inspect = function inspect() {
var str = "";
var max = exports.INSPECT_MAX_BYTES;
str = this.toString("hex", 0, max).replace(/(.{2})/g, "$1 ").trim();
if (this.length > max) str += " ... ";
return "";
};
if (customInspectSymbol) {
Buffer2.prototype[customInspectSymbol] = Buffer2.prototype.inspect;
}
Buffer2.prototype.compare = function compare(target, start, end, thisStart, thisEnd) {
if (isInstance(target, Uint8Array)) {
target = Buffer2.from(target, target.offset, target.byteLength);
}
if (!Buffer2.isBuffer(target)) {
throw new TypeError(
'The "target" argument must be one of type Buffer or Uint8Array. Received type ' + typeof target
);
}
if (start === void 0) {
start = 0;
}
if (end === void 0) {
end = target ? target.length : 0;
}
if (thisStart === void 0) {
thisStart = 0;
}
if (thisEnd === void 0) {
thisEnd = this.length;
}
if (start < 0 || end > target.length || thisStart < 0 || thisEnd > this.length) {
throw new RangeError("out of range index");
}
if (thisStart >= thisEnd && start >= end) {
return 0;
}
if (thisStart >= thisEnd) {
return -1;
}
if (start >= end) {
return 1;
}
start >>>= 0;
end >>>= 0;
thisStart >>>= 0;
thisEnd >>>= 0;
if (this === target) return 0;
var x2 = thisEnd - thisStart;
var y2 = end - start;
var len = Math.min(x2, y2);
var thisCopy = this.slice(thisStart, thisEnd);
var targetCopy = target.slice(start, end);
for (var i = 0; i < len; ++i) {
if (thisCopy[i] !== targetCopy[i]) {
x2 = thisCopy[i];
y2 = targetCopy[i];
break;
}
}
if (x2 < y2) return -1;
if (y2 < x2) return 1;
return 0;
};
function bidirectionalIndexOf(buffer2, val, byteOffset, encoding, dir) {
if (buffer2.length === 0) return -1;
if (typeof byteOffset === "string") {
encoding = byteOffset;
byteOffset = 0;
} else if (byteOffset > 2147483647) {
byteOffset = 2147483647;
} else if (byteOffset < -2147483648) {
byteOffset = -2147483648;
}
byteOffset = +byteOffset;
if (numberIsNaN(byteOffset)) {
byteOffset = dir ? 0 : buffer2.length - 1;
}
if (byteOffset < 0) byteOffset = buffer2.length + byteOffset;
if (byteOffset >= buffer2.length) {
if (dir) return -1;
else byteOffset = buffer2.length - 1;
} else if (byteOffset < 0) {
if (dir) byteOffset = 0;
else return -1;
}
if (typeof val === "string") {
val = Buffer2.from(val, encoding);
}
if (Buffer2.isBuffer(val)) {
if (val.length === 0) {
return -1;
}
return arrayIndexOf(buffer2, val, byteOffset, encoding, dir);
} else if (typeof val === "number") {
val = val & 255;
if (typeof Uint8Array.prototype.indexOf === "function") {
if (dir) {
return Uint8Array.prototype.indexOf.call(buffer2, val, byteOffset);
} else {
return Uint8Array.prototype.lastIndexOf.call(buffer2, val, byteOffset);
}
}
return arrayIndexOf(buffer2, [val], byteOffset, encoding, dir);
}
throw new TypeError("val must be string, number or Buffer");
}
function arrayIndexOf(arr, val, byteOffset, encoding, dir) {
var indexSize = 1;
var arrLength = arr.length;
var valLength = val.length;
if (encoding !== void 0) {
encoding = String(encoding).toLowerCase();
if (encoding === "ucs2" || encoding === "ucs-2" || encoding === "utf16le" || encoding === "utf-16le") {
if (arr.length < 2 || val.length < 2) {
return -1;
}
indexSize = 2;
arrLength /= 2;
valLength /= 2;
byteOffset /= 2;
}
}
function read(buf, i2) {
if (indexSize === 1) {
return buf[i2];
} else {
return buf.readUInt16BE(i2 * indexSize);
}
}
var i;
if (dir) {
var foundIndex = -1;
for (i = byteOffset; i < arrLength; i++) {
if (read(arr, i) === read(val, foundIndex === -1 ? 0 : i - foundIndex)) {
if (foundIndex === -1) foundIndex = i;
if (i - foundIndex + 1 === valLength) return foundIndex * indexSize;
} else {
if (foundIndex !== -1) i -= i - foundIndex;
foundIndex = -1;
}
}
} else {
if (byteOffset + valLength > arrLength) byteOffset = arrLength - valLength;
for (i = byteOffset; i >= 0; i--) {
var found = true;
for (var j2 = 0; j2 < valLength; j2++) {
if (read(arr, i + j2) !== read(val, j2)) {
found = false;
break;
}
}
if (found) return i;
}
}
return -1;
}
Buffer2.prototype.includes = function includes(val, byteOffset, encoding) {
return this.indexOf(val, byteOffset, encoding) !== -1;
};
Buffer2.prototype.indexOf = function indexOf(val, byteOffset, encoding) {
return bidirectionalIndexOf(this, val, byteOffset, encoding, true);
};
Buffer2.prototype.lastIndexOf = function lastIndexOf(val, byteOffset, encoding) {
return bidirectionalIndexOf(this, val, byteOffset, encoding, false);
};
function hexWrite(buf, string2, offset, length) {
offset = Number(offset) || 0;
var remaining = buf.length - offset;
if (!length) {
length = remaining;
} else {
length = Number(length);
if (length > remaining) {
length = remaining;
}
}
var strLen = string2.length;
if (length > strLen / 2) {
length = strLen / 2;
}
for (var i = 0; i < length; ++i) {
var parsed = parseInt(string2.substr(i * 2, 2), 16);
if (numberIsNaN(parsed)) return i;
buf[offset + i] = parsed;
}
return i;
}
function utf8Write(buf, string2, offset, length) {
return blitBuffer(utf8ToBytes(string2, buf.length - offset), buf, offset, length);
}
function asciiWrite(buf, string2, offset, length) {
return blitBuffer(asciiToBytes(string2), buf, offset, length);
}
function base64Write(buf, string2, offset, length) {
return blitBuffer(base64ToBytes(string2), buf, offset, length);
}
function ucs2Write(buf, string2, offset, length) {
return blitBuffer(utf16leToBytes(string2, buf.length - offset), buf, offset, length);
}
Buffer2.prototype.write = function write(string2, offset, length, encoding) {
if (offset === void 0) {
encoding = "utf8";
length = this.length;
offset = 0;
} else if (length === void 0 && typeof offset === "string") {
encoding = offset;
length = this.length;
offset = 0;
} else if (isFinite(offset)) {
offset = offset >>> 0;
if (isFinite(length)) {
length = length >>> 0;
if (encoding === void 0) encoding = "utf8";
} else {
encoding = length;
length = void 0;
}
} else {
throw new Error(
"Buffer.write(string, encoding, offset[, length]) is no longer supported"
);
}
var remaining = this.length - offset;
if (length === void 0 || length > remaining) length = remaining;
if (string2.length > 0 && (length < 0 || offset < 0) || offset > this.length) {
throw new RangeError("Attempt to write outside buffer bounds");
}
if (!encoding) encoding = "utf8";
var loweredCase = false;
for (; ; ) {
switch (encoding) {
case "hex":
return hexWrite(this, string2, offset, length);
case "utf8":
case "utf-8":
return utf8Write(this, string2, offset, length);
case "ascii":
case "latin1":
case "binary":
return asciiWrite(this, string2, offset, length);
case "base64":
return base64Write(this, string2, offset, length);
case "ucs2":
case "ucs-2":
case "utf16le":
case "utf-16le":
return ucs2Write(this, string2, offset, length);
default:
if (loweredCase) throw new TypeError("Unknown encoding: " + encoding);
encoding = ("" + encoding).toLowerCase();
loweredCase = true;
}
}
};
Buffer2.prototype.toJSON = function toJSON2() {
return {
type: "Buffer",
data: Array.prototype.slice.call(this._arr || this, 0)
};
};
function base64Slice(buf, start, end) {
if (start === 0 && end === buf.length) {
return base64.fromByteArray(buf);
} else {
return base64.fromByteArray(buf.slice(start, end));
}
}
function utf8Slice(buf, start, end) {
end = Math.min(buf.length, end);
var res = [];
var i = start;
while (i < end) {
var firstByte = buf[i];
var codePoint = null;
var bytesPerSequence = firstByte > 239 ? 4 : firstByte > 223 ? 3 : firstByte > 191 ? 2 : 1;
if (i + bytesPerSequence <= end) {
var secondByte, thirdByte, fourthByte, tempCodePoint;
switch (bytesPerSequence) {
case 1:
if (firstByte < 128) {
codePoint = firstByte;
}
break;
case 2:
secondByte = buf[i + 1];
if ((secondByte & 192) === 128) {
tempCodePoint = (firstByte & 31) << 6 | secondByte & 63;
if (tempCodePoint > 127) {
codePoint = tempCodePoint;
}
}
break;
case 3:
secondByte = buf[i + 1];
thirdByte = buf[i + 2];
if ((secondByte & 192) === 128 && (thirdByte & 192) === 128) {
tempCodePoint = (firstByte & 15) << 12 | (secondByte & 63) << 6 | thirdByte & 63;
if (tempCodePoint > 2047 && (tempCodePoint < 55296 || tempCodePoint > 57343)) {
codePoint = tempCodePoint;
}
}
break;
case 4:
secondByte = buf[i + 1];
thirdByte = buf[i + 2];
fourthByte = buf[i + 3];
if ((secondByte & 192) === 128 && (thirdByte & 192) === 128 && (fourthByte & 192) === 128) {
tempCodePoint = (firstByte & 15) << 18 | (secondByte & 63) << 12 | (thirdByte & 63) << 6 | fourthByte & 63;
if (tempCodePoint > 65535 && tempCodePoint < 1114112) {
codePoint = tempCodePoint;
}
}
}
}
if (codePoint === null) {
codePoint = 65533;
bytesPerSequence = 1;
} else if (codePoint > 65535) {
codePoint -= 65536;
res.push(codePoint >>> 10 & 1023 | 55296);
codePoint = 56320 | codePoint & 1023;
}
res.push(codePoint);
i += bytesPerSequence;
}
return decodeCodePointsArray(res);
}
var MAX_ARGUMENTS_LENGTH = 4096;
function decodeCodePointsArray(codePoints) {
var len = codePoints.length;
if (len <= MAX_ARGUMENTS_LENGTH) {
return String.fromCharCode.apply(String, codePoints);
}
var res = "";
var i = 0;
while (i < len) {
res += String.fromCharCode.apply(
String,
codePoints.slice(i, i += MAX_ARGUMENTS_LENGTH)
);
}
return res;
}
function asciiSlice(buf, start, end) {
var ret = "";
end = Math.min(buf.length, end);
for (var i = start; i < end; ++i) {
ret += String.fromCharCode(buf[i] & 127);
}
return ret;
}
function latin1Slice(buf, start, end) {
var ret = "";
end = Math.min(buf.length, end);
for (var i = start; i < end; ++i) {
ret += String.fromCharCode(buf[i]);
}
return ret;
}
function hexSlice(buf, start, end) {
var len = buf.length;
if (!start || start < 0) start = 0;
if (!end || end < 0 || end > len) end = len;
var out = "";
for (var i = start; i < end; ++i) {
out += hexSliceLookupTable[buf[i]];
}
return out;
}
function utf16leSlice(buf, start, end) {
var bytes = buf.slice(start, end);
var res = "";
for (var i = 0; i < bytes.length - 1; i += 2) {
res += String.fromCharCode(bytes[i] + bytes[i + 1] * 256);
}
return res;
}
Buffer2.prototype.slice = function slice(start, end) {
var len = this.length;
start = ~~start;
end = end === void 0 ? len : ~~end;
if (start < 0) {
start += len;
if (start < 0) start = 0;
} else if (start > len) {
start = len;
}
if (end < 0) {
end += len;
if (end < 0) end = 0;
} else if (end > len) {
end = len;
}
if (end < start) end = start;
var newBuf = this.subarray(start, end);
Object.setPrototypeOf(newBuf, Buffer2.prototype);
return newBuf;
};
function checkOffset(offset, ext, length) {
if (offset % 1 !== 0 || offset < 0) throw new RangeError("offset is not uint");
if (offset + ext > length) throw new RangeError("Trying to access beyond buffer length");
}
Buffer2.prototype.readUintLE = Buffer2.prototype.readUIntLE = function readUIntLE(offset, byteLength3, noAssert) {
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) checkOffset(offset, byteLength3, this.length);
var val = this[offset];
var mul = 1;
var i = 0;
while (++i < byteLength3 && (mul *= 256)) {
val += this[offset + i] * mul;
}
return val;
};
Buffer2.prototype.readUintBE = Buffer2.prototype.readUIntBE = function readUIntBE(offset, byteLength3, noAssert) {
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) {
checkOffset(offset, byteLength3, this.length);
}
var val = this[offset + --byteLength3];
var mul = 1;
while (byteLength3 > 0 && (mul *= 256)) {
val += this[offset + --byteLength3] * mul;
}
return val;
};
Buffer2.prototype.readUint8 = Buffer2.prototype.readUInt8 = function readUInt8(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 1, this.length);
return this[offset];
};
Buffer2.prototype.readUint16LE = Buffer2.prototype.readUInt16LE = function readUInt16LE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 2, this.length);
return this[offset] | this[offset + 1] << 8;
};
Buffer2.prototype.readUint16BE = Buffer2.prototype.readUInt16BE = function readUInt16BE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 2, this.length);
return this[offset] << 8 | this[offset + 1];
};
Buffer2.prototype.readUint32LE = Buffer2.prototype.readUInt32LE = function readUInt32LE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return (this[offset] | this[offset + 1] << 8 | this[offset + 2] << 16) + this[offset + 3] * 16777216;
};
Buffer2.prototype.readUint32BE = Buffer2.prototype.readUInt32BE = function readUInt32BE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return this[offset] * 16777216 + (this[offset + 1] << 16 | this[offset + 2] << 8 | this[offset + 3]);
};
Buffer2.prototype.readIntLE = function readIntLE(offset, byteLength3, noAssert) {
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) checkOffset(offset, byteLength3, this.length);
var val = this[offset];
var mul = 1;
var i = 0;
while (++i < byteLength3 && (mul *= 256)) {
val += this[offset + i] * mul;
}
mul *= 128;
if (val >= mul) val -= Math.pow(2, 8 * byteLength3);
return val;
};
Buffer2.prototype.readIntBE = function readIntBE(offset, byteLength3, noAssert) {
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) checkOffset(offset, byteLength3, this.length);
var i = byteLength3;
var mul = 1;
var val = this[offset + --i];
while (i > 0 && (mul *= 256)) {
val += this[offset + --i] * mul;
}
mul *= 128;
if (val >= mul) val -= Math.pow(2, 8 * byteLength3);
return val;
};
Buffer2.prototype.readInt8 = function readInt8(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 1, this.length);
if (!(this[offset] & 128)) return this[offset];
return (255 - this[offset] + 1) * -1;
};
Buffer2.prototype.readInt16LE = function readInt16LE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 2, this.length);
var val = this[offset] | this[offset + 1] << 8;
return val & 32768 ? val | 4294901760 : val;
};
Buffer2.prototype.readInt16BE = function readInt16BE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 2, this.length);
var val = this[offset + 1] | this[offset] << 8;
return val & 32768 ? val | 4294901760 : val;
};
Buffer2.prototype.readInt32LE = function readInt32LE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return this[offset] | this[offset + 1] << 8 | this[offset + 2] << 16 | this[offset + 3] << 24;
};
Buffer2.prototype.readInt32BE = function readInt32BE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return this[offset] << 24 | this[offset + 1] << 16 | this[offset + 2] << 8 | this[offset + 3];
};
Buffer2.prototype.readFloatLE = function readFloatLE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return ieee754$1.read(this, offset, true, 23, 4);
};
Buffer2.prototype.readFloatBE = function readFloatBE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 4, this.length);
return ieee754$1.read(this, offset, false, 23, 4);
};
Buffer2.prototype.readDoubleLE = function readDoubleLE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 8, this.length);
return ieee754$1.read(this, offset, true, 52, 8);
};
Buffer2.prototype.readDoubleBE = function readDoubleBE(offset, noAssert) {
offset = offset >>> 0;
if (!noAssert) checkOffset(offset, 8, this.length);
return ieee754$1.read(this, offset, false, 52, 8);
};
function checkInt(buf, value, offset, ext, max, min) {
if (!Buffer2.isBuffer(buf)) throw new TypeError('"buffer" argument must be a Buffer instance');
if (value > max || value < min) throw new RangeError('"value" argument is out of bounds');
if (offset + ext > buf.length) throw new RangeError("Index out of range");
}
Buffer2.prototype.writeUintLE = Buffer2.prototype.writeUIntLE = function writeUIntLE(value, offset, byteLength3, noAssert) {
value = +value;
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) {
var maxBytes = Math.pow(2, 8 * byteLength3) - 1;
checkInt(this, value, offset, byteLength3, maxBytes, 0);
}
var mul = 1;
var i = 0;
this[offset] = value & 255;
while (++i < byteLength3 && (mul *= 256)) {
this[offset + i] = value / mul & 255;
}
return offset + byteLength3;
};
Buffer2.prototype.writeUintBE = Buffer2.prototype.writeUIntBE = function writeUIntBE(value, offset, byteLength3, noAssert) {
value = +value;
offset = offset >>> 0;
byteLength3 = byteLength3 >>> 0;
if (!noAssert) {
var maxBytes = Math.pow(2, 8 * byteLength3) - 1;
checkInt(this, value, offset, byteLength3, maxBytes, 0);
}
var i = byteLength3 - 1;
var mul = 1;
this[offset + i] = value & 255;
while (--i >= 0 && (mul *= 256)) {
this[offset + i] = value / mul & 255;
}
return offset + byteLength3;
};
Buffer2.prototype.writeUint8 = Buffer2.prototype.writeUInt8 = function writeUInt8(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 1, 255, 0);
this[offset] = value & 255;
return offset + 1;
};
Buffer2.prototype.writeUint16LE = Buffer2.prototype.writeUInt16LE = function writeUInt16LE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 2, 65535, 0);
this[offset] = value & 255;
this[offset + 1] = value >>> 8;
return offset + 2;
};
Buffer2.prototype.writeUint16BE = Buffer2.prototype.writeUInt16BE = function writeUInt16BE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 2, 65535, 0);
this[offset] = value >>> 8;
this[offset + 1] = value & 255;
return offset + 2;
};
Buffer2.prototype.writeUint32LE = Buffer2.prototype.writeUInt32LE = function writeUInt32LE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 4, 4294967295, 0);
this[offset + 3] = value >>> 24;
this[offset + 2] = value >>> 16;
this[offset + 1] = value >>> 8;
this[offset] = value & 255;
return offset + 4;
};
Buffer2.prototype.writeUint32BE = Buffer2.prototype.writeUInt32BE = function writeUInt32BE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 4, 4294967295, 0);
this[offset] = value >>> 24;
this[offset + 1] = value >>> 16;
this[offset + 2] = value >>> 8;
this[offset + 3] = value & 255;
return offset + 4;
};
Buffer2.prototype.writeIntLE = function writeIntLE(value, offset, byteLength3, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) {
var limit = Math.pow(2, 8 * byteLength3 - 1);
checkInt(this, value, offset, byteLength3, limit - 1, -limit);
}
var i = 0;
var mul = 1;
var sub = 0;
this[offset] = value & 255;
while (++i < byteLength3 && (mul *= 256)) {
if (value < 0 && sub === 0 && this[offset + i - 1] !== 0) {
sub = 1;
}
this[offset + i] = (value / mul >> 0) - sub & 255;
}
return offset + byteLength3;
};
Buffer2.prototype.writeIntBE = function writeIntBE(value, offset, byteLength3, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) {
var limit = Math.pow(2, 8 * byteLength3 - 1);
checkInt(this, value, offset, byteLength3, limit - 1, -limit);
}
var i = byteLength3 - 1;
var mul = 1;
var sub = 0;
this[offset + i] = value & 255;
while (--i >= 0 && (mul *= 256)) {
if (value < 0 && sub === 0 && this[offset + i + 1] !== 0) {
sub = 1;
}
this[offset + i] = (value / mul >> 0) - sub & 255;
}
return offset + byteLength3;
};
Buffer2.prototype.writeInt8 = function writeInt8(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 1, 127, -128);
if (value < 0) value = 255 + value + 1;
this[offset] = value & 255;
return offset + 1;
};
Buffer2.prototype.writeInt16LE = function writeInt16LE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 2, 32767, -32768);
this[offset] = value & 255;
this[offset + 1] = value >>> 8;
return offset + 2;
};
Buffer2.prototype.writeInt16BE = function writeInt16BE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 2, 32767, -32768);
this[offset] = value >>> 8;
this[offset + 1] = value & 255;
return offset + 2;
};
Buffer2.prototype.writeInt32LE = function writeInt32LE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 4, 2147483647, -2147483648);
this[offset] = value & 255;
this[offset + 1] = value >>> 8;
this[offset + 2] = value >>> 16;
this[offset + 3] = value >>> 24;
return offset + 4;
};
Buffer2.prototype.writeInt32BE = function writeInt32BE(value, offset, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) checkInt(this, value, offset, 4, 2147483647, -2147483648);
if (value < 0) value = 4294967295 + value + 1;
this[offset] = value >>> 24;
this[offset + 1] = value >>> 16;
this[offset + 2] = value >>> 8;
this[offset + 3] = value & 255;
return offset + 4;
};
function checkIEEE754(buf, value, offset, ext, max, min) {
if (offset + ext > buf.length) throw new RangeError("Index out of range");
if (offset < 0) throw new RangeError("Index out of range");
}
function writeFloat(buf, value, offset, littleEndian, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) {
checkIEEE754(buf, value, offset, 4);
}
ieee754$1.write(buf, value, offset, littleEndian, 23, 4);
return offset + 4;
}
Buffer2.prototype.writeFloatLE = function writeFloatLE(value, offset, noAssert) {
return writeFloat(this, value, offset, true, noAssert);
};
Buffer2.prototype.writeFloatBE = function writeFloatBE(value, offset, noAssert) {
return writeFloat(this, value, offset, false, noAssert);
};
function writeDouble(buf, value, offset, littleEndian, noAssert) {
value = +value;
offset = offset >>> 0;
if (!noAssert) {
checkIEEE754(buf, value, offset, 8);
}
ieee754$1.write(buf, value, offset, littleEndian, 52, 8);
return offset + 8;
}
Buffer2.prototype.writeDoubleLE = function writeDoubleLE(value, offset, noAssert) {
return writeDouble(this, value, offset, true, noAssert);
};
Buffer2.prototype.writeDoubleBE = function writeDoubleBE(value, offset, noAssert) {
return writeDouble(this, value, offset, false, noAssert);
};
Buffer2.prototype.copy = function copy(target, targetStart, start, end) {
if (!Buffer2.isBuffer(target)) throw new TypeError("argument should be a Buffer");
if (!start) start = 0;
if (!end && end !== 0) end = this.length;
if (targetStart >= target.length) targetStart = target.length;
if (!targetStart) targetStart = 0;
if (end > 0 && end < start) end = start;
if (end === start) return 0;
if (target.length === 0 || this.length === 0) return 0;
if (targetStart < 0) {
throw new RangeError("targetStart out of bounds");
}
if (start < 0 || start >= this.length) throw new RangeError("Index out of range");
if (end < 0) throw new RangeError("sourceEnd out of bounds");
if (end > this.length) end = this.length;
if (target.length - targetStart < end - start) {
end = target.length - targetStart + start;
}
var len = end - start;
if (this === target && typeof Uint8Array.prototype.copyWithin === "function") {
this.copyWithin(targetStart, start, end);
} else {
Uint8Array.prototype.set.call(
target,
this.subarray(start, end),
targetStart
);
}
return len;
};
Buffer2.prototype.fill = function fill(val, start, end, encoding) {
if (typeof val === "string") {
if (typeof start === "string") {
encoding = start;
start = 0;
end = this.length;
} else if (typeof end === "string") {
encoding = end;
end = this.length;
}
if (encoding !== void 0 && typeof encoding !== "string") {
throw new TypeError("encoding must be a string");
}
if (typeof encoding === "string" && !Buffer2.isEncoding(encoding)) {
throw new TypeError("Unknown encoding: " + encoding);
}
if (val.length === 1) {
var code2 = val.charCodeAt(0);
if (encoding === "utf8" && code2 < 128 || encoding === "latin1") {
val = code2;
}
}
} else if (typeof val === "number") {
val = val & 255;
} else if (typeof val === "boolean") {
val = Number(val);
}
if (start < 0 || this.length < start || this.length < end) {
throw new RangeError("Out of range index");
}
if (end <= start) {
return this;
}
start = start >>> 0;
end = end === void 0 ? this.length : end >>> 0;
if (!val) val = 0;
var i;
if (typeof val === "number") {
for (i = start; i < end; ++i) {
this[i] = val;
}
} else {
var bytes = Buffer2.isBuffer(val) ? val : Buffer2.from(val, encoding);
var len = bytes.length;
if (len === 0) {
throw new TypeError('The value "' + val + '" is invalid for argument "value"');
}
for (i = 0; i < end - start; ++i) {
this[i + start] = bytes[i % len];
}
}
return this;
};
var INVALID_BASE64_RE = /[^+/0-9A-Za-z-_]/g;
function base64clean(str) {
str = str.split("=")[0];
str = str.trim().replace(INVALID_BASE64_RE, "");
if (str.length < 2) return "";
while (str.length % 4 !== 0) {
str = str + "=";
}
return str;
}
function utf8ToBytes(string2, units) {
units = units || Infinity;
var codePoint;
var length = string2.length;
var leadSurrogate = null;
var bytes = [];
for (var i = 0; i < length; ++i) {
codePoint = string2.charCodeAt(i);
if (codePoint > 55295 && codePoint < 57344) {
if (!leadSurrogate) {
if (codePoint > 56319) {
if ((units -= 3) > -1) bytes.push(239, 191, 189);
continue;
} else if (i + 1 === length) {
if ((units -= 3) > -1) bytes.push(239, 191, 189);
continue;
}
leadSurrogate = codePoint;
continue;
}
if (codePoint < 56320) {
if ((units -= 3) > -1) bytes.push(239, 191, 189);
leadSurrogate = codePoint;
continue;
}
codePoint = (leadSurrogate - 55296 << 10 | codePoint - 56320) + 65536;
} else if (leadSurrogate) {
if ((units -= 3) > -1) bytes.push(239, 191, 189);
}
leadSurrogate = null;
if (codePoint < 128) {
if ((units -= 1) < 0) break;
bytes.push(codePoint);
} else if (codePoint < 2048) {
if ((units -= 2) < 0) break;
bytes.push(
codePoint >> 6 | 192,
codePoint & 63 | 128
);
} else if (codePoint < 65536) {
if ((units -= 3) < 0) break;
bytes.push(
codePoint >> 12 | 224,
codePoint >> 6 & 63 | 128,
codePoint & 63 | 128
);
} else if (codePoint < 1114112) {
if ((units -= 4) < 0) break;
bytes.push(
codePoint >> 18 | 240,
codePoint >> 12 & 63 | 128,
codePoint >> 6 & 63 | 128,
codePoint & 63 | 128
);
} else {
throw new Error("Invalid code point");
}
}
return bytes;
}
function asciiToBytes(str) {
var byteArray = [];
for (var i = 0; i < str.length; ++i) {
byteArray.push(str.charCodeAt(i) & 255);
}
return byteArray;
}
function utf16leToBytes(str, units) {
var c2, hi, lo;
var byteArray = [];
for (var i = 0; i < str.length; ++i) {
if ((units -= 2) < 0) break;
c2 = str.charCodeAt(i);
hi = c2 >> 8;
lo = c2 % 256;
byteArray.push(lo);
byteArray.push(hi);
}
return byteArray;
}
function base64ToBytes(str) {
return base64.toByteArray(base64clean(str));
}
function blitBuffer(src, dst, offset, length) {
for (var i = 0; i < length; ++i) {
if (i + offset >= dst.length || i >= src.length) break;
dst[i + offset] = src[i];
}
return i;
}
function isInstance(obj, type) {
return obj instanceof type || obj != null && obj.constructor != null && obj.constructor.name != null && obj.constructor.name === type.name;
}
function numberIsNaN(obj) {
return obj !== obj;
}
var hexSliceLookupTable = function() {
var alphabet2 = "0123456789abcdef";
var table = new Array(256);
for (var i = 0; i < 16; ++i) {
var i16 = i * 16;
for (var j2 = 0; j2 < 16; ++j2) {
table[i16 + j2] = alphabet2[i] + alphabet2[j2];
}
}
return table;
}();
})(buffer);
class GazelleMusic extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName.match(/(DicMusic|Orpheus)/) !== null;
}
async fill(info) {
this.info = info;
const { musicJson } = info;
if (!musicJson) {
return;
}
const { name, year } = musicJson.group;
const groupId = getLocationSearchValueByKey("groupid");
if (!groupId) {
await this.searchAndRedirectToGroup(name, year);
}
await this.processSiteSpecificLogic(musicJson);
this.fillJsonToUploadTable(musicJson, name);
this.fillTorrentFile();
}
async searchAndRedirectToGroup(name, year) {
var _a2;
try {
const searchResult = await GMFetch(
`/ajax.php?action=browse&searchstr=${encodeURIComponent(name)} ${year}`,
{
responseType: "json"
}
);
if (searchResult.status === "success" && searchResult.response.results.length > 0) {
const groupId = searchResult.response.results[0].groupId;
const timestampMatchArray = location.hash && location.hash.match(/(^|#)timestamp=([^#]*)(#|$)/);
const timestamp2 = (_a2 = timestampMatchArray == null ? void 0 : timestampMatchArray[2]) != null ? _a2 : "";
location.href = `${CURRENT_SITE_INFO.url}${CURRENT_SITE_INFO.uploadPath}?groupid=${groupId}#timestamp=${timestamp2}`;
return true;
}
} catch (error) {
console.error("Error searching for group:", error);
}
return false;
}
async processSiteSpecificLogic(musicJson) {
if (CURRENT_SITE_NAME === "Orpheus") {
const { remasterTitle, remasterCatalogueNumber, remasterRecordLabel } = musicJson.torrent;
const { recordLabel, catalogueNumber } = musicJson.group;
if (!remasterCatalogueNumber && !remasterRecordLabel && !remasterTitle && !recordLabel && !catalogueNumber) {
musicJson.torrent.remastered = false;
}
} else if (CURRENT_SITE_NAME === "DicMusic") {
musicJson.group.wikiBody = this.toUnicodeEntities(
musicJson.group.wikiBody
);
}
}
fillJsonToUploadTable(musicJson, name) {
const buf = buffer.Buffer.from(
JSON.stringify({
status: "success",
response: musicJson
})
);
this.attachFile({
data: buf,
selector: "#torrent-json-file",
contentType: "application/json",
fileName: name,
format: "json"
});
}
attachFile(options2) {
const {
data,
selector,
contentType,
fileName,
format: format2,
charset = "UTF-8"
} = options2;
const buf = buffer.Buffer.isBuffer(data) ? data : buffer.Buffer.from(data, charset);
const base64Data = buf.toString("base64");
const fileInput = $$2(selector);
if (base64Data && fileInput.length > 0) {
const blob = base64ToBlob(base64Data, contentType);
const file = new File([blob], `${fileName}.${format2}`, {
type: contentType
});
const dataTransfer = new DataTransfer();
dataTransfer.items.add(file);
const uploadInput = fileInput[0];
uploadInput.files = dataTransfer.files;
uploadInput.dispatchEvent(new Event("change", { bubbles: true }));
}
}
toUnicodeEntities(str) {
const excludedChars = ["<", ">", "&", ";", "/"];
return str.split("").map((char) => {
const code2 = char.charCodeAt(0);
if (code2 > 127 && !excludedChars.includes(char)) {
const hexCode = code2.toString(16);
return `${parseInt(hexCode, 16)};`;
}
return char;
}).join("");
}
}
registry$1.register(new GazelleMusic());
class GPW extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
this.currentSiteInfo = PT_SITE.GPW;
}
canHandle(siteName) {
return siteName === "GPW";
}
fill(info) {
this.info = info;
const isUploadSuccess = !$$2("#mediainfo")[0];
if (isUploadSuccess) {
return;
}
this.transformInfo();
const isAddFormat = getLocationSearchValueByKey("groupid");
if (!isAddFormat) {
this.fillIMDbAndTriggerAutoFill();
}
this.fillCategory();
this.fillEditionInfo();
this.fillMediaInfos();
setTimeout(() => {
this.handleNoAutoCheck();
}, 0);
this.fillScene();
this.fillProcessing();
this.fillDescription();
this.fillTorrentFile();
$$2(".u-bbcodePreview-button").trigger("click");
}
fillIMDbAndTriggerAutoFill() {
if (!this.info) return;
$$2(this.currentSiteInfo.imdb.selector).val(this.info.imdbUrl || 0);
$$2("#imdb_button").trigger("click");
$$2("#upload .collapse").show();
}
fillCategory() {
if (!this.info) return;
$$2(this.currentSiteInfo.category.selector).val(
this.currentSiteInfo.category.map[this.info.category]
);
}
fillEditionInfo() {
if (!this.info) return;
const editionTags = Object.keys(this.info.tags).map(
(tag) => this.info.tags[tag] && this.currentSiteInfo.targetInfo.editionTags[tag]
).filter(Boolean);
let otherTag;
if (this.info.otherTags && Object.keys(this.info.otherTags).length > 0) {
otherTag = Object.keys(this.info.otherTags).join(", ");
}
if (editionTags.length > 0 || otherTag) {
$$2("#movie_edition_information").trigger("click");
}
if (editionTags.length > 0) {
for (const tag of editionTags) {
$$2(`#movie_remaster_tags a[onclick*="'${tag}'"]`).trigger("click");
}
}
if (otherTag) {
$$2("#other-button").trigger("click");
$$2("[name=remaster_custom_title]").val(otherTag);
}
}
fillMediaInfos() {
if (!this.info || !this.info.mediaInfos) {
return;
}
for (let i = 1; i < this.info.mediaInfos.length; i++) {
$$2("#add-mediainfo")[0].click();
}
const textareas = Array.from($$2('[name="mediainfo[]"]'));
for (const [index, textarea] of textareas.entries()) {
textarea.value = this.info.mediaInfos[index];
textarea.dispatchEvent(new Event("input"));
}
$$2('[name="mediainfo[]"]')[0].dispatchEvent(new Event("change"));
}
handleNoAutoCheck() {
if (!this.info) return;
if (!$$2(this.currentSiteInfo.source.selector).val() || !$$2(this.currentSiteInfo.format.selector).val()) {
const { mediaInfos, videoType } = this.info;
const mediaInfo = (mediaInfos == null ? void 0 : mediaInfos[0]) || "";
const isBluray = !!videoType.match(/bluray/i);
const parsedMedia = parseMedia(mediaInfo, isBluray);
if (!parsedMedia) {
console.warn("Failed to parse media info:", mediaInfo);
return;
}
const { format: format2 = "", subtitleTracks = [] } = parsedMedia;
this.info.format = this.getFormat(format2, videoType);
const keyArray = [
"source",
"videoCodec",
"format",
"resolution"
];
keyArray.forEach((key) => {
var _a2, _b;
const { selector = "", map: map2 } = this.currentSiteInfo[key];
if (map2) {
const mapValue = map2[this.info[key]];
$$2(selector).val(mapValue);
if (key === "videoCodec" && mapValue === "Other") {
document.querySelector(selector).dispatchEvent(new Event("change"));
$$2('input[name="codec_other"]').val(
(_b = (_a2 = this.info[key]) == null ? void 0 : _a2.toUpperCase()) != null ? _b : ""
);
}
} else {
$$2(selector).val(this.info[key] || "");
}
});
this.fillSubtitles(subtitleTracks.map((track) => track.language));
}
}
fillSubtitles(subtitles) {
var _a2;
if (subtitles.length > 0) {
$$2("#mixed_subtitles").attr("checked", "true");
$$2('input[name="subtitles[]"][type="checkbox"]').each(function() {
var _a3, _b;
const language = (_b = (_a3 = $$2(this).attr("id")) == null ? void 0 : _a3.replace(
/^\S|(_\S)/g,
(letter) => letter.replace("_", " ").toUpperCase()
)) != null ? _b : "";
if (subtitles.includes(language)) {
$$2(this).attr("checked", "true");
}
});
const event = new Event("change");
(_a2 = document.querySelector("#mixed_subtitles")) == null ? void 0 : _a2.dispatchEvent(event);
const chineseLanguages = subtitles.filter(
(item) => item.match(/Chinese|Traditional|Simplified/i)
);
if (chineseLanguages.length === 1 && chineseLanguages[0] === "Chinese") {
const selector = chineseLanguages[0].match(/Traditional/i) ? "#chinese_traditional" : "#chinese_simplified";
$$2(selector).attr("checked", "true");
} else if (chineseLanguages.length >= 2) {
$$2("#chinese_traditional,#chinese_simplified").attr("checked", "true");
}
}
}
fillScene() {
if (!this.info) return;
if (this.info.tags.scene) {
$$2("#scene").prop("checked", true);
}
}
fillProcessing() {
if (!this.info) return;
let { videoType, size, source, tags: tags2 } = this.info;
if (source.match(/bluray|hddvd|dvd/)) {
if (tags2.diy) {
videoType = "DIY";
}
const videoTypeConfig = this.currentSiteInfo.videoType;
const { selector, map: map2 } = videoTypeConfig;
$$2(selector).val(map2[videoType]);
$$2(selector)[0].dispatchEvent(new Event("change"));
if (map2[videoType] === "Untouched") {
const bdType = getBDTypeBasedOnSize(size);
$$2('select[name="processing_other"]').val(bdType || "");
}
$$2(selector)[0].dispatchEvent(new Event("input"));
}
}
fillDescription() {
if (!this.info) return;
let description = "";
if (this.info.sourceSite === "PTP") {
description = buildPTPDescription(this.info);
} else if (this.info.sourceSite === "BeyondHD") {
description = this.info.originalDescription || "";
} else {
description = this.buildDescription();
}
$$2(this.currentSiteInfo.description.selector).val(description);
$$2(this.currentSiteInfo.description.selector)[0].dispatchEvent(
new Event("input")
);
}
buildDescription() {
if (!this.info) return "";
let description = "";
if (this.info.comparisons && this.info.comparisons.length > 0) {
for (const comparison of this.info.comparisons) {
description += `${comparison.reason}[comparison=${comparison.title}]
${comparison.imgs.join("\n")}
[/comparison]
`;
}
}
if (this.info.screenshots.length > 0) {
description += `${this.info.screenshots.map((v2) => `[img]${v2}[/img]`).join("\n")}
`;
}
return description.trim();
}
transformInfo() {
if (!this.info) return;
if (["encode", "remux"].includes(this.info.videoType) && this.info.mediaInfos) {
const newMediaInfos = [];
for (const mediaInfo of this.info.mediaInfos) {
if (mediaInfo.match(/Disc Title|Disc Label/i)) {
continue;
}
newMediaInfos.push(mediaInfo);
}
this.info.mediaInfos = newMediaInfos;
}
}
getFormat(format2, videoType) {
if (videoType.match(/bluray/) && format2 !== "iso") {
format2 = "m2ts";
} else if (videoType.match(/dvd/)) {
format2 = "vob";
}
return format2 || "mkv";
}
}
registry$1.register(new GPW());
class ZHUQUE extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
this.currentSiteInfo = PT_SITE.ZHUQUE;
}
canHandle(siteName) {
return siteName === "ZHUQUE";
}
fill(info) {
this.info = info;
this.setupMutationObserver();
}
setupMutationObserver() {
if (!this.info) return;
const targetNode = document;
const imdbId = getIdByIMDbUrl(this.info.imdbUrl || "");
const insert = new MutationObserver(() => {
this.fillBasicFields(imdbId);
this.fillZhuqueScreenshots();
this.fillZhuqueMediaInfo();
this.fillZhuqueDescription();
this.setupSelectObserver();
this.fillTorrentFile();
insert.disconnect();
});
insert.observe(targetNode, {
attributes: false,
childList: false,
subtree: true,
characterDataOldValue: false
});
}
fillBasicFields(imdbId) {
if (!this.info) return;
$$2("input.ant-select-selection-search-input[id]").each(function() {
this.dispatchEvent(new Event("keydown"));
});
$$2(this.currentSiteInfo.name.selector).val(this.info.title);
$$2(this.currentSiteInfo.name.selector)[0].dispatchEvent(new Event("input"));
$$2(this.currentSiteInfo.imdb.selector).val(imdbId);
$$2(this.currentSiteInfo.imdb.selector)[0].dispatchEvent(new Event("input"));
if (this.info.subtitle) {
$$2(this.currentSiteInfo.subtitle.selector).val(this.info.subtitle);
$$2(this.currentSiteInfo.subtitle.selector)[0].dispatchEvent(
new Event("input")
);
}
$$2(`.ant-btn-primary span:contains(查 询)`).trigger("click");
}
fillZhuqueScreenshots() {
if (!this.info) return;
let screenshotStr = "";
if (this.info.screenshots.length > 0) {
this.info.screenshots.forEach((img) => {
screenshotStr += `${img}
`;
});
}
$$2(this.currentSiteInfo.screenshots.selector).val(screenshotStr);
$$2(this.currentSiteInfo.screenshots.selector)[0].dispatchEvent(
new Event("input")
);
}
fillZhuqueMediaInfo() {
var _a2, _b;
if (!this.info) return;
$$2(this.currentSiteInfo.mediaInfo.selector).val(
(_b = (_a2 = this.info.mediaInfos) == null ? void 0 : _a2[0]) != null ? _b : ""
);
$$2(this.currentSiteInfo.mediaInfo.selector)[0].dispatchEvent(
new Event("input")
);
}
fillZhuqueDescription() {
var _a2, _b;
if (!this.info) return;
let description = this.info.description.replace(`[quote]${(_b = (_a2 = this.info.mediaInfos) == null ? void 0 : _a2[0]) != null ? _b : ""}[/quote]`, "").trim();
if (this.info.mediaInfos && this.info.mediaInfos.length > 1) {
this.info.mediaInfos.forEach((mediaInfo) => {
description = description.replace(`[quote]${mediaInfo}[/quote]`, "");
});
}
if (this.info.sourceSite === "PTP") {
description = buildPTPDescription(this.info);
description = description.replace(
/\[comparison[^[]*\[\/comparison\]/gi,
""
);
} else if (this.info.sourceSite.match(/BeyondHD|UHDBits/)) {
description = this.info.originalDescription || "";
}
description = description.replace(/\[url.*\[\/url\]/g, "").replace(/\[img.*\[\/img\]/g, "").replace(/\[\/?(i|b|center|quote|size|color)\]/g, "").replace(/\[(size|color)=#?[a-zA-Z0-9]*\]/g, "").replace(/\n\n*/g, "\n");
description = description.replace(/Screen(shot)?s:(\s*)\n?/gi, "").trim();
if (this.info.sourceSite === "KEEPFRDS") {
description = description.replace(/截图对比:[^\n]*\n?/gi, "");
}
if (description !== "") {
description = `\`\`\`
${description}
\`\`\``;
}
if (this.info.comparisons && this.info.comparisons.length > 0) {
for (const comparison in this.info.comparisons) {
description += `
对比图 ${this.info.comparisons[comparison].title}
`;
for (const img in this.info.comparisons[comparison].imgs) {
description += `${this.info.comparisons[comparison].imgs[img]}
`;
}
}
}
const thanksQuoteClosed = GM_getValue(
"easy-upload.thanks-quote-closed",
""
);
if (!thanksQuoteClosed && this.info.sourceSite !== void 0) {
description = `\`\`\`
转自 ${this.info.sourceSite} ,感谢原发布者!
\`\`\`
${description}`;
}
$$2(this.currentSiteInfo.description.selector).val(description);
$$2(this.currentSiteInfo.description.selector)[0].dispatchEvent(
new Event("input")
);
}
setupSelectObserver() {
if (!this.info) return;
const selectNodeParent = document.querySelector("form");
if (!selectNodeParent) return;
const select = new MutationObserver(async () => {
this.fillCategory();
this.fillVideoAndAudioParameters();
select.disconnect();
});
select.observe(selectNodeParent, {
attributes: false,
childList: true,
subtree: true,
characterDataOldValue: false
});
}
fillCategory() {
if (!this.info) return;
const { category: categoryConfig } = this.currentSiteInfo;
$$2(
`div.ant-select-item-option-content:contains(${categoryConfig.map[this.info.category]})`
).trigger("click");
}
fillVideoAndAudioParameters() {
var _a2;
if (!this.info) return;
this.fillTags();
const keyArray = ["videoType", "videoCodec", "audioCodec"];
keyArray.forEach((key) => {
const { map: map2 } = this.currentSiteInfo[key];
if (!map2) return;
const mapValue = map2[this.info[key]];
if (!mapValue) return;
this.clickAppropriateOption(key, mapValue);
});
if (this.info.resolution !== "") {
(_a2 = $$2(
`div.ant-select-item-option-content:contains(${this.info.resolution})`
)[0]) == null ? void 0 : _a2.click();
}
}
async fillTags() {
if (!this.info) return;
const sleep = (ms) => {
return new Promise((resolve28) => setTimeout(resolve28, ms));
};
const { tags: tags2 } = this.currentSiteInfo;
for (const tag in this.info.tags) {
if (tags2[tag]) {
await sleep(100).then(
() => $$2(tags2[tag])[0].click()
);
}
}
}
clickAppropriateOption(key, mapValue) {
if (key !== "videoType" && $$2(`div.ant-select-item-option-content:contains(${mapValue})`).length > 0) {
$$2(`div.ant-select-item-option-content:contains(${mapValue})`)[0].click();
} else if (mapValue === "Blu-ray") {
$$2(`div.ant-select-item-option-content:contains(${mapValue})`)[2].click();
} else if ($$2(`div.ant-select-item-option-content:contains(${mapValue})`).length > 0) {
$$2(`div.ant-select-item-option-content:contains(${mapValue})`)[0].click();
}
}
}
registry$1.register(new ZHUQUE());
class ITS extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "iTS";
}
async fill(info) {
this.info = info;
if (!this.info) return;
this.processTorrentTitle();
this.fillTorrentTitle();
await this.generateAndFillDescription();
this.handleCategoryAndCollection();
this.postProcess();
}
processTorrentTitle() {
super.processTorrentTitle();
const { title } = this.info;
this.info.title = title.replace(/\s/gi, ".");
}
postProcess() {
const { category, title, doubanUrl, imdbUrl, screenshots, subtitle } = this.info;
if (category === "tvPack" || title.match(/Trilogy|Collection/i) || subtitle && subtitle.match(/合集/)) {
$$2('input[name="pack"]').attr("checked", "true");
}
$$2(this.siteInfo.imdb.selector).val(doubanUrl || imdbUrl);
$$2(this.siteInfo.screenshots.selector).val(screenshots.join("\n"));
}
handleCategoryAndCollection() {
if (!this.info) return;
const { resolution, category } = this.info;
if (!resolution.match(/2160|1080|720/) && category === "movie") {
$$2('select[name="type"]').val("67");
}
const collectionMap = this.getCollectionMap();
this.fillCollections(collectionMap);
}
getCollectionMap() {
const collectionMap = {};
$$2('select[name="collection_id1"] option').each(function() {
const option = $$2(this);
collectionMap[option.text()] = option.val();
});
return collectionMap;
}
async fillCollections(collectionMap) {
var _a2, _b, _c;
if (!this.info || !this.info.imdbUrl) return;
const { category } = this.info;
const teamName = getTeamName(this.info.title);
const collectionValues = [];
if (collectionMap[teamName]) {
collectionValues.push(collectionMap[teamName]);
}
if (category.match(/tv|movie|cartoon|documentary/)) {
try {
const imdbData = await getIMDBData(this.info.imdbUrl);
if (imdbData) {
const { directors = [], details } = imdbData;
let language = details.Languages || "";
language = (_c = (_b = (_a2 = language == null ? void 0 : language.split("|")) == null ? void 0 : _a2[0]) == null ? void 0 : _b.trim()) != null ? _c : "";
const director = directors.map((item) => item.name)[0];
if (collectionMap[director]) {
collectionValues.push(collectionMap[director]);
}
if (collectionMap[language]) {
collectionValues.push(collectionMap[language]);
}
}
} catch (error) {
console.error("Error fetching IMDB data:", error);
}
}
collectionValues.forEach((value, index) => {
$$2(`select[name="collection_id${index + 1}"]`).val(value);
});
}
async generateAndFillDescription() {
if (!this.info) return;
const { category, screenshots, comparisons = [] } = this.info;
if (!category.match(/tv|movie|cartoon|documentary/)) return;
let template = this.createDescriptionTemplate();
template = this.fillScreenshotsInTemplate(template, screenshots);
template = this.fillComparisonsInTemplate(template, comparisons);
$$2('textarea[name="descr"]').val($t$1("数据加载中..."));
try {
template = await this.fillTemplateWithMetadata(template);
$$2('textarea[name="descr"]').val(template);
} catch (error) {
$$2('textarea[name="descr"]').val(error.message);
}
}
createDescriptionTemplate() {
return `[center]
[img]$poster$[/img]
[url=$imdbUrl$][img]https://i.ibb.co/KD855ZM/IMDb-Logo-2016.png[/img][/url][size=3]$imdbScore$[/size][*][url=$rtUrl$][img]https://i.ibb.co/BwtmdcV/rottentomatoes-logo.png[/img][/url][size=3]$rtScore$[/size][*][size=3][url=$tmdbUrl$][img]https://i.ibb.co/HhgF1gC/tmdb-logo.png[/img][/url]$tmdbScore$[/size]
[color=DarkOrange][size=2]◢ SYNOPSIS ◣[/size][/color]
$synopsis$
[color=DarkOrange][size=2]◢ TRAILER ◣[/size][/color]
[youtube]$youtubeUrl$[/youtube]
[color=DarkOrange][size=2]◢ SCREENSHOTS ◣[/size][/color]
[box][hide]$SCREENSHOTS$[/hide][/box]
[/center]`;
}
fillScreenshotsInTemplate(template, screenshots) {
const screenshotsBBCode = screenshots.map((img) => `[img]${img}[/img]`);
return template.replace("$SCREENSHOTS$", screenshotsBBCode.join("\n"));
}
fillComparisonsInTemplate(template, comparisons) {
const comparisonImgs = comparisons.flatMap((v2) => v2.imgs);
if (comparisonImgs.length > 0) {
const comparisonImgsBBCode = comparisonImgs.map(
(img) => `[img]${img}[/img]`
);
return template.replace(
/(\[\/center\])$/,
`[color=DarkOrange][size=2]◢ COMPARISONS ◣[/size][/color]
[box][hide]${comparisonImgsBBCode.join(" ")}[/hide][/box]
$1`
);
}
return template;
}
async fillTemplateWithMetadata(template) {
if (!this.info || !this.info.imdbUrl) return template;
const { imdbUrl } = this.info;
const replaceParams = {
tmdbUrl: "",
tmdbScore: "0",
imdbScore: "0",
imdbUrl,
poster: "",
synopsis: "",
rtUrl: "",
rtScore: "0",
youtubeUrl: ""
};
await this.fillIMDBData(replaceParams);
await this.fillTMDBData(replaceParams);
Object.keys(replaceParams).forEach((key) => {
template = template.replace(
`$${key}$`,
replaceParams[key] || ""
);
});
return template;
}
async fillIMDBData(replaceParams) {
var _a2, _b;
if (!this.info || !this.info.imdbUrl) return;
const { category, movieName } = this.info;
try {
const imdbData = await getIMDBData(this.info.imdbUrl);
if (imdbData) {
const {
poster = "",
imdb_rating_average: imdbRate,
description = "",
aka,
year
} = imdbData;
replaceParams.poster = poster;
replaceParams.synopsis = description;
replaceParams.imdbScore = imdbRate;
const searchMovieName = movieName || ((_b = (_a2 = aka.filter((item) => item.country.match(/(World-wide)|UK|USA/))) == null ? void 0 : _a2[0]) == null ? void 0 : _b.title);
await this.fillRottenTomatoesData(
replaceParams,
searchMovieName,
category,
year
);
await this.uploadPosterToPtpImg(replaceParams, poster);
}
} catch (error) {
console.error("Error fetching IMDB data:", error);
}
}
async fillRottenTomatoesData(replaceParams, movieName, category, year) {
if (!movieName) return;
try {
const rtInfo = await getMatchRottenTomatoes(
movieName,
year,
!!category.match(/tv/)
);
if (!rtInfo) return;
const { rottenTomatoes, rtId = "" } = rtInfo;
replaceParams.rtScore = `${rottenTomatoes.audienceScore}%`;
replaceParams.rtUrl = `https://www.rottentomatoes.com/${rtId}`;
} catch (error) {
console.error("Error fetching Rotten Tomatoes data:", error);
}
}
async uploadPosterToPtpImg(replaceParams, poster) {
const ptpImgApiKey = GM_getValue("easy-upload.ptp-img-api-key") || "";
if (ptpImgApiKey && poster) {
try {
const ptpImgPoster = await (await uploadToPtpImg)([poster]);
replaceParams.poster = ptpImgPoster ? ptpImgPoster[0] : "";
} catch (error) {
console.error("Error uploading poster to PTP image host:", error);
}
}
}
async fillTMDBData(replaceParams) {
if (!this.info || !this.info.imdbUrl) return;
try {
const imdbId = getIdByIMDbUrl(this.info.imdbUrl);
const { id: tmdbId, vote_average: tmdbRate } = await getTMDBDataByIMDBId(imdbId);
if (tmdbId) {
replaceParams.tmdbUrl = `https://www.themoviedb.org/movie/${tmdbId}`;
replaceParams.tmdbScore = `${tmdbRate}`;
await this.fillTrailerData(replaceParams, tmdbId);
}
} catch (error) {
console.error("Error fetching TMDB data:", error);
}
}
async fillTrailerData(replaceParams, tmdbId) {
try {
const videos = await getTMDBVideosById(`${tmdbId}`);
const youtubeVideo = videos.find((video) => video.site === "YouTube");
if (youtubeVideo == null ? void 0 : youtubeVideo.key) {
replaceParams.youtubeUrl = `https://www.youtube.com/watch?v=${youtubeVideo.key}`;
}
} catch (error) {
console.error("Error fetching trailer data:", error);
}
}
}
registry$1.register(new ITS());
class BYR extends BaseFiller {
constructor() {
super(...arguments);
this.priority = 10;
}
canHandle(siteName) {
return siteName === "BYR";
}
fill(info) {
this.info = info;
if (!this.info) return;
this.fillBasicInfo();
this.setupDescriptionEditor();
this.fillOriginalTitle();
this.extractAndFillCategoryFields();
this.fillTorrentFile();
}
fillBasicInfo() {
const { subtitle, imdbUrl, doubanUrl } = this.info;
$$2(this.siteInfo.subtitle.selector).val(subtitle || "");
$$2(this.siteInfo.imdb.selector).val(imdbUrl || "");
$$2(this.siteInfo.douban.selector).val(doubanUrl || "");
}
setupDescriptionEditor() {
const { description } = this.info;
CKEDITOR.on("instanceReady", () => {
CKEDITOR.instances.descr.setData(this.convertBBCodeToHtml(description));
});
}
fillOriginalTitle() {
this.processTorrentTitle();
const { title } = this.info;
$$2("#movie_ename0day").val(title);
}
extractAndFillCategoryFields() {
const { category } = this.info;
const torrentInfo = this.extractTorrentInfo();
if (category.match(/movie/)) {
this.fillMovieFields(torrentInfo);
} else if (category.match(/tv/)) {
this.fillTVFields(torrentInfo);
} else if (category.match(/variety/)) {
this.fillVarietyFields(torrentInfo);
}
}
extractTorrentInfo() {
var _a2, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l;
const { description, doubanInfo } = this.info;
const fullDescription = description + doubanInfo;
let area = (_b = (_a2 = fullDescription.match(/(产\s+地|国\s+家)\s+(.+)/)) == null ? void 0 : _a2[2]) != null ? _b : "";
area = area.replace(/\[\/?.+?\]/g, "");
const originalName = (_d = (_c = fullDescription.match(/(片\s+名)\s+(.+)?/)) == null ? void 0 : _c[2]) != null ? _d : "";
const translateName = (_h = (_g = (_f = (_e = fullDescription.match(/(译\s+名)\s+(.+)/)) == null ? void 0 : _e[2]) == null ? void 0 : _f.split("/")) == null ? void 0 : _g[0]) != null ? _h : "";
const movieType = (_j = (_i = fullDescription.match(/(类\s+别)\s+(.+)/)) == null ? void 0 : _i[2]) != null ? _j : "";
const language = (_l = (_k = fullDescription.match(/(语\s+言)\s+(.+)/)) == null ? void 0 : _k[2]) != null ? _l : "";
let chineseName = originalName;
if (!originalName.match(/[\u4e00-\u9fa5]+/)) {
chineseName = translateName.match(/[\u4e00-\u9fa5]+/) ? translateName : "";
}
return {
area,
originalName,
translateName,
movieType,
language,
chineseName
};
}
fillMovieFields(torrentInfo) {
const { area, chineseName, movieType } = torrentInfo;
const { category } = this.info;
const regionSelector = this.determineMovieRegion(area);
const typeMap = {
华语: "11",
欧洲: "12",
北美: "13",
亚洲: "14",
其他: "1"
};
$$2('select[name="second_type"]').val(
typeMap[regionSelector]
);
$$2('select[name="second_type"]')[0].dispatchEvent(new Event("change"));
const movieTypeArr = movieType.split(/\s\//);
$$2("#movie_type").val(movieTypeArr.join("/"));
fillField(
regionSelector,
category === "movie" ? "movie_country" : "show_country"
);
$$2("#movie_cname").val(chineseName);
}
determineMovieRegion(area) {
if (area.match(/华语|台|港/)) {
return "华语";
} else if (area.match(/日本|韩国|泰国/)) {
return "亚洲";
} else if (area.match(/美国|加拿大/)) {
return "北美";
} else if (area.match(/欧|英|法|德|俄|意|苏联|EU/)) {
return "欧洲";
}
return "其他";
}
fillTVFields(torrentInfo) {
var _a2, _b;
const { area, chineseName } = torrentInfo;
const { title, videoType, mediaInfos } = this.info;
const regionSelector = this.determineTVRegion(area);
const typeMap = {
大陆: "15",
港台: "16",
欧美: "17",
日韩: "18",
其他: "2"
};
$$2('select[name="second_type"]').val(
typeMap[regionSelector]
);
$$2('select[name="second_type"]')[0].dispatchEvent(new Event("change"));
fillField(regionSelector, "tv_type");
$$2("#tv_ename").val(title);
$$2("#cname").val(chineseName);
const episode = (_b = (_a2 = title.match(/S\d+(E\d+)?/i)) == null ? void 0 : _a2[0]) != null ? _b : "";
$$2("#tv_season").val(episode);
const isBluray = !!videoType.match(/bluray/i);
const parsedMedia = parseMedia(mediaInfos == null ? void 0 : mediaInfos[0], isBluray);
if (!parsedMedia) {
console.warn("Failed to parse media info:", mediaInfos == null ? void 0 : mediaInfos[0]);
return;
}
const { format: format2 } = parsedMedia;
fillField((format2 == null ? void 0 : format2.toUpperCase()) || "MKV", "tv_filetype");
}
determineTVRegion(area) {
if (area.match(/大陆/)) {
return "大陆";
} else if (area.match(/台|港/)) {
return "港台";
} else if (area.match(/美国|欧|英|法|德|俄|意|苏联|EU/)) {
return "欧美";
} else if (area.match(/日本|韩国/)) {
return "日韩";
}
return "其他";
}
fillVarietyFields(torrentInfo) {
const { area, chineseName, language } = torrentInfo;
const { title } = this.info;
const regionSelector = this.determineTVRegion(area);
const typeMap = {
大陆: "27",
港台: "29",
欧美: "30",
日韩: "28",
其他: "5"
};
$$2('select[name="second_type"]').val(
typeMap[regionSelector]
);
$$2('select[name="second_type"]')[0].dispatchEvent(new Event("change"));
fillField(regionSelector, "show_country");
$$2("#show_cname").val(chineseName);
$$2("#show_ename").val(title);
const languageVal = this.determineLanguage(language);
fillField(languageVal, "show_language");
}
determineLanguage(language) {
if (language.match(/汉语/)) {
return "国语";
} else if (language.match(/粤/)) {
return "粤语";
} else if (language.match(/英语/)) {
return "英语";
} else if (language.match(/日语/)) {
return "日语";
} else if (language.match(/韩语/)) {
return "韩语";
}
return "";
}
convertBBCodeToHtml(bbcode) {
let html2 = bbcode.replace(/\[\*\]([^\n]+)/gi, "$1");
html2 = html2.replace(/(\r\n)|\n/g, "
");
html2 = html2.replace(
/\[(quote|hide=.+?)\]/gi,
"