// ==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
// ==/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 = `
<h3>📚 本地答案答题助手 </h3>
<label class="file-label" for="answer-file">
选择答案文件
</label>
<input type="file" id="answer-file" accept=".txt,.json,.docx,.xlsx,.pdf">
<div class="file-name"></div>
<button id="load-answers">开始导入</button>
<div style="margin: 15px 0; border-top: 1px solid #eee;"></div>
<div class="control-group">
<button id="toggle-auto" class="toggle-button">自动答题</button>
<button id="next-question">下一题</button>
</div>
<div class="setting-item">
<input type="checkbox" id="skip-no-answer" ${CONFIG.autoAnswer.skipNoAnswer ? 'checked' : ''}>
<label for="skip-no-answer">未找到答案时自动跳过</label>
</div>
<div class="setting-item">
<label for="answer-delay">答题延迟(秒)</label>
<input type="number" id="answer-delay" min="0" max="10" step="0.5"
value="${CONFIG.autoAnswer.delay/1000}">
</div>
<div class="setting-item">
<label for="highlight-color">高亮颜色</label>
<input type="color" id="highlight-color" value="#FFEB3B">
<input type="range" id="highlight-opacity" min="0" max="100" value="30">
<span id="opacity-value">30%</span>
</div>
<div id="status"></div>
`;
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();
})();