feat(script): 增加用户脚本服务器设置项

This commit is contained in:
Quan 2025-12-15 17:08:52 +08:00
parent 8382a9abc9
commit 6a182eb310
2 changed files with 120 additions and 93 deletions

View File

@ -19,8 +19,6 @@ from fastmcp import FastMCP
from typing import Annotated from typing import Annotated
from pydantic import Field from pydantic import Field
from types import SimpleNamespace from types import SimpleNamespace
# from aiohttp import web
from pyperclip import copy, paste from pyperclip import copy, paste
from uvicorn import Config, Server from uvicorn import Config, Server
@ -681,53 +679,6 @@ class XHS:
else "" else ""
) )
# @staticmethod
# async def index(request):
# return web.HTTPFound(REPOSITORY)
# async def handle(self, request):
# data = await request.post()
# url = data.get("url")
# download = data.get("download", False)
# index = data.get("index")
# skip = data.get("skip", False)
# url = await self.__extract_links(url, None)
# if not url:
# msg = _("提取小红书作品链接失败")
# data = None
# else:
# if data := await self.__deal_extract(url[0], download, index, None, None, not skip, ):
# msg = _("获取小红书作品数据成功")
# else:
# msg = _("获取小红书作品数据失败")
# data = None
# return web.json_response(dict(message=msg, url=url[0], data=data))
# def init_server(self, ):
# app = web.Application(debug=True)
# app.router.add_get('/', self.index)
# app.router.add_post('/xhs/', self.handle)
# return web.AppRunner(app)
# async def run_server(self, log=None, ):
# try:
# await self.start_server(log)
# while True:
# await sleep(3600) # 保持服务器运行
# except (CancelledError, KeyboardInterrupt):
# await self.close_server(log)
# async def start_server(self, log=None, ):
# await self.runner.setup()
# self.site = web.TCPSite(self.runner, "0.0.0.0")
# await self.site.start()
# logging(log, _("Web API 服务器已启动!"))
# logging(log, _("服务器主机及端口: {0}".format(self.site.name, )))
# async def close_server(self, log=None, ):
# await self.runner.cleanup()
# logging(log, _("Web API 服务器已关闭!"))
async def run_api_server( async def run_api_server(
self, self,
host="0.0.0.0", host="0.0.0.0",

View File

@ -2,7 +2,7 @@
// @name XHS-Downloader // @name XHS-Downloader
// @namespace xhs_downloader // @namespace xhs_downloader
// @homepage https://github.com/JoeanAmier/XHS-Downloader // @homepage https://github.com/JoeanAmier/XHS-Downloader
// @version 2.2.0 // @version 2.2.1
// @tag 小红书 // @tag 小红书
// @tag RedNote // @tag RedNote
// @description 提取小红书作品/用户链接,下载小红书无水印图文/视频作品文件 // @description 提取小红书作品/用户链接,下载小红书无水印图文/视频作品文件
@ -32,6 +32,8 @@
const iconBase64 = ""; const iconBase64 = "";
const defaultsWebSocketURL = "ws://127.0.0.1:5558";
let config = { let config = {
disclaimer: GM_getValue("disclaimer", false), disclaimer: GM_getValue("disclaimer", false),
packageDownloadFiles: GM_getValue("packageDownloadFiles", true), packageDownloadFiles: GM_getValue("packageDownloadFiles", true),
@ -40,10 +42,10 @@
keepMenuVisible: GM_getValue("keepMenuVisible", false), keepMenuVisible: GM_getValue("keepMenuVisible", false),
linkCheckboxSwitch: GM_getValue("linkCheckboxSwitch", true), linkCheckboxSwitch: GM_getValue("linkCheckboxSwitch", true),
imageCheckboxSwitch: GM_getValue("imageCheckboxSwitch", true), imageCheckboxSwitch: GM_getValue("imageCheckboxSwitch", true),
scriptServerURL: GM_getValue("scriptServerURL", "ws://127.0.0.1:5558"), // imageDownloadFormat: GM_getValue("imageDownloadFormat", "JPG"),
scriptServerURL: GM_getValue("scriptServerURL", defaultsWebSocketURL),
scriptServerSwitch: GM_getValue("scriptServerSwitch", false), scriptServerSwitch: GM_getValue("scriptServerSwitch", false),
fileNameFormat: undefined, fileNameFormat: undefined,
imageFileFormat: undefined,
icon: { icon: {
type: 'image', // 可选: image/svg/font type: 'image', // 可选: image/svg/font
image: { image: {
@ -676,9 +678,12 @@
border-radius: 16px; border-radius: 16px;
width: 380px; /* 缩小窗口宽度 */ width: 380px; /* 缩小窗口宽度 */
max-width: 95vw; max-width: 95vw;
max-height: 95vh;
box-shadow: 0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23); box-shadow: 0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23);
overflow: hidden; overflow: hidden;
animation: scaleUp 0.3s; animation: scaleUp 0.3s;
display: flex;
flex-direction: column;
} }
/* 通用头部/内容/底部/按钮(三个弹窗共用) */ /* 通用头部/内容/底部/按钮(三个弹窗共用) */
@ -801,6 +806,31 @@
border-color: #2196F3; border-color: #2196F3;
box-shadow: 0 0 4px rgba(33, 150, 243, 0.3); box-shadow: 0 0 4px rgba(33, 150, 243, 0.3);
} }
.select-input {
width: 100px;
padding: 8px 12px;
border: 1px solid #ddd;
border-radius: 4px;
font-size: 0.9rem;
margin-top: 8px;
background: #fff;
transition: border-color 0.2s, box-shadow 0.2s;
appearance: none;
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 12 12'%3E%3Cpath fill='%23666' d='M3 4.5L6 7.5L9 4.5H3Z'/%3E%3C/svg%3E");
background-repeat: no-repeat;
background-position: right 12px center;
padding-right: 32px;
}
.select-input:focus {
outline: none;
border-color: #2196F3;
box-shadow: 0 0 4px rgba(33, 150, 243, 0.3);
}
.select-input:disabled {
background-color: #f5f5f5;
color: #999;
cursor: not-allowed;
}
.setting-description { .setting-description {
font-size: 0.875rem; font-size: 0.875rem;
color: #757575; color: #757575;
@ -924,15 +954,16 @@
} }
// 创建开关项 // 创建开关项
const createSettingItem = ({label, description, checked}) => { const createSwitchItem = ({label, description, checked, disabled = false}) => {
const item = document.createElement('div'); const item = document.createElement('div');
item.className = 'setting-item'; item.className = 'setting-item';
item.style.opacity = disabled ? 0.6 : 1;
item.innerHTML = ` item.innerHTML = `
<label> <label>
<span>${label}</span> <span>${label}</span>
<div class="toggle-switch"> <div class="toggle-switch">
<input type="checkbox" ${checked ? 'checked' : ''}> <input type="checkbox" ${checked ? 'checked' : ''} ${disabled ? 'disabled' : ''}>
<span class="slider"></span> <span class="slider"></span>
</div> </div>
</label> </label>
@ -987,16 +1018,41 @@
}; };
// 创建文本输入项 // 创建文本输入项
const createTextInput = ({label, description, placeholder, value}) => { const createTextInput = ({label, description, placeholder, value, disabled = false}) => {
const item = document.createElement('div'); const item = document.createElement('div');
item.className = 'setting-item'; item.className = 'setting-item';
item.style.opacity = disabled ? 0.6 : 1;
item.innerHTML = ` item.innerHTML = `
<div> <div>
<span style="font-size: 1rem; font-weight: 500; color: #333;">${label}</span> <span style="font-size: 1rem; font-weight: 500; color: #333;">${label}</span>
</div> </div>
<div class="setting-description">${description}</div> <div class="setting-description">${description}</div>
<input type="text" class="text-input" placeholder="${placeholder}" value="${value}"> <input type="text" class="text-input" placeholder="${placeholder}" value="${value}" ${disabled ?
'disabled' : ''}>
`;
return item;
};
// 创建下拉框项
const createSelectItem = ({label, description, options, value, disabled = false}) => {
const item = document.createElement('div');
item.className = 'setting-item';
item.style.opacity = disabled ? 0.6 : 1;
// 生成选项HTML
const optionsHtml = options.map(
option => `<option value="${option}" ${option === value ? 'selected' : ''}>${option}</option>`).join('');
item.innerHTML = `
<label>
<span>${label}</span>
<select class="select-input" ${disabled ? 'disabled' : ''}>
${optionsHtml}
</select>
</label>
<div class="setting-description">${description}</div>
`; `;
return item; return item;
@ -1037,18 +1093,18 @@
body.className = 'modal-body'; body.className = 'modal-body';
// 自动滚动开关 // 自动滚动开关
const autoScroll = createSettingItem({ const autoScroll = createSwitchItem({
label: '自动滚动页面', label: '自动滚动页面',
description: '启用后,页面将根据规则自动滚动以便加载更多内容', description: '启用后,页面将根据规则自动滚动以便加载更多内容',
checked: GM_getValue("autoScrollSwitch", false), checked: GM_getValue("autoScrollSwitch", false),
}); });
// 文件打包开关 // 文件打包开关
const filePack = createSettingItem({ const filePack = createSwitchItem({
label: '文件打包下载', label: '文件打包下载',
description: '启用后,多个文件的作品将会以压缩包格式下载', description: '启用后,多个文件的作品将会以压缩包格式下载',
checked: GM_getValue("packageDownloadFiles", true), checked: GM_getValue("packageDownloadFiles", true),
}); });
// 滚动次数设置 // 滚动次数设置
const scrollCount = createNumberInput({ const scrollCount = createNumberInput({
@ -1056,47 +1112,65 @@
description: '自动滚动页面的次数(仅在启用自动滚动页面时可用)', description: '自动滚动页面的次数(仅在启用自动滚动页面时可用)',
value: GM_getValue("maxScrollCount", 50), value: GM_getValue("maxScrollCount", 50),
min: 10, min: 10,
max: 5000, max: 9999,
disabled: !GM_getValue("autoScrollSwitch", false), disabled: !GM_getValue("autoScrollSwitch", false),
}); });
const linkCheckboxSwitch = createSettingItem({ const linkCheckboxSwitch = createSwitchItem({
label: '链接提取选择模式', label: '链接提取选择模式',
description: '关闭后,提取作品链接时无需确认直接提取全部链接', description: '关闭后,提取作品链接时无需确认直接提取全部链接',
checked: GM_getValue("linkCheckboxSwitch", true), checked: GM_getValue("linkCheckboxSwitch", true),
});
const imageCheckboxSwitch = createSwitchItem({
label: '图片下载选择模式',
description: '关闭后,下载图文作品时无需确认直接下载全部文件',
checked: GM_getValue("imageCheckboxSwitch", true),
}); });
const imageCheckboxSwitch = createSettingItem({ const keepMenuVisible = createSwitchItem({
label: '图片下载选择模式', label: '菜单保持显示',
description: '关闭后,下载图文作品时无需确认直接下载全部文件', description: '启用后,功能菜单无需鼠标悬停始终保持显示',
checked: GM_getValue("imageCheckboxSwitch", true), checked: GM_getValue("keepMenuVisible", false),
}); });
const keepMenuVisible = createSettingItem({ const scriptServerURL = createTextInput({
label: '菜单保持显示', label: 'WebSocket 服务器地址',
description: '启用后,功能菜单无需鼠标悬停始终保持显示', description: 'WebSocket 服务器地址',
checked: GM_getValue("keepMenuVisible", false), placeholder: defaultsWebSocketURL,
}); value: GM_getValue("scriptServerURL", defaultsWebSocketURL),
disabled: !GM_getValue("scriptServerSwitch", false),
});
const scriptServerSwitch = createSettingItem({ const scriptServerSwitch = createSwitchItem({
label: '连接服务器', label: '连接服务器',
description: '启用后,可以把下载任务推送至服务器', description: '启用后,可以把下载任务推送至服务器',
checked: GM_getValue("scriptServerSwitch", false), checked: GM_getValue("scriptServerSwitch", false),
}); });
// 名称格式设置 // const imageDownloadFormat = createSelectItem({
// label: '图片下载格式',
// description: '选择图片格式',
// options: ["AUTO", "PNG", "JPG",],
// value: GM_getValue("imageDownloadFormat", "JPG"),
// });
//
// const nameFormat = createTextInput({ // const nameFormat = createTextInput({
// label: '文件名称格式', // label: '文件名称格式',
// description: '设置文件的名称格式(例如:{date}-{title})。', // description: '设置文件的名称格式(例如:{date}-{title})。',
// placeholder: '{date}-{title}', // placeholder: '{date}-{title}',
// value: GM_getValue("fileNameFormat",) // value: GM_getValue("fileNameFormat",)
// }); // });
// 绑定自动滚动开关控制次数输入 // 绑定自动滚动开关控制次数输入
autoScroll.querySelector('input').addEventListener('change', (e) => { autoScroll.querySelector('input').addEventListener('change', (e) => {
scrollCount.querySelector('input').disabled = !e.target.checked; scrollCount.querySelector('input').disabled = !e.target.checked;
scrollCount.querySelector('.number-input').style.opacity = e.target.checked ? 1 : 0.6; scrollCount.querySelector('.number-input').style.opacity = e.target.checked ? 1 : 0.6;
}); });
scriptServerSwitch.querySelector('input').addEventListener('change', (e) => {
scriptServerURL.querySelector('input').disabled = !e.target.checked;
scriptServerURL.querySelector('.text-input').style.opacity = e.target.checked ? 1 : 0.6;
});
// 组合内容 // 组合内容
body.appendChild(filePack); body.appendChild(filePack);
@ -1104,9 +1178,10 @@
body.appendChild(scrollCount); body.appendChild(scrollCount);
body.appendChild(linkCheckboxSwitch); body.appendChild(linkCheckboxSwitch);
body.appendChild(imageCheckboxSwitch); body.appendChild(imageCheckboxSwitch);
// body.appendChild(imageDownloadFormat);
body.appendChild(keepMenuVisible); body.appendChild(keepMenuVisible);
body.appendChild(scriptServerURL);
body.appendChild(scriptServerSwitch); body.appendChild(scriptServerSwitch);
// body.appendChild(nameFormat);
// 创建底部按钮 // 创建底部按钮
const footer = document.createElement('div'); const footer = document.createElement('div');
@ -1135,6 +1210,7 @@
updateLinkCheckboxSwitch(linkCheckboxSwitch.querySelector('input').checked); updateLinkCheckboxSwitch(linkCheckboxSwitch.querySelector('input').checked);
updateImageCheckboxSwitch(imageCheckboxSwitch.querySelector('input').checked); updateImageCheckboxSwitch(imageCheckboxSwitch.querySelector('input').checked);
updateMaxScrollCount(parseInt(scrollCount.querySelector('input').value) || 50) updateMaxScrollCount(parseInt(scrollCount.querySelector('input').value) || 50)
updateScriptServerURL(scriptServerURL.querySelector('.text-input').value.trim() || defaultsWebSocketURL);
updateScriptServerSwitch(scriptServerSwitch.querySelector('input').checked); updateScriptServerSwitch(scriptServerSwitch.querySelector('input').checked);
// updateFileNameFormat(nameFormat.querySelector('.text-input').value.trim() || null); // updateFileNameFormat(nameFormat.querySelector('.text-input').value.trim() || null);
closeSettingsModal(); closeSettingsModal();