4chan Vim Navigation

Navigate 4chan threads with vim-style keybindings

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         4chan Vim Navigation
// @namespace    http://tampermonkey.net/
// @version      0.1
// @description  Navigate 4chan threads with vim-style keybindings
// @match        *://boards.4chan.org/*
// @grant        none
// @license      MIT
// ==/UserScript==

(function() {
    'use strict';

    const threads = Array.from(document.querySelectorAll('.thread'));
    let index = 0;
    let buffer = ''; // For capturing gg

    if (threads.length === 0) return;

    function highlightThread(i) {
        threads.forEach(t => t.style.outline = '');
        const thread = threads[i];
        if (thread) {
            thread.scrollIntoView({behavior: "smooth", block: "center"});
            thread.style.outline = '3px solid #4cafef';
        }
    }

    highlightThread(index);

    document.addEventListener('keydown', (e) => {
        // Ignore when typing in inputs/textareas
        if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA' || e.target.isContentEditable) {
            return;
        }

        if (e.key === 'j') {
            index = Math.min(index + 1, threads.length - 1);
            highlightThread(index);
            buffer = '';
        } else if (e.key === 'k') {
            index = Math.max(index - 1, 0);
            highlightThread(index);
            buffer = '';
        } else if (e.key === 'o') {
            const link = threads[index].querySelector('a[href*="/thread/"]');
            if (link) window.location.href = link.href;
            buffer = '';
        } else if (e.key === 'O') {
            const link = threads[index].querySelector('a[href*="/thread/"]');
            if (link) window.open(link.href, '_blank');
            buffer = '';
        } else if (e.key === 'g') {
            if (buffer === 'g') {
                index = 0;
                highlightThread(index);
                buffer = '';
            } else {
                buffer = 'g';
                setTimeout(() => buffer = '', 500); // reset if not pressed quickly
            }
        } else if (e.key === 'G') {
            index = threads.length - 1;
            highlightThread(index);
            buffer = '';
        }
    });
})();