// ==UserScript== // @name BilibiliTimer - B站H5播放器全屏下实时显示当前系统时间和播放进度 // @version 3.3.1 // @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/video/BV* // @match *://www.bilibili.com/html/*layer.htm* // @match *://www.bilibili.com/blackboard/*layer.htm* // @match *://www.bilibili.com/bangumi/play/* // @match *://www.bilibili.com/medialist/play/watchlater* // @match *://live.bilibili.com/* // @run-at document-start // @grant unsafeWindow // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @grant GM.addStyle // @grant GM.setValue // @grant GM.getValue // @icon  // @icon64  // @downloadURL none // ==/UserScript== /* eslint-disable require-atomic-updates */ "use strict"; (function () { /* 防止重复加载 */ if (unsafeWindow.BilibiliTimer) { return; } const multiValueKeys = ["exclude", "grant", "include", "match", "require", "resource"]; const booleanValueKeys = ["noframes"]; const script = { scriptMeta: {}, }; let GM4Detected = false; try { if (Object.prototype.toString.bind(GM)() === "[object Object]") { GM4Detected = true; } else { GM4Detected = false; } } catch (_) { GM4Detected = false; } script.scriptMetaStr = (GM4Detected ? GM.info : GM_info).scriptMetaStr; script.scriptMetaStr.split(/\n+/).forEach((str) => { const string = str.replace(/^\s*\/\/\s*/, ""); const _temp = string.match(/^@([a-z\d:-]+) +(.+)$/i); if (!_temp) { return; } const key = _temp[1], value = _temp[2].trim(); if (multiValueKeys.includes(key)) { if (script.scriptMeta[key]) { script.scriptMeta[key].push(value); } else { script.scriptMeta[key] = [value]; } } else if (booleanValueKeys.includes(key)) { script.scriptMeta[key] = true; } else { script.scriptMeta[key] = value; } }); /* eslint-disable require-await */ script.addStyle = GM4Detected ? GM.addStyle : async (...args) => { return GM_addStyle(...args); }; script.getValue = GM4Detected ? GM.getValue : async (...args) => { return GM_getValue(...args); }; script.setValue = GM4Detected ? GM.setValue : async (...args) => { return GM_setValue(...args); }; /* eslint-enable require-await */ unsafeWindow.BilibiliTimerUninit = false; unsafeWindow.BilibiliTimer = {}; const code = async 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() { const _date = new Date(); ["getDate", "getFullYear", "getHours", "getMilliseconds", "getMinutes", "getMonth", "getSeconds", "getTime", "getUTCDate", "getUTCFullYear", "getUTCHours", "getUTCMilliseconds", "getUTCMinutes", "getUTCMonth", "getUTCSeconds", "getYear"].forEach((key) => { _date[key] = (...a) => { let result = Date.prototype[key].apply(_date, a); if (key.includes("Month")) { result++; } if (typeof result === "number" && `${result}`.length === 1) { return `0${result}`; } return `${result}`; }; }); return _date; }; let isEmbedded; try { isEmbedded = location.host === "www.bilibili.com" && ["/blackboard/html5player.html", "/blackboard/player.html"].includes(location.pathname) && top !== window && top.location.host.includes("bilibili.com"); } catch (_) { isEmbedded = false; } unsafeWindow.BilibiliTimer.isEmbedded = isEmbedded; unsafeWindow.BilibiliTimer.realWindow = 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, autoHideCheck: null, }, { container: ".bilibili-player-video-wrap", controller: ".bilibili-player-video-control", fullscreenSendbar: ".bilibili-player-video-sendbar.active", autoHideCheck: ".bilibili-player-no-cursor", }, ); unsafeWindow.BilibiliTimer.selector.fullscreen = (function () { try { unsafeWindow.document.querySelector(":fullscreen"); // Let's try the standard selector return ":fullscreen"; } catch (_) { // Just keep going } try { unsafeWindow.document.querySelector(":-webkit-full-screen"); // Maybe we should try webkit prefix return ":-webkit-full-screen"; } catch (_) { // Just keep going } try { unsafeWindow.document.querySelector(":-moz-full-screen"); // Or firefox prefix return ":-moz-full-screen"; } catch (_) { // Just keep going } try { unsafeWindow.document.querySelector(":-ms-fullscreen"); // Why you still using IE¿ return ":-ms-fullscreen"; } catch (_) { // 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", }, { timer: "bilibili-player-video-info-container", closeButton: "bilibili-player-video-info-close", panel: "bilibili-player-video-info-panel", restartButton: "bilibili-player-iconfont icon-24repeaton", }, ); unsafeWindow.BilibiliTimer.setting = await script.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() { const controller = $(unsafeWindow.BilibiliTimer.selector.controller); if (controller.closest(".mode-miniscreen")[0]) { return $(window).height(); } let _top = $(window).height() - controller.height(); const fullscreenSendbar = $(unsafeWindow.BilibiliTimer.selector.fullscreenSendbar); if (fullscreenSendbar[0]) { _top -= fullscreenSendbar.outerHeight(true); } return _top; }; $(window).on("resize.BilibiliTimer", () => { try { unsafeWindow.BilibiliTimer.onResizing = 1; } catch (e) { /* */ } }); $(document).on({ "mousemove.BilibiliTimer": function (e) { if (unsafeWindow.BilibiliTimer && unsafeWindow.BilibiliTimer.timer) { if (unsafeWindow.BilibiliTimer.onMousedown) { const maxTop = unsafeWindow.BilibiliTimer.getControllerTop() - unsafeWindow.BilibiliTimer.timer.outerHeight() - 10; const 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; unsafeWindow.BilibiliTimer.timer.removeClass("dragging"); script.setValue("offset", { top: unsafeWindow.BilibiliTimer.timer.css("top"), left: unsafeWindow.BilibiliTimer.timer.css("left"), }); } }, }); unsafeWindow.BilibiliTimer.template = {}; const timer = unsafeWindow.BilibiliTimer.template.timer = $("
"); timer.attr("id", "BilibiliTimer").addClass(unsafeWindow.BilibiliTimer.classList.timer); const closeButton = unsafeWindow.BilibiliTimer.template.closeButton = $(""); closeButton.text(unsafeWindow.BilibiliTimer.closeButtonText).attr({ href: "javascript:void(0);", id: "BilibiliTimerCloseButton", }); closeButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton); const restartButton = unsafeWindow.BilibiliTimer.template.restartButton = $(""); restartButton.attr({ href: "javascript:void(0);", id: "BilibiliTimerRestartButton", title: "如果发现浮窗出现问题,\n例如无法正常拖动,无法正常显示时间等,\n请点击该按钮重建浮窗尝试修复!", }); restartButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton).addClass(unsafeWindow.BilibiliTimer.classList.restartButton); const smallModeButton = unsafeWindow.BilibiliTimer.template.smallModeButton = $(""); smallModeButton.text(unsafeWindow.BilibiliTimer.smallModeButtonText).attr({ href: "javascript:void(0);", id: "BilibiliTimerSmallModeButton", title: "点击切换显示模式", }); smallModeButton.addClass(unsafeWindow.BilibiliTimer.classList.closeButton); const panel = unsafeWindow.BilibiliTimer.template.panel = $(""); panel.addClass(unsafeWindow.BilibiliTimer.classList.panel); panel.append('