NLegs Loader

Loads original images in one page

  1. // ==UserScript==
  2. // @name NLegs Loader
  3. // @version 2025.3.12
  4. // @description Loads original images in one page
  5. // @author 德克斯DEX
  6. // @match *://www.nlegs.com/girls/*.html
  7. // @match *://www.honeyleg.com/article/*.html
  8. // @match *://www.ladylap.com/show/*
  9. // @match *://www.nuyet.com/gallery/*
  10. // @match *://www.legbabe.com/hot/*
  11. // @icon 
  12. // @license MIT
  13. // @namespace https://greasyfork.org/users/20361
  14. // @grant GM_registerMenuCommand
  15. // @grant GM.registerMenuCommand
  16. // @grant GM_openInTab
  17. // @grant GM.openInTab
  18. // @grant GM_getValue
  19. // @grant GM.getValue
  20. // @grant GM_setValue
  21. // @grant GM.setValue
  22. // @grant unsafeWindow
  23. // @require https://update.greasyfork.org/scripts/473358/1237031/JSZip.js
  24. // ==/UserScript==
  25.  
  26. /*
  27. 此站大圖質量還算不錯的,可惜人機驗證神煩!!!
  28.  
  29. 獲取大圖操作
  30. 1.自動取得所有預覽圖
  31. 2.手動點擊載入全部大圖按鈕來獲取大圖
  32. 3.等待替換元素
  33. 4.遇到人機驗證會跳出警告結束取得迴圈
  34. 5.在新開啟的分頁完成人機驗證
  35. 6.回來繼續按載入大圖按鈕取得大圖
  36.  
  37. 東方永頁機用戶請添加黑名單網址避免衝突
  38. https://www.nlegs.com/girls/*.html
  39. https://www.honeyleg.com/article/*.html
  40. https://www.ladylap.com/show/*
  41. https://www.nuyet.com/gallery/*
  42. https://www.legbabe.com/hot/*
  43. */
  44.  
  45. (async () => {
  46. 'use strict';
  47. const language = navigator.language;
  48. let displayLanguage = {};
  49. switch (language) {
  50. case "zh-TW":
  51. case "zh-HK":
  52. case "zh-Hant-TW":
  53. case "zh-Hant-HK":
  54. displayLanguage = {
  55. str_01: "獲取預覽圖遇到了人機驗證,將重新載入頁面",
  56. str_02: "預覽圖連一張都沒有了!",
  57. str_03: "獲取大圖中請勿重複操作!",
  58. str_04: "點擊繼續載入大圖",
  59. str_05: "獲取大圖中斷,遇到了人機驗證,請在新開啟的分頁裡完成人機驗證後,再回來按載入大圖按鈕繼續獲取大圖。",
  60. str_06: "所有大圖獲取完畢",
  61. str_07: "大圖一張也沒有!",
  62. str_08: "獲取大圖或下載或壓縮中請等待完成再操作!",
  63. str_09: "下載第",
  64. str_10: "張",
  65. str_11: "壓縮進度: ",
  66. str_12: "壓縮打包下載圖片",
  67. str_13: "點擊載入全部大圖",
  68. str_14: "鏈接逐張下載大圖",
  69. str_15: "圖片自適應視窗"
  70. };
  71. break;
  72. case "zh-CN":
  73. case "zh-Hans-CN":
  74. displayLanguage = {
  75. str_01: "获取预览图遇到了人机验证,将重新加载页面",
  76. str_02: "预览图连一张都没有了!",
  77. str_03: "获取大图中请勿重复操作!",
  78. str_04: "点击继续加载大图",
  79. str_05: "获取大图中断,遇到了人机验证,请在新开启的标籤页里完成人机验证后,再回来按加载大图按钮继续获取大图。",
  80. str_06: "所有大图获取完毕",
  81. str_07: "大图一张也没有!",
  82. str_08: "获取大图或下载或压缩中请等待完成再操作!",
  83. str_09: "下载第",
  84. str_10: "张",
  85. str_11: "压缩进度: ",
  86. str_12: "压缩打包下载图片",
  87. str_13: "点击加载全部大图",
  88. str_14: "链接逐张下载大图",
  89. str_15: "图片自适应窗口"
  90. };
  91. break;
  92. default:
  93. displayLanguage = {
  94. str_01: "Get preview Encountered human-machine verification will reload the page",
  95. str_02: "There’s not even a single preview image left.",
  96. str_03: "Get original picturesing Do not repeat operations",
  97. str_04: "Click to load",
  98. str_05: "Get original image interrupt Encountered human-machine verification Please complete the human-machine verification in the newly opened tab. come back again Click to load",
  99. str_06: "get completed",
  100. str_07: "There is not a single original picture",
  101. str_08: "Obtaining original image or downloading or compressing Please wait until completion before proceeding",
  102. str_09: "download No.",
  103. str_10: "P",
  104. str_11: "progress: ",
  105. str_12: "zip download",
  106. str_13: "Click to load",
  107. str_14: "link download",
  108. str_15: "Image adaptive viewport"
  109. };
  110. break;
  111. }
  112. const resBlobArray = [];
  113. const ge = (selector, doc) => (doc || document).querySelector(selector);
  114. const gae = (selector, doc) => (doc || document).querySelectorAll(selector);
  115. const gx = (xpath, doc) => (doc || document).evaluate(xpath, (doc || document), null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
  116. const gax = (xpath, doc) => {
  117. let nodes = [];
  118. let results = (doc || document).evaluate(xpath, (doc || document), null, XPathResult.ANY_TYPE, null);
  119. let node;
  120. while (node = results.iterateNext()) {
  121. nodes.push(node);
  122. }
  123. return nodes;
  124. };
  125. const waitEle = selector => new Promise(resolve => {
  126. const loop = setInterval(() => {
  127. let ele = ge(selector);
  128. if (!!ele) {
  129. clearInterval(loop);
  130. resolve(ele);
  131. }
  132. }, 100);
  133. });
  134. const parseHTML = str => new DOMParser().parseFromString(str, 'text/html');
  135. const openInNewTab = () => gae('a[href*=image]').forEach(a => (a.setAttribute('target', '_blank')));
  136. const checkImgStatus = src => new Promise(r => {
  137. const temp = new Image();
  138. temp.onload = r(true);
  139. temp.onerror = r(false);
  140. temp.src = src;
  141. });
  142. const _GM_openInTab = (() => typeof GM_openInTab != "undefined" ? GM_openInTab : GM.openInTab)();
  143. const _GM_getValue = (() => typeof GM_getValue != "undefined" ? GM_getValue : GM.getValue)();
  144. const _GM_setValue = (() => typeof GM_setValue != "undefined" ? GM_setValue : GM.setValue)();
  145. const _GM_registerMenuCommand = (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : GM.registerMenuCommand)();
  146.  
  147. let nlegsImgMode = _GM_getValue("nlegsImgMode");
  148.  
  149. if (nlegsImgMode == undefined) {
  150. _GM_setValue("nlegsImgMode", 0);
  151. nlegsImgMode = 0;
  152. }
  153.  
  154. _GM_registerMenuCommand(nlegsImgMode == 0 ? `❌ ${displayLanguage.str_15}` : `✔️ ${displayLanguage.str_15}`, () => {
  155. nlegsImgMode == 0 ? _GM_setValue("nlegsImgMode", 1) : _GM_setValue("nlegsImgMode", 0);
  156. location.reload();
  157. });
  158.  
  159. const lastPage = await waitEle('.pagination>li:last-child>a');
  160.  
  161. if (lastPage.innerText == 1) {
  162. addButton();
  163. openInNewTab();
  164. } else {
  165. const pages = gae('.pagination>li>a');
  166. const getAllThumb = async () => {
  167. for (let i = 1; i < pages.length; i++) {
  168. const res = await fetch(pages[i].href, {
  169. cache: "no-store"
  170. });
  171. const lastUrl = await res.url;
  172. if (lastUrl.includes("hcaptcha")) {
  173. document.title = displayLanguage.str_01;
  174. return location.reload();
  175. }
  176. const resText = await res.text();
  177. const doc = await parseHTML(resText);
  178. const xpath = "//div[a/div[contains(@style,'thumb')]]";
  179. if (!gx(xpath, doc)) {
  180. document.title = displayLanguage.str_01;
  181. return location.reload();
  182. }
  183. const thumbs = gax(xpath, doc);
  184. console.log(`第${parseInt(i)+1}頁\n`, thumbs);
  185. const fragment = new DocumentFragment();
  186. thumbs.forEach(thumb => fragment.appendChild(thumb));
  187. gx(xpath).parentNode.appendChild(fragment);
  188. const e = '.pagination';
  189. ge(e).outerHTML = ge(e, doc).outerHTML;
  190. }
  191. addButton();
  192. openInNewTab();
  193. };
  194. getAllThumb();
  195. }
  196.  
  197. const getAllOriginal = async () => {
  198. const links = gae('a[href*=image]');
  199. if (!links[0]) {
  200. alert(displayLanguage.str_02);
  201. return;
  202. }
  203. if (/\d+/.test(ge('.getBigImg').innerText)) {
  204. return alert(displayLanguage.str_03);
  205. }
  206. for (let i = 0; i < links.length; i++) {
  207. const res = await fetch(links[i].href, {
  208. cache: "no-store"
  209. });
  210. const lastUrl = await res.url;
  211. if (lastUrl.includes("hcaptcha")) {
  212. ge('.getBigImg').innerText = displayLanguage.str_04;
  213. _GM_openInTab(ge('a[href*=image]').href);
  214. return alert(displayLanguage.str_05);
  215. }
  216. const resText = await res.text();
  217. const doc = await parseHTML(resText);
  218. const imgRes = ge('img[id^=Image]', doc);
  219. if (!imgRes) {
  220. ge('.getBigImg').innerText = displayLanguage.str_04;
  221. _GM_openInTab(ge('a[href*=image]').href);
  222. return alert(displayLanguage.str_05);
  223. } else {
  224. ge('.getBigImg').innerText = `獲取第${parseInt(i)+1}/${links.length}張`;
  225. const res = await fetch(imgRes.src, {
  226. cache: "no-store"
  227. });
  228. const resBlob = await res.blob();
  229. const objectURL = URL.createObjectURL(resBlob);
  230. const check = await checkImgStatus(objectURL);
  231. if (check != true) {
  232. ge('.getBigImg').innerText = displayLanguage.str_04;
  233. _GM_openInTab(ge('a[href*=image]').href);
  234. return alert(displayLanguage.str_05);
  235. }
  236. resBlobArray.push(resBlob);
  237. links[i].parentNode.outerHTML = `<img class="${nlegsImgMode == 0 ? "auto" : "vh"}" src="${objectURL}">`;
  238. }
  239. }
  240. console.log('所有圖片Blob數據\n', resBlobArray);
  241. ge('.getBigImg').innerText = displayLanguage.str_06;
  242. setTimeout(() => (ge('.getBigImg').style.display = "none"), 1000);
  243. };
  244.  
  245. const imgZipDownload = async () => {
  246. const imgs = gae('img[src^=blob]');
  247. if (!imgs[0]) {
  248. return alert(displayLanguage.str_07);
  249. }
  250. if (/\d+/.test(ge('.zipmsg').innerText) || /\d+/.test(ge('.getBigImg').innerText)) {
  251. return alert(displayLanguage.str_08);
  252. }
  253. const imgsNum = resBlobArray.length;
  254. const title = ge('[class^=container] p').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim();
  255. const zip = new JSZip();
  256. const zipFolder = zip.folder(`${title} [${imgsNum}P]`);
  257. for (let i = 0; i < imgsNum; i++) {
  258. const n = parseInt(i) + 1;
  259. const padStart = String(imgsNum).length;
  260. const pn = String(n).padStart(padStart, "0");
  261. const fileName = `${pn}P.jpg`;
  262. ge('.zipmsg').innerText = `${displayLanguage.str_09}${n}/${imgsNum}${displayLanguage.str_10}`;
  263. console.log(`第${n}/${imgsNum}張,檔案名:${fileName},大小:${parseInt(resBlobArray[i].size / 1024)} Kb,下載完成!等待壓縮...`);
  264. zipFolder.file(fileName, resBlobArray[i], {
  265. binary: true
  266. });
  267. }
  268. zip.generateAsync({
  269. type: "blob"
  270. }, (metadata) => {
  271. ge('.zipmsg').innerText = displayLanguage.str_11 + metadata.percent.toFixed(2) + ' %';
  272. console.log('progression: ' + metadata.percent.toFixed(2) + ' %');
  273. }).then(data => {
  274. console.log('ZIP壓縮檔數據\n', data);
  275. ge('.zipmsg').innerText = displayLanguage.str_12;
  276. let a = document.createElement('a');
  277. a.href = URL.createObjectURL(data);
  278. a.download = `${title} [${imgsNum}P].zip`;
  279. try {
  280. a.dispatchEvent(new MouseEvent("click"));
  281. } catch {
  282. let b = document.createEvent("MouseEvents");
  283. b.initMouseEvent("click", !0, !0, window, 0, 0, 0, 80, 20, !1, !1, !1, !1, 0, null);
  284. a.dispatchEvent(b);
  285. }
  286. setTimeout(() => URL.revokeObjectURL(data), 4000);
  287. });
  288. };
  289.  
  290. const imgDownload = async () => {
  291. const imgs = gae('img[src^=blob]');
  292. const imgsNum = imgs.length;
  293. if (!imgs[0]) {
  294. return alert(displayLanguage.str_07);
  295. }
  296. const title = ge('[class^=container] p').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim();
  297. for (let i = 0; i < imgsNum; i++) {
  298. const n = parseInt(i) + 1;
  299. const padStart = String(imgsNum).length;
  300. const pn = String(n).padStart(padStart, "0");
  301. const a = document.createElement('a');
  302. a.href = imgs[i].src;
  303. a.download = `${title}_${pn}P.jpg`;
  304. try {
  305. a.dispatchEvent(new MouseEvent("click"));
  306. } catch {
  307. let b = document.createEvent("MouseEvents");
  308. b.initMouseEvent("click", !0, !0, window, 0, 0, 0, 80, 20, !1, !1, !1, !1, 0, null);
  309. a.dispatchEvent(b);
  310. }
  311. await new Promise(resolve => setTimeout(resolve, 100));
  312. }
  313. };
  314.  
  315. function addButton() {
  316. let ele;
  317. if (location.origin.includes("nlegs")) {
  318. if (ge("span.title")) {
  319. ele = ge('div:has(>span.title)');
  320. } else {
  321. ele = ge('.container div:has(>p)');
  322. }
  323. } else {
  324. if (ge('div:has(>p>input)')) {
  325. ele = ge('div:has(>p>input)');
  326. } else if (ge('#download')) {
  327. ele = ge('#download');
  328. } else {
  329. ele = ge('.col-md-12:has(>p)');
  330. }
  331. }
  332.  
  333. const div = document.createElement('div');
  334. div.innerText = displayLanguage.str_13;
  335. div.className = 'btn btn-primary getBigImg';
  336. div.addEventListener("click", () => {
  337. getAllOriginal();
  338. });
  339. ele.appendChild(div);
  340. const div2 = document.createElement('div');
  341. div2.innerText = displayLanguage.str_14;
  342. div2.className = 'btn btn-primary imgDownload';
  343. div2.addEventListener("click", () => {
  344. imgDownload();
  345. });
  346. ele.appendChild(div2);
  347. const div3 = document.createElement('div');
  348. div3.innerText = displayLanguage.str_12;
  349. div3.className = 'btn btn-primary imgDownload zipmsg';
  350. div3.addEventListener("click", () => {
  351. imgZipDownload();
  352. });
  353. ele.appendChild(div3);
  354. }
  355.  
  356. const addReturnTopButton = () => {
  357. const a = document.createElement('a');
  358. a.href = 'javascript:void(0);';
  359. a.setAttribute('onclick', "window.scrollTo({top:0,behavior:'smooth'});");
  360. const img = new Image();
  361. img.src = '';
  362. img.className = 'returnTop';
  363. a.appendChild(img);
  364. document.body.appendChild(a);
  365. };
  366. addReturnTopButton();
  367.  
  368. const addGlobalStyle = css => {
  369. const style = document.createElement('style');
  370. style.type = 'text/css';
  371. style.innerHTML = css;
  372. document.head.appendChild(style);
  373. };
  374. const css = `
  375. .returnTop {
  376. position: fixed;
  377. right: 10px;
  378. bottom: 60px;
  379. width: 53px;
  380. z-index: 99;
  381. opacity: 0.5;
  382. }
  383. img[src^=blob].auto {
  384. width: auto;
  385. height: auto;
  386. max-width: 100%;
  387. display: block;
  388. margin: 0 auto;
  389. }
  390. img[src^=blob].vh {
  391. width: auto;
  392. height: auto;
  393. max-width: 100%;
  394. max-height: 99vh;
  395. display: block;
  396. margin: 0 auto;
  397. }
  398. .imgDownload {
  399. font-size: 16px;
  400. font-family: Arial,sans-serif!important;
  401. line-height: 24px;
  402. width: 150px;
  403. padding: 4px;
  404. margin-right: 5px;
  405. margin-bottom: 10px;
  406. }
  407. .getBigImg {
  408. font-size: 16px;
  409. font-family: Arial,sans-serif!important;
  410. line-height: 24px;
  411. width: 150px;
  412. position: fixed;
  413. z-index:999;
  414. bottom: 10px;
  415. left: 50%;
  416. margin-left: -75px;
  417. padding: 4px;
  418. }
  419. strong~div {
  420. display: table-cell!important;
  421. }
  422. `;
  423. addGlobalStyle(css);
  424.  
  425. })();