您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Display Real-Time Subscriber Count.
// ==UserScript== // @name YouTube Enhancer (Real-Time Subscriber Count) // @description Display Real-Time Subscriber Count. // @icon  // @version 1.5 // @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 // ==/UserScript== (function () { "use strict"; const FONT_LINK = "https://fonts.googleapis.com/css2?family=Rubik:wght@400;700&display=swap"; const STATS_API_URL = "https://api.livecounts.io/youtube-live-subscriber-counter/stats/"; const UPDATE_INTERVAL = 2000; let overlay = null; let isUpdating = false; let intervalId = null; let currentChannelName = null; const lastSuccessfulStats = new Map(); const previousStats = new Map(); let previousUrl = location.href; let isChecking = false; async function fetchChannel(url) { if (isChecking) return null; isChecking = true; try { const response = await fetch(url, { credentials: "same-origin", }); if (!response.ok) return null; const html = await response.text(); const match = html.match(/var ytInitialData = (.+?);<\/script>/); return match && match[1] ? JSON.parse(match[1]) : null; } catch (error) { return null; } finally { isChecking = false; } } async function getChannelInfo(url) { const data = await fetchChannel(url); if (!data) return null; try { const channelName = data?.metadata?.channelMetadataRenderer?.title || "Unknown"; const channelId = data?.metadata?.channelMetadataRenderer?.externalId || null; return { channelName, channelId }; } catch (e) { return null; } } function isChannelPageUrl(url) { return ( url.includes("youtube.com/") && (url.includes("/channel/") || url.includes("/@")) && !url.includes("/video/") && !url.includes("/watch") ); } function checkUrlChange() { const currentUrl = location.href; if (currentUrl !== previousUrl) { previousUrl = currentUrl; if (isChannelPageUrl(currentUrl)) { setTimeout(() => getChannelInfo(currentUrl), 500); } } } history.pushState = (function (f) { return function () { f.apply(this, arguments); checkUrlChange(); }; })(history.pushState); history.replaceState = (function (f) { return function () { f.apply(this, arguments); checkUrlChange(); }; })(history.replaceState); window.addEventListener("popstate", checkUrlChange); setInterval(checkUrlChange, 1000); function init() { loadFonts(); addStyles(); observePageChanges(); addNavigationListener(); if (isChannelPageUrl(location.href)) { getChannelInfo(location.href); } } function loadFonts() { const fontLink = document.createElement("link"); fontLink.rel = "stylesheet"; fontLink.href = FONT_LINK; document.head.appendChild(fontLink); } function addStyles() { const style = document.createElement("style"); style.textContent = `@keyframes spin { from { transform: rotate(0deg); } to { transform: rotate(360deg); } }`; document.head.appendChild(style); } function createSpinner() { const spinnerWrapper = document.createElement("div"); Object.assign(spinnerWrapper.style, { position: "absolute", top: "0", left: "0", width: "100%", height: "100%", display: "flex", justifyContent: "center", alignItems: "center", zIndex: "10", }); spinnerWrapper.classList.add("spinner-container"); const spinner = document.createElementNS( "http://www.w3.org/2000/svg", "svg" ); spinner.setAttribute("viewBox", "0 0 512 512"); spinner.setAttribute("width", "64"); spinner.setAttribute("height", "64"); spinner.classList.add("loading-spinner"); spinner.style.animation = "spin 1s linear infinite"; const currentColor = getThemeColor(); 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.style.opacity = "0.4"; secondaryPath.style.fill = currentColor; 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.style.fill = currentColor; spinner.appendChild(secondaryPath); spinner.appendChild(primaryPath); spinnerWrapper.appendChild(spinner); return spinnerWrapper; } function createSVGIcon(path) { const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); svg.setAttribute("viewBox", "0 0 640 512"); svg.setAttribute("width", "2rem"); svg.setAttribute("height", "2rem"); svg.style.marginRight = "0.5rem"; svg.style.display = "none"; const svgPath = document.createElementNS( "http://www.w3.org/2000/svg", "path" ); svgPath.setAttribute("d", path); svgPath.setAttribute("fill", getThemeColor()); svg.appendChild(svgPath); return svg; } function createStatContainer(className, iconPath) { const container = document.createElement("div"); Object.assign(container.style, { display: "flex", flexDirection: "column", alignItems: "center", justifyContent: "center", visibility: "hidden", width: "33%", height: "100%", padding: "0 1rem", }); const numberContainer = document.createElement("div"); Object.assign(numberContainer.style, { display: "flex", alignItems: "center", justifyContent: "center", position: "relative", height: "4rem", }); const digitContainer = createNumberContainer(); digitContainer.classList.add(`${className}-number`); Object.assign(digitContainer.style, { fontSize: "4rem", fontWeight: "bold", lineHeight: "1", height: "4rem", fontFamily: "inherit", letterSpacing: "0.025em", }); const differenceElement = document.createElement("div"); differenceElement.classList.add(`${className}-difference`); Object.assign(differenceElement.style, { fontSize: "2.2rem", position: "absolute", right: "-60px", top: "50%", transform: "translateY(-50%)", whiteSpace: "nowrap", opacity: "0.8", }); numberContainer.appendChild(digitContainer); numberContainer.appendChild(differenceElement); const labelContainer = document.createElement("div"); Object.assign(labelContainer.style, { display: "flex", alignItems: "center", marginTop: "0.5rem", }); const icon = createSVGIcon(iconPath); Object.assign(icon.style, { width: "2rem", height: "2rem", marginRight: "0.75rem", }); const labelElement = document.createElement("div"); labelElement.classList.add(`${className}-label`); labelElement.style.fontSize = "2rem"; labelContainer.appendChild(icon); labelContainer.appendChild(labelElement); container.appendChild(numberContainer); container.appendChild(labelContainer); return container; } function getThemeColor() { const isDarkTheme = document.documentElement.getAttribute("dark") !== null || document.documentElement.getAttribute("data-dark") !== null || document.body.getAttribute("dark") !== null || document.querySelector("html[dark]") !== null || document.querySelector('[dark="true"]') !== null; return isDarkTheme ? "white" : "black"; } function updateThemeColors() { if (!overlay) return; const currentColor = getThemeColor(); overlay.style.color = currentColor; overlay .querySelectorAll(".loading-spinner path") .forEach((path) => (path.style.fill = currentColor)); overlay.querySelectorAll("svg path").forEach((path) => { if (!path.closest(".loading-spinner")) path.setAttribute("fill", currentColor); }); } function createOverlay(containerElement) { clearExistingOverlay(); if (!containerElement) return null; const overlay = document.createElement("div"); overlay.classList.add("channel-banner-overlay"); Object.assign(overlay.style, { position: "relative", width: "100%", maxWidth: "1280px", height: "120px", backgroundColor: "transparent", borderRadius: "15px", zIndex: "1", color: getThemeColor(), fontFamily: "Rubik, sans-serif", marginTop: "16px", marginLeft: "auto", marginRight: "auto", }); const contentContainer = document.createElement("div"); Object.assign(contentContainer.style, { display: "flex", justifyContent: "space-around", alignItems: "center", width: "100%", height: "100%", }); contentContainer.classList.add("stats-content"); const subscribersElement = createStatContainer( "subscribers", "M144 160c-44.2 0-80-35.8-80-80S99.8 0 144 0s80 35.8 80 80s-35.8 80-80 80zm368 0c-44.2 0-80-35.8-80-80s35.8-80 80-80s80 35.8 80 80s-35.8 80-80 80zM0 298.7C0 239.8 47.8 192 106.7 192h42.7c15.9 0 31 3.5 44.6 9.7c-1.3 7.2-1.9 14.7-1.9 22.3c0 38.2 16.8 72.5 43.3 96c-.2 0-.4 0-.7 0H21.3C9.6 320 0 310.4 0 298.7zM405.3 320c-.2 0-.4 0-.7 0c26.6-23.5 43.3-57.8 43.3-96c0-7.6-.7-15-1.9-22.3c13.6-6.3 28.7-9.7 44.6-9.7h42.7C592.2 192 640 239.8 640 298.7c0 11.8-9.6 21.3-21.3 21.3H405.3zM416 224c0 53-43 96-96 96s-96-43-96-96s43-96 96-96s96 43 96 96zM128 485.3C128 411.7 187.7 352 261.3 352H378.7C452.3 352 512 411.7 512 485.3c0 14.7-11.9 26.7-26.7 26.7H154.7c-14.7 0-26.7-11.9-26.7-26.7z" ); const viewsElement = createStatContainer( "views", "M288 32c-80.8 0-145.5 36.8-192.6 80.6C48.6 156 17.3 208 2.5 243.7c-3.3 7.9-3.3 16.7 0 24.6C17.3 304 48.6 356 95.4 399.4C142.5 443.2 207.2 480 288 480s145.5-36.8 192.6-80.6c46.8-43.5 78.1-95.4 93-131.1c3.3-7.9 3.3-16.7 0-24.6c-14.9-35.7-46.2-87.7-93-131.1C433.5 68.8 368.8 32 288 32zM144 256a144 144 0 1 1 288 0 144 144 0 1 1 -288 0zm144-64c0 35.3-28.7 64-64 64c-7.1 0-13.9-1.2-20.3-3.3c-5.5-1.8-11.9 1.6-11.7 7.4c.3 6.9 1.3 13.8 3.2 20.7c13.7 51.2 66.4 81.6 117.6 67.9s81.6-66.4 67.9-117.6c-11.1-41.5-47.8-69.4-88.6-71.1c-5.8-.2-9.2 6.1-7.4 11.7c2.1 6.4 3.3 13.2 3.3 20.3z" ); const videosElement = createStatContainer( "videos", "M0 128C0 92.7 28.7 64 64 64H320c35.3 0 64 28.7 64 64V384c0 35.3-28.7 64-64 64H64c-35.3 0-64-28.7-64-64V128zM559.1 99.8c10.4 5.6 16.9 16.4 16.9 28.2V384c0 11.8-6.5 22.6-16.9 28.2s-23 5-32.9-1.6l-96-64L416 337.1V320 192 174.9l14.2-9.5 96-64c9.8-6.5 22.4-7.2 32.9-1.6z" ); contentContainer.append(subscribersElement, viewsElement, videosElement); overlay.append(contentContainer, createSpinner()); containerElement.insertAdjacentElement("afterend", overlay); updateDisplayState(); return overlay; } function fetchWithGM(url, headers = {}) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: "GET", url: url, headers: headers, onload: function (response) { if (response.status === 200) { resolve(JSON.parse(response.responseText)); } else { reject(new Error(`Failed to fetch: ${response.status}`)); } }, onerror: function (error) { reject(error); }, }); }); } async function fetchChannelId(_channelName) { try { const channelInfo = await getChannelInfo(window.location.href); if (channelInfo && channelInfo.channelId) { return channelInfo.channelId; } const metaTag = document.querySelector('meta[itemprop="channelId"]'); if (metaTag && metaTag.content) { return metaTag.content; } const urlMatch = window.location.href.match(/channel\/(UC[\w-]+)/); if (urlMatch && urlMatch[1]) { return urlMatch[1]; } throw new Error("Could not determine channel ID"); } catch (error) { const metaTag = document.querySelector('meta[itemprop="channelId"]'); if (metaTag && metaTag.content) { return metaTag.content; } const urlMatch = window.location.href.match(/channel\/(UC[\w-]+)/); if (urlMatch && urlMatch[1]) { return urlMatch[1]; } throw new Error("Could not determine channel ID"); } } async function fetchChannelStats(channelId) { let retries = 3; while (retries > 0) { try { const stats = await fetchWithGM(`${STATS_API_URL}${channelId}`, { origin: "https://livecounts.io", referer: "https://livecounts.io/", }); if (stats && typeof stats.followerCount !== "undefined") { lastSuccessfulStats.set(channelId, stats); return stats; } } catch (e) { retries--; if (retries > 0) await new Promise((resolve) => setTimeout(resolve, 1000)); } } if (lastSuccessfulStats.has(channelId)) return lastSuccessfulStats.get(channelId); const fallbackStats = { followerCount: 0, bottomOdos: [0, 0], error: true }; const subCountElem = document.querySelector("#subscriber-count"); if (subCountElem) { const subMatch = subCountElem.textContent.match(/[\d,]+/); if (subMatch) fallbackStats.followerCount = parseInt(subMatch[0].replace(/,/g, "")); } return fallbackStats; } function clearExistingOverlay() { const existingOverlay = document.querySelector(".channel-banner-overlay"); if (existingOverlay) { existingOverlay.remove(); } if (intervalId) { clearInterval(intervalId); intervalId = null; } lastSuccessfulStats.clear(); previousStats.clear(); isUpdating = false; overlay = null; } function createDigitElement() { const digit = document.createElement("span"); Object.assign(digit.style, { display: "inline-block", width: "0.6em", textAlign: "center", marginRight: "0.025em", marginLeft: "0.025em", }); return digit; } function createCommaElement() { const comma = document.createElement("span"); comma.textContent = ","; Object.assign(comma.style, { display: "inline-block", width: "0.3em", textAlign: "center", }); return comma; } function createNumberContainer() { const container = document.createElement("div"); Object.assign(container.style, { display: "flex", justifyContent: "center", alignItems: "center", letterSpacing: "0.025em", }); return container; } function updateDigits(container, newValue) { const newValueStr = newValue.toString(); const digits = []; for (let i = newValueStr.length - 1; i >= 0; i -= 3) { const start = Math.max(0, i - 2); digits.unshift(newValueStr.slice(start, i + 1)); } while (container.firstChild) { container.removeChild(container.firstChild); } let digitIndex = 0; for (let i = 0; i < digits.length; i++) { const group = digits[i]; for (let j = 0; j < group.length; j++) { const digitElement = createDigitElement(); digitElement.textContent = group[j]; container.appendChild(digitElement); digitIndex++; } if (i < digits.length - 1) { container.appendChild(createCommaElement()); } } let elementIndex = 0; for (let i = 0; i < digits.length; i++) { const group = digits[i]; for (let j = 0; j < group.length; j++) { const digitElement = container.children[elementIndex]; const newDigit = parseInt(group[j]); const currentDigit = parseInt(digitElement.textContent || "0"); if (currentDigit !== newDigit) { animateDigit(digitElement, currentDigit, newDigit); } elementIndex++; } if (i < digits.length - 1) { elementIndex++; } } } function animateDigit(element, start, end) { const duration = 1000; const startTime = performance.now(); function update(currentTime) { const elapsed = currentTime - startTime; const progress = Math.min(elapsed / duration, 1); const easeOutQuart = 1 - Math.pow(1 - progress, 4); const current = Math.round(start + (end - start) * easeOutQuart); element.textContent = current; if (progress < 1) { requestAnimationFrame(update); } } requestAnimationFrame(update); } function showContent(overlay) { const spinnerContainer = overlay.querySelector(".spinner-container"); if (spinnerContainer) { spinnerContainer.remove(); } const containers = overlay.querySelectorAll( 'div[style*="visibility: hidden"]' ); containers.forEach((container) => { container.style.visibility = "visible"; }); const icons = overlay.querySelectorAll('svg[style*="display: none"]'); icons.forEach((icon) => { icon.style.display = "block"; }); } function updateDifferenceElement(element, currentValue, previousValue) { if (!previousValue) return; const difference = currentValue - previousValue; if (difference === 0) { element.textContent = ""; return; } const sign = difference > 0 ? "+" : ""; element.textContent = `${sign}${difference.toLocaleString()}`; element.style.color = difference > 0 ? "#1ed760" : "#f3727f"; setTimeout(() => { element.textContent = ""; }, 1000); } function updateDisplayState() { const overlay = document.querySelector(".channel-banner-overlay"); const contentContainer = overlay?.querySelector(".stats-content"); const statContainers = contentContainer?.querySelectorAll( 'div[style*="width"]' ); if (!statContainers?.length) return; statContainers.forEach((container) => Object.assign(container.style, { display: "flex", width: "33.33%" }) ); contentContainer.style.display = "flex"; } async function updateOverlayContent(overlay, channelName) { if (isUpdating || channelName !== currentChannelName) return; isUpdating = true; try { const channelId = await fetchChannelId(channelName); const stats = await fetchChannelStats(channelId); if (channelName !== currentChannelName) { isUpdating = false; return; } if (stats.error) { const containers = overlay.querySelectorAll('[class$="-number"]'); containers.forEach((container) => { if ( container.classList.contains("subscribers-number") && stats.followerCount > 0 ) { updateDigits(container, stats.followerCount); } else { container.textContent = "---"; } }); return; } const updateElement = (className, value, label) => { const numberContainer = overlay.querySelector(`.${className}-number`); const differenceElement = overlay.querySelector( `.${className}-difference` ); const labelElement = overlay.querySelector(`.${className}-label`); if (numberContainer) { updateDigits(numberContainer, value); } if (differenceElement && previousStats.has(channelId)) { const previousValue = className === "subscribers" ? previousStats.get(channelId).followerCount : previousStats.get(channelId).bottomOdos[ className === "views" ? 0 : 1 ]; updateDifferenceElement(differenceElement, value, previousValue); } if (labelElement) { labelElement.textContent = label; } }; updateElement("subscribers", stats.followerCount, "Subscribers"); updateElement("views", stats.bottomOdos[0], "Views"); updateElement("videos", stats.bottomOdos[1], "Videos"); if (!previousStats.has(channelId)) { showContent(overlay); } previousStats.set(channelId, stats); } catch (error) { const containers = overlay.querySelectorAll('[class$="-number"]'); containers.forEach((container) => { container.textContent = "---"; }); } finally { isUpdating = false; } } function addOverlay(containerElement) { const channelName = window.location.pathname.split("/")[1].replace("@", ""); if (channelName === currentChannelName && overlay) { return; } currentChannelName = channelName; overlay = createOverlay(containerElement); if (overlay) { if (intervalId) { clearInterval(intervalId); } intervalId = setInterval( () => updateOverlayContent(overlay, channelName), UPDATE_INTERVAL ); updateOverlayContent(overlay, channelName); } } function isChannelPage() { return ( window.location.pathname.startsWith("/@") || window.location.pathname.startsWith("/channel/") || window.location.pathname.startsWith("/c/") ); } function observePageChanges() { const observer = new MutationObserver((mutations) => { for (const mutation of mutations) { if (mutation.type === "childList") { const containerElement = document.getElementById( "page-header-container" ); if (containerElement && isChannelPage()) { addOverlay(containerElement); break; } } if ( mutation.type === "attributes" && (mutation.attributeName === "dark" || mutation.attributeName === "data-dark") && overlay ) { updateThemeColors(); } } }); observer.observe(document.body, { childList: true, subtree: true, }); new MutationObserver((mutations) => { mutations.forEach((mutation) => { if ( mutation.type === "attributes" && (mutation.attributeName === "dark" || mutation.attributeName === "data-dark") && overlay ) { updateThemeColors(); } }); }).observe(document.documentElement, { attributes: true, attributeFilter: ["dark", "data-dark"], }); } function addNavigationListener() { window.addEventListener("yt-navigate-finish", () => { if (!isChannelPage()) { clearExistingOverlay(); currentChannelName = null; } else { const containerElement = document.getElementById( "page-header-container" ); if (containerElement) { addOverlay(containerElement); } } }); } init(); })();