Greasy Fork is available in English.

chinahrt全自动刷课

chinahrt刷课,规则更改了,只能耗时间,可以把视频加入列表放置让自动切换。。

// ==UserScript==
// @name         chinahrt全自动刷课
// @namespace    https://github.com/jsgang/chinahrt
// @version      1.5
// @description  chinahrt刷课,规则更改了,只能耗时间,可以把视频加入列表放置让自动切换。。
// @author       jsgang  yikuaibaiban  https://www.jsgang.top
// @icon         https://www.google.com/s2/favicons?sz=64&domain=jsgang.top
// @match        http://*.chinahrt.com/*
// @match        https://*.chinahrt.com/*
// @match        http://videoadmin.chinahrt.com.cn/videoPlay/play*
// @match        http://videoadmin.chinahrt.com/videoPlay/play*
// @match        https://videoadmin.chinahrt.com.cn/videoPlay/play*
// @match        https://videoadmin.chinahrt.com/videoPlay/play*
//
// @grant        unsafeWindow
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addValueChangeListener
// @grant        GM_notification
//
// @license      GPL
// ==/UserScript==
/**
 * 课程预览页面
 * @type {number}
 */
const COURSE_PREVIEW = 0;
/**
 * 课程播放页面
 * @type {number}
 */
const COURSE_PALY = 1;
/**
 * 视频播放页面
 * @type {number}
 */
const VIDEO_PALY = 2;
/**
 * VUE课程预览页面
 * @type {number}
 */
const VUE_COURSE_PREVIEW = 3;
/**
 * 未知页面
 * @type {number}
 */
const UNKOWN_PAGE = 9999;
/**
 * 课程存储关键字
 * @type {string}
 */
const COURSES = "courses";
/**
 * 自动播放
 * @type {string}
 */
const AUTOPLAY = "autoPlay";
/**
 * 静音
 * @type {string}
 */
const MUTE = "mute";
/**
 * 拖动
 * @type {string}
 */
const DRAG = "drag";
/**
 * 播放速度
 * @type {string}
 */
const SPEED = "speed";
/**
 * 播放模式
 * @type {string}
 */
const PLAY_MODE = "play_mode";
/**
 * 重复次数
 * @type {string}
 */
const REAPT_NUM = "reapt_num";
/**
 * 自刷切换
 * @type {string}
 */
const REAPT_MODE = "reapt_mode";
/**
 * 获取自动播放
 * @returns {*}
 */
function getAutoPlay() {
    return GM_getValue(AUTOPLAY, true);
}

/**
 * 获取静音
 * @returns {*}
 */
function getMute() {
    return GM_getValue(MUTE, true);
}

/**
 * 获取拖动
 * @returns {*}
 */
function getDrag() {
    return GM_getValue(DRAG, 5);
}

/**
 * 获取播放速度
 * @returns {*}
 */
function getSpeed() {
    return GM_getValue(SPEED, 1);
}

/**
 * 获取播放列表
 * @returns {*|*[]}
 */
function getCourses() {
    var value = GM_getValue(COURSES, []);
    if (Array.isArray(value)) {
        return value;
    }
    return [];
}

/**
 * 添加到播放列表
 * @param element
 * @returns {boolean}
 */
function addCourse(element) {
    if (!element.title || !element.url) {
        console.error(element);
        alert("添加失败,缺少必要参数");
        return false;
    }

    var oldValue = getCourses();

    if (oldValue.findIndex(value => value.url == element.url) > -1) {
        alert("已经存在播放列表中");
        return false;
    }

    oldValue.push({title: element.title, url: element.url});

    GM_setValue(COURSES, oldValue);

    return true;
}

/**
 * 从播放列表移除
 * @param index
 */
