release: 1.2.0

This commit is contained in:
mrbunker
2024-09-21 16:46:23 +08:00
parent ea9e20931a
commit 93a51282b6
2 changed files with 208 additions and 120 deletions

326
dist/jop.user.js vendored
View File

@@ -1,7 +1,7 @@
// ==UserScript== // ==UserScript==
// @name JAV 添加跳转在线观看 // @name JAV 添加跳转在线观看
// @namespace https://greasyfork.org/zh-CN/scripts/429173 // @namespace https://greasyfork.org/zh-CN/scripts/429173
// @version 1.1.17 // @version 1.2.0
// @author mission522 // @author mission522
// @description 为 JavDB、JavBus、JavLibrary 这三个站点添加跳转在线观看的链接 // @description 为 JavDB、JavBus、JavLibrary 这三个站点添加跳转在线观看的链接
// @license MIT // @license MIT
@@ -36,7 +36,8 @@
// @connect javbus.com // @connect javbus.com
// @connect javdb.com // @connect javdb.com
// @connect javlibrary.com // @connect javlibrary.com
// @connect javdb008.com // @connect javdb368.com
// @connect javdb369.com
// @connect g64w.com // @connect g64w.com
// @grant GM_addStyle // @grant GM_addStyle
// @grant GM_getValue // @grant GM_getValue
@@ -49,10 +50,10 @@
GM_addStyle(o); GM_addStyle(o);
return; return;
} }
const t = document.createElement("style"); const e = document.createElement("style");
(t.textContent = o), document.head.append(t); (e.textContent = o), document.head.append(e);
})( })(
" .jop-list{box-sizing:border-box;display:flex;flex-wrap:wrap;justify-content:flex-start;gap:10px;width:100%;height:100%;z-index:1;transition:right .2s ease-in-out;font-family:Roboto,Helvetica,Arial,sans-serif;color:#000}.jop-button,.jop-button_def{position:relative;display:flex;align-items:center;justify-content:center;box-sizing:border-box;padding:3px 10px;border-radius:4px;font-weight:500;font-size:14px;border:1px solid #dcdfe6;color:#606266;cursor:pointer}.jop-button_def{margin:10px 0;width:100px}.jop-button:visited{color:#606266}.jop-button:hover{text-decoration:none;color:#409eff;border:1px solid #c6e2ff;background-color:#ecf5ff}.jop-button_label{position:absolute;font-size:10px;padding:4px;border-radius:4px;top:-13px;right:-10px;line-height:.75;color:#67c23a;border:1px solid #e1f3d8;background:white}.jop-button_green{color:#fff!important;background-color:#67c23a}.jop-button_green:hover{color:#fff!important;background-color:#95d475}.jop-button_red{color:#fff!important;background-color:#f56c6c}.jop-button_red:hover{color:#fff!important;background-color:#f89898}.jop-loading{display:inline-block;width:14px;height:14px;margin-right:10px;border:2px dashed #dcdfe6;border-top-color:transparent;border-radius:100%;animation:btnLoading infinite 1s linear}@keyframes btnLoading{0%{transform:rotate(0)}to{transform:rotate(360deg)}}.jop-tag{padding:3px 6px;color:#409eff!important;background:#ecf5ff;border:1px solid #d9ecff;border-radius:4px}.jop-setting-list{display:flex;flex-wrap:wrap}.jop-setting-title{margin:10px 0 5px}.jop-setting-item{display:flex;height:20px;justify-content:center;align-items:center;margin-right:15px;-webkit-user-select:none;user-select:none;cursor:pointer}.db-panel .movie-panel-info div.panel-block{padding:5.5px 12px}.db-panel .jop-app{padding:15px 12px}.lib-panel .jop-app{padding:20px 30px;margin-top:10px}input[type=checkbox],input[type=radio]{margin:0 0 0 5px;cursor:pointer} ", ' .jop-list{box-sizing:border-box;display:flex;flex-wrap:wrap;justify-content:flex-start;gap:10px;width:100%;height:100%;z-index:1;transition:right .2s ease-in-out;color:#000}.jop-button,.jop-button_def{position:relative;display:flex;align-items:center;justify-content:center;box-sizing:border-box;padding:3px 10px;border-radius:4px;font-weight:500;font-size:14px;border:1px solid #dcdfe6;color:#606266;cursor:pointer}.jop-button_def{margin:10px 0;width:100px}.jop-button:visited{color:#606266}.jop-button:hover{text-decoration:none;color:#409eff;border:1px solid #c6e2ff;background-color:#ecf5ff}.jop-button_label{position:absolute;font-size:10px;padding:4px;border-radius:4px;top:-13px;right:-10px;line-height:.75;color:#67c23a;border:1px solid #e1f3d8;background:white}.jop-button_green{color:#fff!important;background-color:#67c23a}.jop-button_green:hover{color:#fff!important;background-color:#95d475}.jop-button_red{color:#fff!important;background-color:#f56c6c}.jop-button_red:hover{color:#fff!important;background-color:#f89898}.jop-loading{display:inline-block;width:14px;height:14px;margin-right:10px;border:2px dashed #dcdfe6;border-top-color:transparent;border-radius:100%;animation:btnLoading infinite 1s linear}@keyframes btnLoading{0%{transform:rotate(0)}to{transform:rotate(360deg)}}.jop-tag{padding:3px 6px;color:#409eff!important;background:#ecf5ff;border:1px solid #d9ecff;border-radius:4px}.jop-setting{margin-top:20px}.jop-setting-list{display:flex;flex-wrap:wrap}.jop-setting-title{margin:10px 0 5px;font-weight:700}.jop-setting-item{display:flex;height:20px;align-items:center;margin-right:15px;-webkit-user-select:none;user-select:none;cursor:pointer}.db-panel .movie-panel-info div.panel-block{padding:5.5px 12px}.db-panel .jop-app{padding:15px 12px}.lib-panel .jop-app{padding:20px 30px;margin-top:10px}input[type=checkbox],input[type=radio]{margin:0 0 0 5px;cursor:pointer}.jop-tooltip-container{position:relative;display:inline-block}.jop-tooltip{position:absolute;bottom:100%;left:50%;transform:translate(-50%);background-color:#333;color:#fff;padding:5px 10px;border-radius:4px;font-size:12px;white-space:nowrap;z-index:1000}.jop-setting-label{cursor:pointer}.jop-checkbox{display:inline-flex;align-items:center;cursor:pointer;margin-right:15px;-webkit-user-select:none;user-select:none}.jop-checkbox-input{position:absolute;opacity:0;cursor:pointer}.jop-checkbox-custom{position:relative;display:inline-block;width:16px;height:16px;background-color:#fff;border:1px solid #dcdfe6;border-radius:2px;transition:all .3s}.jop-checkbox-input:checked+.jop-checkbox-custom{background-color:#409eff;border-color:#409eff}.jop-checkbox-input:checked+.jop-checkbox-custom:after{content:"";position:absolute;top:1px;left:4px;width:5px;height:10px;border:solid white;border-width:0 2px 2px 0;transform:rotate(45deg)}.jop-checkbox-label{margin-left:3px;font-size:14px;color:#606266}.jop-checkbox:hover .jop-checkbox-custom{border-color:#409eff} ',
); );
(function (preact) { (function (preact) {
@@ -128,7 +129,7 @@
// 标签区的第一个一般是字幕标签 // 标签区的第一个一般是字幕标签
subQuery: '.space-y-2 a.text-nord13[href="https://missav.com/chinese-subtitle"]', subQuery: '.space-y-2 a.text-nord13[href="https://missav.com/chinese-subtitle"]',
// 有个「切換無碼」按钮,藏在分享按钮旁边…… // 有个「切換無碼」按钮,藏在分享按钮旁边……
// leakQuery: ".order-first div.rounded-md a[href]:last-child", leakQuery: ".order-first div.rounded-md a[href]:last-child",
}, },
}, },
{ {
@@ -372,16 +373,22 @@
}, },
]; ];
const SP_PREFIX = "300"; const SP_PREFIX = "300";
const gmGet = ({ url }) => {
return new Promise((resolve, reject) => {
_GM_xmlhttpRequest({
method: "GET",
url,
onload: (response) => resolve(response),
onerror: (error) => reject(error),
});
});
};
const isCaseInsensitiveEqual = (str1, str2) => { const isCaseInsensitiveEqual = (str1, str2) => {
return str1.toLowerCase() === str2.toLowerCase(); return str1.toLowerCase() === str2.toLowerCase();
}; };
const isErrorCode = (resCode) => { const isErrorCode = (resCode) => {
return [404, 403].includes(resCode); return [404, 403].includes(resCode);
}; };
const regEnum = {
subtitle: /(中文|字幕|subtitle)/,
leakage: /(无码|無碼|泄漏|Uncensored)/,
};
const getCode = (libItem) => { const getCode = (libItem) => {
const { codeQueryStr } = libItem.querys; const { codeQueryStr } = libItem.querys;
const codeNode = document.querySelector(codeQueryStr); const codeNode = document.querySelector(codeQueryStr);
@@ -394,15 +401,17 @@
if (codeText.startsWith(SP_PREFIX)) return codeText.substring(3); if (codeText.startsWith(SP_PREFIX)) return codeText.substring(3);
return codeText; return codeText;
}; };
const gmGet = ({ url }) => { const regEnum = {
return new Promise((resolve, reject) => { subtitle: /(中文|字幕|subtitle)/,
_GM_xmlhttpRequest({ leakage: /(无码|無碼|泄漏|Uncensored)/,
method: "GET", };
url, const tagsQuery = ({ leakageText, subtitleText }) => {
onload: (response) => resolve(response), const hasLeakage = regEnum.leakage.test(leakageText);
onerror: (error) => reject(error), const hasSubtitle = regEnum.subtitle.test(subtitleText);
}); const tags = [];
}); if (hasLeakage) tags.push("无码");
if (hasSubtitle) tags.push("字幕");
return tags.join(" ");
}; };
var t, var t,
r, r,
@@ -918,54 +927,103 @@
for (c2 in a2) void 0 === p2[c2] && (p2[c2] = a2[c2]); for (c2 in a2) void 0 === p2[c2] && (p2[c2] = a2[c2]);
return preact.options.vnode && preact.options.vnode(l2), l2; return preact.options.vnode && preact.options.vnode(l2), l2;
} }
const Setting = ({ siteList: siteList2, setDisables, disables }) => { const Tooltip = ({ content, children }) => {
const [showSetting, setShowSetting] = p(false); const [isVisible, setIsVisible] = p(false);
const changeCheck = (item, isHidden) => { return u("div", {
className: "jop-tooltip-container",
onMouseEnter: () => setIsVisible(true),
onMouseLeave: () => setIsVisible(false),
children: [
children,
isVisible &&
content &&
u("div", {
className: "jop-tooltip",
children: content,
}),
],
});
};
const Checkbox = ({ label, value, tip, onChange }) => {
const handleChange = (event) => {
onChange(event.currentTarget.checked);
};
return u("label", {
className: "jop-checkbox",
children: [
u("input", {
type: "checkbox",
className: "jop-checkbox-input",
checked: value,
onChange: handleChange,
}),
u("span", {
className: "jop-checkbox-custom",
}),
u(Tooltip, {
content: tip || "",
children: u("span", {
className: "jop-checkbox-label",
children: label,
}),
}),
],
});
};
const Setting = ({
siteList: siteList2,
setDisables,
disables,
multipleNavi,
setMultipleNavi,
}) => {
const [showSetting, setShowSetting] = p(true);
const hanleListChange = (item, isHidden) => {
if (isHidden) { if (isHidden) {
setDisables(disables.filter((disItem) => disItem !== item.name)); setDisables(disables.filter((disItem) => disItem !== item.name));
} else { } else {
setDisables([...disables, item.name]); setDisables([...disables, item.name]);
} }
}; };
const handleNaviChange = (checked) => {
setMultipleNavi(checked);
_GM_setValue("multipleNavi", checked);
};
return u(preact.Fragment, { return u(preact.Fragment, {
children: [ children: [
!showSetting !showSetting &&
? u("div", { u("div", {
className: "jop-button_def", className: "jop-button_def",
onClick: () => { onClick: () => setShowSetting(!showSetting),
setShowSetting(!showSetting); children: "设置",
}, }),
children: "设置",
})
: u("h4", {
className: "jop-setting-title",
children: "勾选默认显示的网站",
}),
showSetting && showSetting &&
u(preact.Fragment, { u(preact.Fragment, {
children: [ children: [
u("div", { u("div", {
className: "jop-setting", className: "jop-setting",
children: u("div", { children: [
className: "jop-setting-list", u(Group, {
children: siteList2.map((item) => { title: "勾选默认展示",
const isHidden = disables.includes(item.name); children: siteList2.map((item) => {
return u("div", { const isHidden = disables.includes(item.name);
className: "jop-setting-item", return u(Checkbox, {
onClick: () => { label: item.name,
changeCheck(item, isHidden); value: !isHidden,
}, onChange: (checked) => hanleListChange(item, checked),
children: [ });
item.name, }),
u("input", {
type: "checkbox",
className: "jop-setting-checkbox",
checked: !isHidden,
}),
],
});
}), }),
}), u(Group, {
title: "其他设置",
children: u(Checkbox, {
label: "展示多个搜索结果",
value: multipleNavi,
tip: "一个站点内出现多条匹配结果时,打开后跳转搜索结果页",
onChange: handleNaviChange,
}),
}),
],
}), }),
u("div", { u("div", {
className: "jop-button_def", className: "jop-button_def",
@@ -979,6 +1037,20 @@
], ],
}); });
}; };
const Group = ({ title, children }) => {
return u(preact.Fragment, {
children: [
u("h4", {
className: "jop-setting-title",
children: title,
}),
u("div", {
className: "jop-setting-list",
children,
}),
],
});
};
function videoPageParser(responseText, { subQuery, leakQuery, videoQuery }) { function videoPageParser(responseText, { subQuery, leakQuery, videoQuery }) {
const doc = new DOMParser().parseFromString(responseText, "text/html"); const doc = new DOMParser().parseFromString(responseText, "text/html");
const subNode = subQuery ? doc.querySelector(subQuery) : ""; const subNode = subQuery ? doc.querySelector(subQuery) : "";
@@ -988,8 +1060,10 @@
const videoNode = videoQuery ? doc.querySelector(videoQuery) : true; const videoNode = videoQuery ? doc.querySelector(videoQuery) : true;
return { return {
isSuccess: !!videoNode, isSuccess: !!videoNode,
hasSubtitle: regEnum.subtitle.test(subNodeText), tag: tagsQuery({
hasLeakage: regEnum.leakage.test(linkNodeText), leakageText: linkNodeText,
subtitleText: subNodeText,
}),
}; };
} }
function serachPageParser( function serachPageParser(
@@ -997,33 +1071,36 @@
{ linkQuery, titleQuery, listIndex = 0 }, { linkQuery, titleQuery, listIndex = 0 },
siteHostName, siteHostName,
CODE, CODE,
searchPageLink,
) { ) {
const doc = new DOMParser().parseFromString(responseText, "text/html"); const doc = new DOMParser().parseFromString(responseText, "text/html");
const linkNode = linkQuery ? doc.querySelectorAll(linkQuery)[listIndex] : null; const titleNodes = titleQuery ? doc.querySelectorAll(titleQuery) : [];
const titleNode = titleQuery ? doc.querySelectorAll(titleQuery)[listIndex] : null; const linkNodes = linkQuery ? doc.querySelectorAll(linkQuery) : [];
const titleNode = titleNodes[listIndex];
const linkNode = linkNodes[listIndex];
const titleNodeText = titleNode ? (titleNode == null ? void 0 : titleNode.outerHTML) : ""; const titleNodeText = titleNode ? (titleNode == null ? void 0 : titleNode.outerHTML) : "";
const codeRegex = /[a-zA-Z]{3,5}-\d{3,5}/; const codeRegex = /[a-zA-Z]{3,5}-\d{3,5}/;
const matchCode = titleNodeText.match(codeRegex); const matchCode = titleNodeText.match(codeRegex);
const isSuccess = const isSuccess =
linkNode && titleNode && matchCode && isCaseInsensitiveEqual(matchCode[0], CODE); linkNode && titleNode && matchCode && isCaseInsensitiveEqual(matchCode[0], CODE);
if (isSuccess) { if (!isSuccess) {
const targetLinkText = linkNode.href.replace(linkNode.hostname, siteHostName);
return { return {
isSuccess: true,
targetLink: targetLinkText,
hasLeakage: regEnum.leakage.test(titleNodeText),
hasSubtitle: regEnum.subtitle.test(titleNodeText),
};
} else {
return {
targetLink: "",
isSuccess: false, isSuccess: false,
hasSubtitle: false,
hasLeakage: false,
}; };
} }
const targetLinkText = linkNode.href.replace(linkNode.hostname, siteHostName);
return {
isSuccess: true,
targetLink: targetLinkText,
multipResLink: searchPageLink,
multipleRes: titleNodes.length > 1,
tag: tagsQuery({
leakageText: titleNodeText,
subtitleText: titleNodeText,
}),
};
} }
const handleFetch = async (siteItem, targetLink, CODE) => { const baseFetcher = async ({ siteItem, targetLink, CODE }) => {
try { try {
const response = await gmGet({ const response = await gmGet({
url: targetLink, url: targetLink,
@@ -1038,45 +1115,61 @@
}; };
} else { } else {
return { return {
...serachPageParser(response.responseText, siteItem.domQuery, siteItem.hostname, CODE), ...serachPageParser(
response.responseText,
siteItem.domQuery,
siteItem.hostname,
CODE,
targetLink,
),
}; };
} }
} catch (error) { } catch (error) {
return { return {
isSuccess: false, isSuccess: false,
targetLink, targetLink,
hasSubtitle: false,
hasLeakage: false,
}; };
} }
}; };
const handleFetchJavBle = async (siteItem, targetLink, CODE) => { const javbleFetcher = async (args) => {
const res = await handleFetch(siteItem, targetLink, CODE); const res = await baseFetcher(args);
const newLink = targetLink.slice(0, -1) + "-c/"; if (res.isSuccess) return res;
return res.isSuccess ? res : await handleFetch(siteItem, newLink, CODE); const newLink = args.targetLink.slice(0, -1) + "-c/";
return await baseFetcher({
...args,
targetLink: newLink,
});
}; };
const SiteBtn = x(({ siteItem, CODE }) => { const fetcher = (args) => {
if (args.siteItem.name === "Jable") {
return javbleFetcher(args);
}
return baseFetcher(args);
};
const SiteBtn = x(({ siteItem, CODE, multipleNavi }) => {
const { name, codeFormater } = siteItem; const { name, codeFormater } = siteItem;
const formatCode = codeFormater ? codeFormater(CODE) : CODE; const formatCode = codeFormater ? codeFormater(CODE) : CODE;
const link = siteItem.url.replace("{{code}}", formatCode); const link = siteItem.url.replace("{{code}}", formatCode);
const [status, setStatus] = p({ const [status, setStatus] = p({
isSuccess: "pedding", isSuccess: "pedding",
hasSubtitle: false, tag: "",
hasLeakage: false, resultLink: "",
targetLink: "",
}); });
const { isSuccess, hasSubtitle, hasLeakage, targetLink } = status; const { isSuccess, tag, resultLink } = status;
_(() => { _(() => {
const fetchMethod = name === "Jable" ? handleFetchJavBle : handleFetch; fetcher({
fetchMethod(siteItem, link, formatCode).then((res) => { siteItem,
targetLink: link,
CODE: formatCode,
}).then((res) => {
const resultLink2 = multipleNavi && res.multipleRes ? res.multipResLink : res.targetLink;
setStatus({ setStatus({
isSuccess: res.isSuccess ? "fulfilled" : "rejected", isSuccess: res.isSuccess ? "fulfilled" : "rejected",
hasSubtitle: res.hasSubtitle, tag: multipleNavi && res.multipleRes ? "多结果" : res.tag,
hasLeakage: res.hasLeakage, resultLink: resultLink2,
targetLink: res.targetLink,
}); });
}); });
}, [handleFetch, siteItem, CODE, link]); }, [fetcher, siteItem, CODE, link, multipleNavi]);
const colorClass = const colorClass =
isSuccess === "pedding" isSuccess === "pedding"
? " " ? " "
@@ -1086,21 +1179,12 @@
return u("a", { return u("a", {
className: "jop-button " + colorClass, className: "jop-button " + colorClass,
target: "_blank", target: "_blank",
href: targetLink === "" ? link : targetLink, href: resultLink === "" ? link : resultLink,
children: [ children: [
(hasSubtitle || hasLeakage) && tag &&
u("div", { u("div", {
className: "jop-button_label", className: "jop-button_label",
children: [ children: tag,
hasSubtitle &&
u("span", {
children: "字幕 ",
}),
hasLeakage &&
u("span", {
children: " 无码",
}),
],
}), }),
u("span", { u("span", {
children: name, children: name,
@@ -1114,36 +1198,40 @@
...["JavBus", "JavDB", "JAVLib", "MISSAV_"], ...["JavBus", "JavDB", "JAVLib", "MISSAV_"],
]; ];
const [disables, setDisables] = p(_GM_getValue("disable", DEF_DIS)); const [disables, setDisables] = p(_GM_getValue("disable", DEF_DIS));
const [multipleNavi, setMultipleNavi] = p(_GM_getValue("multipleNavi", true));
return u(preact.Fragment, { return u(preact.Fragment, {
children: [ children: [
u("div", { u("div", {
class: "jop-list", class: "jop-list",
children: siteList.map((siteItem) => { children: siteList
const hidden = disables.find((disItem) => disItem === siteItem.name) === void 0; .filter(
const sameSite = libItem.name !== siteItem.disableLibItemName; (siteItem) =>
if (hidden && sameSite) { !disables.includes(siteItem.name) && libItem.name !== siteItem.disableLibItemName,
return u( )
.map((siteItem) =>
u(
SiteBtn, SiteBtn,
{ {
siteItem, siteItem,
CODE, CODE,
multipleNavi,
}, },
siteItem.name, siteItem.name,
); ),
} else { ),
return u(preact.Fragment, {});
}
}),
}), }),
u("div", { u(Setting, {
children: u(Setting, { siteList,
siteList, setDisables: (disable) => {
setDisables: (disable) => { setDisables(disable);
setDisables(disable); _GM_setValue("disable", disable);
_GM_setValue("disable", disable); },
}, multipleNavi,
disables, setMultipleNavi: (multipleNavi2) => {
}), setMultipleNavi(multipleNavi2);
_GM_setValue("multipleNavi", multipleNavi2);
},
disables,
}), }),
], ],
}); });

View File

@@ -12,7 +12,7 @@ const includeList = libSites.map((libItem) => libItem.href);
const UserscriptConfig: MonkeyUserScript = { const UserscriptConfig: MonkeyUserScript = {
author: "mission522", author: "mission522",
version: "1.1.17", version: "1.2.0",
license: "MIT", license: "MIT",
name: "JAV 添加跳转在线观看", name: "JAV 添加跳转在线观看",
// match: ["*://*/cn/?v=jav*"], // match: ["*://*/cn/?v=jav*"],