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

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

安裝腳本?
作者推薦腳本

您可能也會喜歡 暴力猴控制台美化

安裝腳本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 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);
})();