Greasy Fork

Greasy Fork is available in English.

成都文理学院刷课助手(自动填充验证码)

成都文理学院数字化实习实训平台刷课,在原基础上,添加了用户交互界面、自动识别填充验证码等功能。

当前为 2024-10-16 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         成都文理学院刷课助手(自动填充验证码)
// @version      1.0.6
// @description  成都文理学院数字化实习实训平台刷课,在原基础上,添加了用户交互界面、自动识别填充验证码等功能。
// @author       Fulling
// @match        *://zxshixun.cdcas.com/user/node*
// @icon         
// @grant        GM_xmlhttpRequest
// @license    	 MIT
// @namespace  	 https://github.com/iFulling/cdcasSK
// ==/UserScript==


let videoElement = null;
let checkCaptchaTimer = null;
let containerTextElement = null;
let timerCnt = 0;

// 下一个视频
function playNext() {
    let links = $('a[target="_self"]');
    let current = 0;

    links.each((index, item) => {
        if ($(item).hasClass("on")) {
            return current = index
        }
    });

    if (current === links.length - 1) {
        addText("最后一个已看完!")
    } else {
        clearInterval(checkCaptchaTimer);
        addText("准备播放下一个视频...")
        setTimeout(() => {
            links[current + 1].click();
        }, 3000);
    }
}

// 输入验证码
async function inputCaptcha(){
    const captchaLayer = $('.layui-layer');

    if (captchaLayer.length && captchaLayer.is(':visible')) {
        addText("验证码弹窗出现,等待填写验证码...");

        // 获取图片
        let imgs = captchaLayer.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];

        // 调用接口,识别验证码
        await getCode(code).then((ans) => {
            let inputs = captchaLayer.find("input")
            let input = inputs[0].style.display === 'none' ? inputs[1] : inputs[0]
            input.value = ans
        });

        const playButton = $('.layui-layer-btn0');  // 选择 "开始播放" 按钮

        if (playButton.length) {
            playButton.click();
            checkCaptchaTimer = setInterval(playVideo, 1000);
            addText("已自动点击开始播放按钮!");
        } else {
            addText("未找到开始播放按钮,尝试刷新页面...");
            location.reload();  // 刷新当前页面
        }
    }
}

// 使用sw1128的接口,油猴链接:http://greasyfork.icu/zh-CN/scripts/459260
function getCode(code){
    return new Promise((resolve, reject) =>{
        const datas = {
            "ImageBase64": String(code),
        }
        GM_xmlhttpRequest({
            method: "POST",
            url: "http://captcha.zwhyzzz.top:8092/identify_GeneralCAPTCHA",
            data: JSON.stringify(datas),
            headers: {
                "Content-Type": "application/json",
            },
            responseType: "json",
            onload: function(response) {
                if (response.status == 200) {
                    if (response.responseText.indexOf("触发限流策略") != -1)
                        addText(response.response["msg"]);
                    try{
                        var result = response.response["result"];
                        addText("识别结果:" + result);
                        return resolve(result);
                    }
                    catch(e){
                        if (response.responseText.indexOf("接口请求频率过高") != -1)
                            addText(response.responseText);
                    }
                }
                else {
                    addText("识别失败,请勿开启代理。");
                }
            }
        });
    });
}

// 播放视频,同时检测验证码
function playVideo(){
    timerCnt++;
    if (timerCnt % 5 === 0) {
        addText("等待加载,已加载:" + timerCnt + "秒")
    }
    if (timerCnt > 20) {
        addText("刷新页面")
        location.reload();
        return
    }
    if (!videoElement) {
        return getVideoElement();
    }
    // 验证码弹窗
    const verifyTags = $('.layui-layer');
    if (verifyTags.length > 0) {
        inputCaptcha()
        clearInterval(checkCaptchaTimer);
        return;
    }

    if (videoElement) {
        if (videoElement.paused) {
            videoElement.play();
            if (videoElement.readyState === 4 && !containerTextElement.text().includes("视频加载完成")) {
                addText("视频加载完成,准备播放");
            }
        } else {
            timerCnt = 0; // 如果视频正在播放,重置计时器
        }
    }
}

// 获取视频元素
const getVideoElement = ()=>{
    videoElement = document.querySelector("video");
    videoElement.muted = true;
    videoElement.playbackRate = 1.0;
    videoElement.volume = 0;
    videoElement.onended = function () {
        playNext();
        setTimeout(() => {}, 2000);
    };
}

// 添加交互显示
const addContainer = () =>{
    const container = $('<container></container>')
    container.addClass('popup');

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

    // 添加移动事件
    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);
    })

    const hr = $("<hr>")
    container.append(hr)

    containerTextElement = $("<div></div>")
    containerTextElement.addClass('container-text')
    container.append(containerTextElement)
    addText("启动成功...")
    addText("提示1:如果开启了系统代理,要先关闭!")
    addText("提示2:因为要获取验证码,在弹出请求跨域资源的页面时,选择 <b>总是允许</b>。")
    addText("提示3:请将浏览器置于前台运行,否则可能上传不了学时!")



    $("body").append(container)
}

// 添加样式
const addStyle = () => {
    const style = $("<style></style>")
    style.prop('type', 'text/css')
    style.html(
        `
.popup {
    position: fixed;
    top: 50px;
    left: 150px;
    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;
}

.container-header {
    height: 30px;
    width: 300px;
    cursor: move;
    line-height: 30px;
}

.container-text {
    margin-top: 10px;
    max-height: 150px;
    min-height: 30px;
    overflow: auto;
}
        `
    )
    $('body').append(style);
}


// 添加交互文本
const addText = text => {
    containerTextElement.append(text + "<br>")
    containerTextElement.scrollTop(containerTextElement[0].scrollHeight)
}

// 初始化程序
const init = () => {
    addContainer()
    addStyle()
    addText("初始化完成...")
}


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

    $(document).ready(function () {
        init()
        checkCaptchaTimer = setInterval(playVideo, 1000);
    });
})();