// ==UserScript== // @name 超星学习通刷课专属 <<< 助你解放双手 // @namespace cx-helper-xqqx // @version 1.2.3 // @description 超星学习通智能刷课助手:静音+强制播放+自动下一节+AI答题+视频监控。支持拖拽缩放面板,位置记忆,Alt+P快捷键。集成DeepSeek API智能答题,检测视频异常自动提示。请合理使用! // @author xqqx // @match *://mooc1-2.chaoxing.com/mooc-ans/* // @match *://mooc1.chaoxing.com/mooc-ans/* // @run-at document-end // @grant GM_getValue // @grant GM_setValue // @grant GM_registerMenuCommand // @icon  // @license MIT // ==/UserScript== (function () { 'use strict'; if (window.top !== window) return; if (window.__PH_INIT__) return; window.__PH_INIT__ = true; // 清理遗留元素,避免重复组件造成冲突 ['ph-panel','ph-fab','ph-video-tip'].forEach(id => { const e = document.getElementById(id); if (e) try { e.remove(); } catch {} }); const Store = (() => { const hasGM = typeof GM_getValue === 'function' && typeof GM_setValue === 'function'; const mem = new Map(); function get(k, d) { try { return hasGM ? GM_getValue(k, d) : (mem.has(k) ? mem.get(k) : d); } catch { return d; } } function set(k, v) { try { hasGM ? GM_setValue(k, v) : mem.set(k, v); } catch {} } return { get, set }; })(); const K = { rate: 'ph_rate', mute: 'ph_mute', playOn: 'ph_play_on', nextOn: 'ph_next_on', apiKey: 'ph_api_key', answerOn: 'ph_answer_on', }; const S = { visible: 'ph_visible', pos: 'ph_pos', tab: 'ph_tab', size: 'ph_size', }; // 默认值 if (Store.get(K.rate, null) == null) Store.set(K.rate, '1.5'); if (Store.get(K.mute, null) == null) Store.set(K.mute, true); if (Store.get(K.playOn, null) == null) Store.set(K.playOn, true); if (Store.get(K.nextOn, null) == null) Store.set(K.nextOn, true); if (Store.get(S.visible, null) == null) Store.set(S.visible, true); if (Store.get(S.tab, null) == null) Store.set(S.tab, 'home'); if (Store.get(S.size, null) == null) Store.set(S.size, {width: 420, height: 450}); if (Store.get(K.apiKey, null) == null) Store.set(K.apiKey, ''); if (Store.get(K.answerOn, null) == null) Store.set(K.answerOn, false); const Log = { info: (...a) => console.log('[PH]', ...a), warn: (...a) => console.warn('[PH]', ...a), }; const Dom = { $(sel, root = document) { return root.querySelector(sel); }, $$(sel, root = document) { return Array.from(root.querySelectorAll(sel)); }, mainDoc() { const f = document.getElementById('iframe'); if (!f) return document; try { return f.contentDocument || (f.contentWindow && f.contentWindow.document) || document; } catch { return document; } }, collectDocs(rootDoc) { const seen = new Set(), out = []; (function dfs(doc) { if (!doc || seen.has(doc)) return; seen.add(doc); out.push(doc); const ifrs = doc.getElementsByTagName('iframe'); for (let i = 0; i < ifrs.length; i++) { try { const d = ifrs[i].contentDocument || (ifrs[i].contentWindow && ifrs[i].contentWindow.document); if (d) dfs(d); } catch {} } })(rootDoc); return out; }, curChapterId() { const el = document.getElementById('chapterIdid'); return el ? el.value : null; }, }; // 视频错误检测与提示 const VideoMonitor = (() => { let errorCount = 0; const MAX_ERRORS = 3; function showSwitchTip() { if (document.getElementById('ph-video-tip')) return; const tip = document.createElement('div'); tip.id = 'ph-video-tip'; tip.style.cssText = [ 'position:fixed','top:50%','left:50%','transform:translate(-50%,-50%)', 'z-index:2147483648','background:#ff4444','color:#fff','padding:20px', 'border-radius:10px','box-shadow:0 8px 20px rgba(0,0,0,.3)','text-align:center' ].join(';'); tip.innerHTML = `
⚠️ 视频播放异常
检测到视频格式不支持或网络问题,建议切换线路:
公网1 → 公网2 → 本校
`; document.body.appendChild(tip); Dom.$('#ph-tip-close').addEventListener('click', () => { tip.remove(); errorCount = 0; // 重置计数 }); // 10秒后自动关闭 setTimeout(() => { if (tip.parentNode) tip.remove(); }, 10000); } function monitorVideos() { const docs = Dom.collectDocs(Dom.mainDoc()); for (const d of docs) { try { d.querySelectorAll('video').forEach(v => { if (v.dataset.phMonitored) return; v.dataset.phMonitored = 'true'; v.addEventListener('error', () => { errorCount++; Log.warn(`视频错误 ${errorCount}/${MAX_ERRORS}:`, v.error); if (errorCount >= MAX_ERRORS) showSwitchTip(); }); v.addEventListener('loadstart', () => errorCount = 0); // 成功加载时重置 }); } catch {} } } return { start: () => setInterval(monitorVideos, 2000) }; })(); const Playback = (() => { let t = null; function allVideos() { const docs = Dom.collectDocs(Dom.mainDoc()), out = []; for (const d of docs) { try { out.push(...d.getElementsByTagName('video')); } catch {} } return out; } function tick() { const rate = parseFloat(Store.get(K.rate, '1.5')); const mute = !!Store.get(K.mute, true); const vids = allVideos(); for (const v of vids) { try { if (v.paused) v.play().catch(() => {}); try { v.playbackRate = rate; } catch {} try { v.muted = mute; } catch {} } catch {} } } return { start() { this.stop(); t = setInterval(tick, 1000); Log.info('播放引擎已启动'); }, stop() { if (t) clearInterval(t); t = null; Log.info('播放引擎已停止'); }, poke() { try { tick(); } catch {} }, }; })(); const Progress = (() => { let t = null, lastChapter = null, obs = null; function currentH4() { const id = Dom.curChapterId(); if (id) { const byId = document.getElementById('cur' + id); if (byId) return byId; } return document.querySelector('#content1 .ncells h4.currents'); } function finishedByCatalog() { const h4 = currentH4(); if (!h4) return null; const hiddenCnt = h4.querySelector('input.jobUnfinishCount'); if (hiddenCnt && hiddenCnt.value != null) { const n = parseInt(hiddenCnt.value, 10); if (!isNaN(n)) return n <= 0; } const spanCnt = h4.querySelector('span.jobCount'); if (spanCnt) { const m = parseInt((spanCnt.textContent || '').trim() || '0', 10); if (!isNaN(m)) return m <= 0; } if (h4.querySelector('.roundpointStudent.blue')) return true; return null; } function allVideosEnded() { const docs = Dom.collectDocs(Dom.mainDoc()); let any = false; for (const d of docs) { try { const arr = d.getElementsByTagName('video'); if (arr.length) any = true; for (const v of arr) { try { if (!(v.ended || (v.duration > 0 && v.currentTime >= v.duration - 0.5))) return false; } catch { return false; } } } catch { return false; } } return any ? true : false; } function goNext() { const btn = document.getElementById('right1'); if (btn && !/\bgray\b/.test(btn.className || '')) { btn.click(); Log.info('已点击"下一节"按钮'); return true; } Log.warn('未找到"下一节"入口'); return false; } function hasQuestions() { const docs = Dom.collectDocs(Dom.mainDoc()); const sels = ['.questionLi', '.examPaper_subject', '.TiMu', '.question-box', '.question']; for (const d of docs) { for (const sel of sels) { if (d.querySelector(sel)) return true; } } return false; } function shouldSkipQuestions() { const apiKey = Store.get(K.apiKey, ''); const answerOn = Store.get(K.answerOn, false); // 如果没有API或答题功能关闭,则跳过题目章节 return !apiKey || !answerOn; } function tick() { const ch = Dom.curChapterId(); if (lastChapter && ch && lastChapter !== ch) { Log.info('检测到章节变化:', lastChapter, '->', ch); setTimeout(() => Playback.poke(), 2500); } if (ch) lastChapter = ch; const fin = finishedByCatalog(); const hasQs = hasQuestions(); const shouldSkip = shouldSkipQuestions(); if (fin === true) { setTimeout(() => goNext(), 600); } else if (fin === false) { Playback.poke(); } else if (hasQs && shouldSkip) { // 有题目但需要跳过:等视频播放完再跳过 if (allVideosEnded()) { Log.info('检测到题目章节,无API配置,跳过...'); setTimeout(() => goNext(), 600); } else { Playback.poke(); // 继续播放视频 } } else if (allVideosEnded()) { setTimeout(() => goNext(), 600); } } return { start() { this.stop(); lastChapter = Dom.curChapterId(); t = setInterval(tick, 3000); Log.info('进度引擎已启动'); }, stop() { if (t) clearInterval(t); t = null; Log.info('进度引擎已停止'); } }; })(); const Panel = (() => { let wrap = null, fab = null; let dragging = false, startX = 0, startY = 0, startLeft = 0, startTop = 0; const clamp = (v, min, max) => Math.max(min, Math.min(max, v)); // 安全事件绑定函数,避免绑定到不存在的元素 function safeBind(selector, event, handler, root = document) { try { const el = root.querySelector(selector); if (!el) { console.warn('[PH] safeBind: missing', selector); return null; } el.addEventListener(event, handler); return el; } catch (err) { console.warn('[PH] safeBind error', selector, err); return null; } } function ensureFab() { fab = document.getElementById('ph-fab'); if (fab) return fab; fab = document.createElement('div'); fab.id = 'ph-fab'; fab.textContent = 'CX'; fab.title = '点击显示面板(Alt+P)'; Object.assign(fab.style, { position: 'fixed', right: '16px', bottom: '16px', zIndex: 2147483647, width: '44px', height: '44px', borderRadius: '50%', background: '#0f1320', color: '#fff', display: 'flex', alignItems: 'center', justifyContent: 'center', cursor: 'pointer', boxShadow: '0 6px 16px rgba(0,0,0,.25)', userSelect: 'none', fontWeight: 600, fontSize: '14px', pointerEvents: 'auto' }); fab.addEventListener('click', () => { open(); }); document.body.appendChild(fab); // 根据存储状态决定初始显示(保证互斥) const vis = !!Store.get(S.visible, true); fab.style.display = vis ? 'none' : 'flex'; return fab; } function switchTab(tab) { const tabs = ['home','answer','search','settings']; tabs.forEach(t => { const page = document.getElementById('ph-page-' + t); const btn = document.querySelector(`.ph-tab[data-tab="${t}"]`); if (page) page.style.display = (t === tab ? 'block' : 'none'); if (btn) btn.classList.toggle('active', t === tab); }); Store.set(S.tab, tab); } function extractQuestions() { const docs = Dom.collectDocs(Dom.mainDoc()); const items = []; const sels = ['.questionLi', '.examPaper_subject', '.TiMu', '.question-box', '.question']; for (const d of docs) { for (const sel of sels) { d.querySelectorAll(sel).forEach(node => { const text = (node.innerText || '').replace(/\s+/g, ' ').trim(); if (text && text.length > 6) items.push(text); }); } } const area = document.getElementById('ph-answer-area'); area.value = items.length ? items.join('\n\n---\n\n') : '未识别到题面,请进入章节测试/作业页面后再试。'; } // API验证和答题相关函数 async function validateAPI(apiKey) { if (!apiKey || apiKey.trim() === '') return false; try { const response = await fetch('https://api.deepseek.com/v1/models', { headers: { 'Authorization': `Bearer ${apiKey}` } }); return response.ok; } catch (error) { Log.warn('API验证失败:', error); return false; } } function updateAPIStatus(status, message) { const statusEl = document.getElementById('ph-api-status'); const toggleBtn = document.getElementById('ph-answer-toggle'); const autoBtn = document.getElementById('ph-auto-answer'); if (statusEl) statusEl.textContent = message; if (toggleBtn) { toggleBtn.style.background = status ? '#0f1320' : '#ccc'; toggleBtn.style.color = status ? '#fff' : '#000'; } if (autoBtn) autoBtn.disabled = !status; } function displayQuestions(questions) { const window = document.getElementById('ph-question-window'); if (!window) return; if (questions.length === 0) { window.innerHTML = '
未找到题目
'; return; } window.innerHTML = questions.map((q, i) => `
题目 ${i+1}:
${q}
` ).join(''); } async function autoAnswer(questions) { const apiKey = Store.get(K.apiKey, ''); if (!apiKey) { alert('请先配置API密钥'); return; } Log.info('开始自动答题,题目数量:', questions.length); for (let i = 0; i < questions.length; i++) { const question = questions[i]; try { // 调用DeepSeek API获取答案 const answer = await callDeepSeekAPI(apiKey, question); if (answer) { // 尝试填入答案到页面 fillAnswer(question, answer, i); Log.info(`题目${i+1}答题完成:`, answer); } } catch (error) { Log.warn(`题目${i+1}答题失败:`, error); } // 添加延迟避免API限制 if (i < questions.length - 1) { await new Promise(resolve => setTimeout(resolve, 1000)); } } } async function callDeepSeekAPI(apiKey, question) { try { const response = await fetch('https://api.deepseek.com/v1/chat/completions', { method: 'POST', headers: { 'Authorization': `Bearer ${apiKey}`, 'Content-Type': 'application/json' }, body: JSON.stringify({ model: 'deepseek-chat', messages: [{ role: 'user', content: `请回答以下题目,只返回答案选项(如A、B、C、D)或简短答案,不要解释:\n\n${question}` }], max_tokens: 100, temperature: 0.1 }) }); const data = await response.json(); return data.choices?.[0]?.message?.content?.trim(); } catch (error) { Log.warn('API调用失败:', error); return null; } } function fillAnswer(question, answer, index) { const docs = Dom.collectDocs(Dom.mainDoc()); // 尝试多种选择器来找到答案输入框 const selectors = [ 'input[type="radio"]', 'input[type="checkbox"]', 'input[type="text"]', 'textarea', '.option-item', '.answer-option' ]; for (const doc of docs) { try { // 根据答案类型选择不同的填入策略 if (/^[A-D]$/i.test(answer)) { // 选择题答案 const options = doc.querySelectorAll('input[type="radio"], input[type="checkbox"]'); for (const option of options) { const label = option.parentElement?.textContent || ''; if (label.includes(answer.toUpperCase())) { option.click(); return; } } } else { // 文本答案 const textInputs = doc.querySelectorAll('input[type="text"], textarea'); if (textInputs[index]) { textInputs[index].value = answer; textInputs[index].dispatchEvent(new Event('input', { bubbles: true })); return; } } } catch (error) { Log.warn('填答失败:', error); } } } function mount() { if (document.getElementById('ph-panel')) { wrap = document.getElementById('ph-panel'); ensureFab(); return; } // 防御性清理:再次删除可能残留的旧元素 ['ph-panel','ph-fab'].forEach(id => { const ex = document.getElementById(id); if (ex) try { ex.remove(); } catch {} }); wrap = document.createElement('div'); wrap.id = 'ph-panel'; wrap.style.cssText = [ 'position:fixed','z-index:2147483647', 'background:#fff','color:#000','padding:12px 15px', 'border-radius:12px','font-size:13px','box-shadow:0 8px 20px rgba(0,0,0,.28)', 'width:420px','min-height:450px','user-select:none', 'right:20px','bottom:20px' ].join(';'); // 设置相对定位以便拖拽手柄正确定位 wrap.style.overflow = 'visible'; wrap.innerHTML = `
学习通助手
首页
答题
搜题
设置
超星学习通刷课助手
v1.2.4 - 助你解放双手,高效学习
✨ 核心功能
🛠️ 使用说明
  1. 面板可拖动、缩放,位置自动记忆
  2. Alt+P 或右下角"CX"可显示/隐藏面板
  3. 设置页面可调节倍速、静音等参数
  4. 答题页面需配置API密钥才能使用AI功能
