Greasy Fork

成都文理学院刷课助手|自动刷课|考试自动答题

成都文理学院刷课助手,(虽不止成文理,但仅在成文理做了测试)🚀目前已支持平台:【数字化实习实训平台、公益课程、在线学堂、英华学堂】。😀目前已具有功能包括:视频自动播放、自动识别填充验证码、考试自动答题等功能。如有bug请留言。🐧QQ交流群:878643471

目前为 2025-04-07 提交的版本。查看 最新版本

// ==UserScript==
// @name         成都文理学院刷课助手|自动刷课|考试自动答题
// @version      2.1.2
// @description  成都文理学院刷课助手,(虽不止成文理,但仅在成文理做了测试)🚀目前已支持平台:【数字化实习实训平台、公益课程、在线学堂、英华学堂】。😀目前已具有功能包括:视频自动播放、自动识别填充验证码、考试自动答题等功能。如有bug请留言。🐧QQ交流群:878643471
// @author       iFulling
// @match        *://zxshixun.cdcas.com/*
// @match        *://*.zjxkeji.com/*
// @match        *://mooc.cdcas.com/*
// @match        *://*.rurenkj.com/*
// @match        *://*/*
// @icon         
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_listValues
// @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
// ==/UserScript==

let videoElement = null;
let checkCaptchaTimer = null;
let containerTextElement = null;
let examTextElement = null;
let sponsorElement = null;
let statementElement = null;
let layuiLayerContent = null;
let links = null;
let current = 0;
let timerCnt = 0;
let version = "2.1.2"
let endpoint_id = "";
let apikey = "";
let examCurrent = 0;
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();  // 刷新当前页面
        }
    }
}

async function test(){
    let img = $("#codeImg")[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)
    console.log(ans)
}

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",
            timeout: 10000,
            ontimeout: async function (e) {
                addText("验证码获取超时,刷新页面...");
                await pause(3)
                location.reload();  // 刷新当前页面
            },
            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");
                }
            },
            onerror: function (error) {
                addText(`${error.statusText} ${error.status} - 出错了`)
            },
        });
    });
}

