Codeforces Rating Hider

Changes the displayed difficulty on Codeforces problemset to XXXX

이 스크립트를 설치하려면 Tampermonkey, Greasemonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

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

이 스크립트를 설치하려면 Tampermonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey 또는 Userscripts와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 유저 스크립트 관리자 확장 프로그램이 필요합니다.

(이미 유저 스크립트 관리자가 설치되어 있습니다. 설치를 진행합니다!)

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

(이미 유저 스타일 관리자가 설치되어 있습니다. 설치를 진행합니다!)

// ==UserScript==
// @name         Codeforces Rating Hider
// @namespace    http://tampermonkey.net/
// @version      1.0.0
// @description  Changes the displayed difficulty on Codeforces problemset to XXXX
// @author       Zafir Hasan Anogh
// @match        https://codeforces.com/
// @match        https://codeforces.com/*
// @grant        none
// @run-at       document-idle
// @license MIT
// ==/UserScript==

(function() {
    'use strict';

    function hideDifficulty() {
        const difficultySpans = document.querySelectorAll('span[title="Difficulty"]');

        difficultySpans.forEach(span => {
            if (span.innerText !== 'XXXX') {
                span.innerText = 'XXXX';
            }
        });
    }

    hideDifficulty();

    const observer = new MutationObserver(function(mutationsList, observer) {
        let needsUpdate = false;
        for (const mutation of mutationsList) {
            if (mutation.type === 'childList' && mutation.addedNodes.length > 0) {
                for (const node of mutation.addedNodes) {
                    if (node.nodeType === Node.ELEMENT_NODE) {
                        if (node.matches('span[title="Difficulty"]') || node.querySelector('span[title="Difficulty"]')) {
                            needsUpdate = true;
                            break;
                        }
                    }
                }
            }
            if (needsUpdate) break;
        }

        if (needsUpdate) {
            hideDifficulty();
        } else {
            const anyVisibleDifficulty = document.querySelector('span[title="Difficulty"]:not(:empty)');
            if (anyVisibleDifficulty && anyVisibleDifficulty.innerText !== 'XXXX') {
                hideDifficulty();
            }
        }
    });

    const targetNode = document.body;
    const config = { childList: true, subtree: true };

    if (targetNode) {
        observer.observe(targetNode, config);
    }

    window.addEventListener('load', hideDifficulty);

})();