- // ==UserScript==
- // @name NLegs Loader
- // @version 2025.3.12
- // @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[a/div[contains(@style,'thumb')]]";
- 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));
- gx(xpath).parentNode.appendChild(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)');
- }
- }
-
- const div = document.createElement('div');
- div.innerText = displayLanguage.str_13;
- div.className = 'btn btn-primary getBigImg';
- div.addEventListener("click", () => {
- getAllOriginal();
- });
- ele.appendChild(div);
- const div2 = document.createElement('div');
- div2.innerText = displayLanguage.str_14;
- div2.className = 'btn btn-primary imgDownload';
- div2.addEventListener("click", () => {
- imgDownload();
- });
- ele.appendChild(div2);
- const div3 = document.createElement('div');
- div3.innerText = displayLanguage.str_12;
- div3.className = 'btn btn-primary imgDownload zipmsg';
- div3.addEventListener("click", () => {
- imgZipDownload();
- });
- ele.appendChild(div3);
- }
-
- 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 {
- 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);
-
- })();