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>/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); });