您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Makes <video> more usable across the web by enabling all controls, downloads, picture-in-picture, etc. May cause usability issues on some sites but is generally an improvment.
当前为
// ==UserScript== // @name Universal <video> Fixer // @namespace http://tampermonkey.net/ // @version 0.1.0 // @description Makes <video> more usable across the web by enabling all controls, downloads, picture-in-picture, etc. May cause usability issues on some sites but is generally an improvment. // @author @varenc // @match *://*/* // @icon  // @license MIT // @run-at document-idle // @grant none // ==/UserScript== (function() { 'use strict'; function fixVideos() { // Process all videos on the page, making them downloadable and fully functional document.querySelectorAll("video").forEach((video, index) => { console.log(`VideoFixer: Processing <video> #${index + 1}:`, video); video.style.position = "relative"; video.style.zIndex = "999999"; // <-- this is the real trick, forcing the native <video> controls to the front video.controls = true; video.style.pointerEvents = "auto"; const removedAttributes = []; const removedControlsListItems = []; // Remove attributes that restrict functionality ["disablePictureInPicture", "disableRemotePlayback"].forEach((attr) => { if (video.hasAttribute(attr)) { removedAttributes.push(attr); video.removeAttribute(attr); } }); // Remove controlsList restrictions if (video.hasAttribute("controlsList")) { removedControlsListItems.push(...video.getAttribute("controlsList").split(/\s+/)); video.removeAttribute("controlsList"); } if (removedAttributes.length > 0 || removedControlsListItems.length > 0) { console.log(`VideoFixer: Removed restrictions from video #${index + 1}:`, { removedAttributes, removedControlsListItems }); } }); console.log("VideoFixer: All videos processed and fixed!"); } // Set up a MutationObserver to detect new videos being added to the page function setupVideoObserver() { const videoObserver = new MutationObserver((mutations) => { let shouldProcess = false; // Check for new/modified <video> for (const mutation of mutations) { if (mutation.type === 'childList') { const addedVideos = Array.from(mutation.addedNodes).filter(node => node.nodeName === 'VIDEO' || (node.nodeType === Node.ELEMENT_NODE && node.querySelector('video')) ); if (addedVideos.length > 0) { shouldProcess = true; break; } } else if (mutation.type === 'attributes' && mutation.target.nodeName === 'VIDEO') { shouldProcess = true; break; } } if (shouldProcess) { fixVideos(); } }); videoObserver.observe(document.body, { childList: true, subtree: true, attributes: true, attributeFilter: ['src', 'controlsList', 'disablePictureInPicture', 'disableRemotePlayback'], attributeOldValue: true }); return videoObserver; } function initialize() { console.log("Universal Video Fixer activated!"); fixVideos(); setupVideoObserver(); } // Wait for the DOM to be fully loaded if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', initialize); } else { initialize(); } // Run a check after a short delay to catch videos that load after initial page load, but were missed by the mutation observer setTimeout(fixVideos, 2000); })();