// ==UserScript== // @name 超星学习通小助手 // @namespace xyz_xyz-%e8%b6%85%e6%98%9f%e5%ad%a6%e4%b9%a0%e9%80%9a%e5%b0%8f%e5%8a%a9%e6%89%8b // @version 1.0.2 // @description 主要用于学习通章节视频刷课以及章节测验,为学生减负,测验仅支持单选,多选,判断。脚本完全免费,可能有BUG,勿喷。有问题请到公众号网络喵反馈。 // @author xyz_xyz // @grant GM_setValue // @grant GM_getValue // @grant GM_xmlhttpRequest // @connect 8.138.189.217 // @require https://html2canvas.hertzen.com/dist/html2canvas.min.js // @match https://mooc1.chaoxing.com/mycourse/studentstudy* // @original-license 非作者同意禁止引用和二次开发 // @icon  // ==/UserScript== (function () { 'use strict'; const BACKEND_URL = 'http://8.138.189.217:31415'; let cardsIframe = document.querySelector('iframe#iframe'); let cardsIframeDocument; async function init() { try { await waitForElement('iframe#iframe', 5000); cardsIframe = document.querySelector('iframe#iframe'); cardsIframeDocument = cardsIframe.contentDocument || cardsIframe.contentWindow.document; if (!cardsIframeDocument) { log("无法获取iframe文档,1秒后重试..."); setTimeout(init, 1000); return; } await xxtCards(); } catch (error) { log(`初始化失败: ${error.message}`); setTimeout(init, 3000); } } async function xxtCards() { try { cardsIframe = document.querySelector('iframe#iframe'); if (!cardsIframe) { log("未找到主iframe,退出流程"); return; } cardsIframeDocument = cardsIframe.contentDocument || cardsIframe.contentWindow.document; if (!cardsIframeDocument) { log("无法获取主iframe文档,退出流程"); return; } await initUI(); await new Promise(resolve => { const checkIframes = () => { const iframes = cardsIframeDocument.querySelectorAll('iframe'); if (iframes.length > 0) { resolve(); } else { setTimeout(checkIframes, 500); } }; checkIframes(); }); const cardIframes = cardsIframeDocument.querySelectorAll('iframe'); if (!cardIframes || cardIframes.length < 1) { log("未找到任务点iframe,准备换页"); await handlePageNavigation(); return; } for (const iframe of cardIframes) { try { const cardIframedoc = await getIframeDocument(iframe); if (!cardIframedoc) continue; if (iframe.src.includes('work')) { log("发现作业iframe,开始处理"); await xxtWork(cardIframedoc); } else if (iframe.src.includes('video')) { log("发现视频iframe,开始处理"); await xxtVideo(cardIframedoc); } else { log("暂不支持该类型的任务点"); } } catch (error) { log(`处理iframe时出错: ${error.message},继续处理下一个`); } } await handlePageNavigation(); } catch (error) { log(`主流程出错: ${error.message}`); } } async function handlePageNavigation() { try { log("准备进行页面导航"); const nextChapter = document.querySelector(`a.nextChapter`); if (nextChapter) { log("等待5秒后执行换页操作"); await delay(1000); nextChapter.click(); log("换页操作已执行"); } else { log("未找到导航元素,无法换页"); } } catch (error) { log(`换页处理出错: ${error.message}`); } await delay(3000); init(); } async function initUI() { try { log('初始化脚本,寻找ans-cc区块...'); const targetDiv = await waitForElement('div.ans-cc', 10000, cardsIframeDocument); if (targetDiv) { log('找到ans-cc区块,创建配置UI...'); createAndInsertUI(targetDiv); loadSavedSettings(); return true; } else { log('超时未找到ans-cc区块'); return false; } } catch (error) { log(`UI初始化出错: ${error.message}`); return false; } } function createAndInsertUI(container) { const configPanel = document.createElement('div'); configPanel.style.cssText = ` padding: 15px; background: #f5f5f5; border-radius: 8px; margin-bottom: 15px; border: 1px solid #e0e0e0; `; configPanel.innerHTML = `
公众号网络喵

