Мовний щит: youtube shorts

Додає на сторінки youtube shorts 2 кнопки: "🚫 канал" і "🚫 відео". Обидві кнопки роблять за вас автоматичні дії, щоб ви не робили це вручну. Першим ділом обидві кнопки ставлять відео на паузу, щоб не відтворювати далі відео. Кнопка "🚫 канал" звітує відео як "пропаганда тероризму" і тицяє за вас "не рекомендувати канал". Кнопка "🚫 відео" тільки звітує відео як "пропаганда тероризму".

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         Мовний щит: youtube shorts
// @namespace    https://constantine-ketskalo.azurewebsites.net/uk/project/46
// @version      1.30
// @description  Додає на сторінки youtube shorts 2 кнопки: "🚫 канал" і "🚫 відео". Обидві кнопки роблять за вас автоматичні дії, щоб ви не робили це вручну. Першим ділом обидві кнопки ставлять відео на паузу, щоб не відтворювати далі відео. Кнопка "🚫 канал" звітує відео як "пропаганда тероризму" і тицяє за вас "не рекомендувати канал". Кнопка "🚫 відео" тільки звітує відео як "пропаганда тероризму".
// @author       Constantine Ketskalo
// @match        https://www.youtube.com/*
// @icon         
// @icon64       
// @run-at       document-end
// @grant        GM_addStyle
// @license      MIT
// ==/UserScript==

GM_addStyle(`
    .anti-moskal-button {
        margin-top: 16px;
        width: 40px;
        height: 40px;
        border-radius: 50%;
        text-align: center;
        cursor: pointer;
        overflow: hidden;
        padding: 0;
        font-size: 12px;
        font-weight: bold;
        text-align: center;
        justify-content: center;
        align-items: center;
        display: flex;
        color: rgb(15, 15, 15);
        border: 4px solid rgba(15, 15, 15, 0.5);
        background-color: rgba(0,0,0,0.05);
    }

    .anti-moskal-button:hover {
        border-color: red;
    }

    .anti-moskal-button::before {
        content: '';
        position: absolute;
        width: 40px;
        height: 4px;
        background-color: rgba(15, 15, 15, 0.5);
        transform: rotate(-45deg);
        pointer-events: none;
    }

        .anti-moskal-button:hover::before {
            background-color: red;
        }

    .anti-moskal-button.video:hover {
        background: yellow;
    }

    .anti-moskal-button.channel:hover {
        background: pink;
    }

    .anti-moskal-button span {
        color: black;
        text-decoration: none;

        z-index: 1;
        padding: 10px;
    }

    .button-blocking-result {
        width: 48px;
        height: 48px;
        border-radius: 50%;
        background-color: #4CAF50; /* Зелений */
        display: flex;
        align-items: center;
        justify-content: center;
        color: white;
        font-size: 24px;
        font-weight: bold;
        font-family: sans-serif;
        user-select: none;
        margin-top: 16px;
    }

    .button-blocking-result .video {

    }

    .button-blocking-result .channel {

    }

    .hidden-button {
        display: none;
    }

    .blocked-video {
        opacity: 0.3;
        filter: grayscale(100%);
    }
`);

/* TODO: зробити 2 окремі функції: для звичайного ютубу і shorts.
 По події зміни посилання вони активуються чи деактивуються, коли відбувається перемикання між звичайним ютубом і shorts.
 */

