// ==UserScript== // @name 视频网站自动网页全屏|倍速播放 // @namespace http://tampermonkey.net/ // @version 2.9.0 // @author Feny // @description 支持哔哩哔哩、B站直播、腾讯视频、优酷视频、爱奇艺、芒果TV、搜狐视频、AcFun弹幕网自动网页全屏;快捷键切换:全屏(F)、网页全屏(P)、下一个视频(N)、弹幕开关(D);支持任意视频倍速播放,提示记忆倍速;B站播放完自动退出网页全屏和取消连播。 // @license GPL-3.0-only // @icon  // @homepage https://github.com/xFeny/UserScript/tree/main/monkey-web-fullscreen // @include *://pages.iqiyi.com/p/zy/* // @include *://www.ezdmw.site/Index/video/* // @include *://player.ezdmw.com/danmuku/* // @include *://*bimiacg*.net/*/play* // @include *://acgfta.com/play* // @include *://ppoft.com/play* // @match *://tv.sohu.com/v/* // @match *://www.mgtv.com/b/* // @match *://www.acfun.cn/v/* // @match *://www.iqiyi.com/v_* // @match *://v.pptv.com/show/* // @match *://v.qq.com/x/page/* // @match *://v.douyu.com/show/* // @match *://v.qq.com/x/cover/* // @match *://live.bilibili.com/* // @match *://v.youku.com/video?* // @match *://haokan.baidu.com/v?* // @match *://live.acfun.cn/live/* // @match *://www.acfun.cn/bangumi/* // @match *://www.bilibili.com/list/* // @match *://www.bilibili.com/video/* // @match *://www.bilibili.com/*/play/* // @match *://v.qq.com/live/p/newtopic/* // @match *://www.bilibili.com/festival/* // @require https://unpkg.com/notyf@3.10.0/notyf.min.js // @require data:application/javascript,%3Bwindow.notyf%3D%7BNotyf%7D%3B // @require https://unpkg.com/sweetalert2@11.20.0/dist/sweetalert2.min.js // @require data:application/javascript,%3Bwindow.sweetalert2%3DSwal%3B // @resource notyf/notyf.min.css https://unpkg.com/notyf@3.10.0/notyf.min.css // @resource sweetalert2 https://unpkg.com/sweetalert2@11.20.0/dist/sweetalert2.min.css // @grant GM_addStyle // @grant GM_addValueChangeListener // @grant GM_deleteValue // @grant GM_getResourceText // @grant GM_getValue // @grant GM_info // @grant GM_listValues // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_unregisterMenuCommand // @grant unsafeWindow // @run-at document-body // @note *://*/* // @downloadURL none // ==/UserScript== (t=>{if(typeof GM_addStyle=="function"){GM_addStyle(t);return}const o=document.createElement("style");o.textContent=t,document.head.append(o)})(' @charset "UTF-8";[part=monkey-show-toast],::part(monkey-show-toast){color:#fff!important;font-size:13.5px!important;padding:5px 15px!important;border-radius:5px!important;position:absolute!important;z-index:2147483647!important;font-weight:400!important;transition:opacity .3s ease-in;background:#000000bf!important}._webFullScreen_{top:0!important;left:0!important;margin:0!important;padding:0!important;border:none!important;width:100vw!important;min-width:0!important;min-height:0!important;height:100vh!important;position:fixed!important;transform:none!important;max-width:none!important;max-height:none!important;border-radius:0!important;z-index:2147483646!important;background-color:#000!important}._webFullScreen_ video{width:100%!important;height:100%!important}video{width:100%;height:100%;object-fit:contain;transform-origin:center;transition:transform .3s;transform:rotate(var(--rotation, 0deg)) scale(var(--scale, 1)) scaleX(var(--mirror, 1))}.hideCursor{cursor:none!important}#bilibili-player .bpx-player-toast-wrap,#bilibili-player .bpx-player-cmd-dm-wrap,#bilibili-player .bpx-player-dialog-wrap,.live-room-app #sidebar-vm,.live-room-app #prehold-nav-vm,.live-room-app #shop-popover-vm,.login-tip,body.lite-room .bili-mini-mask,body.risk-captcha-adapt .bili-mini-mask{display:none!important}.monkey-web-fullscreen{z-index:9999999999!important}.monkey-web-fullscreen h2{font-size:24px}.monkey-web-fullscreen h4{color:red;margin:0 auto}.monkey-web-fullscreen p{color:#999;font-size:12px}.monkey-web-fullscreen .swal2-confirm{background-color:#7066e0!important}.monkey-web-fullscreen .swal2-deny{background-color:#dc3741!important}.monkey-web-fullscreen #monkey-picker{height:auto;max-width:25em;font-size:14px;margin-bottom:0;min-height:10em;resize:vertical}.monkey-web-fullscreen .hide{display:none!important}.monkey-web-fullscreen ._menu_{display:flex;cursor:pointer;font-size:20px;font-weight:400;align-items:center!important;margin-bottom:15px!important;justify-content:space-between!important}.monkey-web-fullscreen ._menu_:last-of-type{margin-bottom:0!important}.monkey-web-fullscreen ._menu_ input{cursor:pointer;width:20px!important;height:20px!important;appearance:auto!important;-webkit-appearance:auto!important}.notyf{z-index:9999999999!important}.notyf .notyf__message{overflow:hidden;display:-webkit-box;line-clamp:4;-webkit-line-clamp:4;text-overflow:ellipsis;-webkit-box-orient:vertical}#buffer,#install,#a1 #tips,.player-overlay,.memory-play-wrap,.atom-notice-click,#loading.not-player,#player #loading-box,.xgplayer [id*=tips],.dplayer-notice strong,.air-player-loading-box,.art-layer-autoPlayback,.art-layer-auto-playback,.invoke-app-floating-tips,.invoke-app-san-container{display:none!important}@supports (selector(:has(div))){#loading:not(:has([class*=player])){display:none!important}} '); (function (notyf, Swal) { 'use strict'; const positions = Object.freeze({ bottomLeft: "bottom: 17%; left: 10px;", bottomRight: "bottom: 17%; right: 10px;", center: "top: 50%; left: 50%; transform: translate(-50%, -50%);" }); const ONE_SECOND = 1e3; const Constants = Object.freeze({ EMPTY: "", DEF_PLAY_RATE: 1, MAX_PLAY_RATE: 16, ONE_SEC: ONE_SECOND, SHOW_TOAST_TIME: ONE_SECOND * 5, SHOW_TOAST_POSITION: positions.bottomLeft, MSG_SOURCE: "SCRIPTS_AUTO_WEB_FULLSCREEN", BILI_VID_REG: /bilibili.com\/video/, IQIYI_VID_REG: /iqiyi.com\/v_*/, ACFUN_VID_REG: /acfun.cn\/v/, QQ_VID_REG: /v.qq.com\/x/ }); function isElement(node) { return node instanceof Element; } function isDocument(node) { return node instanceof Document; } function* getShadowRoots(node, deep) { if (!isElement(node) && !isDocument(node)) { return; } const doc = isDocument(node) ? node : node.getRootNode({ composed: true }); if (isElement(node) && node.shadowRoot) { yield node.shadowRoot; } const toWalk = [node]; let currentNode = void 0; while (currentNode = toWalk.pop()) { const walker = doc.createTreeWalker(currentNode, NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_DOCUMENT_FRAGMENT, { acceptNode(node2) { if (isElement(node2) && node2.shadowRoot) { return NodeFilter.FILTER_ACCEPT; } return NodeFilter.FILTER_SKIP; } }); let walkerNode = walker.nextNode(); while (walkerNode) { if (isElement(walkerNode) && walkerNode.shadowRoot) { { toWalk.push(walkerNode.shadowRoot); } yield walkerNode.shadowRoot; } walkerNode = walker.nextNode(); } } return; } function queryCrossBoundary(selector, subject = document) { const immediate = subject.querySelector(selector); if (immediate) { return immediate; } const shadowRoots = [...getShadowRoots(subject)]; for (const root of shadowRoots) { const child = root.querySelector(selector); if (child) { return child; } } return null; } function queryAllCrossBoundary(selector, subject = document) { const results = [...subject.querySelectorAll(selector)]; const shadowRoots = [...getShadowRoots(subject)]; for (const root of shadowRoots) { const children = root.querySelectorAll(selector); for (const child of children) { results.push(child); } } return results; } function querySelector(selectors, subject = document, _options) { const selectorList = Array.isArray(selectors) ? selectors : [selectors]; if (selectorList.length === 0) { return null; } let currentSubjects = [subject]; let result = null; for (const selector of selectorList) { const newSubjects = []; for (const currentSubject of currentSubjects) { const child = queryCrossBoundary(selector, currentSubject); if (child) { result = child; newSubjects.push(child); } } if (newSubjects.length === 0) { return null; } currentSubjects = newSubjects; } return result; } function querySelectorAll(selectors, subject = document, _options) { const selectorList = Array.isArray(selectors) ? selectors : [selectors]; if (selectorList.length === 0) { return []; } let currentSubjects = [subject]; let results = []; for (const selector of selectorList) { const newSubjects = []; for (const currentSubject of currentSubjects) { const children = queryAllCrossBoundary(selector, currentSubject); for (const child of children) { newSubjects.push(child); } } if (newSubjects.length === 0) { return []; } currentSubjects = newSubjects; results = newSubjects; } return results; } var _GM_addValueChangeListener = /* @__PURE__ */ (() => typeof GM_addValueChangeListener != "undefined" ? GM_addValueChangeListener : void 0)(); var _GM_deleteValue = /* @__PURE__ */ (() => typeof GM_deleteValue != "undefined" ? GM_deleteValue : void 0)(); var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)(); var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : void 0)(); var _GM_listValues = /* @__PURE__ */ (() => typeof GM_listValues != "undefined" ? GM_listValues : void 0)(); var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)(); var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)(); var _GM_unregisterMenuCommand = /* @__PURE__ */ (() => typeof GM_unregisterMenuCommand != "undefined" ? GM_unregisterMenuCommand : void 0)(); var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)(); const { ONE_SEC: ONE_SEC$2, MSG_SOURCE } = Constants; const Tools = { noNumber: (str) => !/\d/.test(str), isTopWin: () => window.top === window, isNumber: (str) => /^[0-9]$/.test(str), alert: (...data) => window.alert(data.join(" ")), scrollTop: (top2) => _unsafeWindow.top.scrollTo({ top: top2 }), query: (selector, context) => querySelector(selector, context), querys: (selector, context) => querySelectorAll(selector, context), validDuration: (video) => !isNaN(video.duration) && video.duration !== Infinity, triggerClick: (ele) => ele?.dispatchEvent(new MouseEvent("click", { bubbles: true })), postMessage: (win = null, data) => win?.postMessage({ source: MSG_SOURCE, ...data }, "*"), isVisible: (ele) => !!(ele?.offsetWidth || ele?.offsetHeight || ele?.getClientRects().length), log: (...data) => console.log(...["%c===== 脚本日志 =====\n\n", "color:green;", ...data, "\n\n"]), preventDefault: (event) => event.preventDefault() & event.stopPropagation() & event.stopImmediatePropagation(), getFrames: () => querySelectorAll("iframe:not([src=''], [src='#'], [id='buffer'], [id='install'])"), getNumbers: (str) => typeof str === "string" ? (str.match(/\d+/g) ?? []).map(Number) : [], haveSiblings: (element) => element?.parentElement?.children.length > 1, getElementRect: (element) => element?.getBoundingClientRect(), notyf(msg, isError = false) { const notyf$1 = new notyf.Notyf({ duration: ONE_SEC$2 * 3, position: { x: "center", y: "top" } }); isError ? notyf$1.error(msg) : notyf$1.success(msg); return false; }, postMsgToFrames(data) { this.getFrames().forEach((iframe) => this.postMessage(iframe?.contentWindow, data)); }, lastTimeMap: /* @__PURE__ */ new Map(), isTooFrequent(key = "default", delay = ONE_SEC$2) { const now = Date.now(); const lastTime = this.lastTimeMap.get(key) ?? 0; const isFrequent = now - lastTime < delay; this.lastTimeMap.set(key, now); return isFrequent; }, getElementCenterPoint(element) { const { top: top2, left, width, height } = this.getElementRect(element); return { centerX: left + width / 2, centerY: top2 + height / 2 }; }, isPointInElement(pointX, pointY, element) { const { top: top2, left, right, bottom } = this.getElementRect(element); return pointX >= left && pointX <= right && pointY >= top2 && pointY <= bottom; }, triggerMousemove(ele) { const { centerY } = this.getElementCenterPoint(ele); for (let x = 0; x < ele.offsetWidth; x += 10) this.dispatchMousemove(ele, x, centerY); }, dispatchMousemove(element, clientX, clientY) { const dict = { clientX, clientY, bubbles: true }; element.dispatchEvent(new MouseEvent("mousemove", dict)); }, triggerHover(element) { if (!element) return; const { centerX, centerY } = this.getElementCenterPoint(element); const dict = { clientX: centerX, clientY: centerY, bubbles: true }; element?.dispatchEvent(new MouseEvent("mouseover", dict)); }, triggerEscape() { const dict = { key: "Escape", keyCode: 27, bubbles: true }; document.body?.dispatchEvent(new KeyboardEvent("keydown", dict)); }, createObserver(target, callback) { target = target instanceof Element ? target : this.query(target); const options = { attributes: true, childList: true, subtree: true }; const observer = new MutationObserver(callback); observer.observe(target, options); return observer; }, closest(element, selector, maxLevel = 3) { for (let level = 0; element && level < maxLevel; level++, element = element.parentElement) { if (element.matches(selector)) return element; } return null; }, findSiblingInParent(element, selector, maxLevel = 3) { for (let parent = element?.parentElement, level = 0; parent && level < maxLevel; parent = parent.parentElement, level++) { for (const child of parent.children) { if (child !== element && child.matches(selector)) return child; } } return null; }, getParentChain(element) { const parents = []; for (let current = element; current && current !== document.body; current = current.parentElement) { parents.unshift(this.getTagInfo(current)); if (current.id && this.noNumber(current.id)) break; } return parents.join(" > "); }, getTagInfo(ele) { if (ele.id && this.noNumber(ele.id)) return `#${ele.id}`; let tagInfo = ele.tagName.toLowerCase(); const classList = Array.from(ele.classList); if (classList.length === 0) return tagInfo; if (/[:\[\]]/.test(ele.className)) return `${tagInfo}[class="${ele.className}"]`; const classes = classList.filter((cls) => this.noNumber(cls)); return classes.length ? `${tagInfo}.${classes.join(".")}` : tagInfo; }, getParents(element, withSelf = false, maxLevel = Infinity) { const parents = withSelf && element ? [element] : []; for (let current = element?.parentElement, level = 0; current && current !== document.body && level < maxLevel; current = current.parentElement, level++) { parents.unshift(current); } return parents; }, simpleHash(str) { let hash = 0; for (let i = 0; i < str.length; i++) { hash = (hash << 5) + hash + str.charCodeAt(i); hash |= 0; } return hash >>> 0; } }; const { EMPTY: EMPTY$1, QQ_VID_REG, BILI_VID_REG, IQIYI_VID_REG, ACFUN_VID_REG } = Constants; const matches = _GM_info.script.matches.filter((match) => match !== "*://*/*").map((match) => new RegExp(match.replace(/\*/g, "\\S+"))); const WebSite = { isDouyu: () => location.host === "v.douyu.com", isBili: () => BILI_VID_REG.test(location.href), isTencent: () => QQ_VID_REG.test(location.href), isIqiyi: () => IQIYI_VID_REG.test(location.href), isAcFun: () => ACFUN_VID_REG.test(location.href), isBiliLive: () => location.host === "live.bilibili.com", isLivePage: () => !location.host.endsWith("live") && /\blive\b/.test(location.href), inMatches: () => matches.some((matche) => matche.test(location.href.replace(location.search, EMPTY$1))) }; const selectorConfig = { "live.bilibili.com": { webfull: "#businessContainerElement" }, "www.bilibili.com": { webfull: ".bpx-player-ctrl-web", next: ".bpx-player-ctrl-next" }, "live.acfun.cn": { full: ".fullscreen-screen", webfull: ".fullscreen-web", danmaku: ".danmaku-enabled" }, "tv.sohu.com": { full: ".x-fullscreen-btn", webfull: ".x-pagefs-btn", danmaku: ".tm-tmbtn", next: ".x-next-btn" }, "haokan.baidu.com": { full: ".art-icon-fullscreen", webfull: ".art-control-fullscreenWeb", next: ".art-control-next" }, "v.qq.com": { full: ".txp_btn_fullscreen", webfull: ".txp_btn_fake", danmaku: ".barrage-switch", next: ".txp_btn_next_u" }, "v.pptv.com": { full: ".w-zoom-container > div", webfull: ".w-expand-container > div", danmaku: ".w-barrage", next: ".w-next" }, "v.youku.com": { full: "#fullscreen-icon", webfull: "#webfullscreen-icon", danmaku: "#barrage-switch", next: ".kui-next-icon-0" }, "www.iqiyi.com": { full: ".iqp-btn-fullscreen", webfull: ".iqp-btn-webscreen", danmaku: "#barrage_switch", next: ".iqp-btn-next" }, "v.douyu.com": { full: ".ControllerBar-WindowFull-Icon", webfull: ".ControllerBar-PageFull-Icon", danmaku: ".BarrageSwitch-icon" }, "www.acfun.cn": { full: ".fullscreen-screen", webfull: ".fullscreen-web", danmaku: ".danmaku-enabled", next: ".btn-next-part div" }, "www.mgtv.com": { full: ".fullscreenBtn i", webfull: ".webfullscreenBtn i", danmaku: "div[class*='danmuSwitch']", next: ".icon-next" } }; const VideoEventHandler = { loadedmetadata() { Tools.querys('[id*="loading"]').filter((ele) => !Tools.query('[class*="player"]', ele)).forEach((ele) => ele.classList.add("not-player")); }, loadeddata() { App.initVideoProperties(this); Tools.query(".conplaying")?.click(); App.tryplay(this); }, timeupdate() { if (isNaN(this.duration)) return; App.useCachePlaybackRate(this); App.experWebFullScreen(this); App.useCachePlayTime(this); App.cachePlayTime(this); }, canplay() { App.tryplay(this); }, play() { this.isEnded = false; App.webFullScreen(this); }, pause() { Tools.query(".ec-no")?.click(); Tools.query('[id*="loading"].not-player')?.remove(); }, ended() { this.isEnded = true; this.hasToast = false; App.delCachePlayTime(); App.exitWebFullScreen(); } }; const { EMPTY, ONE_SEC: ONE_SEC$1, SHOW_TOAST_TIME, SHOW_TOAST_POSITION } = Constants; const App = { init() { this.setupVisibleListener(); this.setupKeydownListener(); this.setupMutationObserver(); this.setupUrlChangeListener(); this.setupMouseMoveListener(); }, normalSite: () => !window?.videoInfo, isLive: () => WebSite.isLivePage() || window?.videoInfo?.isLive, isBackgroudVideo: (video) => video?.muted && video?.hasAttribute("loop"), getVideo: () => Tools.query("video:not([loop]):not([src=''])") || Tools.query("video:not([loop])"), getWebFullElement: () => Tools.query(selectorConfig[location.host]?.webfull), getVideoIframe() { if (!this.videoInfo?.frameSrc) return null; const url = new URL(this.videoInfo.frameSrc); const src = decodeURI(url.pathname + url.search); return Tools.query(`iframe[src*="${src}"]`); }, setupVisibleListener() { window.addEventListener("visibilitychange", () => { if (this.normalSite()) return; const video = this.isLive() ? this.getVideo() : this.video; if (!video || video?.isEnded || !Tools.isVisible(video)) return; document.hidden ? video?.pause() : video?.play(); }); }, setupUrlChangeListener() { const _wr = (method) => { const original = history[method]; history[method] = function() { original.apply(history, arguments); window.dispatchEvent(new Event(method)); }; }; const handler = () => this.setupMutationObserver(); ["popstate", "pushState", "replaceState"].forEach((t) => _wr(t) & window.addEventListener(t, handler)); }, setupMutationObserver() { if (Tools.isTooFrequent()) return; const observer = Tools.createObserver(document.body, () => { this.triggerVideoStart(); const video = this.getVideo(); this.webFullElement = this.getWebFullElement(); if (video?.play && !!video.offsetWidth) this.setupVideoListener(); if (!WebSite.inMatches() && this.topInfo) return observer.disconnect(); if (!this.videoInfo || !this.webFullElement || !this.webFullScreen(video)) return; observer.disconnect(), this.biliLiveExtras(), this.handleLoginPopups(); }); setTimeout(() => observer.disconnect(), ONE_SEC$1 * 10); }, triggerVideoStart() { const element = Tools.query(".ec-no, .conplaying, #start, .choice-true, .close-btn, .closeclick"); if (!element || Tools.isTooFrequent("start")) return; setTimeout(() => element?.click() & element?.remove(), 150); }, setupVideoListener() { const video = this.getVideo(); this.addVideoEvtListener(video); this.healthCurrentVideo(); }, addVideoEvtListener(video) { this.video = video; this.setVideoInfo(video); this.removeVideoEvtListener(); this.videoBoundListeners = []; for (const [type, handler] of Object.entries(VideoEventHandler)) { this.video?.addEventListener(type, handler); this.videoBoundListeners.push([this.video, type, handler]); } }, removeVideoEvtListener() { this.videoBoundListeners?.forEach(([target, type, handler]) => { target?.removeEventListener(type, handler); }); }, healthCurrentVideo() { if (this.healthID || Tools.isTooFrequent("healt")) return; this.healthID = setInterval(() => this.getPlayingVideo(), ONE_SEC$1); }, getPlayingVideo() { const videos = Tools.querys("video"); for (const video of videos) { if (this.video === video || video.paused || !Tools.validDuration(video) || this.isBackgroudVideo(video)) continue; return this.addVideoEvtListener(video); } }, setVideoInfo(video) { const isLive = Object.is(video.duration, Infinity); const videoInfo = { ...Tools.getElementCenterPoint(video), src: video.currentSrc, isLive }; this.setParentVideoInfo(videoInfo); }, setParentVideoInfo(videoInfo) { window.videoInfo = this.videoInfo = videoInfo; if (!Tools.isTopWin()) videoInfo.frameSrc = location.href; if (!Tools.isTopWin()) return Tools.postMessage(window.parent, { videoInfo }); this.setupPickerEpisodeListener(); this.setupScriptMenuCommand(); this.sendTopInfo(); }, sendTopInfo() { const { host, href } = location; window.topInfo = this.topInfo = { innerWidth, host, href, hash: Tools.simpleHash(href) }; Tools.postMsgToFrames({ topInfo }); }, setupMouseMoveListener() { const delay = ONE_SEC$1 * 2; let timer = setTimeout(() => this.showOrHideCursor(), delay); document.addEventListener("mousemove", ({ target, isTrusted }) => { if (!isTrusted) return; clearTimeout(timer); this.showOrHideCursor(false); timer = setTimeout(() => this.showOrHideCursor(), delay); if (this.video === target || !target.matches("video") || this.isBackgroudVideo(target)) return; this.addVideoEvtListener(target); }); }, showOrHideCursor(isHide = true) { if (this.normalSite()) return; const videoWrap = this.getVideoHostContainer(); const elements = Array.from([this?.video, ...Tools.getParents(videoWrap, true, 3)]); elements.forEach((ele) => { if (isHide) ele?.dispatchEvent(new MouseEvent("mouseleave")); isHide ? ele?.classList.add("hideCursor") : ele?.classList.remove("hideCursor"); }); }, showToast(content, duration = SHOW_TOAST_TIME, isRemove = true) { const el = document.createElement("div"); el.setAttribute("part", "monkey-show-toast"); el.setAttribute("style", SHOW_TOAST_POSITION); if (isRemove) Tools.query('[part="monkey-show-toast"]')?.remove(); content instanceof Element ? el.appendChild(content) : el.innerHTML = content; const videoWrap = this.getVideoWrapper(); const target = videoWrap?.matches("video") ? videoWrap?.parentElement : videoWrap; target?.appendChild(el); setTimeout(() => (el.style.opacity = 0, setTimeout(() => el.remove(), ONE_SEC$1 / 3)), duration); } }; class StorageItem { constructor(name, defaultValue, useLocalStorage = false, valueParser = null) { this.name = name; this.valueParser = valueParser; this.defaultValue = defaultValue; this.useLocalStorage = useLocalStorage; } set(value) { this.setItem(this.name, value); } setItem(key, value) { this.useLocalStorage ? localStorage.setItem(key, value) : _GM_setValue(key, value); } get() { return this.parser(this.getItem(this.name) ?? this.defaultValue); } getItem(key) { const value = this.useLocalStorage ? localStorage.getItem(key) : _GM_getValue(key); try { return JSON.parse(value); } catch (e) { return value; } } parser(value) { return this.valueParser ? this.valueParser(value) : value; } del() { this.removeItem(this.name); } removeItem(key) { this.useLocalStorage ? localStorage.removeItem(key) : _GM_deleteValue(key); } } class TimedStorageItem extends StorageItem { constructor(name, defaultValue, useLocalStorage, valueParser) { super(name, defaultValue, useLocalStorage, valueParser); this.clearExpired(); } set(suffix, value, expires) { const key = this.name + suffix; if (expires) expires = Date.now() + expires * 864e5; expires ? this.setItem(key, JSON.stringify({ value, expires })) : this.setItem(key, value); } get(suffix) { const storage = this.getItem(this.name + suffix); if (!storage?.value) return this.parser(storage ?? this.defaultValue); return storage.expires > Date.now() ? this.parser(storage.value) : this.defaultValue; } del(suffix) { this.removeItem(this.name + suffix); } clearExpired() { const keys = this.useLocalStorage ? Object.keys(localStorage) : _GM_listValues(); keys.filter((key) => key.includes(this.name)).forEach((key) => { const storage = this.getItem(key); if (storage?.expires && storage.expires < Date.now()) this.removeItem(key); }); } } const Storage = { PLAY_RATE_STEP: new StorageItem("PLAY_RATE_STEP", 0.25, false, parseFloat), CACHED_PLAY_RATE: new StorageItem("FENY_SCRIPTS_V_PLAYBACK_RATE", 1, true, parseFloat), CLOSE_PLAY_RATE: new StorageItem("CLOSE_PLAY_RATE", false, false, (value) => Boolean(value)), OVERRIDE_KEYBOARD: new StorageItem("OVERRIDE_KEYBOARD", false, false, (value) => Boolean(value)), DISABLE_AUTO: new StorageItem("CLOSE_AUTO_WEB_FULL_SCREEN", false, false, (value) => Boolean(value)), VIDEO_SKIP_INTERVAL: new StorageItem("VIDEO_SKIP_INTERVAL", 5, false, (value) => parseInt(value, 10)), ZERO_KEY_SKIP_INTERVAL: new StorageItem("ZERO_KEY_SKIP_INTERVAL", 30, false, (value) => parseInt(value, 10)), ENABLE_THIS_SITE_AUTO: new TimedStorageItem("ENABLE_THIS_SITE_AUTO_", false, false, (value) => Boolean(value)), DISABLE_MEMORY_TIME: new StorageItem("DISABLE_MEMORY_TIME", false, false, (value) => Boolean(value)), USE_ALTERNATIVE: new StorageItem("USE_ALTERNATIVE", false, false, (value) => Boolean(value)), RELATIVE_EPISODE_SELECTOR: new TimedStorageItem("RELATIVE_EPISODE_SELECTOR_", null), CURRENT_EPISODE_SELECTOR: new TimedStorageItem("CURRENT_EPISODE_SELECTOR_", null), PLAY_TIME: new TimedStorageItem("PLAY_TIME_", 0, true, parseFloat) }; const Keyboard = Object.freeze({ A: "A", S: "S", P: "P", ADD: "+", SUB: "-", KeyA: "KeyA", KeyS: "KeyS", KeyZ: "KeyZ", KeyD: "KeyD", KeyF: "KeyF", KeyN: "KeyN", KeyP: "KeyP", KeyR: "KeyR", Space: "Space", ArrowLeft: "ArrowLeft", ArrowRight: "ArrowRight", NumpadSubtract: "NumpadSubtract", NumpadAdd: "NumpadAdd" }); const { PLAY_RATE_STEP: PLAY_RATE_STEP$2, VIDEO_SKIP_INTERVAL: VIDEO_SKIP_INTERVAL$2, ZERO_KEY_SKIP_INTERVAL: ZERO_KEY_SKIP_INTERVAL$1 } = Storage; const Keydown = { preventDefault(event) { const overrideKey = [Keyboard.Space, Keyboard.ArrowLeft, Keyboard.ArrowRight]; const isOverrideKey = this.isOverrideKeyboard() && overrideKey.includes(event.code); const isNumberKey = Tools.isNumber(event.key) && !this.isDisablePlaybackRate(); if (!isNumberKey && !isOverrideKey) return; Tools.preventDefault(event); }, setupKeydownListener() { window.addEventListener("keyup", (event) => this.preventDefault(event), true); window.addEventListener("keydown", (event) => this.keydownHandler.call(this, event), true); window.addEventListener("message", ({ data }) => { if (!data?.source?.includes(Constants.MSG_SOURCE)) return; if (data?.videoInfo) return this.setParentVideoInfo(data.videoInfo); if (data?.topInfo) window.topInfo = this.topInfo = data.topInfo; if (data?.defaultPlaybackRate) this.defaultPlaybackRate(); this.processEvent(data); }); }, keydownHandler(event, { key, code, ctrlKey, shiftKey } = event) { const target = event.composedPath()[0]; const isInput = ["INPUT", "TEXTAREA"].includes(target.tagName); if (this.normalSite() || isInput || target?.isContentEditable) return; if (!Object.keys(Keyboard).includes(code) && !Tools.isNumber(key)) return; this.preventDefault(event); if (Keyboard.Space === code || shiftKey && Keyboard.KeyR === code) key = code; if ([Keyboard.KeyP, Keyboard.KeyN].includes(code)) return Tools.postMessage(window.top, { key }); this.processEvent({ key }); }, processEvent(data) { if (!this.video) Tools.postMsgToFrames(data); if (data?.key) this.execHotKeyActions(data.key.toUpperCase()); }, execHotKeyActions(key) { const mapping = { Z: () => this.defaultPlaybackRate(), P: () => WebSite.inMatches() ? this.triggerIconElement("webfull", 1) : this.enhance(), N: () => WebSite.inMatches() ? this.triggerIconElement("next") : this.switchEpisode(), ARROWLEFT: () => this.isOverrideKeyboard() && this.adjustVideoTime(-VIDEO_SKIP_INTERVAL$2.get()), ARROWRIGHT: () => this.isOverrideKeyboard() && this.adjustVideoTime(VIDEO_SKIP_INTERVAL$2.get()), SPACE: () => this.isOverrideKeyboard() && this.playOrPause(this.video), 0: () => this.adjustVideoTime(ZERO_KEY_SKIP_INTERVAL$1.get()) ?? true, D: () => this.triggerIconElement("danmaku", 3), F: () => this.triggerIconElement("full", 0), KEYR: () => this.videoRotateOrMirror(true), R: () => this.videoRotateOrMirror() }; [Keyboard.A, Keyboard.ADD].forEach((key2) => mapping[key2] = () => this.adjustPlaybackRate(PLAY_RATE_STEP$2.get())); [Keyboard.S, Keyboard.SUB].forEach((key2) => mapping[key2] = () => this.adjustPlaybackRate(-PLAY_RATE_STEP$2.get())); mapping[key]?.() ?? (Tools.isNumber(key) && this.setPlaybackRate(key)); }, triggerIconElement(name, index) { WebSite.isBiliLive() ? this.getBiliLiveIcons()?.[index]?.click() : Tools.query(selectorConfig[location.host]?.[name])?.click(); } }; const { DISABLE_AUTO, PLAY_RATE_STEP: PLAY_RATE_STEP$1, CLOSE_PLAY_RATE, USE_ALTERNATIVE, OVERRIDE_KEYBOARD, VIDEO_SKIP_INTERVAL: VIDEO_SKIP_INTERVAL$1, DISABLE_MEMORY_TIME: DISABLE_MEMORY_TIME$1, ZERO_KEY_SKIP_INTERVAL, ENABLE_THIS_SITE_AUTO: EN_THIS, CURRENT_EPISODE_SELECTOR: EP_SELECTOR$1, RELATIVE_EPISODE_SELECTOR: RE_SELECTOR$2 } = Storage; const MenuCommand = { isDisableAuto: () => DISABLE_AUTO.get(), isUseAlternative: () => USE_ALTERNATIVE.get(), isOverrideKeyboard: () => OVERRIDE_KEYBOARD.get(), isDisablePlaybackRate: () => CLOSE_PLAY_RATE.get(), isEnbleThisWebSiteAuto: () => EN_THIS.get(Tools.isTopWin() ? location.host : topInfo.host), setupScriptMenuCommand() { if (!Tools.isTopWin() || Tools.isTooFrequent("menu")) return; this.setupMenuChangeListener(); this.registMenuCommand(); }, setupMenuChangeListener() { const host = location.host; [CLOSE_PLAY_RATE.name, OVERRIDE_KEYBOARD.name, EN_THIS.name + host, EP_SELECTOR$1.name + host].forEach( (key) => _GM_addValueChangeListener(key, () => this.registMenuCommand()) ); }, registMenuCommand() { const host = location.host; const isEnble = this.isEnbleThisWebSiteAuto(); const siteFun = () => EN_THIS.set(host, !isEnble); const delPicker = () => EP_SELECTOR$1.del(host) & RE_SELECTOR$2.del(host); [ { title: "设置零键秒数", cache: ZERO_KEY_SKIP_INTERVAL, isDisable: this.isLive() }, { title: "设置倍速步长", cache: PLAY_RATE_STEP$1, isDisable: this.isLive() || this.isDisablePlaybackRate() }, { title: "设置快进/退秒数", cache: VIDEO_SKIP_INTERVAL$1, isDisable: this.isLive() || !this.isOverrideKeyboard() }, { title: `此站${isEnble ? "禁" : "启"}用自动网页全屏`, cache: EN_THIS, isDisable: WebSite.inMatches(), fn: siteFun }, { title: "删除此站的剧集选择器", cache: EP_SELECTOR$1, isDisable: !EP_SELECTOR$1.get(host), fn: delPicker }, { title: "更多设置", cache: OVERRIDE_KEYBOARD, isDisable: false, fn: () => this.moreSettPopup() } ].forEach(({ title, cache, isDisable, fn }) => { const id = `${cache.name}_MENU_ID`; _GM_unregisterMenuCommand(this[id]); if (isDisable) return; this[id] = _GM_registerMenuCommand(title, () => { if (fn) return fn.call(this); const input = prompt(title, cache.get()); if (!isNaN(input) && cache.parser(input)) cache.set(input); }); }); }, moreSettPopup() { const configs = [ { name: "key", label: "空格 ◀▶ 键 控制", cache: OVERRIDE_KEYBOARD }, { name: "auto", label: "禁用自动网页全屏", cache: DISABLE_AUTO, hide: !WebSite.inMatches() }, { name: "rate", label: "禁用视频倍速调节", cache: CLOSE_PLAY_RATE, hide: this.isLive() }, { name: "time", label: "禁用播放进度记录", cache: DISABLE_MEMORY_TIME$1, hide: this.isLive() }, { name: "self", label: "使用自带网页全屏增强", cache: USE_ALTERNATIVE, hide: WebSite.inMatches() } ]; const html = configs.map( ({ name, label, hide }) => `` ); Swal.fire({ width: 350, backdrop: false, title: "更多设置", showCancelButton: true, cancelButtonText: "关闭", showConfirmButton: false, html: html.join(Constants.EMPTY), customClass: { container: "monkey-web-fullscreen" }, didOpen() { Tools.querys("._menu_ input").forEach((ele, i) => { ele.checked = configs[i].cache.get(); ele.addEventListener("click", function() { this.name === "rate" && Tools.postMessage(window, { defaultPlaybackRate: this.checked }); setTimeout(() => configs[i].cache.set(this.checked), 100), Tools.notyf("修改成功!"); }); }); } }); } }; const { ONE_SEC, DEF_PLAY_RATE, MAX_PLAY_RATE } = Constants; const { PLAY_RATE_STEP, CACHED_PLAY_RATE, VIDEO_SKIP_INTERVAL, PLAY_TIME, DISABLE_MEMORY_TIME } = Storage; const VideoControl = { isVideoEnded() { return Math.floor(this.video.currentTime) === Math.floor(this.video.duration); }, initVideoProperties(video) { video.volume = 1; video.hasToast = false; video.isWebFullScreen = false; }, playOrPause: (video) => WebSite.isDouyu() ? Tools.triggerClick(video) : video.paused ? video.play() : video.pause(), tryplay: (video) => video?.paused && (WebSite.isDouyu() ? Tools.triggerClick(video) : video?.play()), checkUsable() { if (!this.video) return false; if (this.isVideoEnded()) return false; if (WebSite.isLivePage()) return false; if (this.isDisablePlaybackRate()) return false; if (!Tools.validDuration(this.video)) return false; return true; }, setPlaybackRate(playRate, show = true) { if (!this.checkUsable()) return; this.video.playbackRate = (+playRate).toFixed(2).replace(/\.?0+$/, ""); if (show) this.customToast("正在以", `${this.video.playbackRate}x`, "倍速播放"); CACHED_PLAY_RATE.set(this.video.playbackRate); }, adjustPlaybackRate(step = PLAY_RATE_STEP.get()) { if (!this.checkUsable()) return; const playRate = Math.max(PLAY_RATE_STEP.get(), this.video.playbackRate + step); this.setPlaybackRate(Math.min(MAX_PLAY_RATE, playRate)); }, defaultPlaybackRate() { if (this.isDisablePlaybackRate()) return; this.setPlaybackRate(DEF_PLAY_RATE, false); this.showToast("已恢复正常倍速播放"); }, useCachePlaybackRate(video) { if (this.isDisablePlaybackRate()) return; const playRate = CACHED_PLAY_RATE.get(); if (video.playbackRate === playRate) return; this.setPlaybackRate(playRate, !video.hasToast); video.hasToast = true; }, adjustVideoTime(second = VIDEO_SKIP_INTERVAL.get()) { if (!this.video || !Tools.validDuration(this.video) || second > 0 && this.video.isEnded) return; const currentTime = Math.min(this.video.currentTime + second, this.video.duration); this.setCurrentTime(currentTime); }, cachePlayTime(video) { if (!this.topInfo || this.isLive()) return; if (DISABLE_MEMORY_TIME.get() || this.isVideoEnded() || this.isMultVideo()) return this.delCachePlayTime(); if (video.currentTime > VIDEO_SKIP_INTERVAL.get()) PLAY_TIME.set(topInfo.hash, video.currentTime - 1, 7); }, useCachePlayTime(video) { if (this.hasUsedPlayTime || !this.topInfo || this.isLive()) return; const time = PLAY_TIME.get(topInfo.hash); if (time <= video.currentTime) return this.hasUsedPlayTime = true; this.customToast("上次观看至", this.formatTime(time), "处,已为您续播", ONE_SEC * 3, false); this.hasUsedPlayTime = true; this.setCurrentTime(time); }, delCachePlayTime: () => PLAY_TIME.del(topInfo.hash), setCurrentTime(currentTime) { if (currentTime) this.video.currentTime = Math.max(0, currentTime); }, rotation: 0, videoRotateOrMirror(mirror = false) { if (!this.video) return; mirror ? this.isMirrored = !this.isMirrored : this.rotation = (this.rotation + 90) % 360; const { videoWidth, videoHeight } = this.video; const isVertical = [90, 270].includes(this.rotation); const scale = isVertical ? videoHeight / videoWidth : 1; this.video.style = `--scale: ${scale}; --rotation: ${this.rotation}deg; --mirror: ${this.isMirrored ? -1 : 1};`; }, customToast(startText, colorText, endText, duration, isCoexist) { const span = document.createElement("span"); span.appendChild(document.createTextNode(startText)); const child = span.cloneNode(true); child.textContent = colorText; child.setAttribute("style", "margin:0 3px!important;color:#ff6101!important;"); span.appendChild(child); span.appendChild(document.createTextNode(endText)); this.showToast(span, duration, isCoexist); }, formatTime(seconds) { if (isNaN(seconds)) return "00:00"; const h = Math.floor(seconds / 3600); const m = Math.floor(seconds % 3600 / 60); const s = Math.floor(seconds % 60); return [...h ? [h] : [], m, s].map((unit) => String(unit).padStart(2, "0")).join(":"); }, isMultVideo() { const currVideoSrc = this.videoInfo.src; const videos = Tools.querys("video").filter((video) => video.currentSrc !== currVideoSrc && !isNaN(video.duration)); return videos.length > 1; } }; const WebSiteLogin = { handleLoginPopups() { this.handleIqyLogin(); this.handleBiliLogin(); this.handleTencentLogin(); }, handleTencentLogin: () => WebSite.isTencent() && Tools.query("#login_win")?.remove(), handleIqyLogin: () => WebSite.isIqiyi() && Tools.query("#qy_pca_login_root")?.remove(), handleBiliLogin() { if (!WebSite.isBili()) return; if (document.cookie.includes("DedeUserID")) return _unsafeWindow.player?.requestQuality(80); setTimeout(() => { _unsafeWindow.__BiliUser__.isLogin = true; _unsafeWindow.__BiliUser__.cache.data.isLogin = true; _unsafeWindow.__BiliUser__.cache.data.mid = Date.now(); }, Constants.ONE_SEC * 3); } }; const WebFullScreen = { webFullScreen(video) { if (!video?.offsetWidth) return false; if (this.isDisableAuto() || video?.offsetWidth >= innerWidth) return true; if (!WebSite.isBiliLive()) return Tools.triggerClick(this.webFullElement); return this.biliLiveWebFullScreen(); }, biliLiveWebFullScreen() { Tools.scrollTop(70); const control = this.getBiliLiveIcons(); const el = Tools.query(":is(.lite-room, #player-ctnr)", top.document); if (el) Tools.scrollTop(Tools.getElementRect(el)?.top ?? 0); return Tools.triggerClick(control?.[1]); }, exitWebFullScreen() { if (!WebSite.isBili() && !WebSite.isAcFun()) return; if (this.video.offsetWidth === innerWidth) this.webFullElement?.click(); const isLast = Tools.query('.video-pod .switch-btn:not(.on), .video-pod__item:last-of-type[data-scrolled="true"]'); if (!Tools.query(".video-pod") || isLast) return Tools.query(".bpx-player-ending-related-item-cancel")?.click(); }, biliLiveExtras() { if (!WebSite.isBiliLive()) return; _unsafeWindow.top?.livePlayer?.volume(100); _unsafeWindow.top?.livePlayer?.switchQuality("10000"); localStorage.setItem("FULLSCREEN-GIFT-PANEL-SHOW", 0); document.body.classList.add("hide-asida-area", "hide-aside-area"); }, getBiliLiveIcons() { Tools.triggerMousemove(this.getVideo()); return Tools.querys("#web-player-controller-wrap-el .right-area .icon"); }, experWebFullScreen(video) { if (!this.topInfo || WebSite.inMatches() || !this.isEnbleThisWebSiteAuto() || video.isWebFullScreen) return; if (video.offsetWidth === topInfo.innerWidth) return video.isWebFullScreen = true; Tools.postMessage(window.top, { key: Keyboard.P }); video.isWebFullScreen = true; } }; const { RELATIVE_EPISODE_SELECTOR: RE_SELECTOR$1 } = Storage; const SwitchEpisode = { switchEpisode(isPrev = false) { const currEpisode = this.getCurrentEpisode() ?? this.getCurrentEpisodeFromAllLink(); const targetEpisode = this.getTargetEpisodeContainer(currEpisode, isPrev); this.jumpToTargetEpisode(targetEpisode); }, getCurrentEpisode() { return RE_SELECTOR$1.get(location.host) ? this.getCurrentEpisodeBySelector() : this.getCurrentEpisodeLinkElement(); }, getCurrentEpisodeLinkElement() { const { pathname, search } = location; const last = pathname.split("/").pop(); const links = Tools.querys(`:is(a[href*="${pathname + search}"], a[href*="${last}"], a[href*="${search}"])`); return links.length <= 1 ? links.shift() : this.findCurrentEpisodeElement(links, pathname + search); }, findCurrentEpisodeElement(eles, pageUrl) { const filter = [ "h1", "header", "[id*='history']", "[id*='guankan']", "[class*='lishi']", "[class*='record']", "[class*='history']", "[class*='tab-item']", "[class*='play-channel']" ]; eles = eles.filter((el) => { const { pathname, search } = new URL(el.href); return !Tools.closest(el, `:is(${filter})`, 5) && pageUrl.includes(pathname + search); }).map(this.getEpisodeContainer).reverse(); return eles.length <= 1 ? eles.shift() : eles.find((el) => el.classList.contains("active") || !!this.getEpisodeNumber(el)); }, getEpisodeNumber: (ele) => Tools.getNumbers(ele?.innerText?.replace(/-/g, ""))?.shift(), getTargetEpisodeContainer(element, isPrev = false) { if (!element) return; const currNumber = this.getEpisodeNumber(element); const episodes = this.getAllEpisodeElement(element); const index = episodes.indexOf(element); const numbers = episodes.map(this.getEpisodeNumber); const { leftSmall, rightLarge } = this.compareLeftRight(numbers, currNumber, index); return (leftSmall || rightLarge) === isPrev ? episodes[index - 1] : episodes[index + 1]; }, compareLeftRight: (numbers, compareNumber = 0, index) => ({ leftSmall: numbers.some((val, i) => i < index && val < compareNumber), rightLarge: numbers.some((val, i) => i > index && val > compareNumber) }), getAllEpisodeElement(element) { if (!element) return []; const eleName = element.tagName; const eleClass = Array.from(element.classList); const sibling = Tools.findSiblingInParent(element, eleName); const children = Array.from(sibling?.parentElement?.children ?? []); return children.filter((ele) => { const currClass = Array.from(ele.classList); const hasSameClass = eleClass.some((value) => currClass.includes(value)); return currClass.length ? hasSameClass : ele.tagName === eleName; }); }, jumpToTargetEpisode(element) { const stack = [element].filter(Boolean); while (stack.length > 0) { const current = stack.pop(); if (current.matches("a, button")) return current.click(); stack.push(...Array.from(current.children).reverse()); current.click(); } }, getEpisodeContainer(element) { while (element && element.parentElement) { const siblings = Array.from(element.parentElement.children); if (siblings.length > 1 && siblings.some((sib) => sib !== element && sib.tagName === element.tagName)) return element; element = element.parentElement; } return null; }, getCurrentEpisodeFromAllLink() { const getNumber = (str) => Tools.getNumbers(str).join(Constants.EMPTY); const lastPath = location.pathname.split("/").pop(); const number = getNumber(lastPath); return number ? Tools.querys("a[onclick]")?.find((el) => Array.from(el.attributes).find((attr) => getNumber(attr.value) === number)) : null; } }; const { RELATIVE_EPISODE_SELECTOR: RE_SELECTOR, CURRENT_EPISODE_SELECTOR: EP_SELECTOR } = Storage; const PickerEpisode = { setupPickerEpisodeListener() { if (WebSite.inMatches()) return; document.body.addEventListener( "click", (event, { target, ctrlKey, altKey, isTrusted } = event) => { if (!ctrlKey || !altKey || !isTrusted || this.isLive()) return; if (!Tools.isTopWin()) return Tools.notyf("此页面不能抓取 (•ิ_•ิ)?", true); Tools.preventDefault(event); const hasCurrentSelector = EP_SELECTOR.get(location.host); const hasRelativeSelector = RE_SELECTOR.get(location.host); if (hasCurrentSelector && hasRelativeSelector) return Tools.notyf("已拾取过剧集元素 ( ̄ー ̄)", true); const number = this.getEpisodeNumber(target); if (!number) return Tools.notyf("点击位置无数字 (•ิ_•ิ)?", true); !hasCurrentSelector ? this.pickerCurrentEpisodeChain(target) : this.pickerRelativeEpisodeChain(target); }, true ); }, pickerCurrentEpisodeChain(element) { if (EP_SELECTOR.get(location.host)) return; this.pickerEpisodeDialog(element, { validBtnCallback(value) { try { const number = this.getEpisodeNumber(Tools.query(value)); !!number ? Tools.notyf(`当前集数:${number}`) : Tools.notyf("获取集数失败 〒▽〒", true); } catch (e) { Tools.notyf("获取集数失败 〒▽〒", true); console.error(e); } }, confirmCallback(value) { EP_SELECTOR.set(location.host, value); Tools.notyf("继续拾取元素 \(>0<)/"); } }); }, pickerRelativeEpisodeChain(element) { if (RE_SELECTOR.get(location.host)) return; this.pickerEpisodeDialog(element, { validBtnCallback(value) { try { const container = this.getEpisodeContainer(Tools.query(value)); const numbers = this.getAllEpisodeElement(container)?.map(this.getEpisodeNumber); !!numbers.length ? Tools.notyf(`所有集数:${numbers.join(" ")}`) : Tools.notyf("获取集数失败 〒▽〒", true); } catch (e) { Tools.notyf("获取集数失败 〒▽〒", true); console.error(e); } }, confirmCallback(value) { RE_SELECTOR.set(location.host, value); Tools.notyf("操作完成 []~( ̄▽ ̄)~* 干杯"); } }); }, getCurrentEpisodeBySelector() { const num = this.getEpisodeNumber(Tools.query(EP_SELECTOR.get(location.host))); const current = this.getEpisodeContainer(Tools.query(EP_SELECTOR.get(location.host))); const episodes = this.getAllEpisodeElement(this.getEpisodeContainer(Tools.query(RE_SELECTOR.get(location.host)))); return episodes.includes(current) ? current : episodes.find((el) => this.getEpisodeNumber(el) === num); }, pickerEpisodeDialog(element, { validBtnCallback, confirmCallback }) { Swal.fire({ html: `
编辑元素选择器,确保能正确获取到集数
`, customClass: { container: "monkey-web-fullscreen" }, title: "拾取剧集元素选择器", confirmButtonText: "保存", denyButtonText: "验证", showCloseButton: true, showDenyButton: true, reverseButtons: true, focusDeny: true, preDeny: () => { const value = Tools.query("#monkey-picker").value.trim(); return value ? validBtnCallback.call(this, value) ?? false : Tools.notyf("元素选择器不能为空!", true); }, preConfirm: () => Tools.query("#monkey-picker").value.trim() || Tools.notyf("元素选择器不能为空!", true), didOpen: () => Tools.query("#monkey-picker").value = Tools.getParentChain(element) }).then((result) => result.isConfirmed && confirmCallback.call(this, result.value)); } }; const ScriptsEnhance = { enhance() { if (this.normalSite()) return; const ele = this.getVideoHostContainer(); ele.oldWidth = ele?.offsetWidth ?? 0, ele.ctrl = ele.ctrl ?? ele?.controls; if (!this.isUseAlternative()) Tools.triggerHover(ele), Tools.triggerEscape(); setTimeout(() => this.alternative(ele), 100); }, alternative(ele) { if (!Tools.isTopWin() || ele.oldWidth !== ele.offsetWidth) return; Tools.getParents(ele, true)?.forEach((el) => el?.classList?.toggle("_webFullScreen_")); if (ele?.matches("video")) ele.controls = ele.classList.contains("_webFullScreen_") ? true : ele.ctrl; this.cleanStubbornElements(ele); }, cleanStubbornElements(element) { if (element.classList.contains("_webFullScreen_")) return; Tools.querys("._webFullScreen_").forEach((el) => el.classList.remove("_webFullScreen_")); Tools.scrollTop(Tools.getElementRect(element)?.top - 100); }, getVideoHostContainer() { if (this.video) return this.getVideoWrapper(); const videoIframe = this.getVideoIframe(); if (videoIframe) return videoIframe; const iframes = Tools.getFrames(); if (iframes.length === 1) return iframes.shift(); const { centerX, centerY } = this.videoInfo; return iframes.find((ele) => Tools.isVisible(ele) && Tools.isPointInElement(centerX, centerY, ele)); }, getVideoWrapper() { const control = this.findVideoControlBar(this.video); return control?.parentElement ?? this.findVideoContainer(this.video?.parentElement) ?? this.video; }, findVideoContainer(target) { return Tools.closest(target, ':is([class*="wrap"], [class*="video"], [class*="Player"], [class*="player"])'); }, findVideoControlBar(element) { const ctrl = ':is([class*="Control"], [class*="control"], [id*="control"], [class*="ctrl"], [class*="bar"])'; return Tools.findSiblingInParent(element, ctrl); } }; const cssLoader = (e) => { const t = GM_getResourceText(e); return GM_addStyle(t), t; }; cssLoader("sweetalert2"); cssLoader("notyf/notyf.min.css"); [Keydown, MenuCommand, VideoControl, WebSiteLogin, WebFullScreen, SwitchEpisode, PickerEpisode, ScriptsEnhance].forEach( (handler) => { Object.entries(handler).forEach(([key, value]) => { App[key] = value instanceof Function ? value.bind(App) : value; }); } ); App.init(); _unsafeWindow.Tools = Tools; _unsafeWindow.MONKEY_WEB_FULLSCREEN = App; })(notyf, sweetalert2);