// ==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-14
// @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';
// Настройки
const MAX_ATTEMPTS = 30; // Максимальное количество попыток прокрутки
const SCROLL_INTERVAL_MS = 750; // Интервал между попытками в миллисекундах
const SCROLL_AMOUNT_PX = window.innerHeight * 0.8; // На сколько прокручивать за раз (80% высоты окна)
const FAST_CHECK_DELAY_MS = 250; // Задержка для быстрой проверки после скролла
const INITIAL_DELAY_MS = 500; // Начальная задержка перед первым запуском
let currentIntervalId = null; // ID текущего интервала прокрутки
let currentSearchAnchorName = ''; // Имя якоря, который активно ищется
function log(message) {
console.log(`[AutoScrollToAnchor] ${message}`);
}
/**
* Останавливает текущий активный поиск якоря.
* @param {string} reason - Причина остановки для логирования.
*/
function stopCurrentSearch(reason = "generic stop") {
if (currentIntervalId) {
clearInterval(currentIntervalId);
currentIntervalId = null;
log(`Search for #${currentSearchAnchorName} stopped. Reason: ${reason}`);
}
// Сбрасываем имя искомого якоря, только если причина не в том, что он был найден
// (чтобы подсветка могла использовать правильное имя)
// Однако, для чистоты лучше всегда сбрасывать, а для подсветки передавать имя якоря отдельно.
// currentSearchAnchorName = ''; // Решим ниже, когда сбрасывать
}
/**
* Пытается найти элемент якоря и прокрутить к нему.
* @param {string} anchorName - Имя якоря для поиска.
* @returns {boolean} - True, если элемент найден и прокрутка выполнена, иначе false.
*/
function findAndScrollToElement(anchorName) {
// Если URL хеш изменился, пока мы искали этот anchorName, значит этот поиск уже не актуален.
const currentUrlAnchor = window.location.hash.substring(1);
if (anchorName && currentUrlAnchor !== anchorName && currentUrlAnchor !== '') {
log(`URL hash changed to #${currentUrlAnchor} while searching for #${anchorName}. Stopping this specific search.`);
// Не останавливаем глобальный поиск здесь, это сделает обработчик hashchange
return false; // Этот конкретный элемент искать больше не нужно
}
if (!anchorName) return false; // Нет якоря для поиска
const elementById = document.getElementById(anchorName);
const elementByName = !elementById ? document.querySelector(`[name="${anchorName}"]`) : null;
const targetElement = elementById || elementByName;
if (targetElement) {
log(`Anchor #${anchorName} 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;
}
/**
* Запускает процесс поиска и прокрутки к указанному якорю.
* @param {string} anchorNameToSearch - Имя якоря для поиска.
*/
function startSearchingForAnchor(anchorNameToSearch) {
stopCurrentSearch(`starting new search for #${anchorNameToSearch}`); // Останавливаем любой предыдущий поиск
if (!anchorNameToSearch) {
log("No anchor specified in URL, nothing to do.");
currentSearchAnchorName = ''; // Убедимся, что нет "зависшего" имени
return;
}
currentSearchAnchorName = anchorNameToSearch; // Устанавливаем новый искомый якорь
log(`Starting search for anchor: #${currentSearchAnchorName}`);
let attempts = 0;
// Попытка найти сразу
if (findAndScrollToElement(currentSearchAnchorName)) {
stopCurrentSearch(`found #${currentSearchAnchorName} immediately`);
// currentSearchAnchorName = ''; // Сбрасываем после успеха
return;
}
currentIntervalId = setInterval(() => {
const currentUrlAnchorWhenIntervalFired = window.location.hash.substring(1);
// Если хеш в URL изменился или был удален, пока работал интервал,
// и он не соответствует тому, что мы ищем, останавливаем этот поиск.
if (currentUrlAnchorWhenIntervalFired !== currentSearchAnchorName) {
log(`URL hash changed to #${currentUrlAnchorWhenIntervalFired} (or removed) while actively searching for #${currentSearchAnchorName}. Stopping this search.`);
stopCurrentSearch("URL hash changed during interval");
// Новый поиск, если он нужен, будет инициирован событием hashchange
return;
}
if (findAndScrollToElement(currentSearchAnchorName)) {
stopCurrentSearch(`found #${currentSearchAnchorName} after scrolling`);
// currentSearchAnchorName = ''; // Сбрасываем после успеха
return;
}
attempts++;
if (attempts > MAX_ATTEMPTS) {
console.warn(`[AutoScrollToAnchor] Anchor #${currentSearchAnchorName} not found after ${MAX_ATTEMPTS} attempts.`);
stopCurrentSearch(`max attempts reached for #${currentSearchAnchorName}`);
// currentSearchAnchorName = ''; // Сбрасываем после неудачи
return;
}
log(`Attempt ${attempts}/${MAX_ATTEMPTS} for #${currentSearchAnchorName}: Scrolling down...`);
window.scrollBy(0, SCROLL_AMOUNT_PX);
// Короткая проверка почти сразу после прокрутки
setTimeout(() => {
if (!currentIntervalId) return; // Поиск мог быть остановлен
const currentUrlAnchorForFastCheck = window.location.hash.substring(1);
if (currentUrlAnchorForFastCheck !== currentSearchAnchorName) {
// Если хеш изменился за время этой короткой задержки
return;
}
if (findAndScrollToElement(currentSearchAnchorName)) {
stopCurrentSearch(`found #${currentSearchAnchorName} after scroll and fast check`);
// currentSearchAnchorName = ''; // Сбрасываем после успеха
}
}, FAST_CHECK_DELAY_MS);
}, SCROLL_INTERVAL_MS);
}
/**
* Обработчик для первоначальной загрузки и изменения хеша URL.
*/
function initialLoadOrHashChangeHandler() {
const anchorNameFromUrl = window.location.hash.substring(1);
// Если якорь в URL тот же, что и текущий искомый, и поиск уже активен, ничего не делаем.
// Это предотвращает лишние перезапуски, если hashchange сработало, но якорь не изменился.
if (anchorNameFromUrl === currentSearchAnchorName && currentIntervalId !== null) {
// log(`Hash event for the same active anchor #${anchorNameFromUrl}. No action needed.`);
return;
}
// Если в URL нет якоря, но какой-то поиск был активен, останавливаем его.
if (!anchorNameFromUrl && currentSearchAnchorName) { // currentSearchAnchorName проверяем, чтобы не логировать без надобности
stopCurrentSearch(`anchor removed from URL (was #${currentSearchAnchorName})`);
currentSearchAnchorName = ''; // Явно сбрасываем
return;
}
// Во всех остальных случаях (новый якорь, или тот же якорь, но поиск неактивен, или якоря нет и не было)
// запускаем/перезапускаем поиск (startSearchingForAnchor сама обработает пустой anchorNameFromUrl)
startSearchingForAnchor(anchorNameFromUrl);
}
// Устанавливаем слушателей
// Используем setTimeout для initialLoadOrHashChangeHandler, чтобы дать странице "успокоиться"
// даже если DOM готов или страница полностью загружена.
function onPageReady() {
setTimeout(initialLoadOrHashChangeHandler, INITIAL_DELAY_MS);
window.addEventListener('hashchange', initialLoadOrHashChangeHandler, false);
}
if (document.readyState === 'complete' || (document.readyState !== 'loading' && !document.documentElement.doScroll)) {
// DOM уже готов или страница загружена
onPageReady();
} else {
// Дожидаемся полной загрузки DOM
document.addEventListener('DOMContentLoaded', onPageReady, { once: true });
}
})();