JavBus工具

暗黑模式、滚动加载、预览视频、离线下载(115)...

ของเมื่อวันที่ 03-12-2021 ดู เวอร์ชันล่าสุด

  1. // ==UserScript==
  2. // @name JavBus工具
  3. // @description 暗黑模式、滚动加载、预览视频、离线下载(115)...
  4. // @version 0.2.2
  5. // @icon https://z3.ax1x.com/2021/10/15/53gMFS.png
  6. // @include *://*.javbus.com/*
  7. // @include *://captchaapi.115.com/*
  8. // @require https://unpkg.com/masonry-layout@4/dist/masonry.pkgd.min.js
  9. // @require https://unpkg.com/infinite-scroll@4/dist/infinite-scroll.pkgd.min.js
  10. // @require https://cdn.jsdelivr.net/npm/localforage@1.10.0/dist/localforage.min.js
  11. // @resource fail https://z3.ax1x.com/2021/10/15/53gcex.png
  12. // @resource info https://z3.ax1x.com/2021/10/15/53g2TK.png
  13. // @resource success https://z3.ax1x.com/2021/10/15/53gqTf.png
  14. // @run-at document-start
  15. // @grant GM_xmlhttpRequest
  16. // @grant GM_addStyle
  17. // @grant GM_getValue
  18. // @grant GM_setValue
  19. // @grant GM_notification
  20. // @grant GM_setClipboard
  21. // @grant GM_getResourceURL
  22. // @grant GM_openInTab
  23. // @grant GM_info
  24. // @grant GM_registerMenuCommand
  25. // @connect *
  26. // @license MIT
  27. // @namespace https://greasyfork.org/users/175514
  28. // ==/UserScript==
  29.  
  30. /**
  31. * 非完全测试,自用Edge
  32. * TODO:
  33. * ✔️ 暗黑模式
  34. * ✔️ 点击事件(新窗口打开,左键前台,右键后台)
  35. * ✔️ 滚动加载
  36. * ✔️ 获取预览大图(javstore.net)
  37. * ✔️ 获取预览视频(r18.com),如有资源点击封面图或样品图像处进行预览
  38. * ✔️ 获取演员名单(javdb.com)
  39. * ✔️ 离线下载(请求离线成功2秒后查询结果汇报),自动修改文件名称,(如已设置操作目录cid,自动移动视频文件至对应目录并删除原目录)
  40. * ✔️ 一键离线(字幕>尺寸>日期)排队执行离线下载,成功一结束排队
  41. * ✔️ 115账号验证弹窗
  42. * ✔️ 查询115是否已有离线资源
  43. * ✔️ 查询115是否已有离线资源并缓存(默认3天有效),3天后重新查询,支持手动刷新“资源刷新”
  44. * ✔️ 根据缓存数据匹配列表页已有资源状态,添加已有资源列表页
  45. */
  46. (function () {
  47. "use strict";
  48. // params
  49. let mousePoint = 0;
  50. const dm = GM_getValue("dm") ?? matchMedia("(prefers-color-scheme: dark)").matches;
  51. const lm = GM_getValue("lm") ?? false;
  52. const ck = GM_getValue("ck") ?? true;
  53. const rootId = GM_getValue("rid") ?? "";
  54. const SUFFIX = " !important";
  55. const { host, pathname } = location;
  56. // document
  57. const doc = document;
  58. doc.create = (tag, attr = {}, child = "") => {
  59. if (!tag) return null;
  60. tag = doc.createElement(tag);
  61. Object.keys(attr).forEach(name => tag.setAttribute(name, attr[name]));
  62. typeof child === "string" && tag.appendChild(doc.createTextNode(child));
  63. typeof child === "object" && tag.appendChild(child);
  64. return tag;
  65. };
  66. const insertBefore = (el, htmlString) => el.insertAdjacentHTML("beforebegin", htmlString);
  67. const insertAfter = (el, htmlString) => el.insertAdjacentHTML("afterend", htmlString);
  68. const appendBefore = (el, htmlString) => el.insertAdjacentHTML("afterbegin", htmlString);
  69. const appendAfter = (el, htmlString) => el.insertAdjacentHTML("beforeend", htmlString);
  70. // localforage
  71. const lf = localforage;
  72. lf.upItem = async (key, val) => {
  73. const original = (await lf.getItem(key)) ?? {};
  74. await lf.setItem(key, Object.assign(original, val));
  75. };
  76. // request
  77. const request = (url, method = "GET", data = {} | "", params = {}) => {
  78. if (!url) return;
  79. method ??= "GET";
  80. method = method.toLocaleUpperCase();
  81. if (method === "POST") {
  82. const cType = { "Content-Type": "application/x-www-form-urlencoded" };
  83. params.headers = Object.assign(params.headers ?? {}, cType);
  84. }
  85. if (typeof data === "object") {
  86. const keys = Object.keys(data);
  87. if (keys.length) {
  88. data = keys.reduce((pre, cur) => `${pre}${pre ? "&" : ""}${cur}=${encodeURIComponent(data[cur])}`, "");
  89. if (method === "GET") url = `${url}?${data}`;
  90. }
  91. }
  92. return new Promise(resolve => {
  93. GM_xmlhttpRequest({
  94. url,
  95. method,
  96. data,
  97. timeout: 20000,
  98. onload: ({ status, responseText }) => {
  99. if (status === 404) resolve(false);
  100. if (/<\/?[a-z][\s\S]*>/i.test(responseText)) {
  101. responseText = new DOMParser().parseFromString(responseText, "text/html");
  102. }
  103. if (/^{.*}$/.test(responseText)) {
  104. responseText = JSON.parse(responseText);
  105. }
  106. if (`${responseText?.errcode}` === "911") verify();
  107. resolve(responseText);
  108. },
  109. ...params,
  110. });
  111. });
  112. };
  113. // notifiy
  114. const notifiy = (title = "" | {}, text = "", image = "info", clickUrl = "") => {
  115. let params = {
  116. title,
  117. text: text ?? GM_info.script.name,
  118. image,
  119. highlight: true,
  120. timeout: 3000,
  121. clickUrl,
  122. };
  123. if (typeof title === "object") params = { ...params, ...title };
  124. if (params.clickUrl) params.onclick = () => GM_openInTab(clickUrl, { active: true });
  125. GM_notification({ ...params, image: GM_getResourceURL(params?.image ?? "info") });
  126. };
  127. // verify
  128. const verify = async () => {
  129. const time = new Date().getTime();
  130. let h = 667;
  131. let w = 375;
  132. let t = (window.screen.availHeight - h) / 2;
  133. let l = (window.screen.availWidth - w) / 2;
  134. window.open(
  135. `https://captchaapi.115.com/?ac=security_code&type=web&cb=Close911_${time}`,
  136. "验证账号",
  137. `height=${h},width=${w},top=${t},left=${l},toolbar=no,menubar=no,scrollbars=no,resizable=no,location=no,status=no`
  138. );
  139. };
  140. // delay
  141. const delay = ms => new Promise(resolve => setTimeout(resolve, ms));
  142. // scriptStart
  143. const scriptStart = () => {
  144. doc.addEventListener("keyup", event => {
  145. const e = event || window.event;
  146. if (e && e.keyCode === 191 && !["INPUT", "TEXTAREA"].includes(doc.activeElement.nodeName)) {
  147. doc.querySelector("#search-input").focus();
  148. }
  149. });
  150.  
  151. const nav = `<a href="https://www.javbus.com/resource">资源</a>`;
  152. let navbar = doc.querySelector("#navbar .nav.navbar-nav");
  153. if (navbar) return appendAfter(navbar, `<li class="resource">${nav}</li>`);
  154. navbar = doc.querySelector("#toptb .wp .z ul");
  155. if (navbar) return appendAfter(navbar, `<li class="nav-title nav-inactive">${nav}</li>`);
  156. };
  157. // handleClick
  158. const handleClick = (selectors, node = doc) => {
  159. for (const item of node.querySelectorAll(selectors)) {
  160. const href = item?.href;
  161. if (!href) continue;
  162. item.addEventListener("contextmenu", e => e.preventDefault());
  163. item.addEventListener("click", e => {
  164. e.preventDefault();
  165. GM_openInTab(href, { active: true });
  166. });
  167. item.addEventListener("mousedown", e => {
  168. if (e.button !== 2) return;
  169. e.preventDefault();
  170. mousePoint = e.screenX + e.screenY;
  171. });
  172. item.addEventListener("mouseup", e => {
  173. const num = e.screenX + e.screenY - mousePoint;
  174. if (e.button !== 2 || num > 5 || num < -5) return;
  175. e.preventDefault();
  176. GM_openInTab(href);
  177. });
  178. }
  179. };
  180. // getDate
  181. const getDate = (str = "") => {
  182. const date = str ? new Date(str) : new Date();
  183. const Y = date.getFullYear();
  184. const M = date.getMonth() + 1 < 10 ? `0${date.getMonth() + 1}` : date.getMonth() + 1;
  185. const D = date.getDate() < 10 ? `0${date.getDate()}` : date.getDate();
  186. return `${Y}-${M}-${D}`;
  187. };
  188.  
  189. class Common {
  190. docStart = () => {};
  191. contentLoaded = () => {};
  192. load = () => {};
  193. }
  194. class Waterfall extends Common {
  195. isHome = /^(\/(page\/\d+)?|\/uncensored(\/page\/\d+)?)+$/i.test(pathname);
  196. docStart = () => {
  197. GM_addStyle(`
  198. .search-header {
  199. padding: 0${SUFFIX};
  200. background: none${SUFFIX};
  201. box-shadow: none${SUFFIX};
  202. }
  203. .photo-frame {
  204. position: relative;
  205. margin: 10px${SUFFIX};
  206. }
  207. .photo-frame img {
  208. height: 100%${SUFFIX};
  209. width: 100%${SUFFIX};
  210. object-fit: cover${SUFFIX};
  211. margin: 0${SUFFIX};
  212. }
  213. .photo-info { padding: 10px${SUFFIX}; }
  214. .alert-page { margin: 20px${SUFFIX}; }
  215. `);
  216. if (!lm) return;
  217. const itemSizer = `167px`;
  218. const gutterSizer = `20px`;
  219. GM_addStyle(`
  220. .pagination, footer { display: none${SUFFIX}; }
  221. .page-load-status {
  222. display: none;
  223. padding-bottom: ${gutterSizer};
  224. text-align: center;
  225. }
  226. body { overflow: hidden; }
  227. .scrollBox {
  228. height: calc(100vh - 50px);
  229. overflow: hidden scroll;
  230. }
  231. #waterfall {
  232. opacity: 0;
  233. margin: ${gutterSizer} auto 0 auto${SUFFIX};
  234. }
  235. .item-sizer, .item a { width: ${itemSizer}${SUFFIX}; }
  236. .gutter-sizer { width: ${gutterSizer}${SUFFIX}; }
  237. .item a { margin: 0 0 ${gutterSizer} 0${SUFFIX}; }
  238. `);
  239. };
  240. contentLoaded = () => {
  241. const nav = doc.querySelector(".search-header .nav");
  242. if (nav) nav.setAttribute("class", "nav nav-pills");
  243. if (!lm) return this.modifyItem();
  244. this.handleLoadMore();
  245. if (!this.isHome) this.modifyLayout();
  246. };
  247. load = () => {
  248. if (lm && this.isHome) this.modifyLayout();
  249. };
  250. handleLoadMore = () => {
  251. const oldWaterfall = doc.querySelector("#waterfall");
  252. if (!oldWaterfall) return GM_addStyle(`#waterfall { opacity: 1; }`);
  253. const newWaterfall = doc.querySelector("#waterfall #waterfall");
  254. if (newWaterfall) oldWaterfall.parentNode.replaceChild(newWaterfall, oldWaterfall);
  255.  
  256. const waterfall = doc.querySelector("#waterfall");
  257. appendBefore(waterfall, `<div class="item-sizer"></div><div class="gutter-sizer"></div>`);
  258. insertAfter(
  259. waterfall,
  260. `<div class="page-load-status"><span class="loader-ellips infinite-scroll-request">Loading...</span><span class="infinite-scroll-last">End of content</span><span class="infinite-scroll-error">No more pages to load</span></div>`
  261. );
  262.  
  263. doc.querySelector(".container-fluid .row").classList.add("scrollBox");
  264. };
  265. modifyLayout = () => {
  266. const waterfall = doc.querySelector("#waterfall");
  267. if (!waterfall) return;
  268. let msnry = new Masonry(waterfall, {
  269. itemSelector: "none",
  270. columnWidth: ".item-sizer",
  271. gutter: ".gutter-sizer",
  272. horizontalOrder: true,
  273. fitWidth: true,
  274. stagger: 30,
  275. visibleStyle: { transform: "translateY(0)", opacity: 1 },
  276. hiddenStyle: { transform: "translateY(120px)", opacity: 0 },
  277. });
  278. imagesLoaded(waterfall, () => {
  279. msnry.options.itemSelector = ".item";
  280. const elems = waterfall.querySelectorAll(".item");
  281. this.modifyItem();
  282. msnry.appended(elems);
  283. GM_addStyle(`#waterfall { opacity: 1; }`);
  284. });
  285. // 搜索页滚动加载需特殊处理
  286. const path = !/^\/(uncensored\/)?(search|searchstar)+\//i.test(pathname)
  287. ? "#next"
  288. : function () {
  289. const items = ["search", "searchstar"];
  290. for (const item of items) {
  291. if (pathname.indexOf(`${item}/`) < 0) continue;
  292. let [prefix, suffix] = pathname.split("&");
  293. suffix ??= "";
  294. prefix = prefix.split("/");
  295. let pre = "";
  296. for (let index = 0; index <= prefix.indexOf(item) + 1; index++) {
  297. pre = `${pre}${prefix[index]}/`;
  298. }
  299. return `${pre}${this.loadCount + 2}&${suffix}`;
  300. }
  301. };
  302. let infScroll = new InfiniteScroll(waterfall, {
  303. path,
  304. append: ".item",
  305. outlayer: msnry,
  306. elementScroll: ".scrollBox",
  307. history: false,
  308. historyTitle: false,
  309. hideNav: ".pagination",
  310. status: ".page-load-status",
  311. debug: false,
  312. });
  313. infScroll.on("load", this.modifyItem);
  314. };
  315. modifyItem = (node = doc) => {
  316. if (ck) handleClick("a.movie-box", node);
  317. const items = node.querySelectorAll(".item");
  318. for (const item of items) {
  319. let [code, date] = item.querySelectorAll("date");
  320. if (code && date) {
  321. code = code.textContent;
  322. date = date.textContent;
  323. const codeKey = `${code.trim().toUpperCase()}/${date.trim().toUpperCase()}`;
  324. lf.keys().then(keys => {
  325. if (!keys.includes(codeKey)) return;
  326. lf.getItem(codeKey).then(val => {
  327. const resource = val?.resource;
  328. if (!resource?.length) return;
  329. const photo = item.querySelector(".photo-frame");
  330. photo.classList.add("playBtn");
  331. photo.setAttribute("title", "点击播放");
  332. photo.addEventListener("click", e => {
  333. e.stopPropagation();
  334. e.preventDefault();
  335. GM_openInTab(resource[0].link, { active: true });
  336. });
  337. });
  338. });
  339. }
  340.  
  341. const info = item.querySelector("a .photo-info span:not(.mleft)");
  342. if (!info) continue;
  343. const [titleNode, secondaryNode] = info.childNodes;
  344. const titleTxt = titleNode.textContent.trim();
  345. const ellipsis = doc.create("div", { class: "ellipsis", title: titleTxt }, titleTxt);
  346. if (secondaryNode?.nodeName === "BR") {
  347. info.removeChild(secondaryNode);
  348. ellipsis.classList.add("line-4");
  349. }
  350. info.replaceChild(ellipsis, titleNode);
  351. }
  352. };
  353. }
  354. class Genre extends Common {
  355. docStart = () => {
  356. GM_addStyle(`
  357. footer { display: none${SUFFIX}; }
  358. button.btn.btn-danger.btn-block.btn-genre {
  359. position: fixed${SUFFIX};
  360. bottom: 0${SUFFIX};
  361. margin: 0${SUFFIX};
  362. left: 0${SUFFIX};
  363. border: 0${SUFFIX};
  364. border-radius: 0${SUFFIX};
  365. }
  366. `);
  367. };
  368. contentLoaded = () => {
  369. if (!doc.querySelector("button.btn.btn-danger.btn-block.btn-genre")) return;
  370. const box = doc.querySelectorAll(".genre-box");
  371. box[box.length - 1].setAttribute("style", "margin-bottom: 65px;");
  372. };
  373. }
  374. class Forum extends Common {
  375. docStart = () => {
  376. GM_addStyle(`
  377. .bcpic,
  378. .banner728,
  379. .sd.sd_allbox > div:last-child { display: none${SUFFIX}; }
  380. .jav-button { margin-top: -3px${SUFFIX}; }
  381. #toptb {
  382. position: fixed${SUFFIX};
  383. top: 0${SUFFIX};
  384. left: 0${SUFFIX};
  385. right: 0${SUFFIX};
  386. z-index: 999${SUFFIX};
  387. }
  388. #wp { margin-top: 55px${SUFFIX}; }
  389. `);
  390. };
  391. }
  392. class Resource extends Common {
  393. docStart = () => {
  394. GM_addStyle(`
  395. .alert.alert-danger.alert-page.error-page, footer { display: none; }
  396. .resItem {
  397. margin: 15px 0;
  398. border-radius: 4px;
  399. overflow: hidden;
  400. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, .1);
  401. background-color: ${dm ? SecondaryBackground : "#fff"};
  402. }
  403. .resItem a { color: unset; }
  404. .thumb { aspect-ratio: 100 / 67; }
  405. .thumb img {
  406. width: 100%;
  407. height: 100%;
  408. object-fit: cover;
  409. }
  410. .info { padding: 15px; }
  411. .title { margin-bottom: 6px; }
  412. `);
  413. };
  414. contentLoaded = async () => {
  415. doc.title = "已有资源(0)";
  416. const navbar = doc.querySelector("#navbar .nav.navbar-nav");
  417. navbar.querySelector(".active").classList.remove("active");
  418. navbar.querySelector(".resource").classList.add("active");
  419. const keys = await lf.keys();
  420. let nodes = [];
  421. for (const key of keys) {
  422. const item = await lf.getItem(key);
  423. const resource = item?.resource;
  424. if (!resource?.length) continue;
  425. nodes.push({ ...item, key: key.split("/")[0] });
  426. }
  427. if (!nodes?.length) {
  428. const alert = doc.querySelector(".alert.alert-danger.alert-page.error-page");
  429. alert.querySelector("h4").textContent = "暂无本地缓存数据";
  430. const [zh, en] = alert.querySelectorAll("p");
  431. zh.setAttribute("style", "display: none");
  432. en.textContent = "请检查浏览器相关设置或重新获取数据";
  433. alert.setAttribute("style", "display: block");
  434. return;
  435. }
  436. doc.title = `已有资源(${nodes.length})`;
  437. nodes.sort((pre, next) => {
  438. let preTime = pre.resource[0].timestamp || Date.parse(pre.resource[0].getDate) / 1000;
  439. let nextTime = next.resource[0].timestamp || Date.parse(next.resource[0].getDate) / 1000;
  440. return preTime > nextTime ? -1 : 1;
  441. });
  442. nodes = nodes.reduce((pre, { thumb, title, resource, href, key, upDate }) => {
  443. return `${pre}
  444. <div class="col-lg-3 col-md-4 col-sm-6">
  445. <div class="resItem">
  446. <a href="${href ?? "/" + key}">
  447. <div class="playBtn thumb" data-link="${resource[0].link}" title="点击播放">
  448. <img src="${thumb}" alt="${title}">
  449. </div>
  450. <div class="info">
  451. <div class="ellipsis title" title="${title}">${title}</div>
  452. <button class="btn btn-xs btn-primary" disabled="disabled">
  453. 更新:${getDate(upDate)}
  454. </button>
  455. <button class="btn btn-xs btn-danger" disabled="disabled">
  456. 资源:${resource.length}
  457. </button>
  458. </div>
  459. </a>
  460. </div>
  461. </div>`;
  462. }, "");
  463. appendAfter(doc.querySelector(".container-fluid .row"), nodes);
  464. for (const item of doc.querySelectorAll(".resItem .thumb")) {
  465. item.addEventListener("click", e => {
  466. e.stopPropagation();
  467. e.preventDefault();
  468. GM_openInTab(item.getAttribute("data-link"), { active: true });
  469. });
  470. }
  471. ck && handleClick(".resItem a");
  472. };
  473. }
  474. class Details extends Common {
  475. code = "";
  476. codeKey = "";
  477. lfItem = {};
  478. config = async (code, date) => {
  479. if (!code || !date) return;
  480. const codeKey = `${code}/${date}`;
  481. this.code = code;
  482. this.codeKey = codeKey;
  483. this.lfItem = (await lf.getItem(codeKey)) ?? {};
  484. };
  485. docStart = () => {
  486. GM_addStyle(`
  487. .info .glyphicon-info-sign,
  488. h4[style="position:relative"],
  489. h4[style="position:relative"] + .row { display: none${SUFFIX}; }
  490. .info ul { margin: 0${SUFFIX}; }
  491. .screencap { max-height: 600px; overflow: hidden; }
  492. #avatar-waterfall, #sample-waterfall, #related-waterfall { margin: -5px${SUFFIX}; }
  493. .photo-info { height: auto${SUFFIX}; }
  494. `);
  495. GM_addStyle(`
  496. #mask {
  497. position: fixed;
  498. width: 100%;
  499. height: 100%;
  500. z-index: 9999;
  501. left: 0;
  502. top: 0;
  503. background: rgba(11,11,11,.8);
  504. display: flex;
  505. justify-content: center;
  506. align-items: center;
  507. display: none;
  508. }
  509. `);
  510. GM_addStyle(`
  511. #exp { display: none; }
  512. #expBtn {
  513. position: absolute;
  514. top: 0;
  515. height: 40px;
  516. line-height: 40px;
  517. left: 0;
  518. right: 0;
  519. cursor: pointer;
  520. background: rgba(0,0,0,.7);
  521. color: #fff;
  522. margin: 560px 15px 0 15px;
  523. z-index: 99;
  524. }
  525. #expBtn::after { content: "展开"; }
  526. #exp:checked + .screencap { max-height: none; }
  527. #exp:checked + .screencap > #expBtn {
  528. top: auto;
  529. margin: 0 15px;
  530. bottom: 0;
  531. }
  532. #exp:checked + .screencap > #expBtn::after { content: "收起"; }
  533. @media screen and (max-width: 480px) {
  534. #btnGrp { margin-bottom: 15px; }
  535. .screencap { max-height: 280px; }
  536. #expBtn { margin-left: 0; margin-right: 0; }
  537. }
  538. #resBox a { color: #CC0000${SUFFIX}; }
  539. `);
  540. doc.addEventListener("DOMNodeInserted", e => {
  541. const node = e.target;
  542. if (node.nodeName.toLowerCase() !== "tr") return;
  543. const href = node.querySelector("td a")?.href;
  544. if (!href) return;
  545. const td = doc.create("td", { style: "text-align:center;white-space:nowrap" });
  546. const copy = doc.create("a", { href, title: href }, "复制");
  547. const offline = doc.create("a", { href, title: href, style: "margin-left:16px" }, "离线下载");
  548. copy.addEventListener("click", this.copyTxt);
  549. offline.addEventListener("click", this.offLine);
  550. td.appendChild(copy);
  551. td.appendChild(offline);
  552. node.appendChild(td);
  553. });
  554. };
  555. contentLoaded = async () => {
  556. ck && handleClick("a.movie-box");
  557. // copy
  558. const handleCopy = selectors => {
  559. const node = doc.querySelector(selectors);
  560. const txt = node?.textContent?.trim();
  561. if (!node || !txt) return;
  562. const copy = doc.create("a", { title: txt, href: txt, style: "margin-left:16px" }, "复制");
  563. copy.addEventListener("click", this.copyTxt);
  564. node.appendChild(copy);
  565. };
  566. handleCopy("h3");
  567. handleCopy("span[style='color:#CC0000;']");
  568. // expBtn
  569. const screencap = doc.querySelector(".col-md-9.screencap");
  570. insertBefore(screencap, `<input type="checkbox" id="exp">`);
  571. appendAfter(screencap, `<label for="exp" id="expBtn"></label>`);
  572.  
  573. const info = doc.querySelector(".col-md-3.info");
  574. // resource
  575. const res = `<p id="resBox"><span class="header">已有资源:</span><span class="genre">查询中...</span></p>`;
  576. appendAfter(info, res);
  577. // btnGrp
  578. const btnGrp = doc.create("div", { id: "btnGrp", class: "btn-group btn-group-justified" });
  579. const btns = { smartRes: "一键离线", refreshRes: "资源刷新" };
  580. Object.keys(btns).forEach(id => {
  581. const btn = doc.create("a", { id, class: "btn btn-default" }, btns[id]);
  582. btn.addEventListener("click", () => this.handleOpt(id));
  583. btnGrp.appendChild(btn);
  584. });
  585. info.appendChild(btnGrp);
  586. // table
  587. const td = `<td style="text-align:center;white-space:nowrap">操作</td>`;
  588. appendAfter(doc.querySelector("#magnet-table tbody tr"), td);
  589. // photoinfo
  590. for (const item of doc.querySelectorAll(".photo-info span")) item.classList.add("ellipsis");
  591. // config
  592. if (!this.codeKey) {
  593. let [code, date] = doc.querySelectorAll("div.col-md-3.info p");
  594. code = code.querySelectorAll("span")[1].childNodes[0].textContent.trim().toUpperCase();
  595. date = date.childNodes[1].textContent.trim().toUpperCase();
  596. await this.config(code, date);
  597. }
  598.  
  599. this.handleStar();
  600. this.handleResource();
  601. this.handlePreview();
  602. this.handleVideo();
  603. };
  604. load = () => {
  605. const maxHei = 600;
  606. const styleHei = doc.querySelector(".col-md-9.screencap .bigImage img").height;
  607. if (styleHei > maxHei) {
  608. GM_addStyle(`.screencap { max-height: ${maxHei}px; } #expBtn { margin-top: ${maxHei - 40}px; }`);
  609. } else {
  610. GM_addStyle(`.screencap { max-height: ${styleHei + 40}px; } #expBtn { margin-top: ${styleHei}px; }`);
  611. }
  612. };
  613. copyTxt = e => {
  614. e.preventDefault();
  615. e.stopPropagation();
  616. const node = e.target;
  617. const txt = node?.title;
  618. if (!node || !txt) return;
  619. GM_setClipboard(txt);
  620. const text = node?.textContent ?? "";
  621. node.textContent = "成功";
  622. setTimeout(() => {
  623. node.textContent = text;
  624. }, 1000);
  625. };
  626. // 预览大图
  627. handlePreview = async () => {
  628. let image = this.lfItem?.image;
  629. if (!image) {
  630. let res = await request(`https://javstore.net/search/${this.code}.html`);
  631. const href = res?.querySelector("#content_news li a")?.href;
  632. if (!href) return;
  633. res = await request(href);
  634. image = res?.querySelector(".news a img[alt*='.th']")?.src?.replace(".th", "");
  635. if (!image) return;
  636. lf.upItem(this.codeKey, { image });
  637. }
  638. if (!(await request(image))) return;
  639. const img = doc.create("img", { src: image, title: "点击收起", style: "cursor:pointer" });
  640. img.addEventListener("click", () => {
  641. doc.querySelector("#exp").checked = false;
  642. });
  643. img.onload = () => doc.querySelector(".col-md-9.screencap").appendChild(img);
  644. };
  645. // 预览视频
  646. handleVideo = async () => {
  647. let video = this.lfItem?.video;
  648. if (!video) {
  649. const res = await request(`https://www.r18.com/common/search/searchword=${this.code}/`);
  650. video = res?.querySelector("a.js-view-sample")?.getAttribute("data-video-high");
  651. if (!video) return;
  652. lf.upItem(this.codeKey, { video });
  653. }
  654. const title = "预览视频";
  655. const playVideo = e => {
  656. e.preventDefault();
  657. e.stopPropagation();
  658. doc.body.setAttribute("style", "overflow: hidden;");
  659. doc.querySelector("#mask").setAttribute("style", "display: flex;");
  660. const video = doc.querySelector("video");
  661. video.play();
  662. video.focus();
  663. doc.onkeydown = event => {
  664. const e = event || window.event;
  665. if (e && e.keyCode === 27) pauseVideo();
  666. };
  667. };
  668. const pauseVideo = () => {
  669. doc.body.setAttribute("style", "overflow: auto;");
  670. doc.querySelector("#mask").setAttribute("style", "display: none;");
  671. doc.querySelector("video").pause();
  672. doc.onkeydown = null;
  673. };
  674. // 视频播放窗口
  675. const videoNode = doc.create("video", { controls: "controls", src: video });
  676. videoNode.currentTime = 5;
  677. videoNode.preload = "auto";
  678. videoNode.muted = true;
  679. const closeBtn = doc.create("button", { title: "Close (Esc)", type: "button", class: "mfp-close" }, "×");
  680. closeBtn.addEventListener("click", pauseVideo);
  681. const mask = doc.create("div", { id: "mask" });
  682. mask.appendChild(closeBtn);
  683. mask.appendChild(videoNode);
  684. doc.body.appendChild(mask);
  685.  
  686. // 封面点击播放
  687. const bigImage = doc.querySelector(".bigImage");
  688. const bImg = bigImage.querySelector("img");
  689. const playBtn = doc.create("div", { class: "playBtn", title });
  690. playBtn.addEventListener("click", playVideo);
  691. playBtn.appendChild(bImg);
  692. bigImage.appendChild(playBtn);
  693.  
  694. // 样品图像
  695. const thumb = bImg.src;
  696. const box = doc.create("a", { class: "sample-box", href: thumb, title });
  697. box.addEventListener("click", playVideo);
  698. appendAfter(box, `<div class="photo-frame playBtn"><img src="${thumb}"></div>`);
  699. if (!doc.querySelector("#sample-waterfall")) {
  700. insertBefore(doc.querySelector("div.clearfix"), `<div id="sample-waterfall"></div>`);
  701. insertBefore(doc.querySelector("div#sample-waterfall"), `<h4>樣品圖像</h4>`);
  702. }
  703. const waterfall = doc.querySelector("div#sample-waterfall");
  704. const ref = waterfall.querySelector("a");
  705. ref ? waterfall.insertBefore(box, ref) : waterfall.appendChild(box);
  706. };
  707. // 演员列表
  708. handleStar = async () => {
  709. const info = doc.querySelector(".col-md-3.info");
  710. if (!/暫無出演者資訊/g.test(info.textContent)) return;
  711. let star = this.lfItem?.star ?? [];
  712. if (!star?.length) {
  713. const site = "https://javdb.com";
  714. let res = await request(`${site}/search?q=${this.code}`);
  715. const href = res.querySelector("#videos .grid-item a").getAttribute("href");
  716. if (!href) return;
  717. res = await request(`${site}${href}`);
  718. res = res?.querySelectorAll(".panel-block");
  719. if (!res?.length) return;
  720. res = res[res.length - 3]?.querySelector(".value").textContent.trim();
  721. res = res.split(/\n/).filter(item => item.indexOf("♀") !== -1);
  722. star = res.map(item => item.trim().replace("♀", ""));
  723. if (!star?.length) return;
  724. lf.upItem(this.codeKey, { star });
  725. }
  726. const p = doc.create("p");
  727. appendAfter(
  728. p,
  729. star.reduce((acc, cur) => `${acc}<span class="genre"><a href="/search/${cur}">${cur}</a></span>`, "")
  730. );
  731. info.replaceChild(p, doc.querySelector("span.glyphicon.glyphicon-info-sign.mb20")?.nextSibling);
  732. };
  733. // 已有资源(本地)
  734. handleResource = async () => {
  735. const lfItem = await lf.getItem(this.codeKey);
  736.  
  737. let resource = lfItem?.resource ?? [];
  738. const upDate = lfItem?.upDate;
  739. const bool = !upDate || Math.floor((new Date().getTime() - upDate) / 24 / 3600 / 1000) > 3;
  740. if (bool) resource = await this.fetchResource();
  741.  
  742. const resBox = doc.querySelector("#resBox");
  743. for (const old of resBox.querySelectorAll(".genre")) resBox.removeChild(old);
  744. if (!resource?.length) return appendAfter(resBox, `<span class="genre">无</span>`);
  745. const child = resource.reduce((acc, { link, getDate, name }) => {
  746. let thunbName = name.length > 20 ? `${name.substr(0, 20)}...` : name;
  747. return `${acc}<span class="genre"><a href="${link}" title="${getDate}/${name}" target="_blank">${thunbName}</a></span>`;
  748. }, "");
  749. appendAfter(resBox, child);
  750.  
  751. if (lfItem?.thumb && lfItem?.title && lfItem?.href) return;
  752. lf.upItem(this.codeKey, {
  753. thumb: doc.querySelector(".screencap .bigImage img").src,
  754. title: doc.querySelector("h3").childNodes[0].textContent,
  755. href: location.href,
  756. });
  757. };
  758. // 已有资源(115)
  759. fetchResource = async (lname = "") => {
  760. let code = this.code;
  761. let codes = [
  762. code,
  763. code.replace(/-/g, ""),
  764. code.replace(/-/g, "-0"),
  765. code.replace(/-/g, "0"),
  766. code.replace(/-/g, "00"),
  767. code.replace(/-/g, "_"),
  768. code.replace(/-/g, "_0"),
  769. code.replace(/-0/g, ""),
  770. code.replace(/-0/g, "-"),
  771. code.replace(/-0/g, "00"),
  772. ];
  773. if (lname) codes.unshift(encodeURIComponent(lname));
  774. let { data } = await request("https://webapi.115.com/files/search", "GET", {
  775. search_value: encodeURIComponent(codes.join(" ")),
  776. format: "json",
  777. });
  778. let resource = [];
  779. if (data?.length) {
  780. const reg = new RegExp(`(${codes.join("|")})`, "gi");
  781. data = data.filter(({ n, play_long }) => n.match(reg) && play_long && play_long > 0);
  782. resource = data.map(item => {
  783. return {
  784. fid: item.fid,
  785. cid: item.cid,
  786. link: `https://v.anxia.com/?pickcode=${item.pc}`,
  787. getDate: item.t,
  788. name: item.n,
  789. timestamp: item.te || item.tp,
  790. };
  791. });
  792. }
  793. await lf.upItem(this.codeKey, { upDate: new Date().getTime(), resource });
  794. return resource;
  795. };
  796. // 离线下载
  797. offLine = async e => {
  798. e.preventDefault();
  799. e.stopPropagation();
  800. const node = e.target;
  801. const link = node?.title;
  802. if (!node || !link) return;
  803. GM_setClipboard(link);
  804. const text = node?.textContent ?? "";
  805. node.textContent = "请求中...";
  806. const firstTd = node.parentNode.parentNode.querySelector("td");
  807. const zh = !!firstTd.querySelector("a.btn.btn-mini-new.btn-warning.disabled");
  808. const lname = firstTd.querySelector("a").textContent.trim();
  809. const res = await this.offLineDownload({ link, zh, lname });
  810. node.textContent = text;
  811. notifiy(res);
  812. };
  813. // 排队离线/资源刷新
  814. handleOpt = async action => {
  815. const node = doc.querySelector(`#${action}`);
  816. if (!node) return;
  817. node.classList.toggle("disabled");
  818. const text = node?.textContent ?? "";
  819. node.textContent = "请求中...";
  820.  
  821. if (action === "refreshRes") {
  822. await lf.upItem(this.codeKey, { upDate: 0 });
  823. await this.handleResource();
  824. }
  825. if (action === "smartRes") {
  826. const trs = doc.querySelector("#magnet-table").querySelectorAll("tr");
  827. let magnetArr = [];
  828. for (let index = 1; index < trs.length; index++) {
  829. const item = { link: "", lname: "", zh: false, size: 0, date: 0 };
  830. let [name, size, date] = trs[index].querySelectorAll("td");
  831. if (!name || !size || !date) continue;
  832. item.zh = !!name.querySelector("a.btn.btn-mini-new.btn-warning.disabled");
  833. name = name.querySelector("a");
  834. item.link = name.href;
  835. item.lname = name.textContent.trim();
  836. size = size.querySelector("a").textContent.trim().replace(/gb/gi, "");
  837. if (/mb/gi.test(size)) size = (parseFloat(size) / 1024).toFixed(2);
  838. item.size = Number(size);
  839. item.date = date.querySelector("a").textContent.trim().replace(/-/g, "");
  840. magnetArr.push(item);
  841. }
  842. magnetArr.sort((pre, next) => {
  843. if (pre.zh === next.zh) {
  844. if (pre.size === next.size) return next.date - pre.date;
  845. return next.size - pre.size;
  846. } else {
  847. return pre.zh > next.zh ? -1 : 1;
  848. }
  849. });
  850. for (let index = 0; index < magnetArr.length; index++) {
  851. const res = await this.offLineDownload(magnetArr[index]);
  852. if (res?.image !== "info") {
  853. notifiy(res);
  854. break;
  855. }
  856. if (index !== magnetArr.length - 1) continue;
  857. notifiy(
  858. "一键离线失败",
  859. "远程未查找到新增资源,接口失效或资源被审核",
  860. "fail",
  861. "http://115.com/?tab=offline&mode=wangpan"
  862. );
  863. }
  864. }
  865.  
  866. node.textContent = text;
  867. node.classList.toggle("disabled");
  868. };
  869. // 请求离线&结果查询
  870. offLineDownload = async ({ link, zh, lname }) => {
  871. const fname = `${zh ? "【中文字幕】" : ""}${doc.querySelector("h3").textContent.replace("复制", "")}`;
  872.  
  873. let notifiyObj = {
  874. title: "操作失败,115未登录",
  875. text: "请登录115账户后再离线下载",
  876. image: "fail",
  877. clickUrl: "http://115.com/?mode=login",
  878. };
  879. let res = await request("http://115.com/", "GET", {
  880. ct: "offline",
  881. ac: "space",
  882. _: new Date().getTime(),
  883. });
  884. if (!res?.sign) return notifiyObj;
  885. const { sign, time } = res;
  886. res = await request("http://115.com/web/lixian/?ct=lixian&ac=add_task_url", "POST", {
  887. url: link.substr(0, 60),
  888. uid: 0,
  889. sign,
  890. time,
  891. });
  892. const { state, errcode, error_msg } = res;
  893. notifiyObj = {
  894. title: "离线失败",
  895. text: error_msg,
  896. image: "info",
  897. clickUrl: "http://115.com/?tab=offline&mode=wangpan",
  898. };
  899. if (`${errcode}` === "911") {
  900. notifiyObj.title += ",账号异常";
  901. notifiyObj.text = "验证后使用";
  902. notifiyObj.image = "fail";
  903. }
  904. if (!state) return notifiyObj;
  905. // 获取旧的本地缓存数据
  906. let lfItem = (await lf.getItem(this.codeKey)) ?? {};
  907. if (!lfItem?.resource) lfItem.resource = [];
  908. // 远程获取搜索结果
  909. await delay(2000);
  910. let resource = await this.fetchResource(lname);
  911. this.handleResource();
  912. const newRes = resource.filter(item => item.getDate === getDate());
  913. // 搜索结果资源数至少比本地缓存多且获取日期为当前日期
  914. if (resource.length <= lfItem.resource.length || newRes.length < 1) {
  915. notifiyObj.text = "未找到新增资源,接口失效或资源审核";
  916. return notifiyObj;
  917. }
  918. for (let index = 0; index < newRes.length; index++) {
  919. const { name } = newRes[index];
  920. const suffix = name.split(".").pop().toLowerCase();
  921. newRes[index]["rename"] = `${fname}.${suffix}`;
  922. }
  923. this.afterAction(newRes.filter(item => item.name.indexOf(fname) === -1));
  924. return {
  925. title: "离线成功",
  926. text: "点击跳转",
  927. image: "success",
  928. clickUrl: `https://115.com/?cid=${rootId ?? 0}&offset=0&mode=wangpan`,
  929. };
  930. };
  931. // 离线后重命名,移动,移除原目录等
  932. afterAction = async items => {
  933. // rename
  934. for (const { fid, rename: file_name } of items) {
  935. request("http://webapi.115.com/files/edit", "POST", { fid, file_name });
  936. }
  937. if (rootId) {
  938. const params = arr => arr.reduce((acc, cur, idx) => `${acc}&fid[${idx}]=${cur}`, "");
  939. const fids = Array.from(new Set(items.map(item => item.fid)));
  940. const cids = Array.from(new Set(items.filter(item => item !== rootId).map(item => item.cid)));
  941. // move
  942. const move_proid = `${new Date()}_${~(100 * Math.random())}_0`;
  943. await request(
  944. "https://webapi.115.com/files/move",
  945. "POST",
  946. `pid=${rootId}&move_proid=${move_proid}${params(fids)}`
  947. );
  948. // del
  949. request("https://webapi.115.com/rb/delete", "POST", `pid=${rootId}&ignore_warn=1${params(cids)}`);
  950. }
  951. lf.upItem(this.codeKey, { upDate: 0 });
  952. await delay(1000);
  953. this.handleResource();
  954. };
  955. }
  956.  
  957. class JavBusScript {
  958. waterfall = new Waterfall();
  959. genre = new Genre();
  960. forum = new Forum();
  961. resource = new Resource();
  962. details = new Details();
  963. }
  964.  
  965. const Background = "rgb(18,18,18)";
  966. const SecondaryBackground = "rgb(32,32,32)";
  967. const LabelColor = "rgba(255,255,255,.95)";
  968. const SecondaryLabelColor = "rgb(170,170,170)";
  969. const Grey = "rgb(49,49,49)";
  970. const Blue = "rgb(10,132,255)";
  971. const Orange = "rgb(255,159,10)";
  972. const Green = "rgb(48,209,88)";
  973. const Pink = "rgb(255,55,95)";
  974. const Red = "rgb(255,69,58)";
  975. const Yellow = "rgb(255,214,10)";
  976. const darkMode = path => {
  977. if (path === "forum") return;
  978. GM_addStyle(`
  979. ::-webkit-scrollbar {
  980. width: 8px;
  981. height: 8px;
  982. }
  983. ::-webkit-scrollbar-thumb {
  984. border-radius: 4px;
  985. background-color: ${Grey};
  986. }
  987. *:not(span) {
  988. border-color: ${Grey}${SUFFIX};
  989. outline: ${Grey}${SUFFIX};
  990. text-shadow: none${SUFFIX};
  991. }
  992. body, footer {
  993. background: ${Background}${SUFFIX};
  994. color: ${SecondaryLabelColor}${SUFFIX};
  995. }
  996. img { filter: brightness(90%)${SUFFIX}; }
  997. nav { background: ${SecondaryBackground}${SUFFIX}; }
  998. input { background: ${Background}${SUFFIX}; }
  999. *::placeholder { color: ${SecondaryLabelColor}${SUFFIX}; }
  1000. button, input, a, h1, h2, h3, h4, h5, h6 {
  1001. color: ${LabelColor}${SUFFIX};
  1002. box-shadow: none${SUFFIX};
  1003. outline: none${SUFFIX};
  1004. }
  1005. .btn.disabled, .btn[disabled], fieldset[disabled] .btn { opacity: .85${SUFFIX}; }
  1006. button, .btn-default, .input-group-addon {
  1007. background: ${Grey}${SUFFIX};
  1008. color: ${LabelColor}${SUFFIX};
  1009. }
  1010. .btn-primary {
  1011. background: ${Blue}${SUFFIX};
  1012. border-color: ${Blue}${SUFFIX};
  1013. }
  1014. .btn-success {
  1015. background: ${Green}${SUFFIX};
  1016. border-color: ${Green}${SUFFIX};
  1017. }
  1018. .btn-danger {
  1019. background: ${Red}${SUFFIX};
  1020. border-color: ${Red}${SUFFIX};
  1021. }
  1022. .btn-warning {
  1023. background: ${Orange}${SUFFIX};
  1024. border-color: ${Orange}${SUFFIX};
  1025. }
  1026. .navbar-nav>.active>a, .navbar-nav>.active>a:focus, .navbar-nav>.active>a:hover, .navbar-nav>.open>a, .dropdown-menu { background: ${Background}${SUFFIX}; }
  1027. .dropdown-menu>li>a:focus, .dropdown-menu>li>a:hover { background: ${Grey}${SUFFIX}; }
  1028. .pagination .active a {
  1029. border: none${SUFFIX};
  1030. color: ${LabelColor}${SUFFIX};
  1031. }
  1032. .pagination>li>a, .pagination>li>span {
  1033. background-color: ${Grey}${SUFFIX};
  1034. border: none${SUFFIX};
  1035. color: ${LabelColor}${SUFFIX};
  1036. }
  1037. tr, .modal-content, .alert {
  1038. background: ${SecondaryBackground}${SUFFIX};
  1039. box-shadow: none${SUFFIX};
  1040. }
  1041. tr:hover { background: ${Grey}${SUFFIX}; }
  1042. `);
  1043. if (path === "waterfall") {
  1044. GM_addStyle(`
  1045. .item a { background: ${SecondaryBackground}${SUFFIX}; }
  1046. .photo-info {
  1047. background: ${SecondaryBackground}${SUFFIX};
  1048. color: ${LabelColor}${SUFFIX};
  1049. }
  1050. date { color: ${SecondaryLabelColor}${SUFFIX}; }
  1051. .nav-pills>li.active>a, .nav-pills>li.active>a:focus, .nav-pills>li.active>a:hover, .nav-pills>li>a:focus, .nav-pills>li>a:hover { background-color: ${Grey}${SUFFIX}; }
  1052. `);
  1053. }
  1054. if (path === "genre") GM_addStyle(`.genre-box { background-color: ${SecondaryBackground}${SUFFIX}; }`);
  1055. if (path === "forum") {
  1056. GM_addStyle(`
  1057. #toptb,
  1058. .mn > div,
  1059. .biaoqicn_show.slidebar a:not(.on),
  1060. .new4_list_top .dspanonhover,
  1061. div[id*="con_NewOne_"],
  1062. .frame,
  1063. .frame-tab,
  1064. .main-right-p15,
  1065. .bm.bmw.cl,
  1066. .bm.bmw.cl .bm_h.cl { background: ${SecondaryBackground}${SUFFIX}; }
  1067. .jav-footer,
  1068. .nav-title.nav-active a,
  1069. .menu-body-panel.login-detail-wrap,
  1070. .menu-body-panel .icon-arrow-t,
  1071. .new4_list_top,
  1072. .comment-excerpt { background: ${Background}${SUFFIX}; }
  1073. .main-right-zuixin .comment-excerpt:before { border-bottom-color: ${Background}${SUFFIX}; }
  1074. .main-right-tit span { color: ${LabelColor}${SUFFIX}; }
  1075. `);
  1076. }
  1077. if (path === "details") {
  1078. GM_addStyle(`
  1079. .movie, .sample-box, .movie-box, .photo-info { background: ${SecondaryBackground}${SUFFIX}; }
  1080. .photo-info { color: ${LabelColor}${SUFFIX}; }
  1081. .avatar-box, .avatar-box span, .info ul li, .info .star-name {
  1082. background: ${SecondaryBackground}${SUFFIX};
  1083. border-color: ${Grey}${SUFFIX};
  1084. color: ${LabelColor}${SUFFIX};
  1085. }
  1086. `);
  1087. }
  1088. };
  1089.  
  1090. if (/javbus\.com/g.test(host)) {
  1091. const menus = [
  1092. { title: "点击事件", name: "ck", val: ck, key: "c" },
  1093. { title: "暗黑模式", name: "dm", val: dm, key: "d" },
  1094. { title: "滚动加载", name: "lm", val: lm, key: "s" },
  1095. ];
  1096. for (const { title, name, val, key } of menus) {
  1097. GM_registerMenuCommand(
  1098. `${val ? "关闭" : "开启"}${title}`,
  1099. () => {
  1100. GM_setValue(name, !val);
  1101. location.reload();
  1102. },
  1103. key
  1104. );
  1105. }
  1106. GM_registerMenuCommand(
  1107. "离线后操作根目录cid",
  1108. () => {
  1109. const rid = prompt("用于离线后移动,删除等操作", rootId);
  1110. GM_setValue("rid", rid);
  1111. location.reload();
  1112. },
  1113. "a"
  1114. );
  1115.  
  1116. lf.config({ name: "JBDB", storeName: "AVS" });
  1117.  
  1118. GM_addStyle(`
  1119. .ad-box { display: none${SUFFIX}; }
  1120. .ellipsis {
  1121. overflow : hidden;
  1122. text-overflow: ellipsis;
  1123. display: -webkit-box;
  1124. -webkit-line-clamp: 1;
  1125. -webkit-box-orient: vertical;
  1126. }
  1127. .line-4 { -webkit-line-clamp: 4; }
  1128. .playBtn { position: relative; }
  1129. .playBtn:after {
  1130. position: absolute;
  1131. background: url(https://javdb.com/packs/media/images/btn-play-b414746c.svg) 50% no-repeat;
  1132. background-color: rgba(0,0,0,.2);
  1133. background-size: 40px;
  1134. content: "";
  1135. top: 0;
  1136. right: 0;
  1137. bottom: 0;
  1138. left: 0;
  1139. transition: all .3s ease-out;
  1140. opacity: .85;
  1141. }
  1142. .playBtn:hover::after { background-color: rgba(0,0,0,0); }
  1143. .playBtn img { filter: none${SUFFIX}; }
  1144. `);
  1145.  
  1146. const pathReg = {
  1147. waterfall:
  1148. /^\/((uncensored|uncensored\/)?(page\/\d+)?$)|((uncensored\/)?((search|searchstar|actresses|genre|star|studio|label|series|director|member)+\/)|actresses(\/\d+)?)+/i,
  1149. genre: /^\/(uncensored\/)?genre$/i,
  1150. forum: /^\/forum\//i,
  1151. resource: /^\/resource$/i,
  1152. details: /^\/[\w]+(-|_)?[\d]*.*$/i,
  1153. };
  1154. const path = Object.keys(pathReg).filter(key => pathReg[key].test(pathname))[0];
  1155. if (!path) return;
  1156. dm && darkMode(path);
  1157. const jav = new JavBusScript()[path];
  1158. if (!jav) return;
  1159. jav.docStart();
  1160. doc.addEventListener("DOMContentLoaded", () => {
  1161. scriptStart();
  1162. jav.contentLoaded();
  1163. });
  1164. window.addEventListener("load", jav.load);
  1165. }
  1166. if (/captchaapi\.115\.com/g.test(host)) {
  1167. doc.addEventListener("DOMContentLoaded", () => {
  1168. window.focus();
  1169. const btn = doc.querySelector("#js_ver_code_box button[rel='verify']");
  1170. btn.addEventListener("click", () => {
  1171. const interval = setInterval(() => {
  1172. if (doc.querySelector("div[rel='error_box']").getAttribute("style").indexOf("none") !== -1) {
  1173. window.open("", "_self");
  1174. window.close();
  1175. }
  1176. }, 300);
  1177. setTimeout(() => clearInterval(interval), 600);
  1178. });
  1179. });
  1180. }
  1181. })();