🏠 Home 

Twitch [Chatters Count]

Shows the amount of people in the chat

// ==UserScript==
// @name         Twitch [Chatters Count]
// @namespace    https://github.com/pabli24
// @version      1.0.0
// @description  Shows the amount of people in the chat
// @author       Pabli
// @license      MIT
// @match        https://www.twitch.tv/*
// @icon         
// @run-at       document-end
// @grant        GM_xmlhttpRequest
// @grant        GM_info
// @grant        GM_notification
// @grant        GM_openInTab
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// ==/UserScript==
(async () => {
'use strict';
let settings = {
previewCards: {
value: await GM_getValue('previewCards', true),
label: 'Show for preview cards in main/directory page',
},
underPlayer: {
value: await GM_getValue('underPlayer', true),
label: 'Show under the player next to the viewer count',
},
theatreMode: {
value: await GM_getValue('theatreMode', true),
label: 'Show in the theatre mode when mousing over the player',
},
topChat: {
value: await GM_getValue('topChat', false),
label: 'Show at the top of the chat',
},
numberFormat: {
value: await GM_getValue('numberFormat', true),
label: 'Number format based on your language (en-US if disabled)',
},
};
let menuCommands = {};
async function updateMenu() {
Object.keys(menuCommands).forEach(key => GM_unregisterMenuCommand(menuCommands[key]));
Object.entries(settings).forEach(([key, config]) => {
menuCommands[key] = GM_registerMenuCommand(
`${config.value ? '☑' : '☐'} ${config.label}`,
async () => {
settings[key].value = !settings[key].value;
await GM_setValue(key, settings[key].value);
GM_notification(`${config.label} is now ${config.value ? 'enabled' : 'disabled'}`, GM_info.script.name, GM_info.script.icon);
updateMenu();
}
);
});
}
updateMenu();
let updater = null;
let currentChannel = '';
setInterval(() => {
const path = window.location.pathname;
if (settings.previewCards.value && (path === '/' || path.startsWith('/directory'))) {
checkForNewCards();
return;
}
let channelName = path.split('/')[1];
if (!channelName || ['settings', 'subscriptions', 'inventory', 'wallet', 'privacy', 'annual-recap'].includes(channelName)) return;
if (channelName === 'popout') {
channelName = path.split('/')[2];
}
if (currentChannel !== channelName) {
currentChannel = channelName;
if (updater) {
clearInterval(updater);
updater = null;
}
}
if (settings.underPlayer.value) {
chattersCount();
} else if (ct) {
ct.remove();
ct = null;
}
if (settings.theatreMode.value) {
tmChattersCount();
} else if (tmCt) {
tmCt.remove();
tmCt = null;
}
if (settings.topChat.value) {
chatChattersCount();
} else if (chatCt) {
chatCt.remove();
chatCt = null;
}
if (!updater) {
updateCount(channelName);
updater = setInterval(() => updateCount(channelName), 60000);
}
}, 1500);
let ct = null;
let tmCt = null;
let chatCt = null;
let chatters = '⏳';
async function updateCount(channelName) {
chatters = await getChatters(channelName);
[ct, tmCt, chatCt].forEach(el => {
if (el) el.textContent = `[${chatters}]`;
});
}
function createChattersCounter(id) {
const counter = document.createElement('span');
counter.id = id;
counter.title = 'Chatters Count';
counter.textContent = `[${chatters}]`;
counter.style.cssText = `
color: var(--color-text-live, #ff8280);
font-size: var(--font-size-base, 1.3rem);
font-weight: var(--font-weight-semibold, 600);
font-feature-settings: "tnum";
line-height: var(--line-height-body, 1.5);
margin-left: 0.5rem;
align-content: center;
`;
return counter;
}
function chattersCount() {
ct = document.getElementById('chatters-count');
if (ct != null) return;
const viewersCount = document.querySelector('p[data-a-target="animated-channel-viewers-count"]')?.parentElement;
if (!viewersCount) return;
ct = createChattersCounter('chatters-count');
viewersCount.appendChild(ct);
}
function tmChattersCount() {
tmCt = document.getElementById('tm-chatters-count');
if (tmCt != null) return;
const tmInfoCard = document.querySelector('p[data-test-selector="stream-info-card-component__description"]');
if (!tmInfoCard) return;
tmCt = createChattersCounter('tm-chatters-count');
tmInfoCard.appendChild(tmCt);
}
function chatChattersCount() {
chatCt = document.getElementById('chat-chatters-count');
if (chatCt != null) return;
const chatHeader = document.querySelector('button[data-test-selector="chat-viewer-list"]')?.parentElement;
if (!chatHeader) return;
chatCt = createChattersCounter('chat-chatters-count');
chatHeader.prepend(chatCt);
}
function checkForNewCards() {
const previewCards = document.querySelectorAll('a[data-a-target="preview-card-image-link"]');
previewCards.forEach(addChattersToCard);
}
async function addChattersToCard(card) {
const cardStat = card.querySelector('div.tw-media-card-stat');
if (!cardStat || cardStat.querySelector('.directory-chatters-count') || card.dataset.chattersLoading === 'true') return;
card.dataset.chattersLoading = 'true';
const channelName = card.getAttribute('href').slice(1);
const counter = document.createElement('span');
counter.className = 'directory-chatters-count';
counter.style.paddingLeft = '0.4rem';
const count = await getChatters(channelName);
counter.textContent = `[${count}]`;
cardStat.appendChild(counter);
delete card.dataset.chattersLoading;
}
let lang = settings.numberFormat.value ? '' : 'en-US';
async function getChatters(channel) {
if (!lang) lang = document.documentElement.getAttribute('lang') || 'en-US';
return new Promise((resolve) => {
GM_xmlhttpRequest({
method: 'POST',
url: 'https://gql.twitch.tv/gql',
headers: {
'Client-Id': 'kimne78kx3ncx6brgo4mv6wki5h1ko',
'Content-Type': 'application/json'
},
data: JSON.stringify({
query: `query { channel(name: "${channel}") { chatters { count } } }`
}),
onload: response => {
const data = JSON.parse(response.responseText);
const count = data?.data?.channel?.chatters?.count;
resolve(count != null && !isNaN(count) ? new Intl.NumberFormat(lang).format(count) : 'N/A');
}
});
});
}
})();