// ==UserScript==
// @name 中国电信网上大学知识中心自动挂课
// @namespace http://tampermonkey.net/
// @version 1.4.1
// @description 鼠标移动到左上角图标点击开始,开启无人值守自动挂课模式!增加对活动中的专题班支持!增加2.1倍数(最大2.2,可能报异常)! 参考:HuangDingYun v.0.5
// @author zsa
// @match https://kc.zhixueyun.com/*
// @match https://cms.myctu.cn/*
// @icon 
// @require https://cdn.jsdelivr.net/npm/jquery@3.6.0/dist/jquery.min.js
// @require https://cdn.jsdelivr.net/npm/blueimp-md5@2.19.0/js/md5.min.js
// @grant unsafeWindow
// @grant GM_openInTab
// @grant GM.openInTab
// @grant GM_getValue
// @grant GM.getValue
// @grant GM_setValue
// @grant GM.setValue
// @grant GM_deleteValue
// @grant GM.deleteValue
// @grant GM_xmlhttpRequest
// @grant GM.xmlHttpRequest
// @grant GM_registerMenuCommand
// @grant GM_addValueChangeListener
// @license MIT
// ==/UserScript==
(function(g) {
g.gooeymenu = function(h, q) {
var rt = g(h);
var k = rt.find('.gooey-menu-nav');
k.addClass("navimenu");
var b = this
, a = b.options = g.extend({}, g.gooeymenu.defaults, q);
b.els = {
item: k.find(".gooey-menu-item"),
checkbox: k.find(".gooey-menu-open"),
button: k.find(".gooey-open-button")
};
b.methods = {
setup: function() {
var d = {
small: 1.4,
medium: 1.8,
large: 2.1
}, f = {
small: 1.3,
medium: 1.6,
large: 2.1
}, c;
for (c in d)
a.margin === c ? a.margin = d[c] : null;
for (var e in f)
a.bounceLength === e ? a.bounceLength = f[e] : null;
b.methods.responsiveStyles();
b.els.item.hover(function() {
a.currentBg = b.els.item.css("background-color");
g(this).css("background-color", a.hover)
}, function() {
g(this).css("background-color", a.currentBg)
});
!0 === a.bounce && b.methods.bounce()
},
setEvents: function() {
["open", "close"].forEach(function(d, b) {
k.on(d, function() {
a[d] && a[d].apply(this, arguments)
})
})
},
bounce: function() {
if (!0 === a.bounce) {
var d = b.els.item.css("transition-timing-function");
b.els.checkbox.on("change", function() {
g(this).is(":checked") ? b.els.item.css({
"transition-timing-function": "cubic-bezier(0.8, 0.84, 0.44, " + a.bounceLength + ")",
"-o-transition-timing-function": "cubic-bezier(0.8, 0.84, 0.44, " + a.bounceLength + ")",
"-moz-transition-timing-function": "cubic-bezier(0.8, 0.84, 0.44, " + a.bounceLength + ")",
"-webkit-transition-timing-function": "cubic-bezier(0.8, 0.84, 0.44, " + a.bounceLength + ")",
"-ms-transition-timing-function": "cubic-bezier(0.8, 0.84, 0.44, " + a.bounceLength + ")"
}) : b.els.item.css({
"transition-timing-function": d,
"-moz-transition-timing-function": d,
"-o-transition-timing-function": d,
"-webkit-transition-timing-function": d
})
})
}
},
position: function(e) {
var p = [[0,1],[3,2]],w = window.innerWidth, h = window.innerHeight, x = b.methods.getCss(e,"left"), y = b.methods.getCss(e,"top");
return p[parseInt(x) < w / 2 ? 0 : 1][parseInt(y) < h / 2 ? 0 : 1];
},
circle: function() {
k.trigger("open");
var pt = b.methods.position(rt[0]), rr = b.els.item.length < 3 ? 180 : 360;
var d, f, c, e, m, h, n, r = b.els.item.length, l = a.transitionStep, q = Math.PI, t = rr / r, p = t = rr / r;
p = (p + (pt * 90) + 180) % 360;
f = a.circle.radius;
b.els.item.each(function() {
b.els.checkbox.is(":checked") ? (d = q * p / 180,
c = Math.abs(Math.cos(d)),
e = f * c,
m = Math.sqrt(f * f - e * e),
h = b.methods.periodCalc(p).x,
n = b.methods.periodCalc(p).y,
g(this).css({
transform: "translate3d(" + h + e + "px," + n + m + "px,0)",
"-o-transform": "translate3d(" + h + e + "px," + n + m + "px,0)",
"-webkit-transform": "translate3d(" + h + e + "px," + n + m + "px,0)",
"-moz-transform": "translate3d(" + h + e + "px," + n + m + "px,0)",
"-ms-transform": "translate3d(" + h + e + "px," + n + m + "px,0)",
"transition-duration": l + "ms",
"-o-transition-duration": l + "ms",
"-webkit-transition-duration": l + "ms",
"-moz-transition-duration": l + "ms"
}),
p += t,
l += a.transitionStep) : (b.els.item.css({
transform: "translate3d(0, 0, 0)",
"-moz-transform": "translate3d(0, 0, 0)",
"-webkit-transform": "translate3d(0, 0, 0)",
"-ms-transform": "translate3d(0, 0, 0)",
"-o-transform": "translate3d(0, 0, 0)"
}),
p = 360 / r,
l = a.transitionStep,
k.trigger("close"))
})
},
periodCalc: function(a) {
return {
x: 90 > a || 270 < a ? "" : "-",
y: 180 < a ? "" : "-"
}
},
linear: function(d) {
k.trigger("open");
var f = "horizontal" === a.style ? a.horizontal.menuItemPosition : a.vertical.menuItemPosition
, c = d[f].init
, e = a.transitionStep;
b.els.item.each(function() {
b.els.checkbox.is(":checked") ? "horizontal" === a.style ? (g(this).css({
transform: "translate3d(" + c + "px, 0, 0)",
"-ms-transform": "translate3d(" + c + "px, 0, 0)",
"-o-transform": "translate3d(" + c + "px, 0, 0)",
"-moz-transform": "translate3d(" + c + "px, 0, 0)",
"-webkit-transform": "translate3d(" + c + "px, 0, 0)",
"transition-duration": e + "ms",
"-o-transition-duration": e + "ms",
"-webkit-transition-duration": e + "ms",
"-moz-transition-duration": e + "ms"
}),
c += d[f].init,
e += a.transitionStep) : "vertical" === a.style && (g(this).css({
"transition-duration": e + "ms",
"-moz-transition-duration": e + "ms",
"-o-transition-duration": e + "ms",
"-webkit-transition-duration": e + "ms"
}),
"down" === a.vertical.direction ? g(this).css({
transform: "translate3d(0, " + c + "px, 0)",
"-moz-transform": "translate3d(0, " + c + "px, 0)",
"-o-transform": "translate3d(0, " + c + "px, 0)",
"-webkit-transform": "translate3d(0, " + c + "px, 0)",
"-ms-transform": "translate3d(0, " + c + "px, 0)"
}) : "up" === a.vertical.direction && g(this).css({
transform: "translate3d(0,-" + c + "px, 0)",
"-moz-transform": "translate3d(0,-" + c + "px, 0)",
"-webkit-transform": "translate3d(0,-" + c + "px, 0)",
"-o-transform": "translate3d(0,-" + c + "px, 0)",
"-ms-transform": "translate3d(0,-" + c + "px, 0)"
}),
c += d[f].init,
e += a.transitionStep) : (b.els.item.css({
transform: "translate3d(0, 0, 0)",
"-moz-transform": "translate3d(0, 0, 0)",
"-webkit-transform": "translate3d(0, 0, 0)",
"-ms-transform": "translate3d(0, 0, 0)",
"-o-transform": "translate3d(0, 0, 0)"
}),
c = d[f].init,
e = a.transitionStep,
k.trigger("close"))
})
},
translate: function() {
var d = {
glue: {
init: a.size
},
spaced: {
init: a.size * a.margin
}
};
b.els.checkbox.on("change", function() {
b._callbacks[a.style](d)
})
},
createOn: function(a, b, c, e) {
b = document.createElementNS("http://www.w3.org/2000/svg", b);
for (var g in c)
c.hasOwnProperty(g) && b.setAttribute(g, c[g]);
e && b.appendChild(document.createTextNode(e));
return a.appendChild(b)
},
responsiveStyles: function() {
var d = 0 < window.innerWidth ? window.innerWidth : screen.width;
320 <= d && 480 >= d ? (a.size /= 1.4,
a.circle.radius /= 1.2) : 480 < d && 768 >= d ? a.size /= 1.2 : 780 < d && 1024 >= d && (a.circle.radius /= 1.2,
a.size /= 1.1);
b.els.item.css({
width: a.size + "px",
height: a.size + "px",
color: a.contentColor,
"background-color": a.bgColor,
"line-height": a.size + "px"
});
b.els.button.css({
width: a.size + "px",
height: a.size + "px",
"background-color": a.bgColor,
"line-height": a.size + "px"
});
k.find(".burger").css({
"font-size": ".8em",
width: a.size / 2 + "px",
height: "3px",
left: a.size / 4 + "px"
})
},
getCss: function(e,a) {
return e.currentStyle?e.currentStyle[a]:document.defaultView.getComputedStyle(e,!1)[a];
},
openMove: function(){
var drags={down:!1,x:0,y:0,winWid:0,winHei:0,clientX:0,clientY:0};
g(b.els.button).on('mousedown', function(e) {
drags.down=!0,
drags.clientX=e.clientX,
drags.clientY=e.clientY,
drags.x=b.methods.getCss(rt[0],"left"),
drags.y=b.methods.getCss(rt[0],"top"),
drags.winHei=$(window).height(),
drags.winWid=$(window).width(),
g(document).on("mousemove",function(e){
if(drags.winWid>640&& e.clientX>=80 && e.clientX <= drags.winWid-90) //50px
rt[0].style.left=parseInt(drags.x)+ (e.clientX-drags.clientX) +"px";
if(e.clientY >= 100 && e.clientY <= drags.winHei-80)//导航高度
rt[0].style.top=parseInt(drags.y)+ (e.clientY-drags.clientY) +"px";
})
});
g(document).on("mouseup",function(){drags.down==!0 && (drags.down=!1,g(document).off("mousemove"))});
},
openHover: function(){
let tt;
g(b.els.button).hover(function(){
clearTimeout(tt);
g(b.els.checkbox).prop("checked", true).change();
},function(){
tt = setTimeout(()=>{
g(b.els.checkbox).prop("checked", false).change();
},200);
});
g(b.els.item).hover(function(){
clearTimeout(tt);
g(b.els.checkbox).prop("checked", true).change();
},function(){
tt = setTimeout(()=>{
g(b.els.checkbox).prop("checked", false).change();
},200)
});
}
};
b._callbacks = {
vertical: b.methods.linear,
horizontal: b.methods.linear,
circle: b.methods.circle
};
b.init = function() {
var a = document.createElementNS("http://www.w3.org/2000/svg", "svg")
, f = g(".navimenu").index(k);
a.setAttribute("id", "gooeySVG" + f);
a.setAttribute("class", "gooeySVG");
k.append(a);
a = document.getElementById("gooeySVG" + f);
b.methods.createOn(a, "defs", {
id: "defs" + f
});
a = document.getElementById("defs" + f);
b.methods.createOn(a, "filter", {
id: "goo-shadow" + f,
overflow: "hidden"
});
var c = document.getElementById("goo-shadow" + f);
b.methods.createOn(c, "feGaussianBlur", {
"in": "SourceGraphic",
result: "blur",
stdDeviation: "10"
});
b.methods.createOn(c, "feColorMatrix", {
"in": "blur",
mode: "matrix",
values: "1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 18 -8",
result: "goo"
});
b.methods.createOn(c, "feGaussianBlur", {
"in": "goo",
stdDeviation: "2",
result: "shadow"
});
b.methods.createOn(c, "feColorMatrix", {
"in": "shadow",
mode: "matrix",
values: "0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0",
result: "shadow"
});
b.methods.createOn(c, "feOffset", {
"in": "shadow",
dx: "1",
dy: "1",
result: "shadow"
});
b.methods.createOn(c, "feComposite", {
in2: "shadow",
"in": "goo",
result: "goo"
});
b.methods.createOn(c, "feComposite", {
in2: "goo",
"in": "SourceGraphic",
result: "mix"
});
b.methods.createOn(a, "filter", {
id: "goo" + f
});
a = document.getElementById("goo" + f);
b.methods.createOn(a, "feGaussianBlur", {
"in": "SourceGraphic",
result: "blur",
stdDeviation: "10"
});
b.methods.createOn(a, "feColorMatrix", {
"in": "blur",
mode: "matrix",
values: "1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 18 -7",
result: "goo"
});
b.methods.createOn(a, "feComposite", {
in2: "goo",
"in": "SourceGraphic",
result: "mix"
});
k.css({
"-webkit-filter": "url('#goo-shadow" + f + "')",
filter: "url('#goo-shadow" + f + "')",
"-ms-filter": "url('#goo-shadow" + f + "')",
"-o-filter": "url('#goo-shadow" + f + "')"
});
b.methods.setEvents();
b.methods.setup();
b.methods.translate.apply(this, arguments);
b.methods.openMove();
b.methods.openHover();
g('head').append('<style>.navimenu{position: relative;min-width: 100px;min-height: 100px;margin: auto auto auto -80px;padding-top: 20px;padding-left: 80px;box-sizing: border-box;text-align: left;-webkit-backface-visibility: hidden;backface-visibility: hidden;-webkit-transform-style: preserve-3d;outline: transparent solid 5px;transform-style: preserve-3d}.navimenu .gooey-menu-item,.navimenu .gooey-open-button{border-radius: 100%;position: absolute;color: #fff;text-align: center;font-size: .9em;transform: translate3d(0,0,0);-ms-transform: translate3d(0,0,0);-moz-transform: translate3d(0,0,0);-o-transform: translate3d(0,0,0);-webkit-transform: translate3d(0,0,0);transition: transform ease-out 200ms;-ms-transition: transform ease-out 200ms;-o-transition: transform ease-out 200ms;-moz-transition: transform ease-out 200ms;-webkit-transition: transform ease-out 200ms;-webkit-backface-visibility: hidden;backface-visibility: hidden;-webkit-transform-style: preserve-3d;outline: transparent solid 5px;transform-style: preserve-3d}.navimenu .gooey-menu-open{display: none}.navimenu .burger{background: #fff;display: block;position: absolute;top: 50%;left: 50%;margin-top: -1.5px;transition: transform 200ms;-ms-transition: transform 200ms;-moz-transition: transform 200ms;-webkit-transition: transform 200ms;-o-transition: transform 200ms}.navimenu .burger-1{transform: translate3d(0,-8px,0);-moz-transform: translate3d(0,-8px,0);-o-transform: translate3d(0,-8px,0);-webkit-transform: translate3d(0,-8px,0);-ms-transform: translate3d(0,-8px,0)}.navimenu .burger-2{transform: translate3d(0,0,0);-ms-transform: translate3d(0,0,0);-moz-transform: translate3d(0,0,0);-o-transform: translate3d(0,0,0);-webkit-transform: translate3d(0,0,0)}.navimenu .burger-3{transform: translate3d(0,8px,0);-ms-transform: translate3d(0,8px,0);-moz-transform: translate3d(0,8px,0);-o-transform: translate3d(0,8px,0);-webkit-transform: translate3d(0,8px,0)}.navimenu .gooey-menu-open:checked+.gooey-open-button .burger-1{transform: translate3d(0,0,0) rotate(45deg);-ms-transform: translate3d(0,0,0) rotate(45deg);-moz-transform: translate3d(0,0,0) rotate(45deg);-o-transform: translate3d(0,0,0) rotate(45deg);-webkit-transform: translate3d(0,0,0) rotate(45deg)}.navimenu .gooey-menu-open:checked+.gooey-open-button .burger-2{transform: translate3d(0,0,0) scale(.1,1);-ms-transform: translate3d(0,0,0) scale(.1,1);-moz-transform: translate3d(0,0,0) scale(.1,1);-o-transform: translate3d(0,0,0) scale(.1,1);-webkit-transform: translate3d(0,0,0) scale(.1,1)}.navimenu .gooey-menu-open:checked+.gooey-open-button .burger-3{transform: translate3d(0,0,0) rotate(-45deg);-ms-transform: translate3d(0,0,0) rotate(-45deg);-moz-transform: translate3d(0,0,0) rotate(-45deg);-webkit-transform: translate3d(0,0,0) rotate(-45deg);-o-transform: translate3d(0,0,0) rotate(-45deg)}.navimenu .gooey-menu-item:hover{background-color: #4682b4;color: #00bcd4}.navimenu .gooey-menu-item{transition-duration: 180ms;-moz-transition-duration: 180ms;-webkit-transition-duration: 180ms;-o-transition-duration: 180ms}.navimenu .gooey-open-button{z-index: 2;transition-timing-function: cubic-bezier(.175,.885,.32,1.275);-ms-transition-timing-function: cubic-bezier(.175,.885,.32,1.275);-moz-transition-timing-function: cubic-bezier(.175,.885,.32,1.275);-webkit-transition-timing-function: cubic-bezier(.175,.885,.32,1.275);-o-transition-timing-function: cubic-bezier(.175,.885,.32,1.275);transition-duration: 400ms;-ms-transition-duration: 400ms;-o-transition-duration: 400ms;-moz-transition-duration: 400ms;-webkit-transition-duration: 400ms;transform: scale(1.1,1.1) translate3d(0,0,0);-ms-transform: scale(1.1,1.1) translate3d(0,0,0);-o-transform: scale(1.1,1.1) translate3d(0,0,0);-moz-transform: scale(1.1,1.1) translate3d(0,0,0);-webkit-transform: scale(1.1,1.1) translate3d(0,0,0);cursor: pointer}.navimenu .gooey-open-button:hover{transform: scale(1.2,1.2) translate3d(0,0,0);-ms-transform: scale(1.2,1.2) translate3d(0,0,0);-moz-transform: scale(1.2,1.2) translate3d(0,0,0);-o-transform: scale(1.2,1.2) translate3d(0,0,0);-webkit-transform: scale(1.2,1.2) translate3d(0,0,0)}.navimenu .gooey-menu-open:checked+.gooey-open-button{transition: 200ms linear;-ms-transition: 200ms linear;-webkit-transition: 200ms linear;-moz-transition: 200ms linear;-o-transition: 200ms linear;transform: scale(.9,.9) translate3d(0,0,0);-ms-transform: scale(.9,.9) translate3d(0,0,0);-o-transform: scale(.9,.9) translate3d(0,0,0);-webkit-transform: scale(.9,.9) translate3d(0,0,0);-moz-transform: scale(.9,.9) translate3d(0,0,0)}.navimenu .gooeySVG{display: none;}</style>')
}
;
b.init()
}
;
g.gooeymenu.defaults = {
style: "horizontal",
size: 70,
margin: "medium",
bgColor: "steelblue",
contentColor: "white",
transitionStep: 100,
bounce: !1,
bounceLength: "medium",
hover: "white",
circle: {
radius: 80
},
horizontal: {
menuItemPosition: "glue"
},
vertical: {
menuItemPosition: "spaced",
direction: "up"
},
open: function() {},
close: function() {}
};
g.fn.gooeymenu = function(h) {
void 0 === h && (h = {});
if (h && "object" === typeof h)
return this.each(function() {
new g.gooeymenu(this,h)
})
}
}
)(unsafeWindow.jQuery || window.jQuery);
const autoPlay = (function ($) {
const CONFIG = {
keys: {
courseStart: '_.unique.name.course.start.',
courseClose: '_.unique.name.course.close.',
subject: '_.unique.name.subject.',
}
}
function GM_onMessage(_this, label, callback) {
// console.log(`增加GM_onMessage,参数为${label}`);
GM_addValueChangeListener(label, function () {
// console.log('onMessage', arguments[2])
callback.apply(_this, arguments[2]);
});
}
function GM_sendMessage(label) {
let prams = Array.from(arguments).slice(1);
prams.push(new Date().getTime());
// console.log(`调用GM_sendMessage,参数为${label}, ${prams}`);
GM_setValue(label, prams);
}
function pageType() {
console.log('window.location.href', window.location.href)
if (window.location.href.match("/study/subject/detail/")) {
return 'List';
} else if(window.location.href.match("/train-new/class-detail")){
return 'NewList';
} else if (window.location.href.match("/vue/paas-designer")) {
return 'PaasList';
} else if (window.location.href.match("/study/course/detail/")) {
return 'Detail';
} else if (window.location.href.match("/safe/topic/resource/")) {
return 'Resource';
}
return 'Unknown';
}
class Filter {
applyRequestFilter(url, method, params) {
// 这里是接口的默认行为,可根据需要在子类中重写
return params
}
applyResponseFilter(url, xhr) {
// 这里是接口的默认行为,可根据需要在子类中重写
}
}
class Filter111 extends Filter {
applyRequestFilter(url, method, argument) {
// 这里是接口的默认行为,可根据需要在子类中重写
console.log('applyRequestFilter', url, argument)
return argument
}
applyResponseFilter(url, xhr) {
// 这里是接口的默认行为,可根据需要在子类中重写
console.log('applyResponseFilter', url, xhr)
}
}
class AbstractPlayPage {
constructor(supportive = true) {
this.targetName = new.target.name;
if (new.target === AbstractPlayPage) {
throw new Error("不能直接创建抽象类");
}
let originalOpen = XMLHttpRequest.prototype.open
let originalSend = XMLHttpRequest.prototype.send
let _that = this;
// 重写 XMLHttpRequest 的 open 方法
XMLHttpRequest.prototype.open = function (method, url) {
// 存储请求的 URL
this._url = url
let _params = arguments
_that.filterList().forEach(f=>{
_params = f.applyRequestFilter(url, method, _params) || _params;
});
originalOpen.apply(this, _params)
}
// 重写 XMLHttpRequest 的 send 方法
XMLHttpRequest.prototype.send = function (body) {
if (this.onreadystatechange) {
let originalReadyStateChange = this.onreadystatechange
this.onreadystatechange = function () {
// 检查特定条件:请求完成、状态成功,并且 URL 包含特定路径
if (this.readyState === 4 && this.status === 200 && (this.responseType == '' || this.responseType == 'text')) {
_that.getFilterList && (_that.getFilterList().forEach(filter => {
filter.applyResponseFilter(this._url, this);
}))
// Object.defineProperty(this, '_responseText', this.responseText)
// console.log('this.responseText', this.responseText)
}
originalReadyStateChange.apply(this)
}
}
originalSend.apply(this, arguments)
}
if(supportive) {
let uuids = window.location.href.match('[0-9a-f]{8}(-[0-9a-f]{4}){3}-[0-9a-f]{12}');
this.pageId = (uuids && uuids[0]) || '';
}
}
filterList() {
// return [new Filter111()]
return [];
}
addNav(menuItems) {
$('body').append(`
<div id="zGooeyMenu">
<nav class="gooey-menu-nav">
<input type="checkbox" class="gooey-menu-open"/>
<label class="gooey-open-button">
<span class="burger burger-1"></span>
<span class="burger burger-2"></span>
<span class="burger burger-3"></span>
</label>
</nav>
</div>
`)
$('head').append(`
<style >
#zGooeyMenu {
width: 130px;
height: 130px;
position: fixed;
z-index: 10000;
top: 40px;
left: 60px
}
#gooey-round .gooey-open-button, #gooey-round .gooey-menu-item {
margin-top: 6.5em;
margin-left: 3.2em;
}
</style>
`)
menuItems.forEach((e)=>{
let item = $(`<a href="javascript:void(0)" id="${e.id}" class="gooey-menu-item"> ${e.name} </a>`)
if (!!e.child) {
item.addClass('drop-down-wrapper');
item.append($(`<div class="drop-down-content "> </div>`).append($(`<div class="content text-center"> </div>`).append(e.child)))
}
$("#zGooeyMenu").find('nav.gooey-menu-nav').append(item);
})
$("#zGooeyMenu").gooeymenu({
bgColor: "#68d099",
contentColor: "white",
style: "circle",
circle: {
radius: 60
},
margin: "small",
size: 60,
bounce: true,
bounceLength: "small",
transitionStep: 100,
hover: "#5dbb89",
open: function() {
$(this).find(".gooey-menu-item").css("background-color", "steelblue");
$(this).find(".gooey-open-button").css("background-color", "steelblue");
},
close: function() {
$(this).find(".gooey-menu-item").css("background-color", "#ffdf00");
$(this).find(".gooey-open-button").css("background-color", "#ffdf00");
}
})
}
start() {
throw new Error(this.targetName + " 未实现 start 方法");
}
}
class UnknownPlayPage extends AbstractPlayPage{
constructor() {
super(false);
}
start() {
// 不支持的页面
}
}
class ListPlayPage extends AbstractPlayPage {
constructor() {
super();
this.playClassList = new Set();
this.maxNum = 1;
this.waitNum = 0;
this.checkTaskList = {};
this.checkCloseTaskList = {};
this.status = 0;
this.repeatNum = 0;
this.repeatMaxNum = 3;
}
start() {
// this.addButton();
this.addListener();
this.addButton2();
}
addListener() {
// 监听课程关闭事件
GM_onMessage(this, CONFIG.keys.courseClose + this.pageId, function (src, data) {
console.log('data',data);
if (data.status == -1) {
GM_setValue(CONFIG.keys.subject + data.courseId, this.pageId)
console.warn('课程播放未完成,可能刷新了页面:', data.courseId)
this.checkCloseTaskList[data.courseId] = setTimeout((courseId) => {
console.warn('课程打开失败:', courseId)
delete this.checkCloseTaskList[courseId]
this.playClassList.delete(courseId);
GM_deleteValue(CONFIG.keys.subject + courseId);
}, 1000 * 8, data.courseId);
}
if (data.status == 0) {
console.warn('课程播放未完成:', data.courseId)
this.playClassList.delete(data.courseId);
}
GM_deleteValue(CONFIG.keys.subject + data.courseId);
if (data.status == 1) {
this.openClassPage();
}
});
// 监听课程成功打开事件
GM_onMessage(this, CONFIG.keys.courseStart + this.pageId, function (src, courseId) {
console.log('课程成功打开', courseId);
this.repeatNum = 0;
this.checkTaskList[courseId] && (clearTimeout(this.checkTaskList[courseId]), delete this.checkTaskList[courseId])
this.checkCloseTaskList[courseId] && (clearTimeout(this.checkCloseTaskList[courseId]), delete this.checkCloseTaskList[courseId])
});
}
addButton2() {
this.addNav([{id:'startAutoPlayer', name: '开始'}, {id:'setUp', name: '未实现', child: ''}]);
$('#startAutoPlayer').click(() => {
if(this.status == 0) {
this.status = 1;
$('#startAutoPlayer').html('停止');
this.playClassList.clear();
this.waitNum = 0;
this.openClassPage(this.maxNum);
} else {
this.stop();
}
})
}
stop(){
this.status = 0;
$('#startAutoPlayer').html('开始');
}
addButton() {
var divButton = '<div class="item" id="startAutoPlayer"> <div class="view"> <i class="iconfont icon-play"></i> <div class="text" id="startAutoPlayerText">自动挂课</div> </div> </div>';
const a = setInterval(() => {
if($("#D60toolbarTab").length > 0 && $("#D60toolbarTab .item").length > 0) {
window.clearInterval(a);
$("#D60toolbarTab").append(divButton);
$('#startAutoPlayer').click(() => {
this.playClassList.clear();
this.waitNum = 0;
this.openClassPage(this.maxNum);
})
}
}, 300);
}
openClassPage(num) {
if (this.status == 0 ) return;
let classList = $("div.item.current-hover").filter((i,e) => ($(e).find("i.icon-reload").length == 0));
// 过滤选修
classList = classList.filter((i,e) => ($(e).find(".attribute .default-skin.is-required").length > 0));
// 过滤考试
classList = classList.filter((i,e) => ($(e).find(".attribute .section-type").text() != '考试'));
if (classList.length == 0) {
console.log('播放完成');
this.stop();
return;
}
this.playClass(classList, num, e => $(e).attr('data-resource-id'));
}
playClass(classList, num, obtainId, play) {
play = play || (e => $(e).click())
let maxNum = Math.min(this.maxNum, (num || Math.max(this.waitNum, 1)));
console.log('playClass - maxNum', maxNum)
let idx = 0;
do {
let courseId = obtainId(classList[idx])
if (!this.playClassList.has(courseId)) {
play(classList[idx]);
// 课程可能没正常打开、监听成功打开事件,没有事件的认为打开失败
this.checkTask(courseId)
this.playClassList.add(courseId);
maxNum --;
this.waitNum = Math.max(this.waitNum - 1, 0);
console.log('playClass - waitNum', this.waitNum)
}
idx ++;
} while(maxNum > 0 && idx < classList.length)
if (maxNum > 0) {
this.openClassPage(maxNum)
}
}
checkTask(courseId) {
GM_setValue(CONFIG.keys.subject + courseId, this.pageId)
this.checkTaskList[courseId] = setTimeout((courseId) => {
console.warn('课程打开失败:', courseId)
delete this.checkTaskList[courseId]
this.playClassList.delete(courseId);
// 尝试播放下一个
if(this.repeatNum <= this.repeatMaxNum) {
this.repeatNum
this.openClassPage();
}
}, 1000 * 10, courseId);
}
}
// 活动专题班
class NewListPlayPage extends ListPlayPage {
openClassPage(num) {
if (this.status == 0 ) return;
let section = $('.train-citem').closest('.section');
if($(section).find("div.node-flag").hasClass('lock')) {
console.log('本部分被锁,暂无法学习');
this.waitNum += num;
return
}
// 有更多按钮就先加载
if($('#D235loadMore').length > 0 ){
$('#D235loadMore').click();
setTimeout(num => this.openClassPage(num), 3000, num);
return
}
let classList = $('.train-citem').filter((i,e)=>$(e).find('.t-business-type').text().trim() == '课程' && $(e).find('div.un-finish').length > 0).find('.pointer');
classList = classList.filter((i,e)=> !this.playClassList.has(this.obtainId(e)))
if (classList.length == 0) {
console.log('本部分播放完成,尝试加载下一部分 (可能是被锁的)');
let nextSection = $(section).next();
if(nextSection.length == 0) {
console.log('学习完成');
this.stop();
return;
}
$(nextSection).find('div.section-title div.right-area i.icon-triangle-down').click()
setTimeout(num => this.openClassPage(num), 3000, num);
return
}
this.playClass(classList, num, this.obtainId);
}
obtainId(e) {
return $(e).attr('id').match('[0-9a-f]{8}(-[0-9a-f]{4}){3}-[0-9a-f]{12}')[0];
}
}
//专题
class PaasListPlayPage extends ListPlayPage {
constructor() {
super();
this.sectionIdx = 0;
if(unsafeWindow.top != unsafeWindow) {
let pme = unsafeWindow.parent.document.getElementById('zGooeyMenu')
if($(pme).length > 0) {
$(pme).hide()
}
}
}
openClassPage(num) {
if (this.status == 0 ) return;
let sections = $('.grade');
let section = sections[this.sectionIdx];
if($(section).find('.title-item .lockbox').length > 0 && $(section).find('.title-item .lockbox').find('.biz-Unlock') == 0) {
console.log('本部分被锁,暂无法学习');
this.waitNum += num;
return
}
// 点击展开
let clickTxt = $(section).find('.title-item .clickTxt');
if($(clickTxt).text().indexOf('展开') >= 0) {
$(clickTxt).click();
setTimeout(num => this.openClassPage(num), 3000, num);
return
}
// 有更多按钮就先加载
if($(section).find('>div').length ==1 ){
console.log('活动未成功加载');
this.waitNum += num;
return
}
let classDiv = $(section).find('>div')[1];
if($(classDiv).find('.more').length > 0) {
$(classDiv).find('.more').click();
setTimeout(num => this.openClassPage(num), 3000, num);
return
}
// 获取课程列表 TODO 获取不到ID 考虑用课程名计算
let classList = $(classDiv).find('.activeContent').filter((i,e) => $(e).find('.btype').text().trim() == '课程' && $(e).find('.requiredY').length > 0 && $(e).find('.operation .finish').length == 0).find('.operation button');
classList = classList.filter((i,e)=> !this.playClassList.has(this.obtainId(e)))
if (classList.length == 0) {
console.log('本部分播放完成,尝试加载下一部分 (可能是被锁的)');
this.sectionIdx++;
if(this.sectionIdx > sections.length -1) {
console.log('学习完成');
this.stop();
return;
}
setTimeout(num => this.openClassPage(num), 2000, num);
return
}
this.playClass(classList, num, this.obtainId);
}
// 考虑用课程名计算
obtainId(e) {
let name = $(e).closest('.activeContent').find('.content .bname').text().trim();
return md5(name);
}
}
class GetCourseIdFilter extends Filter {
constructor(_play) {
super();
this._play = _play;
}
applyRequestFilter(url, method, argument) {
if((url.match("course-study/course-info/related") || url.match("course-study/course-front/is-archive")) && url.indexOf('?') > 0) {
url.split('?')[1].split('&').forEach(e=> {
if(e.indexOf('courseId=')>=0) {
this._play.courseId = e.replaceAll('courseId=','');
}
});
}
return argument;
}
}
// 课程播放页面
class DetailPlayPage extends AbstractPlayPage {
constructor() {
super();
this.courseId = this.pageId
this.state = 0;
}
filterList() {
// return super.filterList();
return [...super.filterList(), new GetCourseIdFilter(this)];
}
start() {
// 多开判断
// unsafeWindow.WebSocket = undefined;
// window.WebSocket = undefined
let nsi,si;
// 获取课程名
nsi = setInterval(()=>{
console.log('this.name', this.name)
if($('.course-title .title .course-title-text').length > 0) {
this.name = $('.course-title .title .course-title-text').text().trim();
clearInterval(nsi);
nsi = null;
// 还没判断到
if (this.state == 0 && !this.subjectId) {
if (!!(this.subjectId = GM_getValue(CONFIG.keys.subject + md5(this.name)))) {
// 告知已经成功打开页面
clearInterval(si);
si = null;
this.state = 1;
this.courseId = md5(this.name);
GM_sendMessage(CONFIG.keys.courseStart + this.subjectId , window.location.href, this.courseId)
this.checkFinished();
}
}
}
}, 500)
// 判断是否插件打开的
si = setInterval(()=>{
console.log('this.courseId', this.courseId)
if(this.courseId) {
clearInterval(si);
si = null;
if (this.state == 0 && !this.subjectId) {
if (!!(this.subjectId = GM_getValue(CONFIG.keys.subject + this.courseId))) {
// 告知已经成功打开页面
clearInterval(nsi);
nsi = null;
this.state = 1;
GM_sendMessage(CONFIG.keys.courseStart + this.subjectId , window.location.href, this.courseId)
this.checkFinished();
}
}
}
}, 500)
setTimeout(()=>{
if (si != null || nsi != null || this.state == 0) {
// 超时未检测到是否插件打开, 增加开始按钮
this.addButton();
}
clearInterval(si);
clearInterval(nsi);
}, 1000 * 5)
}
addButton() {
this.addNav([{id:'startPlayer', name: '开始'}]);
$('#startPlayer').click(() => {
if(this.state == 0) {
this.state = 1;
$('#startPlayer').html('停止');
this.playMedal();
this.checkFinished();
console.log("手动控制开始播放");
} else {
this.state = 0;
$('#startPlayer').html('开始');
console.log("停止播放");
}
})
}
checkFinished() {
let status = 0;
const a = setInterval(() => {
console.log("正在循环判断课程是否完成");
this.playMedal();
let cIndex = 0, nextIndex = 0;
let dls = $("dl.chapter-list-box");
$.each(dls,(i,item)=>{
if($(item).hasClass('focus')){
cIndex = i;
// if($(item).find(".section-item .pointer span").text() == '重新学习'
// || $(item).find('.section-item div.sub-text.focus').text() == '考试'){
// nextIndex = cIndex + 1;
// }
let compulsory = $(item).find(".section-item-wrapper .section-item .sub-text:nth-child(1)");
let type = $(item).find(".section-item-wrapper .section-item .sub-text:nth-child(2)");
let remainder = $(item).find(".section-item-wrapper .section-item .pointer");
let reText = $(remainder).find('span').text() || '';
if ($(compulsory).text() != "必修" || ($(remainder).length == 1 && reText.indexOf('需再学') < 0 && reText.indexOf('需学') < 0) || $(type).text() == '考试') {
nextIndex = cIndex + 1;
}
}
})
console.log("当前学习到第" + (cIndex + 1) + "节课," + (cIndex < nextIndex ? '已学完,开始下一节' : '正在学习'));
if(cIndex < nextIndex && dls.length > nextIndex){
// 开始学下一节
$(dls[nextIndex]).click();
return;
}
if ( ($("div.anew-text").length != 0 && $("div.anew-text").text() == "您已完成该课程的学习") || cIndex < nextIndex) {
console.log("该课程已经完成,开始通信");
window.clearInterval(a);
status = 1;
GM_sendMessage(CONFIG.keys.courseClose + this.subjectId, window.location.href, {status: status, courseId: this.courseId, message: '播放完成' });
window.close();
}
}, 1000 * 15);
window.addEventListener('beforeunload', (event) => {
// 异常关闭
if (status == 0) {
GM_sendMessage(CONFIG.keys.courseClose + this.subjectId, window.location.href, {status: -1, courseId: this.courseId, message: '异常关闭' });
}
});
}
playMedal() {
if ($("button.videojs-referse-btn").length != 0) {
if ($("span.vjs-control-text:contains(', opens captions settings dialog')").length != 0) {
$("span.vjs-control-text:contains(', opens captions settings dialog')").remove();
}
// $("span.vjs-control-text").click();
$("button.vjs-paused").click();
$("button.videojs-referse-btn").click();
}
// 两倍数可被正常统计 最大2.2
let $video = $('#D200container video');
if($video.length > 0 && $video[0].playbackRate < 2.1) {
$video[0].playbackRate = 2.1;
}
}
}
// 在文档页面
class ResourcePlayPage extends DetailPlayPage {
checkFinished() {
let status = 0;
const a = setInterval(() => {
window.clearInterval(a);
status = 1;
GM_sendMessage(CONFIG.keys.courseClose + this.subjectId, window.location.href, {status: status, courseId: this.courseId, message: '播放完成' });
window.close();
}, 1000 * 20);
window.addEventListener('beforeunload', (event) => {
// 异常关闭
if (status == 0) {
GM_sendMessage(CONFIG.keys.courseClose + this.subjectId, window.location.href, {status: -1, courseId: this.courseId, message: '异常关闭' });
}
});
}
}
return {
start: () => {
const play = eval(`new ${pageType()}PlayPage`);
play.start();
}
}
})(unsafeWindow.jQuery || window.jQuery);
(function () {
autoPlay.start();
})();