Greasy Fork is available in English.

BilibiliTimer - B站H5播放器全屏下实时显示当前系统时间和播放进度

B站H5播放器全屏模式下实时显示当前系统时间和播放进度

Mint 2019.08.11.. Lásd a legutóbbi verzió

// ==UserScript==
// @name         BilibiliTimer - B站H5播放器全屏下实时显示当前系统时间和播放进度
// @version      3.1.0
// @description  B站H5播放器全屏模式下实时显示当前系统时间和播放进度
// @author       AnnAngela
// @namespace    https://greasyfork.org/users/129402
// @mainpage     https://greasyfork.org/zh-CN/scripts/30367-bilibilitimer
// @supportURL   https://greasyfork.org/zh-CN/scripts/30367-bilibilitimer/feedback
// @license      GNU General Public License v3.0 or later
// @compatible   chrome
// @match        *://www.bilibili.com/video/av*
// @match        *://www.bilibili.com/watchlater*
// @match        *://www.bilibili.com/html/*layer.htm*
// @match        *://www.bilibili.com/blackboard/*layer.htm*
// @match        *://www.bilibili.com/bangumi/play/*
// @match        *://live.bilibili.com/*
// @run-at       document-start
// @grant        unsafeWindow
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @icon         
// @icon64       
// ==/UserScript==
/* eslint-disable no-magic-numbers */
/* global unsafeWindow, GM_info, GM_addStyle, GM_setValue, GM_getValue */
"use strict";
(function () {
    /* 防止重复加载 */
    if (unsafeWindow.BilibiliTimer) { return; }

    var multiValueKeys = ["exclude", "grant", "include", "match", "require", "resource"];
    var booleanValueKeys = ["noframes"];
    var script = { scriptMetaStr: GM_info.scriptMetaStr };
    GM_info.scriptMetaStr.split(/\n+/).forEach(function (str) {
        var string = str.replace(/^\s*\/\/\s*/, "");
        var _temp = string.match(/^@([a-z\d:-]+) +(.+)$/i);
        if (!_temp) { return; }
        var key = _temp[1],
            value = _temp[2].trim();
        if (multiValueKeys.includes(key)) {
            if (script[key]) {
                script[key].push(value);
            } else {
                script[key] = [value];
            }
        } else if (booleanValueKeys.includes(key)) {
            script[key] = true;
        } else {
            script[key] = value;
        }
    });

    var bilibiliTimerHelper = {
        window: undefined,
        openWin: function (win) {
            var self = this;
            if (self.window) { return self.window.focus(); }
            var doc = win.document;
            var w = win.innerWidth || doc.docElement.clientWidth || doc.body.clientWidth,
                h = win.innerHeight || doc.docElement.clientHeight || doc.body.clientHeight;
            self.window = win.open("about:blank", "bilibiliTimerSetting", "location=1,scrollbars=1,channelmode=1,width=800,height=400,left=" + (w * 0.5 - 500) + ",top=" + (h * 0.5 - 200));
            setTimeout(function () {
                self.window.document.title = script.name + " 设置页面";
                self.window.document.body.innerHTML = "<h1>" + script.name + ' 设置页面</h1><ul><li><input id="SystemTime" type="checkbox"><label for="SystemTime">显示系统时间</label></li><li><input id="Track" type="checkbox"><label for="Track">显示当前进度</label></li><li><input id="BufferTime" type="checkbox"><label for="BufferTime">显示加载进度/缓冲质量</label></li><li><input id="CurrentPageAndWatchlater" type="checkbox"><label for="CurrentPageAndWatchlater">显示当前分页或稍后再看信息</label></li><li><input id="AutoHide" type="checkbox"><label for="AutoHide">无操作时自动隐藏</label></li></ul><div style="margin-top:1em;"><button id="save">保存</button></div>';
                self.window.document.body.style.padding = "2em";
                win.BilibiliTimer.setting.on.forEach(function (set) {
                    (self.window.document.querySelector("#" + set) || {}).checked = true;
                });
                if (GM_getValue("autoHidden")) { self.window.document.querySelector("#AutoHide").checked = true; }
                self.window.document.querySelector("#save").addEventListener("click", function () {
                    var setting = {
                        on: [],
                        off: [],
                    };
                    Array.from(self.window.document.querySelectorAll("input")).forEach(function (ele) {
                        if (ele.id !== "AutoHide") { setting[ele.checked ? "on" : "off"].push(ele.id); }
                        else {
                            GM_setValue("autoHidden", ele.checked);
                        }
                    });
                    GM_setValue("setting", setting);
                    if (!self.window.document.querySelector("#saveSuccessfull")) {
                        var saveSuccessfull = self.window.document.createElement("span");
                        saveSuccessfull.id = "saveSuccessfull";
                        saveSuccessfull.style.marginLeft = "2em";
                        saveSuccessfull.style.transition = "opacity 1s ease-in-out";
                        saveSuccessfull.innerText = "保存成功!";
                        saveSuccessfull.dataset.removeTime = new Date().getTime() + 3000;
                        self.window.document.querySelector("#save").after(saveSuccessfull);
                        win.BilibiliTimer && win.BilibiliTimer.restart && win.BilibiliTimer.restart("Setting Update", null);
                    } else {
                        self.window.document.querySelector("#saveSuccessfull").removeTime = new Date().getTime() + 3000;
                        if (+self.window.document.querySelector("#saveSuccessfull").dataset.removeTime < new Date().getTime()) {
                            self.window.document.querySelector("#saveSuccessfull").style.opacity = "1";
                        }
                    }
                });
                self.window.setInterval(function () {
                    if (self.window.document.querySelector("#saveSuccessfull")) {
                        if (+self.window.document.querySelector("#saveSuccessfull").dataset.removeTime < new Date().getTime()) {
                            if (+self.window.document.querySelector("#saveSuccessfull").dataset.removeTime > new Date().getTime() - 4000) {
                                self.window.document.querySelector("#saveSuccessfull").style.opacity = "0";
                            } else {
                                self.window.document.querySelector("#saveSuccessfull").remove();
                            }
                        }
                    }
                }, 100);
                self.window.focus();
                self.window.addEventListener("beforeunload", function () {
                    self.window = undefined;
                });
            }, 0);

        },
        closeWin: function () {
            if (this.window) { this.window.close(); }
        },
    };
    unsafeWindow.addEventListener("beforeunload", function () {
        bilibiliTimerHelper.closeWin();
    });

    unsafeWindow.BilibiliTimerUninit = false;
    unsafeWindow.BilibiliTimer = {};
    var code = function code() {
        if (unsafeWindow.BilibiliTimerUninit || !unsafeWindow.jQuery) { return false; }
        if (!String.prototype.includes) {
            String.prototype.includes = function includes(s) {
                return this.indexOf(s) !== -1;
            };
        }
        unsafeWindow.BilibiliTimer = {};
        unsafeWindow.BilibiliTimer._loop_count = 0;
        unsafeWindow.BilibiliTimer.date = function bilibiliPlayerDate() {
            var _date = new Date();
            ["getDate", "getFullYear", "getHours", "getMilliseconds", "getMinutes", "getMonth", "getSeconds", "getTime", "getUTCDate", "getUTCFullYear", "getUTCHours", "getUTCMilliseconds", "getUTCMinutes", "getUTCMonth", "getUTCSeconds", "getYear"].forEach(function (key) {
                _date[key] = function () {
                    var result = Date.prototype[key].apply(_date, arguments);
                    if (key.includes("Month")) { result++; }
                    if (typeof result === "number" && (result + "").length === 1) { return "0" + result; }
                    return result + "";
                };
            });
            return _date;
        };
        try {
            unsafeWindow.BilibiliTimer.isEmbedded = location.host === "www.bilibili.com" && ["/blackboard/html5player.html", "/blackboard/player.html"].includes(location.pathname) && top !== window && top.location.host.includes("bilibili.com");
        } catch (_) {
            unsafeWindow.BilibiliTimer.isEmbedded = false;
        }
        unsafeWindow.BilibiliTimer.realWindow = unsafeWindow.BilibiliTimer.isEmbedded ? top : window;
        unsafeWindow.BilibiliTimer.isNewPlayPage = (location.pathname.match(/\/video\/av(\d+)/) || [0, -1])[1] !== -1 && parseInt((document.cookie.match(/stardustvideo=(-?[0-9]+)/i) || [0, "-1"])[1]) > 0 || (location.pathname.match(/\/bangumi\/play\/ep(\d+)/) || [0, -1])[1] !== -1 && document.cookie.includes("stardustpgcv") && document.cookie.match(/stardustpgcv=(-?\d+)/)[1] === "0606";
        unsafeWindow.BilibiliTimer.isNewBangumiPlayPage = (location.pathname.match(/\/bangumi\/play\/ep(\d+)/) || [0, -1])[1] !== -1 && document.cookie.includes("stardustpgcv") && document.cookie.match(/stardustpgcv=(-?\d+)/)[1] === "0606";
        unsafeWindow.BilibiliTimer.isLive = function bilibiliIsLive(a, b) {
            return location.host === "live.bilibili.com" ? a !== undefined ? a : true : b !== undefined ? b : false;
        };
        unsafeWindow.BilibiliTimer.selector = unsafeWindow.BilibiliTimer.isLive(
            {
                container: ".bilibili-live-player-video-area",
                controller: ".bilibili-live-player-video-controller",
                fullscreenSendbar: null,
                settingButton: '.attention-btn-ctnr > .left-part.dp-i-block.pointer.p-relative[role="button"]',
                autoHideCheck: null,
                pageTitle: null,
                watchlaterPageTitle: null,
                watchlaterVideoTitle: null,
                bangumiTitle: null,
                bangumiPartTitle: null,
            },
            {
                container: ".bilibili-player-video-wrap",
                controller: ".bilibili-player-video-control",
                fullscreenSendbar: ".bilibili-player-video-sendbar.active",
                settingButton: unsafeWindow.BilibiliTimer.isNewPlayPage ? "#arc_toolbar_report > .ops, #toolbar_module" : ".bgray-btn-wrap .bgray-btn.show",
                autoHideCheck: ".bilibili-player-no-cursor",
                pageTitle: unsafeWindow.BilibiliTimer.isNewPlayPage ? ".cur-list .on" : "#v_multipage .item.on",
                watchlaterPageTitle: ".bilibili-player-auxiliary-area .bilibili-player-watchlater-part-item[data-state-play=true] .bilibili-player-watchlater-plist-chapter",
                watchlaterVideoTitle: ".bilibili-player-auxiliary-area .bilibili-player-watchlater-item[data-state-play=true] .bilibili-player-watchlater-info-title",
                bangumiTitle: unsafeWindow.BilibiliTimer.isNewPlayPage ? "#h1_module" : "#app .bangumi-info .info-title > a > h2",
                bangumiPartTitle: unsafeWindow.BilibiliTimer.isNewPlayPage ? "#eplist_module .ep-item.cursor" : "#app .bangumi-info .bangumi-list-wrapper .episode-list .episode-item.on",
            }
        );
        unsafeWindow.BilibiliTimer.selector.fullscreen = (function () {
            try {
                unsafeWindow.document.querySelector(":fullscreen"); // Let's try the standard selector
                return ":fullscreen";
            } catch (e) {
                // Just keep going
            }
            try {
                unsafeWindow.document.querySelector(":-webkit-full-screen"); // Maybe we should try webkit prefix
                return ":-webkit-full-screen";
            } catch (e) {
                // Just keep going
            }
            try {
                unsafeWindow.document.querySelector(":-moz-full-screen"); // Or firefox prefix
                return ":-moz-full-screen";
            } catch (e) {
                // Just keep going
            }
            try {
                unsafeWindow.document.querySelector(":-ms-fullscreen"); // Why you still using IE¿
                return ":-ms-fullscreen";
            } catch (e) {
                // Well, I think we should give up
                console.error("BilibiliTimer: Believe it or not, there's no selector which can bring us the fullscreen node.");
                if (unsafeWindow.BilibiliTimer && unsafeWindow.BilibiliTimer.uninit) { unsafeWindow.BilibiliTimer.uninit(); }
                unsafeWindow.BilibiliTimerUninit = true;
            }
        })();
        unsafeWindow.BilibiliTimer.classList = unsafeWindow.BilibiliTimer.isLive(
            {
                timer: "bilibili-live-player-video-info-container",
                closeButton: "bilibili-live-player-video-info-close",
                panel: "bilibili-live-player-video-info-panel",
                restartButton: "live-icon-reload",
                settingButton: "left-part dp-i-block pointer p-relative live-skin-highlight-bg BilibiliTimerSettingButton BilibiliTimerSettingButton-live",
            },
            {
                timer: "bilibili-player-video-info-container",
                closeButton: "bilibili-player-video-info-close",
                panel: "bilibili-player-video-info-panel",
                restartButton: "bilibili-player-iconfont icon-24repeaton",
                settingButton: unsafeWindow.BilibiliTimer.isNewPlayPage ? "BilibiliTimerSettingButton" : "bgray-btn show BilibiliTimerSettingButton",
            }
        );
        unsafeWindow.BilibiliTimer.setting = GM_getValue("setting", {
            on: ["SystemTime", "Track", "BufferTime", "CurrentPageAndWatchlater", "Watchlater"],
            off: [],
        });
        $("body").attr("data-bilibiliTimerSettingOn", unsafeWindow.BilibiliTimer.setting.on.join(","));
        unsafeWindow.BilibiliTimer.closeButtonText = unsafeWindow.BilibiliTimer.isLive("x", "[×]");
        unsafeWindow.BilibiliTimer.smallModeButtonText = unsafeWindow.BilibiliTimer.isLive("_", "[_]");
        unsafeWindow.BilibiliTimer.globallock = false;
        unsafeWindow.BilibiliTimer.widthSet = false;
        unsafeWindow.BilibiliTimer.onResizing = 0;
        if (unsafeWindow.BilibiliTimer.selector.autoHideCheck) { unsafeWindow.BilibiliTimer.mousemoveCount = 0; }
        unsafeWindow.BilibiliTimer.getControllerTop = function BilibiliTimerGetControllerTop() {
            var controller = $(unsafeWindow.BilibiliTimer.selector.controller);
            if (controller.closest(".mode-miniscreen")[0]) { return $(window).height(); }
            var _top = $(window).height() - controller.height();
            var fullscreenSendbar = $(unsafeWindow.BilibiliTimer.selector.fullscreenSendbar);
            if (fullscreenSendbar[0]) { _top -= fullscreenSendbar.outerHeight(true); }
            return _top;
        };
        $(window).on("resize.BilibiliTimer", function () {
            try {
                unsafeWindow.BilibiliTimer.onResizing = 1;
            } catch (e) { /* */ }
        });
        $(document).on({
            "mousemove.BilibiliTimer": function (e) {
                if (unsafeWindow.BilibiliTimer && unsafeWindow.BilibiliTimer.timer) {
                    if (unsafeWindow.BilibiliTimer.onMousedown) {
                        var maxTop = unsafeWindow.BilibiliTimer.getControllerTop() - unsafeWindow.BilibiliTimer.timer.outerHeight() - 10;
                        var maxLeft = $(window).width() - unsafeWindow.BilibiliTimer.timer.outerWidth() - 10;
                        unsafeWindow.BilibiliTimer.timer.css({
                            left: Math.max(Math.min(unsafeWindow.BilibiliTimer.timer.data("baseOffset").left + e.clientX, maxLeft), 10),
                            top: Math.max(Math.min(unsafeWindow.BilibiliTimer.timer.data("baseOffset").top + e.clientY, maxTop), 10),
                        });
                        unsafeWindow.getSelection().removeAllRanges();
                    }
                    if (unsafeWindow.BilibiliTimer.selector.autoHideCheck) { unsafeWindow.BilibiliTimer.mousemoveCount = 0; }
                }
            },
            "mouseup.BilibiliTimer": function () {
                if (unsafeWindow.BilibiliTimer && unsafeWindow.BilibiliTimer.timer && unsafeWindow.BilibiliTimer.onMousedown) {
                    unsafeWindow.BilibiliTimer.onMousedown = false;
                    GM_setValue("offset", {
                        top: unsafeWindow.BilibiliTimer.timer.css("top"),
                        left: unsafeWindow.BilibiliTimer.timer.css("left"),
                    });
                }
            },
        });
        unsafeWindow.BilibiliTimer.template = {};
        var timer = unsafeWindow.BilibiliTimer.template.timer = $("<div/>");
        timer.attr("id", "BilibiliTimer").addClass(unsafeWindow.BilibiliTimer.classList.timer);
        var closeButton = unsafeWindow.BilibiliTimer.template.closeButton = $("<a/>");
        closeButton.text(unsafeWindow.BilibiliTimer.closeButtonText).attr({
            href: "javascript:void(0);",
            id: "BilibiliTimerCloseButton",
        });
        closeButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton);
        var restartButton = unsafeWindow.BilibiliTimer.template.restartButton = $("<a/>");
        restartButton.attr({
            href: "javascript:void(0);",
            id: "BilibiliTimerRestartButton",
            title: "如果发现浮窗出现问题,\n例如无法正常拖动,无法正常显示时间等,\n请点击该按钮重建浮窗尝试修复!",
        });
        restartButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton).addClass(unsafeWindow.BilibiliTimer.classList.restartButton);
        var smallModeButton = unsafeWindow.BilibiliTimer.template.smallModeButton = $("<a/>");
        smallModeButton.text(unsafeWindow.BilibiliTimer.smallModeButtonText).attr({
            href: "javascript:void(0);",
            id: "BilibiliTimerSmallModeButton",
            title: "点击切换显示模式",
        });
        smallModeButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton);
        var panel = unsafeWindow.BilibiliTimer.template.panel = $("<div/>");
        panel.addClass(unsafeWindow.BilibiliTimer.classList.panel);
        panel.append('<div class="info-line BilibiliTimerDisplayInlineInSmallMode" id="BilibiliTimerSystemTime"><span class="info-title">系统时间:</span><span class="info-data" id="BilibiliTimerNowTime"> - </span></div>');
        panel.append(unsafeWindow.BilibiliTimer.isLive('<div class="info-line BilibiliTimerDisplayInlineInSmallMode" id="BilibiliTimerTrack"><span class="info-title">缓冲质量:</span><span class="info-data">当前缓冲时长 <span id="BilibiliTimerVideoBufferedTimeRange"> - </span>s</span></div>', '<div class="info-line BilibiliTimerDisplayInlineInSmallMode" id="BilibiliTimerTrack"><span class="info-title">播放进度:</span><span class="info-data"><span id="BilibiliTimerVideoTime"> - </span><span class="BilibiliTimerHideInSmallMode">(已播放<span id="BilibiliTimerVideoTimePercents"> - </span>%)</span></span></div><div class="info-line BilibiliTimerDisplayInlineInSmallMode" id="BilibiliTimerBufferTime"><span class="info-title">加载进度:</span><span class="info-data"><span class="BilibiliTimerHideInSmallMode"><span id="BilibiliTimerVideoBufferedTime"> - </span>(剩余</span>缓冲时长 <span id="BilibiliTimerVideoBufferedTimeRange"> - </span>s<span class="BilibiliTimerHideInSmallMode">,已缓冲<span id="BilibiliTimerVideoBufferedTimePercents"> - </span>%)</span></span></div><div class="info-line" id="BilibiliTimerCurrentPageAndWatchlater" style="display: none;"><span class="info-title" id="BilibiliTimerTitleDescription">当前分页:</span><span class="info-data" id="BilibiliTimerTitle"> - </span></div>'));
        var settingButton = unsafeWindow.BilibiliTimer.template.settingButton = $(unsafeWindow.BilibiliTimer.isNewPlayPage && !unsafeWindow.BilibiliTimer.isNewBangumiPlayPage ? "<span/>" : "<div/>");
        settingButton.addClass(unsafeWindow.BilibiliTimer.classList.settingButton);
        settingButton.css(unsafeWindow.BilibiliTimer.isLive("width", unsafeWindow.BilibiliTimer.isNewPlayPage ? "width" : "height"), "auto");
        settingButton.html("BilibiliTimer 设置");
        unsafeWindow.BilibiliTimer.init = function BilibiliTimerInit() {
            if (unsafeWindow.BilibiliTimerUninit) { return false; }
            if (!$(unsafeWindow.BilibiliTimer.selector.container)[0] && ++unsafeWindow.BilibiliTimer._loop_count > 150) { return unsafeWindow.BilibiliTimer.uninit(); }
            unsafeWindow.BilibiliTimer.onResizing = 0;
            unsafeWindow.BilibiliTimer.widthSet = false;
            unsafeWindow.BilibiliTimer.timer = unsafeWindow.BilibiliTimer.template.timer.clone();
            unsafeWindow.BilibiliTimer.closeButton = unsafeWindow.BilibiliTimer.template.closeButton.clone();
            unsafeWindow.BilibiliTimer.restartButton = unsafeWindow.BilibiliTimer.template.restartButton.clone();
            unsafeWindow.BilibiliTimer.smallModeButton = unsafeWindow.BilibiliTimer.template.smallModeButton.clone();
            unsafeWindow.BilibiliTimer.panel = unsafeWindow.BilibiliTimer.template.panel.clone();
            unsafeWindow.BilibiliTimer.settingButton = unsafeWindow.BilibiliTimer.template.settingButton.clone();
            unsafeWindow.BilibiliTimer.timer.append(unsafeWindow.BilibiliTimer.closeButton).append(unsafeWindow.BilibiliTimer.restartButton).append(unsafeWindow.BilibiliTimer.smallModeButton).append(unsafeWindow.BilibiliTimer.panel);
            var title = null,
                description = null;
            if ($(unsafeWindow.BilibiliTimer.selector.pageTitle)[0]) {
                var link = $(unsafeWindow.BilibiliTimer.selector.pageTitle).children("a:visible").clone();
                if (unsafeWindow.BilibiliTimer.isNewPlayPage) { link.find(".s1").text(link.find(".s1").text().replace("P", "") + "、"); }
                title = link.text().trim().replace(/^(\d+)、/, "$1/" + $(unsafeWindow.BilibiliTimer.selector.pageTitle).parent().children().length + ":");
            } else if ($(unsafeWindow.BilibiliTimer.selector.watchlaterPageTitle)[0]) {
                description = '当前分页:<div id="BilibiliTimerTitleDescriptionLabel">(稍后再看)</div>';
                title = $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).closest("li").index() + 1 + "/" + $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).closest("li").parent().children().length + ":" + $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).text() + "<br>" + ($(unsafeWindow.BilibiliTimer.selector.watchlaterPageTitle).closest("li").index() + 1) + "/" + $(unsafeWindow.BilibiliTimer.selector.watchlaterPageTitle).closest("li").parent().children().length + ":" + $(unsafeWindow.BilibiliTimer.selector.watchlaterPageTitle).text();
            } else if ($(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle)[0]) {
                description = '当前视频:<div id="BilibiliTimerTitleDescriptionLabel">(稍后再看)</div>';
                title = $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).closest("li").index() + 1 + "/" + $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).closest("li").parent().children().length + ":" + $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle).text();
            } else if ($(unsafeWindow.BilibiliTimer.selector.bangumiTitle)[0]) {
                description = "当前番剧:";
                title = $(unsafeWindow.BilibiliTimer.selector.bangumiTitle).text();
                if ($(unsafeWindow.BilibiliTimer.selector.bangumiPartTitle)[0]) {
                    var cloneNode = $(unsafeWindow.BilibiliTimer.selector.bangumiPartTitle).clone();
                    if (!unsafeWindow.BilibiliTimer.isNewPlayPage) { cloneNode.find(".ep-index").text(cloneNode.find(".ep-index").text() + ":"); }
                    else {
                        title = title.replace(":" + cloneNode.text().trim(), "");
                        cloneNode.text(cloneNode.text().trim().replace(/^(第\d+[话集]) /, "$1:"));
                    }
                    title += "<br>" + "(" + cloneNode.text() + ")";
                }
            }
            if (title) { unsafeWindow.BilibiliTimer.timer.find("#BilibiliTimerTitle").html(title).parent().removeAttr("style"); }
            if (description) { unsafeWindow.BilibiliTimer.timer.find("#BilibiliTimerTitleDescription").html(description); }
            (function loop(BilibiliTimer, $) {
                if (BilibiliTimer.isNewPlayPage && [
                    $("#arc_toolbar_report > .ops > .like").text(),
                    $("#arc_toolbar_report > .ops > .coin, #toolbar_module .coin-info span").text(),
                    $("#arc_toolbar_report > .ops > .collect").text(),
                ].includes("--")) {
                    return setTimeout(loop, 200, BilibiliTimer, $);
                }
                $(".BilibiliTimerSettingButton").remove();
                $(BilibiliTimer.realWindow.document.querySelector(BilibiliTimer.selector.settingButton))[BilibiliTimer.isNewPlayPage ? "append" : "before"](BilibiliTimer.settingButton);
                if (unsafeWindow.BilibiliTimer.isLive()) {
                    $(BilibiliTimer.realWindow.document.querySelector(BilibiliTimer.selector.settingButton)).css("border-radius", "unset");
                    BilibiliTimer.settingButton.css("border-radius", "4px 0 0 4px");
                }
            })(unsafeWindow.BilibiliTimer, $);
            unsafeWindow.BilibiliTimer.timer.on("mousedown", function (e) {
                var baseX = Math.max(e.clientX, 0);
                var baseY = Math.max(e.clientY, 0);
                var baseOffsetX = Math.max(parseInt(unsafeWindow.BilibiliTimer.timer.css("left")), 0);
                var baseOffsetY = Math.max(parseInt(unsafeWindow.BilibiliTimer.timer.css("top")), 0);
                unsafeWindow.BilibiliTimer.timer.data("baseOffset", {
                    left: baseOffsetX - baseX,
                    top: baseOffsetY - baseY,
                });
                unsafeWindow.BilibiliTimer.onMousedown = true;
            });
            unsafeWindow.BilibiliTimer.closeButton.on("click", function () {
                unsafeWindow.BilibiliTimer.globallock = true;
                unsafeWindow.BilibiliTimer.timer.dequeue().clearQueue().css("opacity", "0").delay(370).queue(function () {
                    timer.hide().dequeue();
                });
            });
            unsafeWindow.BilibiliTimer.restartButton.on("click", function () {
                unsafeWindow.BilibiliTimer.restart("User order", "");
            });
            unsafeWindow.BilibiliTimer.smallModeButton.on("click", function () {
                $("body").toggleClass("BilibiliTimerSmallMode");
                localStorage.setItem("BilibiliTimerSmallMode", $("body").is(".BilibiliTimerSmallMode") ? "true" : "false");
            });
            if (localStorage.getItem("BilibiliTimerSmallMode") === "true") {
                $("body").addClass("BilibiliTimerSmallMode");
            }
            unsafeWindow.BilibiliTimer.settingButton.on("click", function () {
                bilibiliTimerHelper.openWin(unsafeWindow);
            });
            if (!GM_getValue("offset")) {
                unsafeWindow.BilibiliTimer.timer.css({
                    left: "auto",
                    right: "10px",
                    top: "10px",
                });
            } else { unsafeWindow.BilibiliTimer.timer.css(GM_getValue("offset")); }
            $(unsafeWindow.BilibiliTimer.selector.container).append(unsafeWindow.BilibiliTimer.timer);
            $(window).resize();
        };
        unsafeWindow.BilibiliTimer.globalWatcher = function BilibiliTimerGlobalWatcher() {
            if (unsafeWindow.BilibiliTimerUninit) { return false; }
            var timer = unsafeWindow.BilibiliTimer.timer;
            if (!timer || !timer[0]) {
                unsafeWindow.BilibiliTimer.init();
                return;
            }
            if ($("object#player_placeholder, object#player_object")[0]) {
                unsafeWindow.BilibiliTimer.uninit();
                return;
            }
            if (!timer.closest("body")[0]) {
                unsafeWindow.BilibiliTimer.restart("Timer did not exist in document.body", timer.closest("body")[0]);
                return;
            }
            if (unsafeWindow.BilibiliTimer.realWindow.document.querySelector(unsafeWindow.BilibiliTimer.selector.fullscreen)) {
                if (GM_getValue("autoHidden") && (unsafeWindow.BilibiliTimer.selector.autoHideCheck ? $(unsafeWindow.BilibiliTimer.selector.autoHideCheck)[0] : unsafeWindow.BilibiliTimer.mousemoveCount >= 3)) { unsafeWindow.BilibiliTimer.autoHidden = true; }
                else { unsafeWindow.BilibiliTimer.autoHidden = false; }
                if (!unsafeWindow.BilibiliTimer.globallock && !unsafeWindow.BilibiliTimer.autoHidden) {
                    if (!timer.is(":visible") || timer.css("opacity") !== "1") {
                        timer.dequeue().clearQueue().show().css("opacity", "1");
                    }
                    if (unsafeWindow.BilibiliTimer.onResizing === 2) {
                        unsafeWindow.BilibiliTimer.onResizing = 0;
                        var maxTop = unsafeWindow.BilibiliTimer.getControllerTop() - timer.outerHeight() - 10;
                        var maxLeft = $(window).width() - timer.outerWidth() - 10;
                        timer.animate({
                            left: Math.max(Math.min(parseInt(timer.css("left")), maxLeft), 10),
                            top: Math.max(Math.min(parseInt(timer.css("top")), maxTop), 10),
                        }, 370);
                        GM_setValue("offset", {
                            top: timer.css("top"),
                            left: timer.css("left"),
                        });
                    } else if (unsafeWindow.BilibiliTimer.onResizing === 1) {
                        unsafeWindow.BilibiliTimer.onResizing = 2;
                    }
                    var date = unsafeWindow.BilibiliTimer.date();
                    timer.find("#BilibiliTimerNowTime").html('<span class="BilibiliTimerHideInSmallMode BilibiliTimerWhiteSpacePre">' + date.getFullYear() + "-" + date.getMonth() + "-" + date.getDate() + " </span>" + date.getHours() + ":" + date.getMinutes() + ":" + date.getSeconds());
                    if (!unsafeWindow.BilibiliTimer.widthSet) { unsafeWindow.BilibiliTimer.heightCalc(); }
                }
            } else {
                unsafeWindow.BilibiliTimer.onResizing = 0;
                unsafeWindow.BilibiliTimer.globallock = false;
                timer.dequeue().clearQueue().css("opacity", "0").delay(370).queue(function () {
                    timer.hide().dequeue();
                });
            }
            var video = $("video")[0];
            if (!video) {
                unsafeWindow.BilibiliTimer.restart("No Video", null);
                return;
            }
            if (!video.dataset.isTrusted) {
                video.dataset.isTrusted = "true";
                video.addEventListener("timeupdate", unsafeWindow.BilibiliTimer.videoPlayListener);
                video.addEventListener("progress", unsafeWindow.BilibiliTimer.videoProgressListener);
            }
            if ((timer.find("#BilibiliTimerVideoTimePercents").text() + timer.find("#BilibiliTimerVideoBufferedTimeRange").text() + timer.find("#BilibiliTimerVideoTime").text()).includes("-")) {
                unsafeWindow.BilibiliTimer.videoPlayListener({
                    target: video[0],
                });
                unsafeWindow.BilibiliTimer.videoProgressListener({
                    target: video[0],
                });
            }
        };
        unsafeWindow.BilibiliTimer.autoHideWatcher = function BilibiliTimerAutoHideWatcher() {
            if (!GM_getValue("autoHidden")) { return; }
            if (unsafeWindow.BilibiliTimer.selector.autoHideCheck ? $(unsafeWindow.BilibiliTimer.selector.autoHideCheck)[0] : unsafeWindow.BilibiliTimer.mousemoveCount >= 3) {
                unsafeWindow.BilibiliTimer.autoHidden = true;
                unsafeWindow.BilibiliTimer.timer.dequeue().clearQueue().css("opacity", "0").delay(370).queue(function () {
                    timer.hide().dequeue();
                });
            }
            if (unsafeWindow.BilibiliTimer.selector.autoHideCheck) { unsafeWindow.BilibiliTimer.mousemoveCount++; }
        };
        unsafeWindow.BilibiliTimer.lazyWatcher = function BilibiliTimerLazyWatcher() {
            if (!unsafeWindow.BilibiliTimer || !unsafeWindow.BilibiliTimer.timer) { return; }
            var timer = unsafeWindow.BilibiliTimer.timer;
            unsafeWindow.BilibiliTimer.heightCalc();
            var flag, node;
            if ($(unsafeWindow.BilibiliTimer.selector.pageTitle)[0]) {
                if (!timer.find("#BilibiliTimerTitleDescription")[0]) {
                    flag = "TitleDescription unset";
                    node = timer.find("#BilibiliTimerTitleDescription")[0];
                } else if (timer.is(":visible") && timer.find("#BilibiliTimerTitleDescription").is(":hidden")) {
                    flag = "TitleDescription invisible";
                    node = timer.find("#BilibiliTimerTitleDescription")[0];
                }
            } else if ($(unsafeWindow.BilibiliTimer.selector.watchlaterPageTitle)[0] || $(unsafeWindow.BilibiliTimer.selector.watchlaterVideoTitle)[0]) {
                if (!timer.find("#BilibiliTimerTitleDescriptionLabel")[0]) {
                    flag = "TitleDescriptionLabel unset";
                    node = timer.find("#BilibiliTimerTitleDescriptionLabel")[0];
                } else if (timer.is(":visible") && timer.find("#BilibiliTimerTitleDescriptionLabel").is(":hidden")) {
                    flag = "TitleDescriptionLabel invisible";
                    node = timer.find("#BilibiliTimerTitleDescriptionLabel")[0];
                }
            } else if ($(unsafeWindow.BilibiliTimer.selector.bangumiTitle)[0]) {
                if (!timer.find("#BilibiliTimerTitleDescription").text().includes("当前番剧:")) {
                    flag = "TitleDescriptionLabel did not include bangumi";
                    node = timer.find("#BilibiliTimerTitleDescription")[0];
                } else if (!/第\d+[话集]/.test(timer.find("#BilibiliTimerTitle").text())) {
                    flag = "bangumiPartTitle is not correct";
                    node = timer.find("#BilibiliTimerTitle")[0];
                }
            } else if (!$("." + unsafeWindow.BilibiliTimer.classList.settingButton.split(" ").join("."))[0] && !location.pathname.includes("/blackboard/")) {
                flag = "SettingButton no found";
                node = undefined;
            }
            if (flag) {
                unsafeWindow.BilibiliTimer.restart(flag, node);
                return;
            }
        };
        unsafeWindow.BilibiliTimer.heightCalc = function BilibiliTimerWidthCalc() {
            if (!unsafeWindow.BilibiliTimer || !unsafeWindow.BilibiliTimer.timer) { return; }
            var timer = unsafeWindow.BilibiliTimer.timer;
            timer.find(".info-line").each(function () {
                if ($(this).css("height", "unset").height() === 0) { return; }
                var maxHeight = 0;
                $(this).css("height", "unset").children().each(function () {
                    var _height = $(this).height();
                    if ($(this).find("#BilibiliTimerTitleDescriptionLabel")[0]) { _height += $(this).find("#BilibiliTimerTitleDescriptionLabel").height(); }
                    if (maxHeight < _height) { maxHeight = _height; }
                });
                $(this).height(maxHeight);
            });
            unsafeWindow.BilibiliTimer.widthSet = true;
            $(window).resize();
        };
        unsafeWindow.BilibiliTimer.timeParse = function BilibiliTimerTimeParse(_time) {
            var time = parseInt(_time);
            var sec = time % 60,
                min = (time - sec) / 60;
            if (sec < 10) { sec = "0" + sec; }
            return min + ":" + sec;
        };
        unsafeWindow.BilibiliTimer.videoPlayListener = function BilibiliTimerVideoPlayListener(e) {
            if (!e.target) { return; }
            var video = e.target;
            var curTime = video.currentTime || 0;
            var durTime = video.duration || 0;
            if (!curTime && !durTime) { return; }
            if (!unsafeWindow.BilibiliTimer || !unsafeWindow.BilibiliTimer.timer) { return; }
            var timer = unsafeWindow.BilibiliTimer.timer;
            timer.find("#BilibiliTimerVideoTime").text(unsafeWindow.BilibiliTimer.timeParse(curTime) + " / " + unsafeWindow.BilibiliTimer.timeParse(durTime));
            if (timer.find("#BilibiliTimerVideoBufferedTime")[0]) {
                var end;
                try {
                    end = video.buffered.end(video.buffered.length - 1);
                } catch (_) {
                    try {
                        end = video.buffered.end(0);
                    } catch (_) {
                        return;
                    }
                }
                if (timer.find("#BilibiliTimerVideoBufferedTime").text() === " - ") { $(video).trigger("progress"); }
                timer.find("#BilibiliTimerVideoBufferedTimeRange").text((end - curTime).toFixed(0));
            }
            timer.find("#BilibiliTimerVideoTimePercents").text((curTime * 100 / durTime).toFixed(2));
        };
        unsafeWindow.BilibiliTimer.videoProgressListener = function BilibiliTimerVideoProgressListener(e) {
            if (!e.target) { return; }
            var video = e.target;
            var curTime = video.currentTime || 0;
            var durTime = video.duration || 0;
            if (!curTime && !durTime) { return; }
            if (!unsafeWindow.BilibiliTimer || !unsafeWindow.BilibiliTimer.timer) { return; }
            var timer = unsafeWindow.BilibiliTimer.timer;
            if (timer.find("#BilibiliTimerVideoBufferedTimeRange")[0]) {
                video = e.target;
                var end;
                try {
                    end = video.buffered.end(video.buffered.length - 1);
                } catch (_) {
                    try {
                        end = video.buffered.end(0);
                    } catch (_) {
                        return;
                    }
                }
                if (timer.find("#BilibiliTimerVideoBufferedTimeRange").text() === " - ") { unsafeWindow.BilibiliTimer.widthSet = false; }
                if (timer.find("#BilibiliTimerVideoTime").text() === " - ") { $(video).trigger("timeupdate"); }
                timer.find("#BilibiliTimerVideoBufferedTime").text(unsafeWindow.BilibiliTimer.timeParse(end));
                unsafeWindow.BilibiliTimer.timer.find("#BilibiliTimerVideoBufferedTimeRange").text((end - video.currentTime).toFixed(0));
                timer.find("#BilibiliTimerVideoBufferedTimePercents").text((end * 100 / video.duration).toFixed(2));
            }
            timer.find("#BilibiliTimerVideoTimePercents").text((curTime * 100 / durTime).toFixed(2));
        };
        unsafeWindow.BilibiliTimer.start = function BilibiliTimerStart() {
            if (unsafeWindow.BilibiliTimerUninit) { return false; }
            if (location.host === "bangumi.bilibili.com") { return false; }
            if (!unsafeWindow.BilibiliTimer.interval) { unsafeWindow.BilibiliTimer.interval = {}; }
            if (!unsafeWindow.BilibiliTimer.interval.globalWatcher) { unsafeWindow.BilibiliTimer.interval.globalWatcher = unsafeWindow.setInterval(unsafeWindow.BilibiliTimer.globalWatcher, 100); }
            if (!unsafeWindow.BilibiliTimer.interval.autoHideWatcher) { unsafeWindow.BilibiliTimer.interval.autoHideWatcher = unsafeWindow.setInterval(unsafeWindow.BilibiliTimer.autoHideWatcher, 1e3); }
            if (!unsafeWindow.BilibiliTimer.interval.lazyWatcher) { unsafeWindow.BilibiliTimer.interval.lazyWatcher = unsafeWindow.setInterval(unsafeWindow.BilibiliTimer.lazyWatcher, 5e3); }
            try {
                var video = $("video");
                setTimeout(function () {
                    unsafeWindow.BilibiliTimer.videoPlayListener({
                        target: video[0],
                    });
                    unsafeWindow.BilibiliTimer.videoProgressListener({
                        target: video[0],
                    });
                }, 100);
            } catch (_) {
                return;
            }
        };
        unsafeWindow.BilibiliTimer.restart = function BilibiliTimerRestart(reason, node) {
            for (var i in unsafeWindow.BilibiliTimer.interval) {
                if (unsafeWindow.BilibiliTimer.interval[i]) { unsafeWindow.clearInterval(unsafeWindow.BilibiliTimer.interval[i]); }
            }
            $(".BilibiliTimerSettingButton").remove();
            if (unsafeWindow.BilibiliTimerUninit) { return false; }
            var timer = $("#BilibiliTimer");
            if (timer[0]) {
                timer.dequeue().clearQueue().css("opacity", "0").delay(370).queue(function () {
                    unsafeWindow.BilibiliTimer.rebuild(reason, node);
                    timer.hide().dequeue();
                });
            } else { unsafeWindow.BilibiliTimer.rebuild(reason, node); }
        };
        unsafeWindow.BilibiliTimer.rebuild = function BilibiliTimerRebuild(reason, node) {
            console.groupCollapsed("BilibiliTimerRebuildTrace:", reason, node);
            console.trace();
            console.groupEnd();
            unsafeWindow.BilibiliTimer = undefined;
            $("#BilibiliTimer").remove();
            if (unsafeWindow.BilibiliTimerUninit) { return false; }
            setTimeout(function () {
                code();
            }, 0);
        };
        unsafeWindow.BilibiliTimer.uninit = function BilibiliTimerUninit() {
            for (var i in unsafeWindow.BilibiliTimer.interval) {
                if (unsafeWindow.BilibiliTimer.interval[i]) { unsafeWindow.clearInterval(unsafeWindow.BilibiliTimer.interval[i]); }
            }
            $(".BilibiliTimerSettingButton").remove();
            $("#BilibiliTimer").remove();
            unsafeWindow.BilibiliTimer = undefined;
            unsafeWindow.BilibiliTimerUninit = true;
        };
        unsafeWindow.BilibiliTimer.start();
    };
    var css = [
        "#BilibiliTimer {",
        "    cursor: move;",
        "    display: block;",
        "    transition-property: opacity, width;",
        "    transition-duration: .37s;",
        "    transition-timing-function: initial;",
        "    transition-delay: initial;",
        "    width: auto;",
        "}",
        "#BilibiliTimer.bilibili-player-video-info-container .bilibili-player-video-info-panel .info-line {",
        "    min-width: auto;",
        "}",
        ".bilibili-player-no-cursor #BilibiliTimer {",
        "    opacity: .73;",
        "    cursor: none;",
        "}",
        "#BilibiliTimer .info-title {",
        "    width: 6em;",
        "    margin: 0;",
        "}",
        "#BilibiliTimer .info-data {",
        "    max-width: 25em;",
        "    white-space: normal;",
        "    vertical-align: top;",
        "}",
        "#BilibiliTimerCloseButton.bilibili-live-player-video-info-close {",
        "    color: rgb(0, 0, 0);",
        "    padding: 0px;",
        "    height: 15px;",
        "    background: rgb(221, 221, 221);",
        "    width: 1em;",
        "    text-align: center;",
        "    top: 8px;",
        "}",
        "#BilibiliTimerRestartButton {",
        "    top: auto;",
        "    bottom: 10px;",
        "}",
        ".BilibiliTimerSettingButton-live {",
        "    -webkit-text-emphasis-color: white;",
        "    -webkit-text-fill-color: white;",
        "    -webkit-text-stroke-color: white;",
        "    background-color: rgb(35, 173, 229);",
        "    border: 0 solid white;",
        "    border-width: 0 1px;",
        "    caret-color: white;",
        "    color: white;",
        "    column-rule-color: white;",
        "    height: 22px;",
        "    line-height: 22px;",
        "    outline-color: white;",
        "    perspective-origin: 33px 11px;",
        "    text-align: center;",
        "    text-decoration: none solid white;",
        "    text-decoration-color: white;",
        "    transform-origin: 33px 11px;",
        "    vertical-align: middle;",
        "    padding: 0 .25em;",
        "}",
        ".bgray-btn-wrap .BilibiliTimer-hr {",
        "    border-style: inset;",
        "    border-width: 1px;",
        "    margin: 0.5em auto;",
        "    overflow: hidden;",
        "}",
        "#BilibiliTimerTitleDescription {",
        "    height: 100%;",
        "    position: relative;",
        "}",
        "#BilibiliTimerTitleDescriptionLabel {",
        "    width: 100%;",
        "    top: 50%;",
        "    left: 0;",
        "    position: absolute;",
        "}",
        ".bgray-btn.show.BilibiliTimerSettingButton {",
        "    writing-mode: tb;",
        "    padding: 7px 8px;",
        "    width: 14px;",
        "}",
        ".BilibiliTimerSettingButton {",
        "    user-select: none;",
        "}",
        "#arc_toolbar_report > .ops > .more > .more-ops-list {",
        "    width: auto;",
        "}",
        "#arc_toolbar_report > .ops > .more > .more-ops-list .BilibiliTimerSettingButton {",
        "    padding: 0 1em;",
        "}",
        "#toolbar_module .BilibiliTimerSettingButton {",
        "    display: block;",
        "    float: left;",
        "    height: 24px;",
        "    margin-left: 12px;",
        "    cursor: pointer;",
        "    line-height: 24px;",
        "    font-size: 14px;",
        "    color: #505050;",
        "}",
        "body:not([data-bilibilitimersettingon*=\"SystemTime\"]) #BilibiliTimerSystemTime,",
        "body:not([data-bilibilitimersettingon*=\"Track\"]) #BilibiliTimerTrack,",
        "body:not([data-bilibilitimersettingon*=\"BufferTime\"]) #BilibiliTimerBufferTime,",
        "body:not([data-bilibilitimersettingon*=\"CurrentPageAndWatchlater\"]) #BilibiliTimerCurrentPageAndWatchlater {",
        "    height: 0!important;",
        "    overflow: hidden;",
        "    width: 0;",
        "}",
        ".bilibili-player-video-info-container .bilibili-player-video-info-close#BilibiliTimerSmallModeButton {",
        "    right: 26px;",
        "}",
        ".BilibiliTimerSmallMode #BilibiliTimer {",
        "    padding-right: 54px;",
        "}",
        ".BilibiliTimerSmallMode .BilibiliTimerHideInSmallMode,",
        ".BilibiliTimerSmallMode .BilibiliTimerDisplayInlineInSmallMode .info-title {",
        "    display: none !important;",
        "}",
        ".BilibiliTimerSmallMode .BilibiliTimerDisplayInlineInSmallMode {",
        "    display: inline !important;",
        "}",
        ".BilibiliTimerSmallMode .BilibiliTimerDisplayInlineInSmallMode:first-child {",
        "    padding-left: 1em;",
        "}",
        ".BilibiliTimerSmallMode .BilibiliTimerDisplayInlineInSmallMode + .BilibiliTimerDisplayInlineInSmallMode::before {",
        '    content: " | ";',
        "}",
        ".BilibiliTimerWhiteSpacePre {",
        "    white-space: pre;",
        "}",
    ].join("\n");
    var c;
    c = unsafeWindow.setInterval(function () {
        if (!unsafeWindow.jQuery) { return false; }
        if (!unsafeWindow.document.querySelector(".bilibili-live-player-video-area, .bilibili-player-video-wrap") || !unsafeWindow.document.querySelector(".bilibili-live-player-video-area, .bilibili-player-video-wrap").querySelector("video")) { return; }
        if (unsafeWindow.BilibiliTimerUninit || unsafeWindow.BilibiliTimer.start) { return unsafeWindow.clearInterval(c); }
        if (/^https:\/\/www\.bilibili\.com\/video\/av\d+/i.test(location.href) && parseInt((unsafeWindow.document.cookie.match(/stardustvideo=(-?\d+)/) || [0, "-1"])[1]) > 0) {
            if ([
                unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .like") ? unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .like").innerText : "--",
                unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .coin") ? unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .coin").innerText : "--",
                unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .collect") ? unsafeWindow.document.querySelector("#arc_toolbar_report > .ops > .collect").innerText : "--",
            ].includes("--")) { return false; }
        }
        unsafeWindow.console.info("%c" + script.name + "@" + script.version + " by " + script.author + " is running!", "padding: 34px 66px 34px 66px; line-height: 64px; background:url('" + script.icon64 + "') top left no-repeat; background-position-y: 4px;");
        GM_addStyle(css);
        code();
    }, 100);
})();