115调试 - 稳定版通用视频控制器 (Netflix 风格 + UI修复 + 人脸拦截 + 跳转后更新 + 修复暂停/继续 + 加载时双重更新 + 99%跳转 + 图标修复)

为 HTML5 视频添加 Netflix 风格悬浮控件,默认播放速度0.1x,修复UI显示问题,支持一键部署(快速跳转至99%、暂停、更新进度、自动取消弹窗按新逻辑、跳过已学习视频),停止/继续部署,拖动控件,拦截人脸识别,跳转后自动更新进度,修复停止/继续部署问题,每次加载视频时更新两次进度,跳转目标调整为99%,修复按钮图标显示。

// ==UserScript==
// @name         115调试 - 稳定版通用视频控制器 (Netflix 风格 + UI修复 + 人脸拦截 + 跳转后更新 + 修复暂停/继续 + 加载时双重更新 + 99%跳转 + 图标修复)
// @namespace    http://tampermonkey.net/
// @version      3.5_zh-CN_fixed_deploy_pause_resume_double_update_99percent_icons
// @description  为 HTML5 视频添加 Netflix 风格悬浮控件,默认播放速度0.1x,修复UI显示问题,支持一键部署(快速跳转至99%、暂停、更新进度、自动取消弹窗按新逻辑、跳过已学习视频),停止/继续部署,拖动控件,拦截人脸识别,跳转后自动更新进度,修复停止/继续部署问题,每次加载视频时更新两次进度,跳转目标调整为99%,修复按钮图标显示。
// @author       Gemini (由 Google 开发), Grok (优化)
// @match        *://gd.aqscwlxy.com/*
// @exclude      https://*.netflix.com/*
// @grant        GM_addStyle
// @run-at       document-start
// @license      MIT
// @icon         
// ==/UserScript==

