// ==UserScript== // @name 🥇【举名继教小助手】无人值守|自动静音|视频助手|考试助手 // @namespace https://juming-helper.example.com // @version 5.8.4 // @description 脚本本身免费,但也希望得到大家的支持。脚本可用于【举名继续教育网】的相关课程,功能包含自动静音、自动签到、连续考试等。 // @author 境界程序员 // @license AGPL License // @match *://*.jumingedu.com/* // @icon  // @grant none // @run-at document-end // @antifeature Donate听说含捐赠功能需要添加此代码(无任何作用) // ==/UserScript== (function () { 'use strict'; // ========== 全局配置与状态 ========== const CONFIG_KEY = 'jm_helper_config_v583'; const COMPLETED_EXAMS_KEY = 'jm_completed_exams_v583'; // 使用短网址替代长base64 const ASSISTANT_IMAGE_SRC = ''; // 默认配置 (autoAnswer 默认为 true, maxExamRetryAttempts 固定为 10) const DEFAULT_CONFIG = { continuousExamMode: false, // 连续考试模式 autoMute: true, autoJumpUnfinished: true, autoSign: true, autoAnswer: true, // 默认开启 logLevel: 'INFO', // DEBUG, INFO, WARN, ERROR clickDelayMin: 300, clickDelayMax: 500, // maxExamRetryAttempts: 10 // 硬编码,不从配置读取 }; let config = { ...DEFAULT_CONFIG }; let currentMode = null; let lastUrl = location.href; let isProcessingExam = false; let shouldStopAutoAnswer = false; let isInitialized = false; // 防止重复初始化 let isPanelInjected = false; // 专门标记UI面板是否已注入 // 日志等级映射 const logLevels = { 'DEBUG': 0, 'INFO': 1, 'WARN': 2, 'ERROR': 3 }; let currentLogLevel = logLevels[config.logLevel]; // 视频模块相关状态 let silenceGuardActive = false; let silenceGuardInterval = null; let videoObserver = null; let signInterval = null; // ========== 工具函数 ========== function log(level, message, force = false) { if (force || logLevels[level] >= currentLogLevel) { const timestamp = new Date().toISOString().slice(11, 23); const styledMessage = `%c[${timestamp}] [${level}] ${message}`; const styles = { 'DEBUG': 'color: #BB86FC;', 'INFO': 'color: #03DAC6; font-weight: bold;', 'WARN': 'color: #FFD60A; font-weight: bold;', 'ERROR': 'color: #CF6679; font-weight: bold;' }; console.log(styledMessage, styles[level]); } } // 获取格式化时间戳 (时:分:秒) function getFormattedTimestamp() { const now = new Date(); const hours = String(now.getHours()).padStart(2, '0'); const minutes = String(now.getMinutes()).padStart(2, '0'); const seconds = String(now.getSeconds()).padStart(2, '0'); return `${hours}:${minutes}:${seconds}`; } // 三行滚动日志函数 function showScrollingLog(message, level = 'INFO') { const statusDiv = document.getElementById('jm-status'); if (!statusDiv) return; const timestamp = getFormattedTimestamp(); const logEntry = `[${timestamp}] ${message}`; const logColor = { 'DEBUG': '#BB86FC', 'INFO': '#03DAC6', 'WARN': '#FFD60A', 'ERROR': '#CF6679' }[level] || '#03DAC6'; // 获取现有的日志数组,如果没有则初始化 let logs = statusDiv.dataset.logs ? JSON.parse(statusDiv.dataset.logs) : []; // 添加新日志到数组开头 logs.unshift({ text: logEntry, color: logColor }); // 限制数组长度为3 if (logs.length > 3) { const removedLog = logs.pop(); // 移除最旧的日志 // 为被移除的日志设置淡出效果和延迟移除 setTimeout(() => { // 检查当前显示的日志是否包含被移除的日志内容 const currentText = statusDiv.innerHTML; if (currentText.includes(removedLog.text)) { // 如果当前显示的包含了要移除的,说明它已经被新的覆盖了,无需操作 // 如果它还在显示,我们需要替换它 const lines = currentText.split('
'); const updatedLines = lines.map(line => { if (line.includes(removedLog.text)) { return line.replace(/opacity:\s*1;/, 'opacity: 0.3; transition: opacity 0.5s ease;') .replace(/color:\s*[^;]+;/, `color: ${removedLog.color};`); } return line; }); statusDiv.innerHTML = updatedLines.join('
'); // 0.5秒后完全移除 setTimeout(() => { const finalLines = updatedLines.filter(line => !line.includes(removedLog.text)); statusDiv.innerHTML = finalLines.join('
'); }, 500); } }, 2000); // 2秒后开始淡出 } // 更新 data 属性 statusDiv.dataset.logs = JSON.stringify(logs); // 重构HTML内容 const htmlLines = logs.map((log, index) => { // 最新的一行 (index === 0) 持续显示,不设淡出 const opacityStyle = index === 0 ? 'opacity: 1;' : 'opacity: 1; transition: opacity 0.5s ease;'; return `${log.text}`; }); statusDiv.innerHTML = htmlLines.join('
'); } function isVisible(el) { if (!el || !(el instanceof HTMLElement)) return false; const style = window.getComputedStyle(el); if ( style.display === 'none' || style.visibility === 'hidden' || parseFloat(style.opacity) <= 0 || el.offsetWidth === 0 || el.offsetHeight === 0 ) { return false; } const rect = el.getBoundingClientRect(); return ( rect.bottom > 0 && rect.top < (window.innerHeight || document.documentElement.clientHeight) && rect.right > 0 && rect.left < (window.innerWidth || document.documentElement.clientWidth) ); } function safeClickElement(el) { if (!el) return false; if (!isVisible(el)) { el.scrollIntoView({ behavior: 'auto', block: 'center' }); // 等待滚动完成 return new Promise(resolve => setTimeout(() => { if (isVisible(el)) { const event = new MouseEvent('click', { view: window, bubbles: true, cancelable: true, button: 0 }); el.dispatchEvent(event); resolve(true); } else { resolve(false); } }, 300)); } else { const event = new MouseEvent('click', { view: window, bubbles: true, cancelable: true, button: 0 }); el.dispatchEvent(event); return Promise.resolve(true); } } function safeClickRadio(input) { if (!input || !(input instanceof HTMLInputElement)) return false; if (input.checked) return true; input.scrollIntoView({ behavior: 'auto', block: 'center' }); const container = input.closest('.el-radio') || input.parentElement; if (container && !isVisible(container)) { container.scrollIntoView({ behavior: 'auto', block: 'center' }); } input.checked = true; input.dispatchEvent(new Event('change', { bubbles: true })); input.dispatchEvent(new Event('input', { bubbles: true })); if (container && container.offsetParent !== null) { setTimeout(() => { if (isVisible(container)) { const event = new MouseEvent('click', { view: window, bubbles: true, cancelable: true, button: 0 }); container.dispatchEvent(event); } }, 50); } return input.checked; } function loadConfig() { try { const stored = localStorage.getItem(CONFIG_KEY); if (stored) { const parsed = JSON.parse(stored); config = { ...DEFAULT_CONFIG, ...parsed }; currentLogLevel = logLevels[config.logLevel]; log('INFO', '配置已从本地加载'); showScrollingLog('配置已从本地加载', 'INFO'); } else { log('INFO', '使用默认配置'); showScrollingLog('使用默认配置', 'INFO'); } } catch (e) { log('ERROR', `加载配置失败: ${e.message}`); showScrollingLog(`加载配置失败: ${e.message}`, 'ERROR'); config = { ...DEFAULT_CONFIG }; } } function saveConfig() { try { localStorage.setItem(CONFIG_KEY, JSON.stringify(config)); log('INFO', '配置已保存到本地'); } catch (e) { log('ERROR', `保存配置失败: ${e.message}`); showScrollingLog(`保存配置失败: ${e.message}`, 'ERROR'); } } function loadCompletedExams() { try { const stored = localStorage.getItem(COMPLETED_EXAMS_KEY); return stored ? JSON.parse(stored) : []; } catch (e) { log('ERROR', `加载已完成考核记录失败: ${e.message}`); showScrollingLog(`加载已完成考核记录失败: ${e.message}`, 'ERROR'); return []; } } function recordCompletedExam(videoTitle) { const completedList = loadCompletedExams(); if (!completedList.includes(videoTitle)) { completedList.push(videoTitle); try { localStorage.setItem(COMPLETED_EXAMS_KEY, JSON.stringify(completedList)); log('INFO', `已记录视频 "${videoTitle}" 为已完成考核。`); showScrollingLog(`已记录视频 "${videoTitle}" 为已完成考核。`, 'INFO'); } catch (e) { log('ERROR', `保存已完成考核记录失败: ${e.message}`); showScrollingLog(`保存已完成考核记录失败: ${e.message}`, 'ERROR'); } } } function hasCompletedExam(videoTitle) { const completedList = loadCompletedExams(); return completedList.includes(videoTitle); } // ========== 视频模块相关函数 ========== function enforceSilence(video) { if (video.hasAttribute('data-jm-auto-disabled')) return; video.muted = true; video.volume = 0; } function handleVideos() { if (!config.autoMute) return; document.querySelectorAll('video').forEach(video => { if (video.hasAttribute('data-jm-listening')) return; video.setAttribute('data-jm-listening', 'true'); enforceSilence(video); video.addEventListener('play', () => { video.removeAttribute('data-jm-auto-disabled'); log('DEBUG', '视频播放,已恢复静音'); }); }); } function startSilenceGuard() { if (silenceGuardActive || !config.autoMute) return; silenceGuardActive = true; silenceGuardInterval = setInterval(() => { document.querySelectorAll('video').forEach(enforceSilence); }, 800); window.addEventListener('beforeunload', () => clearInterval(silenceGuardInterval)); } function stopSilenceGuard() { if (silenceGuardInterval) { clearInterval(silenceGuardInterval); silenceGuardInterval = null; silenceGuardActive = false; log('INFO', '静音守护已停止'); showScrollingLog('静音守护已停止', 'INFO'); } } function autoJumpToUnfinishedVideo() { if (!config.autoJumpUnfinished) return; const items = document.querySelectorAll('.directoryList .item'); let foundUnfinished = false; for (const item of items) { const statusEl = item.querySelector('.status'); if (!statusEl) continue; const text = statusEl.textContent.trim(); const match = text.match(/^已学(\d+)%$/); if (match && parseInt(match[1], 10) < 100) { safeClickElement(item); log('INFO', `自动跳转至未完成视频:${item.querySelector('.title')?.textContent}`); showScrollingLog(`自动跳转至未完成视频`, 'INFO'); foundUnfinished = true; break; } } if (!foundUnfinished) { log('DEBUG', '未找到未完成视频'); } } function trySign() { if (!config.autoSign) return false; const now = Date.now(); const MIN_INTERVAL = 5000; const spans = document.querySelectorAll('button.el-button--primary span'); for (const span of spans) { if (span.textContent.trim() !== '点击签到') continue; const button = span.closest('button'); if (!button || button.disabled || button.classList.contains('is-disabled') || !isVisible(button)) continue; log('INFO', '签到按钮已点击'); showScrollingLog('签到按钮已点击', 'INFO'); safeClickElement(button); return true; } return false; } // 启动签到轮询 function startSignPolling() { if (signInterval) clearInterval(signInterval); if (config.autoSign) { signInterval = setInterval(() => { trySign(); }, 3000); log('INFO', '签到轮询已启动'); showScrollingLog('签到轮询已启动', 'INFO'); } } // 停止签到轮询 function stopSignPolling() { if (signInterval) { clearInterval(signInterval); signInterval = null; log('INFO', '签到轮询已停止'); showScrollingLog('签到轮询已停止', 'INFO'); } } // ========== UI 配置面板 ========== function injectUIPanel() { if (document.getElementById('jm-config-panel')) return; const panel = document.createElement('div'); panel.id = 'jm-config-panel'; panel.style.cssText = ` position: fixed; left: 0; top: 320px; z-index: 99999; background: linear-gradient(145deg, #1a1a2e, #16213e); color: #e6e6e6; /* 高对比度文字颜色 */ width: 280px; border-radius: 12px; box-shadow: 0 8px 24px rgba(0,0,0,0.5); font-family: 'Microsoft YaHei', sans-serif; padding: 12px; backdrop-filter: blur(6px); border: 1px solid rgba(100, 100, 255, 0.3); /* 更明显的边框 */ user-select: none; cursor: move; `; panel.innerHTML = `
🥇举名继教小助手 ${GM_info.script.version}
助手头像
开发者日志等级:
最小延迟 (ms):
最大延迟 (ms):
[${getFormattedTimestamp()}] 状态: 等待配置
`; document.body.appendChild(panel); const autoMuteInput = document.getElementById('jm-auto-mute'); const autoJumpInput = document.getElementById('jm-auto-jump'); const autoSignInput = document.getElementById('jm-auto-sign'); const continuousExamInput = document.getElementById('jm-continuous-exam'); const logLevelSelect = document.getElementById('jm-log-level'); const clickDelayMinInput = document.getElementById('jm-click-delay-min'); const clickDelayMaxInput = document.getElementById('jm-click-delay-max'); const saveBtn = document.getElementById('jm-save-config'); const resetBtn = document.getElementById('jm-reset-config'); const statusDiv = document.getElementById('jm-status'); function updateUIFromConfig() { autoMuteInput.checked = config.autoMute; autoJumpInput.checked = config.autoJumpUnfinished; autoSignInput.checked = config.autoSign; continuousExamInput.checked = config.continuousExamMode; logLevelSelect.value = config.logLevel; clickDelayMinInput.value = config.clickDelayMin; clickDelayMaxInput.value = config.clickDelayMax; currentLogLevel = logLevels[config.logLevel]; showScrollingLog(`配置已加载 - ${config.continuousExamMode ? '连续考试模式开启' : '常规模式'}`, 'INFO'); } // 监听复选框变化,立即生效 autoMuteInput.addEventListener('change', function () { config.autoMute = this.checked; saveConfig(); if (this.checked) { // 开启静音 handleVideos(); startSilenceGuard(); showScrollingLog('自动静音已开启', 'INFO'); } else { // 关闭静音 stopSilenceGuard(); // 恢复视频音量 document.querySelectorAll('video').forEach(video => { if (video.hasAttribute('data-jm-listening')) { video.muted = false; video.volume = 1; // 恢复到默认音量 } }); showScrollingLog('自动静音已关闭', 'INFO'); } }); autoJumpInput.addEventListener('change', function () { config.autoJumpUnfinished = this.checked; saveConfig(); if (this.checked) { // 如果在视频页面,立即执行跳转 if (location.href.includes('/MedicalIndex/videoPlay')) { setTimeout(() => { autoJumpToUnfinishedVideo(); }, 100); } showScrollingLog('自动跳转未学已开启', 'INFO'); } else { showScrollingLog('自动跳转未学已关闭', 'INFO'); } }); autoSignInput.addEventListener('change', function () { config.autoSign = this.checked; saveConfig(); if (this.checked) { startSignPolling(); showScrollingLog('自动签到已开启', 'INFO'); } else { stopSignPolling(); showScrollingLog('自动签到已关闭', 'INFO'); } }); continuousExamInput.addEventListener('change', function () { config.continuousExamMode = this.checked; saveConfig(); showScrollingLog(`连续考试模式已${this.checked ? '开启' : '关闭'}`, 'INFO'); // 如果开启且在视频页面,立即执行 if (this.checked && location.href.includes('/MedicalIndex/videoPlay')) { showScrollingLog('在视频页面开启连续考试模式,立即启动流程...', 'INFO'); setTimeout(() => { startContinuousExam(); // 调用连续考试函数 }, 100); } }); // 监听日志等级变化,立即生效 logLevelSelect.addEventListener('change', function () { config.logLevel = this.value; currentLogLevel = logLevels[this.value]; saveConfig(); showScrollingLog(`日志等级已更改为: ${this.value}`, 'INFO'); }); saveBtn.addEventListener('click', () => { const oldConfig = { ...config }; // 保存旧配置用于比较 config.logLevel = logLevelSelect.value; currentLogLevel = logLevels[config.logLevel]; // 更新当前日志等级 config.clickDelayMin = parseInt(clickDelayMinInput.value) || DEFAULT_CONFIG.clickDelayMin; config.clickDelayMax = parseInt(clickDelayMaxInput.value) || DEFAULT_CONFIG.clickDelayMax; saveConfig(); updateUIFromConfig(); showScrollingLog('配置已保存并应用', 'INFO'); }); resetBtn.addEventListener('click', () => { if (confirm('确定要重置所有配置为默认值吗?')) { config = { ...DEFAULT_CONFIG }; currentLogLevel = logLevels[config.logLevel]; // 更新当前日志等级 saveConfig(); updateUIFromConfig(); showScrollingLog('配置已重置为默认值', 'INFO'); // 重新初始化所有功能 if (config.autoMute) { handleVideos(); startSilenceGuard(); } else { stopSilenceGuard(); } if (config.autoSign) { startSignPolling(); } else { stopSignPolling(); } } }); // 初始加载配置 updateUIFromConfig(); // --- 拖拽功能 --- let isDragging = false; let offsetX, offsetY; panel.addEventListener('mousedown', (e) => { if (e.target === panel || panel.contains(e.target)) { if (e.target.closest('input, select, button')) return; // 避免在控件上拖拽 isDragging = true; offsetX = e.clientX - panel.getBoundingClientRect().left; offsetY = e.clientY - panel.getBoundingClientRect().top; panel.style.cursor = 'grabbing'; e.preventDefault(); } }); document.addEventListener('mousemove', (e) => { if (isDragging) { const x = e.clientX - offsetX; const y = e.clientY - offsetY; const maxX = window.innerWidth - panel.offsetWidth; const maxY = window.innerHeight - panel.offsetHeight; panel.style.left = `${Math.max(0, Math.min(x, maxX))}px`; panel.style.top = `${Math.max(0, Math.min(y, maxY))}px`; } }); document.addEventListener('mouseup', () => { if (isDragging) { isDragging = false; panel.style.cursor = 'move'; } }); // 标记面板已注入 isPanelInjected = true; } // ========== 连续考试核心逻辑 ========== async function startContinuousExam() { if (!config.continuousExamMode) { log('DEBUG', '连续考试模式未开启,不执行。'); return; } log('INFO', '开始连续考试流程'); showScrollingLog('开始连续考试流程', 'INFO'); // 查找所有视频项 const items = document.querySelectorAll('.directoryList .item'); if (items.length === 0) { log('WARN', '未找到视频列表项。'); showScrollingLog('未找到视频列表项', 'WARN'); return; } for (const item of items) { const statusEl = item.querySelector('.status'); if (!statusEl) continue; const text = statusEl.textContent.trim(); log('DEBUG', `检查视频项状态: "${text}"`); if (text === '待考核') { const videoTitleEl = item.querySelector('.title'); const videoTitle = videoTitleEl ? videoTitleEl.textContent.trim() : '未知视频'; // 检查本地记录,避免重复考核 if (hasCompletedExam(videoTitle)) { log('INFO', `视频 "${videoTitle}" 已在本地记录中,跳过。`); showScrollingLog(`视频 "${videoTitle}" 已完成,跳过`, 'INFO'); continue; } log('INFO', `即将进入考试:${videoTitle}`); showScrollingLog(`即将进入考试:${videoTitle}`, 'INFO'); await safeClickElement(item); // 等待页面跳转和加载 await new Promise(resolve => setTimeout(resolve, 5000)); // 查找并点击"开始考核"按钮 let startExamBtn = null; const maxRetries = 10; // 最多重试10次 (硬编码) let attempts = 0; while (attempts < maxRetries) { // **优化查找逻辑:尝试在更具体的上下文中查找** // 1. 尝试在页面主要内容区域查找 (例如 .content, .videoPlay, .main) // 这些类名是猜测的,您可能需要根据实际页面结构调整 const mainContainers = ['.content', '.videoPlay', '.main', '.video-content', '.play-container']; let foundInContainer = false; for (const containerSelector of mainContainers) { const container = document.querySelector(containerSelector); if (container) { log('DEBUG', `在容器 ${containerSelector} 中查找...`); // 在容器内部查找 const allAssessDivsInContainer = container.querySelectorAll('[class*="Assess"]'); log('DEBUG', `在 ${containerSelector} 中找到 ${allAssessDivsInContainer.length} 个 [class*="Assess"] 元素。`); for (const div of allAssessDivsInContainer) { if (div.innerText.trim() === '开始考核' && isVisible(div)) { startExamBtn = div; log('DEBUG', `在容器 ${containerSelector} 中找到可见的"开始考核"按钮:`, startExamBtn); foundInContainer = true; break; } } if (foundInContainer) break; // 如果在某个容器找到,就跳出循环 } } // 2. 如果在容器中没找到,再尝试全局查找 if (!foundInContainer) { log('DEBUG', `在指定容器中未找到,尝试全局查找...`); const allAssessDivs = document.querySelectorAll('[class*="Assess"]'); log('DEBUG', `全局找到 ${allAssessDivs.length} 个 [class*="Assess"] 元素。`); for (const div of allAssessDivs) { // **关键改进:增加可见性检查** if (div.innerText.trim() === '开始考核' && isVisible(div)) { startExamBtn = div; log('DEBUG', `全局找到可见的"开始考核"按钮:`, startExamBtn); break; } } } if (startExamBtn) { log('INFO', '找到"开始考核"按钮,准备点击...'); showScrollingLog('找到"开始考核"按钮,准备点击...', 'INFO'); await safeClickElement(startExamBtn); log('INFO', '"开始考核"按钮已点击,等待二维码出现。'); showScrollingLog('"开始考核"按钮已点击,等待二维码出现。', 'INFO'); // 点击后,脚本任务完成,用户手动扫码 return; // 退出循环,等待用户操作 } else { log('DEBUG', `未找到可见的"开始考核"按钮,等待后重试 (${attempts + 1}/${maxRetries})...`); await new Promise(resolve => setTimeout(resolve, 1000)); // 等待1秒再试 attempts++; } } if (!startExamBtn) { log('ERROR', `在视频 "${videoTitle}" 页面尝试 ${maxRetries} 次后仍未找到可见的"开始考核"按钮。`); showScrollingLog(`未找到"开始考核"按钮,跳过 "${videoTitle}"`, 'ERROR'); // 如果点击了视频但没找到按钮,可能需要回到列表继续下一个 window.history.back(); // 尝试返回列表页 await new Promise(resolve => setTimeout(resolve, 2000)); // 等待页面加载 } } } log('INFO', '连续考试流程结束,未找到更多"待考核"视频。'); showScrollingLog('连续考试流程结束', 'INFO'); } // ========== 视频模块 ========== function initVideoModule() { if (currentMode === 'video') return; currentMode = 'video'; log('INFO', '视频模块已启动'); showScrollingLog('视频模块已启动', 'INFO'); // 初始化视频模块时根据配置启动相应功能 if (config.autoMute) { handleVideos(); startSilenceGuard(); } if (config.autoSign) { startSignPolling(); } // 监听DOM变化,处理新出现的视频元素 if (!videoObserver) { videoObserver = new MutationObserver(() => { if (config.autoMute) handleVideos(); if (config.autoSign) trySign(); }); videoObserver.observe(document.body, { childList: true, subtree: true }); } // 在视频列表页面,如果开启了连续考试模式,启动连续考试 if (config.continuousExamMode) { log('INFO', '页面加载时连续考试模式已开启,启动流程...'); showScrollingLog('连续考试模式已开启,启动流程...', 'INFO'); startContinuousExam(); } // 如果开启了自动跳转未学,立即执行一次 if (config.autoJumpUnfinished) { setTimeout(() => { autoJumpToUnfinishedVideo(); }, 800); } } // ========== 考试模块 ========== function initExamModule() { if (currentMode === 'exam' || !config.autoAnswer) return; // config.autoAnswer 现在始终为 true currentMode = 'exam'; log('INFO', '考试模块已启动'); showScrollingLog('考试模块已启动', 'INFO'); let hasSubmitted = false; let isAnswering = false; let retryAttempts = 0; const MAX_RETRY_ATTEMPTS = 10; // 硬编码为 10 const CACHE_KEY = 'jm_smart_answers_v583'; function getQuestions() { return Array.from(document.querySelectorAll('.itemTopic')).filter(el => { const hasOptions = el.querySelector('input[type="radio"].el-radio__original'); const hasTitleOrStem = el.querySelector('.title') || el.querySelector('.stem') || el.innerText.trim(); return hasOptions && hasTitleOrStem; }); } function getOptions(questionEl) { return Array.from(questionEl.querySelectorAll('input[type="radio"].el-radio__original')); } function getQuestionHash(questionEl) { let stem = questionEl.querySelector('.title')?.innerText || questionEl.querySelector('.stem')?.innerText || (questionEl.innerText.match(/^(.{0,300})/)?.[0] || ''); stem = stem .replace(/^\s*\d+[、.\s]?\s*/, '') .replace(/((单选题|多选题|判断题))/g, '') .replace(/【判断题】/g, '') .trim(); if (!stem) return null; const clean = stem .replace(/[^\u4e00-\u9fa5a-zA-Z0-9\uFF01-\uFF5E!?。,;:]/g, '') .substring(0, 250); if (!clean) return null; let hash = 0; for (let i = 0; i < clean.length; i++) { const char = clean.charCodeAt(i); hash = ((hash << 5) - hash) + char; hash |= 0; } return 'q' + Math.abs(hash).toString(36); } function loadCache() { try { return JSON.parse(localStorage.getItem(CACHE_KEY) || '{}'); } catch (e) { localStorage.removeItem(CACHE_KEY); return {}; } } function saveRecord(hash, record) { const cache = loadCache(); cache[hash] = record; localStorage.setItem(CACHE_KEY, JSON.stringify(cache)); } function learnFromResultDialog() { const questions = getQuestions(); if (questions.length === 0) { log('WARN', '未找到题目,跳过学习'); showScrollingLog('未找到题目,跳过学习', 'WARN'); return; } const errorNumbers = [...document.querySelectorAll('.sheetBox.error')].map(el => parseInt(el.innerText.trim(), 10)); const successNumbers = [...document.querySelectorAll('.sheetBox.success')].map(el => parseInt(el.innerText.trim(), 10)); log('INFO', `正确: [${successNumbers.join(', ')}] 错误: [${errorNumbers.join(', ')}]`); showScrollingLog(`答题结果: 正确${successNumbers.length}, 错误${errorNumbers.length}`, 'INFO'); for (const num of successNumbers) { const idx = num - 1; if (idx >= 0 && idx < questions.length) { const q = questions[idx]; const hash = getQuestionHash(q); const selected = q.querySelector('input.el-radio__original:checked'); if (hash && selected) { const rec = loadCache()[hash] || { tried: [] }; rec.correct = selected.value; rec.tried = [...new Set([...rec.tried, selected.value])]; saveRecord(hash, rec); } } } for (const num of errorNumbers) { const idx = num - 1; if (idx >= 0 && idx < questions.length) { const q = questions[idx]; const hash = getQuestionHash(q); const selected = q.querySelector('input.el-radio__original:checked'); if (hash && selected?.value) { const rec = loadCache()[hash] || { tried: [] }; if (!rec.tried.includes(selected.value)) { rec.tried.push(selected.value); delete rec.correct; saveRecord(hash, rec); } } } } } function clickReAnswer() { const reBtn = [...document.querySelectorAll('.el-dialog .el-button--primary')] .find(btn => btn.innerText.trim() === '重新答题'); if (reBtn) { safeClickElement(reBtn); log('INFO', '点击【重新答题】'); showScrollingLog('点击【重新答题】', 'INFO'); return true; } return false; } function waitForResultDialog() { log('INFO', '开始轮询结果弹窗...'); showScrollingLog('等待考核结果...', 'INFO'); let attempts = 0; const maxAttempts = 20; const check = setInterval(() => { attempts++; const allHeaders = document.querySelectorAll('.el-dialog__header'); const targetHeader = Array.from(allHeaders).find(header => header.innerText.includes('考核未通过') ); if (targetHeader) { clearInterval(check); log('INFO', '成功检测到考核未通过弹窗!'); showScrollingLog('考核未通过,学习答案中...', 'INFO'); learnFromResultDialog(); setTimeout(() => { if (clickReAnswer()) { hasSubmitted = false; isAnswering = false; retryAttempts++; if (retryAttempts <= MAX_RETRY_ATTEMPTS) { log('INFO', `重答第 ${retryAttempts} 次`); showScrollingLog(`重答第 ${retryAttempts} 次`, 'INFO'); waitForAllQuestionsLoaded().then(autoAnswer); } else { log('ERROR', `重答次数已达到上限 ${MAX_RETRY_ATTEMPTS} 次,停止自动答题。`); showScrollingLog(`重答次数已达上限,停止自动答题`, 'ERROR'); shouldStopAutoAnswer = true; } } }, 800); return; } if (attempts >= maxAttempts) { clearInterval(check); log('WARN', '轮询超时:未检测到结果弹窗'); showScrollingLog('未检测到结果弹窗', 'WARN'); // 如果是连续考试模式,且超时,可能需要通知上层 } }, 500); } async function waitForAllQuestionsLoaded() { return new Promise((resolve) => { const MAX_INITIAL_WAIT = 10000; const MAX_TOTAL_TIME = 12000; const startTime = Date.now(); log('INFO', '等待题目初始化(AJAX加载中)...'); showScrollingLog('等待题目加载...', 'INFO'); function waitForInitialQuestion() { const firstQuestion = document.querySelector('.itemTopic'); if (firstQuestion) { log('INFO', '首题已加载,开始渐进滚动...'); showScrollingLog('题目加载中...', 'INFO'); startProgressiveScroll(); return; } if (Date.now() - startTime > MAX_INITIAL_WAIT) { log('WARN', '首题超时,仍尝试滚动'); showScrollingLog('题目加载超时,继续...', 'WARN'); startProgressiveScroll(); return; } setTimeout(waitForInitialQuestion, 300); } let currentCount = 0; let stableRounds = 0; const REQUIRED_STABLE_ROUNDS = 3; function startProgressiveScroll() { const checkAndScroll = () => { if (Date.now() - startTime > MAX_TOTAL_TIME) { log('WARN', '总加载超时'); showScrollingLog('题目加载超时', 'WARN'); resolve(); return; } const questions = document.querySelectorAll('.itemTopic'); const newCount = questions.length; if (newCount === currentCount) { stableRounds++; if (stableRounds >= REQUIRED_STABLE_ROUNDS) { log('INFO', `题目加载完成!共 ${newCount} 道`); showScrollingLog(`题目加载完成!共 ${newCount} 道`, 'INFO'); resolve(); return; } } else { stableRounds = 0; currentCount = newCount; log('DEBUG', `当前题目数: ${newCount}`); } const scrollTop = window.scrollY || document.documentElement.scrollTop; const clientHeight = window.innerHeight || document.documentElement.clientHeight; const scrollHeight = Math.max( document.body.scrollHeight, document.documentElement.scrollHeight ); if (scrollTop < scrollHeight - clientHeight - 10) { window.scrollTo(0, scrollTop + clientHeight); } setTimeout(checkAndScroll, Math.random() * 400 + 600); }; checkAndScroll(); } waitForInitialQuestion(); }); } async function autoAnswer() { if (!config.autoAnswer || shouldStopAutoAnswer) { // config.autoAnswer 现在始终为 true log('INFO', '自动答题已禁用或收到停止信号,停止答题。'); showScrollingLog('自动答题已停止', 'INFO'); isAnswering = false; return; } if (isAnswering || hasSubmitted) return; isAnswering = true; showScrollingLog('开始自动答题...', 'INFO'); const questions = getQuestions(); if (questions.length === 0) { isAnswering = false; showScrollingLog('未找到题目', 'WARN'); return; } log('INFO', `共检测到 ${questions.length} 道题,开始逐题作答...`); for (let i = 0; i < questions.length; i++) { if (!config.autoAnswer || shouldStopAutoAnswer) { // config.autoAnswer 现在始终为 true log('INFO', '自动答题已禁用或收到停止信号,停止答题。'); showScrollingLog('自动答题已停止', 'INFO'); isAnswering = false; return; } const q = questions[i]; const hash = getQuestionHash(q); const options = getOptions(q); if (options.length === 0) continue; const cache = loadCache(); const record = hash ? (cache[hash] || { tried: [] }) : { tried: [] }; let targetOption = null; if (record.correct !== undefined) { targetOption = options.find(opt => opt.value === record.correct); } if (!targetOption) { targetOption = options.find(opt => !record.tried.includes(opt.value)); } if (!targetOption) { targetOption = options[0]; // fallback } log('DEBUG', `正在作答第 ${i + 1} 题...`); safeClickRadio(targetOption); const delay = Math.random() * (config.clickDelayMax - config.clickDelayMin) + config.clickDelayMin; await new Promise(r => setTimeout(r, delay)); } if (!config.autoAnswer || shouldStopAutoAnswer) { // config.autoAnswer 现在始终为 true log('INFO', '自动答题已禁用或收到停止信号,跳过提交。'); showScrollingLog('自动答题已停止,跳过提交', 'INFO'); isAnswering = false; return; } setTimeout(() => { if (hasSubmitted) return; const submitBtn = document.querySelector('.submitBtn .btns'); if (submitBtn) { hasSubmitted = true; log('INFO', '点击提交按钮'); showScrollingLog('点击提交按钮', 'INFO'); safeClickElement(submitBtn); shouldStopAutoAnswer = false; // 提交后重置 setTimeout(waitForResultDialog, 1500); } isAnswering = false; }, 1000); } // 只有在考试页面时才注入UI面板 if (location.href.includes('/MedicalIndex/medicalAssess')) { injectUIPanel(); } if (config.autoAnswer) { // config.autoAnswer 现在始终为 true waitForAllQuestionsLoaded().then(() => { if (getQuestions().length > 0 && !hasSubmitted && config.autoAnswer && !shouldStopAutoAnswer) { autoAnswer(); } }); } else { log('INFO', '自动答题未开启,考试页面不自动答题。'); } const examObserver = new MutationObserver(() => { if (getQuestions().length > 0 && !hasSubmitted && !isAnswering && config.autoAnswer && !shouldStopAutoAnswer) { // config.autoAnswer 现在始终为 true examObserver.disconnect(); waitForAllQuestionsLoaded().then(autoAnswer); } }); examObserver.observe(document.body, { childList: true, subtree: true }); } // ========== 页面类型检测与初始化 ========== function checkCurrentPage() { const url = location.href; const isExam = url.includes('/MedicalIndex/medicalAssess'); const isVideo = url.includes('/MedicalIndex/videoPlay'); // 检查是否为支持的页面 const isSupportedPage = isExam || isVideo; if (url === lastUrl && currentMode) { return; } lastUrl = url; currentMode = null; // 重置模式,允许重新初始化 // 只有在支持的页面才初始化 if (isSupportedPage) { if (!isInitialized) { loadConfig(); isInitialized = true; } // 在支持的页面上才注入UI面板 if (!isPanelInjected) { injectUIPanel(); } if (isExam) { initExamModule(); } else if (isVideo) { initVideoModule(); } } else { // 在非支持页面,确保UI面板被移除 if (isPanelInjected) { const panel = document.getElementById('jm-config-panel'); if (panel) { panel.remove(); isPanelInjected = false; } } // 停止视频相关功能 if (videoObserver) { videoObserver.disconnect(); videoObserver = null; } stopSilenceGuard(); stopSignPolling(); } } checkCurrentPage(); setInterval(checkCurrentPage, 1000); document.addEventListener('visibilitychange', () => { if (!document.hidden) { setTimeout(checkCurrentPage, 300); } }); })();