@editframe/elements 0.17.6-beta.0 → 0.18.3-beta.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (218) hide show
  1. package/dist/EF_FRAMEGEN.js +1 -1
  2. package/dist/ScrubTrackManager.d.ts +2 -2
  3. package/dist/elements/EFAudio.d.ts +21 -2
  4. package/dist/elements/EFAudio.js +41 -11
  5. package/dist/elements/EFImage.d.ts +1 -0
  6. package/dist/elements/EFImage.js +11 -3
  7. package/dist/elements/EFMedia/AssetIdMediaEngine.d.ts +18 -0
  8. package/dist/elements/EFMedia/AssetIdMediaEngine.js +41 -0
  9. package/dist/elements/EFMedia/AssetMediaEngine.d.ts +47 -0
  10. package/dist/elements/EFMedia/AssetMediaEngine.js +116 -0
  11. package/dist/elements/EFMedia/BaseMediaEngine.d.ts +55 -0
  12. package/dist/elements/EFMedia/BaseMediaEngine.js +96 -0
  13. package/dist/elements/EFMedia/BufferedSeekingInput.d.ts +43 -0
  14. package/dist/elements/EFMedia/BufferedSeekingInput.js +159 -0
  15. package/dist/elements/EFMedia/JitMediaEngine.browsertest.d.ts +0 -0
  16. package/dist/elements/EFMedia/JitMediaEngine.d.ts +31 -0
  17. package/dist/elements/EFMedia/JitMediaEngine.js +62 -0
  18. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.d.ts +9 -0
  19. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.d.ts +16 -0
  20. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.js +48 -0
  21. package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.d.ts +3 -0
  22. package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.js +138 -0
  23. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.d.ts +9 -0
  24. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.d.ts +4 -0
  25. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.js +16 -0
  26. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.d.ts +9 -0
  27. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.d.ts +3 -0
  28. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.js +22 -0
  29. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.d.ts +7 -0
  30. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.js +24 -0
  31. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.d.ts +4 -0
  32. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.js +18 -0
  33. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.d.ts +4 -0
  34. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.js +16 -0
  35. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.d.ts +3 -0
  36. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.js +104 -0
  37. package/dist/elements/EFMedia/services/AudioElementFactory.d.ts +22 -0
  38. package/dist/elements/EFMedia/services/AudioElementFactory.js +72 -0
  39. package/dist/elements/EFMedia/services/MediaSourceService.browsertest.d.ts +1 -0
  40. package/dist/elements/EFMedia/services/MediaSourceService.d.ts +47 -0
  41. package/dist/elements/EFMedia/services/MediaSourceService.js +73 -0
  42. package/dist/elements/EFMedia/shared/AudioSpanUtils.d.ts +7 -0
  43. package/dist/elements/EFMedia/shared/AudioSpanUtils.js +54 -0
  44. package/dist/elements/EFMedia/shared/BufferUtils.d.ts +70 -0
  45. package/dist/elements/EFMedia/shared/BufferUtils.js +89 -0
  46. package/dist/elements/EFMedia/shared/MediaTaskUtils.d.ts +23 -0
  47. package/dist/elements/EFMedia/shared/RenditionHelpers.browsertest.d.ts +1 -0
  48. package/dist/elements/EFMedia/shared/RenditionHelpers.d.ts +19 -0
  49. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.d.ts +1 -0
  50. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.d.ts +18 -0
  51. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.js +60 -0
  52. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.test.d.ts +1 -0
  53. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.d.ts +9 -0
  54. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.d.ts +16 -0
  55. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.js +46 -0
  56. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.d.ts +9 -0
  57. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.d.ts +4 -0
  58. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.js +16 -0
  59. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.d.ts +9 -0
  60. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.d.ts +3 -0
  61. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.js +27 -0
  62. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.d.ts +7 -0
  63. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.js +25 -0
  64. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.d.ts +9 -0
  65. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.d.ts +4 -0
  66. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.js +18 -0
  67. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.d.ts +9 -0
  68. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.d.ts +4 -0
  69. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.js +16 -0
  70. package/dist/elements/EFMedia.browsertest.d.ts +1 -0
  71. package/dist/elements/EFMedia.d.ts +75 -111
  72. package/dist/elements/EFMedia.js +141 -1111
  73. package/dist/elements/EFTemporal.d.ts +1 -1
  74. package/dist/elements/EFTemporal.js +1 -1
  75. package/dist/elements/EFTimegroup.d.ts +11 -0
  76. package/dist/elements/EFTimegroup.js +88 -13
  77. package/dist/elements/EFVideo.d.ts +60 -29
  78. package/dist/elements/EFVideo.js +103 -203
  79. package/dist/elements/EFWaveform.js +2 -2
  80. package/dist/elements/SampleBuffer.d.ts +14 -0
  81. package/dist/elements/SampleBuffer.js +52 -0
  82. package/dist/getRenderInfo.d.ts +2 -2
  83. package/dist/getRenderInfo.js +2 -1
  84. package/dist/gui/ContextMixin.js +17 -70
  85. package/dist/gui/EFFilmstrip.d.ts +3 -3
  86. package/dist/gui/EFFilmstrip.js +1 -1
  87. package/dist/gui/EFFitScale.d.ts +2 -2
  88. package/dist/gui/TWMixin.js +1 -1
  89. package/dist/gui/services/ElementConnectionManager.browsertest.d.ts +1 -0
  90. package/dist/gui/services/ElementConnectionManager.d.ts +59 -0
  91. package/dist/gui/services/ElementConnectionManager.js +128 -0
  92. package/dist/gui/services/PlaybackController.browsertest.d.ts +1 -0
  93. package/dist/gui/services/PlaybackController.d.ts +103 -0
  94. package/dist/gui/services/PlaybackController.js +290 -0
  95. package/dist/services/MediaSourceManager.d.ts +62 -0
  96. package/dist/services/MediaSourceManager.js +211 -0
  97. package/dist/style.css +1 -1
  98. package/dist/transcoding/cache/CacheManager.d.ts +73 -0
  99. package/dist/transcoding/cache/RequestDeduplicator.d.ts +29 -0
  100. package/dist/transcoding/cache/RequestDeduplicator.js +53 -0
  101. package/dist/transcoding/cache/RequestDeduplicator.test.d.ts +1 -0
  102. package/dist/transcoding/types/index.d.ts +242 -0
  103. package/dist/transcoding/utils/MediaUtils.d.ts +9 -0
  104. package/dist/transcoding/utils/UrlGenerator.d.ts +26 -0
  105. package/dist/transcoding/utils/UrlGenerator.js +45 -0
  106. package/dist/transcoding/utils/constants.d.ts +27 -0
  107. package/dist/utils/LRUCache.d.ts +34 -0
  108. package/dist/utils/LRUCache.js +115 -0
  109. package/package.json +3 -2
  110. package/src/elements/EFAudio.browsertest.ts +183 -43
  111. package/src/elements/EFAudio.ts +59 -13
  112. package/src/elements/EFImage.browsertest.ts +42 -0
  113. package/src/elements/EFImage.ts +23 -3
  114. package/src/elements/EFMedia/AssetIdMediaEngine.test.ts +222 -0
  115. package/src/elements/EFMedia/AssetIdMediaEngine.ts +70 -0
  116. package/src/elements/EFMedia/AssetMediaEngine.ts +210 -0
  117. package/src/elements/EFMedia/BaseMediaEngine.test.ts +164 -0
  118. package/src/elements/EFMedia/BaseMediaEngine.ts +170 -0
  119. package/src/elements/EFMedia/BufferedSeekingInput.browsertest.ts +400 -0
  120. package/src/elements/EFMedia/BufferedSeekingInput.ts +267 -0
  121. package/src/elements/EFMedia/JitMediaEngine.browsertest.ts +165 -0
  122. package/src/elements/EFMedia/JitMediaEngine.ts +110 -0
  123. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.ts +554 -0
  124. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.ts +81 -0
  125. package/src/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.ts +241 -0
  126. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.ts +59 -0
  127. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.ts +23 -0
  128. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.ts +55 -0
  129. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.ts +35 -0
  130. package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.ts +42 -0
  131. package/src/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.ts +34 -0
  132. package/src/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.ts +23 -0
  133. package/src/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.ts +174 -0
  134. package/src/elements/EFMedia/services/AudioElementFactory.browsertest.ts +325 -0
  135. package/src/elements/EFMedia/services/AudioElementFactory.ts +119 -0
  136. package/src/elements/EFMedia/services/MediaSourceService.browsertest.ts +257 -0
  137. package/src/elements/EFMedia/services/MediaSourceService.ts +102 -0
  138. package/src/elements/EFMedia/shared/AudioSpanUtils.ts +128 -0
  139. package/src/elements/EFMedia/shared/BufferUtils.ts +310 -0
  140. package/src/elements/EFMedia/shared/MediaTaskUtils.ts +44 -0
  141. package/src/elements/EFMedia/shared/RenditionHelpers.browsertest.ts +247 -0
  142. package/src/elements/EFMedia/shared/RenditionHelpers.ts +79 -0
  143. package/src/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.ts +128 -0
  144. package/src/elements/EFMedia/tasks/makeMediaEngineTask.test.ts +233 -0
  145. package/src/elements/EFMedia/tasks/makeMediaEngineTask.ts +89 -0
  146. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.ts +555 -0
  147. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.ts +79 -0
  148. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.ts +59 -0
  149. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.ts +23 -0
  150. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.ts +55 -0
  151. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.ts +45 -0
  152. package/src/elements/EFMedia/videoTasks/makeVideoSeekTask.ts +44 -0
  153. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.ts +57 -0
  154. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.ts +32 -0
  155. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.ts +56 -0
  156. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.ts +23 -0
  157. package/src/elements/EFMedia.browsertest.ts +658 -265
  158. package/src/elements/EFMedia.ts +173 -1763
  159. package/src/elements/EFTemporal.ts +3 -4
  160. package/src/elements/EFTimegroup.browsertest.ts +6 -3
  161. package/src/elements/EFTimegroup.ts +152 -21
  162. package/src/elements/EFVideo.browsertest.ts +115 -37
  163. package/src/elements/EFVideo.ts +123 -452
  164. package/src/elements/EFWaveform.ts +1 -1
  165. package/src/elements/MediaController.ts +2 -12
  166. package/src/elements/SampleBuffer.ts +97 -0
  167. package/src/gui/ContextMixin.ts +23 -104
  168. package/src/gui/services/ElementConnectionManager.browsertest.ts +263 -0
  169. package/src/gui/services/ElementConnectionManager.ts +224 -0
  170. package/src/gui/services/PlaybackController.browsertest.ts +437 -0
  171. package/src/gui/services/PlaybackController.ts +521 -0
  172. package/src/services/MediaSourceManager.ts +333 -0
  173. package/src/transcoding/cache/CacheManager.ts +208 -0
  174. package/src/transcoding/cache/RequestDeduplicator.test.ts +170 -0
  175. package/src/transcoding/cache/RequestDeduplicator.ts +65 -0
  176. package/src/transcoding/types/index.ts +265 -0
  177. package/src/transcoding/utils/MediaUtils.ts +63 -0
  178. package/src/transcoding/utils/UrlGenerator.ts +68 -0
  179. package/src/transcoding/utils/constants.ts +36 -0
  180. package/src/utils/LRUCache.ts +153 -0
  181. package/test/EFVideo.framegen.browsertest.ts +38 -29
  182. package/test/__cache__/GET__api_v1_transcode_audio_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__32da3954ba60c96ad732020c65a08ebc/data.bin +0 -0
  183. package/test/__cache__/GET__api_v1_transcode_audio_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__32da3954ba60c96ad732020c65a08ebc/metadata.json +21 -0
  184. package/test/__cache__/GET__api_v1_transcode_audio_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__b0b2b07efcf607de8ee0f650328c32f7/data.bin +0 -0
  185. package/test/__cache__/GET__api_v1_transcode_audio_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__b0b2b07efcf607de8ee0f650328c32f7/metadata.json +21 -0
  186. package/test/__cache__/GET__api_v1_transcode_audio_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a75c2252b542e0c152c780e9a8d7b154/data.bin +0 -0
  187. package/test/__cache__/GET__api_v1_transcode_audio_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a75c2252b542e0c152c780e9a8d7b154/metadata.json +21 -0
  188. package/test/__cache__/GET__api_v1_transcode_audio_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a64ff1cfb1b52cae14df4b5dfa1e222b/data.bin +0 -0
  189. package/test/__cache__/GET__api_v1_transcode_audio_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a64ff1cfb1b52cae14df4b5dfa1e222b/metadata.json +21 -0
  190. package/test/__cache__/GET__api_v1_transcode_audio_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__91e8a522f950809b9f09f4173113b4b0/data.bin +0 -0
  191. package/test/__cache__/GET__api_v1_transcode_audio_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__91e8a522f950809b9f09f4173113b4b0/metadata.json +21 -0
  192. package/test/__cache__/GET__api_v1_transcode_audio_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__e66d2c831d951e74ad0aeaa6489795d0/data.bin +0 -0
  193. package/test/__cache__/GET__api_v1_transcode_audio_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__e66d2c831d951e74ad0aeaa6489795d0/metadata.json +21 -0
  194. package/test/__cache__/GET__api_v1_transcode_high_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__26197f6f7c46cacb0a71134131c3f775/data.bin +0 -0
  195. package/test/__cache__/GET__api_v1_transcode_high_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__26197f6f7c46cacb0a71134131c3f775/metadata.json +21 -0
  196. package/test/__cache__/GET__api_v1_transcode_high_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__4cb6774cd3650ccf59c8f8dc6678c0b9/data.bin +0 -0
  197. package/test/__cache__/GET__api_v1_transcode_high_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__4cb6774cd3650ccf59c8f8dc6678c0b9/metadata.json +21 -0
  198. package/test/__cache__/GET__api_v1_transcode_high_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0b3b2b1c8933f7fcf8a9ecaa88d58b41/data.bin +0 -0
  199. package/test/__cache__/GET__api_v1_transcode_high_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0b3b2b1c8933f7fcf8a9ecaa88d58b41/metadata.json +21 -0
  200. package/test/__cache__/GET__api_v1_transcode_high_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0798c479b44aaeef850609a430f6e613/data.bin +0 -0
  201. package/test/__cache__/GET__api_v1_transcode_high_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0798c479b44aaeef850609a430f6e613/metadata.json +21 -0
  202. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/data.bin +1 -0
  203. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/metadata.json +19 -0
  204. package/test/createJitTestClips.ts +320 -188
  205. package/test/recordReplayProxyPlugin.js +302 -0
  206. package/test/useAssetMSW.ts +1 -1
  207. package/test/useMSW.ts +35 -22
  208. package/types.json +1 -1
  209. package/dist/JitTranscodingClient.d.ts +0 -167
  210. package/dist/JitTranscodingClient.js +0 -373
  211. package/dist/ScrubTrackManager.js +0 -216
  212. package/dist/elements/printTaskStatus.js +0 -11
  213. package/src/elements/__screenshots__/EFMedia.browsertest.ts/EFMedia-JIT-audio-playback-audioBufferTask-should-work-in-JIT-mode-without-URL-errors-1.png +0 -0
  214. package/test/EFVideo.frame-tasks.browsertest.ts +0 -524
  215. /package/dist/{JitTranscodingClient.browsertest.d.ts → elements/EFMedia/AssetIdMediaEngine.test.d.ts} +0 -0
  216. /package/dist/{JitTranscodingClient.test.d.ts → elements/EFMedia/BaseMediaEngine.test.d.ts} +0 -0
  217. /package/dist/{ScrubTrackIntegration.test.d.ts → elements/EFMedia/BufferedSeekingInput.browsertest.d.ts} +0 -0
  218. /package/dist/{SegmentSwitchLoading.test.d.ts → elements/EFMedia/services/AudioElementFactory.browsertest.d.ts} +0 -0
