nlegs.com 聚圖&下載

如題,由於此站人機驗證出現頻繁,一部寫真可能需要分1~3次才能全部載入大圖。

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name         nlegs.com 聚圖&下載
// @version      1.1.1
// @description  如題,由於此站人機驗證出現頻繁,一部寫真可能需要分1~3次才能全部載入大圖。
// @author       tony0809
// @match        http*://www.nlegs.com/girls/*.html
// @match        http*://www.honeyleg.com/article/*.html
// @match        http*://www.ladylap.com/show/*
// @match        http*://www.nuyet.com/gallery/*
// @match        http*://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://cdn.jsdelivr.net/npm/jszip@3.9.1/dist/jszip.min.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/*
*/

(() => {
    '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 parseHTML = str => new DOMParser().parseFromString(str, 'text/html');
    const openInNewTab = () => {
        gae('a[href*=image]').forEach(a => {
            a.setAttribute('target', '_blank');
        });
    };
    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();
    });

    let loopFind = setInterval(() => {
        let set = ge('.pagination>li:last-child>a');
        if (set || /legbabe/.test(location.origin)) {
            clearInterval(loopFind);
            if (/legbabe/.test(location.origin)) {
                addButton();
                openInNewTab();
            } else if (set.innerText == 1) {
                addButton();
                openInNewTab();
            } else {
                let pages = gae('.pagination>li>a');
                const getAllThumb = async () => {
                    for (let i = 1; i < pages.length; i++) {
                        let res = await fetch(pages[i].href);
                        let resText = await res.text();
                        let doc = await parseHTML(resText);
                        let xpath = "//div[a/div[contains(@style,'thumb') and span]]";
                        if (!gx(xpath, doc)) {
                            alert(displayLanguage.str_01);
                            location.reload();
                            return;
                        }
                        let thumbs = gax(xpath, doc);
                        console.log(`第${parseInt(i)+1}頁\n`, thumbs);
                        let fragment = new DocumentFragment();
                        thumbs.forEach(thumb => {
                            fragment.appendChild(thumb);
                        });
                        gx(xpath).parentNode.appendChild(fragment);
                        let e = '.pagination';
                        ge(e).outerHTML = ge(e, doc).outerHTML;
                    }
                    addButton();
                    openInNewTab();
                };
                getAllThumb();
            }
        }
    }, 100);

    const getAllOriginal = async () => {
        let links = gae('a[href*=image]');
        if (!links[0]) {
            alert(displayLanguage.str_02);
            return;
        }
        if (/\d+/.test(ge('.getBigImg').innerText)) {
            alert(displayLanguage.str_03);
            return;
        }
        for (let i = 0; i < links.length; i++) {
            let res = await fetch(links[i].href);
            let resText = await res.text();
            let doc = await parseHTML(resText);
            let imgRes = ge('.img-res', doc);
            if (!imgRes) {
                ge('.getBigImg').innerText = displayLanguage.str_04;
                alert(displayLanguage.str_05);
                _GM_openInTab(ge('a[href*=image]').href);
                return;
            } else {
                ge('.getBigImg').innerText = `獲取第${parseInt(i)+1}/${links.length}張`;
                let res = await fetch(imgRes.src);
                let resBlob = await res.blob();
                resBlobArray.push(resBlob);
                let objectURL = URL.createObjectURL(resBlob);
                console.log(objectURL);
                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]) {
            alert(displayLanguage.str_07);
            return;
        }
        if (/\d+/.test(ge('.zipmsg').innerText) || /\d+/.test(ge('.getBigImg').innerText)) {
            alert(displayLanguage.str_08);
            return;
        }
        const imgsNum = resBlobArray.length;
        const title = ge('strong').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim();
        const zip = new JSZip();
        const zipFolder = zip.folder(`${title} [${imgsNum}P]`);
        for (let i = 0; i < imgsNum; i++) {
            let n = parseInt(i) + 1;
            let padStart = String(imgsNum).length;
            let pn = String(n).padStart(padStart, "0");
            let 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`;
            document.body.appendChild(a);
            a.click();
            a.remove();
            URL.revokeObjectURL(data);
        });
    };

    const imgDownload = async () => {
        const imgs = gae('img[src^=blob]');
        const imgsNum = imgs.length;
        if (!imgs[0]) {
            alert(displayLanguage.str_07);
            return;
        }
        let title = ge('strong').innerText.replace(/\[\d+[-\.\+\w]+\]/, '').trim();
        for (let i = 0; i < imgsNum; i++) {
            let n = parseInt(i) + 1;
            let padStart = String(imgsNum).length;
            let pn = String(n).padStart(padStart, "0");
            let a = document.createElement('a');
            a.href = imgs[i].src;
            a.download = `${title}_${pn}P.jpg`;
            document.body.appendChild(a);
            a.click();
            a.remove();
            await new Promise(resolve => setTimeout(resolve, 100));
        }
    };

    const addButton = () => {
        let ele;
        if (/nlegs/.test(location.origin)) {
            try {
                ele = ge('span.title').parentNode;
            } catch (e) {
                try {
                    ele = ge('strong').parentNode.parentNode;
                    ele.style.textAlign = "center";
                    ele.querySelector('div').style.display = "none";
                } catch (e) {
                    ele = ge('strong').parentNode;
                }
            }
        } else if (/nuyet/.test(location.origin)) {
            if (ge('#download')) {
                try {
                    ele = ge('.btn.btn-danger,.btn.btn-primary').parentNode.parentNode;
                } catch (e) {
                    ele = ge('#download').parentNode;
                }
            } else {
                ele = ge('strong').parentNode;
            }
        } else if (/legbabe/.test(location.origin)) {
            try {
                ele = ge('.btn.btn-danger,.btn.btn-primary').parentNode.parentNode;
            } catch (e) {
                ele = ge('strong').parentNode;
            }
        } else if (ge('#download')) {
            try {
                ele = ge('.btn.btn-danger,.btn.btn-primary').parentNode.parentNode;
            } catch (e) {
                ele = ge('strong').parentNode;
            }
        } else {
            ele = ge('strong').parentNode;
        }

        let div = document.createElement('div');
        div.innerText = displayLanguage.str_13;
        div.className = 'btn btn-primary getBigImg';
        div.addEventListener("click", () => {
            getAllOriginal();
        });
        ele.appendChild(div);
        let div2 = document.createElement('div');
        div2.innerText = displayLanguage.str_14;
        div2.className = 'btn btn-primary imgDownload';
        div2.addEventListener("click", () => {
            imgDownload();
        });
        ele.appendChild(div2);
        let 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 = () => {
        let a = document.createElement('a');
        a.href = 'javascript:void(0);';
        a.setAttribute('onclick', "window.scrollTo({top:0,behavior:'smooth'});");
        let img = new Image();
        img.src = '';
        img.className = 'returnTop';
        a.appendChild(img);
        document.body.appendChild(a);
    };
    addReturnTopButton();

    const addGlobalStyle = css => {
        let 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);

})();