🎉 欢迎访问GreasyFork.Org 镜像站!本镜像站由公众号【爱吃馍】搭建,用于分享脚本。联系邮箱📮

Greasy fork 爱吃馍镜像

抖音播完自动暂停

在抖音网页版中,当视频剩余 0.2 秒时自动暂停,防止自动跳播下一条视频。同时支持快捷键P控制启用/暂停

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey, Greasemonkey или Violentmonkey.

You will need to install an extension such as Tampermonkey to install this script.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Userscripts.

За да инсталирате скрипта, трябва да инсталирате разширение като Tampermonkey.

За да инсталирате този скрипт, трябва да имате инсталиран скриптов мениджър.

(Вече имам скриптов мениджър, искам да го инсталирам!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

(Вече имам инсталиран мениджър на стиловете, искам да го инсталирам!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

// ==UserScript==
// @name          抖音播完自动暂停
// @namespace     http://tampermonkey.net/
// @version       1.1
// @description   在抖音网页版中,当视频剩余 0.2 秒时自动暂停,防止自动跳播下一条视频。同时支持快捷键P控制启用/暂停
// @author        ChatGPT & Gemini
// @icon         
// @match         https://www.tiktok.com/*
// @match         https://www.douyin.com/*
// @match         https://www.douyin.com/video/*
// @exclude       https://www.tiktok.com/*/live/*
// @exclude       https://www.tiktok.com/live/*
// @exclude       https://live.douyin.com/*
// @grant         GM_setValue
// @grant         GM_getValue
// @grant         GM_notification
// ==/UserScript==

(function() {
    'use strict';

    // --- 脚本状态管理 ---
    const SCRIPT_ENABLED_KEY = 'douyin_auto_pause_script_enabled';
    const FIRST_RUN_KEY = 'douyin_auto_pause_first_run_v09'; // 更新首次运行标记版本
    let scriptEnabled = GM_getValue(SCRIPT_ENABLED_KEY, true); // 默认启用

    function setScriptEnabled(enabled) {
        scriptEnabled = enabled;
        GM_setValue(SCRIPT_ENABLED_KEY, enabled);
        showNotification(`脚本已${enabled ? '启用' : '禁用'}`, enabled ? 'success' : 'warning');
        if (!enabled) {
            // 脚本禁用时,移除所有视频监听器并停止观察者
            if (currentVideo) {
                currentVideo.removeEventListener('timeupdate', handleVideoTimeUpdate);
                currentVideo.removeEventListener('ended', handleVideoEnded);
                currentVideo = null; // 清理当前视频引用
            }
            if (observer) {
                observer.disconnect(); // 停止观察
            }
            console.log('Tampermonkey: 脚本已禁用,已移除所有监听器和观察者。');
        } else {
            // 脚本启用时,重新初始化观察者和视频监听
            if (observer) {
                observer.observe(document.body, {
                    childList: true,
                    subtree: true,
                    attributes: true,
                    attributeFilter: ['data-e2e-state', 'src']
                });
            }
            setupVideoListener();
            console.log('Tampermonkey: 脚本已启用,重新设置监听器和观察者。');
        }
    }

    // --- 提示框相关变量和函数 ---
    let notificationTimeoutId = null;

    function showNotification(message, type = 'info', duration = 1000) {
        const existingNotification = document.getElementById('douyin-tampermonkey-notification');
        if (existingNotification) {
            existingNotification.remove();
        }
        if (notificationTimeoutId) {
            clearTimeout(notificationTimeoutId);
        }

        const notificationDiv = document.createElement('div');
        notificationDiv.id = 'douyin-tampermonkey-notification';
        notificationDiv.textContent = message;

        Object.assign(notificationDiv.style, {
            position: 'fixed',
            top: '20px',
            left: '50%',
            transform: 'translateX(-50%)',
            padding: '10px 20px',
            borderRadius: '8px',
            color: '#fff',
            fontSize: '14px',
            zIndex: '99999',
            opacity: '0',
            transition: 'opacity 0.3s ease-in-out',
            pointerEvents: 'none', // 默认不响应鼠标事件
            boxShadow: '0 4px 12px rgba(0, 0, 0, 0.15)'
        });

        switch (type) {
            case 'info':
                notificationDiv.style.backgroundColor = 'rgba(0, 0, 0, 0.7)';
                break;
            case 'success':
                notificationDiv.style.backgroundColor = 'rgba(76, 175, 80, 0.7)';
                break;
            case 'warning':
                notificationDiv.style.backgroundColor = 'rgba(255, 152, 0, 0.7)';
                break;
            case 'error':
                notificationDiv.style.backgroundColor = 'rgba(244, 67, 54, 0.7)';
                break;
            default:
                notificationDiv.style.backgroundColor = 'rgba(0, 0, 0, 0.7)';
        }

        document.body.appendChild(notificationDiv);
        notificationDiv.offsetHeight; // 强制 reflow,确保动画生效
        notificationDiv.style.opacity = '1';

        notificationTimeoutId = setTimeout(() => {
            notificationDiv.style.opacity = '0';
            notificationDiv.addEventListener('transitionend', () => {
                notificationDiv.remove();
            }, { once: true });
        }, duration);
    }

    // --- 定制首次启动提示框 ---
    function showCustomFirstRunNotification() {
        const existingNotification = document.getElementById('douyin-tampermonkey-custom-notification');
        if (existingNotification) {
            existingNotification.remove();
        }

        const customNotificationDiv = document.createElement('div');
        customNotificationDiv.id = 'douyin-tampermonkey-custom-notification';
        customNotificationDiv.style.cssText = `
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background-color: rgba(0, 0, 0, 0.85);
            padding: 25px 35px;
            border-radius: 10px;
            color: #fff;
            font-size: 16px;
            text-align: center;
            z-index: 100000;
            box-shadow: 0 5px 15px rgba(0, 0, 0, 0.3);
            display: flex;
            flex-direction: column;
            gap: 20px;
            max-width: 80%;
            pointer-events: auto; /* 允许鼠标事件 */
            opacity: 0;
            transition: opacity 0.3s ease-in-out;
        `;

        const messageSpan = document.createElement('span');
        messageSpan.innerHTML = `
            按下键盘上的快捷键 P 来启动或关闭播完暂停功能,<br>
            如果要开启连播功能,请自行关闭播完暂停功能,否则连播功能会失效。
        `;
        messageSpan.style.lineHeight = '1.5';
        messageSpan.style.fontWeight = 'bold';

        const button = document.createElement('button');
        button.textContent = '我知道了';
        button.style.cssText = `
            background-color: #fe2c55; /* 抖音红 */
            color: white;
            border: none;
            padding: 10px 20px;
            border-radius: 5px;
            cursor: pointer;
            font-size: 16px;
            transition: background-color 0.2s ease-in-out;
        `;
        button.onmouseover = () => button.style.backgroundColor = '#d02648';
        button.onmouseout = () => button.style.backgroundColor = '#fe2c55';
        button.onclick = () => {
            customNotificationDiv.style.opacity = '0';
            customNotificationDiv.addEventListener('transitionend', () => {
                customNotificationDiv.remove();
            }, { once: true });
        };

        customNotificationDiv.appendChild(messageSpan);
        customNotificationDiv.appendChild(button);
        document.body.appendChild(customNotificationDiv);

        // 强制 reflow,确保动画生效
        customNotificationDiv.offsetHeight;
        customNotificationDiv.style.opacity = '1';
    }


    // --- 辅助函数 ---

    /**
     * 获取当前正在播放的视频元素。
     */
    function getCurrentVideoElement() {
        const videoElements = document.querySelectorAll('video');

        for (const video of videoElements) {
            if (video.offsetWidth > 0 && video.offsetHeight > 0 && video.duration > 0 && !video.paused) {
                // 优先选择抖音主视频播放器容器内的视频
                const parentContainer = video.closest('.web-player-video-container, .xgplayer-container, .player-container');
                if (parentContainer) {
                    return video;
                }
                // 如果没有找到特定的父容器,但它是一个可见且正在播放的视频,也可能就是我们要找的
                return video;
            }
        }
        return null;
    }

    // --- 核心逻辑 ---

    let currentVideo = null;
    const PAUSE_THRESHOLD = 0.2; // 提前暂停的秒数

    // 用于跟踪视频是否已经被处理过,避免重复触发
    let videoProcessedFlag = false;

    /**
     * 处理视频 timeupdate 事件的逻辑。
     */
    function handleVideoTimeUpdate() {
        if (!scriptEnabled) return;

        if (this.duration > 0 && !this.paused && !this.ended && !videoProcessedFlag) {
            const remainingTime = this.duration - this.currentTime;

            if (remainingTime <= PAUSE_THRESHOLD) {
                this.pause();
                console.log(`Tampermonkey: 已在视频结束前 ${PAUSE_THRESHOLD} 秒暂停。`);
                showNotification(`视频已暂停`, 'success');
                videoProcessedFlag = true;
                // 移除监听器,等待下一个视频重新绑定
                if (currentVideo) {
                    currentVideo.removeEventListener('timeupdate', handleVideoTimeUpdate);
                    currentVideo.removeEventListener('ended', handleVideoEnded);
                }
            }
        }
    }

    /**
     * 备用:处理视频播放完全结束的逻辑 (以防 timeupdate 不够精确)
     */
    function handleVideoEnded() {
        if (!scriptEnabled) return;

        console.log('Tampermonkey: 视频播放完全结束(备用触发)。');
        this.pause();
        showNotification('视频已暂停', 'success');
        videoProcessedFlag = true;
        // 移除监听器
        if (currentVideo) {
            currentVideo.removeEventListener('timeupdate', handleVideoTimeUpdate);
            currentVideo.removeEventListener('ended', handleVideoEnded);
        }
    }

    /**
     * 监听视频播放事件,并在视频变化时更新监听器。
     */
    function setupVideoListener() {
        if (!scriptEnabled) return;

        const video = getCurrentVideoElement();

        if (video && (video !== currentVideo || (currentVideo && video.src !== currentVideo.src))) {
            console.log('Tampermonkey: 检测到新视频或视频源改变,正在重新设置监听器。');

            if (currentVideo) {
                currentVideo.removeEventListener('timeupdate', handleVideoTimeUpdate);
                currentVideo.removeEventListener('ended', handleVideoEnded);
                console.log('Tampermonkey: 已移除旧视频监听器。');
            }

            currentVideo = video;
            videoProcessedFlag = false; // 重置处理标记

            currentVideo.addEventListener('timeupdate', handleVideoTimeUpdate);
            currentVideo.addEventListener('ended', handleVideoEnded);
            console.log('Tampermonkey: 已为新视频设置 timeupdate 和 ended 监听器。');
        } else if (!video && currentVideo) {
            currentVideo.removeEventListener('timeupdate', handleVideoTimeUpdate);
            currentVideo.removeEventListener('ended', handleVideoEnded);
            currentVideo = null;
            videoProcessedFlag = false;
            console.log('Tampermonkey: 当前视频元素已消失,已清理监听器。');
        }
    }

    // --- 快捷键监听 ---
    document.addEventListener('keydown', (event) => {
        if (event.target.tagName === 'INPUT' || event.target.tagName === 'TEXTAREA') {
            return;
        }

        if (event.key === 'P' || event.key === 'p') {
            event.preventDefault();
            setScriptEnabled(!scriptEnabled);
        }
    });

    // --- 页面URL检测 ---
    function checkPageUrlAndDisableScript() {
        if (window.location.href.includes('https://live.douyin.com/')) {
            if (scriptEnabled) {
                setScriptEnabled(false);
                showNotification('检测到直播页面,脚本已自动禁用', 'error', 3000);
            }
        }
    }

    // --- 首次启动判断 ---
    function checkFirstRunAndShowNotification() {
        const isFirstRun = GM_getValue(FIRST_RUN_KEY, true);
        if (isFirstRun) {
            showCustomFirstRunNotification(); // 显示定制的首次启动提示
            GM_setValue(FIRST_RUN_KEY, false); // 标记已运行过
        }
    }

    // --- 启动脚本 ---

    // 使用 MutationObserver 监视 DOM 变化
    const observer = new MutationObserver(mutations => {
        if (!scriptEnabled) return;

        let videoRelatedChange = false;
        for (const mutation of mutations) {
            if (mutation.type === 'childList' || (mutation.type === 'attributes' && mutation.attributeName === 'src')) {
                videoRelatedChange = true;
                break;
            }
        }

        if (videoRelatedChange) {
            setupVideoListener();
        }
    });

    // 只有在脚本启用时才开始观察
    if (scriptEnabled) {
        observer.observe(document.body, {
            childList: true,
            subtree: true,
            attributes: true,
            attributeFilter: ['data-e2e-state', 'src']
        });
    }

    // 页面加载或从其他页面导航回来时
    window.addEventListener('load', () => {
        checkPageUrlAndDisableScript();
        checkFirstRunAndShowNotification(); // 检查并显示首次启动提示
        if (scriptEnabled) {
            setupVideoListener();
            showNotification(`脚本已${scriptEnabled ? '启用' : '禁用'} (按 'P' 切换)`, scriptEnabled ? 'info' : 'warning');
        } else {
            showNotification('脚本已禁用 (按 \'P\' 启用)', 'warning');
        }
    });

    // 监听URL变化(适用于单页应用,如抖音)
    let lastUrl = location.href;
    new MutationObserver(() => {
        if (location.href !== lastUrl) {
            lastUrl = location.href;
            checkPageUrlAndDisableScript();
            if (scriptEnabled) {
                setupVideoListener();
            }
        }
    }).observe(document, { subtree: true, childList: true });

    // 初始检查URL和显示脚本状态
    checkPageUrlAndDisableScript();
    checkFirstRunAndShowNotification(); // 初始加载时也检查一次
    showNotification(`脚本已${scriptEnabled ? '启用' : '禁用'} (按 'P' 切换)`, scriptEnabled ? 'info' : 'warning');

})();