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