Greasy Fork

来自缓存

Greasy Fork is available in English.

济南专业技术人员继续教育—公需课

公需科目辅助|自动答题|自动切换下一小节

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name        济南专业技术人员继续教育—公需课
// @namespace   Violentmonkey Scripts
// @match       *://*.ghlearning.com/*
// @match       http://221.214.69.254:9091/*
// @grant       none
// @version     0.1.10
// @author      aliha
// @description 公需科目辅助|自动答题|自动切换下一小节
// @license      GPL-3.0
// @run-at       document-end
// @icon         
// ==/UserScript==
(function () {

    // 定义一些设置变量
    const NEXT_SUBMIT_DEALY = 6;    // 尝试错误后下一次提交间隔秒,目前测试6秒刚好,不然会在弹窗出现之前往下执行,跳过一些答案选项组合
    const SUBMIT_DELAY = 1;         // 填写完成答案后,提交间隔
    const FIRST_RUN_DELAY = 15;     // 首次运行延迟15秒,防止在player没加载前运行出错
    const RUN_INTERVAL = 180;       // 脚本运行间隔,默认180秒
    const PLAY_TYPE = 1;            // 播放方式,1-方式一;2-方式二


    // 延迟s秒
    function delay(s) {
        return new Promise(resolve => setTimeout(resolve, s * 1000));
    }

    // 检测答题元素,获取选项
    function getItems() {
        if (document.querySelector(".pv-ask-modal")) {
            let qusCard = document.querySelector(".pv-ask-modal")
            let inputs = qusCard.querySelectorAll("input")
            return inputs
        }

        return null
    }

    // 生成数组所有的穷举组合并剔除空数组
    function generateCombinations(arr) {
        const combinations = [
            []
        ];

        // 遍历数组元素
        for (let i = 0; i < arr.length; i++) {
            const currentLength = combinations.length;

            // 遍历当前已生成的组合
            for (let j = 0; j < currentLength; j++) {
                const currentCombination = combinations[j];

                // 生成新的组合,包含当前数组元素
                const newCombination = currentCombination.concat(arr[i]);

                // 如果组合不为空,则将新组合添加到二维数组中
                if (newCombination.length > 0) {
                    combinations.push(newCombination);
                }
            }
        }

        // 剔除空数组
        return combinations.filter(combination => combination.length > 1);
    }

    // 判断是否暂停
    function isPuased() {
        var elements = document.getElementsByClassName("pv-icon-btn-play");
        for (var i = 0; i < elements.length; i++) {
            var element = elements[i];
            var computedStyle = window.getComputedStyle(element);
            if (computedStyle.getPropertyValue("display") === "block") {
                // console.log(element);
                return true;
            } else {
                return false;
            }
        }
    }

    // 获取进度,自动播放
    function getProcess() {
        let jindu = document.querySelector("#a span[du-html=sumschedule]"); // 获取总进度
        if (jindu) {

            if (jindu.innerText === "100.00") {
                console.info("本课程已完成");
                return 0;
            } else {
                // 获取当前进度,如果播放完毕或者未开始,点击播放按钮开始播放,(因为系统播放完一节会自动切换下一节,所以光点击播放按钮就可以了,不用手动切换小节)
                let dangqian = document.querySelector(".videoLi.active");
                let ispaused = isPuased();
                if (dangqian.innerText.match(/[0-9]+%/)[0] == "100%" || dangqian.innerText.match(/[0-9]+%/)[0] == "0%" || ispaused) {
                    console.info("本课程未完成,继续播放");

                    if (PLAY_TYPE === 1) {
                        console.info("使用方式一自动播放")
                        const pauseBtn = document.querySelector('button[type="button"].pv-playpause.pv-iconfont.pv-icon-btn-play');
                        if (pauseBtn) {
                            pauseBtn.click();
                        }

                    } else if (PLAY_TYPE === 2) {
                        console.info("使用方式二自动播放")
                        const video = document.getElementById('video');
                        video.muted = true;
                        video.play()
                            .then(() => {
                                console.info("自动播放成功");
                            })
                            .catch(error => {
                                console.error('自动播放失败,请手动点击播放:', error);
                            });
                    }
                }
            }
        }
    }

    // 挨个尝试,检测到回答错误继续,检测到回答正确跳出,同时清空答案
    async function answer(res_ls) {

        for (let i = res_ls.length - 1; i != 0; i--) {
            await delay(NEXT_SUBMIT_DEALY)

            let inputs = getItems()
            if (inputs) {
                console.log(`尝试第${res_ls.length - i}次作答`)
                // console.log(...res_ls[i])
                for (let j = 0; j < res_ls[i].length; j++) {
                    inputs[res_ls[i][j]].checked = true;
                }

                // 提交答案
                let button = document.querySelector('button.pv-ask-submit[data-type="pvSubmit"]');
                await delay(SUBMIT_DELAY)
                if (button) {
                    button.click()
                }
            }else{
                console.log("回答正确");
                break;
            }
        }
    }

    // 主函数
    async function main() {
        console.info("开始答题")

        await getProcess()

        // 自动答题
        if (getItems()) {
            let inputs = getItems()
            const array = Array.from({
                length: inputs.length
            }, (_, index) => index);
            let num = generateCombinations(array)
            await answer(num)
            console.info("答题脚本执行完毕")
        } else {
            console.info(`未检测到答题卡,${RUN_INTERVAL}秒后再次运行`)
        }
    }


    setTimeout(main, FIRST_RUN_DELAY*1000);
    setInterval(main, RUN_INTERVAL*1000);
})();