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