// ==UserScript==
// @name Return YouTube Comment Username
// @name:ja YouTubeコメント欄の名前を元に戻す
// @name:zh-CN 恢復 YouTube 评论用户名
// @name:zh-TW 恢復 YouTube 評論名稱
// @version 0.3.4.1
// @author yakisova41
// @license MIT
// @icon 
// @namespace https://yt-returnname-api.pages.dev/extension/
// @description This script replaces the "handle" in the YouTube comments section to user name
// @description:ja YouTubeのコメント欄の名前をハンドル(@...)からユーザー名に書き換えます。
// @description:zh-TW 此腳本將 YouTube 評論部分中的“handle”替換為用戶名
// @description:zh-CN 此脚本将 YouTube 评论部分中的“handle”替换为用户名
// @match https://www.youtube.com/*
// @grant unsafeWindow
// @run-at document-end
// ==/UserScript==
const scriptString = `// src/getUserName.ts
async function getUserName(id) {
const data = await fetch(
\`https://www.youtube.com/youtubei/v1/browse?key=AIzaSyAO_FJ2SlqU8Q4STEHLGCilw_Y9_11qcW8&prettyPrint=false\`,
{
method: "POST",
headers: {
accept: "*/*",
"accept-encoding": "gzip, deflate, br",
"accept-language": "ja",
"content-type": "application/json",
cookie: \`GPS=1; YSC=YajuSEnP; DEVICE_INFO=DEVICE_INFO; VISITOR_INFO1_LIVE=LLIIVVEE; PREF=f6=40000000&tz=Asia.Tokyo; ST-o2eza2=itct=itct&endpoint=%7B%22clickTrackingParams%22%3A%22CBQQ8JMBGAciEwjNqtCAASAhXnm1YBHABY%3D%22%2C%22commandMetadata%22%3A%7B%22webCommandMetadata%22%3A%7B%22url%22%3A%22%2F%40FUCKYOUTUBE%2Fchannels%22%2C%22webPageType%22%3A%22WEB_PAGE_TYPE_CHANNEL%22%2C%22rootVe%22%3A3611%2C%22apiUrl%22%3A%22%2Fyoutubei%2Fv1%2Fbrowse%22%7D%7D%2C%22browseEndpoint%22%3A%7B%22browseId%22%3A%22\${id}%22%2C%22params%22%3A%22EghjaGFubmVsc_IGBAoCUgA%253D%22%2C%22canonicalBaseUrl%22%3A%22%2F%40FUCK_YOUTUBE%22%7D%7D\`,
dnt: "1",
referer: \`https://www.youtube.com/channel/\${id}\`,
"sec-ch-ua": \`"Chromium";v="110", "Not A(Brand";v="24", "Google Chrome";v="110"\`,
"sec-ch-ua-arch": "x86",
"sec-ch-ua-bitness": "64",
"sec-ch-ua-full-version": "110.0.5481.104",
"sec-ch-ua-full-version-list": \`"Chromium";v="110.0.5481.104", "Not A(Brand";v="24.0.0.0", "Google Chrome";v="110.0.5481.104"\`,
"sec-ch-ua-mobile": "?0",
"sec-ch-ua-platform": "Windows",
"sec-ch-ua-platform-version": "15.0.0",
"sec-ch-ua-wow64": "?0",
"sec-fetch-dest": "empty",
"sec-fetch-mode": "same-origin",
"sec-fetch-site": "same-origin",
"user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36",
"x-client-data": "x-client-data",
"x-goog-authuser": "0",
"x-goog-visitor-id": "visitorData",
"x-origin": "https://www.youtube.com",
"x-youtube-bootstrap-logged-in": "true",
"x-youtube-client-name": "1",
"x-youtube-client-version": "2.20230217.01.00"
},
body: JSON.stringify({
context: {
client: {
hl: "ja",
gl: "JP",
remoteHost: "1919:8a10:1145:1419:e1c9:b81a:09db:ff3a",
deviceMake: "",
deviceModel: "",
visitorData: "visitorData",
userAgent: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36,gzip(gfe)",
clientName: "WEB",
clientVersion: "2.20230217.01.00",
osName: "Windows",
osVersion: "10.0",
originalUrl: "https://www.youtube.com/@FUCK_YOUTUBE/channels",
platform: "DESKTOP",
clientFormFactor: "UNKNOWN_FORM_FACTOR",
configInfo: {
appInstallData: "appInstallData"
},
userInterfaceTheme: "USER_INTERFACE_THEME_DARK",
timeZone: "Asia/Tokyo",
browserName: "Chrome",
browserVersion: "110.0.0.0",
acceptHeader: "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
deviceExperimentId: "deviceExperimentId",
screenWidthPoints: 599,
screenHeightPoints: 937,
screenPixelDensity: 1,
screenDensityFloat: 1,
utcOffsetMinutes: 540,
memoryTotalKbytes: "8000000",
mainAppWebInfo: {
graftUrl: "/@FUCK_YOUTUBE/channels",
pwaInstallabilityStatus: "PWA_INSTALLABILITY_STATUS_CAN_BE_INSTALLED",
webDisplayMode: "WEB_DISPLAY_MODE_BROWSER",
isWebNativeShareAvailable: true
}
},
user: { lockedSafetyMode: false },
request: {
useSsl: true,
internalExperimentFlags: [],
consistencyTokenJars: []
},
clickTracking: {
clickTrackingParams: "UnkoBuuriiiiiiiicuuusssaiMAJIDE="
},
adSignalsInfo: {
params: [
{ key: "dt", value: "1145141919810" },
{ key: "flash", value: "0" },
{ key: "frm", value: "0" },
{ key: "u_tz", value: "540" },
{ key: "u_his", value: "1" },
{ key: "u_h", value: "1080" },
{ key: "u_w", value: "1920" },
{ key: "u_ah", value: "1040" },
{ key: "u_aw", value: "1920" },
{ key: "u_cd", value: "24" },
{ key: "bc", value: "31" },
{ key: "bih", value: "937" },
{ key: "biw", value: "582" },
{
key: "brdim",
value: "-1920,0,-1920,0,1920,0,1920,1040,599,937"
},
{ key: "vis", value: "1" },
{ key: "wgl", value: "true" },
{ key: "ca_type", value: "image" }
]
}
},
browseId: id,
params: "YajuSenpaiInmu1919%3D"
})
}
).then(async (res) => await res.text()).then((text) => {
const data2 = JSON.parse(text);
const name = data2.header.c4TabbedHeaderRenderer.title;
return name;
});
return data;
}
// src/index.ts
function main() {
const handleYtAction = (e) => {
const { actionName } = e.detail;
switch (actionName) {
case "yt-append-continuation-items-action":
handleYtAppendContinuationItemsAction(e.detail);
break;
case "yt-reload-continuation-items-command":
handleYtReloadContinuationItemsCommand(e.detail);
break;
case "yt-history-load":
handleYtHistory(e.detail);
break;
case "yt-get-multi-page-menu-action":
handleYtGetMultiPageMenuAction(e.detail);
break;
case "yt-create-comment-action":
handleYtCreateCommentAction(e.detail);
break;
}
};
document.addEventListener("yt-action", handleYtAction);
document.addEventListener("yt-navigate-finish", ({ detail }) => {
document.removeEventListener("yt-action", handleYtAction);
document.addEventListener("yt-action", handleYtAction);
});
}
function handleYtAppendContinuationItemsAction(detail) {
const continuationItems = detail.args[0].appendContinuationItemsAction.continuationItems;
if (isCommentRenderer(continuationItems)) {
const replyDetail = detail;
setTimeout(() => {
rewriteReplytNameFromContinuationItems(
replyDetail.args[0].appendContinuationItemsAction.continuationItems
);
}, 1);
} else {
const commentDetail = detail;
setTimeout(() => {
rewriteCommentNameFromContinuationItems(
commentDetail.args[0].appendContinuationItemsAction.continuationItems
);
}, 100);
}
}
function handleYtReloadContinuationItemsCommand(detail) {
const reloadDetail = detail;
const { slot } = reloadDetail.args[0].reloadContinuationItemsCommand;
if (slot === "RELOAD_CONTINUATION_SLOT_BODY") {
const continuationItems = reloadDetail.args[0].reloadContinuationItemsCommand.continuationItems;
setTimeout(() => {
rewriteCommentNameFromContinuationItems(continuationItems);
}, 100);
}
}
function handleYtHistory(detail) {
const historyDetail = detail;
const continuationItems = historyDetail.args[1].historyEntry?.rootData.response.contents.twoColumnWatchNextResults?.results?.results?.contents[3]?.itemSectionRenderer?.contents;
if (continuationItems !== void 0) {
setTimeout(() => {
rewriteCommentNameFromContinuationItems(continuationItems);
}, 100);
}
}
function handleYtGetMultiPageMenuAction(detail) {
const getMultiPageMenuDetail = detail;
const continuationItems = getMultiPageMenuDetail.args[0].getMultiPageMenuAction.menu.multiPageMenuRenderer.sections[1].itemSectionRenderer?.contents;
const highLightedTeaserContents = getMultiPageMenuDetail.args[0]?.getMultiPageMenuAction?.menu?.multiPageMenuRenderer.sections[1].itemSectionRenderer?.contents[0]?.commentThreadRenderer.replies?.commentRepliesRenderer?.teaserContents;
if (continuationItems !== void 0) {
setTimeout(() => {
rewriteCommentNameFromContinuationItems(continuationItems);
if (highLightedTeaserContents !== void 0) {
const highLightedReplyRenderer = highLightedTeaserContents[0]?.commentRenderer;
let isContainer = highLightedReplyRenderer.authorIsChannelOwner;
if (highLightedReplyRenderer.authorCommentBadge !== void 0) {
isContainer = true;
}
rewriteHighlightedReply(
highLightedReplyRenderer.trackingParams,
isContainer,
highLightedReplyRenderer.authorEndpoint.browseEndpoint.browseId
);
}
}, 100);
}
}
function rewriteHighlightedReply(trackedParams, isContainer, userId) {
const elem = findElementByTrackingParams(
trackedParams,
"ytd-comment-renderer"
);
const rewriteHighlightedReplyElem = (elem2) => {
nameRewriteOfCommentRenderer(elem2, isContainer, userId);
};
if (elem === null) {
void reSearchElement(trackedParams, "ytd-comment-renderer").then((elem2) => {
rewriteHighlightedReplyElem(elem2);
});
} else {
rewriteHighlightedReplyElem(elem);
}
}
function handleYtCreateCommentAction(detail) {
const createCommentDetail = detail;
const continuationItems = [
{
commentThreadRenderer: createCommentDetail.args[0].createCommentAction.contents.commentThreadRenderer
}
];
setTimeout(() => {
rewriteCommentNameFromContinuationItems(continuationItems);
}, 100);
}
function rewriteReplytNameFromContinuationItems(continuationItems) {
continuationItems.forEach((continuationItem) => {
const { commentRenderer } = continuationItem;
if (commentRenderer !== void 0) {
const replyCommentRenderer = findElementByTrackingParams(
commentRenderer.trackingParams,
"ytd-comment-renderer"
);
const reWriteReplyCommentRenderer = (replyCommentRenderer2) => {
let isContainer = commentRenderer.authorIsChannelOwner;
if (commentRenderer.authorCommentBadge !== void 0) {
isContainer = true;
}
nameRewriteOfCommentRenderer(
replyCommentRenderer2,
isContainer,
commentRenderer.authorEndpoint.browseEndpoint.browseId
);
mentionRewriteOfCommentRenderer(replyCommentRenderer2);
};
if (replyCommentRenderer !== null) {
reWriteReplyCommentRenderer(replyCommentRenderer);
} else {
void reSearchElement(
commentRenderer.trackingParams,
"ytd-comment-renderer"
).then((el) => {
reWriteReplyCommentRenderer(el);
});
}
}
});
}
function rewriteCommentNameFromContinuationItems(continuationItems) {
continuationItems.forEach((continuationItem) => {
const { commentThreadRenderer } = continuationItem;
if (commentThreadRenderer !== void 0) {
const { trackingParams } = commentThreadRenderer;
const commentElem = findElementByTrackingParams(
trackingParams,
"#comments > #sections > #contents > ytd-comment-thread-renderer"
);
const reWriteCommentElem = (commentElem2) => {
const commentRenderer = commentElem2?.querySelector(
"ytd-comment-renderer"
);
if (commentRenderer !== null && commentRenderer !== void 0) {
let isContainer = commentThreadRenderer.comment.commentRenderer.authorIsChannelOwner;
if (commentThreadRenderer.comment.commentRenderer.authorCommentBadge !== void 0) {
isContainer = true;
}
nameRewriteOfCommentRenderer(
commentRenderer,
isContainer,
commentThreadRenderer.comment.commentRenderer.authorEndpoint.browseEndpoint.browseId
);
}
};
if (commentElem !== null) {
reWriteCommentElem(commentElem);
} else {
void reSearchElement(
trackingParams,
"ytd-comment-thread-renderer"
).then((commentElem2) => {
reWriteCommentElem(commentElem2);
});
}
}
});
}
function nameRewriteOfCommentRenderer(commentRenderer, isNameContainerRender, userId) {
let nameElem = commentRenderer.querySelector(
"#body > #main > #header > #header-author > h3 > a > span"
);
if (isNameContainerRender) {
nameElem = commentRenderer.querySelector(
"#body > #main > #header > #header-author > #author-comment-badge > ytd-author-comment-badge-renderer > a > #channel-name > #container > #text-container > yt-formatted-string"
);
}
void getUserName(userId).then((name) => {
if (nameElem !== null) {
nameElem.textContent = name;
}
});
}
function mentionRewriteOfCommentRenderer(commentRenderer) {
const aTags = commentRenderer.querySelectorAll(
"#body > #main > #comment-content > ytd-expander > #content > #content-text > a"
);
aTags.forEach((aTag) => {
if (aTag.textContent?.match("@.*") !== null) {
const href = aTag.getAttribute("href");
if (href !== null) {
void getUserName(href.split("/")[2]).then((name) => {
aTag.textContent = \`@\${name} \`;
});
}
}
});
}
function isCommentRenderer(continuationItems) {
if (continuationItems.length > 0) {
if (continuationItems[0].hasOwnProperty("commentThreadRenderer")) {
return false;
}
if (continuationItems[0].hasOwnProperty("commentRenderer")) {
return true;
}
}
return false;
}
function findElementByTrackingParams(trackingParams, elementSelector) {
let returnElement = null;
const elems = document.querySelectorAll(elementSelector);
elems.forEach((elem) => {
if (elem.trackedParams === trackingParams) {
returnElement = elem;
}
});
return returnElement;
}
async function reSearchElement(trackingParams, elementType) {
return await new Promise((resolve) => {
let isFinding = true;
const search = () => {
const el = findElementByTrackingParams(trackingParams, elementType);
if (el !== null) {
resolve(el);
isFinding = false;
}
if (isFinding) {
setTimeout(() => {
search();
}, 100);
}
};
search();
});
}
// node_modules/ts-extension-builder/tmp/entry.ts
var args = {};
if (typeof GM_info !== "undefined" && GM_info.script.grant !== void 0) {
GM_info.script.grant.forEach((propatyName) => {
let keyName = propatyName.split("GM_")[1];
if (keyName === "xmlhttpRequest") {
keyName = "xmlHttpRequest";
}
args[propatyName] = GM[keyName];
});
}
main(args);
`;
const script = document.createElement("script");
script.innerHTML = scriptString
unsafeWindow.document.body.appendChild(script)