NLegs Loader

Loads original images in one page

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         NLegs Loader
// @version      2025.9.19
// @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/*.html
// @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)();

    const item_xpath = "//div[a[@class='thumbnail']] | //div[a[div[@class='img-div']]] | //div[@class='col-md-2 col-sm-4 col-xs-12']";

    if (!gx(item_xpath)) {
        return;
    }

    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();
    });

    let get_error = false;

    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"
                });
                console.log(pages[i].href, "\n", res);
                if (!res.ok) {
                    get_error = true;
                    return alert(`Get image thumbs\n${pages[i].href}\nHTTP response status codes:${res.status}`);
                }
                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);
                if (!gx(item_xpath, doc)) {
                    document.title = displayLanguage.str_01;
                    return location.reload();
                }
                const thumbs = gax(item_xpath, doc);
                console.log(`第${parseInt(i)+1}頁\n`, thumbs);
                const fragment = new DocumentFragment();
                thumbs.forEach(thumb => fragment.appendChild(thumb));
                gx(item_xpath).parentNode.appendChild(fragment);
                const e = '.pagination';
                ge(e).outerHTML = ge(e, doc).outerHTML;
            }
            addButton();
            openInNewTab();
        };
        getAllThumb();
    }

    if (get_error) return;

    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);

})();