From d0849440d26ea20a7905f422166d40882b26f8f8 Mon Sep 17 00:00:00 2001 From: Thomas E Tuoti Date: Sun, 22 Dec 2024 17:55:46 -0700 Subject: [PATCH] Allow time seeking in videoparser --- core/modules/parsers/audioparser.js | 89 +++------------- core/modules/parsers/videoparser.js | 156 +++++++++++++++++++++++++--- 2 files changed, 161 insertions(+), 84 deletions(-) diff --git a/core/modules/parsers/audioparser.js b/core/modules/parsers/audioparser.js index ef4543c57..5eb2ff985 100644 --- a/core/modules/parsers/audioparser.js +++ b/core/modules/parsers/audioparser.js @@ -2,89 +2,33 @@ title: $:/core/modules/parsers/audioparser.js type: application/javascript module-type: parser + +The audio parser parses an audio tiddler into an embeddable HTML element + \*/ -(function () { +(function(){ /*jslint node: true, browser: true */ /*global $tw: false */ "use strict"; -var AudioParser = function (type, text, options) { +var AudioParser = function(type,text,options) { var element = { - type: "element", - tag: "audio", - attributes: { - controls: { type: "string", value: "controls" }, - style: { type: "string", value: "width: 100%; object-fit: contain" }, - preload: { type: "string", value: "auto" }, // Changed to auto for full loading - class: { type: "string", value: "tw-audio-element" } - } - }; - - // Set source with range support - if (options._canonical_uri) { - element.children = [{ type: "element", - tag: "source", + tag: "audio", attributes: { - src: { type: "string", value: options._canonical_uri }, - type: { type: "string", value: type } + controls: {type: "string", value: "controls"}, + style: {type: "string", value: "width: 100%; object-fit: contain"} } - }]; - } else if (text) { - element.attributes.src = { type: "string", value: "data:" + type + ";base64," + text }; + }, + src; + if(options._canonical_uri) { + element.attributes.src = {type: "string", value: options._canonical_uri}; + } else if(text) { + element.attributes.src = {type: "string", value: "data:" + type + ";base64," + text}; } - - if ($tw.browser) { - $tw.hooks.addHook("th-page-refreshed", function () { - setTimeout(function () { - Array.from(document.getElementsByClassName("tw-audio-element")).forEach(function (audio) { - // Force aggressive loading - audio.preload = "auto"; - audio.autobuffer = true; - - // Create XMLHttpRequest to load full file - var xhr = new XMLHttpRequest(); - xhr.open('GET', audio.currentSrc, true); - xhr.responseType = 'blob'; - - xhr.onprogress = function(e) { - if (e.lengthComputable) { - var percentComplete = (e.loaded / e.total) * 100; - console.log("Loading: " + percentComplete.toFixed(2) + "%"); - } - }; - - xhr.onload = function() { - if (xhr.status === 200) { - var blob = new Blob([xhr.response], { type: type }); - var url = URL.createObjectURL(blob); - audio.src = url; - } - }; - - xhr.send(); - - // Monitor buffer state - audio.addEventListener("progress", function() { - var buffered = this.buffered; - if(buffered.length > 0) { - var total = 0; - for(var i = 0; i < buffered.length; i++) { - var start = buffered.start(i); - var end = buffered.end(i); - total += (end - start); - console.log(`Buffer ${i}: ${start}-${end} (${((end-start)/this.duration*100).toFixed(2)}%)`); - } - console.log(`Total buffered: ${(total/this.duration*100).toFixed(2)}%`); - } - }); - }); - }, 100); - }); - } - this.tree = [element]; + this.source = text; this.type = type; }; @@ -93,4 +37,5 @@ exports["audio/mpeg"] = AudioParser; exports["audio/mp3"] = AudioParser; exports["audio/mp4"] = AudioParser; -})(); \ No newline at end of file +})(); + diff --git a/core/modules/parsers/videoparser.js b/core/modules/parsers/videoparser.js index 1c8a38bb2..27c4f7e73 100644 --- a/core/modules/parsers/videoparser.js +++ b/core/modules/parsers/videoparser.js @@ -6,29 +6,161 @@ module-type: parser The video parser parses a video tiddler into an embeddable HTML element \*/ -(function(){ +(function () { /*jslint node: true, browser: true */ /*global $tw: false */ "use strict"; -var VideoParser = function(type,text,options) { +var VideoParser = function (type, text, options) { var element = { + type: "element", + tag: "video", + attributes: { + controls: { type: "string", value: "controls" }, + style: { type: "string", value: "width: 100%; object-fit: contain" }, + preload: { type: "string", value: "auto" }, + class: { type: "string", value: "tw-video-element" } + } + }; + + if (options._canonical_uri) { + element.children = [{ type: "element", - tag: "video", + tag: "source", attributes: { - controls: {type: "string", value: "controls"}, - style: {type: "string", value: "width: 100%; object-fit: contain"} + src: { type: "string", value: options._canonical_uri }, + type: { type: "string", value: type } } - }, - src; - if(options._canonical_uri) { - element.attributes.src = {type: "string", value: options._canonical_uri}; - } else if(text) { - element.attributes.src = {type: "string", value: "data:" + type + ";base64," + text}; + }]; + } else if (text) { + element.attributes.src = { type: "string", value: "data:" + type + ";base64," + text }; } + + if ($tw.browser) { + const processedVideos = new WeakMap(); + const bufferThreshold = 0.1; // 10% buffered before play + + $tw.hooks.addHook("th-page-refreshed", function() { + Array.from(document.getElementsByClassName("tw-video-element")).forEach(function(video) { + if (processedVideos.has(video)) return; + + // Create loading overlay + const overlay = document.createElement('div'); + overlay.style.cssText = 'position:absolute;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.5);display:flex;align-items:center;justify-content:center;color:white;'; + overlay.innerHTML = 'Loading 0%'; + video.parentNode.style.position = 'relative'; + video.parentNode.appendChild(overlay); + + video.preload = "auto"; + video.autobuffer = true; + + // Prevent play until buffered + video.addEventListener('play', function(e) { + if (video.buffered.length === 0 || (video.buffered.end(0) / video.duration) < bufferThreshold) { + console.log('Waiting for buffer...'); + video.pause(); + } + }, { passive: true }); + + // Monitor buffering + video.addEventListener('progress', function() { + if (video.buffered.length > 0) { + const progress = (video.buffered.end(0) / video.duration * 100).toFixed(2); + console.log(`Buffer: ${progress}%`); + overlay.innerHTML = `Loading ${progress}%`; + + if ((video.buffered.end(0) / video.duration) >= bufferThreshold) { + overlay.style.display = 'none'; + } + } + }, { passive: true }); + + const xhr = new XMLHttpRequest(); + xhr.open('GET', video.currentSrc, true); + xhr.responseType = 'blob'; + + xhr.onprogress = function(e) { + if (e.lengthComputable) { + console.log(`Download: ${(e.loaded / e.total * 100).toFixed(2)}%`); + } + }; + + xhr.onload = function() { + if (xhr.status === 200) { + const blob = new Blob([xhr.response], { type: video.type || 'video/mp4' }); + const url = URL.createObjectURL(blob); + video._blob = blob; + video.src = url; + processedVideos.set(video, { + blob: blob, + url: url + }); + + // Handle seeking with passive listener + video.addEventListener('seeking', function() { + if (!video.src || video.src === '') { + video.src = URL.createObjectURL(video._blob); + } + }, { passive: true }); + + // Cleanup + const observer = new MutationObserver(function(mutations) { + mutations.forEach(function(mutation) { + if ([...mutation.removedNodes].includes(video)) { + URL.revokeObjectURL(url); + processedVideos.delete(video); + observer.disconnect(); + if (overlay.parentNode) { + overlay.parentNode.removeChild(overlay); + } + } + }); + }); + + observer.observe(video.parentNode, { + childList: true, + subtree: true + }); + } + }; + + xhr.send(); + + video.addEventListener('loadedmetadata', async () => { + // Use requestAnimationFrame instead of direct style changes + requestAnimationFrame(async () => { + const xhr = new XMLHttpRequest(); + xhr.open('HEAD', video.currentSrc); + + xhr.onload = () => { + const observer = new MutationObserver((mutations) => { + requestAnimationFrame(() => { + mutations.forEach((mutation) => { + if (mutation.addedNodes.length) { + const overlay = mutation.target.querySelector('.play-overlay'); + if (overlay) { + overlay.parentNode.removeChild(overlay); + } + } + }); + }); + }); + + observer.observe(video.parentNode, { + childList: true, + subtree: true + }); + }; + + xhr.send(); + }); + }, { passive: true }); + }); + }, { passive: true }); + } + this.tree = [element]; - this.source = text; this.type = type; };