您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Loads original images in one page
// ==UserScript== // @name NLegs Loader // @version 2025.5.25 // @description Loads original images in one page // @author 德克斯DEX // @match *://www.nlegs.com/girls/*.html // @match *://www.honeyleg.com/article/*.html // @match *://www.ladylap.com/show/* // @match *://www.nuyet.com/gallery/* // @match *://www.legbabe.com/hot/* // @icon  // @license MIT // @namespace https://greasyfork.org/users/20361 // @grant GM_registerMenuCommand // @grant GM.registerMenuCommand // @grant GM_openInTab // @grant GM.openInTab // @grant GM_getValue // @grant GM.getValue // @grant GM_setValue // @grant GM.setValue // @grant unsafeWindow // @require https://update.greasyfork.org/scripts/473358/1237031/JSZip.js // ==/UserScript== /* 此站大圖質量還算不錯的,可惜人機驗證神煩!!! 獲取大圖操作 1.自動取得所有預覽圖 2.手動點擊載入全部大圖按鈕來獲取大圖 3.等待替換元素 4.遇到人機驗證會跳出警告結束取得迴圈 5.在新開啟的分頁完成人機驗證 6.回來繼續按載入大圖按鈕取得大圖 東方永頁機用戶請添加黑名單網址避免衝突 https://www.nlegs.com/girls/*.html https://www.honeyleg.com/article/*.html https://www.ladylap.com/show/* https://www.nuyet.com/gallery/* https://www.legbabe.com/hot/* */ (async () => { 'use strict'; const language = navigator.language; let displayLanguage = {}; switch (language) { case "zh-TW": case "zh-HK": case "zh-Hant-TW": case "zh-Hant-HK": displayLanguage = { str_01: "獲取預覽圖遇到了人機驗證,將重新載入頁面", str_02: "預覽圖連一張都沒有了!", str_03: "獲取大圖中請勿重複操作!", str_04: "點擊繼續載入大圖", str_05: "獲取大圖中斷,遇到了人機驗證,請在新開啟的分頁裡完成人機驗證後,再回來按載入大圖按鈕繼續獲取大圖。", str_06: "所有大圖獲取完畢", str_07: "大圖一張也沒有!", str_08: "獲取大圖或下載或壓縮中請等待完成再操作!", str_09: "下載第", str_10: "張", str_11: "壓縮進度: ", str_12: "壓縮打包下載圖片", str_13: "點擊載入全部大圖", str_14: "鏈接逐張下載大圖", str_15: "圖片自適應視窗" }; break; case "zh-CN": case "zh-Hans-CN": displayLanguage = { str_01: "获取预览图遇到了人机验证,将重新加载页面", str_02: "预览图连一张都没有了!", str_03: "获取大图中请勿重复操作!", str_04: "点击继续加载大图", str_05: "获取大图中断,遇到了人机验证,请在新开启的标籤页里完成人机验证后,再回来按加载大图按钮继续获取大图。", str_06: "所有大图获取完毕", str_07: "大图一张也没有!", str_08: "获取大图或下载或压缩中请等待完成再操作!", str_09: "下载第", str_10: "张", str_11: "压缩进度: ", str_12: "压缩打包下载图片", str_13: "点击加载全部大图", str_14: "链接逐张下载大图", str_15: "图片自适应窗口" }; break; default: displayLanguage = { str_01: "Get preview Encountered human-machine verification will reload the page", str_02: "There’s not even a single preview image left.", str_03: "Get original picturesing Do not repeat operations", str_04: "Click to load", 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", str_06: "get completed", str_07: "There is not a single original picture", str_08: "Obtaining original image or downloading or compressing Please wait until completion before proceeding", str_09: "download No.", str_10: "P", str_11: "progress: ", str_12: "zip download", str_13: "Click to load", str_14: "link download", str_15: "Image adaptive viewport" }; break; } const resBlobArray = []; const ge = (selector, doc) => (doc || document).querySelector(selector); const gae = (selector, doc) => (doc || document).querySelectorAll(selector); const gx = (xpath, doc) => (doc || document).evaluate(xpath, (doc || document), null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; const gax = (xpath, doc) => { let nodes = []; let results = (doc || document).evaluate(xpath, (doc || document), null, XPathResult.ANY_TYPE, null); let node; while (node = results.iterateNext()) { nodes.push(node); } return nodes; }; const waitEle = selector => new Promise(resolve => { const loop = setInterval(() => { let ele = ge(selector); if (!!ele) { clearInterval(loop); resolve(ele); } }, 100); }); const parseHTML = str => new DOMParser().parseFromString(str, 'text/html'); const openInNewTab = () => gae('a[href*=image]').forEach(a => (a.setAttribute('target', '_blank'))); const checkImgStatus = src => new Promise(r => { const temp = new Image(); temp.onload = r(true); temp.onerror = r(false); temp.src = src; }); const _GM_openInTab = (() => typeof GM_openInTab != "undefined" ? GM_openInTab : GM.openInTab)(); const _GM_getValue = (() => typeof GM_getValue != "undefined" ? GM_getValue : GM.getValue)(); const _GM_setValue = (() => typeof GM_setValue != "undefined" ? GM_setValue : GM.setValue)(); const _GM_registerMenuCommand = (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : GM.registerMenuCommand)(); let nlegsImgMode = _GM_getValue("nlegsImgMode"); if (nlegsImgMode == undefined) { _GM_setValue("nlegsImgMode", 0); nlegsImgMode = 0; } _GM_registerMenuCommand(nlegsImgMode == 0 ? `❌ ${displayLanguage.str_15}` : `✔️ ${displayLanguage.str_15}`, () => { nlegsImgMode == 0 ? _GM_setValue("nlegsImgMode", 1) : _GM_setValue("nlegsImgMode", 0); location.reload(); }); const lastPage = await waitEle('.pagination>li:last-child>a'); if (lastPage.innerText == 1) { addButton(); openInNewTab(); } else { const pages = gae('.pagination>li>a'); const getAllThumb = async () => { for (let i = 1; i < pages.length; i++) { const res = await fetch(pages[i].href, { cache: "no-store" }); const lastUrl = await res.url; if (lastUrl.includes("hcaptcha")) { document.title = displayLanguage.str_01; return location.reload(); } const resText = await res.text(); const doc = await parseHTML(resText); const xpath = "//div[div[a[@class='thumbnail']]]"; if (!gx(xpath, doc)) { document.title = displayLanguage.str_01; return location.reload(); } const thumbs = gax(xpath, doc); console.log(`第${parseInt(i)+1}頁\n`, thumbs); const fragment = new DocumentFragment(); thumbs.forEach(thumb => fragment.appendChild(thumb)); ge("div:has(>.pagination)").before(fragment); const e = '.pagination'; ge(e).outerHTML = ge(e, doc).outerHTML; } addButton(); openInNewTab(); }; getAllThumb(); } const getAllOriginal = async () => { const links = gae('a[href*=image]'); if (!links[0]) { alert(displayLanguage.str_02); return; } if (/\d+/.test(ge('.getBigImg').innerText)) { return alert(displayLanguage.str_03); } for (let i = 0; i < links.length; i++) { const res = await fetch(links[i].href, { cache: "no-store" }); const lastUrl = await res.url; if (lastUrl.includes("hcaptcha")) { ge('.getBigImg').innerText = displayLanguage.str_04; _GM_openInTab(ge('a[href*=image]').href); return alert(displayLanguage.str_05); } const resText = await res.text(); const doc = await parseHTML(resText); const imgRes = ge('img[id^=Image]', doc); if (!imgRes) { ge('.getBigImg').innerText = displayLanguage.str_04; _GM_openInTab(ge('a[href*=image]').href); return alert(displayLanguage.str_05); } else { ge('.getBigImg').innerText = `獲取第${parseInt(i)+1}/${links.length}張`; const res = await fetch(imgRes.src, { cache: "no-store" }); const resBlob = await res.blob(); const objectURL = URL.createObjectURL(resBlob); const check = await checkImgStatus(objectURL); if (check != true) { ge('.getBigImg').innerText = displayLanguage.str_04; _GM_openInTab(ge('a[href*=image]').href); return alert(displayLanguage.str_05); } resBlobArray.push(resBlob); links[i].parentNode.outerHTML = `<img class="${nlegsImgMode == 0 ? "auto" : "vh"}" src="${objectURL}">`; } } console.log('所有圖片Blob數據\n', resBlobArray); ge('.getBigImg').innerText = displayLanguage.str_06; setTimeout(() => (ge('.getBigImg').style.display = "none"), 1000); }; const imgZipDownload = async () => { const imgs = gae('img[src^=blob]'); if (!imgs[0]) { return alert(displayLanguage.str_07); } if (/\d+/.test(ge('.zipmsg').innerText) || /\d+/.test(ge('.getBigImg').innerText)) { return alert(displayLanguage.str_08); } const imgsNum = resBlobArray.length; const title = ge('[class^=container] p').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim(); const zip = new JSZip(); const zipFolder = zip.folder(`${title} [${imgsNum}P]`); for (let i = 0; i < imgsNum; i++) { const n = parseInt(i) + 1; const padStart = String(imgsNum).length; const pn = String(n).padStart(padStart, "0"); const fileName = `${pn}P.jpg`; ge('.zipmsg').innerText = `${displayLanguage.str_09}${n}/${imgsNum}${displayLanguage.str_10}`; console.log(`第${n}/${imgsNum}張,檔案名:${fileName},大小:${parseInt(resBlobArray[i].size / 1024)} Kb,下載完成!等待壓縮...`); zipFolder.file(fileName, resBlobArray[i], { binary: true }); } zip.generateAsync({ type: "blob" }, (metadata) => { ge('.zipmsg').innerText = displayLanguage.str_11 + metadata.percent.toFixed(2) + ' %'; console.log('progression: ' + metadata.percent.toFixed(2) + ' %'); }).then(data => { console.log('ZIP壓縮檔數據\n', data); ge('.zipmsg').innerText = displayLanguage.str_12; let a = document.createElement('a'); a.href = URL.createObjectURL(data); a.download = `${title} [${imgsNum}P].zip`; try { a.dispatchEvent(new MouseEvent("click")); } catch { let b = document.createEvent("MouseEvents"); b.initMouseEvent("click", !0, !0, window, 0, 0, 0, 80, 20, !1, !1, !1, !1, 0, null); a.dispatchEvent(b); } setTimeout(() => URL.revokeObjectURL(data), 4000); }); }; const imgDownload = async () => { const imgs = gae('img[src^=blob]'); const imgsNum = imgs.length; if (!imgs[0]) { return alert(displayLanguage.str_07); } const title = ge('[class^=container] p').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim(); for (let i = 0; i < imgsNum; i++) { const n = parseInt(i) + 1; const padStart = String(imgsNum).length; const pn = String(n).padStart(padStart, "0"); const a = document.createElement('a'); a.href = imgs[i].src; a.download = `${title}_${pn}P.jpg`; try { a.dispatchEvent(new MouseEvent("click")); } catch { let b = document.createEvent("MouseEvents"); b.initMouseEvent("click", !0, !0, window, 0, 0, 0, 80, 20, !1, !1, !1, !1, 0, null); a.dispatchEvent(b); } await new Promise(resolve => setTimeout(resolve, 100)); } }; function addButton() { let ele; if (location.origin.includes("nlegs")) { if (ge("span.title")) { ele = ge('div:has(>span.title)'); } else { ele = ge('.container div:has(>p)'); } } else { if (ge('div:has(>p>input)')) { ele = ge('div:has(>p>input)'); } else if (ge('#download')) { ele = ge('#download'); } else { ele = ge('.col-md-12:has(>p) p'); } } const p_div = document.createElement('div'); p_div.style.marginBottom = "10px"; const div = document.createElement('div'); div.innerText = displayLanguage.str_13; div.className = 'btn btn-primary getBigImg'; div.addEventListener("click", () => { getAllOriginal(); }); p_div.appendChild(div); const div2 = document.createElement('div'); div2.innerText = displayLanguage.str_14; div2.className = 'btn btn-primary imgDownload'; div2.addEventListener("click", () => { imgDownload(); }); p_div.appendChild(div2); const div3 = document.createElement('div'); div3.innerText = displayLanguage.str_12; div3.className = 'btn btn-primary imgDownload zipmsg'; div3.addEventListener("click", () => { imgZipDownload(); }); p_div.appendChild(div3); ele.appendChild(p_div); } const addReturnTopButton = () => { const a = document.createElement('a'); a.href = 'javascript:void(0);'; a.setAttribute('onclick', "window.scrollTo({top:0,behavior:'smooth'});"); const img = new Image(); img.src = ''; img.className = 'returnTop'; a.appendChild(img); document.body.appendChild(a); }; addReturnTopButton(); const addGlobalStyle = css => { const style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = css; document.head.appendChild(style); }; const css = ` .returnTop { position: fixed; right: 10px; bottom: 60px; width: 53px; z-index: 99; opacity: 0.5; } img[src^=blob].auto { width: auto; height: auto; max-width: 100%; display: block; margin: 0 auto; } img[src^=blob].vh { width: auto; height: auto; max-width: 100%; max-height: 99vh; display: block; margin: 0 auto; } .imgDownload { display: inline; padding: 6px 12px!important; font-size: 16px; font-family: Arial,sans-serif!important; line-height: 24px; width: 150px; padding: 4px; margin-right: 5px; margin-bottom: 10px; } .getBigImg { font-size: 16px; font-family: Arial,sans-serif!important; line-height: 24px; width: 150px; position: fixed; z-index:999; bottom: 10px; left: 50%; margin-left: -75px; padding: 4px; } strong~div { display: table-cell!important; } `; addGlobalStyle(css); })();