// ==UserScript== // @name MP自定义站点索引配置助手 // @author wangzijian0@vip.qq.com // @description 自动获取 RSS订阅 的分类并生成 NexusPHP JSON和Base64配置,搭配MoviePilot的 自定义索引站点 插件使用。 // @version 1.0.2 // @icon  // @match https://*/* // @match http://*/* // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_setClipboard // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @run-at document-idle // @license MIT // @namespace https://greasyfork.org/users/1453515 // @downloadURL https://update.greasyfork.icu/scripts/554748/MP%E8%87%AA%E5%AE%9A%E4%B9%89%E7%AB%99%E7%82%B9%E7%B4%A2%E5%BC%95%E9%85%8D%E7%BD%AE%E5%8A%A9%E6%89%8B.user.js // @updateURL https://update.greasyfork.icu/scripts/554748/MP%E8%87%AA%E5%AE%9A%E4%B9%89%E7%AB%99%E7%82%B9%E7%B4%A2%E5%BC%95%E9%85%8D%E7%BD%AE%E5%8A%A9%E6%89%8B.meta.js // ==/UserScript== (function () { 'use strict'; const SECTION_NAME_MAP = new Map([ ['分类', 'category'], ['类别', 'category'], ['類別', 'category'], ['分類', 'category'], ['类型', 'category'], ['類型', 'category'], ['檢索分類', 'category'], ['检索分类', 'category'], ['來源', 'source'], ['来源', 'source'] ]); const STORAGE_KEYS = { togglerPosition: 'mp_custom_toggler_position', panelPosition: 'mp_custom_panel_position', customDesc: 'mp_custom_category_desc', togglerVisible: 'mp_custom_toggler_visible' }; let customDescRaw = ''; let customDescMap = new Map(); let togglerPosition = null; let togglerVisible = true; let panelPosition = null; let panelDragging = false; const ALLOWED_PREFIX_RE = /^cat\d*$/i; const DEFAULT_SCHEMA = 'NexusPhp'; const DEFAULT_ENCODING = 'UTF-8'; const SECOND_LEVEL_TLDS = new Set(['co', 'com', 'net', 'org', 'gov', 'edu', 'ac']); const TOGGLER_DRAG_THRESHOLD_PX = 4; const CATEGORY_LOCALIZATION = new Map([ ['movies', '电影'], ['movie', '电影'], ['tv series', '电视剧'], ['tv shows', '综艺'], ['tv show', '综艺'], ['animations', '动漫'], ['animation', '动漫'], ['documentaries', '纪录片'], ['documentary', '纪录片'], ['music videos', 'MV'], ['music video', 'MV'], ['music', '音乐'], ['misc', '音乐'], ['other', '其他'], ['3d', '3D'], ['sports', '体育'], ['sport', '体育'], ['photo', '写真'], ['Books', '书籍'], ['pc games', 'PC游戏'], ['pc game', 'PC游戏'], ['hqaudio', '音频'], ['hq audio', '音频'] ]); const state = { panel: null, toggler: null, status: null, output: null, base64Output: null, generateBtn: null, copyJsonBtn: null, copyBase64Btn: null, jsonSplit: null, base64Split: null, customDescInput: null, typingTimeout: null, loadingCustomDesc: false, maximized: false, previousDimensions: null, inputs: {}, toggleJsonBtn: null, toggleBase64Btn: null, customView: null, jsonView: null, base64View: null, activeView: 'custom' }; const menuCommandHandles = []; let menuInitialized = false; function clamp(value, min, max) { if (Number.isNaN(value)) return min; if (max < min) return min; return Math.min(Math.max(value, min), max); } function constrainPanelPosition(left, top, width, height) { const viewportWidth = window.innerWidth || document.documentElement.clientWidth || width; const viewportHeight = window.innerHeight || document.documentElement.clientHeight || height; const maxLeft = Math.max(0, viewportWidth - width); const maxTop = Math.max(0, viewportHeight - height); return { left: clamp(left, 0, maxLeft), top: clamp(top, 0, maxTop) }; } function constrainPanelWithinViewport() { if (!state.panel || state.maximized) return; const rect = state.panel.getBoundingClientRect(); const width = rect.width || state.panel.offsetWidth || state.panel.clientWidth; const height = rect.height || state.panel.offsetHeight || state.panel.clientHeight; if (!width || !height) return; const { left, top } = constrainPanelPosition(rect.left, rect.top, width, height); state.panel.style.left = `${left}px`; state.panel.style.top = `${top}px`; state.panel.style.right = 'auto'; state.panel.style.bottom = 'auto'; } function cssEscape(ident) { if (typeof CSS !== 'undefined' && typeof CSS.escape === 'function') { return CSS.escape(ident); } return ident.replace(/[^\w-]/g, '\\$&'); } function slugify(str) { if (!str) return 'section'; return str .normalize('NFKC') .replace(/[::]/g, '') .replace(/[\s\u00A0]+/g, '_') .replace(/[^\p{L}\p{N}_]+/gu, '_') .replace(/^_+|_+$/g, '') .toLowerCase() || 'section'; } function sanitizeTrackerName(raw) { if (!raw) return ''; let name = raw; name = name.replace(/\s*(?:(?::|:){1,2}|-+)?\s*RSS\s*订阅.*$/i, ''); name = name.split(/\s*::\s*/)[0]; name = name.split(/\s*::\s*/)[0]; return name.trim(); } function deriveTrackerName(hostname) { const separators = [' - ', ' | ', ' — ', ' – ']; let candidate = sanitizeTrackerName(document.title || ''); for (const separator of separators) { if (candidate.includes(separator)) { candidate = candidate.split(separator)[0]; break; } } if (!candidate) { candidate = hostname; } return candidate.trim() || hostname; } function sanitizeTrackerId(raw) { return (raw || '').replace(/[^a-z0-9]+/gi, '').toLowerCase(); } function parseToUrl(domain) { if (!domain) { throw new Error('Invalid domain'); } try { return new URL(domain); } catch (error) { const trimmed = domain.trim(); return new URL(trimmed.startsWith('http') ? trimmed : `https://${trimmed}`); } } function normalizeDomainOrigin(domain) { const url = parseToUrl(domain); return `${url.protocol}//${url.host}`; } function normalizeHostname(domain) { const url = parseToUrl(domain); return url.hostname.replace(/^www\./i, '') || url.hostname; } function extractBaseDomain(domain) { const hostname = normalizeHostname(domain); if (!hostname) return ''; const segments = hostname.split('.').filter(Boolean); if (segments.length <= 2) { return segments.join('.'); } const secondLast = segments[segments.length - 2].toLowerCase(); if (SECOND_LEVEL_TLDS.has(secondLast) && segments.length >= 3) { return segments.slice(-3).join('.'); } return segments.slice(-2).join('.'); } function deriveTrackerId(domain) { if (!domain) return ''; const hostname = normalizeHostname(domain); const segments = hostname.split('.').filter(Boolean); let base = ''; if (segments.length >= 2) { base = segments[segments.length - 2]; } else { base = segments[0] || hostname; } return sanitizeTrackerId(base || hostname); } function encodeToBase64(str) { try { return btoa(unescape(encodeURIComponent(str))); } catch (error) { console.error('[NexusPHP RSS Config Helper] Base64 encode failed:', error); throw new Error('Base64 编码失败'); } } function prettifyCustomKey(key) { if (!key) return ''; let result = key.trim().replace(/_/g, ' '); result = result.replace(/(^|[\s/\-])([a-z])/g, (match, sep, char) => `${sep}${char.toUpperCase()}`); if (/\d/.test(result)) { result = result.replace(/[a-z]+/gi, (segment) => segment.toUpperCase()); } return result.replace(/\s+/g, ' ').trim(); } function getDefaultCustomDescText() { const pairs = new Map(); CATEGORY_LOCALIZATION.forEach((value, key) => { if (!value) return; const formattedKey = prettifyCustomKey(key); if (!formattedKey || pairs.has(formattedKey)) return; pairs.set(formattedKey, value); }); return Array.from(pairs.entries()) .map(([k, v]) => `${k}=${v}`) .join('\n'); } function createPanel() { if (state.panel) return; GM_addStyle(` :root { --tm-color-white: #ffffff; --tm-color-text: #1f2c46; --tm-color-subtle: #506690; --tm-color-primary: #4285f4; --tm-color-primary-dark: #1f63ff; --tm-color-secondary: #1f3f72; --tm-color-border: #dce4f7; --tm-color-surface: #f7f9ff; --tm-shadow-large: 0 18px 40px rgba(30, 60, 110, 0.16); --tm-shadow-medium: 0 12px 30px rgba(30, 60, 110, 0.12); } .tm-rss-panel { position: fixed; top: 24px; left: 24px; width: clamp(460px, 52vw, 720px); min-width: 360px; max-width: calc(100vw - 48px); max-height: calc(100vh - 48px); resize: horizontal; background: var(--tm-color-white); color: var(--tm-color-text); border-radius: 14px; font-family: 'Segoe UI', system-ui, -apple-system, sans-serif; display: none; flex-direction: column; overflow: hidden; z-index: 99999; border: 1px solid var(--tm-color-border); box-shadow: var(--tm-shadow-large); } .tm-rss-panel.is-visible { display: flex; } .tm-rss-panel.is-maximized { left: 12px !important; top: 12px !important; right: 12px !important; bottom: 12px !important; width: auto !important; height: auto !important; max-width: calc(100vw - 24px) !important; max-height: calc(100vh - 24px) !important; border-radius: 10px; box-shadow: none; resize: none; } .tm-rss-header { display: flex; align-items: center; justify-content: space-between; padding: 14px 20px; border-bottom: 1px solid var(--tm-color-border); background: linear-gradient(135deg, rgba(66, 133, 244, 0.08), rgba(66, 133, 244, 0.02)); font-size: 16px; font-weight: 600; letter-spacing: 0.2px; cursor: move; } .tm-rss-close { background: none; border: none; color: var(--tm-color-subtle); font-size: 20px; cursor: pointer; padding: 0 4px; transition: transform 0.25s ease, color 0.25s ease; } .tm-rss-close:hover { transform: rotate(90deg); color: var(--tm-color-primary); } .tm-rss-body { padding: 18px 22px 22px; overflow-y: auto; background: var(--tm-color-white); } .tm-rss-layout { display: grid; grid-template-columns: minmax(220px, 230px) minmax(380px, 1fr); column-gap: 20px; row-gap: 12px; margin-bottom: 18px; align-items: start; } .tm-rss-panel.is-maximized .tm-rss-layout { grid-template-columns: minmax(220px, 230px) minmax(540px, 1fr); column-gap: 24px; row-gap: 16px; } .tm-rss-column { display: flex; flex-direction: column; gap: 18px; } .tm-rss-column--custom { height: auto; margin-left: 0; } .tm-rss-panel.is-maximized .tm-rss-column--custom { margin-left: 0; } .tm-rss-field label { font-size: 13px; letter-spacing: 0.3px; font-weight: 600; color: var(--tm-color-subtle); } .tm-rss-field input[type="text"], .tm-rss-field input[type="url"] { border-radius: 8px; border: 1px solid var(--tm-color-border); background: var(--tm-color-surface); color: var(--tm-color-text); padding: 9px 12px; font-size: 13px; transition: border 0.15s ease, box-shadow 0.15s ease; width: 100%; box-sizing: border-box; } .tm-rss-field textarea { border-radius: 8px; border: 1px solid var(--tm-color-border); background: var(--tm-color-surface); color: var(--tm-color-text); width: 100%; padding: 10px 12px; box-sizing: border-box; overflow: auto; white-space: pre-wrap; word-break: break-word; font-family: inherit; font-size: 12px; resize: vertical; line-height: 1.45; transition: border 0.15s ease, box-shadow 0.15s ease; min-height: 286px; } .tm-rss-field input[type="text"]:focus, .tm-rss-field input[type="url"]:focus, .tm-rss-field textarea:focus { border-color: var(--tm-color-primary); outline: none; box-shadow: 0 0 0 4px rgba(66, 133, 244, 0.16); } .tm-rss-field input[type="checkbox"] { vertical-align: middle; } .tm-rss-hint { display: block; font-size: 11px; opacity: 0.7; margin-top: 4px; color: var(--tm-color-subtle); } .tm-rss-actions { display: grid; grid-template-columns: repeat(3, minmax(0, 1fr)); gap: 10px; margin: 6px 0 12px; align-items: stretch; } .tm-rss-actions > * { display: flex; min-width: 0; } .tm-rss-actions button { flex: 1 1 auto; border: none; border-radius: 999px; padding: 10px 16px; font-size: 13px; font-weight: 600; cursor: pointer; transition: transform 0.15s ease, box-shadow 0.15s ease; display: flex; align-items: center; justify-content: center; } .tm-rss-actions .tm-generate { background: linear-gradient(135deg, var(--tm-color-primary), var(--tm-color-primary-dark)); color: #ffffff; } .tm-rss-actions button:hover { transform: translateY(-1px); box-shadow: var(--tm-shadow-medium); border-color: transparent; } .tm-rss-split { display: inline-flex; align-items: stretch; border: 1px solid var(--tm-color-border); border-radius: 999px; overflow: hidden; background: var(--tm-color-white); flex: 1 1 auto; min-width: 0; } .tm-rss-split button { border: none; background: transparent; padding: 9px 16px; font-size: 12px; font-weight: 600; cursor: pointer; color: var(--tm-color-primary); transition: background 0.15s ease, color 0.15s ease; display: flex; align-items: center; justify-content: center; min-width: 0; } .tm-rss-split button + button { border-left: 1px solid var(--tm-color-border); } .tm-rss-split button:first-child { border-radius: 999px 0 0 999px; } .tm-rss-split button:last-child { border-radius: 0 999px 999px 0; } .tm-rss-split .tm-copy { background: rgba(66, 133, 244, 0.08); color: var(--tm-color-primary); } .tm-rss-split .tm-toggle { background: transparent; color: var(--tm-color-secondary); } .tm-rss-split button:hover { background: rgba(66, 133, 244, 0.12); color: var(--tm-color-primary); } .tm-rss-split.is-active { border-color: rgba(66, 133, 244, 0.45); background: rgba(66, 133, 244, 0.08); } .tm-rss-split.is-active .tm-toggle { color: var(--tm-color-primary); } .tm-rss-status { min-height: 20px; font-size: 12px; color: var(--tm-color-secondary); margin-bottom: 10px; letter-spacing: 0.3px; } .tm-rss-field--views { position: relative; } .tm-rss-views { position: relative; min-height: 260px; } .tm-rss-view { display: none; flex-direction: column; gap: 8px; } .tm-rss-view.is-active { display: flex; } .tm-rss-view label { font-size: 13px; letter-spacing: 0.3px; font-weight: 600; color: var(--tm-color-subtle); } .tm-rss-view .tm-rss-hint { display: block; font-size: 11px; opacity: 0.7; margin-top: -2px; color: var(--tm-color-subtle); } .tm-rss-output, .tm-rss-base64 { width: 100%; padding: 10px 12px; box-sizing: border-box; min-height: 286px; border-radius: 8px; border: 1px solid var(--tm-color-border); background: var(--tm-color-surface); color: var(--tm-color-text); font-size: 12px; line-height: 1.45; resize: vertical; transition: border 0.15s ease, box-shadow 0.15s ease; overflow: auto; white-space: pre-wrap; word-break: break-word; font-family: inherit; } .tm-rss-toggler { position: fixed; left: 28px; bottom: 32px; width: 48px; height: 48px; background: #ffffff; color: #4a5568; border: 1px solid rgba(79, 84, 101, 0.25); border-radius: 18px; display: flex; align-items: center; justify-content: center; z-index: 99998; cursor: grab; transition: background 0.25s ease, color 0.25s ease, border-color 0.25s ease; } .tm-rss-toggler:hover { background: #f3f6ff; color: #1f2c46; border-color: rgba(66, 133, 244, 0.35); } .tm-rss-toggler.is-dragging { cursor: grabbing; background: #e2e8f6; border-color: rgba(66, 133, 244, 0.4); } .tm-rss-toggler-icon { display: inline-flex; width: 20px; height: 20px; align-items: center; justify-content: center; } .tm-rss-toggler svg { width: 100%; height: 100%; fill: currentColor; } @media (max-width: 640px) { .tm-rss-panel { right: 12px; left: 12px; width: auto; } } `); state.panel = document.createElement('div'); state.panel.className = 'tm-rss-panel'; state.panel.innerHTML = `
MP自定义站点索引配置生成
自动读取全站分类,“id:cat=desc”,可编辑翻译
生成后可在此查看并复制 JSON 配置内容。
生成后可在此查看并复制 Base64 编码配置。
`; document.body.appendChild(state.panel); state.status = state.panel.querySelector('.tm-rss-status'); state.output = state.panel.querySelector('.tm-rss-output'); state.base64Output = state.panel.querySelector('.tm-rss-base64'); state.generateBtn = state.panel.querySelector('.tm-generate'); state.copyJsonBtn = state.panel.querySelector('.tm-copy-json'); state.copyBase64Btn = state.panel.querySelector('.tm-copy-base64'); state.toggleJsonBtn = state.panel.querySelector('.tm-toggle-json'); state.toggleBase64Btn = state.panel.querySelector('.tm-toggle-base64'); state.jsonSplit = state.panel.querySelector('.tm-rss-split--json'); state.base64Split = state.panel.querySelector('.tm-rss-split--base64'); state.customView = state.panel.querySelector('.tm-rss-view[data-view="custom"]'); state.jsonView = state.panel.querySelector('.tm-rss-view[data-view="json"]'); state.base64View = state.panel.querySelector('.tm-rss-view[data-view="base64"]'); state.customDescInput = state.panel.querySelector('[data-field="customDesc"]'); state.inputs.schema = state.panel.querySelector('[data-field="schema"]'); state.inputs.trackerId = state.panel.querySelector('[data-field="trackerId"]'); state.inputs.trackerName = state.panel.querySelector('[data-field="trackerName"]'); state.inputs.domain = state.panel.querySelector('[data-field="domain"]'); state.inputs.encoding = state.panel.querySelector('[data-field="encoding"]'); const closeBtn = state.panel.querySelector('.tm-rss-close'); closeBtn.addEventListener('click', hidePanel); const header = state.panel.querySelector('.tm-rss-header'); enablePanelDrag(header); header.addEventListener('dblclick', handleHeaderDoubleClick); state.generateBtn.addEventListener('click', handleGenerate); state.copyJsonBtn.addEventListener('click', handleCopyJson); state.copyBase64Btn.addEventListener('click', handleCopyBase64); if (state.toggleJsonBtn) { state.toggleJsonBtn.addEventListener('click', () => { handleToggleView('json'); }); } if (state.toggleBase64Btn) { state.toggleBase64Btn.addEventListener('click', () => { handleToggleView('base64'); }); } if (state.customDescInput) { state.customDescInput.value = customDescRaw || ''; state.customDescInput.addEventListener('change', handleCustomDescChange); state.customDescInput.addEventListener('input', handleCustomDescInput); } const host = location.hostname; state.inputs.trackerId.value = deriveTrackerId(host); state.inputs.trackerName.value = deriveTrackerName(host); state.inputs.domain.value = location.origin.endsWith('/') ? location.origin : `${location.origin}`; applyPanelPosition(); setActiveView('custom'); } function ensurePanelVisible() { createPanel(); refreshCustomDescValue(true).catch((error) => { console.warn('[NexusPHP RSS Config Helper] Failed to refresh custom descriptions', error); }); state.panel.classList.add('is-visible'); requestAnimationFrame(constrainPanelWithinViewport); } function hidePanel() { if (!state.panel) return; state.panel.classList.remove('is-visible'); } function isPanelVisible() { return Boolean(state.panel && state.panel.classList.contains('is-visible')); } function updateToggleButtonStates() { if (state.toggleJsonBtn) { const active = state.activeView === 'json'; state.toggleJsonBtn.textContent = active ? '返回 分类描述' : '查看 JSON'; if (state.jsonSplit) { state.jsonSplit.classList.toggle('is-active', active); } } if (state.toggleBase64Btn) { const active = state.activeView === 'base64'; state.toggleBase64Btn.textContent = active ? '返回 分类描述' : '查看 Base64'; if (state.base64Split) { state.base64Split.classList.toggle('is-active', active); } } } function setActiveView(view) { if (!state.customView || !state.jsonView || !state.base64View) return; const allowed = new Set(['custom', 'json', 'base64']); const target = allowed.has(view) ? view : 'custom'; state.customView.classList.toggle('is-active', target === 'custom'); state.jsonView.classList.toggle('is-active', target === 'json'); state.base64View.classList.toggle('is-active', target === 'base64'); state.activeView = target; updateToggleButtonStates(); } function handleToggleView(view) { if (state.activeView === view) { setActiveView('custom'); } else { setActiveView(view); } } function applyTogglerVisibility(visible) { if (!state.toggler) return; if (visible) { state.toggler.style.display = ''; state.toggler.removeAttribute('aria-hidden'); } else { state.toggler.style.display = 'none'; state.toggler.setAttribute('aria-hidden', 'true'); } } function setTogglerVisibility(visible, options = {}) { const { persist = true } = options; const nextVisible = Boolean(visible); if (!state.toggler) { createToggler(); } if (togglerVisible === nextVisible) { applyTogglerVisibility(nextVisible); if (persist) { setPreference(STORAGE_KEYS.togglerVisible, nextVisible); } return; } togglerVisible = nextVisible; applyTogglerVisibility(togglerVisible); if (persist) { setPreference(STORAGE_KEYS.togglerVisible, togglerVisible); } refreshMenuCommands(); } function refreshMenuCommands() { if (typeof GM_registerMenuCommand !== 'function') { return; } const canUnregister = typeof GM_unregisterMenuCommand === 'function'; if (canUnregister && menuCommandHandles.length) { menuCommandHandles.splice(0, menuCommandHandles.length).forEach((id) => { try { GM_unregisterMenuCommand(id); } catch (error) { console.warn('[NexusPHP RSS Config Helper] Failed to unregister menu command', error); } }); } else if (!canUnregister && menuInitialized) { return; } const openId = GM_registerMenuCommand('打开 MP自定义站点索引配置器', ensurePanelVisible); if (canUnregister && openId) { menuCommandHandles.push(openId); } if (canUnregister) { const toggleId = GM_registerMenuCommand(togglerVisible ? '隐藏悬浮按钮' : '显示悬浮按钮', () => { setTogglerVisibility(!togglerVisible); }); if (toggleId) { menuCommandHandles.push(toggleId); } } else { GM_registerMenuCommand('显示悬浮按钮', () => { setTogglerVisibility(true); }); GM_registerMenuCommand('隐藏悬浮按钮', () => { setTogglerVisibility(false); }); } menuInitialized = true; } function createToggler() { if (state.toggler) return; state.toggler = document.createElement('button'); state.toggler.type = 'button'; state.toggler.className = 'tm-rss-toggler'; state.toggler.setAttribute('aria-label', '打开配置面板'); state.toggler.innerHTML = ` `; document.body.appendChild(state.toggler); state.toggler.addEventListener('click', (event) => { if (state.toggler.dataset.dragging === 'true' || state.toggler.dataset.dragMoved === 'true') { event.preventDefault(); event.stopPropagation(); return; } if (isPanelVisible()) { hidePanel(); } else { ensurePanelVisible(); } }); applyTogglerPosition(); enableTogglerDrag(); applyTogglerVisibility(togglerVisible); } function registerMenu() { refreshMenuCommands(); } function setStatus(message, isError = false) { if (!state.status) return; state.status.textContent = message || ''; state.status.style.color = isError ? '#ff5c5c' : '#1f3f72'; } function handleHeaderDoubleClick(event) { event.preventDefault(); if (state.maximized) { restorePanelSize(); } else { maximizePanel(); } } function maximizePanel() { if (!state.panel || state.maximized) return; const style = state.panel.style; state.previousDimensions = { left: style.left, top: style.top, right: style.right, bottom: style.bottom, width: style.width, height: style.height, maxWidth: style.maxWidth, maxHeight: style.maxHeight, borderRadius: style.borderRadius, boxShadow: style.boxShadow }; state.maximized = true; state.panel.classList.add('is-maximized'); style.left = ''; style.top = ''; style.right = ''; style.bottom = ''; style.width = ''; style.height = ''; style.maxWidth = ''; style.maxHeight = ''; style.borderRadius = ''; style.boxShadow = ''; } function restorePanelSize() { if (!state.panel || !state.maximized) return; state.maximized = false; state.panel.classList.remove('is-maximized'); const prev = state.previousDimensions || {}; const style = state.panel.style; style.left = prev.left !== undefined ? prev.left : style.left; style.top = prev.top !== undefined ? prev.top : style.top; style.right = prev.right !== undefined ? prev.right : style.right; style.bottom = prev.bottom !== undefined ? prev.bottom : style.bottom; style.width = prev.width !== undefined ? prev.width : style.width; style.height = prev.height !== undefined ? prev.height : style.height; style.maxWidth = prev.maxWidth !== undefined ? prev.maxWidth : style.maxWidth; style.maxHeight = prev.maxHeight !== undefined ? prev.maxHeight : style.maxHeight; style.borderRadius = prev.borderRadius !== undefined ? prev.borderRadius : style.borderRadius; style.boxShadow = prev.boxShadow !== undefined ? prev.boxShadow : style.boxShadow; state.previousDimensions = null; applyPanelPosition(); } function isNexusPhpSite() { if (document.querySelector('form[action*="getrss.php"]')) return true; if (document.querySelector('a[href*="getrss.php"]')) return true; if (document.querySelector('table.torrents')) return true; const generator = document.querySelector('meta[name="generator"]'); if (generator && /NexusPHP/i.test(generator.content)) return true; return false; } function parseRssResponse(html) { const parser = new DOMParser(); return parser.parseFromString(html, 'text/html'); } function resolveSectionInfo(input) { let row = input.closest('tr'); while (row) { const embedded = row.querySelector('td.embedded'); if (embedded) { const rawTitle = embedded.textContent.replace(/\s+/g, '').replace(/[::]/g, ''); const label = embedded.textContent.trim().replace(/[::]/g, ''); const key = SECTION_NAME_MAP.get(rawTitle) || SECTION_NAME_MAP.get(label) || slugify(label || rawTitle); return { key, label: label || rawTitle }; } row = row.previousElementSibling; } return null; } function getLabelText(input, doc) { const labelByFor = input.id ? doc.querySelector(`label[for="${cssEscape(input.id)}"]`) : null; const directLabel = input.closest('label'); const labelNode = labelByFor || directLabel; if (labelNode) { const text = labelNode.textContent.replace(/\s+/g, ' ').trim(); if (text) return text; } const container = input.closest('td') || input.parentElement; if (!container) return ''; const img = container.querySelector('img[alt], img[title]'); if (img) { return (img.getAttribute('title') || img.getAttribute('alt') || '').trim(); } const anchor = container.querySelector('a'); if (anchor) { const text = anchor.textContent.replace(/\s+/g, ' ').trim(); if (text) return text; } const text = container.textContent.replace(/\s+/g, ' ').trim(); return text; } function extractOption(input, doc) { const idSource = input.id || input.name || ''; const idMatch = idSource.match(/\d+/); const optionId = idMatch ? Number(idMatch[0]) : idSource; const labelText = getLabelText(input, doc); const normalizedKey = (labelText || '').trim().toLowerCase(); const idKey = (idSource || '').trim().toLowerCase(); const customDesc = customDescMap.get(normalizedKey) || customDescMap.get(idKey); const localizedDesc = customDesc || CATEGORY_LOCALIZATION.get(normalizedKey) || CATEGORY_LOCALIZATION.get(idKey) || labelText || idSource; return { id: optionId, cat: labelText || idSource, desc: localizedDesc }; } function collectSections(doc) { const inputs = Array.from(doc.querySelectorAll('input[type="checkbox"]')); const sections = {}; inputs.forEach((input) => { const fieldId = input.id || input.name || ''; if (!ALLOWED_PREFIX_RE.test(fieldId)) return; const sectionInfo = resolveSectionInfo(input); if (!sectionInfo) return; if (!sections[sectionInfo.key]) { sections[sectionInfo.key] = { title: sectionInfo.label, items: [] }; } sections[sectionInfo.key].items.push(extractOption(input, doc)); }); const normalized = {}; Object.entries(sections).forEach(([key, payload]) => { const unique = []; const seen = new Set(); payload.items.forEach((item) => { const identity = `${item.id}-${item.cat}`; if (!seen.has(identity)) { seen.add(identity); unique.push(item); } }); unique.sort((a, b) => { if (typeof a.id === 'number' && typeof b.id === 'number') { return a.id - b.id; } return String(a.cat).localeCompare(String(b.cat), 'zh-Hans-CN'); }); normalized[key] = unique; }); return normalized; } function buildBaseConfig(meta, sections) { const config = { schema: meta.schema || DEFAULT_SCHEMA, id: meta.trackerId, name: meta.trackerName, domain: meta.domain, encoding: meta.encoding || DEFAULT_ENCODING, public: meta.isPublic, search: { paths: [ { path: 'torrents.php', method: 'get' } ], params: { search: '{keyword}', search_area: 0 }, batch: { delimiter: ' ', space_replace: '_' } }, category: sections, torrents: { list: { selector: 'table.torrents > tr:has(table.torrentname)' }, fields: { id: { selector: 'a[href*="details.php?id="]', attribute: 'href', filters: [ { name: 'regexp', args: ['id=(\\d+)'] } ] }, title_default: { selector: 'a[href*="details.php?id="]' }, title_optional: { optional: true, selector: 'a[title][href*="details.php?id="]', attribute: 'title' }, title: { text: '{% if fields.title_optional %}{{ fields.title_optional }}{% else %}{{ fields.title_default }}{% endif %}' }, details: { selector: 'a[href*="details.php?id="]', attribute: 'href' }, download: { selector: 'a[href*="download.php?id="]', attribute: 'href' }, date_elapsed: { selector: 'td:nth-child(4) > span', optional: true }, date_added: { selector: 'td:nth-child(4) > span', attribute: 'title', optional: true }, size: { selector: 'td:nth-child(5)' }, seeders: { selector: 'td:nth-child(6)' }, leechers: { selector: 'td:nth-child(7)' }, grabs: { selector: 'td:nth-child(8)' }, downloadvolumefactor: { case: { 'img.pro_free': 0, 'img.pro_free2up': 0, 'img.pro_50pctdown': 0.5, 'img.pro_50pctdown2up': 0.5, 'img.pro_30pctdown': 0.3, '*': 1 } }, uploadvolumefactor: { case: { 'img.pro_50pctdown2up': 2, 'img.pro_free2up': 2, 'img.pro_2up': 2, '*': 1 } } } } }; if (!Object.keys(config.category || {}).length) { delete config.category; } return config; } async function handleGenerate() { if (!state.generateBtn) return; const meta = { schema: state.inputs.schema.value.trim() || DEFAULT_SCHEMA, trackerId: state.inputs.trackerId.value.trim(), trackerName: sanitizeTrackerName(state.inputs.trackerName.value.trim()), domain: state.inputs.domain.value.trim(), encoding: state.inputs.encoding.value.trim() || DEFAULT_ENCODING, isPublic: false }; state.inputs.trackerName.value = meta.trackerName; meta.trackerId = sanitizeTrackerId(meta.trackerId); if (!meta.trackerId) { meta.trackerId = deriveTrackerId(meta.domain || location.hostname); } state.inputs.trackerId.value = meta.trackerId; if (!meta.trackerName) { meta.trackerName = deriveTrackerName(location.hostname); state.inputs.trackerName.value = meta.trackerName; } if (!meta.trackerId) { setStatus('请填写 Tracker ID。', true); state.inputs.trackerId.focus(); return; } if (!meta.trackerName) { setStatus('请填写 Tracker 名称。', true); state.inputs.trackerName.focus(); return; } if (!meta.domain) { setStatus('请填写 Domain。', true); state.inputs.domain.focus(); return; } let requestUrl; try { const normalizedDomain = normalizeDomainOrigin(meta.domain); meta.domain = normalizedDomain; requestUrl = new URL('getrss.php', normalizedDomain).toString(); } catch (error) { setStatus('Domain 格式不正确,请确认包含协议(例如 https://example.com)。', true); return; } state.generateBtn.disabled = true; setStatus('正在获取 getrss.php ...'); try { const response = await fetch(requestUrl, { credentials: 'include', headers: { 'X-Requested-With': 'Tampermonkey' } }); if (!response.ok) { throw new Error(`HTTP ${response.status}`); } const html = await response.text(); const doc = parseRssResponse(html); const sections = collectSections(doc); if (!Object.keys(sections).length) { setStatus('未能从 getrss.php 中解析到分类,请确认已登录且页面结构未变。', true); state.base64Output.value = ''; return; } const config = buildBaseConfig(meta, sections); const json = JSON.stringify(config, null, 2); state.output.value = json; try { const baseDomain = extractBaseDomain(meta.domain) || normalizeHostname(meta.domain); const host = baseDomain.replace(/^www\./i, ''); const encoded = encodeToBase64(json); state.base64Output.value = `${host}|${encoded}`; setActiveView('base64'); } catch (base64Error) { console.error('[NexusPHP RSS Config Helper] Base64 error:', base64Error); state.base64Output.value = ''; setStatus(`解析完成,但 Base64 生成失败:${base64Error.message}`, true); return; } setStatus('解析完成。可以复制 JSON,或继续调整字段。'); } catch (error) { console.error('[NexusPHP RSS Config Helper] Failed:', error); setStatus(`获取或解析失败:${error.message}`, true); state.base64Output.value = ''; } finally { state.generateBtn.disabled = false; } } function handleCopyJson() { if (!state.output || !state.output.value) { setStatus('没有可复制的内容,请先生成。', true); return; } try { const text = state.output.value; if (typeof GM_setClipboard === 'function') { GM_setClipboard(text, { type: 'text', mimetype: 'text/plain' }); } else if (navigator.clipboard && navigator.clipboard.writeText) { navigator.clipboard.writeText(text); } else { throw new Error('浏览器不支持自动复制'); } setStatus('已复制到剪贴板。'); } catch (error) { setStatus(`复制失败:${error.message}`, true); } } function handleCopyBase64() { if (!state.base64Output || !state.base64Output.value) { setStatus('没有 Base64 内容,请先生成。', true); return; } try { const text = state.base64Output.value; if (typeof GM_setClipboard === 'function') { GM_setClipboard(text, { type: 'text', mimetype: 'text/plain' }); } else if (navigator.clipboard && navigator.clipboard.writeText) { navigator.clipboard.writeText(text); } else { throw new Error('浏览器不支持自动复制'); } setStatus('Base64 已复制到剪贴板。'); } catch (error) { setStatus(`Base64 复制失败:${error.message}`, true); } } function handleCustomDescChange(event) { updateCustomDesc(event.target.value); } function handleCustomDescInput(event) { if (state.typingTimeout) { clearTimeout(state.typingTimeout); } state.typingTimeout = setTimeout(() => { updateCustomDesc(event.target.value); }, 500); } function updateCustomDesc(rawText) { if (customDescRaw === rawText) { return; } customDescRaw = rawText; customDescMap = parseCustomDesc(customDescRaw); setPreference(STORAGE_KEYS.customDesc, customDescRaw); } function parseCustomDesc(rawText) { const map = new Map(); if (!rawText) return map; rawText .split(/\r?\n/) .map((line) => line.trim()) .filter(Boolean) .forEach((line) => { const [keyPartRaw, valuePartRaw] = line.split('=').map((segment) => segment.trim()); if (!keyPartRaw || !valuePartRaw) return; let idToken = ''; let labelToken = ''; let keyToken = keyPartRaw; if (keyPartRaw.includes(':')) { const [left, right] = keyPartRaw.split(':'); idToken = (left || '').trim(); labelToken = (right || '').trim(); } else { labelToken = keyPartRaw.trim(); } const register = (token) => { if (!token) return; map.set(token.toLowerCase(), valuePartRaw); }; register(keyToken); register(labelToken); const registerId = (token) => { if (!token) return; const cleaned = token.replace(/^cat/i, ''); if (cleaned) { register(cleaned); register(`cat${cleaned}`); } register(token); }; registerId(idToken); }); return map; } async function refreshCustomDescValue(forceFetch = false) { if (!state.customDescInput) return; if (!forceFetch && customDescRaw.trim()) { state.customDescInput.value = customDescRaw; return; } if (state.loadingCustomDesc) { return; } state.loadingCustomDesc = true; try { const domain = state.inputs?.domain?.value?.trim() || location.origin; let requestUrl; try { requestUrl = new URL('getrss.php', domain).toString(); } catch (error) { console.warn('[NexusPHP RSS Config Helper] Invalid domain for custom desc fetch:', domain, error); fallbackCustomDesc(); return; } const response = await fetch(requestUrl, { credentials: 'include' }); if (!response.ok) { throw new Error(`HTTP ${response.status}`); } const html = await response.text(); const doc = parseRssResponse(html); const { entries, map } = extractCustomDescFromDoc(doc); if (entries.length) { customDescRaw = entries .map(({ idFragment, label, desc }) => `${idFragment}:${label}=${desc}`) .join('\n'); customDescMap = map; setPreference(STORAGE_KEYS.customDesc, customDescRaw); state.customDescInput.value = customDescRaw; } else { fallbackCustomDesc(); } } catch (error) { console.warn('[NexusPHP RSS Config Helper] Failed to fetch custom descriptions', error); fallbackCustomDesc(); } finally { state.loadingCustomDesc = false; } } function fallbackCustomDesc() { if (!state.customDescInput) return; const stored = (customDescRaw || '').trim(); if (stored) { state.customDescInput.value = customDescRaw; return; } const defaults = getDefaultCustomDescText(); customDescRaw = defaults; customDescMap = parseCustomDesc(defaults); state.customDescInput.value = defaults; setPreference(STORAGE_KEYS.customDesc, defaults); } function extractCustomDescFromDoc(doc) { if (!doc) return { entries: [], map: new Map() }; const inputs = Array.from(doc.querySelectorAll('input[type="checkbox"]')); const map = new Map(); const entries = []; inputs.forEach((input) => { const fieldId = input.id || input.name || ''; if (!fieldId) return; if (!ALLOWED_PREFIX_RE.test(fieldId)) return; const sectionInfo = resolveSectionInfo(input); if (!sectionInfo || sectionInfo.key !== 'category') return; const label = getLabelText(input, doc); if (!label) return; const normalizedId = fieldId.trim(); const displayLabel = label.trim(); const numericId = (normalizedId.match(/\d+/) || [''])[0]; const localizationKeyCandidates = [ displayLabel.trim().toLowerCase(), normalizedId.toLowerCase() ]; if (numericId) { localizationKeyCandidates.push(`cat${numericId}`.toLowerCase(), numericId.toLowerCase()); } const localizedDesc = localizationKeyCandidates.reduce((result, key) => { if (result) return result; return CATEGORY_LOCALIZATION.get(key); }, null) || displayLabel; const entry = { id: normalizedId, numericId, label: displayLabel, desc: localizedDesc, idFragment: numericId || normalizedId }; entries.push(entry); const register = (token) => { if (!token) return; map.set(token.toLowerCase(), localizedDesc); }; register(displayLabel); register(normalizedId); if (numericId) { register(numericId); register(`cat${numericId}`); } }); return { entries, map }; } function enableTogglerDrag() { if (!state.toggler) return; let startX = 0; let startY = 0; let initialLeft = 0; let initialTop = 0; let hasDraggedBeyondThreshold = false; const onPointerMove = (event) => { if (state.toggler.dataset.dragging !== 'true') return; const deltaX = event.clientX - startX; const deltaY = event.clientY - startY; if (!hasDraggedBeyondThreshold) { const distance = Math.hypot(deltaX, deltaY); if (distance < TOGGLER_DRAG_THRESHOLD_PX) { return; } hasDraggedBeyondThreshold = true; } const newLeft = initialLeft + deltaX; const newTop = initialTop + deltaY; state.toggler.style.left = `${newLeft}px`; state.toggler.style.top = `${newTop}px`; state.toggler.style.right = 'auto'; state.toggler.style.bottom = 'auto'; state.toggler.dataset.dragMoved = 'true'; }; const onPointerUp = (event) => { if (state.toggler.dataset.dragging !== 'true') return; event.preventDefault(); state.toggler.classList.remove('is-dragging'); state.toggler.dataset.dragging = 'false'; document.removeEventListener('pointermove', onPointerMove); document.removeEventListener('pointerup', onPointerUp); document.removeEventListener('pointercancel', onPointerUp); if (state.toggler.dataset.dragMoved === 'true') { const rect = state.toggler.getBoundingClientRect(); const position = { left: rect.left, top: rect.top }; togglerPosition = position; setPreference(STORAGE_KEYS.togglerPosition, position); setTimeout(() => { state.toggler.dataset.dragMoved = 'false'; }, 50); } else { state.toggler.dataset.dragMoved = 'false'; } hasDraggedBeyondThreshold = false; }; state.toggler.addEventListener('pointerdown', (event) => { if (event.button !== 0) return; event.preventDefault(); state.toggler.dataset.dragging = 'true'; state.toggler.dataset.dragMoved = 'false'; hasDraggedBeyondThreshold = false; state.toggler.classList.add('is-dragging'); startX = event.clientX; startY = event.clientY; const rect = state.toggler.getBoundingClientRect(); initialLeft = rect.left; initialTop = rect.top; state.toggler.style.left = `${initialLeft}px`; state.toggler.style.top = `${initialTop}px`; state.toggler.style.right = 'auto'; state.toggler.style.bottom = 'auto'; document.addEventListener('pointermove', onPointerMove); document.addEventListener('pointerup', onPointerUp); document.addEventListener('pointercancel', onPointerUp); }); } function applyTogglerPosition() { if (!state.toggler) return; if (!togglerPosition || typeof togglerPosition.left !== 'number' || typeof togglerPosition.top !== 'number') { return; } state.toggler.style.left = `${togglerPosition.left}px`; state.toggler.style.top = `${togglerPosition.top}px`; state.toggler.style.right = 'auto'; state.toggler.style.bottom = 'auto'; } function enablePanelDrag(handle) { if (!handle || !state.panel) return; let startX = 0; let startY = 0; let initialLeft = 0; let initialTop = 0; let panelWidth = 0; let panelHeight = 0; const onPointerMove = (event) => { if (!panelDragging) return; const deltaX = event.clientX - startX; const deltaY = event.clientY - startY; const desiredLeft = initialLeft + deltaX; const desiredTop = initialTop + deltaY; const { left, top } = constrainPanelPosition(desiredLeft, desiredTop, panelWidth, panelHeight); state.panel.style.left = `${left}px`; state.panel.style.top = `${top}px`; state.panel.style.right = 'auto'; state.panel.style.bottom = 'auto'; }; const onPointerUp = () => { if (!panelDragging) return; panelDragging = false; panelPosition = getPanelPosition(); setPreference(STORAGE_KEYS.panelPosition, panelPosition); document.removeEventListener('pointermove', onPointerMove); document.removeEventListener('pointerup', onPointerUp); document.removeEventListener('pointercancel', onPointerUp); }; handle.addEventListener('pointerdown', (event) => { if (event.button !== 0) return; if (state.maximized) { return; } event.preventDefault(); panelDragging = true; const rect = state.panel.getBoundingClientRect(); panelWidth = rect.width || state.panel.offsetWidth || state.panel.clientWidth; panelHeight = rect.height || state.panel.offsetHeight || state.panel.clientHeight; const constrainedStart = constrainPanelPosition(rect.left, rect.top, panelWidth, panelHeight); initialLeft = constrainedStart.left; initialTop = constrainedStart.top; state.panel.style.left = `${initialLeft}px`; state.panel.style.top = `${initialTop}px`; state.panel.style.right = 'auto'; state.panel.style.bottom = 'auto'; startX = event.clientX; startY = event.clientY; document.addEventListener('pointermove', onPointerMove); document.addEventListener('pointerup', onPointerUp); document.addEventListener('pointercancel', onPointerUp); }); } function getPanelPosition() { if (!state.panel) return null; const rect = state.panel.getBoundingClientRect(); return { left: rect.left, top: rect.top }; } function applyPanelPosition() { if (!state.panel) return; if (state.maximized) return; if (!panelPosition || typeof panelPosition.left !== 'number' || typeof panelPosition.top !== 'number') { state.panel.style.left = '24px'; state.panel.style.top = '24px'; state.panel.style.right = 'auto'; state.panel.style.bottom = 'auto'; requestAnimationFrame(constrainPanelWithinViewport); return; } state.panel.style.left = `${panelPosition.left}px`; state.panel.style.top = `${panelPosition.top}px`; state.panel.style.right = 'auto'; state.panel.style.bottom = 'auto'; requestAnimationFrame(constrainPanelWithinViewport); } function getPreference(key, fallback) { try { if (typeof GM_getValue === 'function') { const value = GM_getValue(key, fallback); return value === undefined ? fallback : value; } const value = localStorage.getItem(key); return value ? JSON.parse(value) : fallback; } catch (error) { console.warn('[NexusPHP RSS Config Helper] Failed to get preference', key, error); return fallback; } } function setPreference(key, value) { try { if (typeof GM_setValue === 'function') { GM_setValue(key, value); } else { localStorage.setItem(key, JSON.stringify(value)); } } catch (error) { console.warn('[NexusPHP RSS Config Helper] Failed to set preference', key, error); } } function init() { if (!isNexusPhpSite()) { return; } customDescRaw = getPreference(STORAGE_KEYS.customDesc, ''); customDescMap = parseCustomDesc(customDescRaw); togglerPosition = getPreference(STORAGE_KEYS.togglerPosition, null); const storedVisibility = getPreference(STORAGE_KEYS.togglerVisible, null); togglerVisible = storedVisibility === null ? true : Boolean(storedVisibility); panelPosition = getPreference(STORAGE_KEYS.panelPosition, null); createPanel(); createToggler(); registerMenu(); window.addEventListener('resize', constrainPanelWithinViewport); } if (document.readyState === 'complete' || document.readyState === 'interactive') { init(); } else { window.addEventListener('DOMContentLoaded', init, { once: true }); } })();