// ==UserScript== // @name 本地答案答题助手 // @namespace http://tampermonkey.net/ // @version 0.5 // @description 从本地文档检索答案并自动选中 // @author 侯钰熙 // @match *://*/* // @icon  // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_xmlhttpRequest // @require https://cdnjs.cloudflare.com/ajax/libs/pdf.js/3.11.174/pdf.min.js // @require https://unpkg.com/xlsx/dist/xlsx.full.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/mammoth/1.4.0/mammoth.browser.min.js // @license MIT // @downloadURL https://update.greasyfork.icu/scripts/521835/%E6%9C%AC%E5%9C%B0%E7%AD%94%E6%A1%88%E7%AD%94%E9%A2%98%E5%8A%A9%E6%89%8B.user.js // @updateURL https://update.greasyfork.icu/scripts/521835/%E6%9C%AC%E5%9C%B0%E7%AD%94%E6%A1%88%E7%AD%94%E9%A2%98%E5%8A%A9%E6%89%8B.meta.js // ==/UserScript== (function() { 'use strict'; // 初始化 PDF.js const pdfjsLib = window['pdfjs-dist/build/pdf']; pdfjsLib.GlobalWorkerOptions.workerSrc = 'https://cdnjs.cloudflare.com/ajax/libs/pdf.js/3.11.174/pdf.worker.min.js'; // 配置项 const CONFIG = { // 题目选择器,根据实际网站调整 questionSelector: '.subject-item .subject-question', // 选项选择器 optionSelector: '.subject-item .subject-option input[type="radio"]', // 本地答案数据 answers: { // 示例数据格式 '题目1': '答案1', '题目2': '答案2' }, autoAnswer: { enabled: false, delay: 2000, // 答题延迟,单位毫秒 skipNoAnswer: true, // 是否跳过没有答案的题目 }, articleContent: '', // 存储文章内容 matchThreshold: 0.8, // 文本相似度匹配阈值 highlight: { color: 'rgba(255, 235, 59, 0.3)', // 黄色半透明 borderColor: '#FFC107', currentQuestion: null // 存储当前题目元素 }, nextButtonSelector: '.next-btn, .submit-btn', // 下一题按钮选择器 }; // 添加控制面板样式 GM_addStyle(` #answer-helper-panel { position: fixed; top: 20px; right: 20px; background: #ffffff; padding: 15px; border-radius: 12px; box-shadow: 0 4px 20px rgba(0, 0, 0, 0.1); z-index: 9999; font-family: 'PingFang SC', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; min-width: 240px; transition: all 0.3s ease; } #answer-helper-panel:hover { box-shadow: 0 6px 25px rgba(0, 0, 0, 0.15); } #answer-helper-panel h3 { margin: 0 0 15px 0; color: #333; font-size: 16px; font-weight: 600; text-align: center; } #answer-helper-panel input[type="file"] { display: none; } #answer-helper-panel .file-label { display: block; padding: 8px 12px; background: #f0f2f5; border-radius: 8px; color: #666; cursor: pointer; margin-bottom: 10px; text-align: center; transition: all 0.2s ease; font-size: 14px; } #answer-helper-panel .file-label:hover { background: #e6e8eb; color: #333; } #answer-helper-panel .file-name { font-size: 12px; color: #666; margin: 5px 0; text-align: center; word-break: break-all; } #answer-helper-panel button { width: 100%; padding: 8px 15px; background: #4CAF50; color: white; border: none; border-radius: 8px; cursor: pointer; font-size: 14px; font-weight: 500; transition: all 0.2s ease; } #answer-helper-panel button:hover { background: #43A047; transform: translateY(-1px); } #answer-helper-panel button:active { transform: translateY(1px); } #status { margin-top: 10px; padding: 8px; border-radius: 6px; background: #f5f5f5; font-size: 12px; color: #666; text-align: center; min-height: 20px; } .status-success { color: #4CAF50 !important; background: #E8F5E9 !important; } .status-error { color: #F44336 !important; background: #FFEBEE !important; } .status-loading { color: #2196F3 !important; background: #E3F2FD !important; } .control-group { display: flex; gap: 8px; margin-bottom: 10px; } .control-group button { flex: 1; min-width: 0; padding: 6px 8px; } .toggle-button { background: #FF9800 !important; } .toggle-button.active { background: #E65100 !important; } .setting-item { display: flex; align-items: center; margin: 8px 0; font-size: 12px; color: #666; } .setting-item input[type="checkbox"] { margin-right: 8px; } .setting-item input[type="number"] { width: 60px; padding: 2px 4px; margin-left: 8px; border: 1px solid #ddd; border-radius: 4px; } .question-highlight { position: relative; background-color: ${CONFIG.highlight.color} !important; border: 2px solid ${CONFIG.highlight.borderColor} !important; border-radius: 4px; transition: all 0.3s ease; } .question-highlight::before { content: '当前题目'; position: absolute; top: -20px; left: 50%; transform: translateX(-50%); background: ${CONFIG.highlight.borderColor}; color: #fff; padding: 2px 8px; border-radius: 4px; font-size: 12px; z-index: 1000; } `); // 创建控制面板 function createPanel() { const panel = document.createElement('div'); panel.id = 'answer-helper-panel'; panel.innerHTML = `