(function() {
    'use strict';

    // ################################
    // Оголошення коду
    // ################################

    const ELEMENT_LOAD_TIMEOUT_SEC = 10000; // 10 секунд
    const ELEMENT_LOAD_INTERVAL_MS = 300; // 0.3 секунди

    const youtubeShortsMenuSelector = '#experiment-overlay #actions';

    async function pauseVideoAsync() {
        let videoElement = document.querySelector('#shorts-container video');

        if (videoElement) {
            videoElement.pause();
        }
    }

    async function waitForThingToHappenAsync(thing, errorMessage = undefined, timeout = undefined) {
        timeout = timeout ?? ELEMENT_LOAD_TIMEOUT_SEC;

        const start = Date.now();
        return new Promise((resolve, reject) => {
            const interval = setInterval(() => {
                if (thing()) {
                    clearInterval(interval);
                    resolve();
                } else if (Date.now() - start > timeout) {
                    clearInterval(interval);
                    reject(errorMessage ?? `waitForThingToHappenAsync: Timeout for thing: ${thing}`);
                }
            }, ELEMENT_LOAD_INTERVAL_MS);
        });
    }

    function selectElementByText(text, containerCssSelector = undefined) {
        const container = containerCssSelector ? document.querySelector(containerCssSelector) : document;
        if (!container) {
            return null;
        }
        // отримати по тексту через xpath, по точному тексту, цільовий елемент без дочірніх елементів
        const xpath = `.//*/text()[normalize-space()="${text}"]/parent::*`;
        return document.evaluate(xpath, container, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
    }

    // Очікує на появу елемента
    async function waitForElementAsync(selector, errorMessage = undefined, timeout = undefined) {
        return waitForThingToHappenAsync(() => {
            const el = typeof(selector) === 'function'
                        ? selector()
                        : document.querySelector(selector);

            return el ? true : false;
        },
        errorMessage,
        timeout);
    }

    function inputText(element, text) {
        element.value = text;
        element.dispatchEvent(new Event('input', { bubbles: true }));
    }

    function confirmIsUserLoggedIn() {
        const isLoggedIn = document.querySelectorAll('#avatar-btn').length > 0;
        if (!isLoggedIn) {
            alert('Вам потрібно увійти в акаунт Google, щоб поскаржитися на відео.');
        }
        return isLoggedIn;
    }

    async function markVideoAsReportedAsync() {
        document.querySelector('ytd-player#player video').classList.add('blocked-video');
        document.querySelector('.anti-moskal-button.video').classList.add('hidden-button');
        document.querySelector('.button-blocking-result.video').classList.remove('hidden-button');
    }

    async function reportVideoAsync() {
        if (!confirmIsUserLoggedIn()) {
            return;
        }

        // меню 3 крапки
        const threeDotsButtonSelector = '#button-shape .yt-spec-touch-feedback-shape__fill';
        await waitForElementAsync(threeDotsButtonSelector, `Didn't find ${threeDotsButtonSelector}`);
        document.querySelector(threeDotsButtonSelector).click();

        // кнопка "Поскаржитись"
        const reportButtonSelector = () => selectElementByText('Поскаржитись', 'ytd-popup-container');
        await waitForElementAsync(reportButtonSelector, `Didn't find ${reportButtonSelector}`);
        reportButtonSelector().scrollIntoView();
        reportButtonSelector().click();

        // радіо "Пропаганда тероризму"
        const radioTerrorismSelector = '[id="radio:8"]';
        await waitForElementAsync(radioTerrorismSelector, `Didn't find ${radioTerrorismSelector}`);
        document.querySelector(radioTerrorismSelector).scrollIntoView();
        document.querySelector(radioTerrorismSelector).click();

        // кнопка "Далі"
        const nextButtonSelector = '#bottom-bar button';
        await waitForElementAsync(nextButtonSelector, `Didn't find ${nextButtonSelector}`);
        document.querySelector(nextButtonSelector).scrollIntoView();
        document.querySelector(nextButtonSelector).click();

        // ввести причину звітування "russian propaganda"
        const reportReasonInputSelector = 'textarea';
        await waitForElementAsync(reportReasonInputSelector, `Didn't find ${reportReasonInputSelector}`);
        const reportReasonInputElement = document.querySelector(reportReasonInputSelector);
        reportReasonInputElement.scrollIntoView();
        inputText(reportReasonInputElement, 'russian propaganda');

        // кнопка "Поскаржитися"
        const submitButtonSelector = '#bottom-bar button';
        await waitForElementAsync(submitButtonSelector, `Didn't find ${submitButtonSelector}`);
        document.querySelector(submitButtonSelector).scrollIntoView();
        document.querySelector(submitButtonSelector).click();

        // індикатор зміни вікна
        const indicatorImageSelector = '.ytWebReportFormConfirmationPageViewModelImageDialog';
        await waitForElementAsync(indicatorImageSelector, `Didn't find ${indicatorImageSelector}`);

        // кнопка "OK"
        const exitButtonSelector = '#bottom-bar button';
        await waitForElementAsync(exitButtonSelector, `Didn't find ${exitButtonSelector}`);
        document.querySelector(exitButtonSelector).scrollIntoView();
        document.querySelector(exitButtonSelector).click();
    }

    async function rejectChannelRecommendationAsync() {
        // меню 3 крапки
        const threeDotsButtonSelector = '#button-shape .yt-spec-touch-feedback-shape__fill';
        await waitForElementAsync(threeDotsButtonSelector, `Didn't find ${threeDotsButtonSelector}`);
        document.querySelector(threeDotsButtonSelector).click();

        // кнопка "Не рекомендувати цей канал"
        const notInterestedButtonSelector = () => selectElementByText('Не рекомендувати цей канал', 'ytd-popup-container');
        await waitForElementAsync(notInterestedButtonSelector, `Didn't find ${notInterestedButtonSelector}`);
        notInterestedButtonSelector().scrollIntoView();
        notInterestedButtonSelector().click();
    }

    async function resetStylesAsync() {
        document.querySelector('ytd-player#player video').classList.remove('blocked-video');
        for (let button of document.querySelectorAll('.anti-moskal-button')) {
            button.classList.remove('hidden-button');
        }
        for (let resultButton of document.querySelectorAll('.button-blocking-result')) {
            resultButton.classList.add('hidden-button');
        }
    }

    async function addReportButtonsToShortsMenuAsync() {
        const menu = document.querySelector(youtubeShortsMenuSelector);

        // Створюємо елемент кнопки "🚫 відео"
        const videoButtonWrapper = document.createElement('div');
        videoButtonWrapper.className = 'anti-moskal-button video';

        const videoText = document.createElement('span');
        videoText.href = '#';
        videoText.innerText = 'відео';

        videoButtonWrapper.appendChild(videoText);

        videoButtonWrapper.onclick = async (event) => {
            event.preventDefault();
            await pauseVideoAsync();

            if (confirm('Поскаржитись на москальське відео?')) {
                await reportVideoAsync()
                    .then(() => {
                        return markVideoAsReportedAsync();
                    })
                    .catch((error) => {
                        console.error('Виникла помилка при спробі поскаржитися на відео.', error);
                    });
            }
        };

        // Створюємо елемент кнопки відео успішно заблоковане
        const videoBlockingResult = document.createElement('div');
        videoBlockingResult.className = 'button-blocking-result video hidden-button';
        videoBlockingResult.textContent = '✓';

        // Створюємо елемент кнопки "🚫 канал"
        const channelButtonWrapper = document.createElement('div');
        channelButtonWrapper.className = 'anti-moskal-button channel';

        const channelText = document.createElement('span');
        channelText.href = '#';
        channelText.innerText = 'канал';

        channelButtonWrapper.appendChild(channelText);

        channelButtonWrapper.onclick = async (event) => {
            event.preventDefault();

            await pauseVideoAsync();

            if (!confirm('Поскаржитись на москальське відео і видалити канал з рекомендацій?')) {
                return;
            }

            const isVideoAlreadyReported = document.querySelector('.button-blocking-result.video:not(.hidden-button)');

            if (!isVideoAlreadyReported) {
                await reportVideoAsync();
            }
            
            await rejectChannelRecommendationAsync();
        };

        // створення елементів для меню
        menu.appendChild(videoButtonWrapper);
        menu.appendChild(videoBlockingResult);
        menu.appendChild(channelButtonWrapper);
    }

    // ################################
    // Виконання коду
    // ################################

    // скинути стилі відео і кнопок при прокручуванні на інше відео shorts
    // або додати кнопки, коли користувач переходить на shorts з основного ютубу
    window.navigation.addEventListener("navigate", async (event) => {
        let initialUrl = window.location.href;
        await waitForThingToHappenAsync(() => {
            return window.location.href !== initialUrl;
        });

        if (!window.location.href.includes('youtube.com/shorts')) {
            return;
        }

        // дочекатись появи меню youtube shorts
        await waitForElementAsync(youtubeShortsMenuSelector, `Didn't find ${youtubeShortsMenuSelector}`)
            .then(() => {
                if (document.querySelectorAll('#experiment-overlay #actions .anti-moskal-button').length == 0) {
                    return addReportButtonsToShortsMenuAsync();
                }
                else {
                    return resetStylesAsync();
                }
            })
            .catch((error) => {
                console.error('waiting for youtube shorts menu failed', error);
            });
    });

    // дочекатись появи меню youtube shorts
    waitForElementAsync(youtubeShortsMenuSelector, `Didn't find ${youtubeShortsMenuSelector}`)
    .then(() => {
        if (document.querySelectorAll('#experiment-overlay #actions .anti-moskal-button').length == 0) {
            return addReportButtonsToShortsMenuAsync();
        }
    });
})();