nlegs.com 聚圖&下載

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

05.04.2023 itibariyledir. En son verisyonu görün.

Bu betiği kurabilmeniz için Tampermonkey, Greasemonkey ya da Violentmonkey gibi bir kullanıcı betiği eklentisini kurmanız gerekmektedir.

Bu betiği kurabilmeniz için Tampermonkey ya da Violentmonkey gibi bir kullanıcı betiği eklentisini kurmanız gerekmektedir.

Bu betiği kurabilmeniz için Tampermonkey ya da Violentmonkey gibi bir kullanıcı betiği eklentisini kurmanız gerekmektedir.

Bu betiği kurabilmeniz için Tampermonkey ya da Userscripts gibi bir kullanıcı betiği eklentisini kurmanız gerekmektedir.

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

Bu komut dosyasını yüklemek için bir kullanıcı komut dosyası yöneticisi uzantısı yüklemeniz gerekecek.

(Zaten bir kullanıcı komut dosyası yöneticim var, kurmama izin verin!)

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.

(Zateb bir user-style yöneticim var, yükleyeyim!)

// ==UserScript==
// @name         nlegs.com 聚圖&下載
// @version      1.0.3
// @description  如題,由於此站人機驗證出現頻繁,一部寫真可能需要分1~3次才能全部載入大圖。
// @author       tony0809
// @match        https://www.nlegs.com/girls/*.html
// @icon         
// @grant        none
// @license      MIT
// @namespace    https://greasyfork.org/users/20361
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/FileSaver.min.js
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/jszip.min.js
// ==/UserScript==

/*
腳本"图聚合展示by xhua",https://greasyfork.org/scripts/442098
對於此站也不能取得全部大圖,遇到人機驗證一樣撞牆,所以才寫了一個堪用的取得大圖腳本。
此站大圖質量算是不錯的,可惜人機驗證神煩!!!

獲取大圖操作
1.自動取得所有預覽圖
2.手動點擊載入全部大圖按鈕來獲取大圖
3.等待替換元素
4.遇到人機驗證會跳出警告結束取得迴圈
5.在新開啟的分頁完成人機驗證
6.回來繼續按載入大圖按鈕取得大圖

東方永頁機用戶請添加黑名單網址避免衝突
*://www.nlegs.com/girls/*.html

WIN10把默認圖片保存格式變成「jfif」了,怎麼變回「JPG」格式?
首先按鍵盤的「Win鍵+R鍵」,彈出「運行」對話框,輸入「regedit」,然後點Enter進入註冊表編輯器。
然後把路徑貼到地址欄裡Enter:
HKEY_CLASSES_ROOT\MIME\Database\Content Type\image/jpeg
再然後,右面的列表框中有個「Extension」選項,雙擊這一行點開,在「編輯字符串」對話框中,把「jfif」改爲「jpg」,最後點確定就可以了。
*/

(() => {
    'use strict';
    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');
        });
    };
    let loopFind = setInterval(() => {
        let set = ge('.pagination>li:last-child>a');
        if (set) {
            clearInterval(loopFind);
            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('獲取預覽圖遇到了人機驗證,將重新載入頁面');
                            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('預覽圖連一張都沒有了!');
            return;
        }
        if (/\d+/.test(ge('.getBigImg').innerText)) {
            alert('獲取大圖中請勿重複操作!');
            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 = '點擊繼續載入大圖';
                alert('獲取大圖中斷,遇到了人機驗證,請在新開啟的分頁裡完成人機驗證後,再回來按載入大圖按鈕繼續獲取大圖。');
                ge('a[href*=image][target=_blank]').click();
                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 = await URL.createObjectURL(resBlob);
                console.log(objectURL);
                links[i].parentNode.outerHTML = `<img src="${objectURL}">`;
            }
        }
        console.log('所有圖片Blob數據\n', resBlobArray);
        ge('.getBigImg').innerText = '所有大圖獲取完畢';
    };

    const imgZipDownload = async () => {
        const imgs = gae('img[src^=blob]');
        if (!imgs[0]) {
            alert('大圖一張也沒有!');
            return;
        }
        if (/\d+/.test(ge('.zipmsg').innerText) || /\d+/.test(ge('.getBigImg').innerText)) {
            alert('獲取大圖或下載或壓縮中請等待完成再操作!');
            return;
        }
        /*
        const imgsNum = imgs.length;
        const title = ge('strong').innerText.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 pn = n;
            if (i < 9) {
                pn = '00' + n;
            } else if (i < 99) {
                pn = '0' + n;
            }
            let fileName = `${pn}P.jpg`;
            ge('.zipmsg').innerText = `下載第${n}/${imgsNum}張`;
            await fetch(imgs[i].src).then(res => res.blob()).then(data => {
                zipFolder.file(fileName, data, {
                    binary: true
                });
                console.log(`第${n}/${imgsNum}張,檔案名:${fileName},大小:${parseInt(data.size / 1024)} Kb,下載完成!等待壓縮...`);
            });
        }
        */
        const imgsNum = resBlobArray.length;
        const title = ge('strong').innerText.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 pn = n;
            if (i < 9) {
                pn = '00' + n;
            } else if (i < 99) {
                pn = '0' + n;
            }
            let fileName = `${pn}P.jpg`;
            ge('.zipmsg').innerText = `下載第${n}/${imgsNum}張`;
            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 = '壓縮進度: ' + metadata.percent.toFixed(2) + ' %';
            console.log('progression: ' + metadata.percent.toFixed(2) + ' %');
        }).then(data => {
            console.log('ZIP壓縮檔數據\n', data);
            ge('.zipmsg').innerText = '壓縮打包下載';
            //saveAs(data, `${title} [${imgsNum}P].zip`);
            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 = () => {
        const imgs = gae('img[src^=blob]');
        const imgsNum = imgs.length;
        if (!imgs[0]) {
            alert('大圖一張也沒有!');
            return;
        }
        let title = ge('strong').innerText.trim();
        for (let i = 0; i < imgsNum; i++) {
            let n = parseInt(i) + 1;
            let pn = n;
            if (i < 9) {
                pn = '00' + n;
            } else if (i < 99) {
                pn = '0' + n;
            }
            let yes = confirm(`下載第 ${n}/${imgsNum} 張`);
            if (yes) {
                let a = document.createElement('a');
                a.href = imgs[i].src;
                a.download = `${title}_${pn}P.jpg`;
                document.body.appendChild(a);
                a.click();
                a.remove();
            }
        }
    };

    const addButton = () => {
        let ele = ge('span.title').parentNode;
        let div = document.createElement('div');
        div.innerText = '點擊載入全部大圖';
        div.className = 'btn btn-primary getBigImg';
        div.addEventListener("click", () => {
            getAllOriginal();
        });
        ele.appendChild(div);
        let div2 = document.createElement('div');
        div2.innerText = '鏈接逐張下載大圖';
        div2.className = 'btn btn-primary imgDownload';
        div2.addEventListener("click", () => {
            imgDownload();
        });
        ele.appendChild(div2);
        let div3 = document.createElement('div');
        div3.innerText = '壓縮打包下載圖片';
        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] {
    width: auto;
    height: auto;
    max-width: 100%;
    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;
}
    `;
    addGlobalStyle(css);

})();