function removeCourse(index) {
    var courses = getCourses();

    if (Number.isNaN(index)) {
        for (let i = courses.length; i >= 0; i--) {
            const element = courses[i];
            // 正则提取 href 中  sectionId courseId trainplanId
            var jsonHref = element.url;
            var jsonSectionId = jsonHref.match(/sectionId=([^&]*)/)[1];
            var jsonCourseId = jsonHref.match(/courseId=([^&]*)/)[1];
            var jsonTrainplanId = jsonHref.match(/trainplanId=([^&]*)/)[1];

            // 正则提取 window.location.href 中  sectionId courseId trainplanId
            var href = window.location.href;
            var sectionId = href.match(/sectionId=([^&]*)/)[1];
            var courseId = href.match(/courseId=([^&]*)/)[1];
            var trainplanId = href.match(/trainplanId=([^&]*)/)[1];

            if (jsonCourseId == courseId && jsonSectionId == sectionId && jsonTrainplanId == trainplanId) {
                courses.splice(i, 1);
            }
        }
    } else {
        courses.splice(index, 1);
    }

    GM_setValue(COURSES, courses);
}

/**
 * 生成可以添加到播放列表的容器
 * @returns {*|jQuery|HTMLElement}
 */
function createCanPlayList() {
    // 生成容器
    var playListBox = $("<div>", {
        id: "canPlayBox",
        css: {
            width: "300px",
            height: "500px",
            position: "fixed",
            top: "100px",
            background: "rgba(255,255,255,1)",
            right: "20px",
            border: "1px solid #c1c1c1"
        }
    });

    var status = $("<div></div>", {
        text: "获取中...",
        css: {
            height: "30px",
            "border-bottom": "1px solid",
            "text-align": "center",
            "line-height": "30px",
            "color": "#4bccf2",
            "font-weight": "bold"
        },
        click: function () {
            playListBox.trigger("clear");
            if (getPageNumber() == VUE_COURSE_PREVIEW) {
                vue_findCourses(playListBox);
            } else {
                findCourses(playListBox);
            }
        }
    });
    status.appendTo(playListBox);

    var listBox = $("<div></div>", {
        css: {
            height: "470px",
            "overflow-y": "auto"
        }
    }).appendTo(playListBox);

    playListBox.on("clear", function () {
        status.text("获取中...");
        listBox.empty();
    });

    // 添加绑定事件
    playListBox.on("bind", function (e, data) {
        if (status.text() == "获取中...") {
            status.text("点击刷新");
        }
        var box = $("<div>", {
            css: {
                "border-bottom": "1px solid #c1c1c1",
                "padding": "8px",
                "line-height": "150%",
                "border-bottom": "1px solid #c1c1c1",
                "margin-bottom": "3px"
            }
        });

        var ptitle = $("<p>", {
            text: data.title,
            title: data.title,
            css: {
                "font-size": "13px",
                "white-space": "nowrap",
                "overflow": "hidden",
                "text-overflow": "ellipsis",
            }
        });
        ptitle.appendTo(box);

        var pstatus = $("<p>", {
            text: "学习状态: " + data.status,
            title: "学习状态: " + data.status,
            css: {
                "font-size": "12px",
                "white-space": "nowrap",
                "overflow": "hidden",
                "text-overflow": "ellipsis",
                "color": "#c1c1c1"
            }
        });
        pstatus.appendTo(box);

        var disabled = getCourses().findIndex(value => value.url == data.url) > -1;
        var button = $("<button>", {
            text: disabled ? "已在列表中" : "添加到播放列表",
            type: "button",
            disabled: disabled,
            css: {
                color: disabled ? "#000" : "#FFF",
                backgroundColor: disabled ? "#c3c3c3" : "#4bccf2",
                border: "none",
                padding: "5px 10px",
                "margin-top": "4px"
            },
            click: function () {
                if (addCourse({title: data.title, url: data.url})) {
                    $(this).attr("disabled", true);
                    $(this).css({
                        color: "#000",
                        backgroundColor: "#c3c3c3",
                    }).text("已在列表中");
                }
            }
        });
        button.appendTo(box);

        box.appendTo(listBox);
    });

    playListBox.appendTo('body');

    return playListBox;
}

/**
 * 显示通知
 * @param content
 */
function showNotification(content) {
    GM_notification({
        text: content,
        title: "Chinahrt自动刷课",
        image: "",
    });
}

/**
 * 创建配置窗口
 */
