aboutsummaryrefslogtreecommitdiffstats
path: root/youtube/static/js/transcript-table.js
diff options
context:
space:
mode:
authorLibravatarLibravatar Biswakalyan Bhuyan <biswa@surgot.in> 2024-09-19 15:33:11 +0530
committerLibravatarLibravatar Biswakalyan Bhuyan <biswa@surgot.in> 2024-09-19 15:33:11 +0530
commita4e01da27c08e43a67b2618ad1e71c1f8f86d5cd (patch)
tree5b8f407dbb7e9d1ab2106ac0cc8564897e7a2098 /youtube/static/js/transcript-table.js
downloadyt-local-master.tar.gz
yt-local-master.tar.bz2
yt-local-master.zip
youtube fronendHEADmaster
Diffstat (limited to 'youtube/static/js/transcript-table.js')
-rw-r--r--youtube/static/js/transcript-table.js151
1 files changed, 151 insertions, 0 deletions
diff --git a/youtube/static/js/transcript-table.js b/youtube/static/js/transcript-table.js
new file mode 100644
index 0000000..5cee97e
--- /dev/null
+++ b/youtube/static/js/transcript-table.js
@@ -0,0 +1,151 @@
+let details_tt, select_tt, table_tt;
+
+function renderCues() {
+ const selectedTrack = QId("js-video-player").textTracks[select_tt.selectedIndex];
+ const cuesList = [...selectedTrack.cues];
+ const is_automatic = cuesList[0].text.startsWith(" \n");
+
+ // Firefox ignores cues starting with a blank line containing a space
+ // Automatic captions contain such a blank line in the first cue
+ let ff_bug = false;
+ if (!cuesList[0].text.length) { ff_bug = true; is_automatic = true };
+ let rows;
+
+ function forEachCue(callback) {
+ for (let i=0; i < cuesList.length; i++) {
+ let txt, startTime = selectedTrack.cues[i].startTime;
+ if (is_automatic) {
+ // Automatic captions repeat content. The new segment is displayed
+ // on the bottom row; the old one is displayed on the top row.
+ // So grab the bottom row only. Skip every other cue because the bottom
+ // row is empty.
+ if (i % 2) continue;
+ if (ff_bug && !selectedTrack.cues[i].text.length) {
+ txt = selectedTrack.cues[i+1].text;
+ } else {
+ txt = selectedTrack.cues[i].text.split('\n')[1].replace(/<[\d:.]*?><c>(.*?)<\/c>/g, "$1");
+ }
+ } else {
+ txt = selectedTrack.cues[i].text;
+ }
+ callback(startTime, txt);
+ }
+ }
+
+ function createTimestampLink(startTime, txt, title=null) {
+ a = document.createElement("a");
+ a.appendChild(text(txt));
+ a.href = "javascript:;"; // TODO: replace this with ?t parameter
+ if (title) a.title = title;
+ a.addEventListener("click", (e) => {
+ QId("js-video-player").currentTime = startTime;
+ })
+ return a;
+ }
+
+ clearNode(table_tt);
+ console.log("render cues..", selectedTrack.cues.length);
+ if (Q("input#transcript-use-table").checked) {
+ forEachCue((startTime, txt) => {
+ let tr, td, a;
+ tr = document.createElement("tr");
+
+ td = document.createElement("td")
+ td.appendChild(createTimestampLink(startTime, toTimestamp(startTime)));
+ tr.appendChild(td);
+
+ td = document.createElement("td")
+ td.appendChild(text(txt));
+ tr.appendChild(td);
+
+ table_tt.appendChild(tr);
+ });
+ rows = table_tt.rows;
+ }
+ else {
+ forEachCue((startTime, txt) => {
+ span = document.createElement("span");
+ let idx = txt.indexOf(" ", 1);
+ let [firstWord, rest] = [txt.slice(0, idx), txt.slice(idx)];
+
+ span.appendChild(createTimestampLink(startTime, firstWord, toTimestamp(startTime)));
+ if (rest) span.appendChild(text(rest + " "));
+ table_tt.appendChild(span);
+ });
+ rows = table_tt.childNodes;
+ }
+
+ let lastActiveRow = null;
+ let row;
+ function colorCurRow(e) {
+ // console.log("cuechange:", e);
+ let activeCueIdx = cuesList.findIndex((c) => c == selectedTrack.activeCues[0]);
+ let activeRowIdx = is_automatic ? Math.floor(activeCueIdx / 2) : activeCueIdx;
+
+ if (lastActiveRow) lastActiveRow.style.backgroundColor = "";
+ if (activeRowIdx < 0) return;
+ row = rows[activeRowIdx];
+ row.style.backgroundColor = "#0cc12e42";
+ lastActiveRow = row;
+ }
+ selectedTrack.addEventListener("cuechange", colorCurRow);
+}
+
+function loadCues() {
+ const textTracks = QId("js-video-player").textTracks;
+ const selectedTrack = textTracks[select_tt.selectedIndex];
+
+ // See https://developer.mozilla.org/en-US/docs/Web/API/TextTrack/mode
+ // This code will (I think) make sure that the selected track's cues
+ // are loaded even if the track subtitles aren't on (showing). Setting it
+ // to hidden will load them.
+ let selected_track_target_mode = "hidden";
+
+ for (let track of textTracks) {
+ // Want to avoid unshowing selected track if it's showing
+ if (track.mode === "showing") selected_track_target_mode = "showing";
+
+ if (track !== selectedTrack) track.mode = "disabled";
+ }
+ if (selectedTrack.mode == "disabled") {
+ selectedTrack.mode = selected_track_target_mode;
+ }
+
+ let intervalID = setInterval(() => {
+ if (selectedTrack.cues && selectedTrack.cues.length) {
+ clearInterval(intervalID);
+ renderCues();
+ }
+ }, 100);
+}
+
+window.addEventListener('DOMContentLoaded', function() {
+ const textTracks = QId("js-video-player").textTracks;
+ if (!textTracks.length) return;
+
+ details_tt = Q("details#transcript-details");
+ details_tt.addEventListener("toggle", () => {
+ if (details_tt.open) loadCues();
+ });
+
+ select_tt = Q("select#select-tt");
+ select_tt.selectedIndex = getDefaultTranscriptTrackIdx();
+ select_tt.addEventListener("change", loadCues);
+
+ table_tt = Q("table#transcript-table");
+ table_tt.appendChild(text("loading..."));
+
+ textTracks.addEventListener("change", (e) => {
+ // console.log(e);
+ let idx = getActiveTranscriptTrackIdx(); // sadly not provided by 'e'
+ if (textTracks[idx].mode == "showing") {
+ select_tt.selectedIndex = idx;
+ loadCues();
+ }
+ else if (details_tt.open && textTracks[idx].mode == "disabled") {
+ textTracks[idx].mode = "hidden"; // so we still receive 'oncuechange'
+ }
+ })
+
+ Q("input#transcript-use-table").addEventListener("change", renderCues);
+});