// ==UserScript== // @name 成都文理学院刷课助手|自动刷课|考试自动答题 // @version 2.0.5 // @description 成都文理学院刷课助手,🚀目前已支持平台:【数字化实习实训平台、公益课程、在线学堂、英华学堂】。😀目前已具有功能包括:视频自动播放、自动识别填充验证码、考试自动答题等功能。如有bug请留言。🐧QQ交流群:878643471 // @author iFulling // @match *://zxshixun.cdcas.com/* // @match *://gyxy.cdcas.com/* // @match *://mooc.cdcas.com/* // @match *://cdcas.rurenkj.com/* // @icon  // @grant GM_xmlhttpRequest // @license MIT // @namespace https://github.com/iFulling/cdcasSK // @connect 119.8.102.43 // @connect 119.8.102.43:5000 // @connect ark.cn-beijing.volces.com // @downloadURL none // ==/UserScript== /** /* @downloadURL https://update.greasyfork.org/scripts/512596/%E6%88%90%E9%83%BD%E6%96%87%E7%90%86%E5%AD%A6%E9%99%A2%E5%88%B7%E8%AF%BE%E5%8A%A9%E6%89%8B%EF%BC%88%E8%87%AA%E5%8A%A8%E5%A1%AB%E5%85%85%E9%AA%8C%E8%AF%81%E7%A0%81%EF%BC%89.user.js /* @updateURL https://update.greasyfork.org/scripts/512596/%E6%88%90%E9%83%BD%E6%96%87%E7%90%86%E5%AD%A6%E9%99%A2%E5%88%B7%E8%AF%BE%E5%8A%A9%E6%89%8B%EF%BC%88%E8%87%AA%E5%8A%A8%E5%A1%AB%E5%85%85%E9%AA%8C%E8%AF%81%E7%A0%81%EF%BC%89.meta.js */ let videoElement = null; let checkCaptchaTimer = null; let containerTextElement = null; let examTextElement = null; let layuiLayerContent = null; let links = null; let current = 0; let timerCnt = 0; let version = "2.0.5" let token = ""; let auth = ""; let examCurrent = 0; let examTimer = null; let startFlag = false; // 获取当前课程 function getCurrent() { links = $('a[target="_self"]'); links.each((index, item) => { if ($(item).hasClass("on")) { return current = index } }); } // 下一个视频 async function playNext() { clearInterval(checkCaptchaTimer); if (current === links.length - 1) { addText("最后一个已看完!") } else { addText("准备播放下一个视频...") await pause(3) links[current + 1].click(); } } // 输入验证码 async function inputCaptcha() { if (layuiLayerContent.length && layuiLayerContent.is(':visible')) { addText("验证码弹窗出现,准备填写验证码..."); await pause(2, 5) // 获取图片 let imgs = layuiLayerContent.find("img") let img = imgs[0].style.opacity === '0' ? imgs[1] : imgs[0] // 图片转base64 let canvas = document.createElement("canvas"); let ctx = canvas.getContext("2d"); canvas.width = img.width; canvas.height = img.height; ctx.drawImage(img, 0, 0, img.width, img.height); let code = canvas.toDataURL("image/png").split("base64,")[1]; // 调用接口,识别验证码 let ans = await getCode(code) // 获取input,填入验证码 let inputs = layuiLayerContent.find("input") let input = inputs[0].style.display === 'none' ? inputs[1] : inputs[0] $(input).mousedown() input.value = ans // 点击开始播放按钮 await pause(2, 5) const playButton = $('.layui-layer-btn0'); if (playButton.length) { playButton.click(); checkCaptchaTimer = setInterval(playVideo, 1000); addText("已自动点击开始播放按钮!"); } else { addText("未找到开始播放按钮,尝试刷新页面..."); location.reload(); // 刷新当前页面 } } } function getCode(code) { return new Promise((resolve, reject) => { const datas = { "ImageBase64": String(code), } GM_xmlhttpRequest({ method: "POST", url: "http://119.8.102.43:5000/get_captcha", data: JSON.stringify(datas), headers: { "Content-Type": "application/json", }, responseType: "json", onload: function (response) { if (response.status == 200) { let result = response.response["message"]; addText("识别结果:" + result); return resolve(result); } else { let result = JSON.parse(response.response)["message"]; addText(result); addText("识别失败,请勿开启代理,或联系管理员。🐧群:878643471"); } } }); }); } // 播放视频,同时检测验证码 async function playVideo() { timerCnt++; if (timerCnt % 5 === 0) { addText("等待加载,已加载:" + timerCnt + "秒") } if (timerCnt > 20) { addText("刷新页面") location.reload(); return } if (!videoElement) { if (links[current].title && current === links.length - 1) { addText("课程已看完,自动停止!") clearInterval(checkCaptchaTimer) }else if (links[current].title && /考试|章节作业|自学教材/.test(links[current].title)){ addText("没有视频,准备跳过...") clearInterval(checkCaptchaTimer) await playNext(); }else { getVideoElement(); } return } // 验证码弹窗 layuiLayerContent = $('.layui-layer-content'); if (layuiLayerContent.length > 0) { clearInterval(checkCaptchaTimer); await inputCaptcha() return; } // 检测视频是否加载成功且暂停 // if (!videoElement) return; if (videoElement.paused) { videoElement.play(); if (videoElement.readyState === 4) { const message = containerTextElement.text().includes("视频加载完成") ? "请将浏览器置于前台运行。(若学时会增加可忽略)" : "视频加载完成,准备播放"; addText(message); } } else { timerCnt = 0; } } // 获取视频元素 const getVideoElement = () => { videoElement = document.querySelector("video"); videoElement.muted = true; videoElement.playbackRate = 1.0; videoElement.volume = 0; videoElement.onended = async function () { await playNext(); }; } // 添加交互显示 const addContainer = () => { const mini = $("
刷课
助手
") const container = $('') container.addClass('popup'); const header = $("
") header.addClass('container-header') header.text("成都文理学院刷课助手 " + version) container.append(header) const btnGroup = $("
") const classTab = $("") const examTab = $("") const minimize = $("") btnGroup.append(classTab) btnGroup.append(examTab) btnGroup.append(minimize) header.append(btnGroup) classTab.on("click", () => { examTextElement.hide() containerTextElement.show() }) examTab.on("click", ()=>{ containerTextElement.hide() examTextElement.show() }) minimize.on("click", ()=>{ container.hide() mini.css("display", "flex") }) let moveFlag = false; mini.on("mouseup", ()=>{ if (moveFlag) return; container.show() mini.hide() }) // 添加移动事件 header.on("mousedown", function (event) { // 获取鼠标相对盒子的偏移量 let shiftX = event.clientX - header.offset().left; let shiftY = event.clientY - header.offset().top; // 当鼠标移动时 function onMouseMove(event) { container.css({ left: event.pageX - shiftX + 'px', top: event.pageY - shiftY + 'px' }) } // 鼠标提起来 function onMouseUp() { $(document).off('mousemove', onMouseMove); $(document).off('mouseup', onMouseUp); } $(document).on('mousemove', onMouseMove); $(document).on('mouseup', onMouseUp); }) mini.on("mousedown", function (event) { // 获取鼠标相对盒子的偏移量 let shiftX = event.clientX - mini.offset().left; let shiftY = event.clientY - mini.offset().top; // 当鼠标移动时 function onMouseMove(event) { moveFlag = true; mini.css({ left: event.pageX - shiftX + 'px', top: event.pageY - shiftY + 'px' }) } // 鼠标提起来 function onMouseUp() { moveFlag = false; $(document).off('mousemove', onMouseMove); $(document).off('mouseup', onMouseUp); } $(document).on('mousemove', onMouseMove); $(document).on('mouseup', onMouseUp); }) const hr = $("
") container.append(hr) $("body").append(container) $("body").append(mini) } const showClassOption = () => { containerTextElement = $("
") containerTextElement.addClass('container-text') $(".popup").append(containerTextElement) addText("