function createConfigBox() {
    var box = $("<div>", {
        css: {
            position: "fixed",
            right: 0,
            top: 0,
            width: "250px",
            height: "240px",
            "background-color": "#FFF",
            "z-index": 9999,
            border: "1px solid #ccc"
        }
    });

    $("<div>", {
        text: "视频控制配置",
        css: {
            "border-bottom": "1px solid #ccc",
            padding: "5px",
            "font-weight": "bold"
        }
    }).appendTo(box);

    var configBox = $("<div>", {
        css: {
            padding: "5px",
            "padding-bottom": "5px",
            "font-size": "12px",
            "line-height": "150%"
        }
    });

    // 自动播放
    var autoPlayBox = $("<div>", {css: {"border-bottom": "1px dotted #ccc", "padding-bottom": "5px"}});
    $("<p>", {text: "是否自动播放:"}).appendTo(autoPlayBox);
    $("<input>", {
        type: "radio", name: "autoPlay", value: true, checked: getAutoPlay(), click: function () {
            GM_setValue(AUTOPLAY, true);
        }
    }).appendTo(autoPlayBox);
    $("<label>", {text: "是"}).appendTo(autoPlayBox);
    $("<input>", {
        type: "radio", name: "autoPlay", value: false, checked: !getAutoPlay(), click: function () {
            GM_setValue(AUTOPLAY, false);
        }
    }).appendTo(autoPlayBox);
    $("<label>", {text: "否"}).appendTo(autoPlayBox);
    autoPlayBox.appendTo(configBox);

    // 是否静音
    var mutePlayBox = $("<div>", {css: {"border-bottom": "1px dotted #ccc", "padding-bottom": "5px"}});
    $("<p>", {text: "是否静音:"}).appendTo(mutePlayBox);
    $("<input>", {
        type: "radio", name: "mute", value: true, checked: getMute(), click: function () {
            GM_setValue(MUTE, true);
        }
    }).appendTo(mutePlayBox);
    $("<label>", {text: "是"}).appendTo(mutePlayBox);
    $("<input>", {
        type: "radio", name: "mute", value: false, checked: !getMute(), click: function () {
            GM_setValue(MUTE, false);
        }
    }).appendTo(mutePlayBox);
    $("<label>", {text: "否"}).appendTo(mutePlayBox);
    $("<p>", {
        text: "注意:不静音,视频可能会出现不会自动播放",
        css: {"font-size": "13px", "font-weight": "bold"}
    }).appendTo(mutePlayBox);
    mutePlayBox.appendTo(configBox);

    // 启用拖放
    var dragPlayBox = $("<div>", {css: {"border-bottom": "1px dotted #ccc", "padding-bottom": "5px"}});
    $("<p>", {text: "启用拖放(慎用):"}).appendTo(dragPlayBox);
    $("<input>", {
        type: "radio", name: "drag", value: 5, checked: getDrag() == 5, click: function () {
            GM_setValue(DRAG, 5);
        }
    }).appendTo(dragPlayBox);
    $("<label>", {text: "还原"}).appendTo(dragPlayBox);
    $("<input>", {
        type: "radio", name: "drag", value: 1, checked: getDrag() == 1, click: function () {
            GM_setValue(DRAG, 1);
        }
    }).appendTo(dragPlayBox);
    $("<label>", {text: "启用"}).appendTo(dragPlayBox);
    dragPlayBox.appendTo(configBox);
 var playModeBox = $("<div>", {css: {"border-bottom": "1px dotted #ccc", "padding-bottom": "5px"}});

    playModeBox.appendTo(configBox);

    configBox.appendTo(box);

    box.appendTo("body");
}

/**
 * 创建播放列表窗口
 */