📚 本地答案答题助手

30%
`; document.body.appendChild(panel); // 文件选择事件 const fileInput = panel.querySelector('#answer-file'); const fileLabel = panel.querySelector('.file-label'); const fileName = panel.querySelector('.file-name'); fileInput.addEventListener('change', (e) => { if (e.target.files.length > 0) { fileName.textContent = e.target.files[0].name; fileLabel.textContent = '更换文件'; } else { fileName.textContent = ''; fileLabel.textContent = '选择答案文件'; } }); document.getElementById('load-answers').addEventListener('click', loadAnswersFromFile); // 添加控制事件监听 const toggleButton = document.getElementById('toggle-auto'); const nextButton = document.getElementById('next-question'); const skipCheckbox = document.getElementById('skip-no-answer'); const delayInput = document.getElementById('answer-delay'); toggleButton.addEventListener('click', () => { CONFIG.autoAnswer.enabled = !CONFIG.autoAnswer.enabled; toggleButton.classList.toggle('active'); toggleButton.textContent = CONFIG.autoAnswer.enabled ? '停止答题' : '自动答题'; if (CONFIG.autoAnswer.enabled) { startAutoAnswer(); } }); nextButton.addEventListener('click', () => { clickNextQuestion(); }); skipCheckbox.addEventListener('change', (e) => { CONFIG.autoAnswer.skipNoAnswer = e.target.checked; }); delayInput.addEventListener('change', (e) => { CONFIG.autoAnswer.delay = Math.max(0, parseFloat(e.target.value) * 1000); }); // 添加高亮颜色设置事件 const colorInput = document.getElementById('highlight-color'); const opacityInput = document.getElementById('highlight-opacity'); const opacityValue = document.getElementById('opacity-value'); function updateHighlightColor() { const color = colorInput.value; const opacity = opacityInput.value / 100; CONFIG.highlight.color = `${color}${Math.round(opacity * 255).toString(16).padStart(2, '0')}`; CONFIG.highlight.borderColor = color; // 更新样式 GM_addStyle(` .question-highlight { background-color: ${CONFIG.highlight.color} !important; border: 2px solid ${CONFIG.highlight.borderColor} !important; } .question-highlight::before { background: ${CONFIG.highlight.borderColor}; } `); // 如果当前有高亮的题目,刷新其样式 if (CONFIG.highlight.currentQuestion) { CONFIG.highlight.currentQuestion.classList.remove('question-highlight'); setTimeout(() => { CONFIG.highlight.currentQuestion.classList.add('question-highlight'); }, 0); } } colorInput.addEventListener('input', updateHighlightColor); opacityInput.addEventListener('input', () => { opacityValue.textContent = `${opacityInput.value}%`; updateHighlightColor(); }); } // 文件加载处理函数 async function loadAnswersFromFile() { const fileInput = document.getElementById('answer-file'); const file = fileInput.files[0]; if (!file) { updateStatus('请选择文件', 'error'); return; } updateStatus('正在解析文件...', 'loading'); try { let answers = {}; const fileExt = file.name.split('.').pop().toLowerCase(); switch (fileExt) { case 'json': case 'txt': answers = await parseTextFile(file); break; case 'docx': answers = await parseWordFile(file); break; case 'xlsx': answers = await parseExcelFile(file); break; case 'pdf': answers = await parsePDFFile(file); break; default: throw new Error('不支持的文件格式'); } CONFIG.answers = answers; GM_setValue('answers', answers); updateStatus('答案加载成功 ✨', 'success'); startAutoMatch(); } catch (error) { updateStatus('文件解析错误: ' + error.message, 'error'); } } // 解析文本文件(JSON/TXT) function parseTextFile(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onload = (e) => { try { const content = e.target.result; try { // 尝试解析为JSON const answers = JSON.parse(content); resolve(answers); } catch { // 如果不是JSON,则作为文章内容处理 CONFIG.articleContent = content; resolve({}); // 返回空对象,因为答案需要实时搜索 } } catch (error) { reject(new Error('文件解析失败')); } }; reader.onerror = () => reject(new Error('文件读取失败')); reader.readAsText(file); }); } // 解析Word文件 function parseWordFile(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onload = async (e) => { try { const arrayBuffer = e.target.result; const result = await mammoth.extractRawText({ arrayBuffer }); const text = result.value; // 假设Word文档的格式是每行一个题目答案对,用制表符或特定分隔符分隔 const answers = {}; const lines = text.split('\n'); lines.forEach(line => { const [question, answer] = line.split('\t'); if (question && answer) { answers[question.trim()] = answer.trim(); } }); resolve(answers); } catch (error) { reject(new Error('Word文件解析失败')); } }; reader.onerror = () => reject(new Error('文件读取失败')); reader.readAsArrayBuffer(file); }); } // 解析Excel���件 function parseExcelFile(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onload = (e) => { try { const data = new Uint8Array(e.target.result); const workbook = XLSX.read(data, { type: 'array' }); const firstSheet = workbook.Sheets[workbook.SheetNames[0]]; const answers = {}; // 假设Excel的A列是题目,B列是答案 const jsonData = XLSX.utils.sheet_to_json(firstSheet, { header: ['question', 'answer'] }); jsonData.forEach(row => { if (row.question && row.answer) { answers[row.question.trim()] = row.answer.trim(); } }); resolve(answers); } catch (error) { reject(new Error('Excel文件解析失败')); } }; reader.onerror = () => reject(new Error('文件读取失败')); reader.readAsArrayBuffer(file); }); } // 解析PDF文件 function parsePDFFile(file) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onload = async (e) => { try { const typedArray = new Uint8Array(e.target.result); const loadingTask = pdfjsLib.getDocument({ data: typedArray }); updateStatus('正在解析PDF...', 'loading'); const pdf = await loadingTask.promise; const answers = {}; const numPages = pdf.numPages; // 读取所有页面的文本 for (let i = 1; i <= numPages; i++) { updateStatus(`正在解析第 ${i}/${numPages} 页...`, 'loading'); const page = await pdf.getPage(i); const textContent = await page.getTextContent(); let pageText = ''; // 更好的文本提取逻辑 let lastY = null; textContent.items.forEach(item => { if (lastY !== item.transform[5]) { pageText += '\n'; lastY = item.transform[5]; } pageText += item.str + ' '; }); // 处理每一行 const lines = pageText.split('\n').filter(line => line.trim()); lines.forEach(line => { // 尝试多种分隔符 let parts = line.split(/[\t|||│||]/); if (parts.length < 2) { parts = line.split(/\s{2,}/); // 使用两个或更多空格作为分隔符 } if (parts.length >= 2) { const question = parts[0].trim(); const answer = parts[parts.length - 1].trim(); if (question && answer) { answers[question] = answer; } } }); } if (Object.keys(answers).length === 0) { reject(new Error('未能从PDF中提取到答案数据')); } else { updateStatus('PDF解析完成', 'success'); resolve(answers); } } catch (error) { console.error('PDF解析错误:', error); reject(new Error('PDF文件解析失败: ' + error.message)); } }; reader.onerror = () => reject(new Error('文件读取失败')); reader.readAsArrayBuffer(file); }); } // 更新状态显示 function updateStatus(message, type = 'normal') { const statusEl = document.getElementById('status'); statusEl.textContent = message; // 移除所有状态类 statusEl.classList.remove('status-success', 'status-error', 'status-loading'); // 根据类型添加对应的状态类 switch (type) { case 'success': statusEl.classList.add('status-success'); break; case 'error': statusEl.classList.add('status-error'); break; case 'loading': statusEl.classList.add('status-loading'); break; } } // 添加高亮当前题目的函数 function highlightCurrentQuestion(question) { // 移除之前的高亮 if (CONFIG.highlight.currentQuestion) { CONFIG.highlight.currentQuestion.classList.remove('question-highlight'); } // 添加新的高亮 if (question) { question.classList.add('question-highlight'); CONFIG.highlight.currentQuestion = question; // 滚动到当前题目 question.scrollIntoView({ behavior: 'smooth', block: 'center' }); } } // 开始自动匹配答案 async function startAutoMatch() { // 获取当前题目 const questionItem = document.querySelector('.subject-item'); if (!questionItem) return false; // 高亮当前题目 highlightCurrentQuestion(questionItem); // 获取题目文本(去除分数等额外信息) const questionText = questionItem.querySelector('.subject-question') ?.textContent.trim() .replace(/(\d+分)/, '') // 移除分数提示 .replace(/^\d+\.\s*/, '') // 移除题号 .trim(); if (!questionText) return false; // 先尝试精确匹配 let answer = CONFIG.answers[questionText]; if (answer) { return selectAnswer(answer, questionItem); } // 如果有文章内容,尝试从文章中查找答案 if (CONFIG.articleContent) { const options = Array.from(questionItem.querySelectorAll('.subject-option')); if (!options.length) return false; // 获取问题上下文 const context = findQuestionContext(questionText); // 计算每个选项在上下文中的相似度 const optionScores = options.map(option => { const optionText = option.textContent.trim().replace(/^[A-Z]\./, '').trim(); const score = calculateContextSimilarity(context, optionText); return { option, optionText, score }; }); // 获取最高分的选项 const bestMatch = optionScores.reduce((best, current) => { return current.score > best.score ? current : best; }, { score: 0 }); if (bestMatch.score >= CONFIG.matchThreshold) { const radioInput = bestMatch.option.querySelector('input[type="radio"]'); if (radioInput) { radioInput.click(); updateStatus(`已选中最佳匹配答案: ${bestMatch.optionText} (相似度: ${bestMatch.score.toFixed(2)})`, 'success'); return true; } } } updateStatus('未找到匹配的答案', 'error'); return false; } // 查找问题上下文 function findQuestionContext(question) { const normalizedContent = CONFIG.articleContent.replace(/\s+/g, ''); const normalizedQuestion = question.replace(/\s+/g, ''); // 在文章中查找问题相关内容 let bestMatchIndex = -1; let bestMatchScore = 0; // 使用滑动窗口查找最相关的段落 const windowSize = 100; // 上下文窗口大小 for (let i = 0; i < normalizedContent.length - windowSize; i++) { const window = normalizedContent.slice(i, i + windowSize); const score = similarity(window, normalizedQuestion); if (score > bestMatchScore) { bestMatchScore = score; bestMatchIndex = i; } } if (bestMatchIndex >= 0) { // 返回最佳匹配位置的上下文 const start = Math.max(0, bestMatchIndex - 50); const end = Math.min(normalizedContent.length, bestMatchIndex + windowSize + 50); return normalizedContent.slice(start, end); } return ''; } // 计算选项在上下文中的相似度 function calculateContextSimilarity(context, option) { if (!context || !option) return 0; // 将上下文分成小段落 const segments = []; const segmentLength = option.length * 2; for (let i = 0; i < context.length - segmentLength; i += segmentLength / 2) { segments.push(context.slice(i, i + segmentLength)); } // 计算选项与每个段落的相似度,取最高值 return segments.reduce((maxScore, segment) => { const score = similarity(segment, option); return Math.max(maxScore, score); }, 0); } // 选择答案的辅助函数 function selectAnswer(answer, questionItem) { const options = questionItem.querySelectorAll('.subject-option'); for (const option of options) { const optionText = option.textContent.trim().replace(/^[A-Z]\./, '').trim(); if (similarity(optionText, answer) >= CONFIG.matchThreshold) { const radioInput = option.querySelector('input[type="radio"]'); if (radioInput) { radioInput.click(); updateStatus(`已选中答案: ${answer}`, 'success'); return true; } } } return false; } // 添加自动答题相关函数 function startAutoAnswer() { if (!CONFIG.autoAnswer.enabled) return; startAutoMatch().then(matched => { if (!matched && CONFIG.autoAnswer.skipNoAnswer) { updateStatus('未找到答案,准备跳过...', 'loading'); setTimeout(() => { clickNextQuestion(); }, 1000); } else if (matched) { updateStatus('答题成功,等待下一题...', 'success'); setTimeout(() => { clickNextQuestion(); startAutoAnswer(); }, CONFIG.autoAnswer.delay); } else { CONFIG.autoAnswer.enabled = false; document.getElementById('toggle-auto').classList.remove('active'); document.getElementById('toggle-auto').textContent = '自动答题'; updateStatus('自动答题已停止', 'error'); } }); } // 点击下一题按钮 function clickNextQuestion() { // 移除当前高亮 if (CONFIG.highlight.currentQuestion) { CONFIG.highlight.currentQuestion.classList.remove('question-highlight'); CONFIG.highlight.currentQuestion = null; } // 查找下一题或提交按钮 const nextButton = document.querySelector(CONFIG.nextButtonSelector); if (nextButton) { nextButton.click(); // 等待新题目加载完成后高亮 setTimeout(() => { const newQuestion = document.querySelector('.subject-item'); if (newQuestion) { highlightCurrentQuestion(newQuestion); } }, 500); } else { updateStatus('未找到下一题按钮', 'error'); CONFIG.autoAnswer.enabled = false; document.getElementById('toggle-auto').classList.remove('active'); document.getElementById('toggle-auto').textContent = '自动答题'; } } // 初始化 function init() { createPanel(); // 从存储中恢复答案数据 const savedAnswers = GM_getValue('answers'); if (savedAnswers) { CONFIG.answers = savedAnswers; startAutoMatch(); } } // 启动脚本 init(); })();