⚠️ 使用提醒
本脚本仅供学习交流使用,请遵守学校相关规定,合理使用辅助工具。
`; document.body.appendChild(wrap); ensureFab(); // 初始化API相关控件 const apiKey = Store.get(K.apiKey, ''); const answerOn = !!Store.get(K.answerOn, false); if (Dom.$('#ph-api-key')) Dom.$('#ph-api-key').value = apiKey; if (Dom.$('#ph-answer-toggle')) { Dom.$('#ph-answer-toggle').textContent = answerOn ? '停止答题' : '开始答题'; Dom.$('#ph-answer-toggle').style.background = answerOn ? '#ff4444' : (apiKey ? '#0f1320' : '#ccc'); Dom.$('#ph-answer-toggle').style.color = answerOn ? '#fff' : (apiKey ? '#fff' : '#000'); } // 更新API状态显示 updateAPIStatus(!!apiKey, apiKey ? 'API已配置' : '未配置API'); // 大小调整功能 - 确保DOM元素已存在 const resizeHandle = Dom.$('#ph-resize'); if (resizeHandle) { resizeHandle.addEventListener('mousedown', (e) => { let resizing = true; const startWidth = wrap.offsetWidth; const startHeight = wrap.offsetHeight; const startX = e.clientX; const startY = e.clientY; function onResize(e) { if (!resizing) return; const newWidth = Math.max(300, startWidth + (e.clientX - startX)); const newHeight = Math.max(200, startHeight + (e.clientY - startY)); wrap.style.width = newWidth + 'px'; wrap.style.height = newHeight + 'px'; } function onResizeEnd() { if (!resizing) return; resizing = false; document.removeEventListener('mousemove', onResize); document.removeEventListener('mouseup', onResizeEnd); Store.set(S.size, {width: wrap.offsetWidth, height: wrap.offsetHeight}); } document.addEventListener('mousemove', onResize); document.addEventListener('mouseup', onResizeEnd); e.preventDefault(); e.stopPropagation(); }); } // 安全事件绑定 safeBind('#ph-min', 'click', () => hide()); safeBind('#ph-close', 'click', () => hide()); safeBind('#ph-rate', 'change', (e) => { Store.set(K.rate, e.target.value); Playback.poke(); setStatus(); }); safeBind('#ph-mute', 'change', (e) => { Store.set(K.mute, e.target.checked); Playback.poke(); setStatus(); }); safeBind('#ph-play-toggle', 'click', () => { const on = !Store.get(K.playOn, true); Store.set(K.playOn, on); const btn = document.getElementById('ph-play-toggle'); if (btn) btn.textContent = on ? '关闭' : '开启'; if (on) Playback.start(); else Playback.stop(); setStatus(); }); safeBind('#ph-next-toggle', 'click', () => { const on = !Store.get(K.nextOn, true); Store.set(K.nextOn, on); const btn = document.getElementById('ph-next-toggle'); if (btn) btn.textContent = on ? '关闭' : '开启'; if (on) Progress.start(); else Progress.stop(); setStatus(); }); safeBind('#ph-extract', 'click', extractQuestions); // API相关事件处理 safeBind('#ph-api-save', 'click', async () => { const apiKey = document.getElementById('ph-api-key').value.trim(); if (!apiKey) { updateAPIStatus(false, '请输入API密钥'); return; } updateAPIStatus(false, '验证中...'); const isValid = await validateAPI(apiKey); if (isValid) { Store.set(K.apiKey, apiKey); updateAPIStatus(true, 'API有效'); } else { updateAPIStatus(false, 'API无效'); } }); safeBind('#ph-answer-toggle', 'click', () => { const apiKey = Store.get(K.apiKey, ''); if (!apiKey) { updateAPIStatus(false, '请先配置API'); return; } const on = !Store.get(K.answerOn, false); Store.set(K.answerOn, on); const btn = document.getElementById('ph-answer-toggle'); if (btn) { btn.textContent = on ? '停止答题' : '开始答题'; btn.style.background = on ? '#ff4444' : '#0f1320'; } }); safeBind('#ph-auto-answer', 'click', () => { const docs = Dom.collectDocs(Dom.mainDoc()); const items = []; const sels = ['.questionLi', '.examPaper_subject', '.TiMu', '.question-box', '.question']; for (const d of docs) { for (const sel of sels) { d.querySelectorAll(sel).forEach(node => { const text = (node.innerText || '').replace(/\s+/g, ' ').trim(); if (text && text.length > 6) items.push(text); }); } } displayQuestions(items); if (items.length > 0) autoAnswer(items); }); safeBind('#ph-copy', 'click', () => { const t = document.getElementById('ph-answer-area'); if (t) { t.select(); document.execCommand('copy'); } }); safeBind('#ph-search-btn', 'click', () => { const q = (document.getElementById('ph-search-q').value || '').trim(); const result = document.getElementById('ph-search-result'); if (result) result.textContent = q ? `本地占位:你输入了「${q}」` : '请输入关键词'; }); document.querySelectorAll('.ph-tab').forEach(btn => { btn.addEventListener('click', () => switchTab(btn.dataset.tab)); }); // 拖拽:使用 pointerdown/pointermove/pointerup const header = Dom.$('#ph-header'); if (header) { header.addEventListener('pointerdown', function (e) { // 仅左键(pointerType 仍然通过 button 判断) if (e.button !== 0) return; // 如果点击的是可交互控件,则不触发拖拽(保留按钮、输入的点击) if (e.target.closest('button, input, select, textarea, .ph-tab')) return; // 记录初始坐标 dragging = true; const rect = wrap.getBoundingClientRect(); startX = e.clientX; startY = e.clientY; startLeft = rect.left; startTop = rect.top; // 切换为 left/top 定位以避免与 right/bottom 冲突 wrap.style.right = 'auto'; wrap.style.bottom = 'auto'; wrap.style.left = rect.left + 'px'; wrap.style.top = rect.top + 'px'; // 禁用文本选择体验 document.body.style.userSelect = 'none'; // 捕获指针,保证 move/up 正确分发 try { header.setPointerCapture(e.pointerId); } catch (err) {} function onPointerMove(ev) { if (!dragging) return; const dx = ev.clientX - startX, dy = ev.clientY - startY; const left = clamp(startLeft + dx, 0, window.innerWidth - wrap.offsetWidth); const top = clamp(startTop + dy, 0, window.innerHeight - wrap.offsetHeight); wrap.style.left = left + 'px'; wrap.style.top = top + 'px'; } function onPointerUp(ev) { if (!dragging) return; dragging = false; // 恢复可选中 document.body.style.userSelect = ''; try { header.releasePointerCapture(e.pointerId); } catch (err) {} document.removeEventListener('pointermove', onPointerMove); document.removeEventListener('pointerup', onPointerUp); // 保存位置 const r = wrap.getBoundingClientRect(); Store.set(S.pos, { left: Math.round(r.left), top: Math.round(r.top) }); } document.addEventListener('pointermove', onPointerMove); document.addEventListener('pointerup', onPointerUp); e.preventDefault(); }); } // 应用保存的位置和大小 const pos = Store.get(S.pos, null); const size = Store.get(S.size, null); if (pos && Number.isFinite(pos.left) && Number.isFinite(pos.top)) { wrap.style.right = 'auto'; wrap.style.bottom = 'auto'; Object.assign(wrap.style, { left: pos.left + 'px', top: pos.top + 'px' }); } // 如果没有保存的位置,保持CSS中设置的right和bottom if (size && Number.isFinite(size.width) && Number.isFinite(size.height)) { wrap.style.width = size.width + 'px'; wrap.style.height = size.height + 'px'; } // 初始化控件 Dom.$('#ph-rate').value = String(Store.get(K.rate, '1.5')); Dom.$('#ph-mute').checked = !!Store.get(K.mute, true); const playOn = !!Store.get(K.playOn, true); const nextOn = !!Store.get(K.nextOn, true); Dom.$('#ph-play-toggle').textContent = playOn ? '关闭' : '开启'; Dom.$('#ph-next-toggle').textContent = nextOn ? '关闭' : '开启'; if (Store.get(S.visible, true)) open(); else hide(); switchTab(Store.get(S.tab, 'home')); setStatus(); // 强制同步显示状态,确保面板与FAB互斥 ensureFab(); // 确保fab存在 const visible = !!Store.get(S.visible, true); if (visible) { if (wrap) wrap.style.display = 'block'; if (fab) fab.style.display = 'none'; } else { if (wrap) wrap.style.display = 'none'; if (fab) fab.style.display = 'flex'; } } function setStatus() { const s = Dom.$('#ph-status'); if (!s) return; const r = Store.get(K.rate, '1.5'); const m = !!Store.get(K.mute, true); const p = !!Store.get(K.playOn, true); const n = !!Store.get(K.nextOn, true); s.textContent = `状态:${p?'强制播放开':'强制播放关'} / ${n?'自动下一节开':'自动下一节关'} / 倍速 ${r}x / ${m?'静音':'出声'}`; } function open() { if (!wrap) mount(); wrap.style.display = 'block'; if (fab) fab.style.display = 'none'; // 隐藏悬浮球 Store.set(S.visible, true); } function hide() { if (!wrap) return; wrap.style.display = 'none'; if (fab) fab.style.display = 'flex'; // 显示悬浮球 Store.set(S.visible, false); } function toggle() { if (!wrap || wrap.style.display === 'none') open(); else hide(); } return { mount, setStatus, open, hide, toggle }; })(); // 扩展菜单与快捷键 if (typeof GM_registerMenuCommand === 'function') { GM_registerMenuCommand('显示/隐藏面板 (Alt+P)', () => Panel.toggle()); } document.addEventListener('keydown', (e) => { if (e.altKey && String(e.key).toLowerCase() === 'p') { Panel.toggle(); e.preventDefault(); } }); // 启动 function boot() { Panel.mount(); VideoMonitor.start(); // 启动视频监控 if (Store.get(K.playOn, true)) Playback.start(); else Playback.stop(); if (Store.get(K.nextOn, true)) Progress.start(); else Progress.stop(); setTimeout(() => Playback.poke(), 1200); window.PHPanel = Panel; } if (document.readyState === 'loading') document.addEventListener('DOMContentLoaded', boot); else boot(); })();