// ==UserScript==
// @name chinahrt继续教育;chinahrt全自动刷课;解除系统限制;
// @version 4.0.0
// @namespace https://github.com/yikuaibaiban/chinahrt
// @description 【❤全自动刷课❤】功能可自由配置,只需将视频添加到播放列表,后续刷课由系统自动完成;使用教程:https://www.cnblogs.com/ykbb/p/16695563.html
// @author yikuaibaiban;https://www.cnblogs.com/ykbb/;https://github.com/yikuaibaiban
// @icon 
// @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
// @grant GM_addStyle
//
// @license GPL
// @downloadURL none
// ==/UserScript==
class VueHandler {
static getInstance() {
return document.querySelector("article")?.__vue__
}
static pageCategory() {
const path = this.getInstance()?.$route?.path;
if (path === "/v_courseDetails") {
return General.pageCategory.detail
}
return General.pageCategory.other
}
static registerRouterChange() {
this.getInstance().$router.afterEach((to, from, failure) => {
PlayPage.removeConfigBox();
PlayPage.removePlaylistBox();
PlayPage.removeFeedbackBox();
PlayPage.removeNoticeBox();
experimentalHandler.removeExperimentalBox();
DetailPage.removeCanPlaylist();
if (to.path === "/v_courseDetails") {
DetailPage.appendToCanPlaylist(this.getCourses())
}
})
}
static getCourses() {
let results = [];
let query = this.getInstance()?.$route?.query;
const chapters = this.getInstance()?._data?.pageData?.course?.chapter_list;
if (chapters && chapters.length > 0) {
for (let i = 0; i < chapters.length; i++) {
const sections = chapters[i]?.section_list;
if (sections && sections.length > 0) {
for (let j = 0; j < sections.length; j++) {
const section = sections[j];
const url = window.location.protocol + "//" + window.location.host + window.location.pathname + "#/v_video?platformId=" + query.platformId + "&trainplanId=" + query.trainplanId + "&courseId=" + query.courseId + "§ionId=" + section.id;
results.push({
title: section.name,
url: url,
status: section.study_status + "( " + section.studyTimeStr + " )"
})
}
}
}
}
return results
}
}
class BasicHandler {
static pageCategory() {
const href = window.location.href;
if (href.indexOf("/course/play_video") > -1 || href.indexOf("/videoPlay/play") > -1) {
return General.pageCategory.play
}
if (href.indexOf("/course/preview") > -1) {
return General.pageCategory.detail
}
return General.pageCategory.other
}
static findPageCourses() {
let results = [];
if (this.pageCategory() === General.pageCategory.detail) {
const allLinks = document.querySelectorAll("a");
for (let i = 0; i < allLinks.length; i++) {
const element = allLinks[i];
if (element.href.indexOf("/course/play_video") > -1) {
results.push({title: element.innerText, url: element.href, status: $(element).prev().text()})
}
}
}
return results
}
static generateBoxItem(item, parent) {
let box = document.createElement("div");
box.className = "item";
let title = document.createElement("p");
title.innerText = item.title;
title.className = "title";
box.appendChild(title);
for (let i = 0; i < item.options.length; i++) {
const option = item.options[i];
let label = document.createElement("label");
label.innerText = option.text;
box.appendChild(label);
let input = document.createElement("input");
input.type = "radio";
input.name = item.name;
input.value = option.value;
input.checked = item.action() === option.value;
input.onclick = function () {
item.action(option.value)
};
label.appendChild(input)
}
if (item.remark) {
let remark = document.createElement("p");
remark.innerText = item.remark;
remark.className = "remark";
box.appendChild(remark)
}
parent.appendChild(box)
}
}
class General {
static coursesKey = "courses";
static pageCategory = {play: 0, detail: 1, other: 99};
static addCourse(value) {
if (!value.title || !value.url) {
this.notification("课程添加失败,缺少必要参数。");
return false
}
let courses = this.courses();
if (this.courseAdded(courses, value.url)) {
this.notification("课程已经在播放列表中。");
return false
}
courses.push({title: value.title, url: value.url});
this.courses(courses);
return true
}
static removeCourse(index) {
let courses = this.courses();
if (Number.isNaN(index)) {
for (let i = courses.length; i >= 0; i--) {
const element = courses[i];
let jsonHref = element.url;
let jsonSectionId = jsonHref.match(/sectionId=([^&]*)/)[1];
let jsonCourseId = jsonHref.match(/courseId=([^&]*)/)[1];
let jsonTrainplanId = jsonHref.match(/trainplanId=([^&]*)/)[1];
let href = window.location.href;
let sectionId = href.match(/sectionId=([^&]*)/)[1];
let courseId = href.match(/courseId=([^&]*)/)[1];
let trainplanId = href.match(/trainplanId=([^&]*)/)[1];
if (jsonCourseId === courseId && jsonSectionId === sectionId && jsonTrainplanId === trainplanId) {
courses.splice(i, 1)
}
}
} else {
courses.splice(index, 1)
}
this.courses(courses)
}
static courseAdded(courses, url) {
if (courses && Array.isArray(courses)) {
return courses.findIndex(value => value.url === url) > -1
}
return this.courses().findIndex(value => value.url === url) > -1
}
static getValue(key, defaultValue) {
return GM_getValue(key, defaultValue)
}
static setValue(key, value) {
GM_setValue(key, value);
return value
}
static autoPlay(value) {
if (value !== undefined) {
General.setValue("autoPlay", value);
if (value) {
if (player) {
player.videoPlay()
}
}
return value
} else {
return General.getValue("autoPlay", true)
}
}
static mute(value) {
if (value !== undefined) {
General.setValue("mute", value);
if (player) {
if (value) {
player.videoMute()
} else {
player.videoEscMute()
}
}
return value
} else {
return General.getValue("mute", true)
}
}
static drag(value) {
if (value !== undefined) {
General.setValue("drag", value);
if (player) {
player.changeConfig("config", "timeScheduleAdjust", value)
}
return value
} else {
return General.getValue("drag", 5)
}
}
static speed(value) {
if (attrset) {
attrset.playbackRate = 1
}
if (value !== undefined) {
General.setValue("speed", value);
if (player) {
player.changePlaybackRate(value)
}
return value
} else {
return General.getValue("speed", 1)
}
}
static playModel(value) {
return value !== undefined ? General.setValue("play_mode", value) : General.getValue("play_mode", 0)
}
static notification(content) {
GM_notification({
text: content,
title: "Chinahrt自动刷课",
image: ""
})
}
static courses(value) {
if (value) {
if (!Array.isArray(value)) {
this.notification("保存课程数据失败,数据格式异常。");
return []
}
return General.setValue(this.coursesKey, value)
}
let courses = General.getValue(this.coursesKey, []);
if (!Array.isArray(courses)) {
return []
}
return courses
}
}
class PlayPage {
static#configBoxId = "configBox";
static#playlistBoxId = "playlistBox";
static#feedbackBoxId = "feedbackBox";
static#configContent = [{
title: "自动播放",
name: "autoPlay",
action: General.autoPlay,
remark: "",
options: [{text: "是", value: true}, {text: "否", value: false}]
}, {
title: "静音",
name: "mute",
action: General.mute,
remark: "注意:不静音,视频可能会出现不会自动播放",
options: [{text: "是", value: true}, {text: "否", value: false}]
}, {
title: "拖放",
name: "drag",
action: General.drag,
remark: "注意:慎用此功能,后台可能会检测播放数据。",
options: [{text: "还原", value: 5}, {text: "启用", value: 1}]
}, {
title: "播放速度",
name: "speed",
action: General.speed,
remark: "注意:慎用此功能,后台可能会检测播放数据。",
options: [{text: "0.5倍", value: 0}, {text: "正常", value: 1}, {text: "1.25倍", value: 2}, {
text: "1.5倍",
value: 3
}, {text: "2倍", value: 4}]
}];
static playerInit() {
player.changeControlBarShow(true);
player.changeConfig("config", "timeScheduleAdjust", General.drag());
if (General.mute()) {
player.videoMute()
} else {
player.videoEscMute()
}
player.changePlaybackRate(General.speed());
if (General.autoPlay()) {
player.videoPlay()
}
}
static init() {
removePauseBlur();
PlayPage.createConfigBox();
PlayPage.createPlaylistBox();
PlayPage.createFeedbackBox();
experimentalHandler.createExperimentalBox();
GM_addValueChangeListener(General.coursesKey, function (name, oldValue, newValue, remote) {
PlayPage.removePlaylistBox();
PlayPage.createPlaylistBox()
});
PlayPage.playerInit();
player.addListener("loadedmetadata", PlayPage.playerInit);
player.addListener("ended", function () {
General.removeCourse(window.location.href);
let courses = General.courses();
if (courses.length === 0) {
General.notification("所有视频已经播放完毕")
} else {
General.notification("即将播放下一个视频:" + courses[0].title);
window.top.location.href = courses[0].url
}
});
player.addListener("time", function (t) {
experimentalHandler.timeHandler(t)
})
}
static #getConfigBox() {
return document.getElementById(this.#configBoxId)
}
static #getPlaylistBox() {
return document.getElementById(this.#playlistBoxId)
}
static #getFeedbackBox() {
return document.getElementById(this.#feedbackBoxId)
}
static createConfigBox() {
const existBox = this.#getConfigBox();
if (existBox) {
return existBox
}
let configBox = document.createElement("div");
configBox.id = this.#configBoxId;
configBox.className = "configBox";
document.body.appendChild(configBox);
let title = document.createElement("div");
title.innerText = "视频控制配置";
title.className = "title";
configBox.appendChild(title);
for (let i = 0; i < this.#configContent.length; i++) {
const element = this.#configContent[i];
BasicHandler.generateBoxItem(element, configBox)
}
return configBox
}
static removeConfigBox() {
const configBox = this.#getConfigBox();
if (configBox) {
configBox.remove()
}
}
static createPlaylistBox() {
const existBox = this.#getPlaylistBox();
if (existBox) {
return existBox
}
let playlistBox = document.createElement("div");
playlistBox.id = this.#playlistBoxId;
playlistBox.className = "playlistBox";
document.body.appendChild(playlistBox);
let oneClear = document.createElement("button");
oneClear.innerText = "一键清空";
oneClear.className = "oneClear";
oneClear.onclick = function () {
if (confirm("确定要清空播放列表么?")) {
General.courses([])
}
};
playlistBox.appendChild(oneClear);
let title = document.createElement("div");
title.innerText = "视频列表";
title.className = "title";
playlistBox.appendChild(title);
const courses = General.courses();
for (let i = 0; i < courses.length; i++) {
const course = courses[i];
let childTitle = document.createElement("p");
childTitle.innerText = course.title;
childTitle.title = course.title;
childTitle.className = "child_title";
playlistBox.appendChild(childTitle);
let childBtn = document.createElement("button");
childBtn.innerText = "移除";
childBtn.type = "button";
childBtn.setAttribute("data", i);
childBtn.className = "child_remove";
childBtn.onclick = function () {
if (confirm("确定要删除这个视频任务么?")) {
General.removeCourse(this.getAttribute("data"))
}
};
playlistBox.appendChild(childBtn)
}
}
static removePlaylistBox() {
const playlistBox = this.#getPlaylistBox();
if (playlistBox) {
playlistBox.remove()
}
}
static createFeedbackBox() {
const existBox = this.#getFeedbackBox();
if (existBox) {
return existBox
}
let box = document.createElement("div");
box.className = "feedbackBox";
document.body.appendChild(box);
let changelog = document.createElement("div");
changelog.className = "changelog";
changelog.innerHTML = "1.本次重构了所有代码。
" + "2.修复了鼠标移出视频会暂停的问题。
" + "3.增加了'一键添加'到播放列表与'一键清空'的功能。
" + "4.修复一些已知错误。
";
box.appendChild(changelog);
let notice = document.createElement("div");
notice.innerHTML = "点击课程详情页中的插件提供的【添加到播放列表】按钮添加需要自动播放的课程。
受到浏览器策略影响第一次可能无法自动播放,请手动点击播放或在控制配置中设置为静音,再刷新。";
notice.className = "notice";
box.appendChild(notice);
const links = [{title: "使用教程", link: "https://www.cnblogs.com/ykbb/p/16695563.html"}, {
title: "博客园",
link: "https://www.cnblogs.com/ykbb/"
}, {title: "留言", link: "https://msg.cnblogs.com/send/ykbb"}, {
title: "GitHub",
link: "https://github.com/yikuaibaiban/chinahrt-autoplay/issues"
}];
for (const link of links) {
let a = document.createElement("a");
a.innerText = link.title;
a.target = "_blank";
a.href = link.link;
a.className = "link";
box.appendChild(a)
}
return box
}
static removeFeedbackBox() {
this.#getFeedbackBox()?.remove()
}
}
class DetailPage {
static#canPlaylistId = "canPlaylist";
static createCanPlaylist() {
const existBox = document.getElementById(this.#canPlaylistId);
if (existBox) {
return existBox
}
let playlist = document.createElement("div");
playlist.id = this.#canPlaylistId;
playlist.className = "canPlaylist";
let oneClick = document.createElement("button");
oneClick.innerText = "一键添加";
oneClick.type = "button";
oneClick.className = "oneClick";
oneClick.onclick = function () {
const items = playlist.getElementsByClassName("item");
for (let item of items) {
const buttons = item.getElementsByTagName("button");
for (let button of buttons) {
if (button.disabled) {
continue
}
button.click()
}
}
};
playlist.appendChild(oneClick);
playlist.addEventListener("clear", function () {
while (playlist.firstChild) {
playlist.removeChild(playlist.firstChild)
}
});
playlist.addEventListener("append", function (data) {
let child = document.createElement("div");
child.className = "item";
this.appendChild(child);
let title = document.createElement("p");
title.innerText = data.detail.title;
title.title = data.detail.title;
title.className = "title";
child.appendChild(title);
let status = document.createElement("p");
status.innerText = data.detail.status;
status.title = data.detail.status;
status.className = "status";
child.appendChild(status);
let added = General.courseAdded(undefined, data.detail.url);
let addBtn = document.createElement("button");
addBtn.innerText = added ? "已在列表中" : "添加到播放列表";
addBtn.type = "button";
addBtn.disabled = added;
addBtn.className = added ? "addBtn disable" : "addBtn";
addBtn.onclick = function () {
if (General.addCourse(data.detail)) {
this.setAttribute("disabled", true);
this.setAttribute("class", "addBtn disable");
this.innerText = "已在列表中"
}
};
child.appendChild(addBtn)
});
document.body.append(playlist);
return playlist
}
static removeCanPlaylist() {
document.getElementById(this.#canPlaylistId)?.remove()
}
static appendToCanPlaylist(courses) {
const box = document.getElementById(this.#canPlaylistId);
if (Array.isArray(courses) && box) {
for (let i = 0; i < courses.length; i++) {
let course = courses[i];
box.dispatchEvent(new CustomEvent("append", {detail: course}))
}
}
}
}
class experimentalHandler {
static#experimentalBoxId = "experimentalBox";
static#experimentalContent = [{
title: "播放模式",
name: "playMode",
action: General.playModel,
remark: "",
options: [{text: "正常", value: 0}, {
text: "二段播放",
value: 3,
title: "将视频分为二段:开始,结束各播放90秒"
}, {text: "三段播放", value: 1, title: "将视频分为三段:开始,中间,结束各播放90秒"}, {
text: "秒播",
value: 2,
title: "将视频分为两段:开始,结束各播放一秒"
}]
}];
static #getExperimentalBox() {
return document.getElementById(this.#experimentalBoxId)
}
static createExperimentalBox() {
const existBox = this.#getExperimentalBox();
if (existBox) {
return existBox
}
let box = document.createElement("div");
box.className = "experimentalBox";
document.body.appendChild(box);
let tip = document.createElement("div");
tip.innerText = "此功能只适用个别地区。无法使用的就不要使用了。";
tip.className = "tip";
box.appendChild(tip);
for (let i = 0; i < this.#experimentalContent.length; i++) {
BasicHandler.generateBoxItem(this.#experimentalContent[i], box)
}
}
static removeExperimentalBox() {
const box = this.#getExperimentalBox();
if (box) {
box.remove()
}
}
static timeHandler(t) {
if (player !== undefined) {
let videoDuration = parseInt(player.getMetaDate().duration);
if (General.playModel() === 1) {
if (videoDuration <= 270) {
return
}
var videoMiddleStart = videoDuration / 2 - 45;
var videoMiddleEnd = videoDuration / 2 + 45;
var videoEndStart = videoDuration - 90;
if (t > 90 && t < videoMiddleStart) {
player.videoSeek(videoMiddleStart);
return
}
if (t > videoMiddleEnd && t < videoEndStart) {
player.videoSeek(videoEndStart);
return
}
return
}
if (General.playModel() === 2) {
if (t > 1 && t < videoDuration - 1) {
player.videoSeek(videoDuration - 1)
}
return
}
if (General.playModel() === 3) {
if (videoDuration <= 180) {
return
}
if (t > 90 && t < videoDuration - 90) {
player.videoSeek(videoDuration - 90)
}
}
} else {
General.notification("找不到播放器")
}
}
}
window.onload = function () {
setTimeout(function () {
let inVue;
try {
inVue = Vue !== undefined;
console.log("当前模式:Vue", window.location.href)
} catch (e) {
inVue = false;
console.log("当前模式:JQuery", window.location.href)
}
const pageCategory = inVue ? VueHandler.pageCategory() : BasicHandler.pageCategory();
if (pageCategory === General.pageCategory.play || General.pageCategory.detail === pageCategory) {
GM_addStyle(".canPlaylist{width:300px;height:500px;position:fixed;top:100px;background:#fff;right:80px;border:1px solid #c1c1c1;overflow-y:auto}.canPlaylist .oneClick{margin:0 auto;width:100%;border:0;padding:6px 0;background:linear-gradient(180deg,#4bce31,#4bccf2);height:50px;border-radius:5px;color:#fff;font-weight:700;letter-spacing:4px;font-size:18px}.canPlaylist .item{padding:8px;line-height:150%;border-bottom:1px solid #c1c1c1;margin-bottom:3px}.canPlaylist .item .status,.canPlaylist .item .title{font-size:13px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.canPlaylist .item .status{font-size:12px;color:#c1c1c1}.canPlaylist .item .addBtn{color:#fff;background-color:#4bccf2;border:0;padding:5px 10px;margin-top:4px}.canPlaylist .item .addBtn.disable{color:#000;background-color:#c3c3c3}.configBox{right:0;top:0;height:280px;overflow-y:auto}.configBox>.title{border-bottom:1px solid #ccc;padding:5px}.configBox .item{border-bottom:1px dotted #ccc;padding:5px 0;font-size:14px}.configBox .item .title{padding-bottom:5px;display:inline-block;font-weight:700}.configBox .item .title:after{content:\":\";margin-right:5px}.configBox .item .remark{font-size:12px;color:#c1c1c1}.configBox,.experimentalBox,.playlistBox{position:fixed;width:250px;background-color:#fff;z-index:9999;border:1px solid #ccc}.playlistBox{right:0;top:290px;height:450px;overflow-y:auto}.playlistBox .oneClear{margin:0 auto;width:100%;border:0;padding:6px 0;background:linear-gradient(180deg,#4bce31,#4bccf2);height:50px;border-radius:5px;color:#fff;font-weight:700;letter-spacing:4px;font-size:18px}.playlistBox .title{border-bottom:1px solid #ccc;padding:5px;font-weight:700}.playlistBox .child_title{font-size:13px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.playlistBox .child_remove{color:#fff;background-color:#fd1952;border:0;padding:5px 10px;margin:4px 0 10px}.experimentalBox{right:255px;top:0;height:280px}.experimentalBox .tip{border-bottom:1px solid #ccc;padding:5px;font-weight:700;color:red}.experimentalBox .item,.feedbackBox{font-size:14px}.feedbackBox{font-weight:700;background-color:#fff;padding:4px 7px;position:absolute;top:0;line-height:30px;z-index:99999;display:flex;flex-direction:row;left:30px;flex-wrap:wrap;width:226px}.feedbackBox .link{padding:8px 0 8px 10px}.feedbackBox .notice{font-size:14px;color:red;border-bottom:2px dotted #c1c1c1}.feedbackBox .changelog{margin:10px 0;border-bottom:2px dotted #c1c1c1}.configBox label,.experimentalBox label{margin-right:3px}.configBox label input,.experimentalBox label input{margin-left:2px}");
if (pageCategory === General.pageCategory.play) {
let playTimer = setInterval(function () {
try {
console.log(player);
if (player) {
PlayPage.init();
clearInterval(playTimer)
}
} catch (error) {
console.log(error);
console.log("未获取到播放器")
}
}, 500)
} else if (pageCategory === General.pageCategory.detail) {
DetailPage.createCanPlaylist();
if (inVue) {
let checkTimer = setInterval(function () {
if (VueHandler.getInstance()) {
VueHandler.registerRouterChange();
DetailPage.appendToCanPlaylist(VueHandler.getCourses());
clearInterval(checkTimer)
}
}, 500)
} else {
DetailPage.appendToCanPlaylist(BasicHandler.findPageCourses())
}
}
}
}, 1e3)
};