您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Check the Monetization Status.
// ==UserScript== // @name YouTube Enhancer (Monetization Checker) // @description Check the Monetization Status. // @icon  // @version 1.8 // @author exyezed // @namespace https://github.com/exyezed/youtube-enhancer/ // @supportURL https://github.com/exyezed/youtube-enhancer/issues // @license MIT // @match https://youtube.com/* // @match https://www.youtube.com/* // @grant GM_xmlhttpRequest // @connect monetizationcheck.vercel.app // @connect monetcheck.vercel.app // ==/UserScript== (function () { "use strict"; let lastCheckedUrl = ""; let lastCheckedChannelId = null; let isCheckingMonetization = false; let checkQueue = Promise.resolve(); const API_ENDPOINTS = [ "https://monetizationcheck.vercel.app", "https://monetcheck.vercel.app" ]; let currentApiIndex = 0; let statusCache = {}; function addGlobalStyles() { const styleElement = document.createElement('style'); styleElement.id = 'yt-monetization-global-styles'; styleElement.textContent = ` ytd-channel-name { display: flex !important; align-items: center !important; flex-wrap: wrap !important; } ytd-channel-name #container { display: flex !important; align-items: center !important; } ytd-channel-name ytd-badge-supported-renderer { display: flex !important; align-items: center !important; } .yt-video-monetization-icon { display: flex !important; align-items: center !important; justify-content: center !important; margin-left: 4px !important; height: 15px !important; position: relative !important; top: 0 !important; } .yt-video-monetization-icon svg { width: 15px !important; height: 15px !important; display: block !important; } .yt-monetization-icon { display: flex !important; align-items: center !important; justify-content: center !important; margin-left: 8px !important; height: 20px !important; position: relative !important; } .yt-monetization-icon svg { width: 20px !important; height: 20px !important; display: block !important; } .dynamic-text-view-model-wiz__h1 { display: flex !important; align-items: center !important; } .dynamic-text-view-model-wiz__h1 .yt-core-attributed-string { display: flex !important; align-items: center !important; } .dynamicTextViewModelH1 { display: flex !important; align-items: center !important; } .dynamicTextViewModelH1 .yt-core-attributed-string { display: flex !important; align-items: center !important; } `; document.head.appendChild(styleElement); } addGlobalStyles(); const commonIconStyles = { position: "relative", display: "flex", alignItems: "center", justifyContent: "center", marginLeft: "8px", height: "20px", cursor: "pointer" }; const videoIconStyles = { position: "relative", display: "flex", alignItems: "center", justifyContent: "center", marginLeft: "4px", height: "15px", cursor: "pointer" }; function createMonetizationIcon(isMonetized, isVideo = false) { const container = document.createElement("div"); Object.assign(container.style, isVideo ? videoIconStyles : commonIconStyles); const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); svg.setAttribute("viewBox", "0 0 512 512"); svg.setAttribute("fill", "none"); if (isVideo) { svg.setAttribute("width", "15"); svg.setAttribute("height", "15"); } else { svg.setAttribute("width", "20"); svg.setAttribute("height", "20"); } svg.style.display = "block"; const path = document.createElementNS("http://www.w3.org/2000/svg", "path"); path.setAttribute( "d", "M256 512A256 256 0 1 0 256 0a256 256 0 1 0 0 512zm20.8-378.4l0 14.2c9.7 1.2 19.4 3.9 29 6.6c1.9 .5 3.7 1 5.6 1.6c11.5 3.2 18.3 15.1 15.1 26.6s-15.1 18.2-26.6 15.1c-1.6-.4-3.1-.9-4.7-1.3c-7-2-14-3.9-21.1-5.3c-13.2-2.5-28.5-1.3-40.8 4c-11 4.8-20.1 16.4-7.6 24.4c9.8 6.3 21.8 9.5 33.2 12.6c2.4 .6 4.7 1.3 7 1.9c15.6 4.4 35.5 10.1 50.4 20.3c19.4 13.3 28.5 34.9 24.2 58.1c-4.1 22.4-19.7 37.1-38.4 44.7c-7.8 3.2-16.3 5.2-25.2 6.2l0 15.2c0 11.9-9.7 21.6-21.6 21.6s-21.6-9.7-21.6-21.6l0-17.4c-14.5-3.3-28.7-7.9-42.8-12.5c-11.3-3.7-17.5-16-13.7-27.3s16-17.5 27.3-13.7c2.5 .8 5 1.7 7.5 2.5c11.3 3.8 22.9 7.7 34.5 9.6c17 2.5 30.6 1 39.5-2.6c12-4.8 17.7-19.1 5.9-27.1c-10.1-6.9-22.6-10.3-34.5-13.5c-2.3-.6-4.5-1.2-6.8-1.9c-15.1-4.3-34-9.6-48.2-18.7c-19.5-12.5-29.4-33.3-25.2-56.4c4-21.8 21-36.3 39-44.1c5.5-2.4 11.4-4.3 17.5-5.7l0-16.1c0-11.9 9.7-21.6 21.6-21.6s21.6 9.7 21.6 21.6z" ); path.setAttribute("fill", isMonetized ? "#16a34a" : "#dc2626"); svg.appendChild(path); container.appendChild(svg); return container; } function createSpinnerIcon(isVideo = false) { const container = document.createElement("div"); Object.assign(container.style, isVideo ? videoIconStyles : commonIconStyles); const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); svg.setAttribute("viewBox", "0 0 512 512"); if (isVideo) { svg.setAttribute("width", "15"); svg.setAttribute("height", "15"); } else { svg.setAttribute("width", "20"); svg.setAttribute("height", "20"); } svg.style.display = "block"; svg.style.animation = "yt-monetization-spinner-rotate 1.5s linear infinite"; if (!document.querySelector("#yt-monetization-spinner-style")) { const style = document.createElement("style"); style.id = "yt-monetization-spinner-style"; style.textContent = ` @keyframes yt-monetization-spinner-rotate { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } `; document.head.appendChild(style); } const secondaryPath = document.createElementNS( "http://www.w3.org/2000/svg", "path" ); secondaryPath.setAttribute( "d", "M0 256C0 114.9 114.1 .5 255.1 0C237.9 .5 224 14.6 224 32c0 17.7 14.3 32 32 32C150 64 64 150 64 256s86 192 192 192c69.7 0 130.7-37.1 164.5-92.6c-3 6.6-3.3 14.8-1 22.2c1.2 3.7 3 7.2 5.4 10.3c1.2 1.5 2.6 3 4.1 4.3c.8 .7 1.6 1.3 2.4 1.9c.4 .3 .8 .6 1.3 .9s.9 .6 1.3 .8c5 2.9 10.6 4.3 16 4.3c11 0 21.8-5.7 27.7-16c-44.3 76.5-127 128-221.7 128C114.6 512 0 397.4 0 256z" ); secondaryPath.setAttribute("fill", "#ffffff"); secondaryPath.setAttribute("opacity", "0.4"); const primaryPath = document.createElementNS( "http://www.w3.org/2000/svg", "path" ); primaryPath.setAttribute( "d", "M224 32c0-17.7 14.3-32 32-32C397.4 0 512 114.6 512 256c0 46.6-12.5 90.4-34.3 128c-8.8 15.3-28.4 20.5-43.7 11.7s-20.5-28.4-11.7-43.7c16.3-28.2 25.7-61 25.7-96c0-106-86-192-192-192c-17.7 0-32-14.3-32-32z" ); primaryPath.setAttribute("fill", "#ffffff"); svg.appendChild(secondaryPath); svg.appendChild(primaryPath); container.appendChild(svg); return container; } function extractChannelIdFromVideoPage() { const channelLinks = document.querySelectorAll('a.ytd-channel-name, ytd-video-owner-renderer a'); for (const link of channelLinks) { const href = link.getAttribute('href'); if (href) { if (href.includes('/channel/')) { return href.split('/channel/')[1].split('/')[0]; } if (href.includes('/@')) { return '@' + href.split('/@')[1].split('/')[0]; } } } const ownerRenderer = document.querySelector('ytd-video-owner-renderer'); if (ownerRenderer) { const channelLink = ownerRenderer.querySelector('a'); if (channelLink) { const href = channelLink.getAttribute('href'); if (href) { if (href.includes('/channel/')) { return href.split('/channel/')[1].split('/')[0]; } if (href.includes('/@')) { return '@' + href.split('/@')[1].split('/')[0]; } } } } return null; } function extractChannelIdentifier() { const url = window.location.href; const urlObj = new URL(url); const pathname = urlObj.pathname; if (pathname === '/watch') { const channelId = extractChannelIdFromVideoPage(); if (channelId) { return channelId; } } if (pathname.includes('/channel/')) { return pathname.split('/channel/')[1].split('/')[0]; } if (pathname.includes('/@')) { return '@' + pathname.split('/@')[1].split('/')[0]; } return null; } function extractVideoId() { const url = window.location.href; const urlObj = new URL(url); const pathname = urlObj.pathname; const searchParams = urlObj.searchParams; if (pathname === '/watch') { return searchParams.get('v'); } if (pathname.startsWith('/shorts/')) { return pathname.split('/shorts/')[1].split('/')[0]; } return null; } function isVideoPage() { const url = window.location.href; return url.includes('/watch') || url.includes('/shorts/'); } function isShortsPage() { const url = window.location.href; return url.includes('/shorts/'); } function isChannelPage() { const channelIdentifier = extractChannelIdentifier(); return channelIdentifier !== null && !isVideoPage(); } function getNextApiEndpoint() { const endpoint = API_ENDPOINTS[currentApiIndex]; currentApiIndex = (currentApiIndex + 1) % API_ENDPOINTS.length; return endpoint; } function checkMonetizationStatus(type, id) { const cacheKey = `${type}_${id}`; if (statusCache[cacheKey] !== undefined) { return Promise.resolve(statusCache[cacheKey]); } return new Promise((resolve, reject) => { const tryWithApi = (apiEndpoint, retryCount = 0) => { const apiUrl = `${apiEndpoint}/${type}/${id}`; GM_xmlhttpRequest({ method: "GET", url: apiUrl, timeout: 10000, onload: function(response) { if (response.status === 200) { try { const result = JSON.parse(response.responseText); const isMonetized = result.monetization === true; statusCache[cacheKey] = isMonetized; resolve(isMonetized); } catch (error) { // Try next API endpoint if JSON parsing fails if (retryCount < API_ENDPOINTS.length - 1) { const nextApiEndpoint = getNextApiEndpoint(); tryWithApi(nextApiEndpoint, retryCount + 1); } else { reject(new Error("Invalid API response from all endpoints")); } } } else if (response.status === 429 || response.status >= 500) { // Rate limited or server error, try next API endpoint if (retryCount < API_ENDPOINTS.length - 1) { const nextApiEndpoint = getNextApiEndpoint(); tryWithApi(nextApiEndpoint, retryCount + 1); } else { reject(new Error(`API request failed on all endpoints: ${response.status}`)); } } else { // Other errors, try next API endpoint if (retryCount < API_ENDPOINTS.length - 1) { const nextApiEndpoint = getNextApiEndpoint(); tryWithApi(nextApiEndpoint, retryCount + 1); } else { reject(new Error(`API request failed: ${response.status}`)); } } }, onerror: function(error) { // Network error, try next API endpoint if (retryCount < API_ENDPOINTS.length - 1) { const nextApiEndpoint = getNextApiEndpoint(); tryWithApi(nextApiEndpoint, retryCount + 1); } else { reject(error); } }, ontimeout: function() { // Timeout, try next API endpoint if (retryCount < API_ENDPOINTS.length - 1) { const nextApiEndpoint = getNextApiEndpoint(); tryWithApi(nextApiEndpoint, retryCount + 1); } else { reject(new Error("Request timed out on all endpoints")); } } }); }; // Start with the next API endpoint const firstApiEndpoint = getNextApiEndpoint(); tryWithApi(firstApiEndpoint, 0); }); } function queueMonetizationCheck(forceCheck = false) { if (isCheckingMonetization && !forceCheck) { return; } checkQueue = checkQueue.then(() => { return performMonetizationCheck(forceCheck); }).catch((error) => { console.error("Error checking monetization:", error); isCheckingMonetization = false; }); } async function performMonetizationCheck(forceCheck = false) { const currentUrl = window.location.href; const currentChannelId = extractChannelIdentifier(); if (currentUrl !== lastCheckedUrl || currentChannelId !== lastCheckedChannelId || forceCheck) { lastCheckedUrl = currentUrl; lastCheckedChannelId = currentChannelId; isCheckingMonetization = false; } if (isCheckingMonetization) { return; } isCheckingMonetization = true; try { if (isVideoPage()) { const videoId = extractVideoId(); if (!videoId) { throw new Error("Could not determine video ID"); } await showVideoSpinner(); const endpoint = isShortsPage() ? "shorts" : "video"; const isMonetized = await checkMonetizationStatus(endpoint, videoId); await updateVideoPage(isMonetized); } else if (isChannelPage()) { const channelId = extractChannelIdentifier(); if (!channelId) { throw new Error("Could not determine channel ID"); } await showChannelSpinner(); const isMonetized = await checkMonetizationStatus("channel", channelId); await updateChannelPage(isMonetized); } } catch (error) { console.error("Monetization check failed:", error); if (isVideoPage()) { await updateVideoPage(false, true); } else if (isChannelPage()) { await updateChannelPage(false, true); } } finally { isCheckingMonetization = false; } } function findChannelTitleElement(retryCount = 0, maxRetries = 5) { const possibleTitleSelectors = [ "h1.dynamicTextViewModelH1 > span.yt-core-attributed-string", "h1.dynamic-text-view-model-wiz__h1 .yt-core-attributed-string", "h1 .yt-core-attributed-string", "ytd-channel-name #channel-name", "#channel-header-container #channel-name", ]; for (const selector of possibleTitleSelectors) { const element = document.querySelector(selector); if (element) return element; } if (retryCount < maxRetries) { return new Promise((resolve) => { setTimeout(() => { resolve(findChannelTitleElement(retryCount + 1, maxRetries)); }, 500); }); } return null; } function findVideoChannelElement(retryCount = 0, maxRetries = 5) { const element = document.querySelector("ytd-video-owner-renderer #channel-name"); if (element) return element; if (retryCount < maxRetries) { return new Promise((resolve) => { setTimeout(() => { resolve(findVideoChannelElement(retryCount + 1, maxRetries)); }, 500); }); } return null; } async function showChannelSpinner() { const titleElement = await findChannelTitleElement(); if (!titleElement) { return null; } const existingIcon = document.querySelector(".yt-monetization-icon"); if (existingIcon) { existingIcon.remove(); } const spinner = createSpinnerIcon(false); spinner.classList.add("yt-monetization-icon"); const channelNameContainer = titleElement.closest('h1') || titleElement.parentNode; if (channelNameContainer) { channelNameContainer.appendChild(spinner); } else { titleElement.appendChild(spinner); } return spinner; } async function showVideoSpinner() { const channelElement = await findVideoChannelElement(); if (!channelElement) { return null; } const existingIcon = document.querySelector(".yt-video-monetization-icon"); if (existingIcon) { existingIcon.remove(); } const spinner = createSpinnerIcon(true); spinner.classList.add("yt-video-monetization-icon"); const badgeContainer = channelElement.querySelector("ytd-badge-supported-renderer"); if (badgeContainer) { badgeContainer.parentNode.insertBefore(spinner, badgeContainer.nextSibling); } else { channelElement.appendChild(spinner); } return spinner; } async function updateChannelPage(isMonetized, isError = false) { const titleElement = await findChannelTitleElement(); if (!titleElement) { return; } const existingIcon = document.querySelector(".yt-monetization-icon"); if (existingIcon) { existingIcon.remove(); } const monetizationIcon = createMonetizationIcon(isMonetized, false); monetizationIcon.classList.add("yt-monetization-icon"); if (isError) { monetizationIcon.title = "Failed to check monetization status"; } else { monetizationIcon.title = isMonetized ? "Channel is monetized" : "Channel is not monetized"; } const channelNameContainer = titleElement.closest('h1') || titleElement.parentNode; if (channelNameContainer) { channelNameContainer.appendChild(monetizationIcon); } else { titleElement.appendChild(monetizationIcon); } monetizationIcon.addEventListener('click', () => { queueMonetizationCheck(true); }); } async function updateVideoPage(isMonetized, isError = false) { const channelElement = await findVideoChannelElement(); if (!channelElement) { return; } const existingIcon = document.querySelector(".yt-video-monetization-icon"); if (existingIcon) { existingIcon.remove(); } const monetizationIcon = createMonetizationIcon(isMonetized, true); monetizationIcon.classList.add("yt-video-monetization-icon"); if (isError) { monetizationIcon.title = "Failed to check monetization status"; } else { monetizationIcon.title = isMonetized ? "Channel is monetized" : "Channel is not monetized"; } const badgeContainer = channelElement.querySelector("ytd-badge-supported-renderer"); if (badgeContainer) { badgeContainer.parentNode.insertBefore(monetizationIcon, badgeContainer.nextSibling); } else { channelElement.appendChild(monetizationIcon); } monetizationIcon.addEventListener('click', () => { queueMonetizationCheck(true); }); } function setupUrlChangeDetection() { let lastPath = window.location.pathname; let lastSearch = window.location.search; let lastVideoId = extractVideoId(); let lastChannelId = extractChannelIdentifier(); let lastHref = window.location.href; setInterval(() => { statusCache = {}; }, 30 * 60 * 1000); function checkForChanges() { const currentPath = window.location.pathname; const currentSearch = window.location.search; const currentVideoId = extractVideoId(); const currentChannelId = extractChannelIdentifier(); const currentHref = window.location.href; if (currentPath !== lastPath || currentSearch !== lastSearch || currentVideoId !== lastVideoId || currentChannelId !== lastChannelId || currentHref !== lastHref) { lastPath = currentPath; lastSearch = currentSearch; lastVideoId = currentVideoId; lastChannelId = currentChannelId; lastHref = currentHref; setTimeout(() => queueMonetizationCheck(true), 500); } } const observer = new MutationObserver((mutations) => { for (const mutation of mutations) { if (mutation.type === 'childList' && mutation.addedNodes.length > 0) { for (const node of mutation.addedNodes) { if (node.nodeType === Node.ELEMENT_NODE) { if (node.id === 'content' || node.id === 'page-manager' || node.tagName === 'YTD-WATCH-FLEXY' || node.tagName === 'YTD-BROWSE' || node.tagName === 'YTD-REEL-WATCH-RENDERER') { checkForChanges(); break; } } } } } }); observer.observe(document.body, { childList: true, subtree: true, }); const originalPushState = history.pushState; const originalReplaceState = history.replaceState; history.pushState = function () { originalPushState.apply(this, arguments); setTimeout(checkForChanges, 100); }; history.replaceState = function () { originalReplaceState.apply(this, arguments); setTimeout(checkForChanges, 100); }; window.addEventListener("popstate", () => { setTimeout(checkForChanges, 100); }); setInterval(checkForChanges, 1000); document.addEventListener('click', (event) => { let target = event.target; while (target && target !== document) { if (target.tagName === 'A' || target.id === 'thumbnail' || target.classList.contains('yt-simple-endpoint')) { setTimeout(checkForChanges, 500); break; } target = target.parentNode; } }, true); } queueMonetizationCheck(true); setupUrlChangeDetection(); window.checkYoutubeMonetization = () => queueMonetizationCheck(true); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址