// ==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 = `