@@ -1,17 +1,29 @@
1
1
  import { EFMedia } from "./EFMedia.js";
2
- import { DelayedLoadingState } from "../DelayedLoadingState.js";
3
2
  import { TWMixin } from "../gui/TWMixin2.js";
4
- import { ScrubTrackManager } from "../ScrubTrackManager.js";
5
- import { printTaskStatus } from "./printTaskStatus.js";
3
+ import { DelayedLoadingState } from "../DelayedLoadingState.js";
4
+ import { makeVideoBufferTask } from "./EFMedia/videoTasks/makeVideoBufferTask.js";
5
+ import { makeVideoInitSegmentFetchTask } from "./EFMedia/videoTasks/makeVideoInitSegmentFetchTask.js";
6
+ import { makeVideoInputTask } from "./EFMedia/videoTasks/makeVideoInputTask.js";
7
+ import { makeVideoSeekTask } from "./EFMedia/videoTasks/makeVideoSeekTask.js";
8
+ import { makeVideoSegmentFetchTask } from "./EFMedia/videoTasks/makeVideoSegmentFetchTask.js";
9
+ import { makeVideoSegmentIdTask } from "./EFMedia/videoTasks/makeVideoSegmentIdTask.js";
6
10
  import { Task } from "@lit/task";
7
11
  import debug from "debug";
