GitHub PR Load All Comments

Adds "Load all!" button to load all hidden conversations in GitHub PRs

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, Greasemonkey alebo Violentmonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, % alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey alebo Userscripts.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie správcu používateľských skriptov.

(Už mám správcu používateľských skriptov, nechajte ma ho nainštalovať!)

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

(Už mám správcu používateľských štýlov, nechajte ma ho nainštalovať!)

// ==UserScript==
// @name         GitHub PR Load All Comments
// @namespace    http://tampermonkey.net/
// @icon         https://github.githubassets.com/favicons/favicon-dark.png
// @version      1.0.2
// @description  Adds "Load all!" button to load all hidden conversations in GitHub PRs
// @author       KakkoiDev
// @match        https://github.com/*
// @grant        none
// @license      MIT
// ==/UserScript==

// Broad @match needed: GitHub uses SPA navigation, so navigating to a PR
// doesn't reload the page. MutationObserver handles detecting "Load more" buttons.

(function() {
    'use strict';

    let isLoadingAll = false;

    function getLoadMoreButtons() {
        return [...document.querySelectorAll('button')].filter(btn =>
            btn.textContent.trim().toLowerCase().includes('load more')
        );
    }

    function clickAllLoadMore() {
        const buttons = getLoadMoreButtons();
        buttons.forEach(btn => btn.click());
        return buttons.length;
    }

    function addLoadAllButtons() {
        const buttons = getLoadMoreButtons();
        buttons.forEach(btn => {
            const parent = btn.parentElement;
            if (!parent || parent.querySelector('.load-all-btn')) return;

            const loadAllBtn = document.createElement('button');
            loadAllBtn.type = 'button';
            loadAllBtn.className = btn.className + ' load-all-btn';
            loadAllBtn.textContent = 'Load all!';

            loadAllBtn.addEventListener('click', (e) => {
                e.preventDefault();
                e.stopPropagation();
                isLoadingAll = true;
                clickAllLoadMore();
            });

            parent.appendChild(loadAllBtn);
        });

        if (isLoadingAll && buttons.length > 0) {
            clickAllLoadMore();
        }
    }

    addLoadAllButtons();

    const observer = new MutationObserver(() => {
        addLoadAllButtons();
    });

    observer.observe(document.body, {
        childList: true,
        subtree: true
    });
})();