// 播放视频,同时检测验证码
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 = $("<div class='mini'>刷课<br>助手</div>")
    const container = $('<container></container>')
    container.addClass('popup');

    const header = $("<div></div>")
    header.addClass('container-header')
    header.text("成都文理学院刷课助手 " + version)
    container.append(header)

    const btnGroup = $("<div></div>")
    const classTab = $("<button class='classTabBtn'>刷课配置</button>")
    const examTab = $("<button class='examTabBtn'>搜题配置</button>")
    const statementTab = $("<button class='examTabBtn'>使用说明</button>")
    const sponsorTab = $("<button class='examTabBtn'>捐赠</button>")
    const minimize = $("<button class='examTabBtn'>缩小窗口</button>")
    btnGroup.append(classTab)
    btnGroup.append(examTab)
    btnGroup.append(statementTab)
    btnGroup.append(sponsorTab)
    btnGroup.append(minimize)
    header.append(btnGroup)

    classTab.on("click", () => {
        containerTextElement.show()
        examTextElement.hide()
        statementElement.hide()
        sponsorElement.hide()
    })
    examTab.on("click", ()=>{
        containerTextElement.hide()
        examTextElement.show()
        statementElement.hide()
        sponsorElement.hide()
    })
    statementTab.on("click", () => {
        containerTextElement.hide()
        examTextElement.hide()
        statementElement.show()
        sponsorElement.hide()
    })
    sponsorTab.on("click", () => {
        containerTextElement.hide()
        examTextElement.hide()
        statementElement.hide()
        sponsorElement.show()
    })
    classTab.on("mousedown", e=>{
        e.stopPropagation()
    })
    examTab.on("mousedown", e=>{
        e.stopPropagation()
    })
    statementTab.on("mousedown", e=>{
        e.stopPropagation()
    })
    sponsorTab.on("mousedown", e=>{
        e.stopPropagation()
    })
    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) {
        event.stopPropagation();
        event.preventDefault()
        document.body.style.userSelect = 'none';
        // 获取鼠标相对盒子的偏移量
        let shiftX = event.clientX - header.offset().left;
        let shiftY = event.clientY - header.offset().top;
        // 当鼠标移动时
        function onMouseMove(event) {
            event.stopPropagation();
            event.preventDefault()
            container.css({
                left: event.pageX - shiftX + 'px',
                top: event.pageY - shiftY + 'px'
            })
        }
        // 鼠标提起来
        function onMouseUp() {
            document.body.style.userSelect = 'auto';
            $(document).off('mousemove', onMouseMove);
            $(document).off('mouseup', onMouseUp);
        }
        $(document).on('mousemove', onMouseMove);
        $(document).on('mouseup', onMouseUp);
    })
    mini.on("mousedown", function (event) {
        event.stopPropagation();
        event.preventDefault()
        document.body.style.userSelect = 'none';
        // 获取鼠标相对盒子的偏移量
        let shiftX = event.clientX - mini.offset().left;
        let shiftY = event.clientY - mini.offset().top;
        // 当鼠标移动时
        function onMouseMove(event) {
            event.stopPropagation();
            event.preventDefault()
            moveFlag = true;
            mini.css({
                left: event.pageX - shiftX + 'px',
                top: event.pageY - shiftY + 'px'
            })
        }
        // 鼠标提起来
        function onMouseUp() {
            document.body.style.userSelect = 'auto';
            moveFlag = false;
            $(document).off('mousemove', onMouseMove);
            $(document).off('mouseup', onMouseUp);
        }
        $(document).on('mousemove', onMouseMove);
        $(document).on('mouseup', onMouseUp);
    })

    const hr = $("<hr>")
    container.append(hr)
    $("body").append(container)
    $("body").append(mini)
}
const showSponsor = () => {
    sponsorElement = $("<div></div>")
    sponsorElement.addClass('container-sponsor')
    $(".popup").append(sponsorElement)

    sponsorElement.append("<br><b>二维码是自愿捐赠!插件完全免费,但是依然需要一台服务器来处理数据。</b><br>")
    sponsorElement.append("如果您觉得它对您有所帮助,您可以选择自愿捐赠支持它。<br><br>")
    sponsorElement.append("PS: 所有捐赠将用于维护免费插件运行<br>")
    sponsorElement.append("<img alt='zfb' width='35%' src=''/>&nbsp;")
    sponsorElement.append("<img alt='wx' width='35%' src=''/>")
    sponsorElement.scrollTop(sponsorElement[0].scrollHeight)
}
const showStatement = () => {
    statementElement = $("<div></div>")
    statementElement.addClass('container-text')
    $(".popup").append(statementElement)

    statementElement.append("<h4 style='width: 100%; text-align: center;'>成都文理学院刷课助手 使用说明</h4><br><br>")
    statementElement.append("1. 您使用本插件,即表示您同意本使用条款;如果您不同意,请勿使用本插件。<br>")
    statementElement.append("2. 此脚本仅用于学习研究,您必须在下载后24小时内将所有内容从您的计算机或手机或任何存储设备中完全删除,若违反规定引起任何事件本人对此均不负责。<br>")
    statementElement.append("3. 请勿将此脚本用于任何商业或非法目的,若违反规定请自行对此负责。<br>")
    statementElement.append("4. 本人对此脚本引发的问题概不负责,包括但不限于由脚本错误引起的任何损失和损害(如封禁、限制等)。<br>")
    statementElement.append("5. 若有信息功能侵犯到您的权益,请及时联系作者。<br>")
    statementElement.append("6. 任何以任何方式查看此脚本的人或直接或间接使用此脚本的使用者都应仔细阅读此条款。<br>")
    statementElement.append("7. 本人保留随时更改或补充此条款的权利,一旦您使用或复制或修改了此脚本,即视为您已接受此免责条款。<br>")
    statementElement.scrollTop(statementElement[0].scrollHeight)
}
const showClassOption = () => {
    containerTextElement = $("<div></div>")
    containerTextElement.addClass('container-text')
    $(".popup").append(containerTextElement)
    addText("<h4>提示1</h4>:如果开启了系统代理,要先关闭!")
    addText("<h4>提示2</h4>:本脚本仅支持PC端,如果不起作用,点油猴图标看是否有提示 \"<b>Please enable developer mode...</b>\",若有,点击查看 <a target='_blank' href='https://www.baidu.com/s?wd=%E6%B2%B9%E7%8C%B4%20Please%20enable%20developer'>油猴插件不能使用</a>")
    addText("<h4>提示3</h4>:安装过老版本的需要把老版本删除或者禁用。")
    addText("<h4>提示4</h4>:因不同浏览器的优化策略问题,如果发现<b>学时没变</b>,看视频时请<b>将浏览器置于前台运行</b>。<br>")
    addText("启动成功...")
}
const showExamOption = () => {
    examTextElement = $("<div></div>")
    examTextElement.addClass('container-exam')
    $(".popup").append(examTextElement)
    examTextElement.append("<h4>提示1</h4>:如果开启了系统代理,要先关闭!<br>")
    examTextElement.append("<h4>提示2</h4>:本脚本仅支持PC端,如果不起作用,点油猴图标看是否有提示 \"<b>Please enable developer mode...</b>\",若有,点击查看 <a target='_blank' href='https://www.baidu.com/s?wd=%E6%B2%B9%E7%8C%B4%20Please%20enable%20developer'>油猴插件不能使用</a><br>")
    examTextElement.append("<h4>提示3</h4>:安装过老版本的需要把老版本删除或者禁用。<br>")
    examTextElement.append("<h4>提示4</h4>:对接的是抖音豆包,因为是AI,<b>所以不能保证完全正确,分数高低与作者无关</b>,如果有所担心可在搜完后再自己手动搜一遍<br>")
    examTextElement.append("启动成功...<br><br>")

    let endpoint_id = GM_getValue("endpoint_id", " ")
    let apikey = GM_getValue("apikey", "")
    let date = new Date();
    date.setTime(date.getTime() + (365 * 24 * 60 * 60 * 1000));

    examTextElement.append("搜题配置:点击链接 👉 <a target='_blank' href='https://pan.baidu.com/s/1YMk6Fqv6Bmr1jU0FlQXqNQ?pwd=6666'>视频教程</a> | <a target='_blank' href='https://kdocs.cn/l/clJtV1RU8GDe'>获取搜题接入点ID和API Key</a><br>")
    let endpointDiv = $("<div></div>")
    endpointDiv.append("<span>接入点ID:</span>")
    let endpointInput = $("<input type='text' value='"+endpoint_id+"'/>")
    endpointInput.on("keyup", e=>{
        endpoint_id = e.target.value
        GM_setValue("endpoint_id", endpoint_id)
    })
    endpointDiv.append(endpointInput)

    let apikeyDiv = $("<div></div>")
    apikeyDiv.append("<span>API Key: </span>")
    let apikeyInput = $("<input type='text' value='"+apikey+"'/>")
    apikeyInput.on("keyup", e=>{
        apikey = e.target.value
        GM_setValue("apikey", apikey)
    })
    apikeyDiv.append(apikeyInput)

    examTextElement.append(endpointDiv)
    examTextElement.append(apikeyDiv)

    let startBtn = $("<button>开始搜题</button>")
    let stopBtn = $("<button>停止搜题</button>")
    let saveBtn = $("<button>保存配置</button>")
    examTextElement.append(saveBtn)
    examTextElement.append(startBtn)
    examTextElement.append(stopBtn)
    examCurrent = parseInt($(".topic-head.on").text()) - 1
    examTextElement.append("<span id='examStatus'></span>")
    setExamStatus("已停止。搜题将从当前题开始");
    if ($("#startArea").length == 1){
        setExamStatus("等待答题...");
    }
    if ($(".courseexam-list").find(".time").text().includes("已交卷")) {
        setExamStatus("已交卷,不可继续答题");
    }
    saveBtn.on("click", ()=>{
        GM_setValue("endpoint_id", endpointInput[0].value)
        GM_setValue("apikey", apikeyInput[0].value)
        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></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: 250px;
    min-height: 30px;
    overflow: auto;
}
.container-exam {
    max-height: 300px;
}

.container-sponsor {
    max-height: 400px;
}

.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 + "<br>")
    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": endpoint_id,
                "messages": [
                    {
                        "role": "user",
                        "content": "只说选项号,不要说理由。\n" + question
                    }
                ]
            }),
            headers: {
                "Authorization": "Bearer " + apikey,
                "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()
    showStatement()
    showSponsor()
}