function createPlayListBox() {
    var box = $("<div>", {
        id: "playListBox",
        css: {
            position: "fixed",
            right: 0,
            top: 250,
            width: "250px",
            height: "450px",
            "background-color": "#FFF",
            "z-index": 9999,
            border: "1px solid #ccc",
            "overflow-y": "auto"
        }
    });

    $("<div>", {
        text: "视频列表",
        css: {
            "border-bottom": "1px solid #ccc",
            padding: "5px",
            "font-weight": "bold"
        }
    }).appendTo(box);

    // 渲染课程列表
    var courses = getCourses();
    for (let index = 0; index < courses.length; index++) {
        const element = courses[index];

        var ptitle = $("<p>", {
            text: element.title,
            title: element.title,
            css: {
                "font-size": "13px",
                "white-space": "nowrap",
                "overflow": "hidden",
                "text-overflow": "ellipsis",
            }
        });
        ptitle.appendTo(box);

        var button = $("<button>", {
            text: "移除",
            type: "button",
            data: {index: index},
            css: {
                color: "#FFF",
                backgroundColor: "#fd1952",
                border: "none",
                padding: "5px 10px",
                "margin": "4px 0 10px 0",
            },
            click: function () {
                if (confirm("确定删除这个视频么?")) {
                    removeCourse($(this).data("index"));
                }
            }
        });
        button.appendTo(box);
    }

    box.appendTo("body");
}

/**
 * 获取当前页面编号
 * @returns {number}
 */
function getPageNumber() {
    var href = window.location.href;
    // 以下是Vue版的请求地址
    if (href.indexOf("/index.html#/v_courseDetails") > -1) {
        return VUE_COURSE_PREVIEW;
    }
    // 默认课程详情地址
    if (href.indexOf("/course/preview") > -1) {
        return COURSE_PREVIEW;
    }
    if (href.indexOf("/course/play_video") > -1) {
        return COURSE_PALY;
    }
    if (href.indexOf("/videoPlay/play") > -1) {
        return VIDEO_PALY;
    }

    return UNKOWN_PAGE;
}

/**
 * 获取课程信息
 * @param playListBox
 */
function findCourses(playListBox) {
    // 提取所有链接
    var allLinks = document.querySelectorAll("a");
    // 提取所有可以播放的数据
    for (let i = 0; i < allLinks.length; i++) {
        const element = allLinks[i];
        if (element.href.indexOf("/course/play_video") > -1) {
            playListBox.trigger("bind", {
                title: element.innerText,
                url: element.href,
                status: $(element).prev().text()
            });
        }
    }
}

/**
 * VUE版本获取课程信息
 * @param playListBox
 */
function vue_findCourses(playListBox) {
    // 获取data
    var data = document.querySelector("article")?.__vue__?._data;
    if (!data) {
        return;
    }

    // 获取页面信息
    var pageData = data?.pageData;
    if (!pageData) {
        return;
    }

    // 获取所有章节信息
    var chapters = pageData?.course?.chapter_list;

    // 循环获取章节信息
    if (chapters && chapters.length > 0) {
        for (let i = 0; i < chapters.length; i++) {
            const chapter = chapters[i];
            // 循环分段(课时)
            var sections = chapter?.section_list;
            if (sections && sections.length > 0) {
                for (let j = 0; j < sections.length; j++) {
                    const section = sections[j];
                    // 拼接课时网址
                    var url = window.location.protocol + "//" + window.location.host + window.location.pathname + "#/v_video?platformId=" + data.platformId + "&trainplanId=" + data.trainplanId + "&courseId=" + data.courseId + "&sectionId=" + section.id;
                    playListBox.trigger("bind", {
                        title: section.name,
                        url: url,
                        status: section.study_status + "( " + section.studyTimeHHmmss + " )"
                    });
                }
            }
        }
    }
}

