// ==UserScript==
// @name YouTube Enhancer (Real-Time Subscriber Count)
// @description Display Real-Time Subscriber Count.
// @icon data:image/svg+xml;base64,<svg xmlns="http://www.w3.org/2000/svg"  viewBox="0 0 100 100" width="100px" height="100px" baseProfile="basic"><path fill="#de333b" d="M89.437,39.23c-0.841-0.794-2.113-0.996-3.253-1.303c-6.326-1.704-10.654-8.81-9.266-15.213	c0.263-1.212,0.699-2.481,0.305-3.657c-0.269-0.802-0.89-1.431-1.518-1.998c-2.302-2.08-4.969-3.756-7.842-4.927	c-1.004-0.409-2.071-0.763-3.152-0.671c-2.18,0.185-3.761,2.054-5.53,3.342c-3.165,2.305-7.539,2.836-11.164,1.355	c-2.785-1.138-5.089-3.345-7.992-4.135c-5.97-1.624-12.029,3.563-12.528,9.371c-0.228,2.655,0.108,5.368-0.489,7.965	c-0.918,3.996-4.18,7.362-8.145,8.405c-1.309,0.344-2.765,0.496-3.748,1.427c-0.944,0.894-1.2,2.281-1.352,3.573	c-0.364,3.109-0.413,6.256-0.145,9.375c0.066,0.772,0.162,1.573,0.575,2.228c0.886,1.407,2.796,1.616,4.408,2.022	c6.248,1.572,10.437,8.793,8.701,14.997c-0.45,1.607-1.24,3.244-0.852,4.867c0.356,1.489,1.623,2.566,2.877,3.444	c2.133,1.494,4.445,2.734,6.869,3.685c4.813,1.889,7.341-3.734,11.307-5.198c3.455-1.275,7.517-1.036,10.726,0.809	c4.013,2.307,5.67,7.06,10.959,4.862c3.061-1.272,8.887-5.175,8.43-9.124c-0.532-4.59-1.557-7.851,0.862-12.212	c1.847-3.33,5-6.006,8.762-6.84c0.984-0.218,2.044-0.35,2.823-0.989c1.048-0.861,1.263-2.362,1.322-3.717	c0.14-3.18-0.159-6.38-0.885-9.479c-0.17-0.726-0.377-1.474-0.858-2.043C89.58,39.372,89.51,39.299,89.437,39.23z"/><path fill="none" d="M72.846,23.741c0.263-1.212,0.699-2.481,0.305-3.657c-0.269-0.802-0.89-1.431-1.518-1.998 c-2.302-2.08-4.969-3.756-7.842-4.927c-1.004-0.409-2.071-0.763-3.152-0.671c-2.18,0.185-3.761,2.054-5.53,3.342 c-3.165,2.305-7.539,2.836-11.164,1.355c-2.785-1.138-5.089-3.345-7.992-4.135c-5.97-1.624-12.029,3.563-12.528,9.371 c-0.228,2.655,0.108,5.368-0.489,7.965c-0.918,3.996-4.18,7.362-8.145,8.405c-1.309,0.344-2.766,0.496-3.748,1.427 c-0.944,0.894-1.2,2.281-1.352,3.573c-0.364,3.109-0.413,6.256-0.145,9.375c0.066,0.772,0.162,1.573,0.575,2.228 c0.886,1.407,2.796,1.616,4.408,2.022c6.248,1.572,10.437,8.793,8.701,14.997c-0.45,1.607-1.24,3.244-0.852,4.867 c0.356,1.489,1.623,2.566,2.877,3.444c2.133,1.494,4.445,2.734,6.869,3.685c4.813,1.889,7.341-3.734,11.307-5.198 c3.455-1.275,7.517-1.036,10.726,0.809c4.013,2.307,5.67,7.06,10.959,4.862c3.061-1.272,8.887-5.175,8.43-9.124 c-0.532-4.59-1.557-7.851,0.862-12.212c1.847-3.33,5-6.006,8.762-6.84c0.984-0.218,2.044-0.35,2.823-0.989 c1.048-0.861,1.263-2.362,1.322-3.717c0.14-3.18-0.159-6.38-0.885-9.479c-0.17-0.726-0.377-1.474-0.858-2.043 c-0.066-0.078-0.136-0.152-0.209-0.221c-0.841-0.794-2.113-0.996-3.253-1.303"/><path d="M72.846,23.741c0,0,0.054-0.205,0.16-0.604c0.098-0.401,0.297-0.988,0.37-1.805c0.032-0.406,0.015-0.88-0.151-1.372 c-0.164-0.493-0.487-0.971-0.89-1.417c-0.825-0.872-1.85-1.774-3.091-2.703c-1.242-0.92-2.705-1.848-4.412-2.663 c-0.855-0.396-1.747-0.831-2.817-1.067c-1.059-0.259-2.333-0.106-3.383,0.43c-1.066,0.519-1.966,1.278-2.868,1.99 c-0.898,0.727-1.817,1.343-2.892,1.8c-2.128,0.897-4.696,1.186-7.151,0.528c-2.471-0.581-4.566-2.479-7.384-3.967 c-1.407-0.737-3.124-1.248-4.874-1.218c-1.75,0.016-3.503,0.516-5.062,1.378c-3.104,1.718-5.617,4.861-6.209,8.76 c-0.272,1.892-0.176,3.642-0.265,5.339c-0.074,1.7-0.3,3.251-0.991,4.67c-0.673,1.424-1.7,2.719-2.975,3.717 c-0.64,0.495-1.338,0.917-2.076,1.244c-0.764,0.335-1.449,0.534-2.402,0.73c-0.926,0.219-2.135,0.444-3.28,1.356 c-1.161,0.943-1.651,2.303-1.88,3.385c-0.231,1.129-0.301,2.073-0.403,3.09c-0.088,1.006-0.141,2.02-0.166,3.039 c-0.023,1.019-0.012,2.044,0.029,3.072l0.091,1.544c0.039,0.501,0.073,1.066,0.218,1.726c0.136,0.641,0.448,1.44,0.983,2.053 c0.522,0.62,1.183,1.055,1.789,1.327c1.226,0.549,2.317,0.701,3.213,0.936c1.655,0.438,3.207,1.34,4.478,2.599 c1.243,1.276,2.272,2.842,2.815,4.577c0.564,1.722,0.72,3.554,0.331,5.243c-0.137,0.823-0.542,1.757-0.834,2.902 c-0.148,0.57-0.266,1.212-0.278,1.911c-0.018,0.695,0.117,1.465,0.391,2.147c0.569,1.374,1.532,2.296,2.419,3.008 c0.912,0.715,1.761,1.254,2.667,1.815c1.815,1.093,3.646,2.007,5.651,2.747c1.146,0.451,2.58,0.547,3.829,0.221 c1.26-0.32,2.278-0.958,3.137-1.593c1.707-1.29,2.988-2.679,4.356-3.492c0.342-0.217,0.656-0.351,1.011-0.501 c0.389-0.142,0.784-0.265,1.185-0.366c0.803-0.201,1.626-0.314,2.448-0.342c1.644-0.056,3.283,0.233,4.749,0.848 c1.482,0.597,2.664,1.592,3.971,2.79c0.654,0.592,1.333,1.219,2.138,1.795c0.797,0.568,1.765,1.101,2.861,1.29 c1.091,0.2,2.167,0.06,3.127-0.2c0.943-0.311,1.776-0.689,2.544-1.118c1.544-0.855,2.953-1.85,4.226-3.013 c0.629-0.591,1.234-1.208,1.754-1.922c0.531-0.695,0.99-1.483,1.281-2.374c0.165-0.435,0.23-0.918,0.291-1.393 c0.005-0.476-0.014-1.021-0.079-1.36c-0.115-0.758-0.229-1.51-0.341-2.254c-0.23-1.467-0.418-2.863-0.384-4.193 c0.035-1.328,0.308-2.592,0.807-3.77c0.981-2.377,2.62-4.368,4.489-5.763c0.944-0.689,1.947-1.248,2.983-1.625 c0.523-0.189,1.029-0.338,1.572-0.46c0.545-0.126,1.121-0.25,1.687-0.49c0.564-0.23,1.115-0.628,1.471-1.132 c0.369-0.497,0.57-1.052,0.697-1.583c0.244-1.073,0.214-2.069,0.217-3.006c-0.017-1.885-0.189-3.6-0.43-5.124 c-0.245-1.526-0.55-2.855-0.889-4.002c-0.181-0.572-0.433-1.095-0.789-1.476c-0.352-0.385-0.765-0.618-1.139-0.781 c-0.759-0.311-1.371-0.41-1.772-0.506c-0.403-0.09-0.61-0.136-0.61-0.136s0.204,0.059,0.601,0.173 c0.394,0.119,1.002,0.257,1.719,0.602c0.353,0.179,0.729,0.424,1.032,0.795c0.306,0.367,0.512,0.86,0.656,1.418 c0.268,1.131,0.502,2.475,0.663,3.984c0.158,1.511,0.242,3.201,0.167,5.043c-0.045,0.921-0.074,1.889-0.328,2.784 c-0.25,0.906-0.769,1.647-1.668,1.947c-0.879,0.32-2.088,0.388-3.256,0.786c-1.166,0.368-2.305,0.937-3.383,1.657 c-2.139,1.464-4.03,3.567-5.251,6.213c-0.624,1.321-1.013,2.827-1.107,4.365c-0.096,1.538,0.063,3.065,0.242,4.555 c0.086,0.738,0.172,1.483,0.26,2.234c0.059,0.415,0.026,0.64,0.032,0.931c-0.059,0.285-0.092,0.572-0.22,0.862 c-0.418,1.169-1.414,2.292-2.528,3.266c-1.136,0.975-2.445,1.845-3.834,2.561c-0.696,0.366-1.408,0.648-2.075,0.853 c-0.674,0.154-1.322,0.211-1.924,0.084c-2.453-0.466-4.451-4.3-8.331-5.944c-1.856-0.828-3.903-1.229-5.965-1.2 c-1.031,0.015-2.067,0.137-3.088,0.372c-0.51,0.118-1.016,0.265-1.516,0.438c-0.519,0.198-1.076,0.438-1.541,0.722 c-1.908,1.126-3.265,2.591-4.671,3.591c-0.695,0.503-1.38,0.88-2.021,1.028c-0.641,0.142-1.258,0.121-1.948-0.156 c-1.682-0.642-3.421-1.522-5.025-2.514c-1.602-1.005-3.315-2.165-3.695-3.242c-0.2-0.498-0.186-1.104,0.038-1.958 c0.206-0.843,0.637-1.81,0.875-3.021c0.538-2.345,0.322-4.796-0.413-7.036c-0.717-2.257-2.025-4.265-3.662-5.927 c-1.661-1.639-3.768-2.873-6.055-3.454c-1.074-0.256-2.008-0.423-2.63-0.71c-0.316-0.137-0.505-0.289-0.636-0.435 c-0.132-0.147-0.209-0.307-0.293-0.64c-0.076-0.318-0.119-0.755-0.159-1.247l-0.099-1.437c-0.049-0.957-0.07-1.912-0.059-2.862 c0.012-0.95,0.05-1.896,0.119-2.836c0.076-0.925,0.154-1.925,0.304-2.693c0.157-0.791,0.417-1.336,0.765-1.619 c0.353-0.319,1.067-0.555,1.97-0.761c0.9-0.195,2.036-0.527,2.975-0.977c0.967-0.451,1.865-1.019,2.677-1.675 c1.62-1.322,2.898-3,3.729-4.868c0.863-1.875,1.095-3.953,1.121-5.765c0.043-1.825-0.077-3.571,0.111-5.084 c0.341-2.948,2.302-5.637,4.768-7.082c1.232-0.733,2.6-1.166,3.958-1.225c1.365-0.063,2.674,0.26,3.934,0.858 c2.505,1.175,4.9,3.196,7.871,3.807c2.884,0.648,5.766,0.19,8.124-0.922c1.179-0.555,2.234-1.34,3.105-2.115 c0.883-0.766,1.695-1.502,2.574-1.977c0.875-0.484,1.796-0.644,2.719-0.476c0.921,0.153,1.818,0.526,2.666,0.872 c1.695,0.709,3.167,1.54,4.43,2.378c1.257,0.841,2.33,1.688,3.17,2.484c0.409,0.405,0.732,0.832,0.908,1.278 c0.179,0.444,0.223,0.891,0.215,1.287c-0.024,0.796-0.186,1.399-0.261,1.803C72.888,23.532,72.846,23.741,72.846,23.741z"/><path fill="#f2f2f2" d="M66.847,49.099c0.01-1.192-0.639-2.243-1.695-3.036c-5.193-3.901-10.51-7.632-15.939-11.198	c-0.577-0.379-1.167-0.531-1.734-0.528c-0.159-0.143-0.314-0.292-0.476-0.433c-1.226-1.064-2.773-1.308-4.225-0.545	c-2.279,1.199-2.655,3.534-2.495,5.861c0.04,0.577,0.08,1.155,0.119,1.732c0,0.001,0,0.002,0,0.004	c-0.086,2.46,0.148,4.974,0.222,7.435c0.074,2.478,0.148,4.957,0.222,7.435c0.039,1.322,0.079,2.644,0.118,3.965	c0.042,1.424-0.081,2.971,0.655,4.248c1.695,2.943,5.139,2.268,7.581,0.835c1.976-1.16,3.934-2.351,5.855-3.601	c3.891-2.534,7.656-5.323,11.008-8.544C67.036,51.793,67.287,50.327,66.847,49.099z"/></svg>
// @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();
})();