// ==UserScript==
// @name 超星学习通刷课专属 <<< 助你解放双手
// @namespace cx-helper-xqqx
// @version 1.3.0
// @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','database','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', '.subject_type', '.TiMu',
'.questionLi .clearfix', '.ZyTop', '.question-box', '.question',
'.subject_content', 'li.topic-item'
];
for (const d of docs) {
for (const sel of sels) {
d.querySelectorAll(sel).forEach((node, index) => {
// 保留原始HTML结构,然后转换为格式化文本
let html = node.innerHTML || '';
let text = node.innerText || '';
// 保留换行和段落结构
text = text.replace(/\n\s*\n/g, '\n'); // 合并多个空行
text = text.replace(/^\s+|\s+$/g, ''); // 去除首尾空白
// 格式化选项(A、B、C、D)
text = text.replace(/([ABCD])[\.、]\s*/g, '\n$1. '); // 选项换行
text = text.replace(/\s*([ABCD])\s*[\.、]\s*/g, '\n$1. '); // 处理选项格式
// 清理题号但保留结构
text = text.replace(/^(\d+[\.\)]\s*)/, '【题目】 ');
if (text && text.length > 10) {
items.push(`=== 题目 ${items.length + 1} ===\n${text}\n`);
}
});
}
}
const area = document.getElementById('ph-answer-area');
if (area) {
if (items.length > 0) {
area.value = items.join('\n' + '-'.repeat(60) + '\n\n');
} else {
area.value = '未识别到题面,请确保在章节测试/作业页面。';
}
}
displayQuestions(items);
}
// 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;
}
// 清理题目格式,避免重复显示
const cleanQuestions = questions.map(q => {
// 移除已有的格式标记
let cleaned = q.replace(/^=== 题目 \d+ ===\n?/g, '');
cleaned = cleaned.replace(/^【题目】\s*/g, '');
return cleaned.trim();
});
window.innerHTML = cleanQuestions.map((q, i) =>
``
).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());
// 改进的题目匹配策略
for (const doc of docs) {
try {
// 找到所有题目容器
const questionContainers = doc.querySelectorAll('.questionLi, .examPaper_subject, .TiMu, .question-box, .question');
if (questionContainers[index]) {
const container = questionContainers[index];
// 根据答案类型选择不同的填入策略
if (/^[A-D]$/i.test(answer.trim())) {
// 选择题答案
const options = container.querySelectorAll('input[type="radio"], input[type="checkbox"]');
for (const option of options) {
const label = option.closest('label')?.textContent || option.parentElement?.textContent || '';
if (label.includes(answer.toUpperCase()) || label.startsWith(answer.toUpperCase())) {
option.click();
Log.info(`题目${index+1}选择答案: ${answer}`);
return true;
}
}
} else {
// 文本答案
const textInputs = container.querySelectorAll('input[type="text"], textarea');
if (textInputs.length > 0) {
textInputs[0].value = answer.trim();
textInputs[0].dispatchEvent(new Event('input', { bubbles: true }));
textInputs[0].dispatchEvent(new Event('change', { bubbles: true }));
Log.info(`题目${index+1}填入答案: ${answer}`);
return true;
}
}
}
} catch (error) {
Log.warn(`题目${index+1}填答失败:`, error);
}
}
Log.warn(`题目${index+1}未找到合适的输入元素`);
return false;
}
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 - 助你解放双手,高效学习
✨ 核心功能
- 强制播放:自动播放视频,支持倍速调节
- 自动下一节:完成后自动跳转下一章节
- 智能答题:集成DeepSeek AI自动答题
- 视频监控:检测播放异常并提示切换线路
🛠️ 使用说明
- 面板可拖动、缩放,位置自动记忆
- Alt+P 或右下角"CX"可显示/隐藏面板
- 设置页面可调节倍速、静音等参数
- 答题页面需配置API密钥才能使用AI功能
⚠️ 使用提醒
本脚本仅供学习交流使用,请遵守学校相关规定,合理使用辅助工具。
点击搜索按钮开始查找答案
题库搜题免费使用,AI搜题需配置DeepSeek API
`;
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);
// 定期检查是否需要自动答题
setInterval(checkAndAutoAnswer, 3000);
// 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'); }
});
// 搜题模式切换和功能实现
let searchMode = 'database'; // 'database' 或 'ai'
safeBind('#ph-search-mode-db', 'click', () => {
searchMode = 'database';
document.getElementById('ph-search-mode-db').className = 'ph-btn primary';
document.getElementById('ph-search-mode-ai').className = 'ph-btn';
document.getElementById('ph-search-mode-desc').textContent = '题库搜题:直接给出标准答案';
});
safeBind('#ph-search-mode-ai', 'click', () => {
searchMode = 'ai';
document.getElementById('ph-search-mode-db').className = 'ph-btn';
document.getElementById('ph-search-mode-ai').className = 'ph-btn primary';
document.getElementById('ph-search-mode-desc').textContent = 'AI搜题:提供答案解析和推理过程';
});
// 题库答题页面功能
safeBind('#ph-db-answer-toggle', 'click', () => {
const on = !Store.get('ph_db_answer_on', false);
Store.set('ph_db_answer_on', on);
const btn = document.getElementById('ph-db-answer-toggle');
if (btn) {
btn.textContent = on ? '停止题库答题' : '开始题库答题';
btn.style.background = on ? '#dc3545' : '#28a745';
}
if (on) {
// 开始自动检测和答题
startDatabaseAnswering();
} else {
// 停止自动答题
stopDatabaseAnswering();
}
});
safeBind('#ph-db-extract', 'click', () => {
const docs = Dom.collectDocs(Dom.mainDoc());
const items = [];
const sels = [
'.questionLi', '.examPaper_subject', '.subject_type', '.TiMu',
'.questionLi .clearfix', '.ZyTop', '.question-box', '.question',
'.subject_content', 'li.topic-item'
];
for (const d of docs) {
for (const sel of sels) {
d.querySelectorAll(sel).forEach((node, index) => {
let html = node.innerHTML || '';
let text = node.innerText || '';
text = text.replace(/\n\s*\n/g, '\n');
text = text.replace(/^\s+|\s+$/g, '');
text = text.replace(/([ABCD])[\.、]\s*/g, '\n$1. ');
text = text.replace(/\s*([ABCD])\s*[\.、]\s*/g, '\n$1. ');
text = text.replace(/^(\d+[\.\)]\s*)/, '【题目】 ');
if (text && text.length > 10) {
items.push(`=== 题目 ${items.length + 1} ===\n${text}\n`);
}
});
}
}
const area = document.getElementById('ph-db-answer-area');
const window = document.getElementById('ph-db-question-window');
if (area) {
if (items.length > 0) {
area.value = items.join('\n' + '-'.repeat(60) + '\n\n');
} else {
area.value = '未识别到题面,请确保在章节测试/作业页面。';
}
}
// 显示在题目窗口
if (window) {
if (items.length === 0) {
window.innerHTML = '未找到题目
';
return;
}
const cleanQuestions = items.map(q => {
let cleaned = q.replace(/^=== 题目 \d+ ===\n?/g, '');
cleaned = cleaned.replace(/^【题目】\s*/g, '');
return cleaned.trim();
});
window.innerHTML = cleanQuestions.map((q, i) =>
``
).join('');
}
});
safeBind('#ph-db-copy', 'click', () => {
const t = document.getElementById('ph-db-answer-area');
if (t) { t.select(); document.execCommand('copy'); }
});
// 题库自动答题逻辑
let dbAnsweringInterval = null;
function startDatabaseAnswering() {
if (dbAnsweringInterval) return;
dbAnsweringInterval = setInterval(async () => {
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);
});
}
}
if (items.length > 0) {
Log.info('检测到题目,开始题库答题');
for (let i = 0; i < items.length; i++) {
try {
const answer = await searchFromDatabase(items[i]);
if (answer) {
fillAnswer(items[i], answer, i);
Log.info(`题目${i+1}答题完成:`, answer);
}
} catch (error) {
Log.warn(`题目${i+1}答题失败:`, error);
}
if (i < items.length - 1) {
await new Promise(resolve => setTimeout(resolve, 1000));
}
}
}
}, 5000); // 每5秒检测一次
}
function stopDatabaseAnswering() {
if (dbAnsweringInterval) {
clearInterval(dbAnsweringInterval);
dbAnsweringInterval = null;
}
}
// 题库搜题函数
async function searchFromDatabase(question) {
try {
const response = await fetch(`https://api.pearktrue.cn/api/question/?question=${encodeURIComponent(question)}`);
const data = await response.json();
if (data.code === 200 && data.data) {
return `【题库答案】\n题目:${data.data.question}\n答案:${data.data.answer}`;
} else {
return '题库中未找到该题目';
}
} catch (error) {
throw new Error('题库API请求失败');
}
}
// AI搜题函数
async function searchFromAI(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: 'system',
content: '你是一个专业的题目解答助手。请按照以下格式回答:\n1. 首先给出明确的答案(如选择题直接给A/B/C/D,判断题给对/错)\n2. 然后简要说明理由(控制在100字以内)\n3. 回答要简洁明了,避免冗长的解释'
},
{
role: 'user',
content: `请回答以下题目:\n\n${question}`
}
],
max_tokens: 200,
temperature: 0.1
})
});
const data = await response.json();
if (data.choices && data.choices[0]) {
const content = data.choices[0].message.content;
return `【AI解析】\n${content}`;
} else {
throw new Error('AI响应格式错误');
}
} catch (error) {
throw new Error('AI API请求失败');
}
}
safeBind('#ph-search-btn', 'click', async () => {
const question = document.getElementById('ph-search-q').value.trim();
const resultDiv = document.getElementById('ph-search-result');
if (!question) {
resultDiv.textContent = '请输入题目内容';
return;
}
resultDiv.textContent = '搜索中...';
try {
let result;
if (searchMode === 'database') {
result = await searchFromDatabase(question);
} else {
const apiKey = Store.get(K.apiKey, '');
if (!apiKey) {
resultDiv.textContent = 'AI搜题需要先配置DeepSeek API密钥';
return;
}
result = await searchFromAI(apiKey, question);
}
resultDiv.textContent = result || '未找到相关答案';
} catch (error) {
resultDiv.textContent = `搜索失败: ${error.message}`;
}
});
// 清空搜索结果按钮
safeBind('#ph-search-clear', 'click', () => {
document.getElementById('ph-search-q').value = '';
document.getElementById('ph-search-result').textContent = '点击搜索按钮开始查找答案';
});
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();
})();