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 - bring back the seek bar, enable download, PiP, etc
// @namespace http://tampermonkey.net/
// @version 0.1.1
// @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);
})();