// ==UserScript== // @name Auto-Select YouTube Subtitles by Sapioit // @namespace Sapioit // @copyright Sapioit, 2020 - Present // @author sapioitgmail.com // @license GPL-2.0-only; http://www.gnu.org/licenses/gpl-2.0.txt // @version 2.13.1.6 // @description Automatically selects the YouTube subtitles with the text "English (auto-generated)" // @match https://*.youtube.com/* // @match https://youtube.com/* // @match https://youtu.be/* // @icon https://youtube.com/favicon.ico // @grant none // @downloadURL none // ==/UserScript== let repeatInterval = 5000; // 1000 milliseconds = 1 second; 10000 milliseconds = 10 second; let waitInterval = 5; // Adjust the delay (in milliseconds) if needed. 1000 = 1 second. function title_changed (count = 3) { console.log("Auto-Select YouTube Subtitles by Sapioit: Checking title."); // Check if the value of the title (the first "yt-formatted-string.style-scope.ytd-watch-metadata" element) has changed const firstWatchMetadataElement = ( document.querySelectorAll('yt-formatted-string.style-scope.ytd-watch-metadata') )[0]; if (firstWatchMetadataElement) { // Get the current value of the title const currentWatchMetadataValue = firstWatchMetadataElement.textContent.trim(); // Get the consecutive unchanged count from localStorage, or set it to 0 if it's not available let consecutiveUnchangedCount = parseInt(localStorage.getItem('consecutiveUnchangedCount')) || 0; console.log("Auto-Select YouTube Subtitles by Sapioit: Try nr #" + (1+consecutiveUnchangedCount) ); // Check if the current value is the same as the previous one stored in localStorage if (localStorage.getItem('currentWatchMetadataValue') === currentWatchMetadataValue) { // Increment the consecutiveUnchangedCount if the value is the same consecutiveUnchangedCount++; // Update the stored value and consecutiveUnchangedCount in localStorage localStorage.setItem('currentWatchMetadataValue', currentWatchMetadataValue); localStorage.setItem('consecutiveUnchangedCount', consecutiveUnchangedCount.toString()); // Stop the function if the title value hasn't changed for the last three checks (consecutiveUnchangedCount is less than 4) if ( consecutiveUnchangedCount < (count+1) ) { console.log("Auto-Select YouTube Subtitles by Sapioit: First 'yt-formatted-string.style-scope.ytd-watch-metadata' element has not changed for the last three checks. Stopping check."); return false; // Return that the title has not changed. } else { // Reset the consecutiveUnchangedCount if the value has changed consecutiveUnchangedCount = 0; // Update the stored value and consecutiveUnchangedCount in localStorage localStorage.setItem('currentWatchMetadataValue', currentWatchMetadataValue); localStorage.setItem('consecutiveUnchangedCount', consecutiveUnchangedCount.toString()); return true; // Return that the title has changed. } } else { // Reset the consecutiveUnchangedCount if the value has changed consecutiveUnchangedCount = 0; // Update the stored value and consecutiveUnchangedCount in localStorage localStorage.setItem('currentWatchMetadataValue', currentWatchMetadataValue); localStorage.setItem('consecutiveUnchangedCount', consecutiveUnchangedCount.toString()); return true; // Return that the title has changed. } } else { console.log("Auto-Select YouTube Subtitles by Sapioit: Checking title FAILED."); } return true; // Return that the title has changed. // Usage example: if ( title_changed() ){ return; } if ( title_changed(3) ){ return; } } function checkSubtitles() { if ( title_changed() ){ return; } console.log("Auto-Select YouTube Subtitles by Sapioit: Checking subtitles."); // Check if the value of ytp-menuitem-content is 'English (auto-generated)' const menuItems = document.querySelectorAll('.ytp-menuitem'); const subtitlesButton = document.querySelector('.ytp-subtitles-button'); let autoGeneratedMenuItem; for (const menuItem of menuItems) { const menuLabel = menuItem.querySelector('.ytp-menuitem-label span:first-child'); const menuContent = menuItem.querySelector('.ytp-menuitem-content'); if ( menuLabel && menuLabel.textContent.trim() === 'Subtitles/CC' && menuContent && menuContent.textContent.trim() !== 'English (auto-generated)' ){ autoGeneratedMenuItem = menuItem; break; } } // Run the code if the condition is not met if (subtitlesButton && subtitlesButton.getAttribute('title') !== 'Subtitles/closed captions unavailable') { if (autoGeneratedMenuItem || subtitlesButton.getAttribute('aria-pressed') !== 'true' ) { // Change the subtitles //alert(subtitlesButton.outerHTML); ChangeSubtitles(); } else { console.log("Auto-Select YouTube Subtitles by Sapioit: Subtitles did not need to be updated."); } } else { console.log("Auto-Select YouTube Subtitles by Sapioit: There are no subtitles available."); } } // Schedule to run the code every so often setInterval(checkSubtitles, repeatInterval); // 1000 milliseconds = 1 second; 10000 milliseconds = 10 second; window.addEventListener('load', ChangeSubtitles() ); function ChangeSubtitles() { 'use strict'; // Delay function to add a short delay between clicks function delay(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } // Find the gear icon //const gearIcon = document.querySelector('.ytp-settings-button'); const gearIcon = document.querySelector('.ytp-settings-button'); // Find the subtitles icon const subtitlesButton = document.querySelector('.ytp-subtitles-button'); // Click the gear icon to open the menu (gearIcon).click(); //document.querySelector('.ytp-settings-button').click(); // Wait for the menu to open setTimeout(() => { // Find the third menu item const menuItems = document.querySelectorAll('.ytp-menuitem-label span'); let thirdMenuItem; // Loop through each element in the 'menuItems' array. for (let j = 0; j < menuItems.length; j++) { // Get the current menu item element and its closest ancestor with the class 'ytp-menuitem'. const menuItem = menuItems[j].closest('.ytp-menuitem'); // If 'menuItem' exists and has a child with the class 'ytp-menuitem-label' that has the first child a 'span', and its text content is 'Subtitles/CC'. if (menuItem && menuItem.querySelector('.ytp-menuitem-label span:first-child').textContent.trim() === 'Subtitles/CC') { // If the condition is true, assign the 'menuItem' element to the variable 'thirdMenuItem'. thirdMenuItem = menuItem; // Exit the loop immediately since the desired menu item is found. break; } } // Click the third menu item if (thirdMenuItem) { (thirdMenuItem).click(); } // Wait for the "Subtitles/CC" menu to open setTimeout(() => { // Find the "English (auto-generated)" menu item const menuLabels = document.querySelectorAll('.ytp-menuitem-label'); let autoGeneratedMenuItem; for (let i = 0; i < menuLabels.length; i++) { if (menuLabels[i].textContent.trim() === 'English (auto-generated)') { autoGeneratedMenuItem = menuLabels[i]; break; } } // Click the "English (auto-generated)" menu item if found if (autoGeneratedMenuItem) { (autoGeneratedMenuItem).click(); } // Wait for the "English (auto-generated)" option to be selected setTimeout(() => { const element = document.querySelector('.ytp-settings-menu'); const computedStyle = getComputedStyle(element); if (computedStyle.display !== 'none') { // Click the gear icon to open the menu (gearIcon).click(); } setTimeout(() => { if (subtitlesButton.getAttribute('aria-pressed') !== 'true') { //alert(subtitlesButton.outerHTML); (subtitlesButton).click(); } }, waitInterval); // Adjust the delay (in milliseconds) if needed }, waitInterval); // Adjust the delay (in milliseconds) if needed }, waitInterval); // Adjust the delay (in milliseconds) if needed }, waitInterval); // Adjust the delay (in milliseconds) if needed console.log("Auto-Select YouTube Subtitles by Sapioit: Subtitles changed."); }