From 51c249e9f49a880b8451121d396a60ff73ce5600 Mon Sep 17 00:00:00 2001
From: Kelvin <kelvin@futo.org>
Date: Tue, 7 Nov 2023 20:33:45 +0100
Subject: [PATCH] Channel memberships and live chat donation support

---
 YoutubeConfig.json |  2 +-
 YoutubeScript.js   | 52 +++++++++++++++++++++++++++++++++++++++-------
 2 files changed, 45 insertions(+), 9 deletions(-)

diff --git a/YoutubeConfig.json b/YoutubeConfig.json
index e5db5b3..47745c3 100644
--- a/YoutubeConfig.json
+++ b/YoutubeConfig.json
@@ -7,7 +7,7 @@
 	"sourceUrl": "https://plugins.grayjay.app/Youtube/YoutubeConfig.json",
 	"repositoryUrl": "https://futo.org",
 	"scriptUrl": "./YoutubeScript.js",
-	"version": 138,
+	"version": 139,
 	"iconUrl": "./youtube.png",
 	"id": "35ae969a-a7db-11ed-afa1-0242ac120002",
 
diff --git a/YoutubeScript.js b/YoutubeScript.js
index a8e72b6..16e2b39 100644
--- a/YoutubeScript.js
+++ b/YoutubeScript.js
@@ -17,7 +17,7 @@ const URL_SUBSCRIPTIONS_M = "https://m.youtube.com/feed/subscriptions";
 const URL_PLAYLIST = "https://youtube.com/playlist?list=";
 const URL_PLAYLISTS_M = "https://m.youtube.com/feed/library";
 
-const URL_LIVE_CHAT_HTML = "https://www.youtube.com/live_chat?v=";
+const URL_LIVE_CHAT_HTML = "https://www.youtube.com/live_chat";
 const URL_LIVE_CHAT = "https://www.youtube.com/youtubei/v1/live_chat/get_live_chat";
 
 const URL_WATCHTIME = "https://www.youtube.com/api/stats/watchtime";
@@ -474,13 +474,44 @@ source.getContentChapters = function(url, initialData) {
 
     return result;
 }
+function getVideoDetailsHtml(url, useLogin) {
+	const shouldUseLogin = useLogin && bridge.isLoggedIn();
 
+	const headersUsed = (shouldUseLogin) ? getAuthContextHeaders(false) : {};
+	headersUsed["Accept-Language"] = "en-US";
+
+	const result = http.GET(url, headersUsed, shouldUseLogin);
+	if(result.isOk)
+	    return result.body;
+	else
+	    throw new ScriptException("Failed to get video details [" + url + "] (" + result.code + ")");
+}
 source.getLiveChatWindow = function(url) {
 	const id = extractVideoIDFromUrl(url);
 	if(!id)
 		throw new ScriptException("No valid id found");
 
-	const chatUrl = URL_LIVE_CHAT_HTML + id;
+	let chatUrl = URL_LIVE_CHAT_HTML + "?v=" + id;
+    if(bridge.isLoggedIn()) {
+        try {
+            //Try live version
+            const html = getVideoDetailsHtml(url, true);
+            const initialData = getInitialData(html)
+
+            const continuations = initialData?.contents?.twoColumnWatchNextResults?.conversationBar?.liveChatRenderer?.continuations;
+            if(continuations) {
+                const continuation = continuations.find(x=>x.reloadContinuationData?.continuation);
+                if(continuation) {
+                    chatUrl = URL_LIVE_CHAT_HTML + "?continuation=" + continuation.reloadContinuationData?.continuation;
+
+                }
+            }
+        }
+        catch(ex) {
+            log("Failed to get live chat window continuation, fallback to standard\n" + ex)
+        }
+    }
+
 
 	const chatHtmlResp = http.GET(chatUrl, {}, false);
 	if(!chatHtmlResp.isOk)
@@ -497,7 +528,7 @@ source.getLiveEvents = function(url) {
 	if(!id)
 		throw new ScriptException("No valid id found");
 
-	const chatHtmlResp = http.GET(URL_LIVE_CHAT_HTML + id, {}, false);
+	const chatHtmlResp = http.GET(URL_LIVE_CHAT_HTML + "?v=" + id, {}, false);
 	if(!chatHtmlResp.isOk)
 		throw new ScriptException("Failed to get chat html");
 	const chatHtml = chatHtmlResp.body;
@@ -2058,7 +2089,7 @@ function extractVideoPage_VideoDetails(initialData, initialPlayerData, contextDa
 		id: new PlatformID(PLATFORM, videoDetails.videoId, config.id),
 		name: videoDetails.title,
 		thumbnails: new Thumbnails(videoDetails.thumbnail?.thumbnails.map(x=>new Thumbnail(x.url, x.height)) ?? []),
-		author: new PlatformAuthorLink(new PlatformID(PLATFORM, videoDetails.channelId, config.id, PLATFORM_CLAIMTYPE), videoDetails.author, URL_BASE + "/channel/" + videoDetails.channelId, null),
+		author: new PlatformAuthorLink(new PlatformID(PLATFORM, videoDetails.channelId, config.id, PLATFORM_CLAIMTYPE), videoDetails.author, URL_BASE + "/channel/" + videoDetails.channelId, null, null),
 		duration: parseInt(videoDetails.lengthSeconds),
 		viewCount: parseInt(videoDetails.viewCount),
 		url: contextData.url,
@@ -2294,6 +2325,12 @@ function toSRTTime(sec, withDot) {
 }
 
 function extractVideoOwnerRenderer_AuthorLink(renderer) {
+	const id = renderer?.navigationEndpoint?.browseEndpoint?.browseId;
+    const url = (!id) ? extractRuns_Url(renderer.title.runs) : URL_BASE + "/channel/" + id;
+
+    const hasMembership = !!(renderer?.membershipButton?.buttonRenderer)
+    let membershipUrl = (hasMembership) ? url + "/join" : null;
+
 	let bestThumbnail = null;
 	if(renderer.thumbnail?.thumbnails)
 		bestThumbnail = renderer.thumbnail.thumbnails[renderer.thumbnail.thumbnails.length - 1].url;
@@ -2301,13 +2338,12 @@ function extractVideoOwnerRenderer_AuthorLink(renderer) {
 	let subscribers = null;
 	if(renderer.subscriberCountText)
 		subscribers = extractHumanNumber_Integer(extractText_String(renderer.subscriberCountText));
-	
-	const id = renderer?.navigationEndpoint?.browseEndpoint?.browseId;
+
 	return new PlatformAuthorLink(new PlatformID(PLATFORM, id, config.id, PLATFORM_CLAIMTYPE), 
 		extractRuns_String(renderer.title.runs),
-		(!id) ? extractRuns_Url(renderer.title.runs) : URL_BASE + "/channel/" + id,
+		url,
 		bestThumbnail,
-		subscribers);
+		subscribers, membershipUrl);
 }
 function extractTwoColumnWatchNextResultContents_CommentsPager(contextUrl, contents) {
 	//Add additional/better details
-- 
GitLab