(function() {
    'use strict';

    // --- 配置 ---
    const PANEL_ID = 'universal-video-controller-panel-zh-v13';
    const SPEEDS = [0.1, 0.5, 1.0];
    const FIXED_JUMP_PERCENTAGES = [
        { label: '50%', value: 50 },
        { label: '99%', value: 99 }
    ];
    const SKIP_AMOUNT = 10;
    const HIDE_DELAY = 1800;
    const MIN_VIDEO_SIZE = 100;
    const SAFE_ACTION_FLAG = '_UVC_safeAction_v13';
    const AUTO_SLOWDOWN_THRESHOLD = 3.0;
    const AUTO_SLOWDOWN_FLAG = '_UVC_isAutoSlowdown_v13';
    const SLOWDOWN_RATE = 0.1;
    const PIN_PANEL_FLAG = '_UVC_isPanelPinned_v13';
    const CONFIRM_TEXT = '确认';
    const CANCEL_TEXT = '取消';
    const GOTO_TEST_TEXT = '去测试';
    const UPDATE_TEXT = '更新进度';
    const BLOCKED_URL = "https://gd.aqscwlxy.com/gd_api/face/launch_face_auth.php";
    const STUDYRECORD_URL = '/gd_api/study/studyrecord.php';
    const BLOCK_EVENTS = ['blur', 'visibilitychange', 'pagehide', 'freeze'];
    const UPDATE_CLICK_COUNT = 5;
    const FAKE_SUCCESS_RESPONSE = { code: 200, msg: "上传成功 (由脚本修改)", data: null };
    const FAKE_SUCCESS_RESPONSE_TEXT = JSON.stringify(FAKE_SUCCESS_RESPONSE);
    const DOUBLE_UPDATE_DELAY = 100; // 双重更新之间的延迟(毫秒)

    // --- 状态变量 ---
    let currentVideo = null;
    let controlPanel = null;
    let hideTimeout = null;
    let activeSpeedButton = null;
    let playPauseButton = null;
    let dragButton = null;
    let muteButton = null;
    let deployButton = null;
    let stopDeployButton = null;
    let currentTimeDisplay = null;
    let durationDisplay = null;
    let timeUpdateInterval = null;
    let progressBarContainer = null;
    let progressTrack = null;
    let progressBuffer = null;
    let progressFill = null;
    let progressThumb = null;
    let isDraggingProgress = false;
    let wasPausedBeforeDrag = false;
    let progressPercentDisplay = null;
    let originalSpeedBeforeSlowdown = null;
    let pinPanelButton = null;
    let allowPauseAtSlowSpeed = true;
    let isDeploying = false;
    let isDeploymentPaused = false;
    let isDraggingPanel = false;
    let dragStartX = 0;
    let dragStartY = 0;
    window[AUTO_SLOWDOWN_FLAG] = false;
    window[PIN_PANEL_FLAG] = true;

    // --- 日志函数 ---
    function log(message) { console.log('[VideoController]', message); }

    // --- 拦截人脸识别和强制更新学习记录 ---
    if (window.location.hostname === 'gd.aqscwlxy.com') {
        const originalOpen = XMLHttpRequest.prototype.open;
        XMLHttpRequest.prototype.open = function(method, url) {
            this._url = url;
            this._method = method;
            if (typeof url === 'string') {
                if (url.includes(STUDYRECORD_URL) && method.toUpperCase() === 'POST') {
                    this.addEventListener('readystatechange', function() {
                        if (this.readyState === 4) {
                            Object.defineProperties(this, {
                                response: { value: FAKE_SUCCESS_RESPONSE_TEXT, writable: true, configurable: true },
                                responseText: { value: FAKE_SUCCESS_RESPONSE_TEXT, writable: true, configurable: true },
                                status: { value: 200, writable: true, configurable: true },
                                statusText: { value: 'OK (faked)', writable: true, configurable: true }
                            });
                            log('学习记录已强制更新为成功');
                        }
                    }, false);
                }
                if (url === BLOCKED_URL) {
                    this.addEventListener('readystatechange', function() {
                        if (this.readyState === 4) {
                            Object.defineProperties(this, {
                                status: { value: 0, writable: true, configurable: true },
                                statusText: { value: 'Blocked by script', writable: true, configurable: true }
                            });
                            log('人脸识别请求被拦截');
                        }
                    }, false);
                }
            }
            originalOpen.apply(this, arguments);
        };

        const originalFetch = window.fetch;
        window.fetch = function(input, init) {
            let url = (typeof input === 'string') ? input : (input ? input.url : '');
            let method = (init && init.method) ? init.method.toUpperCase() : 'GET';
            if (typeof url === 'string') {
                if (url.includes(STUDYRECORD_URL) && method === 'POST') {
                    const fakeResponse = new Response(FAKE_SUCCESS_RESPONSE_TEXT, {
                        status: 200,
                        statusText: 'OK (faked)',
                        headers: { 'Content-Type': 'application/json;charset=utf-8' }
                    });
                    log('学习记录 Fetch 请求已强制返回成功');
                    return Promise.resolve(fakeResponse);
                }
                if (url === BLOCKED_URL) {
                    log('人脸识别 Fetch 请求被拦截');
                    return Promise.reject(new Error("人脸识别请求被脚本拦截"));
                }
            }
            return originalFetch.apply(this, arguments);
        };
    }

    // --- 防护功能 ---
    if (window.location.hostname === 'gd.aqscwlxy.com') {
        const winAdd = window.addEventListener;
        window.addEventListener = function(type, listener, options) {
            if (BLOCK_EVENTS.includes(type)) return;
            winAdd.call(this, type, listener, options);
        };

        const docAdd = document.addEventListener;
        document.addEventListener = function(type, listener, options) {
            if (BLOCK_EVENTS.includes(type)) return;
            docAdd.call(this, type, listener, options);
        };

        Object.defineProperty(document, 'visibilityState', { get: () => 'visible', configurable: true });
        Object.defineProperty(document, 'hidden', { get: () => false, configurable: true });
        document.hasFocus = () => true;

        const originalPause = HTMLMediaElement.prototype.pause;
        HTMLMediaElement.prototype.pause = function() {
            if (window[SAFE_ACTION_FLAG]) {
                window[SAFE_ACTION_FLAG] = false;
                return originalPause.apply(this, arguments);
            }
            const fromScript = !(new Error().stack.includes('native code'));
            if (fromScript && !allowPauseAtSlowSpeed) {
                return;
            }
            return originalPause.apply(this, arguments);
        };

        window.addEventListener('load', () => {
            const originalCurrentTime = Object.getOwnPropertyDescriptor(HTMLMediaElement.prototype, 'currentTime');
            Object.defineProperty(HTMLMediaElement.prototype, 'currentTime', {
                get: function() { return originalCurrentTime.get.call(this); },
                set: function(val) {
                    if (window[SAFE_ACTION_FLAG]) {
                        window[SAFE_ACTION_FLAG] = false;
                        return originalCurrentTime.set.call(this, val);
                    }
                    if (typeof val === 'number' && val < this.currentTime - 1) {
                        return;
                    }
                    return originalCurrentTime.set.call(this, val);
                },
                configurable: true
            });
        });
    }

    // --- 样式 ---
    GM_addStyle(`
        #${PANEL_ID} { position: fixed; bottom: 20px; left: 50%; transform: translateX(-50%); background-color: rgba(20, 20, 20, 0.9); border-radius: 8px; padding: 10px 15px; flex-direction: column; gap: 8px; z-index: 2147483647; font-family: Arial, sans-serif; backdrop-filter: blur(6px); box-shadow: 0 3px 12px rgba(0, 0, 0, 0.6); color: #fff; transition: opacity 0.3s ease; width: clamp(550px, 75vw, 950px); box-sizing: border-box; cursor: move; }
        #${PANEL_ID}.visible { display: flex; opacity: 1; bottom: 30px; }
        #${PANEL_ID} .progress-bar-container { width: 100%; height: 16px; display: flex; align-items: center; cursor: pointer; padding: 5px 0; }
        #${PANEL_ID} .progress-track { width: 100%; height: 5px; background-color: rgba(100, 100, 100, 0.6); border-radius: 3px; position: relative; overflow: hidden; }
        #${PANEL_ID} .progress-buffer { position: absolute; height: 100%; width: 0%; background-color: rgba(150, 150, 150, 0.6); }
        #${PANEL_ID} .progress-fill { position: absolute; height: 100%; width: 0%; background-color: #E50914; z-index: 1; }
        #${PANEL_ID} .progress-thumb { position: absolute; top: 50%; width: 14px; height: 14px; background-color: #E50914; border: 1px solid rgba(255,255,255,0.8); border-radius: 50%; transform: translate(-50%, -50%); z-index: 2; cursor: pointer; }
        #${PANEL_ID} .controls-row { display: flex; width: 100%; justify-content: space-between; align-items: center; gap: 10px; }
        #${PANEL_ID} .controls-left, #${PANEL_ID} .controls-center, #${PANEL_ID} .controls-right { display: flex; align-items: center; gap: 8px; }
        #${PANEL_ID} .controls-center { flex-grow: 1; justify-content: center; gap: 6px; }
        #${PANEL_ID} .progress-percent-display { font-size: 13px; font-weight: bold; color: #E0E0E0; min-width: 45px; text-align: right; margin-right: 10px; }
        #${PANEL_ID} .speed-controls { display: flex; align-items: center; gap: 6px; }
        #${PANEL_ID} .speed-controls button.active { background-color: rgba(229, 9, 20, 0.85); border-color: #fff; font-weight: bold; }
        #${PANEL_ID} .jump-controls-group { display: flex; align-items: center; gap: 5px; border-left: 1px solid #555; padding-left: 10px; }
        #${PANEL_ID} button { background: rgba(70, 70, 70, 0.75); color: #fff; border: 1px solid rgba(255, 255, 255, 0.35); border-radius: 5px; padding: 5px 9px; cursor: pointer; font-size: 12px; transition: background-color 0.2s ease; min-width: 28px; display: flex; align-items: center; justify-content: center; }
        #${PANEL_ID} button:hover { background: rgba(100, 100, 100, 0.9); }
        #${PANEL_ID} .deploy-button { background: rgba(0, 128, 0, 0.75); padding: 5px 12px; }
        #${PANEL_ID} .deploy-button:hover { background: rgba(0, 160, 0, 0.9); }
        #${PANEL_ID} .stop-deploy-button { background: rgba(255, 0, 0, 0.75); padding: 5px 12px; }
        #${PANEL_ID} .stop-deploy-button:hover { background: rgba(255, 50, 50, 0.9); }
        #${PANEL_ID} .continue-deploy-button { background: rgba(0, 128, 0, 0.75); padding: 5px 12px; }
        #${PANEL_ID} .continue-deploy-button:hover { background: rgba(0, 160, 0, 0.9); }
        #${PANEL_ID} .time-display { font-size: 12px; color: #ddd; min-width: 80px; text-align: center; }
        #${PANEL_ID} .pin-panel-button.pinned { background-color: rgba(229, 9, 20, 0.85); }
        #${PANEL_ID} .drag-button::before { content: '✥'; font-size: 15px; }
        #${PANEL_ID} .play-button::before { content: '▶'; font-size: 12px; }
        #${PANEL_ID} .pause-button::before { content: '❚❚'; font-size: 12px; }
        #${PANEL_ID} .skip-back-button::before { content: '⏪'; font-size: 14px; }
        #${PANEL_ID} .skip-forward-button::before { content: '⏩'; font-size: 14px; }
        #${PANEL_ID} .mute-button::before { content: '🔇'; font-size: 14px; }
        #${PANEL_ID} .unmute-button::before { content: '🔊'; font-size: 14px; }
        #${PANEL_ID} .deploy-button::before { content: '🚀'; font-size: 14px; margin-right: 4px; }
        #${PANEL_ID} .stop-deploy-button::before { content: '⏹'; font-size: 12px; margin-right: 4px; }
        #${PANEL_ID} .continue-deploy-button::before { content: '▶'; font-size: 12px; margin-right: 4px; }
        #${PANEL_ID} .pin-panel-button::before { content: '📌'; font-size: 14px; }
        .auto-dismiss-alert { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background: rgba(255, 0, 0, 0.9); color: white; padding: 15px 25px; border-radius: 5px; z-index: 2147483647; font-size: 16px; }
        .chapter-end-notification { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background-color: rgba(0, 0, 0, 0.8); color: white; padding: 15px 25px; border-radius: 8px; z-index: 10000; }
    `);

    // --- 核心功能函数 ---
    function setSafeAction(enable) { window[SAFE_ACTION_FLAG] = enable; }

    function findButtons(text) {
        return Array.from(document.querySelectorAll('button')).filter(button => {
            const isVisible = button.offsetParent !== null;
            const isEnabled = !button.disabled;
            return isVisible && isEnabled && button.textContent?.trim().includes(text);
        });
    }

    function checkForCancelButton() {
        const cancelButtons = findButtons(CANCEL_TEXT);
        const gotoTestButtons = findButtons(GOTO_TEST_TEXT);
        const confirmButtons = findButtons(CONFIRM_TEXT);

        if (cancelButtons.length > 0) {
            if (gotoTestButtons.length > 0) {
                log(`发现 "${CANCEL_TEXT}" 和 "${GOTO_TEST_TEXT}",点击 "${CANCEL_TEXT}"`);
                cancelButtons[0].click();
                return true;
            }
            if (confirmButtons.length > 0) {
                log(`发现 "${CANCEL_TEXT}" 和 "${CONFIRM_TEXT}",点击 "${CONFIRM_TEXT}"`);
                confirmButtons[0].click();
                return true;
            }
        }
        return false;
    }

    function isUpdateButtonPresent() {
        const buttons = document.querySelectorAll('div.ro-comment > span:nth-child(2)');
        return Array.from(buttons).some(button => button.textContent.trim() === UPDATE_TEXT && isVisible(button));
    }

    function isVideoStudied(item) {
        const finishElement = item.querySelector('.ro-finish');
        return finishElement && finishElement.textContent.includes('已学习');
    }

    function showAutoDismissAlert(message) {
        const alertDiv = document.createElement('div');
        alertDiv.className = 'auto-dismiss-alert';
        alertDiv.textContent = message;
        document.body.appendChild(alertDiv);
        setTimeout(() => alertDiv.remove(), 2500);
    }

    function showChapterEndNotification(message) {
        const notification = document.createElement('div');
        notification.className = 'chapter-end-notification';
        notification.textContent = message;
        document.body.appendChild(notification);
        setTimeout(() => notification.remove(), 2500);
    }

    function updateProgress(actionType) {
        const buttons = document.querySelectorAll('div.ro-comment > span:nth-child(2)');
        buttons.forEach(button => {
            if (button.textContent.trim() === UPDATE_TEXT && isVisible(button)) {
                for (let i = 0; i < UPDATE_CLICK_COUNT; i++) {
                    setTimeout(() => {
                        button.click();
                        log(`点击 "${UPDATE_TEXT}" 第 ${i + 1} 次 (${actionType})`);
                    }, i * 50);
                }
            }
        });
    }

    async function doubleUpdateProgress(actionType) {
        if (isUpdateButtonPresent()) {
            updateProgress(`${actionType} - 第一次`);
            await new Promise(resolve => setTimeout(resolve, UPDATE_CLICK_COUNT * 50 + DOUBLE_UPDATE_DELAY));
            updateProgress(`${actionType} - 第二次`);
            await new Promise(resolve => setTimeout(resolve, UPDATE_CLICK_COUNT * 50 + 50));
        } else {
            log(`未找到“${UPDATE_TEXT}”按钮,跳过${actionType}双重更新`);
        }
    }

    function updateProgressOnDeploy() {
        return new Promise(resolve => {
            if (isUpdateButtonPresent()) {
                updateProgress("部署或跳转后更新");
                setTimeout(resolve, UPDATE_CLICK_COUNT * 50 + 50);
            } else {
                log("未找到“更新进度”按钮,跳过更新");
                resolve();
            }
        });
    }

    function getVideoItems() {
        const container = document.querySelector('div.items');
        return container ? Array.from(container.querySelectorAll('div.item')) : [];
    }

    function getActiveItem() {
        const container = document.querySelector('div.items');
        return container ? container.querySelector('div.item.item-active') : null;
    }

    function isVisible(elem) {
        if (!(elem instanceof Element)) return false;
        const style = window.getComputedStyle(elem);
        return style.display !== 'none' && style.visibility !== 'hidden' && elem.offsetParent !== null;
    }

    function createControlPanel() {
        if (document.getElementById(PANEL_ID)) return;

        controlPanel = document.createElement('div');
        controlPanel.id = PANEL_ID;
        controlPanel.style.display = 'none';

        progressBarContainer = document.createElement('div');
        progressBarContainer.className = 'progress-bar-container';
        progressTrack = document.createElement('div');
        progressTrack.className = 'progress-track';
        progressBuffer = document.createElement('div');
        progressBuffer.className = 'progress-buffer';
        progressFill = document.createElement('div');
        progressFill.className = 'progress-fill';
        progressThumb = document.createElement('div');
        progressThumb.className = 'progress-thumb';
        progressTrack.appendChild(progressBuffer);
        progressTrack.appendChild(progressFill);
        progressTrack.appendChild(progressThumb);
        progressBarContainer.appendChild(progressTrack);
        controlPanel.appendChild(progressBarContainer);
        progressBarContainer.addEventListener('click', handleProgressBarClick);
        progressBarContainer.addEventListener('mousedown', handleDragStart);

        const controlsRow = document.createElement('div');
        controlsRow.className = 'controls-row';

        const controlsLeft = document.createElement('div');
        controlsLeft.className = 'controls-left';
        playPauseButton = document.createElement('button');
        playPauseButton.className = 'play-button';
        playPauseButton.title = "播放/暂停";
        playPauseButton.addEventListener('click', handlePlayPause);
        controlsLeft.appendChild(playPauseButton);
        const skipBackButton = document.createElement('button');
        skipBackButton.className = 'skip-back-button';
        skipBackButton.title = `后退 ${SKIP_AMOUNT} 秒`;
        skipBackButton.addEventListener('click', handleSkipBack);
        controlsLeft.appendChild(skipBackButton);
        const skipForwardButton = document.createElement('button');
        skipForwardButton.className = 'skip-forward-button';
        skipForwardButton.title = `快进 ${SKIP_AMOUNT} 秒`;
        skipForwardButton.addEventListener('click', handleSkipForward);
        controlsLeft.appendChild(skipForwardButton);

        const controlsCenter = document.createElement('div');
        controlsCenter.className = 'controls-center';
        progressPercentDisplay = document.createElement('span');
        progressPercentDisplay.className = 'progress-percent-display';
        progressPercentDisplay.textContent = '0%';
        controlsCenter.appendChild(progressPercentDisplay);
        const speedContainer = document.createElement('div');
        speedContainer.className = 'speed-controls';
        const speedLabel = document.createElement('span');
        speedLabel.textContent = '速度:';
        speedContainer.appendChild(speedLabel);
        SPEEDS.forEach(speed => {
            const speedButton = document.createElement('button');
            speedButton.textContent = `${speed}x`;
            speedButton.dataset.speed = speed;
            speedButton.addEventListener('click', handleSpeedChange);
            if (speed === 0.1) {
                speedButton.classList.add('active');
                activeSpeedButton = speedButton;
            }
            speedContainer.appendChild(speedButton);
        });
        controlsCenter.appendChild(speedContainer);

        const jumpGroupContainer = document.createElement('div');
        jumpGroupContainer.className = 'jump-controls-group';
        FIXED_JUMP_PERCENTAGES.forEach(jump => {
            const jumpButton = document.createElement('button');
            jumpButton.textContent = jump.label;
            jumpButton.dataset.percent = jump.value;
            jumpButton.addEventListener('click', () => jumpToPercentAndPause(jump.value));
            jumpGroupContainer.appendChild(jumpButton);
        });
        controlsCenter.appendChild(jumpGroupContainer);

        const controlsRight = document.createElement('div');
        controlsRight.className = 'controls-right';
        const timeContainer = document.createElement('div');
        timeContainer.className = 'time-display';
        currentTimeDisplay = document.createElement('span');
        durationDisplay = document.createElement('span');
        timeContainer.appendChild(currentTimeDisplay);
        timeContainer.appendChild(document.createTextNode(' / '));
        timeContainer.appendChild(durationDisplay);
        controlsRight.appendChild(timeContainer);
        pinPanelButton = document.createElement('button');
        pinPanelButton.className = 'pin-panel-button';
        pinPanelButton.title = '固定/取消固定面板';
        pinPanelButton.addEventListener('click', togglePinPanel);
        controlsRight.appendChild(pinPanelButton);
        deployButton = document.createElement('button');
        deployButton.className = 'deploy-button';
        deployButton.innerHTML = '一键部署';
        deployButton.addEventListener('click', handleOneClickDeploy);
        controlsRight.appendChild(deployButton);
        stopDeployButton = document.createElement('button');
        stopDeployButton.className = 'stop-deploy-button';
        stopDeployButton.innerHTML = '停止部署';
        stopDeployButton.addEventListener('click', toggleDeployPause);
        controlsRight.appendChild(stopDeployButton);
        muteButton = document.createElement('button');
        muteButton.className = 'mute-button';
        muteButton.title = '静音/取消静音';
        muteButton.addEventListener('click', handleMuteToggle);
        controlsRight.appendChild(muteButton);
        dragButton = document.createElement('button');
        dragButton.className = 'drag-button';
        dragButton.title = '拖动面板';
        dragButton.addEventListener('mousedown', startDraggingPanel);
        controlsRight.appendChild(dragButton);

        controlsRow.appendChild(controlsLeft);
        controlsRow.appendChild(controlsCenter);
        controlsRow.appendChild(controlsRight);
        controlPanel.appendChild(controlsRow);
        document.body.appendChild(controlPanel);
        updatePinButtonState();
        log("控制面板已创建 (v3.5 - 图标修复)");
    }

    function formatTime(seconds) {
        if (isNaN(seconds) || !isFinite(seconds)) return "--:--";
        const date = new Date(0);
        date.setSeconds(Math.round(seconds));
        return date.toISOString().substr(11, 8).startsWith('00:') ? date.toISOString().substr(14, 5) : date.toISOString().substr(11, 8);
    }

    function updateProgressAndTimers() {
        if (!currentVideo || !controlPanel) return;
        const currentTime = currentVideo.currentTime;
        const duration = currentVideo.duration;
        currentTimeDisplay.textContent = formatTime(currentTime);
        durationDisplay.textContent = formatTime(duration);
        if (duration && isFinite(duration)) {
            const progressPercent = (currentTime / duration) * 100;
            progressFill.style.width = `${progressPercent}%`;
            progressThumb.style.left = `${progressPercent}%`;
            progressBuffer.style.width = `${calculateBufferPercent(currentVideo, currentTime, duration)}%`;
            progressPercentDisplay.textContent = `${Math.round(progressPercent)}%`;
            const remainingTime = duration - currentTime;
            if (remainingTime <= AUTO_SLOWDOWN_THRESHOLD && !currentVideo.paused && !window[AUTO_SLOWDOWN_FLAG]) {
                originalSpeedBeforeSlowdown = currentVideo.playbackRate;
                setSafeAction(true);
                currentVideo.playbackRate = SLOWDOWN_RATE;
                setSafeAction(false);
                window[AUTO_SLOWDOWN_FLAG] = true;
                updateControlState(currentVideo);
            } else if (remainingTime > AUTO_SLOWDOWN_THRESHOLD && window[AUTO_SLOWDOWN_FLAG]) {
                restoreOriginalSpeed();
            }
        }
    }

    function calculateBufferPercent(video, currentTime, duration) {
        let bufferedEnd = currentTime;
        if (video.buffered && video.buffered.length > 0) {
            for (let i = 0; i < video.buffered.length; i++) {
                if (video.buffered.start(i) <= currentTime && video.buffered.end(i) > currentTime) {
                    bufferedEnd = video.buffered.end(i);
                    break;
                }
            }
        }
        return (bufferedEnd / duration) * 100;
    }

    function restoreOriginalSpeed() {
        if (window[AUTO_SLOWDOWN_FLAG] && currentVideo) {
            const speedToRestore = originalSpeedBeforeSlowdown || 0.1;
            setSafeAction(true);
            currentVideo.playbackRate = speedToRestore;
            setSafeAction(false);
            window[AUTO_SLOWDOWN_FLAG] = false;
            originalSpeedBeforeSlowdown = null;
            updateControlState(currentVideo);
        }
    }

    function updateControlState(video) {
        if (!controlPanel || !video) return;
        playPauseButton.className = video.paused ? 'play-button' : 'pause-button';
        playPauseButton.title = video.paused ? "播放" : "暂停";
        const currentRate = video.playbackRate;
        controlPanel.querySelectorAll('.speed-controls button').forEach(btn => {
            btn.classList.toggle('active', Math.abs(parseFloat(btn.dataset.speed) - currentRate) < 0.01);
        });
        muteButton.className = video.muted ? 'mute-button' : 'unmute-button';
        muteButton.title = video.muted ? "取消静音" : "静音";
        updateProgressAndTimers();
        if (!video.paused && !timeUpdateInterval) {
            timeUpdateInterval = setInterval(updateProgressAndTimers, 250);
        } else if (video.paused && timeUpdateInterval) {
            clearInterval(timeUpdateInterval);
            timeUpdateInterval = null;
        }
    }

    function showControls(video) {
        cancelHide();
        if (!controlPanel) createControlPanel();
        currentVideo = video;
        controlPanel.style.display = 'flex';
        controlPanel.classList.add('visible');
        updateControlState(video);
    }

    function scheduleHide() {
        cancelHide();
        if (!window[PIN_PANEL_FLAG] && !isDraggingPanel && !isDraggingProgress) {
            hideTimeout = setTimeout(() => {
                if (controlPanel) controlPanel.classList.remove('visible');
            }, HIDE_DELAY);
        }
    }

    function cancelHide() {
        if (hideTimeout) clearTimeout(hideTimeout);
        hideTimeout = null;
    }

    function handlePlayPause(e) {
        e.stopPropagation();
        if (!currentVideo) return;
        setSafeAction(true);
        if (currentVideo.paused) {
            currentVideo.play();
        } else {
            currentVideo.pause();
        }
        setSafeAction(false);
        updateControlState(currentVideo);
    }

    function resetAutoSlowdownIfNeeded() {
        if (!currentVideo || !currentVideo.duration || !isFinite(currentVideo.duration)) return;
        const remainingTime = currentVideo.duration - currentVideo.currentTime;
        if (remainingTime > AUTO_SLOWDOWN_THRESHOLD && window[AUTO_SLOWDOWN_FLAG]) {
            restoreOriginalSpeed();
        }
    }

    function handleSkipBack(e) {
        e.stopPropagation();
        if (!currentVideo || !currentVideo.duration) return;
        setSafeAction(true);
        currentVideo.currentTime = Math.max(0, currentVideo.currentTime - SKIP_AMOUNT);
        setSafeAction(false);
        resetAutoSlowdownIfNeeded();
        updateProgressAndTimers();
    }

    function handleSkipForward(e) {
        e.stopPropagation();
        if (!currentVideo || !currentVideo.duration) return;
        setSafeAction(true);
        currentVideo.currentTime = Math.min(currentVideo.duration, currentVideo.currentTime + SKIP_AMOUNT);
        setSafeAction(false);
        resetAutoSlowdownIfNeeded();
        updateProgressAndTimers();
    }

    function handleSpeedChange(e) {
        e.stopPropagation();
        if (!currentVideo || isDeploying) return;
        const speed = parseFloat(e.target.dataset.speed);
        setSafeAction(true);
        currentVideo.playbackRate = speed;
        setSafeAction(false);
        allowPauseAtSlowSpeed = speed <= 0.5;
        if (!allowPauseAtSlowSpeed && currentVideo.paused) {
            setSafeAction(true);
            currentVideo.play();
            setSafeAction(false);
        }
        if (window[AUTO_SLOWDOWN_FLAG]) {
            window[AUTO_SLOWDOWN_FLAG] = false;
            originalSpeedBeforeSlowdown = null;
        }
        updateControlState(currentVideo);
    }

    function handleMuteToggle(e) {
        e.stopPropagation();
        if (!currentVideo) return;
        currentVideo.muted = !currentVideo.muted;
        updateControlState(currentVideo);
    }

    async function handleOneClickDeploy(e) {
        e.stopPropagation();
        if (!currentVideo || isDeploying) return;
        if (!isUpdateButtonPresent()) {
            showAutoDismissAlert("请等待“更新进度”按钮出现后再部署!");
            return;
        }

        isDeploying = true;
        isDeploymentPaused = false;
        let currentItem = getActiveItem();
        if (!currentItem) {
            isDeploying = false;
            log("未找到活动项目,部署中止");
            return;
        }

        const startItem = currentItem;
        let passesCompleted = 0;

        log("开始一键部署");

        while (passesCompleted < 2 && isDeploying) {
            let currentPassItem = currentItem;
            log(`第 ${passesCompleted + 1} 轮部署开始,当前项目: ${currentPassItem.textContent.trim().substring(0, 10)}...`);

            while (isDeploying) {
                if (isDeploymentPaused) {
                    log("部署暂停,等待继续");
                    await waitForResume();
                    if (!isDeploying) break;
                    log("部署继续");
                }

                if (isVideoStudied(currentPassItem)) {
                    const nextElement = currentPassItem.nextElementSibling;
                    if (nextElement && nextElement.classList.contains('item')) {
                        currentPassItem = nextElement;
                        nextElement.click();
                        log(`跳过已学习视频,切换至: ${nextElement.textContent.trim().substring(0, 10)}...`);
                        await waitForVideoLoad();
                        continue;
                    } else {
                        log("没有更多视频可切换,结束当前轮次");
                        break;
                    }
                }

                setSafeAction(true);
                currentVideo.playbackRate = 0.1;
                setSafeAction(false);

                if (passesCompleted === 0) {
                    jumpToPercentAndPause(99);
                    await updateProgressOnDeploy();
                } else {
                    await updateProgressOnDeploy();
                }

                checkForCancelButton();

                const nextElement = currentPassItem.nextElementSibling;
                if (nextElement && nextElement.classList.contains('item')) {
                    nextElement.click();
                    currentPassItem = nextElement;
                    log(`切换至下一视频: ${nextElement.textContent.trim().substring(0, 10)}...`);
                    await waitForVideoLoad();
                } else {
                    log("本轮部署完成,无下一视频");
                    break;
                }
            }

            passesCompleted++;
            if (passesCompleted < 2 && isDeploying) {
                startItem.click();
                currentItem = startItem;
                log("返回起始项目,开始第二轮部署");
                await waitForVideoLoad();
            } else if (isDeploying) {
                showChapterEndNotification("🎉 本章已全部部署完成!");
                log("部署完成");
            }
        }
        isDeploying = false;
        log("一键部署结束");
    }

    function toggleDeployPause(e) {
        e.stopPropagation();
        if (!isDeploying) return;
        isDeploymentPaused = !isDeploymentPaused;
        stopDeployButton.className = isDeploymentPaused ? 'continue-deploy-button' : 'stop-deploy-button';
        stopDeployButton.innerHTML = isDeploymentPaused ? '继续部署' : '停止部署';
        log(isDeploymentPaused ? "部署已暂停" : "部署已恢复");
    }

    function waitForResume() {
        return new Promise(resolve => {
            const checkResume = () => {
                if (!isDeploymentPaused || !isDeploying) {
                    resolve();
                } else {
                    setTimeout(checkResume, 100);
                }
            };
            checkResume();
        });
    }

    async function waitForVideoLoad() {
        return new Promise(resolve => {
            const timeout = setTimeout(() => {
                resolve();
            }, 2000);
            const checkVideo = () => {
                const newVideo = document.querySelector('video');
                if (newVideo && newVideo !== currentVideo && newVideo.readyState >= 2) {
                    currentVideo = newVideo;
                    clearTimeout(timeout);
                    log("新视频已加载,准备双重更新");
                    doubleUpdateProgress("视频加载完成").then(() => {
                        resolve();
                    });
                } else {
                    setTimeout(checkVideo, 50);
                }
            };
            checkVideo();
        });
    }

    function handleProgressBarClick(e) {
        if (!currentVideo || !currentVideo.duration || isDraggingProgress) return;
        e.stopPropagation();
        const rect = progressTrack.getBoundingClientRect();
        const seekRatio = Math.max(0, Math.min(1, (e.clientX - rect.left) / rect.width));
        setSafeAction(true);
        currentVideo.currentTime = seekRatio * currentVideo.duration;
        setSafeAction(false);
        resetAutoSlowdownIfNeeded();
        updateProgressAndTimers();
    }

    function handleDragStart(e) {
        if (!currentVideo || !currentVideo.duration) return;
        e.stopPropagation();
        cancelHide();
        isDraggingProgress = true;
        wasPausedBeforeDrag = currentVideo.paused;
        if (timeUpdateInterval) clearInterval(timeUpdateInterval);
        document.addEventListener('mousemove', handleDragMove);
        document.addEventListener('mouseup', handleDragEnd);
        handleDragMove(e);
    }

    function handleDragMove(e) {
        if (!isDraggingProgress || !currentVideo) return;
        e.preventDefault();
        const rect = progressTrack.getBoundingClientRect();
        const seekRatio = Math.max(0, Math.min(1, (e.clientX - rect.left) / rect.width));
        const seekTime = seekRatio * currentVideo.duration;
        progressFill.style.width = `${seekRatio * 100}%`;
        progressThumb.style.left = `${seekRatio * 100}%`;
        currentTimeDisplay.textContent = formatTime(seekTime);
    }

    async function handleDragEnd(e) {
        if (!isDraggingProgress || !currentVideo) return;
        e.stopPropagation();
        document.removeEventListener('mousemove', handleDragMove);
        document.removeEventListener('mouseup', handleDragEnd);
        const rect = progressTrack.getBoundingClientRect();
        const seekRatio = Math.max(0, Math.min(1, (e.clientX - rect.left) / rect.width));
        setSafeAction(true);
        currentVideo.currentTime = seekRatio * currentVideo.duration;
        currentVideo.pause();
        setSafeAction(false);
        resetAutoSlowdownIfNeeded();
        isDraggingProgress = false;
        updateProgressAndTimers();
        if (isUpdateButtonPresent()) {
            await updateProgressOnDeploy();
            log("进度条拖动后已更新进度");
        }
        scheduleHide();
    }

    async function jumpToPercentAndPause(percent) {
        if (!currentVideo || !currentVideo.duration) return;
        const seekTime = (percent / 100) * currentVideo.duration;
        setSafeAction(true);
        currentVideo.currentTime = seekTime;
        currentVideo.pause();
        setSafeAction(false);
        resetAutoSlowdownIfNeeded();
        updateControlState(currentVideo);
        if (isUpdateButtonPresent()) {
            await updateProgressOnDeploy();
            log(`跳转至 ${percent}% 后已更新进度`);
        }
    }

    function updatePinButtonState() {
        if (!pinPanelButton) return;
        pinPanelButton.classList.toggle('pinned', window[PIN_PANEL_FLAG]);
    }

    function togglePinPanel(e) {
        e.stopPropagation();
        window[PIN_PANEL_FLAG] = !window[PIN_PANEL_FLAG];
        updatePinButtonState();
        if (!window[PIN_PANEL_FLAG]) scheduleHide();
        else cancelHide();
    }

    function startDraggingPanel(e) {
        e.preventDefault();
        isDraggingPanel = true;
        dragStartX = e.clientX - (parseInt(controlPanel.style.left) || window.innerWidth / 2 - controlPanel.offsetWidth / 2);
        dragStartY = e.clientY - (window.innerHeight - (parseInt(controlPanel.style.bottom) || 20) - controlPanel.offsetHeight);
        document.addEventListener('mousemove', dragPanel);
        document.addEventListener('mouseup', stopDraggingPanel);
        controlPanel.style.transition = 'none';
    }

    function dragPanel(e) {
        if (!isDraggingPanel) return;
        e.preventDefault();
        const x = e.clientX - dragStartX;
        const y = window.innerHeight - (e.clientY - dragStartY) - controlPanel.offsetHeight;
        controlPanel.style.left = `${x}px`;
        controlPanel.style.bottom = `${y}px`;
        controlPanel.style.transform = 'none';
    }

    function stopDraggingPanel() {
        isDraggingPanel = false;
        document.removeEventListener('mousemove', dragPanel);
        document.removeEventListener('mouseup', stopDraggingPanel);
        controlPanel.style.transition = 'opacity 0.3s ease';
        scheduleHide();
    }

    function isValidVideo(video) {
        return video.offsetWidth >= MIN_VIDEO_SIZE && video.offsetHeight >= MIN_VIDEO_SIZE && !video._hasListener_v13;
    }

    async function attachListenersToVideo(video) {
        if (!isValidVideo(video)) return;
        video.addEventListener('mouseenter', () => showControls(video));
        video.addEventListener('mouseleave', scheduleHide);
        video.addEventListener('play', () => showControls(video));
        video.addEventListener('pause', () => updateControlState(video));
        video.addEventListener('ratechange', () => updateControlState(video));
        video.addEventListener('loadedmetadata', async () => {
            video.muted = true;
            setSafeAction(true);
            video.playbackRate = 0.1;
            setSafeAction(false);
            showControls(video);
            await doubleUpdateProgress("视频元数据加载完成");
        });
        video.addEventListener('seeked', updateProgressAndTimers);
        video.addEventListener('progress', updateProgressAndTimers);
        video._hasListener_v13 = true;
    }

    function scanForVideos() {
        document.querySelectorAll('video').forEach(attachListenersToVideo);
    }

    createControlPanel();
    if (document.readyState === 'complete' || document.readyState === 'interactive') {
        scanForVideos();
    } else {
        document.addEventListener('DOMContentLoaded', scanForVideos);
    }

    const observer = new MutationObserver(mutations => {
        mutations.forEach(mutation => {
            mutation.addedNodes.forEach(node => {
                if (node.nodeType === Node.ELEMENT_NODE) {
                    if (node.tagName === 'VIDEO') attachListenersToVideo(node);
                    else node.querySelectorAll('video').forEach(attachListenersToVideo);
                }
            });
            checkForCancelButton();
        });
    });
    observer.observe(document.body, { childList: true, subtree: true });

    log("通用视频控制器 (v3.5 - UI修复、人脸拦截、跳转后更新、修复暂停/继续、加载时双重更新、99%跳转、图标修复) 已加载");
})();