// ==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');
})();