window.onload = function () {
    try {
        // 检测到是Vue
        if (Vue) {
            console.log("当前是Vue模式");
            // 创建播放列表
            var playListBox;
            // 循环检测
            setInterval(() => {
                if (getPageNumber() == VUE_COURSE_PREVIEW) {
                    if (!playListBox) {
                        playListBox = createCanPlayList();
                        setTimeout(function () {
                            vue_findCourses(playListBox);
                        }, 1000);
                    }
                } else {
                    if (playListBox) {
                        playListBox.remove();
                        playListBox = undefined;
                    }
                }
            }, 1000);
        }
    } catch (error) {
        console.log("当前不是Vue模式");
    }


    if (getPageNumber() == COURSE_PREVIEW) {
        $(document).ready(function () {
            // 创建播放列表
            var playListBox = createCanPlayList();
            findCourses(playListBox);
        });
    }

    if (getPageNumber() == VIDEO_PALY) {
        $(document).ready(function () {

            // 增加提示信息
            $('<div></div>', {
                html: "点课程详情页中的【添加到播放列表】按钮添加需要学习的课程。受到浏览器策略影响第一次可能无法自动播放,请手动点击播放。视频播放标签放置最前面",
                css: {
                    "font-size": "14px",
                    "font-weight": "bold",
                    color: "red",
                    background: "#FFF",
                    position: "absolute",
                    "line-height": "30px",
                    "z-index": "99999",
                    "left": "30px",
                    bottom: "10px"
                }
            }).prependTo("#body");

            $("video").prop("muted", "muted");

            // 移除讨厌的事件
            removePauseBlur();

            // 视频播放初始化
            function run() {
                 // 总是显示播放进度
                player.changeControlBarShow(true);

                // 拖动开关
                player.changeConfig('config', 'timeScheduleAdjust', getDrag());

                // 静音
                if (getMute()) {
                    player.videoMute();
                } else {
                    +
                        player.videoEscMute();
                }

                // 播放速度
                player.changePlaybackRate(getSpeed());

                // 自动播放
                if (getAutoPlay()) {
                    player.videoPlay();
                }
            }

            // 视频总长度
            var videoDuration = 0;
            var videocurrentTime=0;
            var startstaus=0;
            var actisok=0;
            var ctimes=0;
            var rtimes=0;
            var rtimess=0;
            var jumpas=0;
            var nonum=0;
            var nplay=0;

             var tmp = setInterval(function () {
                if (player != undefined) {
                    player.addListener('loadedmetadata', run);
                    run();
                    clearInterval(tmp);


                    // 移除本课程学习完毕
                    attrset.proxyUrl = "";

                    // 播放结束
                    player.addListener('ended', function () {
                        removeCourse(window.location.href);
                        var courses = getCourses();
                        if (courses.length == 0) {
                            GM_setValue(REAPT_MODE, 0);
                            GM_setValue(REAPT_NUM, 0);
                            showNotification("所有视频已经播放完毕");
                        } else {
                            GM_setValue(REAPT_MODE, 0);
                            GM_setValue(REAPT_NUM, 0);
                           
                            showNotification("即将播放下一个视频:" + courses[0].title);
                            window.top.location.href = courses[0].url;
                         
                        }
                    });


                    player.addListener('time', function (t) {
                        videoDuration = parseInt(player.getMetaDate().duration);
                        videocurrentTime = parseInt(attrset.currentTime);
                        
                        
                      // console.log(attrset.currentTime+"/"+videoDuration);

                    });
                }
            }, 1000);

            // 创建配置窗口
            createConfigBox();

            // 创建播放列表窗口
            createPlayListBox();



            // 检测播放列表
            GM_addValueChangeListener(COURSES, function (name, oldValue, newValue, remote) {
                console.log("检测播放列表变动");
                $("#playListBox").remove();
                createPlayListBox();
            });

            // 监测自动播放
            GM_addValueChangeListener(AUTOPLAY, function (name, oldValue, newValue, remote) {
                console.log("监测自动播放变动");
                if (newValue) {
                    player.videoPlay();
                }
            });

            // 检测静音
            GM_addValueChangeListener(MUTE, function (name, oldValue, newValue, remote) {
                console.log("检测静音变动");
                if (newValue) {
                    player.videoMute();
                } else {
                    player.videoEscMute();
                }
            });

            // 检测拖动
            GM_addValueChangeListener(DRAG, function (name, oldValue, newValue, remote) {
                console.log("检测拖动变动");
                player.changeConfig('config', 'timeScheduleAdjust', newValue);
            });

            // 检测速度
            GM_addValueChangeListener(SPEED, function (name, oldValue, newValue, remote) {
                console.log("检测速度变动");
                player.changePlaybackRate(newValue);
            });
        });
    }
}