From 984901b1dfa2e9882abedfc6f46d3b11e43a655f Mon Sep 17 00:00:00 2001 From: absidue <48293849+absidue@users.noreply.github.com> Date: Mon, 12 Sep 2022 18:24:27 +0200 Subject: [PATCH 1/2] Fix the localised description containing broken URLs --- src/renderer/views/Watch/Watch.js | 66 ++++++++++++++++++++++++++++++- 1 file changed, 64 insertions(+), 2 deletions(-) diff --git a/src/renderer/views/Watch/Watch.js b/src/renderer/views/Watch/Watch.js index f18e9808d91eb..cb26943643edd 100644 --- a/src/renderer/views/Watch/Watch.js +++ b/src/renderer/views/Watch/Watch.js @@ -308,8 +308,20 @@ export default Vue.extend({ this.videoPublished = new Date(result.videoDetails.publishDate.replace('-', '/')).getTime() try { // workaround for description localization - const descriptionLines = result.response.contents.twoColumnWatchNextResults.results.results.contents[1].videoSecondaryInfoRenderer.description?.runs - this.videoDescription = descriptionLines?.map(line => line.text).join('') ?? '' + const descriptionRuns = result.response.contents.twoColumnWatchNextResults.results.results.contents[1].videoSecondaryInfoRenderer.description?.runs + + if (!Array.isArray(descriptionRuns)) { + // eslint-disable-next-line no-throw-literal + throw ['not an array', descriptionRuns] + } + + const fallbackDescription = result.player_response.videoDetails.shortDescription + + // YouTube truncates links in the localised description + // so we need to fix them here, so that autolinker can do it's job properly later on + this.videoDescription = descriptionRuns + .map(run => this.processDescriptionPart(run, fallbackDescription)) + .join('') } catch (err) { console.error('Failed to extract localised video description, falling back to the standard one.', err) // if the workaround for localization fails, this sets the description to the potentially non-localized value @@ -787,6 +799,56 @@ export default Vue.extend({ }) }, + processDescriptionPart(part, fallbackDescription) { + const timestampRegex = /^([0-9]+:)?[0-9]+:[0-9]+$/ + + if (typeof part.navigationEndpoint === 'undefined' || part.navigationEndpoint === null) { + return part.text + } + + if (part.navigationEndpoint.urlEndpoint) { + const urlWithTracking = part.navigationEndpoint.urlEndpoint.url + const url = new URL(urlWithTracking) + + if (url.hostname === 'www.youtube.com' && url.pathname === '/redirect' && url.searchParams.has('q')) { + // remove utm tracking parameters + const realURL = new URL(url.searchParams.get('q')) + + realURL.searchParams.delete('utm_source') + realURL.searchParams.delete('utm_medium') + realURL.searchParams.delete('utm_campaign') + realURL.searchParams.delete('utm_term') + realURL.searchParams.delete('utm_content') + + return realURL.toString() + } else if (fallbackDescription.includes(urlWithTracking)) { + // this is probably a special YouTube URL like http://www.youtube.com/approachingnirvana + // only use it if it exists in the fallback description + // otherwise assume YouTube has changed it's tracking URLs and throw an error + return urlWithTracking + } + + // eslint-disable-next-line no-throw-literal + throw `Failed to extract real URL from tracking URL: ${urlWithTracking}` + } else if (part.navigationEndpoint.watchEndpoint) { + if (timestampRegex.test(part.text)) { + return part.text + } + const watchEndpoint = part.navigationEndpoint.watchEndpoint + + let videoURL = `https://www.youtube.com/watch?v=${watchEndpoint.videoId}` + if (watchEndpoint.startTimeSeconds !== 0) { + videoURL += `&t=${watchEndpoint.startTimeSeconds}s` + } + return videoURL + } else { + // Some YouTube URLs don't have the urlEndpoint so we handle them here + + const path = part.navigationEndpoint.commandMetadata.webCommandMetadata.url + return `https://www.youtube.com${path}` + } + }, + addToHistory: function (watchProgress) { const videoData = { videoId: this.videoId, From 3d8fe26c7c2b51f4fdd6f6031dbe42460d2daccf Mon Sep 17 00:00:00 2001 From: absidue <48293849+absidue@users.noreply.github.com> Date: Sat, 17 Sep 2022 14:52:06 +0200 Subject: [PATCH 2/2] Fix hashtag handling in title and description --- src/renderer/views/Watch/Watch.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/renderer/views/Watch/Watch.js b/src/renderer/views/Watch/Watch.js index cb26943643edd..3fab671c62c62 100644 --- a/src/renderer/views/Watch/Watch.js +++ b/src/renderer/views/Watch/Watch.js @@ -277,7 +277,7 @@ export default Vue.extend({ } try { // workaround for title localization - this.videoTitle = result.response.contents.twoColumnWatchNextResults.results.results.contents[0].videoPrimaryInfoRenderer.title.runs[0].text + this.videoTitle = result.response.contents.twoColumnWatchNextResults.results.results.contents[0].videoPrimaryInfoRenderer.title.runs.map(run => run.text).join('') } catch (err) { console.error('Failed to extract localised video title, falling back to the standard one.', err) // if the workaround for localization fails, this sets the title to the potentially non-localized value @@ -802,7 +802,7 @@ export default Vue.extend({ processDescriptionPart(part, fallbackDescription) { const timestampRegex = /^([0-9]+:)?[0-9]+:[0-9]+$/ - if (typeof part.navigationEndpoint === 'undefined' || part.navigationEndpoint === null) { + if (typeof part.navigationEndpoint === 'undefined' || part.navigationEndpoint === null || part.text.startsWith('#')) { return part.text }