说明

关注公众号网络喵,获取ai免费题库token。

配置设置

操作日志

`; container.insertBefore(configPanel, container.firstChild); bindEvents(); log('配置UI创建完成'); } function bindEvents() { cardsIframeDocument.getElementById('saveBtn').addEventListener('click', saveSettings); cardsIframeDocument.getElementById('verifyBtn').addEventListener('click', verifyCurrentToken); } function loadSavedSettings() { log('加载保存的设置...'); const savedToken = GM_getValue('work_token', ''); const savedSpeed = GM_getValue('video_speed', '1'); cardsIframeDocument.getElementById('tokenInput').value = savedToken; cardsIframeDocument.getElementById('speedSelect').value = savedSpeed; log('设置加载完成'); } function saveSettings() { const token = cardsIframeDocument.getElementById('tokenInput').value; const speed = cardsIframeDocument.getElementById('speedSelect').value; GM_setValue('work_token', token); GM_setValue('video_speed', speed); showMessage('设置已保存', 'green'); log('设置已保存到本地存储'); } async function verifyCurrentToken() { try { const token = cardsIframeDocument.getElementById('tokenInput').value; if (!token) { showMessage('请输入Token', 'red'); log('验证失败:未输入Token'); return; } showMessage('验证中...', 'blue'); log('开始验证Token...'); const valid = await verifyToken(token); if (valid) { showMessage('Token验证成功', 'green'); saveSettings(); log('Token验证成功'); } else { showMessage('Token验证失败', 'red'); log('Token验证失败'); } } catch (error) { log(`Token验证出错: ${error.message}`); showMessage('验证过程出错', 'red'); } } function showMessage(text, color) { const messageEl = cardsIframeDocument.getElementById('message'); if (messageEl) { messageEl.textContent = text; messageEl.style.color = color; setTimeout(() => { messageEl.textContent = ''; }, 3000); } } function log(message) { if (!cardsIframeDocument) return; const logArea = cardsIframeDocument.getElementById('logArea'); if (!logArea) return; const now = new Date(); const timeString = `${now.getHours().toString().padStart(2, '0')}:${now.getMinutes().toString().padStart(2, '0')}:${now.getSeconds().toString().padStart(2, '0')}`; const logEntry = document.createElement('div'); logEntry.style.cssText = 'margin: 3px 0; border-bottom: 1px solid #f0f0f0; padding-bottom: 2px;'; logEntry.innerHTML = `[${timeString}] ${message}`; logArea.appendChild(logEntry); logArea.scrollTop = logArea.scrollHeight; } function verifyToken(token) { return new Promise((resolve) => { GM_xmlhttpRequest({ method: "POST", url: `${BACKEND_URL}/api/verify-token`, data: JSON.stringify({ token }), headers: { "Content-Type": "application/json" }, onload: (res) => { try { const data = JSON.parse(res.responseText); log(`Token验证响应: ${data.msg}`); resolve(data.code === 200); } catch (e) { log(`Token验证解析错误: ${e.message}`); resolve(false); } }, onerror: (err) => { log(`Token验证请求错误: ${err.message || '未知错误'}`); resolve(false); } }); }); } function xxtVideo(vedioDoc) { return new Promise((resolve) => { (async () => { try { log("开始处理视频..."); const video = await waitForElement('video', 10000, vedioDoc); if (!video) { log("未找到视频元素,结束视频处理"); resolve(); return; } if (video.ended) { log("视频已播放完成"); resolve(); return; } log(`找到视频元素,开始自动播放处理...`); let playbackRate = parseFloat(GM_getValue('video_speed', '1.0')); video.playbackRate = playbackRate; video.autoplay = true; video.muted = true; async function forcePlay() { if (video.paused && !video.ended) { try { await video.play(); log(`视频已恢复播放,播放速度: ${playbackRate}x`); } catch (error) { log(`视频播放失败: ${error.message},1秒后重试`); if (!video.ended) { await delay(1000); await forcePlay(); } } } } await forcePlay(); video.addEventListener('timeupdate', () => { const newSpeed = parseFloat(GM_getValue('video_speed', '1.0')); if (newSpeed !== playbackRate) { playbackRate = newSpeed; video.playbackRate = playbackRate; log(`视频速度已调整为: ${playbackRate}x`); } const currentTime = video.currentTime; const duration = video.duration; const progressValue = (currentTime / duration) * 100; console.log(`${playbackRate}X,${progressValue.toFixed(1)}%`); }); video.addEventListener('pause', async () => { if (!video.ended) { log(`检测到视频被暂停,正在恢复播放...`); await forcePlay(); } }); await new Promise(resolveVideo => { const checkEnded = () => { if (video.ended) { log(`视频播放完成`); resolveVideo(); } else { setTimeout(checkEnded, 1000); } }; checkEnded(); }); resolve(); } catch (error) { log(`视频处理出错: ${error.message}`); resolve(); } })(); }); } async function xxtWork(Doc) { try { log("开始处理作业..."); const workIframe = await waitForElement('iframe#frame_content', 10000, Doc); if (!workIframe) { log("未找到作业内容iframe"); return; } const workIframedoc = await getIframeDocument(workIframe); if (!workIframedoc) { log("无法获取作业iframe文档"); return; } const secretStyleInHead = workIframedoc.head.querySelector('#cxSecretStyle'); if (secretStyleInHead) { const cssContent = secretStyleInHead.textContent; let style = document.head.querySelector('#cxSecretStyle'); if (!style) { style = document.createElement('style'); style.id = 'cxSecretStyle'; document.head.appendChild(style); } style.textContent = cssContent; log("已处理字体样式"); } await initWork(workIframedoc); } catch (error) { log(`作业处理出错: ${error.message}`); } } async function initWork(workDoc) { function captureQuestion(questionEl) { return new Promise((resolve) => { const clone = questionEl.cloneNode(true); clone.style.position = "fixed"; clone.style.top = "0"; clone.style.left = "-9999px"; workDoc.body.appendChild(clone); html2canvas(clone, { useCORS: true, allowTaint: false }).then(canvas => { const base64 = canvas.toDataURL("image/png").replace("data:image/png;base64,", ""); workDoc.body.removeChild(clone); resolve(base64); }).catch(error => { log(`截图失败: ${error.message}`); workDoc.body.removeChild(clone); resolve(null); }); }); } function getAnswer(token, questionInfo) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: "POST", url: `${BACKEND_URL}/api/get-answer`, data: JSON.stringify({ token, question_type: questionInfo.type, question_image: questionInfo.image }), headers: { "Content-Type": "application/json" }, onload: (res) => { try { const data = JSON.parse(res.responseText); if (data.code === 200) { resolve(data.answer); } else { reject(new Error(data.msg)); } } catch (error) { reject(new Error(`解析答案失败: ${error.message}`)); } }, onerror: () => reject(new Error("后端请求失败")) }); }); } function selectAnswer(questionEl, questionType, answer) { const questionId = questionEl.getAttribute("data"); if (!questionId) { throw new Error("未找到题目ID"); } switch (questionType) { case "单选题": { const answerInput = questionEl.querySelector(`input#answer${questionId}`); if (answerInput) { answerInput.value = answer; log(`已选择单选题答案: ${answer}`); } else { throw new Error(`未找到单选题输入框: answer${questionId}`); } break; } case "多选题": { const answerInput = questionEl.querySelector(`input#answer${questionId}`); if (answerInput) { answerInput.value = answer.join(""); log(`已选择多选题答案: ${answer.join("")}`); } else { throw new Error(`未找到多选题输入框: answer${questionId}`); } break; } case "判断题": { const answerInput = questionEl.querySelector(`input#answer${questionId}`); if (answerInput) { answerInput.value = answer; log(`已选择判断题答案: ${answer}`); } else { throw new Error(`未找到判断题输入框: answer${questionId}`); } break; } default: throw new Error(`不支持的题目类型: ${questionType}`); } } async function processAllQuestions(token) { const DividerBox = workDoc.querySelector("div.DividerBox"); if (DividerBox) { log("题目已完成,无需重复处理"); return; } const questions = await waitForElement('div.singleQuesId', 10000, workDoc, true); if (!questions || questions.length === 0) { log("未找到题目"); return; } log(`发现 ${questions.length} 道题目,开始逐一处理`); for (let i = 0; i < questions.length; i++) { const questionEl = questions[i]; const questionIndex = i + 1; try { log(`------ 开始处理第 ${questionIndex}/${questions.length} 题 ------`); showMessage(`处理第 ${questionIndex}/${questions.length} 题`, `blue`); const questionId = questionEl.getAttribute("data"); if (!questionId) { log(`第 ${questionIndex} 题未找到ID,跳过`); continue; } const answerTypeInput = questionEl.querySelector(`span.newZy_TItle`); if (!answerTypeInput) { log(`第 ${questionIndex} 题未找到题型,跳过`); continue; } let typeText = answerTypeInput.textContent.substring(1, answerTypeInput.textContent.length - 1); if (!['单选题', '多选题', '判断题'].includes(typeText)) { log(`第 ${questionIndex} 题为 ${typeText},暂不支持,跳过`); continue; } const answerInput = questionEl.querySelector(`input#answer${questionId}`); if (answerInput && answerInput.getAttribute("value")) { log(`第 ${questionIndex} 题已作答,跳过`); continue; } log(`正在截图第 ${questionIndex} 题`); const base64Image = await captureQuestion(questionEl); if (!base64Image) { log(`第 ${questionIndex} 题截图失败,跳过`); continue; } const questionInfo = { type: typeText, image: base64Image }; log(`题目类型: ${questionInfo.type}`); log(`正在请求第 ${questionIndex} 题答案`); const answer = await getAnswer(token, questionInfo); if (!answer) { log(`未获取到第 ${questionIndex} 题答案,跳过`); continue; } log(`获取到答案: ${JSON.stringify(answer)}`); selectAnswer(questionEl, questionInfo.type, answer); log(`第 ${questionIndex} 题答题完成`); await delay(1000); } catch (err) { log(`第 ${questionIndex} 题处理失败: ${err.message},继续处理下一题`); } } await saveAndSubmitAnswers(workDoc); } async function saveAndSubmitAnswers(workDoc) { try { log("准备保存答案"); const tempsave = await waitForElement(`span#tempsave`, 5000, workDoc); if (tempsave) { tempsave.click(); log("已点击保存按钮"); let btnSubmit = await waitForElement(`.btnSubmit`, 5000, workDoc); let popok = await waitForElement(`#popok`, 5000); if (popok && btnSubmit) { log("已点击确认按钮"); btnSubmit.click(); await delay(2000); popok.click(); } else { log("未找到确认按钮"); } await delay(1000); } else { log("未找到保存按钮"); } } catch (error) { log(`保存答案出错: ${error.message}`); } } const token = GM_getValue('work_token', ''); if (!token) { log("未设置Token,无法处理作业"); showMessage("请先设置并验证Token", "red"); return; } await delay(2000); await processAllQuestions(token); } function waitForElement(selector, timeout = 5000, doc = document, returnAll = false) { return new Promise((resolve) => { const start = Date.now(); const checkInterval = 200; const checkElement = () => { const elements = returnAll ? doc.querySelectorAll(selector) : doc.querySelector(selector); if ((!returnAll && elements) || (returnAll && elements.length > 0)) { resolve(elements); return; } if (Date.now() - start > timeout) { resolve(returnAll ? [] : null); return; } setTimeout(checkElement, checkInterval); }; checkElement(); }); } function getIframeDocument(iframe) { return new Promise((resolve) => { const checkDocument = () => { const doc = iframe.contentDocument || iframe.contentWindow.document; if (doc && doc.readyState === 'complete') { resolve(doc); } else { setTimeout(checkDocument, 200); } }; iframe.onload = checkDocument; checkDocument(); }); } function delay(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } setTimeout(init, 5000); })();