BiibiliTimer

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

目前为 2017-10-22 提交的版本。查看 最新版本

// ==UserScript==
// @name         BiibiliTimer
// @version      2.1.0
// @description  B站H5播放器全屏时实时显示当前系统时间和播放进度
// @author       AnnAngela
// @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        *://live.bilibili.com/*
// @compatible   chrome 自Chrome 50开始兼容
// @icon         
// @icon64       
// @run-at       document-start
// @grant        unsafeWindow
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @namespace    https://greasyfork.org/users/129402
// ==/UserScript==

(function() {
    'use strict';
    /* 防止重复加载 */
    if (unsafeWindow.BilibiliTimer) return;

    unsafeWindow.console.debug('%c' + GM_info.script.name + '@' + GM_info.script.version + ' by ' + GM_info.script.author + ' is running!', "padding: 32px 66px 32px 64px; line-height: 64px; background:url('" + GM_info.script.icon64 + "') top left no-repeat;");
    unsafeWindow.BilibiliTimerUninit = false;
    unsafeWindow.BilibiliTimerGM = {
        set: GM_setValue,
        get: GM_getValue,
        delete: GM_deleteValue
    };
    unsafeWindow.BilibiliTimer = {};
    /*
     * 出于实现上的考虑,以下代码被保存为纯文本,并通过unsafeWindow.eval运行,以操作真实环境下的DOM节点
     */
    var code = [
        "(function() {",
        "    'use strict';",
        "    if (window.BilibiliTimerUninit || !window.jQuery) return false;",
        "    if (!String.prototype.includes) String.prototype.includes = function includes(s) {",
        "        return this.indexOf(s) !== -1;",
        "    };",
        "    var BilibiliTimer = window.BilibiliTimer || (window.BilibiliTimer = {});",
        "    var LF = String.fromCharCode(10);",
        "    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;",
        "                else return result + '';",
        "            };",
        "        });",
        "        return _date;",
        "    };",
        "    try {",
        "        BilibiliTimer.isEmbedded = location.host === 'www.bilibili.com' && location.pathname === '/blackboard/html5player.html' && top !== window && top.location.host === 'bangumi.bilibili.com';",
        "    } catch(_) {",
        "        BilibiliTimer.isEmbedded = false;",
        "    }",
        "    BilibiliTimer.isLive = function bilibiliIsLive(a, b) {",
        "        return location.host === 'live.bilibili.com' ? (a !== undefined ? a : true) : (b !== undefined ? b : false);",
        "    };",
        "    BilibiliTimer.selector = BilibiliTimer.isLive({",
        "        container: '.bilibili-live-player-video-area',",
        "        controller: '.bilibili-live-player-video-controller',",
        "        fullscreenSendbar: null,",
        "        autoHideButton: '.attend-button .right-part',",
        "        autoHideButtonText: '.BilibiliTimerAutoHideButtonText',",
        "        autoHideCheck: null,",
        "        pageTitle: null,",
        "        watchlaterPageTitle: null,",
        "        watchlaterVideoTitle: null",
        "    }, {",
        "        container: '.bilibili-player-video-wrap',",
        "        controller: '.bilibili-player-video-control',",
        "        fullscreenSendbar: '.bilibili-player-video-sendbar.active',",
        "        autoHideButton: '.bgray-btn-wrap .bgray-btn.show',",
        "        autoHideButtonText: '.BilibiliTimerAutoHideButtonText',",
        "        autoHideCheck: '.bilibili-player-no-cursor',",
        "        pageTitle: '#plist .curPage ',",
        "        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'",
        "    });",
        "    BilibiliTimer.classList = 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',",
        "        autoHideButton: 'mid-part BilibiliTimerAutoHideButton',",
        "        autoHideButtonText: 'BilibiliTimerAutoHideButtonText'",
        "    }, {",
        "        timer: 'bilibili-player-video-info-container',",
        "        closeButton: 'bilibili-player-video-info-close',",
        "        panel: 'bilibili-player-video-info-panel',",
        "        restartButton: 'bilibili-player-iconfont icon-24repeaton',",
        "        autoHideButton: 'bgray-btn show  BilibiliTimerAutoHideButton',",
        "        autoHideButtonText: 'BilibiliTimerAutoHideButtonText'",
        "    });",
        "    BilibiliTimer.closeButtonText = BilibiliTimer.isLive('x', '[x]');",
        "    BilibiliTimer.globallock = false;",
        "    BilibiliTimer.widthSet = false;",
        "    BilibiliTimer.onResizing = 0;",
        "    if (BilibiliTimer.selector.autoHideCheck) BilibiliTimer.mousemoveCount = 0;",
        "    BilibiliTimer.getControllerTop = function BilibiliTimerGetControllerTop() {",
        "        var controller = $(BilibiliTimer.selector.controller);",
        "        if (controller.closest('.mode-miniscreen')[0]) return $(window).height();",
        "        var _top = $(window).height() - controller.height();",
        "        var fullscreenSendbar = $(BilibiliTimer.selector.fullscreenSendbar);",
        "        if (fullscreenSendbar[0]) _top -= fullscreenSendbar.outerHeight(true);",
        "        return _top;",
        "    };",
        "    $(window).on('resize.BilibiliTimer', function() {",
        "        BilibiliTimer.onResizing = 1;",
        "    });",
        "    $(document).on({",
        "        'mousemove.BilibiliTimer': function(e) {",
        "            var BilibiliTimer = window.BilibiliTimer;",
        "            if (BilibiliTimer && BilibiliTimer.timer) {",
        "                if (BilibiliTimer.timer.data('onMousedown')) {",
        "                    var maxTop = BilibiliTimer.getControllerTop() - BilibiliTimer.timer.outerHeight() - 10;",
        "                    var maxLeft = $(window).width() - BilibiliTimer.timer.outerWidth() - 10;",
        "                    BilibiliTimer.timer.css({",
        "                        left: Math.max(Math.min(BilibiliTimer.timer.data('baseOffset').left + e.clientX, maxLeft), 10),",
        "                        top: Math.max(Math.min(BilibiliTimer.timer.data('baseOffset').top + e.clientY, maxTop), 10)",
        "                    });",
        "                    window.getSelection().removeAllRanges();",
        "                }",
        "                if (BilibiliTimer.selector.autoHideCheck) BilibiliTimer.mousemoveCount = 0;",
        "            }",
        "        },",
        "        'mouseup.BilibiliTimer': function(e) {",
        "            var BilibiliTimer = window.BilibiliTimer;",
        "            if (BilibiliTimer && BilibiliTimer.timer && BilibiliTimer.timer.data('onMousedown')) {",
        "                BilibiliTimer.timer.data('onMousedown', false);",
        "                BilibiliTimerGM.set('offset', {",
        "                    top: BilibiliTimer.timer.css('top'),",
        "                    left: BilibiliTimer.timer.css('left')",
        "                });",
        "            }",
        "        }",
        "    });",
        "    BilibiliTimer.template = {};",
        "    var timer = BilibiliTimer.template.timer = $('<div/>');",
        "    timer.attr('id', 'BilibiliTimer').addClass(BilibiliTimer.classList.timer);",
        "    var closeButton = BilibiliTimer.template.closeButton = $('<a/>');",
        "    closeButton.text(BilibiliTimer.closeButtonText).attr({",
        "        href: 'javascript:void(0);',",
        "        id: 'BilibiliTimerCloseButton'",
        "    });",
        "    closeButton.addClass(BilibiliTimer.classList.closeButton);",
        "    var restartButton = BilibiliTimer.template.restartButton = $('<a/>');",
        "    restartButton.attr({",
        "        href: 'javascript:void(0);',",
        "        id: 'BilibiliTimerRestartButton',",
        "        title: '如果发现浮窗出现问题,' + LF + '例如无法正常拖动,无法正常显示时间等,' + LF + '请点击该按钮重建浮窗尝试修复!'",
        "    });",
        "    restartButton.addClass(BilibiliTimer.classList.closeButton).addClass(BilibiliTimer.classList.restartButton);",
        "    var panel = BilibiliTimer.template.panel = $('<div/>');",
        "    panel.addClass(BilibiliTimer.classList.panel);",
        "    panel.append(\"<div class='info-line'><span class='info-title'>系统时间:</span><span class='info-data' id='BilibiliTimerNowTime'> - </span></div>\");",
        "    panel.append(BilibiliTimer.isLive(\"<div class='info-line'><span class='info-title'>缓冲质量:</span><span class='info-data'>当前缓冲时长 <span id='BilibiliTimerVideoBufferedTimeRange'> - </span>s</span></div>\", \"<div class='info-line'><span class='info-title'>播放进度:</span><span class='info-data' id='BilibiliTimerVideoTime'> - </span></div><div class='info-line'><span class='info-title'>加载进度:</span><span class='info-data'><span id='BilibiliTimerVideoBufferedTime'> - </span>(剩余缓冲时长<span id='BilibiliTimerVideoBufferedTimeRange'> - </span>s,已缓冲<span id='BilibiliTimerVideoBufferedTimePercents'> - </span>%)</span></div><div class='info-line' style='display: none;'><span class='info-title' id='BilibiliTimerTitleDescription'>当前分页:</span><span class='info-data' id='BilibiliTimerTitle'> - </span></div>\"));",
        "    var autoHideButton = BilibiliTimer.template.autoHideButton = $('<div/>');",
        "    autoHideButton.addClass(BilibiliTimer.classList.autoHideButton);",
        "    autoHideButton.css(BilibiliTimer.isLive('width', 'height'), 'auto');",
        "    autoHideButton.html('浮窗自动隐藏' + BilibiliTimer.isLive('|', '<hr/>'));",
        "    var autoHideButtonText = $('<span/>');",
        "    autoHideButtonText.addClass(BilibiliTimer.classList.autoHideButtonText);",
        "    if (!BilibiliTimerGM.get('autoHidden')) {",
        "        BilibiliTimerGM.set('autoHidden', false);",
        "        autoHideButtonText.text('OFF');",
        "    } else autoHideButtonText.text('ON');",
        "    autoHideButton.append(autoHideButtonText);",
        "    BilibiliTimer.init = function BilibiliTimerInit() {",
        "        if (window.BilibiliTimerUninit) return false;",
        "        BilibiliTimer.onResizing = 0;",
        "        BilibiliTimer.widthSet = false;",
        "        BilibiliTimer.timer = BilibiliTimer.template.timer.clone();",
        "        BilibiliTimer.closeButton = BilibiliTimer.template.closeButton.clone();",
        "        BilibiliTimer.restartButton = BilibiliTimer.template.restartButton.clone();",
        "        BilibiliTimer.panel = BilibiliTimer.template.panel.clone();",
        "        BilibiliTimer.autoHideButton = BilibiliTimer.template.autoHideButton.clone();",
        "        BilibiliTimer.timer.append(BilibiliTimer.closeButton).append(BilibiliTimer.restartButton).append(BilibiliTimer.panel);",
        "        var title = null,",
        "            description = null;",
        "        if ($(BilibiliTimer.selector.pageTitle)[0]) title = $(BilibiliTimer.selector.pageTitle).text();",
        "        else if ($(BilibiliTimer.selector.watchlaterPageTitle)[0]) {",
        "            description = '当前分页:<br>(稍后再看)';",
        "            title = ($(BilibiliTimer.selector.watchlaterPageTitle).closest('li').index() + 1) + '、' + $(BilibiliTimer.selector.watchlaterPageTitle).text();",
        "        }",
        "        if ($(BilibiliTimer.selector.watchlaterVideoTitle)[0]) {",
        "            if (!description) description = '当前视频:<br>(稍后再看)';",
        "            title = ($(BilibiliTimer.selector.watchlaterVideoTitle).closest('li').index() + 1) + '、' + $(BilibiliTimer.selector.watchlaterVideoTitle).text() + '<br>' + (title ? '(' + title + ')' : '');",
        "        }",
        "        if (title) BilibiliTimer.timer.find('#BilibiliTimerTitle').html(title).parent().removeAttr('style');",
        "        if (description) BilibiliTimer.timer.find('#BilibiliTimerTitleDescription').html(description);",
        "        $((BilibiliTimer.isEmbedded ? top : window).document.querySelector(BilibiliTimer.selector.autoHideButton)).before(BilibiliTimer.autoHideButton);",
        "        BilibiliTimer.autoHideButtonText = BilibiliTimer.autoHideButton.find(BilibiliTimer.selector.autoHideButtonText);",
        "        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(BilibiliTimer.timer.css('left')), 0);",
        "            var baseOffsetY = Math.max(parseInt(BilibiliTimer.timer.css('top')), 0);",
        "            BilibiliTimer.timer.data({",
        "                baseOffset: {",
        "                    left: baseOffsetX - baseX,",
        "                    top: baseOffsetY - baseY",
        "                },",
        "                onMousedown: true",
        "            });",
        "        });",
        "        BilibiliTimer.closeButton.on('click', function() {",
        "            BilibiliTimer.globallock = true;",
        "            BilibiliTimer.timer.fadeOut(370);",
        "        });",
        "        BilibiliTimer.restartButton.on('click', BilibiliTimer.restart.bind(BilibiliTimer));",
        "        BilibiliTimer.autoHideButton.on('click', function() {",
        "            if (BilibiliTimerGM.get('autoHidden')) {",
        "                BilibiliTimerGM.set('autoHidden', false);",
        "                BilibiliTimer.autoHideButtonText.text('OFF');",
        "            } else {",
        "                BilibiliTimerGM.set('autoHidden', true);",
        "                BilibiliTimer.autoHideButtonText.text('ON');",
        "            }",
        "        });",
        "        if (!BilibiliTimerGM.get('offset')) {",
        "            BilibiliTimer.timer.css({",
        "                right: '10px',",
        "                top: '10px'",
        "            }).css({",
        "                left: BilibiliTimer.timer.offset().left + 'px',",
        "                right: 'auto'",
        "            });",
        "        } else BilibiliTimer.timer.css(BilibiliTimerGM.get('offset'));",
        "        $(BilibiliTimer.selector.container).append(BilibiliTimer.timer);",
        "        $(window).resize();",
        "    };",
        "    BilibiliTimer.globalWatcher = function BilibiliTimerGlobalWatcher() {",
        "        if (window.BilibiliTimerUninit) return false;",
        "        var timer = BilibiliTimer.timer;",
        "        if (!timer || !timer[0]) {",
        "            BilibiliTimer.init();",
        "            return;",
        "        }",
        "        if ($('object#player_placeholder, object#player_object')[0]) {",
        "            BilibiliTimer.uninit();",
        "            return;",
        "        }",
        "        if (!timer.closest('body')[0]) {",
        "            BilibiliTimer.restart();",
        "            return;",
        "        }",
        "        if (!!$(':-webkit-full-screen')[0]) {",
        "            if ((BilibiliTimer.selector.autoHideCheck ? !$(BilibiliTimer.selector.autoHideCheck)[0] : BilibiliTimer.mousemoveCount < 3) && BilibiliTimer.autoHidden) BilibiliTimer.autoHidden = false;",
        "            if (!BilibiliTimer.globallock && !BilibiliTimer.autoHidden) {",
        "                if (!timer.is(':visible')) timer.fadeIn();",
        "                if (BilibiliTimer.onResizing === 2) {",
        "                    BilibiliTimer.onResizing = 0;",
        "                    var maxTop = BilibiliTimer.getControllerTop() - timer.outerHeight() - 10;",
        "                    var maxLeft = $(window).width() - timer.outerWidth() - 10;",
        "                    timer.css({",
        "                        left: Math.max(Math.min(parseInt(timer.css('left')), maxLeft), 10),",
        "                        top: Math.max(Math.min(parseInt(timer.css('top')), maxTop), 10)",
        "                    });",
        "                    BilibiliTimerGM.set('offset', {",
        "                        top: timer.css('top'),",
        "                        left: timer.css('left')",
        "                    });",
        "                } else if (BilibiliTimer.onResizing === 1) {",
        "                    BilibiliTimer.onResizing = 2;",
        "                }",
        "            }",
        "        } else {",
        "            BilibiliTimer.onResizing = 0;",
        "            BilibiliTimer.globallock = false;",
        "            timer.fadeOut();",
        "        }",
        "        if (timer.is(':visible')) {",
        "            var date = BilibiliTimer.date();",
        "            timer.find('#BilibiliTimerNowTime').text(date.getFullYear() + '-' + date.getMonth() + '-' + date.getDate() + ' ' + date.getHours() + ':' + date.getMinutes() + ':' + date.getSeconds());",
        "            if (!BilibiliTimer.widthSet) BilibiliTimer.widthCalc();",
        "        }",
        "        var video = $('video');",
        "        if (!video.data('onListened')) {",
        "            video.data('onListened', true);",
        "            video.on({",
        "                'timeupdate.BilibiliTimer': BilibiliTimer.videoPlayListener.bind(BilibiliTimer),",
        "                'progress.BilibiliTimer': BilibiliTimer.videoProgressListener.bind(BilibiliTimer)",
        "            });",
        "        }",
        "    };",
        "    BilibiliTimer.autoHideWatcher = function BilibiliTimerAutoHideWatcher() {",
        "        if (!BilibiliTimerGM.get('autoHidden')) return;",
        "        if ((BilibiliTimer.selector.autoHideCheck ? $(BilibiliTimer.selector.autoHideCheck)[0] : BilibiliTimer.mousemoveCount >= 3)) {",
        "            BilibiliTimer.autoHidden = true;",
        "            BilibiliTimer.timer.fadeOut(370);",
        "        }",
        "        if (BilibiliTimer.selector.autoHideCheck) BilibiliTimer.mousemoveCount++;",
        "    };",
        "    BilibiliTimer.widthWatcher = function BilibiliTimerWidthWatcher() {",
        "        var BilibiliTimer = window.BilibiliTimer;",
        "        if (!BilibiliTimer || !BilibiliTimer.timer) return;",
        "        if (BilibiliTimer.timer.find('.info-line').width() > BilibiliTimer.timer.width()) BilibiliTimer.widthCalc();",
        "    };",
        "    BilibiliTimer.widthCalc = function BilibiliTimerWidthCalc() {",
        "        var BilibiliTimer = window.BilibiliTimer;",
        "        if (!BilibiliTimer || !BilibiliTimer.timer) return;",
        "        var timer = BilibiliTimer.timer;",
        "        var maxWidth = 0;",
        "        timer.find('.info-line').each(function() {",
        "            var width = 0,",
        "                maxHeight = 0;",
        "            $(this).children().each(function() {",
        "                width += $(this).outerWidth(true);",
        "                if (maxHeight < $(this).height()) maxHeight = $(this).height();",
        "            });",
        "            if (maxWidth < width) maxWidth = width;",
        "            $(this).height(maxHeight);",
        "        });",
        "        timer.width(maxWidth);",
        "        BilibiliTimer.widthSet = true;",
        "    };",
        "    BilibiliTimer.timeParse = function BilibiliTimerTimeParse(time) {",
        "        time = parseInt(time)",
        "        var sec = time % 60,",
        "            min = (time - sec) / 60;",
        "        if (sec < 10) sec = '0' + sec;",
        "        return min + ':' + sec;",
        "    };",
        "    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;",
        "        var BilibiliTimer = window.BilibiliTimer;",
        "        if (BilibiliTimer && BilibiliTimer.timer) BilibiliTimer.timer.find('#BilibiliTimerVideoTime').text(BilibiliTimer.timeParse(curTime) + '/' + BilibiliTimer.timeParse(durTime));",
        "        if (BilibiliTimer.timer.find('#BilibiliTimerVideoBufferedTime')[0]) {",
        "            var video = e.target,",
        "                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');",
        "            BilibiliTimer.timer.find('#BilibiliTimerVideoBufferedTimeRange').text((end - curTime).toFixed(0))",
        "        }",
        "    };",
        "    BilibiliTimer.videoProgressListener = function BilibiliTimerVideoProgressListener(e) {",
        "        var BilibiliTimer = window.BilibiliTimer;",
        "        if (!BilibiliTimer || !BilibiliTimer.timer) return;",
        "        var timer = BilibiliTimer.timer;",
        "        if (timer.find('#BilibiliTimerVideoBufferedTimeRange')[0]) {",
        "            var video = e.target,",
        "                end;",
        "            try {",
        "                end = video.buffered.end(video.buffered.length - 1);",
        "            } catch (_) {",
        "                try {",
        "                    end = video.buffered.end(0);",
        "                } catch (_) {",
        "                    return;",
        "                }",
        "            }",
        "            if (timer.find('#BilibiliTimerVideoBufferedTimeRange').text() === ' - ') BilibiliTimer.widthSet = false;",
        "            timer.find('#BilibiliTimerVideoBufferedTime').text(BilibiliTimer.timeParse(end));",
        "            BilibiliTimer.timer.find('#BilibiliTimerVideoBufferedTimeRange').text((end - video.currentTime).toFixed(0))",
        "            timer.find('#BilibiliTimerVideoBufferedTimePercents').text((end * 100 / video.duration).toFixed(2));",
        "        }",
        "    }",
        "    BilibiliTimer.start = function BilibiliTimerStart() {",
        "        if (window.BilibiliTimerUninit) return false;",
        "        if (location.host === 'bangumi.bilibili.com') return false;",
        "        if (!BilibiliTimer.interval) BilibiliTimer.interval = {};",
        "        if (!BilibiliTimer.interval.globalWatcher) BilibiliTimer.interval.globalWatcher = setInterval(BilibiliTimer.globalWatcher, 100);",
        "        if (!BilibiliTimer.interval.autoHideWatcher) BilibiliTimer.interval.autoHideWatcher = setInterval(BilibiliTimer.autoHideWatcher, 1000);",
        "        if (!BilibiliTimer.interval.widthWatcher) BilibiliTimer.interval.widthWatcher = setInterval(BilibiliTimer.widthWatcher, 5000);",
        "        try {",
        "            var video = $('video');",
        "            setTimeout(function() {",
        "                BilibiliTimer.videoPlayListener({ target: video[0] });",
        "                BilibiliTimer.videoProgressListener({ target: video[0] });",
        "            }, 100);",
        "        } catch (_) {",
        "            return;",
        "        }",
        "    };",
        "    BilibiliTimer.restart = function BilibiliTimerRestart() {",
        "        for (var i in BilibiliTimer.interval) {",
        "            if (BilibiliTimer.interval[i]) clearInterval(BilibiliTimer.interval[i]);",
        "        }",
        "        $('.BilibiliTimerAutoHideButton').remove();",
        "        if (window.BilibiliTimerUninit) return false;",
        "        var timer = $('#BilibiliTimer');",
        "        if (timer[0]) timer.fadeOut(370, BilibiliTimer.rebuild);",
        "        else BilibiliTimer.rebuild();",
        "    };",
        "    BilibiliTimer.rebuild = function BilibiliTimerRebuild() {",
        "        $('#BilibiliTimer').remove();",
        "        window.BilibiliTimer = undefined;",
        "        if (window.BilibiliTimerUninit) return false;",
        "        if (window.BilibiliTimerCode) setTimeout(function() {",
        "            eval(window.BilibiliTimerCode);",
        "        }, 0);",
        "    };",
        "    BilibiliTimer.uninit = function BilibiliTimerUninit() {",
        "        for (var i in BilibiliTimer.interval) {",
        "            if (BilibiliTimer.interval[i]) clearInterval(BilibiliTimer.interval[i]);",
        "        }",
        "        $('.BilibiliTimerAutoHideButton').remove();",
        "        $('#BilibiliTimer').remove();",
        "        window.BilibiliTimer = undefined;",
        "        window.BilibiliTimerUninit = true;",
        "    };",
        "    BilibiliTimer.start();",
        "})();"
    ].join('\n');
    var css = [
        "#BilibiliTimer {",
        "    cursor: move;",
        "    display: block;",
        "    transition-property: opacity, width;",
        "    transition-duration: .37s;",
        "    transition-timing-function: initial;",
        "    transition-delay: initial;",
        "}",
        ".bilibili-player-no-cursor #BilibiliTimer {",
        "    opacity: .73;",
        "}",
        "#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;",
        "}",
        ".attention-btn-ctrl .mid-part {",
        "    background-color: #4fc1e9;",
        "    color: #fff;",
        "    float: left;",
        "    cursor: pointer;",
        "    -webkit-user-select: none;",
        "    width: 90px;",
        "    height: 26px;",
        "    padding: 0 5px;",
        "    line-height: 26px;",
        "    font-size: 12px;",
        "    text-align: center;",
        "    text-overflow: ellipsis;",
        "    white-space: nowrap;",
        "    overflow: hidden;",
        "    box-sizing: border-box;",
        "    box-shadow: 0 0 0.1em 0.1em #ddd;",
        "}"
    ].join('\n');
    unsafeWindow.addEventListener('load', function() {
        GM_addStyle(css);
        unsafeWindow.BilibiliTimerCode = code;
        unsafeWindow.eval([
            "setInterval(function() {",
            "    if (!window.BilibiliTimerUninit && (!window.BilibiliTimer || !window.BilibiliTimer.init)) eval(window.BilibiliTimerCode);",
            "}, 500);",
        ].join('\n'));
    });
})();