中国电信网上大学知识中心自动挂课

鼠标移动到左上角图标点击开始,开启无人值守自动挂课模式!增加对活动中的专题班支持!增加2.1倍数(最大2.2,可能报异常)! 参考:HuangDingYun v.0.5

// ==UserScript==
// @name         中国电信网上大学知识中心自动挂课
// @namespace    http://tampermonkey.net/
// @version      1.3
// @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: `aaa`}]);
      $('#startAutoPlayer').click(() => {
        if(this.status == 0) {
          this.status = 1;
          $('#startAutoPlayer').html('停止');
          this.playClassList.clear();
          this.waitNum = 0;
          this.openClassPage(this.maxNum);
        } else {
          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));
      if (classList.length == 0) {
        console.log('播放完成');
      }

      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('学习完成');
        }
        $(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('学习完成');
          $('#startAutoPlayer').click();
          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.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();
})();