Auto scroll to anchor on dynamic pages

Tries to scroll to an anchor on pages with dynamic content loading by repeatedly scrolling down. Handles hash changes.

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

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         Auto scroll to anchor on dynamic pages
// @name:en      Auto scroll to anchor on dynamic pages
// @name:ru      Автоматическая прокрутка к якорю на динамических страницах
// @namespace    http://tampermonkey.net/
// @version      2025-05-15_15-5 // Не забывайте обновлять версию
// @description  Tries to scroll to an anchor on pages with dynamic content loading by repeatedly scrolling down. Handles hash changes.
// @description:en  Tries to scroll to an anchor on pages with dynamic content loading by repeatedly scrolling down. Handles hash changes.
// @description:ru  Пытается прокрутить до якоря на страницах с динамической загрузкой контента, многократно прокручивая вниз. Обрабатывает изменения хеша.
// @author       Igor Lebedev + (DeepSeek and Gemini Pro)
// @license        GPL-3.0-or-later
// @match        *://*/*
// @icon         
// @grant        none
// @run-at       document-start
// ==/UserScript==

(function() {
    'use strict';

    // --- Определение среды выполнения ---
    let executionEnvironment = 'userscript'; // По умолчанию считаем, что это userscript
    let logPrefix = "[AutoScrollToAnchor]";    // Префикс для логов userscript

    try {
        if (typeof browser !== 'undefined' && browser.runtime && browser.runtime.id) {
            executionEnvironment = 'extension_firefox';
            logPrefix = "[AutoScrollExt FF]";
        } else if (typeof chrome !== 'undefined' && chrome.runtime && chrome.runtime.id) {
            // Это может быть Chrome расширение или Firefox (где chrome является псевдонимом browser)
            // Для большей точности можно проверить chrome.runtime.getURL("").startsWith("moz-extension://") для Firefox
            if (chrome.runtime.getURL && chrome.runtime.getURL("").startsWith("moz-extension://")) {
                executionEnvironment = 'extension_firefox';
                logPrefix = "[AutoScrollExt FF]";
            } else {
                executionEnvironment = 'extension_chrome_or_edge'; // или другое на базе Chromium
                logPrefix = "[AutoScrollExt Cr]";
            }
        }
    } catch (e) {
        // Ошибка доступа к browser.runtime или chrome.runtime может возникнуть, если они не определены.
        // В этом случае остаемся со значением по умолчанию 'userscript'.
    }
    // ------------------------------------

    // --- Попытка сохранить исходный якорь ---
    let initialAnchorOnLoad = window.location.hash.substring(1);
    if (initialAnchorOnLoad) {
        log(`Initial anchor detected at document-start: #${initialAnchorOnLoad}`);
    }
    // --------------------------------------------

    // Настройки (остаются общими)
    const MAX_ATTEMPTS = 30;
    const SCROLL_INTERVAL_MS = 750;
    const SCROLL_AMOUNT_PX = window.innerHeight * 0.8;
    const FAST_CHECK_DELAY_MS = 250;
    const INITIAL_DELAY_MS = 500;

    let currentIntervalId = null;
    let currentSearchAnchorName = '';

    // Используем определенный ранее logPrefix
    function log(message) {
        console.log(`${logPrefix} ${message}`);
    }
    if (executionEnvironment !== 'userscript') { // Для расширений можно добавить ID расширения, если нужно
        log(`Running as ${executionEnvironment}. Extension ID (if applicable): ${ (typeof browser !== 'undefined' && browser.runtime && browser.runtime.id) || (typeof chrome !== 'undefined' && chrome.runtime && chrome.runtime.id) || 'N/A'}`);
    } else {
        log(`Running as ${executionEnvironment}.`);
    }


    function stopCurrentSearch(reason = "generic stop") {
        if (currentIntervalId) {
            clearInterval(currentIntervalId);
            currentIntervalId = null;
            log(`Search for #${currentSearchAnchorName} stopped. Reason: ${reason}`);
        }
    }

    // ... (функции log, stopCurrentSearch, findAndScrollToElement, startSearchingForAnchor) ...
    // Важно: findAndScrollToElement должна быть готова к тому, что currentUrlAnchor может быть пуст,
    // если сайт успел его удалить, но мы все еще ищем initialAnchorOnLoad.

    /**
     * Модифицированная findAndScrollToElement
     * @param {string} anchorNameToFind - Имя якоря для поиска.
     * @param {string} currentExpectedUrlAnchor - Якорь, который мы ОЖИДАЕМ сейчас в URL (может быть '' если сайт его удалил).
     * @returns {boolean}
     */
    function findAndScrollToElement(anchorNameToFind, currentExpectedUrlAnchor) {
        if (!anchorNameToFind) return false;

        // Проверка, не изменился ли ЦЕЛЕВОЙ якорь, к которому мы стремимся
        // (например, если пользователь кликнул на другой якорь уже после начала поиска)
        const actualCurrentUrlAnchor = window.location.hash.substring(1);
        if (actualCurrentUrlAnchor && actualCurrentUrlAnchor !== anchorNameToFind && actualCurrentUrlAnchor !== currentExpectedUrlAnchor) {
            log(`User navigated to a new anchor #${actualCurrentUrlAnchor} while searching for #${anchorNameToFind}. Stopping this search.`);
            return false; // Пользователь перешел на другой якорь, этот поиск неактуален
        }

        const elementById = document.getElementById(anchorNameToFind);
        const elementByName = !elementById ? document.querySelector(`[name="${anchorNameToFind}"]`) : null;
        const targetElement = elementById || elementByName;

        if (targetElement) {
            log(`Anchor #${anchorNameToFind} found.`);
            targetElement.scrollIntoView({ behavior: 'smooth', block: 'center' });

            const originalBg = targetElement.style.backgroundColor;
            targetElement.style.backgroundColor = 'yellow';
            setTimeout(() => {
                targetElement.style.backgroundColor = originalBg;
            }, 2000);

            return true;
        }
        return false;
    }

    /**
     * Модифицированная startSearchingForAnchor
     * @param {string} anchorNameToSearch - Имя якоря для поиска.
     * @param {string} currentUrlAnchorAtStart - Якорь, который был в URL в момент инициации этого поиска
     */
    function startSearchingForAnchor(anchorNameToSearch, currentUrlAnchorAtStart) {
        stopCurrentSearch(`starting new search for #${anchorNameToSearch}`);

        if (!anchorNameToSearch) {
            log("No anchor specified for search, nothing to do.");
            currentSearchAnchorName = '';
            return;
        }

        currentSearchAnchorName = anchorNameToSearch; // Это якорь, который мы ИЩЕМ на странице
        log(`Starting search for anchor: #${currentSearchAnchorName} (URL hash was #${currentUrlAnchorAtStart || 'empty'} at initiation)`);

        let attempts = 0;

        if (findAndScrollToElement(currentSearchAnchorName, currentUrlAnchorAtStart)) {
            stopCurrentSearch(`found #${currentSearchAnchorName} immediately`);
            return;
        }

        currentIntervalId = setInterval(() => {
            // Важно: передаем currentUrlAnchorAtStart, чтобы findAndScrollToElement
            // знала, какой якорь мы ожидали в URL на момент начала поиска.
            if (findAndScrollToElement(currentSearchAnchorName, currentUrlAnchorAtStart)) {
                stopCurrentSearch(`found #${currentSearchAnchorName} after scrolling`);
                return;
            }

            attempts++;
            if (attempts > MAX_ATTEMPTS) {
                console.warn(`${logPrefix} Anchor #${currentSearchAnchorName} not found after ${MAX_ATTEMPTS} attempts.`);
                stopCurrentSearch(`max attempts reached for #${currentSearchAnchorName}`);
                return;
            }

            log(`Attempt ${attempts}/${MAX_ATTEMPTS} for #${currentSearchAnchorName}: Scrolling down...`);
            window.scrollBy(0, SCROLL_AMOUNT_PX);

            setTimeout(() => {
                if (!currentIntervalId) return;
                if (findAndScrollToElement(currentSearchAnchorName, currentUrlAnchorAtStart)) {
                    stopCurrentSearch(`found #${currentSearchAnchorName} after scroll and fast check`);
                }
            }, FAST_CHECK_DELAY_MS);
        }, SCROLL_INTERVAL_MS);
    }

    function initialLoadOrHashChangeHandler() {
        let anchorToActUpon = window.location.hash.substring(1);
        let currentUrlAnchorForContext = anchorToActUpon; // Якорь, который СЕЙЧАС в URL

        if (!anchorToActUpon && initialAnchorOnLoad) {
            // Если в URL якоря нет, НО он был при самой первой загрузке
            log(`URL hash is empty, but an initial anchor #${initialAnchorOnLoad} was detected. Attempting to use it.`);
            anchorToActUpon = initialAnchorOnLoad;
            // currentUrlAnchorForContext остается пустым, так как сайт его удалил
        }

        // Сброс initialAnchorOnLoad после первой попытки его использовать,
        // чтобы при последующих hashchange (если пользователь кликает по другим якорям на странице)
        // мы не пытались вернуться к самому первому якорю.
        // Но делаем это только если мы действительно собираемся действовать (т.е. anchorToActUpon не пуст)
        if (anchorToActUpon) {
            initialAnchorOnLoad = null; // Используем его только один раз
        }


        // Это условие для предотвращения перезапуска, если хеш не изменился, немного усложняется.
        // Мы должны сравнивать anchorToActUpon с тем, что мы активно ищем (currentSearchAnchorName).
        if (anchorToActUpon === currentSearchAnchorName && currentIntervalId !== null) {
             // Если мы уже ищем этот якорь (или пытались искать), и поиск активен, ничего не делаем.
            return;
        }

        if (!anchorToActUpon && currentSearchAnchorName) {
            stopCurrentSearch(`Anchor removed or no longer relevant (was #${currentSearchAnchorName})`);
            currentSearchAnchorName = '';
            return;
        }

        // Запускаем поиск, передавая якорь, который нужно найти,
        // и якорь, который был в URL в момент принятия решения о поиске.
        startSearchingForAnchor(anchorToActUpon, currentUrlAnchorForContext);
    }

    function onPageReady() {
        // initialAnchorOnLoad уже должен быть установлен здесь
        log(`onPageReady. Initial anchor was: #${initialAnchorOnLoad || 'none'}. Current hash: #${window.location.hash.substring(1) || 'none'}`);
        setTimeout(initialLoadOrHashChangeHandler, INITIAL_DELAY_MS);
        window.addEventListener('hashchange', initialLoadOrHashChangeHandler, false);
    }

    // Код для запуска при загрузке страницы
    // Эта логика подходит для обоих случаев, так как @run-at document-start и
    // "run_at": "document_start" в manifest.json ведут себя схожим образом.
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', onPageReady, { once: true });
    } else {
        // DOMContentLoaded уже сработал, или мы находимся в состоянии interactive/complete
        onPageReady();
    }


})();