8
12
  import { css, html } from "lit";
9
- import { customElement, state } from "lit/decorators.js";
13
+ import { customElement, property, state } from "lit/decorators.js";
10
14
  import _decorate from "@oxc-project/runtime/helpers/decorate";
11
- import { VideoAsset } from "@editframe/assets/EncodedAsset.js";
12
15
  import { createRef, ref } from "lit/directives/ref.js";
13
16
  const log = debug("ef:elements:EFVideo");
14
17
  let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
18
+ static get observedAttributes() {
19
+ const parentAttributes = EFMedia.observedAttributes || [];
20
+ return [
21
+ ...parentAttributes,
22
+ "video-buffer-duration",
23
+ "max-video-buffer-fetches",
24
+ "enable-video-buffering"
25
+ ];
26
+ }
15
27
  static {
16
28
  this.styles = [css`
17
29
  :host {
@@ -76,7 +88,15 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
76
88
  constructor() {
77
89
  super();
78
90
  this.canvasRef = createRef();
79
- this.lastSeekTimeMs = 0;
91
+ this.videoBufferDurationMs = 6e4;
92
+ this.maxVideoBufferFetches = 2;
93
+ this.enableVideoBuffering = true;
94
+ this.videoSegmentIdTask = makeVideoSegmentIdTask(this);
95
+ this.videoInitSegmentFetchTask = makeVideoInitSegmentFetchTask(this);
96
+ this.videoSegmentFetchTask = makeVideoSegmentFetchTask(this);
97
+ this.videoInputTask = makeVideoInputTask(this);
98
+ this.videoSeekTask = makeVideoSeekTask(this);
99
+ this.videoBufferTask = makeVideoBufferTask(this);
80
100
  this.loadingState = {
81
101
  isLoading: false,
82
102
  operation: null,
@@ -87,71 +107,28 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
87
107
  onError: (error) => {
88
108
  console.error("frameTask error", error);
89
109
  },
110
+ onComplete: () => {},
90
111
  task: async ([_desiredSeekTimeMs], { signal }) => {
91
- await this.seekTask.taskComplete;
92
- if (signal.aborted) return;
93
- await this.fragmentIndexTask.taskComplete;
94
- if (signal.aborted) return;
95
- await this.mediaSegmentsTask.taskComplete;
96
- if (signal.aborted) return;
97
- await this.videoAssetTask.taskComplete;
98
- if (signal.aborted) return;
112
+ await this.videoSeekTask.taskComplete;
99
113
  await this.paintTask.taskComplete;
100
114
  if (signal.aborted) return;
101
115
  }
102
116
  });
103
- this.videoAssetTask = new Task(this, {
104
- autoRun: true,
105
- args: () => [this.effectiveMode, this.mediaSegmentsTask.value],
106
- onError: (error) => {
107
- console.error("videoAsset task error", error);
108
- },
109
- task: async ([mode, _files], { signal: _signal }) => {
110
- await this.mediaSegmentsTask.taskComplete;
111
- if (_signal.aborted) return void 0;
112
- await this.fragmentIndexTask.taskComplete;
113
- if (_signal.aborted) return void 0;
114
- const files = this.mediaSegmentsTask.value;
115
- const fragmentIndex = this.fragmentIndexTask.value;
116
- if (!files) {
117
- log("trace: videoAsset task aborted - no files");
118
- throw new Error(`Video asset creation failed: No media segment files available. This indicates a problem with media segment loading for source: "${this.src}"`);
119
- }
120
- const computedVideoTrackId = Object.values(fragmentIndex ?? {}).find((track) => track.type === "video")?.track;
121
- if (computedVideoTrackId === void 0) {
122
- log("trace: videoAsset task aborted - no video track");
123
- throw new Error(`Video asset creation failed: No video track found in media segments. Source may not contain video content: "${this.src}"`);
124
- }
125
- const videoFile = files[computedVideoTrackId];
126
- if (!videoFile) {
127
- log("trace: videoAsset task aborted - no video file");
128
- throw new Error(`Video asset creation failed: Video file not available for track ${computedVideoTrackId}. Media segment loading may have failed for source: "${this.src}"`);
129
- }
130
- const existingAsset = this.videoAssetTask.value;
131
- if (existingAsset) {
132
- for (const frame of existingAsset?.decodedFrames || []) frame.close();
133
- const decoder = existingAsset?.videoDecoder;
134
- if (decoder && decoder.state !== "closed") decoder.close();
135
- }
136
- if (_signal.aborted) return void 0;
137
- log("trace: creating video asset", { mode });
138
- const videoTrackFragmentIndex = Object.values(fragmentIndex ?? {}).find((track) => track.type === "video");
139
- const startTimeOffsetMs = Number((videoTrackFragmentIndex?.startTimeOffsetMs ?? 0).toFixed(5));
140
- if (mode === "jit-transcode") {
141
- const result$1 = await VideoAsset.createFromCompleteMP4(`jit-segment-${computedVideoTrackId}`, videoFile, { startTimeOffsetMs });
142
- return result$1;
143
- }
144
- const result = await VideoAsset.createFromReadableStream("video.mp4", videoFile.stream(), videoFile, { startTimeOffsetMs });
145
- return result;
146
- }
147
- });
148
117
  this.paintTask = new Task(this, {
149
118
  args: () => [this.desiredSeekTimeMs],
150
119
  onError: (error) => {
151
120
  console.error("paintTask error", error);
152
121
  },
122
+ onComplete: () => {},
153
123
  task: async ([_seekToMs], { signal }) => {
124
+ await this.videoSeekTask.taskComplete;
154
125
  const isProductionRendering = this.isInProductionRenderingMode();
126
+ const sample = this.videoSeekTask.value;
127
+ if (sample) {
128
+ const videoFrame = sample.toVideoFrame();
129
+ this.displayFrame(videoFrame, _seekToMs);
130
+ videoFrame.close();
131
+ }
155
132
  if (!isProductionRendering) {
156
133
  if (!this.rootTimegroup || this.rootTimegroup.currentTimeMs === 0 && this.desiredSeekTimeMs === 0) return;
157
134
  } else {
@@ -159,83 +136,6 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
159
136
  if (!this.isFrameRenderingActive()) return;
160
137
  }
161
138
  if (signal.aborted) return;
162
- await this.mediaSegmentsTask.taskComplete;
163
- if (signal.aborted) return;
164
- await this.videoAssetTask.taskComplete;
165
- if (signal.aborted) return;
166
- const videoAsset = this.videoAssetTask.value;
167
- const currentSeekToMs = this.desiredSeekTimeMs;
168
- if (!videoAsset) {
169
- log("trace: paintTask aborted - no video asset");
170
- throw new Error(`Frame rendering failed: No video asset available. This may indicate a problem with video loading or an invalid source: "${this.src}"`);
171
- }
172
- if (this.#decoderNeedsReset) try {
173
- if (videoAsset?.videoDecoder) videoAsset.configureDecoder();
174
- else console.warn("No video decoder available for reset");
175
- this.#decoderNeedsReset = false;
176
- } catch (resetError) {
177
- console.error("reset error", resetError);
178
- throw new Error(`Frame rendering failed: Unable to reset video decoder after previous error. Decoder state: ${resetError instanceof Error ? resetError.message : "Unknown error"}. Try refreshing the page or reloading the video.`);
179
- }
180
- if (signal.aborted) return;
181
- if (this.#decoderLock) return;
182
- try {
183
- this.#decoderLock = true;
184
- const currentVideoAsset = this.videoAssetTask.value;
185
- if (videoAsset !== currentVideoAsset) return;
186
- const decoderState = videoAsset?.videoDecoder?.state;
187
- if (decoderState === "closed") return;
188
- if (this.effectiveMode === "jit-transcode" && this.scrubTrackManager) {
189
- const shouldUseScrub = this.scrubTrackManager.shouldUseScrubTrack(currentSeekToMs);
190
- const isFastSeeking = this.scrubTrackManager.isFastSeeking(this.lastSeekTimeMs, currentSeekToMs);
191
- if (shouldUseScrub || isFastSeeking) try {
192
- this.startDelayedLoading("scrub-segment-load", "Loading scrub segment...");
193
- const scrubFrame = await this.scrubTrackManager.getScrubFrame(currentSeekToMs);
194
- if (scrubFrame && this.canvasElement) {
195
- this.scrubTrackManager.recordCacheMiss();
196
- this.lastSeekTimeMs = currentSeekToMs;
197
- this.clearDelayedLoading("scrub-segment-load");
198
- return this.displayFrame(scrubFrame, currentSeekToMs);
199
- }
200
- console.warn("Scrub track returned null frame, falling back to normal video");
201
- this.clearDelayedLoading("scrub-segment-load");
202
- this.startDelayedLoading("video-segment-fallback", "Loading high quality video...");
203
- } catch (error) {
204
- this.clearDelayedLoading("scrub-segment-load");
205
- console.warn("Scrub track failed, falling back to normal video:", error);
206
- this.startDelayedLoading("video-segment-fallback", "Loading high quality video...");
207
- }
208
- else this.scrubTrackManager?.recordCacheHit();
209
- }
210
- const shouldShowLoading = !this.delayedLoadingState.isLoading && (this.effectiveMode !== "asset" || !videoAsset);
211
- if (shouldShowLoading) this.startDelayedLoading("video-segment", "Loading video segment...");
212
- this.lastSeekTimeMs = currentSeekToMs;
213
- const result = await this.renderNormalVideo(videoAsset, currentSeekToMs);
214
- this.clearDelayedLoading("video-segment");
215
- this.clearDelayedLoading("video-segment-fallback");
216
- return result;
217
- } catch (error) {
218
- this.clearDelayedLoading("scrub-segment-load");
219
- this.clearDelayedLoading("video-segment");
220
- this.clearDelayedLoading("video-segment-fallback");
221
- if (error instanceof Error) {
222
- if (error.name === "DataError" && error.message.includes("key frame is required")) {
223
- console.warn("Decoder reset during VideoAsset due to key frame requirement");
224
- this.#decoderNeedsReset = true;
225
- if (this.effectiveMode === "jit-transcode") this.requestUpdate();
226
- throw error;
227
- }
228
- if (error.name === "AbortError") throw new Error("Frame rendering cancelled: Operation was aborted, likely due to a new seek request or component unmounting.");
229
- if (error.message.includes("VideoAsset decoder closed") || error.message.includes("recreation in progress")) return;
230
- if (error.name === "InvalidStateError" && error.message.includes("closed codec")) return;
231
- console.warn("Decoder reset during VideoAsset recreation", error);
232
- this.#decoderNeedsReset = true;
233
- throw error;
234
- }
235
- throw new Error(`Frame rendering failed: Unknown error during video rendering at ${currentSeekToMs}ms. Error: ${String(error)}`);
236
- } finally {
237
- this.#decoderLock = false;
238
- }
239
139
  }
240
140
  });
241
141
  this.delayedLoadingState = new DelayedLoadingState(250, (isLoading, message) => {
@@ -261,46 +161,8 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
261
161
  get canvasElement() {
262
162
  return this.canvasRef.value;
263
163
  }
264
- #decoderLock = false;
265
- #decoderNeedsReset = false;
266
- get frameTaskStatus() {
267
- return {
268
- desiredSeekTimeMs: this.desiredSeekTimeMs,
269
- fragmentIndexTask: printTaskStatus(this.fragmentIndexTask.status),
270
- seekTask: printTaskStatus(this.seekTask.status),
271
- mediaSegmentsTask: printTaskStatus(this.mediaSegmentsTask.status),
272
- assetSegmentLoader: printTaskStatus(this.assetSegmentLoader.status),
273
- assetSegmentKeysTask: printTaskStatus(this.assetSegmentKeysTask.status),
274
- assetInitSegmentsTask: printTaskStatus(this.assetInitSegmentsTask.status),
275
- videoAssetTask: printTaskStatus(this.videoAssetTask.status),
276
- paintTask: printTaskStatus(this.paintTask.status),
277
- frameTask: printTaskStatus(this.frameTask.status)
278
- };
279
- }
280
- #lastVideoAsset = null;
281
164
  updated(changedProperties) {
282
165
  super.updated(changedProperties);
283
- const currentVideoAsset = this.videoAssetTask.value;
284
- if (currentVideoAsset !== this.#lastVideoAsset) this.#lastVideoAsset = currentVideoAsset;
285
- this.initializeScrubTrackManager();
286
- }
287
- /**
288
- * Initialize scrub track manager if needed
289
- */
290
- async initializeScrubTrackManager() {
291
- const mode = this.effectiveMode;
292
- if (mode === "jit-transcode" && this.src && !this.scrubTrackManager) {
293
- const jitClient = this.jitClientTask.value;
294
- if (jitClient) try {
295
- this.scrubTrackManager = new ScrubTrackManager(this.src, jitClient, { onLoadingStateChange: (isLoading, message) => {
296
- if (isLoading) this.startDelayedLoading("scrub-segment", message || "Loading scrub track...");
297
- else this.clearDelayedLoading("scrub-segment");
298
- } });
299
- await this.scrubTrackManager.initialize();
300
- } catch (error) {
301
- console.warn("Failed to initialize scrub track manager:", error);
302
- }
303
- }
304
166
  }
305
167
  /**
306
168
  * Start a delayed loading operation for testing
@@ -325,34 +187,6 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
325
187
  };
326
188
  }
327
189
  /**
328
- * Render normal video using existing logic
329
- */
330
- async renderNormalVideo(videoAsset, seekToMs) {
331
- let targetSeekTimeSeconds = seekToMs / 1e3;
332
- try {
333
- const currentVideoAsset = this.videoAssetTask.value;
334
- if (videoAsset !== currentVideoAsset) throw new Error("VideoAsset decoder closed during seek - recreation in progress");
335
- const decoderState = videoAsset?.videoDecoder?.state;
336
- if (decoderState === "closed") throw new Error("VideoAsset decoder closed during seek - recreation in progress");
337
- if (this.effectiveMode === "jit-transcode") targetSeekTimeSeconds %= 2;
338
- const frame = await videoAsset.seekToTime(targetSeekTimeSeconds);
339
- if (frame) {
340
- const finalVideoAsset = this.videoAssetTask.value;
341
- if (videoAsset !== finalVideoAsset) {
342
- frame.close();
343
- throw new Error("VideoAsset decoder closed during seek - recreation in progress");
344
- }
345
- const finalSeekToMs = this.desiredSeekTimeMs;
346
- return this.displayFrame(frame, finalSeekToMs);
347
- }
348
- log("trace: no frame returned from seekToTime");
349
- throw new Error(`Frame rendering failed: No frame available at time ${seekToMs}ms (${targetSeekTimeSeconds}s). This may indicate seeking beyond video duration, corrupted video data, or an incompatible video format.`);
350
- } catch (error) {
351
- if (error instanceof Error && (error.message.includes("VideoAsset decoder closed") || error.message.includes("recreation in progress"))) throw error;
352
- throw error;
353
- }
354
- }
355
- /**
356
190
  * Display a video frame on the canvas
357
191
  */
358
192
  displayFrame(frame, seekToMs) {
@@ -383,7 +217,7 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
383
217
  log("trace: displayFrame aborted - null frame format");
384
218
  throw new Error(`Frame display failed: Video frame has null format at time ${seekToMs}ms. This indicates corrupted or incompatible video data.`);
385
219
  }
386
- log("trace: drawing frame to canvas");
220
+ log("trace: drawing frame to canvas", frame.timestamp / 1e3);
387
221
  ctx.drawImage(frame, 0, 0, this.canvasElement.width, this.canvasElement.height);
388
222
  log("trace: frame drawn to canvas", { seekToMs });
389
223
  return seekToMs;
@@ -415,6 +249,60 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
415
249
  return this.scrubTrackManager?.getCacheStats() || null;
416
250
  }
417
251
  /**
252
+ * Effective mode - always returns "asset" for EFVideo
253
+ */
254
+ get effectiveMode() {
255
+ return "asset";
256
+ }
257
+ /**
258
+ * Legacy getter for asset index loader (maps to mediaEngine task)
259
+ */
260
+ get assetIndexLoader() {
261
+ return this.mediaEngineTask;
262
+ }
263
+ /**
264
+ * Legacy getter for fragment index task (maps to videoSegmentIdTask)
265
+ */
266
+ get fragmentIndexTask() {
267
+ return this.videoSegmentIdTask;
268
+ }
269
+ /**
270
+ * Legacy getter for seek task (maps to videoSeekTask)
271
+ */
272
+ get seekTask() {
273
+ return this.videoSeekTask;
274
+ }
275
+ /**
276
+ * Legacy getter for media segments task (maps to videoSegmentFetchTask)
277
+ */
278
+ get mediaSegmentsTask() {
279
+ return this.videoSegmentFetchTask;
280
+ }
281
+ /**
282
+ * Legacy getter for asset segment keys task (maps to videoSegmentIdTask)
283
+ */
284
+ get assetSegmentKeysTask() {
285
+ return this.videoSegmentIdTask;
286
+ }
287
+ /**
288
+ * Legacy getter for asset init segments task (maps to videoInitSegmentFetchTask)
289
+ */
290
+ get assetInitSegmentsTask() {
291
+ return this.videoInitSegmentFetchTask;
292
+ }
293
+ /**
294
+ * Legacy getter for asset segment loader (maps to videoSegmentFetchTask)
295
+ */
296
+ get assetSegmentLoader() {
297
+ return this.videoSegmentFetchTask;
298
+ }
299
+ /**
300
+ * Legacy getter for video asset task (maps to videoBufferTask)
301
+ */
302
+ get videoAssetTask() {
303
+ return this.videoBufferTask;
304
+ }
305
+ /**
418
306
  * Clean up resources when component is disconnected
419
307
  */
420
308
  disconnectedCallback() {
@@ -423,6 +311,18 @@ let EFVideo = class EFVideo$1 extends TWMixin(EFMedia) {
423
311
  this.delayedLoadingState.clearAllLoading();
424
312
  }
425
313
  };
314
+ _decorate([property({
315
+ type: Number,
316
+ attribute: "video-buffer-duration"
317
+ })], EFVideo.prototype, "videoBufferDurationMs", void 0);
318
+ _decorate([property({
319
+ type: Number,
320
+ attribute: "max-video-buffer-fetches"
321
+ })], EFVideo.prototype, "maxVideoBufferFetches", void 0);
322
+ _decorate([property({
323
+ type: Boolean,
324
+ attribute: "enable-video-buffering"
325
+ })], EFVideo.prototype, "enableVideoBuffering", void 0);
426
326
  _decorate([state()], EFVideo.prototype, "loadingState", void 0);
427
327
  EFVideo = _decorate([customElement("ef-video")], EFVideo);
428
328
  export { EFVideo };
@@ -1,9 +1,9 @@
1
1
  import { EF_INTERACTIVE } from "../EF_INTERACTIVE.js";
2
+ import { EF_RENDERING } from "../EF_RENDERING.js";
2
3
  import { EFTemporal } from "./EFTemporal.js";
3
4
  import { TargetController } from "./TargetController.js";
4
5
  import { TWMixin } from "../gui/TWMixin2.js";
5
6
  import { CrossUpdateController } from "./CrossUpdateController.js";
6
- import { EF_RENDERING } from "../EF_RENDERING.js";
7
7
  import { Task } from "@lit/task";
8
8
  import { LitElement, css, html } from "lit";
9
9
  import { customElement, property, state } from "lit/decorators.js";
@@ -80,8 +80,8 @@ let EFWaveform = class EFWaveform$1 extends EFTemporal(TWMixin(LitElement)) {
80
80
  static {
81
81
  this.styles = css`
82
82
  :host {
83
- display: block;
84
83
  all: inherit;
84
+ display: block;
85
85
  position: relative;
86
86
  }
87
87
 
@@ -0,0 +1,14 @@
1
+ import { AudioSample, VideoSample } from 'mediabunny';
2
+ export type MediaSample = VideoSample | AudioSample;
3
+ export declare class SampleBuffer {
4
+ private buffer;
5
+ private bufferSize;
6
+ constructor(bufferSize?: number);
7
+ push(sample: MediaSample): void;
8
+ clear(): void;
9
+ peek(): MediaSample | undefined;
10
+ find(desiredSeekTimeMs: number): MediaSample | undefined;
11
+ get length(): number;
12
+ get firstTimestamp(): number;
13
+ getContents(): MediaSample[];
14
+ }
@@ -0,0 +1,52 @@
1
+ var SampleBuffer = class {
2
+ constructor(bufferSize = 10) {
3
+ this.buffer = [];
4
+ this.bufferSize = bufferSize;
5
+ }
6
+ push(sample) {
7
+ const currentBuffer = [...this.buffer];
8
+ currentBuffer.push(sample);
9
+ if (currentBuffer.length > this.bufferSize) {
10
+ const shifted = currentBuffer.shift();
11
+ if (shifted) try {
12
+ shifted.close();
13
+ } catch (_error) {}
14
+ }
15
+ this.buffer = currentBuffer;
16
+ }
17
+ clear() {
18
+ const toClose = this.buffer;
19
+ this.buffer = [];
20
+ for (const sample of toClose) try {
21
+ sample.close();
22
+ } catch (_error) {}
23
+ }
24
+ peek() {
25
+ const currentBuffer = this.buffer;
26
+ return currentBuffer[0];
27
+ }
28
+ find(desiredSeekTimeMs) {
29
+ const currentBuffer = [...this.buffer];
30
+ if (currentBuffer.length === 0) return void 0;
31
+ const roundToMicroseconds = (timeMs) => Math.round(timeMs * 1e3) / 1e3;
32
+ const targetTimeMs = roundToMicroseconds(desiredSeekTimeMs);
33
+ for (const sample of currentBuffer) {
34
+ const sampleStartMs = roundToMicroseconds((sample.timestamp || 0) * 1e3);
35
+ const sampleDurationMs = roundToMicroseconds((sample.duration || 0) * 1e3);
36
+ const sampleEndMs = sampleStartMs + sampleDurationMs;
37
+ if (targetTimeMs >= sampleStartMs && targetTimeMs < sampleEndMs) return sample;
38
+ }
39
+ return void 0;
40
+ }
41
+ get length() {
42
+ return this.buffer.length;
43
+ }
44
+ get firstTimestamp() {
45
+ const currentBuffer = this.buffer;
46
+ return currentBuffer[0]?.timestamp || 0;
47
+ }
48
+ getContents() {
49
+ return [...this.buffer];
50
+ }
51
+ };
52
+ export { SampleBuffer };
@@ -18,20 +18,20 @@ export declare const RenderInfo: z.ZodObject<{
18
18
  efImage: string[];
19
19
  }>;
20
20
  }, "strip", z.ZodTypeAny, {
21
+ fps: number;
21
22
  width: number;
22
23
  height: number;
23
24
  durationMs: number;
24
- fps: number;
25
25
  assets: {
26
26
  efMedia: Record<string, any>;
27
27
  efCaptions: string[];
28
28
  efImage: string[];
29
29
  };
30
30
  }, {
31
+ fps: number;
31
32
  width: number;
32
33
  height: number;
33
34
  durationMs: number;
34
- fps: number;
35
35
  assets: {
36
36
  efMedia: Record<string, any>;
37
37
  efCaptions: string[];
@@ -30,7 +30,8 @@ const getRenderInfo = async () => {
30
30
  case "EF-VIDEO": {
31
31
  const src = element.src;
32
32
  console.error("Processing element", element.tagName, src);
33
- assets.efMedia[src] = element.fragmentIndexTask.value;
33
+ const mediaEngine = element.mediaEngineTask.value;
34
+ if (mediaEngine && "data" in mediaEngine) assets.efMedia[src] = mediaEngine.data;
34
35
  break;
35
36
  }
36
37
  case "EF-IMAGE": {
@@ -4,6 +4,8 @@ import { fetchContext } from "./fetchContext.js";
4
4
  import { focusContext } from "./focusContext.js";
5
5
  import { focusedElementContext } from "./focusedElementContext.js";
6
6
  import { loopContext, playingContext } from "./playingContext.js";
7
+ import { ElementConnectionManager } from "./services/ElementConnectionManager.js";
8
+ import { PlaybackController } from "./services/PlaybackController.js";
7
9
  import { consume, createContext, provide } from "@lit/context";
8
10
  import { property, state } from "lit/decorators.js";
9
11
  import _decorate from "@oxc-project/runtime/helpers/decorate";
@@ -40,6 +42,19 @@ function ContextMixin(superClass) {
40
42
  this.loop = false;
41
43
  this.rendering = false;
42
44
  this.currentTimeMs = 0;
45
+ this._elementConnectionManager = new ElementConnectionManager(3e3);
46
+ this.playbackController = new PlaybackController({
47
+ fps: 30,
48
+ onTimeUpdate: (timeMs) => {
49
+ this.currentTimeMs = timeMs;
50
+ },
51
+ onPlayStateChange: (playing) => {
52
+ this.playing = playing;
53
+ },
54
+ onError: (error) => {
55
+ console.error("🎵 [PLAYBACK_ERROR]:", error);
56
+ }
57
+ });
43
58
  }
44
59
  static {
45
60
  this[contextMixinSymbol] = true;
@@ -63,8 +78,6 @@ function ContextMixin(superClass) {
63
78
  set signingURL(value) {
64
79
  this.#signingURL = value;
65
80
  }
66
- #FPS = 30;
67
- #MS_PER_FRAME = 1e3 / this.#FPS;
68
81
  #timegroupObserver = new MutationObserver((mutations) => {
69
82
  for (const mutation of mutations) if (mutation.type === "childList") {
70
83
  const newTimegroup = this.querySelector("ef-timegroup");
@@ -106,79 +119,13 @@ function ContextMixin(superClass) {
106
119
  pause() {
107
120
  this.playing = false;
108
121
  }
109
- #playbackAudioContext = null;
110
- #playbackAnimationFrameRequest = null;
111
- #AUDIO_PLAYBACK_SLICE_MS = 1e3;
112
- #syncPlayheadToAudioContext(target, startMs) {
113
- const rawTimeMs = startMs + (this.#playbackAudioContext?.currentTime ?? 0) * 1e3;
114
- const nextTimeMs = Math.round(rawTimeMs / this.#MS_PER_FRAME) * this.#MS_PER_FRAME;
115
- if (nextTimeMs !== this.currentTimeMs) this.currentTimeMs = nextTimeMs;
116
- this.#playbackAnimationFrameRequest = requestAnimationFrame(() => {
117
- this.#syncPlayheadToAudioContext(target, startMs);
118
- });
119
- }
120
122
  async stopPlayback() {
121
- if (this.#playbackAudioContext) {
122
- if (this.#playbackAudioContext.state !== "closed") await this.#playbackAudioContext.close();
123
- }
124
- if (this.#playbackAnimationFrameRequest) cancelAnimationFrame(this.#playbackAnimationFrameRequest);
125
- this.#playbackAudioContext = null;
123
+ await this.playbackController.stopPlayback();
126
124
  }
127
125
  async startPlayback() {
128
- await this.stopPlayback();
129
126
  const timegroup = this.targetTimegroup;
130
127
  if (!timegroup) return;
131
- await timegroup.waitForMediaDurations();
132
- let currentMs = timegroup.currentTimeMs;
133
- const fromMs = currentMs;
134
- const toMs = timegroup.endTimeMs;
135
- if (fromMs >= toMs) {
136
- this.pause();
137
- return;
138
- }
139
- let bufferCount = 0;
140
- this.#playbackAudioContext = new AudioContext({ latencyHint: "playback" });
141
- if (this.#playbackAnimationFrameRequest) cancelAnimationFrame(this.#playbackAnimationFrameRequest);
142
- this.#syncPlayheadToAudioContext(timegroup, currentMs);
143
- const playbackContext = this.#playbackAudioContext;
144
- if (playbackContext.state === "suspended") {
145
- console.warn("AudioContext is suspended, media playback will not work until user has interacted with page.");
146
- this.playing = false;
147
- return;
148
- }
149
- await playbackContext.suspend();
150
- const fillBuffer = async () => {
151
- if (bufferCount > 1) return;
152
- const canFillBuffer = await queueBufferSource();
153
- if (canFillBuffer) fillBuffer();
154
- };
155
- const queueBufferSource = async () => {
156
- if (currentMs >= toMs) return false;
157
- const startMs = currentMs;
158
- const endMs = currentMs + this.#AUDIO_PLAYBACK_SLICE_MS;
159
- currentMs += this.#AUDIO_PLAYBACK_SLICE_MS;
160
- const audioBuffer = await timegroup.renderAudio(startMs, endMs);
161
- bufferCount++;
162
- const source = playbackContext.createBufferSource();
163
- source.buffer = audioBuffer;
164
- source.connect(playbackContext.destination);
165
- source.start((startMs - fromMs) / 1e3);
166
- source.onended = () => {
167
- bufferCount--;
168
- if (endMs >= toMs) {
169
- this.pause();
170
- if (this.loop) this.updateComplete.then(() => {
171
- this.currentTimeMs = 0;
172
- this.updateComplete.then(() => {
173
- this.play();
174
- });
175
- });
176
- } else fillBuffer();
177
- };
178
- return true;
179
- };
180
- await fillBuffer();
181
- await playbackContext.resume();
128
+ await this.playbackController.startPlayback(timegroup, timegroup.currentTimeMs);
182
129
  }
183
130
  }
184
131
  _decorate([consume({
@@ -121,9 +121,9 @@ export declare class EFFilmstrip extends EFFilmstrip_base {
121
121
  scrub(e: MouseEvent): void;
122
122
  startScrub(e: MouseEvent): void;
123
123
  scrollScrub(e: WheelEvent): void;
124
- gutterRef: import('lit-html/directives/ref.js').Ref<HTMLDivElement>;
125
- hierarchyRef: import('lit-html/directives/ref.js').Ref<HTMLDivElement>;
126
- playheadRef: import('lit-html/directives/ref.js').Ref<HTMLDivElement>;
124
+ gutterRef: import('lit-html/directives/ref').Ref<HTMLDivElement>;
125
+ hierarchyRef: import('lit-html/directives/ref').Ref<HTMLDivElement>;
126
+ playheadRef: import('lit-html/directives/ref').Ref<HTMLDivElement>;
127
127
  get gutter(): HTMLDivElement | undefined;
128
128
  render(): TemplateResult<1>;
129
129
  updated(changes: PropertyValueMap<any> | Map<PropertyKey, unknown>): void;
@@ -5,8 +5,8 @@ import { targetTimegroupContext } from "./ContextMixin.js";
5
5
  import { TimegroupController } from "../elements/TimegroupController.js";
6
6
  import { EFTimegroup } from "../elements/EFTimegroup.js";
7
7
  import { EFImage } from "../elements/EFImage.js";
8
- import { EFAudio } from "../elements/EFAudio.js";
9
8
  import { TWMixin } from "./TWMixin2.js";
9
+ import { EFAudio } from "../elements/EFAudio.js";
10
10
  import { EFVideo } from "../elements/EFVideo.js";
11
11
  import { EFCaptions, EFCaptionsActiveWord } from "../elements/EFCaptions.js";
12
12
  import { EFWaveform } from "../elements/EFWaveform.js";
@@ -1,7 +1,7 @@
1
1
  import { LitElement } from 'lit';
2
2
  export declare class EFFitScale extends LitElement {
3
- containerRef: import('lit-html/directives/ref.js').Ref<HTMLDivElement>;
4
- contentRef: import('lit-html/directives/ref.js').Ref<HTMLSlotElement>;
3
+ containerRef: import('lit-html/directives/ref').Ref<HTMLDivElement>;
4
+ contentRef: import('lit-html/directives/ref').Ref<HTMLSlotElement>;
5
5
  createRenderRoot(): this;
6
6
  uniqueId: string;
7
7
  private scale;