提示1

:如果开启了系统代理,要先关闭!") addText("

提示2

:本脚本仅支持PC端,如果不起作用,点油猴图标看是否有提示 \"Please enable developer mode...\",若有,点击查看 油猴插件不能使用") addText("

提示3

:安装过老版本的需要把老版本删除或者禁用。") addText("

提示4

:因不同浏览器的优化策略问题,如果发现学时没变,看视频时请将浏览器置于前台运行
") addText("启动成功...") } const showExamOption = () => { examTextElement = $("
") examTextElement.addClass('container-exam') $(".popup").append(examTextElement) examTextElement.append("

提示1

:如果开启了系统代理,要先关闭!
") examTextElement.append("

提示2

:本脚本仅支持PC端,如果不起作用,点油猴图标看是否有提示 \"Please enable developer mode...\",若有,点击查看 油猴插件不能使用
") examTextElement.append("

提示3

:安装过老版本的需要把老版本删除或者禁用。
") examTextElement.append("

提示4

:对接的是抖音豆包,因为是AI,所以不能保证完全正确,分数高低与作者无关,如果有所担心可在搜完后再自己手动搜一遍
") examTextElement.append("启动成功...

") let cookies = document.cookie.split("; "); let cookieObj = {}; cookies.forEach(cookie=>{ let part = cookie.split("=") cookieObj[part[0]] = part[1] }) token = cookieObj["_db_token"] || "" auth = cookieObj["_db_auth"] || "" let date = new Date(); date.setTime(date.getTime() + (365 * 24 * 60 * 60 * 1000)); examTextElement.append("搜题配置:点击链接 👉 视频教程 | 获取搜题接入点ID和API Key
") let tokenDiv = $("
") tokenDiv.append("接入点ID:") let tokenInput = $("") tokenInput.on("keyup", e=>{ token = e.target.value document.cookie = "_db_token="+token+"; domain=.cdcas.com; expires="+date.toUTCString()+"; path=/"; }) tokenDiv.append(tokenInput) let authDiv = $("
") authDiv.append("API Key: ") let authInput = $("") authInput.on("keyup", e=>{ auth = e.target.value document.cookie = "_db_auth="+auth+"; domain=.cdcas.com; expires="+date.toUTCString()+"; path=/"; }) authDiv.append(authInput) examTextElement.append(tokenDiv) examTextElement.append(authDiv) let startBtn = $("") let stopBtn = $("") let saveBtn = $("") examTextElement.append(saveBtn) examTextElement.append(startBtn) examTextElement.append(stopBtn) examCurrent = parseInt($(".topic-head.on").text()) - 1 examTextElement.append("") setExamStatus("已停止。搜题将从当前题开始"); if ($("#startArea").length == 1){ setExamStatus("等待答题..."); } if ($(".courseexam-list").find(".time").text().includes("已交卷")) { setExamStatus("已交卷,不可继续答题"); } saveBtn.on("click", ()=>{ document.cookie = "_db_token="+tokenInput[0].value+"; domain=.cdcas.com; expires="+date.toUTCString()+"; path=/"; document.cookie = "_db_auth="+authInput[0].value+"; domain=.cdcas.com; expires="+date.toUTCString()+"; path=/"; setExamStatus("保存成功!"); }) startBtn.on("click",async ()=>{ if (startFlag) return; startFlag = true; examCurrent = parseInt($(".topic-head.on").text()) - 1 let n = $(".courseexamcon-intro").find("ul").children("li").length; for (; examCurrent < n; examCurrent++) { if (!startFlag) break; let tab = $("#topic-tab-" + examCurrent) setExamStatus("进行中...正在搜索第 " + ( examCurrent + 1 ) + " 题") await pause(1) // 1 单选 2 多选 3 判断 let type = parseInt(tab.find(".courseexamcon-main").data("type")) if ([1, 2, 3].includes(type)){ let question = tab.find(".courseexamcon-main")[0].innerText.replaceAll("\n.\n", ".") let answer = await getAnswer(question) answer = answer.match(/[a-zA-Z]+/)[0]; setExamStatus("第 "+ (examCurrent + 1) +" 题答案:" + answer) switch (type) { case 1: case 3: tab.find("input[value='"+answer.toUpperCase()+"']").click() break; case 2: for (let item of answer) { tab.find("input[value='"+item.toUpperCase()+"']").click() } break; } }else{ setExamStatus("未添加该题型,跳过...") } await pause(3) let btn = tab.find("input[value='保存修改']") if (btn.css("display") == "none"){ tab.find("input[value='继续下一题']").click() }else{ btn.click() $(".courseexamcon-intro").find("ul").children("li")[examCurrent + 1].querySelector("a").click() } } startFlag = false setExamStatus("已停止。下次搜题将从当前题开始"); }) stopBtn.on("click", ()=>{ startFlag = false setExamStatus("已停止。下次搜题将从当前题开始") }) } const setExamStatus = text => { $("#examStatus").text("当前状态:"+text) } // 添加样式 const addStyle = () => { const style = $("") style.prop('type', 'text/css') style.html( ` .mini { position: fixed; top: 50px; left: 150px; width: 50px; height: 50px; display: none; z-index: 99999999999999999; background: #bb241d; color: white; border-radius: 50%; cursor: pointer; align-items: center; justify-content: center; box-shadow: rgba(0, 0, 0, 0.5) 0px 0px 8px 2px; } .popup { position: fixed; top: 50px; left: 150px; width: 540px; font: 14px Menlo, Monaco, Consolas, "Courier New", monospace; z-index: 9999999999999999999999; background-color: #fff; box-shadow: 0 0 5px 1px rgba(0, 0, 0, .3); padding: 10px; border-radius: 5px; } .popup h4{ display: inline-block; } .popup b{ color: red; } .popup a{ color: blue; } .container-header { height: 30px; cursor: move; line-height: 30px; display: flex; justify-content: space-between; } .container-header button{ margin-left: 10px; border: none; background: none; cursor: pointer; border-bottom: 1px solid; } .container-header button:hover{ color: red; } .container-text, .container-exam { margin-top: 10px; max-height: 200px; min-height: 30px; overflow: auto; } .container-exam { max-height: 300px; } .container-exam div{ margin: 10px 0; } .container-exam input{ border: 1px solid #000000; width: 400px; padding: 3px; padding-left: 5px; border-radius: 3px; } .container-exam button{ color: red; padding: 3px 10px; background: #ba251e; color: white; border-radius: 3px; cursor: pointer; margin: 10px 10px 10px 0; border: none; } .container-exam button:hover{ background: pink; } ` ) $('body').append(style); } // 添加交互文本 const addText = text => { containerTextElement.append(text + "
") containerTextElement.scrollTop(containerTextElement[0].scrollHeight) } // 暂停函数 function pause(start, end = undefined) { let delay = start; if (end) { delay = Math.floor(Math.random() * (end - start)) + start; addText(`等待 ${delay} 秒后继续...`); } return new Promise(resolve => { setTimeout(() => { resolve(); }, delay * 1000); // 将秒转换为毫秒 }); } const getAnswer = (question) => { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: "POST", url: "https://ark.cn-beijing.volces.com/api/v3/chat/completions", data: JSON.stringify({ "model": token, "messages": [ { "role": "user", "content": "只说选项号,不要说理由。\n" + question } ] }), headers: { "Authorization": "Bearer " + auth, "Content-Type": "application/json", }, responseType: "json", onload: function (response) { if (response.status == 401) { setExamStatus("作者关闭了搜题接口,开启时间等待更新..."); } else if (response.status == 200) { try { var answer = response.response["choices"][0].message.content; return resolve(answer); } catch (e) { setExamStatus("异常捕获:接口错误!"); } } else { setExamStatus("接口错误!"); } } }); }) } // 初始化程序 const init = async () => { addContainer() addStyle() showClassOption() showExamOption() } // 运行程序 (function () { 'use strict'; $(document).ready(async function () { await init() if (window.location.href.includes("/node")) { $(".classTabBtn").click() getCurrent() addText("初始化完成,可以解放双手了;为了更像人为点击,将会延时一段时间再播放
") await pause(5, 10) checkCaptchaTimer = setInterval(playVideo, 1000); }else if (window.location.href.includes("/exam")){ $(".examTabBtn").click() }else{ $(".classTabBtn").click() addText("请点进课程内容,进行学习...
") } }); })();