function FirstUse(){
    var fisrtUse = GM_getValue("fisrtUse", true);
    if (fisrtUse) {
        var mzsm = prompt("成都文理学院刷课助手|自动刷课|考试自动答题\n若要使用,请阅读并同意以下免责条款。\n\n \
1. 此脚本仅用于学习研究,您必须在下载后24小时内将所有内容从您的计算机或手机或任何存储设备中完全删除,若违反规定引起任何事件本人对此均不负责。\n \
2. 请勿将此脚本用于任何商业或非法目的,若违反规定请自行对此负责。\n \
3. 本人对此脚本引发的问题概不负责,包括但不限于由脚本错误引起的任何损失和损害(如封禁、限制等)。\n \
4. 若有信息功能侵犯到您的权益,请及时联系作者。\n \
5. 任何以任何方式查看此脚本的人或直接或间接使用此脚本的使用者都应仔细阅读此条款。\n \
6. 本人保留随时更改或补充此条款的权利,一旦您使用或复制或修改了此脚本,即视为您已接受此免责条款。\n\n \
若您同意以上内容,请输入“我已阅读并同意以上内容” 然后开始使用。", "");
        if (mzsm == "我已阅读并同意以上内容") {
            GM_setValue("fisrtUse", false);
            return false;
        }
        else {
            alert("免责条款未同意,脚本停止运行。\n若不想使用,请自行禁用脚本,以免每个页面都弹出该提示。");
            return true;
        }
    }
}

function matchIcon() {
    let iconLink = document.querySelector("link[rel='shortcut icon']");
    return iconLink && /yinghua|canghui|gyxy|ruren|zjxkeji|yuncanjykeji/.test(iconLink.getAttribute("href"));
}

// 运行程序
(function () {
    'use strict';

    if (!matchIcon()) return;
    if (FirstUse()) return;

    window.addEventListener("load", async function (){
        await init()
        // await test()
        if (window.location.href.includes("/node")) {
            $(".classTabBtn").click()
            getCurrent()
            addText("初始化完成,可以解放双手了;为了更像人为点击,将会延时一段时间再播放<br>")
            await pause(5, 10)
            checkCaptchaTimer = setInterval(playVideo, 1000);
        }else if (window.location.href.includes("/exam")){
            $(".examTabBtn").click()
        }else{
            $(".classTabBtn").click()
            addText("请点进课程内容,进行学习...<br>")
        }
    });
})();