Autoplay Dynamic Feeds Fix

Forces muted autoplay on all videos, including new ones in feeds (Safari/iOS)

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Autoplay Dynamic Feeds Fix
// @description  Forces muted autoplay on all videos, including new ones in feeds (Safari/iOS)
// @match        *://*/*  // Runs on all sites; edit to e.g., https://x.com/* for specific
// @grant        none
// @run-at       document-start
// @version 0.0.1.20251122130515
// @namespace https://greasyfork.org/users/1540459
// ==/UserScript==

(function() {
    'use strict';

    function enableAutoplay(video) {
        if (!video || video.dataset.autoplayFixed) return;

        video.muted = true;
        video.volume = 0;
        video.playsInline = true;
        video.setAttribute('muted', 'muted');
        video.setAttribute('playsinline', 'playsinline');
        video.setAttribute('autoplay', 'autoplay');
        video.setAttribute('loop', 'loop');
        video.currentTime = 0;

        const playPromise = video.play();
        if (playPromise !== undefined) {
            playPromise.catch(e => {
                console.warn('Autoplay retry:', e);
                setTimeout(() => video.play(), 200);
            });
        }

        video.dataset.autoplayFixed = 'true';
    }

    function scanVideos() {
        document.querySelectorAll('video').forEach(enableAutoplay);
    }

    // Run on load
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', scanVideos);
    } else {
        scanVideos();
    }

    // Watch for dynamic videos (scrolling feeds)
    const observer = new MutationObserver(scanVideos);
    observer.observe(document.body || document.documentElement, {
        childList: true,
        subtree: true
    });

    // Event for loaded videos
    document.addEventListener('loadeddata', (e) => {
        if (e.target.tagName === 'VIDEO') enableAutoplay(e.target);
    }, true);

    // Backup scan every 800ms for stubborn sites
    setInterval(scanVideos, 800);

})();