- /*
-
- MIT License
-
- Copyright 2023 CY Fung
-
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
-
- The above copyright notice and this permission notice shall be included in all
- copies or substantial portions of the Software.
-
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- SOFTWARE.
-
- */
- // ==UserScript==
- // @name Restore YouTube Username from Handle to Custom
- // @namespace http://tampermonkey.net/
- // @version 0.5.19
- // @license MIT License
-
- // @author CY Fung
- // @match https://www.youtube.com/*
- // @match https://m.youtube.com/*
- // @exclude /^https?://\S+\.(txt|png|jpg|jpeg|gif|xml|svg|manifest|log|ini)[^\/]*$/
- // @icon https://github.com/cyfung1031/userscript-supports/raw/main/icons/general-icon.png
- // @supportURL https://github.com/cyfung1031/userscript-supports
- // @run-at document-start
- // @grant none
- // @unwrap
- // @allFrames true
- // @inject-into page
-
- // @compatible firefox 55
- // @compatible chrome 61
- // @compatible opera 48
- // @compatible safari 11.1
- // @compatible edge 16
-
- // @description To restore YouTube Username to the traditional custom name
- // @description:ja YouTubeのユーザー名を伝統的なカスタム名に復元するために。
- // @description:zh-TW 將 YouTube 使用者名稱從 Handle 恢復為自訂名稱
- // @description:zh-CN 将 YouTube 用户名从 Handle 恢复为自定义名称
-
- // @description:ko YouTube 사용자 이름을 전통적인 사용자 지정 이름으로 복원합니다.
- // @description:ru Восстановление имени пользователя YouTube с помощью обычного настраиваемого имени
- // @description:af Herstel YouTube-gebruikersnaam vanaf Handvat na Aangepaste Naam
- // @description:az YouTube İstifadəçi Adını Özəl Adından İstifadə Etmək
- // @description:id Mengembalikan Nama Pengguna YouTube dari Handle ke Kustom
- // @description:ms Pulihkan Nama Pengguna YouTube dari Handle ke Tersuai
- // @description:bs Vrati YouTube korisničko ime sa ručke na prilagođeno ime
- // @description:ca Restaurar el nom d'usuari de YouTube de la màniga al personalitzat
- // @description:cs Obnovte uživatelské jméno YouTube z rukojeti na vlastní
- // @description:da Gendan YouTube-brugernavn fra Håndtag til Brugerdefineret
- // @description:de Stellt den YouTube-Benutzernamen von Handle auf Benutzerdefiniert wieder her
- // @description:et Taasta YouTube'i kasutajanimi käepidemelt kohandatud nimeks
- // @description:es Restaurar el nombre de usuario de YouTube de la manija al personalizado
- // @description:eu Berrezarri YouTube Erabiltzaile-izena Manipulatik Pertsonalizatuera
- // @description:fr Restaurer le nom d'utilisateur YouTube de la poignée au nom personnalisé
- // @description:gl Restaura o nome de usuario de YouTube da manexa ao personalizado
- // @description:hr Vrati YouTube korisničko ime s ručke na prilagođeno ime
- // @description:zu Hlanganisa Igama Lokusebenza lwe-YouTube kusuka kwi-Handle kuze kube kusebenza kakhulu
- // @description:is Endurheimtu YouTube Notandanafn frá Handfangi til Sérsniðins
- // @description:it Ripristina il nome utente di YouTube da Handle a Personalizzato
- // @description:sw Rejesha Jina la Mtumiaji wa YouTube kutoka kwa Kishughulikia hadi Desturi
- // @description:lv Atjaunot YouTube lietotāja vārdu no roktura uz pielāgotu
- // @description:lt Atkurti „YouTube“ naudotojo vardą iš rankenos į tinkamą
- // @description:hu Visszaállítja a YouTube felhasználónevet a fogantyútól a testreszabottra
- // @description:nl Herstel YouTube-gebruikersnaam van Handle naar Aangepaste naam
- // @description:uz YouTube foydalanuvchining nomini Hurda dan Shaxsiylashtirilgan nomga qaytarish
- // @description:pl Przywróć nazwę użytkownika YouTube z uchwytu na niestandardową
- // @description:pt Restaurar o nome de usuário do YouTube de Handle para Personalizado
- // @description:pt-BR Restaurar o nome de usuário do YouTube de Handle para Personalizado
- // @description:ro Restaurați Numele de Utilizator YouTube de la Mâner la Personalizat
- // @description:sq Rikthe emrin e përdoruesit të YouTube nga Maja në Personalizuar
- // @description:sk Obnovte používateľské meno YouTube z rukoväte na vlastné
- // @description:sl Obnovite uporabniško ime YouTube iz ročaja v prilagojeno ime
- // @description:sr Вратите YouTube корисничко име са ручке на прилагођено име
- // @description:fi Palauta YouTube-käyttäjänimi kahvasta räätälöityyn nimeen
- // @description:sv Återställ YouTube-användarnamnet från handtaget till anpassat namn
- // @description:vi Khôi phục Tên người dùng YouTube từ Tay cầm thành Tùy chỉnh
- // @description:tr Özel İsme Kullanıcı Adını İade Etme
- // @description:be Вярнуць імя карыстальніка YouTube з ручкі на наладжваецца
- // @description:bg Възстановяване на потребителско име на YouTube от дръжка до персонализирано име
- // @description:ky YouTube колдонуучунун атты боюнча атын тамашалау
- // @description:kk YouTube пайдаланушының атын Handle танымасынан Жеке атауға қайта түсіру
- // @description:mk Врати го името на YouTube корисникот од држачот во прилагодено име
- // @description:mn YouTube хэрэглэгчийн нэрийг хүчирхийлэгчидээс Байгууллагад нь солих
- // @description:uk Відновлення імені користувача YouTube з ручки на налаштоване ім'я
- // @description:el Επαναφορά Ονόματος Χρήστη YouTube από τη Λαβή σε Προσαρμοσμένο
- // @description:hy Վերականգնել YouTube-ի Օգտագործողի Անունը Ձեռքից Հատուցվածությանով
- // @description:ur ہینڈل سے یوٹیوب یوزر نیم کو کسٹم میں بحال کریں
- // @description:ar استعادة اسم مستخدم YouTube من المقبض إلى مخصص
- // @description:fa بازیابی نام کاربری یوتیوب از دستگیره به سفارشی
- // @description:ne ह्यान्डलबाट यूट्युब प्रयोगकर्तानाम अनुकूलमा उपयोग गर्नुहोस्
- // @description:mr हॅंडलमधून यूट्यूब वापरकर्तानाव अनुकूलित करा
- // @description:hi हैंडल से यूट्यूब यूजरनेम को कस्टम में बदलें
- // @description:as কাস্টম হতে হ্যান্ডেল পৰা ইউটিউব ব্যৱহাৰকাৰী নাম পুনৰ্স্থাপন কৰক
- // @description:bn হ্যান্ডেল থেকে কাস্টম করে ইউটিউব ব্যবহারকারীর নাম পুনরুদ্ধার করুন
- // @description:pa ਯੂਟਿਊਬ ਯੂਜਰਨਾਂ ਦਾ ਨਾਂ ਹੈਂਡਲ ਤੋਂ ਕਸਟਮ ਵਿੱਚ ਮੁੜ ਲਾਓ
- // @description:gu હેન્ડલથી કસ્ટમમાંથી YouTube વપરાશકર્તાનું નામ પુનર્સ્થાપિત કરો
- // @description:or କଷ୍ଟମରେ ହେନ୍ଡେଲରୁ YouTube ବ୍ୟବହାରକାରୀଙ୍କ ନାମ ପୁନର୍ପ୍ରାପ୍ତ କରନ୍ତୁ
- // @description:ta ஹேண்டில் இருந்து கஸ்டம் ஆக்கும் யூடியூப் பயனர்பெயரை மீட்டமைக்கவும்
- // @description:te హాండును నుంచి YouTube వాడుకరి పేరును కస్టమ్కు పునరుద్ధరించండి
- // @description:kn ಹ್ಯಾಂಡಲ್ನಿಂದ YouTube ಬಳಕೆದಾರ ಹೆಸರನ್ನು ಅನುಕೂಲಿತ ಮಾಡಿ
- // @description:ml ഹാന്റിൽ നിന്ന് കസ്റ്റം ആക്കാംമെന്ന് യുട്യൂബ് ഉപയോക്തൃനാമം മറ്റുള്ളവരെ മാറ്റുക
- // @description:si හැන්ඩල් සහිතව YouTube භාවිතයේ පරිශීලක නාමය ස්වයං කරයි
- // @description:th กู้คืนชื่อผู้ใช้ YouTube จากแฮนดิลไปเป็นชื่อที่กำหนดเอง
- // @description:lo ເຮັດຊະນິດຊື່ຜູ້ໃຊ້ YouTube ຈາກ Handle ຈົດໄວ້ເປັນຊື່ປັດຈຸບັນ
- // @description:my Handle မှ YouTube အသုံးပြုသူအမည်ကို စတင်ပြန်စစ်ဆေးပါ
- // @description:ka YouTube მომხმარებლის სახელის აღდგენა ხანდლიდან მორგებულ სახელში
- // @description:am ስለ YouTube የተጠቃሚ ስም ማስቀመጫዎቹን ከ Handle ወደ Custom ውስጥ እንደሚመጣ ይረዳሉ
- // @description:km កំណត់ YouTube ពាក្យឈ្មោះពី Handle ទៅជាឈ្មោះផ្ទាល់ខ្លួន
-
- // ==/UserScript==
-
- /* jshint esversion:8 */
-
-
- /**
- @typedef DisplayNameResultObject
- @type {Object}
- @property {string} title
- @property {string?} externalId
- @property {string[]} ownerUrls
- @property {string} channelUrl
- @property {string} vanityChannelUrl
- @property {boolean|undefined} verified123
-
- */
-
- (function (__CONTEXT__) {
- 'use strict';
-
- const { Promise, fetch } = __CONTEXT__; // YouTube hacks Promise in WaterFox Classic and "Promise.resolve(0)" nevers resolve.
-
- const fxOperator = (proto, propertyName) => {
- let propertyDescriptorGetter = null;
- try {
- propertyDescriptorGetter = Object.getOwnPropertyDescriptor(proto, propertyName).get;
- // https://github.com/cyfung1031/userscript-supports/issues/9
- // .parentNode becomes DocumentFragment
- /**
- *
- * Issue Description: YtCustomElements - Custom DOM Operations overrided in FireFox
- *
- * e.g. ytd-comment-renderer#comment > div#body
- *
- * ${div#body}.parentNode = DocumentFragment <Node.parentNode>
- * ${ytd-comment-renderer#comment}.firstElementChild <Element.firstElementChild>
- *
- * Cofirmed Affected: parentNode, firstChild, firstElementChild, lastChild, lastElementChild, querySelector, & querySelectorAll
- *
- * Alternative way: ytCustomElement.$.xxxxxxx
- *
- */
- } catch (e) { }
- return typeof propertyDescriptorGetter === 'function' ? (e) => propertyDescriptorGetter.call(e) : (e) => e[propertyName];
- };
-
- const fxAPI = (proto, propertyName) => {
- const methodFunc = proto[propertyName];
- return typeof methodFunc === 'function' ? (e, ...args) => methodFunc.apply(e, args) : (e, ...args) => e[propertyName](...args);
- };
-
- const nodeParent = fxOperator(Node.prototype, 'parentNode');
- const nodeFirstChild = fxOperator(Node.prototype, 'firstChild');
- const nodeNextSibling = fxOperator(Node.prototype, 'nextSibling');
-
- const elementQS = fxAPI(Element.prototype, 'querySelector');
- const elementQSA = fxAPI(Element.prototype, 'querySelectorAll');
-
- /*
-
-
- init
-
- initialdata
- state-progress
- state-responsereceived
- player-autonav-pause
- state-change
- state-navigateend
- player-initialized
- renderer-module-load-start
- video-data-change
- player-state-change
- updateui
- renderer-module-load-end -> channel owner DOM available
- player-autonav-pause
- updateui
- renderer-module-load-start
- updateui
- renderer-module-load-end
-
-
- playing
-
- h5player.video-progress
- h5player.video-progress
- h5player.video-progress
- h5player.video-progress
- ...
-
-
-
-
- navigate new video
-
- state-navigatestart
- state-change => channel owener DOM vanished
- state-progress
- ...
- h5player.player-state-change
- h5player.video-data-change
- h5player.player-state-change
- h5player.muted-autoplay-change
- h5player.volume-change
- h5player.video-data-change
- h5player.volume-change
- ...
-
- state-progress
- ...
-
- state-progress => channel owner DOM appear [just before state-responsereceived]
- state-responsereceived
-
- video-data-change
- state-change
- state-navigateend
-
-
-
-
- UX interaction
-
- user-activity
- player-autonav-pause
-
-
-
-
-
- EventTarget.prototype.uhfsE = EventTarget.prototype.dispatchEvent
-
- EventTarget.prototype.dispatchEvent = function (ev) {
- if (ev instanceof Event) {
- if (ev.type === 'video-progress') { } else {
- console.log(ev.type, this, ev.target)
- if (ev.target === null) {
- console.log(ev.type, (document.querySelector('ytm-app') || 0).data)
- console.log(ev.type, (document.querySelector('ytm-slim-owner-renderer') || 0).textContent)
- }
- }
- }
- return this.uhfsE.apply(this, arguments);
- }
-
-
-
- */
-
- const isMobile = location.hostname === 'm.youtube.com';
-
- const cfg = {};
- class Mutex {
-
- constructor() {
- this.p = Promise.resolve()
- }
-
- /**
- * @param {(lockResolve: () => void)} f
- */
- lockWith(f) {
- this.p = this.p.then(() => new Promise(f).catch(console.warn))
- }
-
- }
-
- class OrderedMutex extends Mutex {
- constructor() {
- super();
- this.nextIndex = 0;
- this.arr = [];
- this.lockFunc = resolve => {
- if (this.arr.length >= 1) {
- let f = this.arr[0];
- if (typeof f === 'function') {
- this.arr.shift();
- if (this.nextIndex > 0) this.nextIndex--;
- f(resolve);
- return;
- } else {
- // reset if unknown error
- this.arr.length = 0;
- this.nextIndex = 0;
- }
- }
- resolve();
- };
- }
- /**
- * @param {(lockResolve: () => void)} f
- */
- add(f) {
- if (this.nextIndex === this.arr.length) {
- this.arr.push(f); this.nextIndex++;
- } else {
- this.arr.splice(this.nextIndex++, 0, f);
- }
- this.lockWith(this.lockFunc);
- }
- }
- const mutex = new OrderedMutex();
-
- /** @type {Map<string, AsyncValue | Object>} */
- const displayNameResStore = new Map();
-
- /** @type {WeakMap<Function, Function>} */
- const dataChangedFuncStore = new WeakMap();
-
- /** @type {Map<string, AsyncValue | string>} */
- const handleToChannelId = new Map();
-
- class AsyncValue {
- constructor() {
- this.__value__ = null;
- this.__resolve__ = null;
- this.__promise__ = new Promise(resolve => {
- this.__resolve__ = resolve;
- });
- }
- setValue(value) {
- const promise = this.__promise__;
- if (promise === null) throw 'Value has already been set.';
- this.__value__ = value;
- this.__promise__ = null;
- this.__resolve__ ? this.__resolve__() : Promise.resolve().then(() => this.__resolve__());
- }
- async getValue() {
- const promise = this.__promise__;
- if (promise === null) return this.__value__;
- await promise; // promise become null
- this.__promise__ = null; // just in case
- this.__resolve__ = null;
- return this.__value__;
- }
- }
-
- /**
- *
- * @param {DisplayNameResultObject} resultInfo
- * @param {string} channelId
- */
- const cacheHandleToChannel = (resultInfo, channelId) => {
-
- const { vanityChannelUrl, ownerUrls } = resultInfo;
-
- let handleText = urlToHandle(vanityChannelUrl || '');
-
- if (handleText) {
- // match = true;
- } else if ((ownerUrls || 0).length >= 1) {
- for (const ownerUrl of ownerUrls) {
- handleText = urlToHandle(ownerUrl || '');
-
- if (handleText) {
- // match = true;
- break;
- }
- }
- }
- if (handleText) {
-
- const asyncValue = handleToChannelId.get(handleText); // nothing if no pending promise
- if (asyncValue instanceof AsyncValue) {
- asyncValue.setValue(channelId);
- }
- handleToChannelId.set(handleText, channelId);
- }
-
-
- }
-
- /**
- *
- * @param {string} channelId
- */
- function stackNewRequest(channelId) {
-
- mutex.add(lockResolve => {
-
- let bResult = displayNameResStore.get(channelId);
- if (!(bResult instanceof AsyncValue)) {
- // resolved or removed
- lockResolve(); lockResolve = null;
- return;
- }
-
- let setResult = (result) => {
- setResult = null;
- if (!result) {
- bResult.fetchingState = 0;
- bResult.setValue(null);
- displayNameResStore.delete(channelId); // create another network response in the next request
- } else {
- bResult.fetchingState = 4;
- bResult.setValue(result);
- displayNameResStore.set(channelId, result); // update store result to resolved value
- }
- bResult = null;
- }
-
- if (bResult.fetchingState >= 2) { // fetchingState == 3 or 4
- // request is already done. no need to stack any request
- lockResolve(); lockResolve = null;
- return;
- }
-
- if (!document.querySelector(`[jkrgy="${channelId}"]`)) {
- // element has already been removed
- lockResolve(); lockResolve = null;
- setResult(null);
- return;
- }
-
- //INNERTUBE_API_KEY = ytcfg.data_.INNERTUBE_API_KEY
-
- bResult.fetchingState = 2;
-
- fetch(new window.Request(`/youtubei/v1/browse?key=${cfg.INNERTUBE_API_KEY}&prettyPrint=false`, {
- "method": "POST",
- "mode": "same-origin",
- "credentials": "same-origin",
-
- referrerPolicy: "no-referrer",
- cache: "default",
- redirect: "error", // there shall be no redirection in this API request
- integrity: "",
- keepalive: false,
- signal: undefined,
-
- "headers": {
- "Content-Type": "application/json",
- "Accept-Encoding": "gzip, deflate, br",
- // X-Youtube-Bootstrap-Logged-In: false,
- // X-Youtube-Client-Name: 1, // INNERTUBE_CONTEXT_CLIENT_NAME
- // X-Youtube-Client-Version: "2.20230622.06.00" // INNERTUBE_CONTEXT_CLIENT_VERSION
- },
- "body": JSON.stringify({
- "context": {
- "client": {
- "clientName": "MWEB",
- "clientVersion": `${cfg.INNERTUBE_CLIENT_VERSION || '2.20230614.01.00'}`,
- "originalUrl": `https://m.youtube.com/channel/${channelId}`,
- "playerType": "UNIPLAYER",
- "platform": "MOBILE",
- "clientFormFactor": "SMALL_FORM_FACTOR",
- "acceptHeader": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
- "mainAppWebInfo": {
- "graftUrl": `/channel/${channelId}`,
- "webDisplayMode": "WEB_DISPLAY_MODE_BROWSER",
- "isWebNativeShareAvailable": true
- }
- },
- "user": {
- "lockedSafetyMode": false
- },
- "request": {
- "useSsl": true,
- "internalExperimentFlags": [],
- "consistencyTokenJars": []
- }
- },
- "browseId": `${channelId}`
- })
- })).then(res => {
- bResult.fetchingState = 3;
- lockResolve();
- lockResolve = null;
- return res.json();
- }).then(res => {
-
- // console.log(displayNameRes, urlToHandle(vanityChannelUrl))
- // displayNameCacheStore.set(channelId, displayNameRes);
-
-
- let resultInfo = ((res || 0).metadata || 0).channelMetadataRenderer;
- if (!resultInfo) {
- // invalid json format
- setResult(null);
- return;
- }
-
- cacheHandleToChannel(resultInfo, channelId);
-
- const { title, externalId, ownerUrls, channelUrl, vanityChannelUrl } = resultInfo;
-
- const displayNameRes = { title, externalId, ownerUrls, channelUrl, vanityChannelUrl, verified123: false };
-
- setResult(displayNameRes);
-
-
- }).catch(e => {
- lockResolve && lockResolve();
- console.warn(e);
- setResult && setResult(null);
- });
-
- });
-
- }
-
- /**
- *
- * @param {string} url Example: _www\.youtube\.com/\@yr__kd_
- * @returns Example: _\@yr__kd_
- */
- function urlToHandle(url) {
-
- if (typeof url !== 'string') return '';
- let i = url.indexOf('.youtube.com/');
- if (i >= 1) url = url.substring(i + '.youtube.com/'.length);
- else if (url.charAt(0) === '/') url = url.substring(1);
- return isDisplayAsHandle(url) ? url : '';
-
- }
-
- /**
- *
- * @param {string} channelId Example: UC0gmRdmpDWJ4dt7DAeRaawA
- * @returns {Promise<DisplayNameResultObject | null>}
- */
- async function getDisplayName(channelId) {
-
- let cache = displayNameResStore.get(channelId);
- let isStackNewRequest = false;
-
- /** @type {AsyncValue | null | undefined} */
- let aResult;
-
- if (cache instanceof AsyncValue) {
- aResult = cache;
- if (aResult.fetchingState >= 2) {
- // aResult.fetchingState
- // 2: network fetch started
- // 3: network fetch ended
- // isStackNewRequest = false;
- } else {
- // 0: invalid before
- // 1: scheduled but not yet fetch
- isStackNewRequest = true;
- }
- } else if (cache) {
- // resolved result
- return cache;
- } else {
- // else no cached value found
- aResult = new AsyncValue();
- displayNameResStore.set(channelId, aResult);
- isStackNewRequest = true;
- }
-
- if (isStackNewRequest) {
- aResult.fetchingState = 1;
- stackNewRequest(channelId);
- }
- return await aResult.getValue();
- }
-
-
- /**
- *
- * @param {string} href Example: https\:\/\/www\.youtube\.com/channel/UC0gmRdmpDWJ4dt7DAeRaawA
- * @returns
- */
- const obtainChannelId = (href) => {
- let m = /\/channel\/(UC[-_a-zA-Z0-9+=.]+)/.exec(`/${href}`);
- // let m = /\/channel\/([^/?#\s]+)/.exec(`/${href}`);
- return !m ? '' : (m[1] || '');
- };
-
-
- /**
- *
- * @param {Element} Node
- */
- const resetWhenDataChanged = (ytElm) => {
- if (ytElm instanceof Element) {
- const anchors = elementQSA(ytElm, 'a[id][href*="channel/"][jkrgy]');
- if ((anchors || 0).length >= 1 && (ytElm.data || 0).jkrgx !== 1) {
- for (const anchor of anchors) {
- anchor.removeAttribute('jkrgy');
- }
- }
- }
- }
-
- /**
- *
- * @param {Function} dataChanged original dataChanged function
- * @returns new dataChanged function
- */
- const dataChangeFuncProducer = (dataChanged) => {
- return function () {
- resetWhenDataChanged(this);
- return dataChanged.apply(this, arguments);
- }
- };
-
- /**
- *
- * @param {HTMLAnchorElement} anchor Example: https\:\/\/www\.youtube\.com/channel/UCRmLncxsQFcOOC8OhzUIfxQ/videos
- * @param {string} channelHref Example: /channel/UCRmLncxsQFcOOC8OhzUIfxQ
- * @param {string} channelId Example: UCRmLncxsQFcOOC8OhzUIfxQ
- * @returns
- */
- const anchorIntegrityCheck = (anchor, channelHref, channelId) => {
-
- let currentHref = anchor.getAttribute('href');
- if (currentHref === channelHref) return true; // /channel/UCRmLncxsQFcOOC8OhzUIfxQ // /channel/UCRmLncxsQFcOOC8OhzUIfxQ
-
- return (currentHref + '/').indexOf(channelHref + '/') >= 0;
-
- };
-
- /**
- *
- * @param {string} currentDisplayed
- * @param {DisplayNameResultObject} fetchResult
- * @returns
- */
- const verifyAndConvertHandle = (currentDisplayed, fetchResult) => {
-
- const { title, externalId, ownerUrls, channelUrl, vanityChannelUrl, verified123 } = fetchResult;
-
- const currentDisplayTrimmed = currentDisplayed.trim();
- let match = false;
- if (verified123) {
- match = true;
- } else if ((vanityChannelUrl || '').endsWith(`/${currentDisplayTrimmed}`)) {
- match = true;
- } else if ((ownerUrls || 0).length >= 1) {
- for (const ownerUrl of ownerUrls) {
- if ((ownerUrl || '').endsWith(`/${currentDisplayTrimmed}`)) {
- match = true;
- break;
- }
- }
- }
- if (match) {
- return currentDisplayTrimmed;
- }
- return '';
-
- };
-
- /**
- *
- *
-
- ### [Handle naming guidelines](https://support.google.com/youtube/answer/11585688?hl=en&co=GENIE.Platform%3DAndroid)
-
- - Is between 3-30 characters
- - Is made up of alphanumeric characters (A–Z, a–z, 0–9)
- - Your handle can also include: underscores (_), hyphens (-), dots (.)
- - Is not URL-like or phone number-like
- - Is not already being used
- - Follows YouTube's Community Guidelines
-
- ### Handle automatically generated by YouTube
- - `@user-XXXX`
- - without dot (.)
- *
- * @param {string} text
- * @returns
- */
- const isDisplayAsHandle = (text) => {
-
- if (typeof text !== 'string') return false;
- if (text.length < 4) return false;
- if (text.indexOf('@') < 0) return false;
- return /^\s*@[-_a-zA-Z0-9.]{3,30}\s*$/.test(text);
-
- };
-
- /**
- *
- * @param {Object[]} contentTexts
- * @param {number} idx
- * @returns
- */
- const contentTextProcess = (contentTexts, idx) => {
- const contentText = contentTexts[idx];
- const text = (contentText || 0).text;
- const url = (((contentText.navigationEndpoint || 0).commandMetadata || 0).webCommandMetadata || 0).url;
- if (typeof url === 'string' && typeof text === 'string') {
-
- if (!isDisplayAsHandle(text)) return null;
- const channelId = obtainChannelId(url);
-
- return getDisplayName(channelId).then(fetchResult => {
- let resolveResult = null;
- if (fetchResult) {
- // note: if that user shown is not found in `a[id]`, the hyperlink would not change
-
- const textTrimmed = verifyAndConvertHandle(text, fetchResult);
- if (textTrimmed) {
- resolveResult = (md) => {
- let runs = ((md || 0).contentText || 0).runs;
- if (!runs || !runs[idx]) return;
- if (runs[idx].text !== text) return;
- runs[idx].text = text.replace(textTrimmed, `@${fetchResult.title}`); // HyperLink always @SomeOne
- md.contentText = Object.assign({}, md.contentText);
- };
- }
- }
- return (resolveResult); // function as a Promise
- });
- }
-
- return null;
- };
-
- /**
- *
- * @param {ParentNode} parentNode
- * @param {Function} callback
- */
- function findTextNodes(parentNode, callback) {
- /**
- *
- * @param {Node} node
- * @returns
- */
- function traverse(node) {
- if (node.nodeType === Node.TEXT_NODE) {
- let r = callback(node);
- if (r === true) return true;
- }
-
- let childNode = nodeFirstChild(node);
- while (childNode) {
- let r = traverse(childNode);
- if (r === true) return true;
- childNode = nodeNextSibling(childNode);
- }
- }
-
- traverse(parentNode);
- }
-
- /**
- *
- * @param {ParentNode} parentNode
- * @param {Object[]} contentTexts
- * @param {number} aidx
- * @returns
- */
- const mobileContentHandleAsync = async (parentNode, contentTexts, aidx) => {
-
-
- let commentText = elementQS(parentNode, '.comment-text');
- if (!commentText) return;
- commentText = elementQS(commentText, '.yt-core-attributed-string') || commentText;
- let currentText = commentText.textContent;
- let textMatch = commentText.textContent === contentTexts.map(e => e.text).join('');
- if (!textMatch) return;
-
- const contentText = contentTexts[aidx];
-
- /** @type {string} */
- const handleText = contentText.text.trim();
- /** @type {string | null | undefined} */
- let channelId;
-
- let cache = handleToChannelId.get(handleText);
-
- if (typeof cache === 'string') {
- channelId = cache;
- } else if (cache instanceof AsyncValue) {
- channelId = await cache.getValue();
- } else {
- const asyncValue = new AsyncValue();
- handleToChannelId.set(handleText, asyncValue);
- channelId = await asyncValue.getValue(); // note: it could be never resolved
- }
-
- if (!channelId) return; // just in case
-
- if (commentText.isConnected !== true) return; // already removed
- if (commentText.textContent !== currentText) return; // already changed
-
- const fetchResult = await getDisplayName(channelId);
-
- if (fetchResult === null) return;
-
- const { title, externalId } = fetchResult;
- if (externalId !== channelId) return; // channel id must be the same
-
- let search = contentText.text;
- if (typeof search === 'string') {
-
- const searchTrimmed = search.trim();
- if (searchTrimmed === handleText) { // ensure integrity after getDisplayName
- contentText.text = search.replace(searchTrimmed, "@" + title);
-
- findTextNodes(commentText, (textnode) => {
- if (textnode.nodeValue.indexOf(search) >= 0) {
- textnode.nodeValue = textnode.nodeValue.replace(search, contentText.text);
- return true;
- }
- });
- }
-
- }
-
- }
-
- /**
- * Process Checking when there is new (unprocessed) anchor DOM element with hyperlink of channel
- * @type {(anchor: HTMLElement, channelHref: string?, mt: string?)}
- */
- const domCheckAsync = isMobile ? async (anchor, channelHref, mt) => {
-
- try {
- if (!channelHref || !mt) return;
- let parentNode = nodeParent(anchor);
- while (parentNode instanceof Node) {
- if (parentNode.nodeName === 'YTM-COMMENT-RENDERER' || ('_commentData' in parentNode)) break;
- parentNode = nodeParent(parentNode);
- }
- if (parentNode instanceof Node) { } else return;
-
- let displayTextDOM = elementQS(parentNode, '.comment-header .comment-title');
- if (!displayTextDOM) return;
- displayTextDOM = elementQS(displayTextDOM, '.yt-core-attributed-string') || displayTextDOM;
- let airaLabel = anchor.getAttribute('aria-label')
- const parentNodeData = parentNode.data;
- let runs = ((parentNodeData || 0).authorText || 0).runs;
-
- if (displayTextDOM && airaLabel && displayTextDOM.textContent.trim() === airaLabel.trim() && isDisplayAsHandle(airaLabel) && runs && (runs[0] || 0).text === airaLabel) {
-
- const fetchResult = await getDisplayName(mt);
-
- if (fetchResult === null) return;
-
- const { title, externalId } = fetchResult;
-
- if (externalId !== mt) return; // channel id must be the same
-
- // anchor href might be changed by external
- if (!anchorIntegrityCheck(anchor, channelHref, externalId)) return;
-
- let found = false;
-
- findTextNodes(displayTextDOM, (textnode) => {
- if (textnode.nodeValue === airaLabel) {
- textnode.nodeValue = title;
- found = true;
- return true;
- }
- });
-
- if (!found) return;
- anchor.setAttribute('aria-label', title);
- runs[0].text = title;
-
-
- const contentTexts = (parentNodeData.contentText || 0).runs;
- if (contentTexts && contentTexts.length >= 2) {
- for (let aidx = 0; aidx < contentTexts.length; aidx++) {
- const contentText = contentTexts[aidx] || 0;
- if (contentText.italics !== true) {
- let isHandle = isDisplayAsHandle(contentText.text);
- if (isHandle) {
- mobileContentHandleAsync(parentNode, contentTexts, aidx);
- }
- }
- }
- }
-
- }
- } catch (e) {
- console.warn(e);
- }
-
- } : async (anchor, channelHref, mt) => {
-
- try {
- if (!channelHref || !mt) return;
- let parentNode = nodeParent(anchor);
- while (parentNode instanceof Node) {
- if (typeof parentNode.is === 'string' && typeof parentNode.dataChanged === 'function') break;
- parentNode = nodeParent(parentNode);
- }
- if (parentNode instanceof Node) { } else return;
- const authorText = (parentNode.data || 0).authorText;
- const currentDisplayed = (authorText || 0).simpleText;
- if (typeof currentDisplayed !== 'string') return;
- if (!isDisplayAsHandle(currentDisplayed)) return;
-
- const oldDataChanged = parentNode.dataChanged;
- if (typeof oldDataChanged === 'function' && !oldDataChanged.jkrgx) {
- let newDataChanged = dataChangedFuncStore.get(oldDataChanged);
- if (!newDataChanged) {
- newDataChanged = dataChangeFuncProducer(oldDataChanged);
- newDataChanged.jkrgx = 1;
- dataChangedFuncStore.set(oldDataChanged, newDataChanged);
- }
- parentNode.dataChanged = newDataChanged;
- }
-
- const fetchResult = await getDisplayName(mt);
-
- if (fetchResult === null) return;
-
- const { title, externalId } = fetchResult;
-
- if (externalId !== mt) return; // channel id must be the same
-
- // anchor href might be changed by external
- if (!anchorIntegrityCheck(anchor, channelHref, externalId)) return;
-
- const parentNodeData = parentNode.data;
- const funcPromises = [];
-
- if (parentNode.isAttached === true && parentNode.isConnected === true && typeof parentNodeData === 'object' && parentNodeData && parentNodeData.authorText === authorText) {
-
- if (authorText.simpleText !== currentDisplayed) return;
- const currentDisplayTrimmed = verifyAndConvertHandle(currentDisplayed, fetchResult);
- const cSimpleText = ((parentNodeData.authorText || 0).simpleText || '');
- if (currentDisplayTrimmed && currentDisplayed !== title && cSimpleText === currentDisplayed) {
-
- // the inside hyperlinks will be only converted if its parent author name is handle
- const contentTexts = (parentNodeData.contentText || 0).runs;
- if (contentTexts && contentTexts.length >= 1) {
- for (let aidx = 0; aidx < contentTexts.length; aidx++) {
- const r = contentTextProcess(contentTexts, aidx);
- if (r instanceof Promise) funcPromises.push(r);
- }
- }
-
- const md = Object.assign({}, parentNodeData);
- let setBadge = false;
- if (((((md.authorCommentBadge || 0).authorCommentBadgeRenderer || 0).authorText || 0).simpleText || '').trim() === cSimpleText.trim()) {
- setBadge = true;
- }
- // parentNode.data = Object.assign({}, { jkrgx: 1 });
- md.authorText = Object.assign({}, md.authorText, { simpleText: currentDisplayed.replace(currentDisplayTrimmed, title) });
- if (setBadge) {
- md.authorCommentBadge = Object.assign({}, md.authorCommentBadge);
- md.authorCommentBadge.authorCommentBadgeRenderer = Object.assign({}, md.authorCommentBadge.authorCommentBadgeRenderer);
- md.authorCommentBadge.authorCommentBadgeRenderer.authorText = Object.assign({}, md.authorCommentBadge.authorCommentBadgeRenderer.authorText, { simpleText: title });
-
- }
- if (funcPromises.length >= 1) {
- let funcs = await Promise.all(funcPromises);
-
- for (const func of funcs) {
- if (typeof func === 'function') {
- func(md);
- }
- }
- }
- parentNode.data = Object.assign({}, md, { jkrgx: 1 });
- }
-
- }
- } catch (e) {
- console.warn(e);
- }
-
- }
-
- /**
- *
- * @param {string} url
- * @returns
- */
- const channelUrlUnify = (url) => {
-
- if (typeof url !== 'string') return url; // type unchanged
- let c0 = url.charAt(0);
- if (c0 === '/') return url;
- let i = url.indexOf('.youtube.com/');
- if (i >= 1) url = url.substring(i + '.youtube.com/'.length - 1);
- else if (url.charAt(0) !== '/') url = '/' + url;
- return url
-
- }
-
- let firstDOMCheck = false;
- const firstDOMChecker = isMobile ? () => {
-
- const channelNameDOM = document.querySelector('ytm-slim-owner-renderer');
- const channelNameDOMData = (channelNameDOM || 0).data;
-
- if (channelNameDOMData) {
- let mainChannelUrl = null;
- let mainFormattedNameUrl = null;
- let mainChannelText = null;
- let mainFormattedNameText = null;
-
- try {
- mainChannelUrl = channelUrlUnify(channelNameDOMData.channelUrl)
- mainFormattedNameUrl = channelUrlUnify(channelNameDOMData.title.runs[0].navigationEndpoint.browseEndpoint.canonicalBaseUrl)
- mainChannelText = channelNameDOMData.channelName
- mainFormattedNameText = channelNameDOMData.title.runs[0].text
- } catch (e) { }
-
- if (mainChannelUrl === mainFormattedNameUrl && mainChannelText === mainFormattedNameText && typeof mainChannelUrl === 'string' && typeof mainChannelText === 'string') {
-
-
- let channelId = obtainChannelId(mainChannelUrl);
- if (channelId) {
- if (mainChannelText.startsWith('@')) return;
- if (mainChannelText.trim() !== mainChannelText) return;
-
- displayNameResStore.set(channelId, {
- channelUrl: `https://www.youtube.com/channel/${channelId}`,
- externalId: `${channelId}`,
- ownerUrls: [],
- title: mainChannelText,
- vanityChannelUrl: null,
- verified123: true
- });
- }
-
-
- }
- }
-
- } : () => {
-
- const channelNameDOM = document.querySelector('ytd-channel-name.ytd-video-owner-renderer');
- const channelNameDOMData = (channelNameDOM || 0).__data;
- if (channelNameDOMData) {
- let mainChannelUrl = null;
- let mainFormattedNameUrl = null;
- let mainChannelText = null;
- let mainFormattedNameText = null;
-
- try {
- mainChannelUrl = channelNameDOMData.channelName.runs[0].navigationEndpoint.browseEndpoint.canonicalBaseUrl
- mainFormattedNameUrl = channelNameDOMData.formattedName.runs[0].navigationEndpoint.browseEndpoint.canonicalBaseUrl
- mainChannelText = channelNameDOMData.channelName.runs[0].text
- mainFormattedNameText = channelNameDOMData.formattedName.runs[0].text
- } catch (e) { }
-
- if (mainChannelUrl === mainFormattedNameUrl && mainChannelText === mainFormattedNameText && typeof mainChannelUrl === 'string' && typeof mainChannelText === 'string') {
-
- let m = /^\/(@[-_a-zA-Z0-9.]{3,30})$/.test(mainChannelUrl);
-
- if (m && m[1] && mainChannelText !== m[1]) {
-
- let channelId = obtainChannelId(mainChannelUrl);
- if (channelId) {
- if (mainChannelText.startsWith('@')) return;
- if (mainChannelText.trim() !== mainChannelText) return;
-
- displayNameResStore.set(channelId, {
- channelUrl: `https://www.youtube.com/channel/${channelId}`,
- externalId: `${channelId}`,
- ownerUrls: [],
- title: mainChannelText,
- vanityChannelUrl: null,
- verified123: true
- });
- }
-
- }
- }
- }
-
-
-
-
- };
-
- // let newAnchorAdded = false;
- /*
-
- const intersectionobserver = new IntersectionObserver((entries) => {
- let anchorAppear = false;
- for (const entry of entries) {
- if (entry.isIntersecting === true) {
- anchorAppear = true;
- break;
- }
- }
- if (anchorAppear && newAnchorAdded) {
- newAnchorAdded = false; // stacking will be only reset when one or more anchor added to DOM.
- mutex.nextIndex = 0; // higher pirority for new elements being shown
- }
- }, {
- rootMargin:"0px 0px 0px 0px",
- threshold:1
- })
- */
-
-
- /* globals WeakRef:false */
-
- /** @type {(o: Object | null) => WeakRef | null} */
- const mWeakRef = typeof WeakRef === 'function' ? (o => o ? new WeakRef(o) : null) : (o => o || null); // typeof InvalidVar == 'undefined'
-
- /** @type {(wr: Object | null) => Object | null} */
- const kRef = (wr => (wr && wr.deref) ? wr.deref() : wr);
-
- let lastNewAnchorLastWR = null;
-
- let domCheckTimeStamp = 0;
-
- const mutexForDOMMutations = new Mutex();
- const domCheckSelector = isMobile ? 'a[aria-label^="@"][href*="channel/"]:not([jkrgy])' : 'a[id][href*="channel/"]:not([jkrgy])';
-
- let domCheckScheduled = false;
- const domChecker = () => {
-
- if(domCheckScheduled) return;
- const newAnchors = document.querySelectorAll(domCheckSelector);
- if (newAnchors.length === 0) {
- return;
- }
- domCheckScheduled = true;
-
- mutexForDOMMutations.lockWith(lockResolve => {
- domCheckScheduled = false;
-
- const newAnchors = document.querySelectorAll(domCheckSelector);
- if (newAnchors.length === 0) {
- lockResolve();
- return;
- }
-
- const cNewAnchorFirst = newAnchors[0]; // non-null
- const cNewAnchorLast = newAnchors[newAnchors.length - 1]; // non-null
- /** @type {HTMLElement | null} */
- const lastNewAnchorLast = kRef(lastNewAnchorLastWR); // HTMLElement | null
- lastNewAnchorLastWR = mWeakRef(cNewAnchorLast);
-
- const ltDOMCheck = domCheckTimeStamp;
- const ctDOMCheck = Date.now();
- domCheckTimeStamp = ctDOMCheck;
-
- if (!firstDOMCheck) {
- firstDOMCheck = true;
- firstDOMChecker();
- }
-
- const isDocumentModifying = ctDOMCheck - ltDOMCheck < 230;
-
- Promise.resolve()
- .then(() => {
- if (lastNewAnchorLast && mutex.nextIndex >= 1) {
-
- if (isDocumentModifying) {
-
- } else {
- if ((lastNewAnchorLast.compareDocumentPosition(cNewAnchorFirst) & 2) === 2) {
- mutex.nextIndex = 0;
- }
- }
-
- /*
- if (mutex.nextIndex === 0) {
- // no change
- } else if ((lastNewAnchorLast.compareDocumentPosition(cNewAnchorLast) & 2) === 2) { // when "XX replies" clicked
- mutex.nextIndex = 0; // highest priority
- } else if (cNewAnchorLast !== cNewAnchorFirst && (lastNewAnchorLast.compareDocumentPosition(cNewAnchorFirst) & 2) === 1) { // rarely
- mutex.nextIndex = Math.floor(mutex.nextIndex / 2); // relatively higher priority
- }
- */
- }
- }).then(() => {
-
- // newAnchorAdded = true;
- for (const anchor of newAnchors) {
- // author-text or name
- // normal url: /channel/xxxxxxx
- // Improve YouTube! - https://www.youtube.com/channel/xxxxxxx/videos
- const href = anchor.getAttribute('href');
- const channelId = obtainChannelId(href); // string, can be empty
- anchor.setAttribute('jkrgy', channelId);
- // intersectionobserver.unobserve(anchor);
- // intersectionobserver.observe(anchor); // force first occurance
- domCheckAsync(anchor, href, channelId);
- }
-
- }).then(lockResolve).catch(e => {
- lockResolve();
- console.warn(e);
- });
-
- });
-
-
- };
-
-
- /** @type {MutationObserver | null} */
- let domObserver = null;
-
- /** @type {(callback: (event: Event)=>{})} */
- const onPageFetched = isMobile ? (callback) => {
-
- let state = 0;
-
- window.addEventListener('state-change', function (evt) {
- if (state === 1) return; // ignore intermediate events
- state = 1; // before value: 0 or 2;
- callback(evt);
- }, false);
-
- window.addEventListener('state-navigateend', function (evt) { // fallback
- let inSeq = state === 1;
- state = 2; // 2 when loaded
- if (!inSeq) callback(evt);
- }, false);
-
- } : (callback) => {
-
- let state = 0;
-
- document.addEventListener('yt-page-data-fetched', function (evt) {
- if (state === 1) return; // ignore intermediate events if any
- state = 1; // before value: 0 or 2;
- callback(evt);
- }, false);
-
- document.addEventListener('yt-navigate-finish', function (evt) { // fallback
- let inSeq = state === 1;
- state = 2; // 2 when loaded
- if (!inSeq) callback(evt);
- }, false);
-
- };
-
- /**
- *
- * @param {Node} app
- * @returns
- */
- function setupOnPageFetched(app) {
-
- firstDOMCheck = false;
-
- if (!cfg['INNERTUBE_API_KEY']) {
- console.warn("Userscript Error: INNERTUBE_API_KEY is not found.");
- return;
- }
-
- if (!domObserver) {
- domObserver = new MutationObserver(domChecker);
- } else {
- domObserver.takeRecords();
- domObserver.disconnect();
- }
-
- domObserver.observe(app, { childList: true, subtree: true });
- domChecker();
-
- }
-
- /**
- *
- * @param {Object|null} dest
- * @param {Object} ytcfg
- * @param {string[]} keys
- * @returns
- */
- function getYtConfig(dest, ytcfg, keys) {
- dest = dest || {};
- if (typeof ytcfg.get === 'function') { // mobile
- for (const key of keys) {
- dest[key] = ytcfg.get(key);
- }
- } else {
- const cfgData = ytcfg.data_ || ytcfg.__data || ytcfg._data || ytcfg.data || ytcfg; // dekstop - ytcfg.data_
- for (const key of keys) {
- dest[key] = cfgData[key];
- }
- }
- return dest;
- }
-
- /** @type { (evt: Object) => Element | null } */
- const getPageApp = isMobile
- ? ((evt) => (document.querySelector('ytm-app#app') || document.querySelector('ytm-app')))
- : ((evt) => ((evt || 0).target || document.querySelector('ytd-app')));
-
- onPageFetched(async (evt) => {
- const app = getPageApp(evt);
- console.assert(app);
- const ytcfg = ((window || 0).ytcfg || 0);
- getYtConfig(cfg, ytcfg, ['INNERTUBE_API_KEY', 'INNERTUBE_CLIENT_VERSION']);
- setupOnPageFetched(app);
- });
-
- })({ Promise, fetch });