Greasy Fork

Greasy Fork is available in English.

平滑滚动翻页

使用w和s键进行平滑滚动翻页。键盘左右键或A/D键模拟点击页面中【上一页|上一章】【下一页|下一章】按钮,双击只点击【上一章】【下一章】按钮。

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         平滑滚动翻页
// @namespace    http://tampermonkey.net/
// @version      1.8
// @description  使用w和s键进行平滑滚动翻页。键盘左右键或A/D键模拟点击页面中【上一页|上一章】【下一页|下一章】按钮,双击只点击【上一章】【下一章】按钮。
// @author       coccvo
// @include      *
// @exclude      https://www.bilibili.com/*
// @grant        none
// @icon         
// @license MIT
// ==/UserScript==

(function() {
    'use strict';

    let initialized = false;
    let clickTimer = null;

    // 模拟点击函数
    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 hasNextPageButton() {
        return simulateClickByText('下一页') || simulateClickByText('下一页>');
    }

    function hasLastPageButton() {
        return simulateClickByText('上一页') || simulateClickByText('上一页>');
    }

    function initialize() {
        if (initialized) return;
        initialized = true;

        let viewportHeight = window.innerHeight;
        let scrollDistance = viewportHeight * 0.9;

        // 监听按键事件
        document.addEventListener('keydown', function(event) {
            if (event.ctrlKey || event.altKey || event.shiftKey) return;
            // 检查事件是否发生在输入框内部
            const tag = event.target.tagName.toLowerCase();
            if (tag === 'input' || tag === 'textarea') return;

            switch(event.key.toLowerCase()) {
                case 'w':
                case 'arrowup':
                    event.preventDefault();
                    window.scrollBy({
                        top: -scrollDistance,
                        left: 0,
                        behavior: 'smooth'
                    });
                    break;
                case 's':
                case 'arrowdown':
                    event.preventDefault();
                    window.scrollBy({
                        top: scrollDistance,
                        left: 0,
                        behavior: 'smooth'
                    });
                    break;
                case 'arrowleft':
                case 'a':
                    handlePageNavigation('prev');
                    break;
                case 'arrowright':
                case 'd':
                    handlePageNavigation('next');
                    break;
            }
        });
    }

    // 处理页面导航
    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();
        }
    }

    // 页面加载完毕后执行
    window.addEventListener('load', function() {
        initialize();
    });

    // 窗口获取焦点时初始化(仅第一次)
    window.addEventListener('focus', function() {
        initialize();
    });
})();