Greasy Fork

Greasy Fork is available in English.

平滑滚动翻页

使用上下键或WS键进行平滑滚动翻页,一次翻页0.9倍页面距离。键盘左右键或A/D键模拟点击页面中【上一页|上一章】【下一页|下一章】按钮,双击只点击【上一章】【下一章】按钮。可以与自动滚屏助手脚本联动。

当前为 2025-05-17 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         平滑滚动翻页
// @namespace    http://tampermonkey.net/
// @version      2.1
// @description  使用上下键或WS键进行平滑滚动翻页,一次翻页0.9倍页面距离。键盘左右键或A/D键模拟点击页面中【上一页|上一章】【下一页|下一章】按钮,双击只点击【上一章】【下一章】按钮。可以与自动滚屏助手脚本联动。
// @author       coccvo
// @include      *
// @exclude      https://www.bilibili.com/*
// @grant        none
// @icon         
// @license MIT
// ==/UserScript==

(function() {
    'use strict';

    // 状态变量
    let isScrollKeyPressed = false;
    let autoScrollWasActive = false;
    let lastKeyPressTime = 0;
    let clickTimer = null;
    let scrollDistance = window.innerHeight * 0.9; // 默认滚动距离

    // 检查自动滚动接口
    const isAutoScrollAvailable = () => typeof window.getAutoScrollState === 'function';

    // 检测页面按钮
    function hasNextPageButton() {
        return simulateClickByText('下一页') || simulateClickByText('下一页>');
    }
    function hasLastPageButton() {
        return simulateClickByText('上一页') || simulateClickByText('上一页>');
    }

    // 模拟点击函数
    function simulateClickByText(textContent) {
        const links = document.querySelectorAll('a');
        for (let i = 0; i < links.length; i++) {
            if (links[i].textContent.trim() === textContent) {
                links[i].click();
                return true;
            }
        }
        return false;
    }

    // 处理页面导航
    function handlePageNavigation(direction) {
        if (clickTimer == null) {
            clickTimer = setTimeout(() => {
                const isPrev = direction === 'prev';
                if (isPrev ? hasLastPageButton() : hasNextPageButton()) {
                    if (isPrev) {
                        simulateClickByText('上一页') || simulateClickByText('<上一页');
                    } else {
                        simulateClickByText('下一页') || simulateClickByText('下一页>');
                    }
                } else {
                    simulateClickByText(isPrev ? '上一章' : '下一章');
                }
                document.body.focus();
                clickTimer = null;
            }, 300);
        } else {
            clearTimeout(clickTimer);
            clickTimer = null;
            simulateClickByText(direction === 'prev' ? '上一章' : '下一章');
            document.body.focus();
        }
    }


    // 按键按下事件
    function handleKeyDown(event) {
        const target = event.target;
        if (
            target.tagName.toLowerCase() === 'input' ||
            target.tagName.toLowerCase() === 'textarea' ||
            target.isContentEditable ||
            target.closest('[contenteditable="true"]')
        ) return;

        const key = event.key.toLowerCase();
        const isScrollKey = ['w', 'arrowup', 's', 'arrowdown'].includes(key);

        if (isScrollKey) {
            isScrollKeyPressed = true;

            // 暂停自动滚动
            if (isAutoScrollAvailable() && window.getAutoScrollState()) {
                autoScrollWasActive = true;
                window.turnOffAutoScroll();
            }

            // WS键单次滚动
            if (key === 'w' || key === 'arrowup') {
                event.preventDefault();
                window.scrollBy({ top: -scrollDistance, left: 0, behavior: 'smooth' });
            } else if (key === 's' || key === 'arrowdown') {
                event.preventDefault();
                window.scrollBy({ top: scrollDistance, left: 0, behavior: 'smooth' });
            }
        }
    }
  // 统一按键处理
    function handleKeyEvents(e) {
        if (event.ctrlKey || event.altKey || event.shiftKey) return;
        // 忽略输入框中的按键
        if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA') {
            return;
        }
        switch(e.key.toLowerCase()) {
            case 'arrowleft':
            case 'a':
                handlePageNavigation('prev');
                break;
            case 'arrowright':
            case 'd':
                handlePageNavigation('next');
                break;
        }
    }
    document.addEventListener('keydown', handleKeyEvents);
    // 按键释放事件
    function handleKeyUp(event) {
        const key = event.key.toLowerCase();
        const isScrollKey = ['w', 'arrowup', 's', 'arrowdown'].includes(key);

        if (isScrollKey) {
            isScrollKeyPressed = false;
        }

        // WS键释放后延迟恢复自动滚动
        if (isScrollKey && autoScrollWasActive) {
            setTimeout(() => {
                if (!isScrollKeyPressed && isAutoScrollAvailable()) {
                    window.startAutoScroll();
                    autoScrollWasActive = false;
                }
            }, 500); // 延迟500ms恢复
        }
    }

    // 页面加载初始化
    window.addEventListener('load', () => {
        scrollDistance = window.innerHeight * 0.9; // 动态计算滚动距离
    });

    // 窗口大小变化时更新滚动距离
    window.addEventListener('resize', () => {
        scrollDistance = window.innerHeight * 0.9;
    });

    // 添加事件监听
    document.addEventListener('keydown', handleKeyDown);
    document.addEventListener('keyup', handleKeyUp);
})();