// ==UserScript==
// @name Мовний щит: youtube shorts
// @namespace https://constantine-ketskalo.azurewebsites.net/uk/project/46
// @version 1.20
// @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 1 (1.1, 1.2):
Коли висота екрану не дуже велика (наприклад 731 px), то кнопки стискаються вертикально і погано виглядають.
Якщо обгорнути в додаткові дівки з потрібними класами, як в закоментованому коді, то вони не сплескуються по висоті,
але виникає інша проблема: перший елемент в меню вилізає за меню наверх. Якщо пропорційно їх зменшувати в такій ситуації,
то мало би виглядати добре. Але на разі спроби з flex-shrink тут не працюють. Допрацювати потім.
*/
(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);
});
}
// Очікує на появу елемента
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 = 'ytd-popup-container #items ytd-menu-service-item-renderer:has(svg path[d="m13.18 4 .24 1.2.16.8H19v7h-5.18l-.24-1.2-.16-.8H6V4h7.18M14 3H5v18h1v-9h6.6l.4 2h7V5h-5.6L14 3z"])';
await waitForElementAsync(reportButtonSelector, `Didn't find ${reportButtonSelector}`);
document.querySelector(reportButtonSelector).click();
// радіо "Пропаганда тероризму"
const radioTerrorismSelector = '[id="radio:8"]';
await waitForElementAsync(radioTerrorismSelector, `Didn't find ${radioTerrorismSelector}`);
document.querySelector(radioTerrorismSelector).click();
// кнопка "Далі"
const nextButtonSelector = '#bottom-bar button';
await waitForElementAsync(nextButtonSelector, `Didn't find ${nextButtonSelector}`);
document.querySelector(nextButtonSelector).click();
// ввести причину звітування "russian propaganda"
const reportReasonInputSelector = 'textarea';
await waitForElementAsync(reportReasonInputSelector, `Didn't find ${reportReasonInputSelector}`);
const reportReasonInputElement = document.querySelector(reportReasonInputSelector);
inputText(reportReasonInputElement, 'russian propaganda');
// кнопка "Поскаржитися"
const submitButtonSelector = '#bottom-bar button';
await waitForElementAsync(submitButtonSelector, `Didn't find ${submitButtonSelector}`);
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).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 = 'ytd-popup-container #items ytd-menu-service-item-renderer:has(svg path[d="M12 3c-4.96 0-9 4.04-9 9s4.04 9 9 9 9-4.04 9-9-4.04-9-9-9m0-1c5.52 0 10 4.48 10 10s-4.48 10-10 10S2 17.52 2 12 6.48 2 12 2zm7 11H5v-2h14v2z"])';
await waitForElementAsync(notInterestedButtonSelector, `Didn't find ${notInterestedButtonSelector}`);
document.querySelector(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');
}
}
function fitItemsToMenuSize() {
return; // TODO 1.1
const menu = document.querySelector(youtubeShortsMenuSelector);
if (menu.areFittingStylesAssigned) {
return;
}
const menuItems = Array.from(document.querySelector('#experiment-overlay #actions').children);
/*const items = allItems.filter(el => {
const style = window.getComputedStyle(el);
return style.display !== 'none';
});*/
// const menuHeight = menu.clientHeight;
const menuItemsHeights = menuItems.map(item => item.clientHeight);
menu.style.display = 'flex';
menu.style.flexDirection = 'column';
menuItemsHeights.forEach((height, index, array) => {
menuItems[index].style.flex = '1 1 0';// `${height} ${height} ${height}`;
});
menu.areFittingStylesAssigned = true;
}
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);
// TODO 1.2
/*function createMenuItemFrom(element) {
const menuItem = document.createElement('div');
menuItem.className = 'button-container style-scope ytd-reel-player-overlay-renderer';
menuItem.appendChild(element);
return menuItem;
}
const videoButtonMenuItem = createMenuItemFrom(videoButtonWrapper);
const videoResultMenuItem = createMenuItemFrom(videoBlockingResult);
const channelButtonMenuItem = createMenuItemFrom(channelButtonWrapper);
menu.appendChild(videoButtonMenuItem);
menu.appendChild(videoResultMenuItem);
menu.appendChild(channelButtonMenuItem);*/
fitItemsToMenuSize();
}
// ################################
// Виконання коду
// ################################
// скинути стилі відео і кнопок при прокручуванні на інше відео 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();
}
});
})();