@omnimedia/omnitool 1.1.0-8 → 1.1.0-80
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.
- package/README.md +293 -76
- package/package.json +19 -17
- package/s/context.ts +0 -7
- package/s/demo/demo.bundle.ts +104 -43
- package/s/demo/demo.css +244 -14
- package/s/demo/routines/export-test.ts +16 -0
- package/s/demo/routines/filmstrip-test.ts +22 -18
- package/s/demo/routines/load-video.ts +2 -3
- package/s/demo/routines/playback-test.ts +61 -0
- package/s/demo/routines/timeline-setup.ts +50 -0
- package/s/demo/routines/transcode-test.ts +19 -5
- package/s/demo/routines/transitions-test.ts +2 -2
- package/s/demo/routines/waveform-test.ts +35 -8
- package/s/driver/driver-worker.ts +9 -0
- package/s/driver/driver.test.ts +1 -1
- package/s/driver/driver.ts +43 -42
- package/s/driver/fns/schematic.ts +26 -6
- package/s/driver/fns/work.ts +65 -189
- package/s/driver/parts/compositor.ts +316 -0
- package/s/driver/parts/machina.ts +19 -20
- package/s/driver/utils/find-pixi-filter.ts +24 -0
- package/s/index.html.ts +103 -25
- package/s/index.ts +1 -0
- package/s/tests.bundle.ts +11 -0
- package/s/tests.html.ts +28 -0
- package/s/timeline/index.ts +8 -2
- package/s/timeline/parts/animations.ts +56 -0
- package/s/timeline/parts/filmstrip.ts +43 -16
- package/s/timeline/parts/filters.ts +453 -0
- package/s/timeline/parts/item.ts +70 -7
- package/s/timeline/parts/media.ts +12 -2
- package/s/timeline/parts/resource-pool.ts +8 -5
- package/s/timeline/parts/resource.ts +3 -0
- package/s/timeline/parts/waveform/parts/collect.ts +72 -0
- package/s/timeline/parts/waveform/parts/render.ts +45 -0
- package/s/timeline/parts/waveform/parts/types.ts +24 -0
- package/s/timeline/parts/waveform/waveform.ts +161 -0
- package/s/timeline/renderers/export/parts/audio-gain.ts +17 -0
- package/s/timeline/renderers/export/parts/audio-mix.ts +133 -0
- package/s/timeline/renderers/export/parts/cursor.ts +129 -0
- package/s/timeline/renderers/export/parts/produce-audio.ts +64 -0
- package/s/timeline/renderers/export/parts/produce-video.ts +49 -0
- package/s/timeline/renderers/export/parts/resamplers.ts +48 -0
- package/s/timeline/renderers/export/produce.ts +28 -0
- package/s/timeline/renderers/parts/handy.ts +377 -0
- package/s/timeline/renderers/parts/samplers/audio/parts/find.ts +19 -0
- package/s/timeline/renderers/parts/samplers/audio/parts/init.ts +60 -0
- package/s/timeline/renderers/parts/samplers/audio/parts/sink.ts +38 -0
- package/s/timeline/renderers/parts/samplers/audio/parts/types.ts +16 -0
- package/s/timeline/renderers/parts/samplers/audio/sampler.ts +35 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/defaults.ts +16 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/sample.ts +69 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/sequence.ts +111 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/sink.ts +38 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/transition.ts +28 -0
- package/s/timeline/renderers/parts/samplers/visual/parts/types.ts +10 -0
- package/s/timeline/renderers/parts/samplers/visual/sampler.ts +28 -0
- package/s/timeline/renderers/parts/schedulers.ts +96 -0
- package/s/timeline/renderers/player/parts/playback.ts +159 -0
- package/s/timeline/renderers/player/player.ts +77 -0
- package/s/timeline/renderers/renderers.test.ts +385 -0
- package/s/timeline/sugar/helpers.ts +190 -0
- package/s/timeline/sugar/o.ts +271 -59
- package/s/timeline/sugar/omni.test.ts +210 -0
- package/s/timeline/sugar/omni.ts +35 -12
- package/s/timeline/types.ts +28 -3
- package/s/timeline/utils/anim.ts +71 -0
- package/s/timeline/utils/checksum.ts +3 -1
- package/s/timeline/utils/datafile.ts +15 -4
- package/s/timeline/utils/dummy-data.ts +3 -3
- package/s/timeline/utils/matrix.ts +21 -0
- package/s/timeline/utils/terps.ts +81 -0
- package/s/units/fps.ts +8 -0
- package/s/units/ms.ts +8 -0
- package/s/units/seconds.ts +8 -0
- package/x/WebGLRenderer-4CHIZDHY.js +2 -0
- package/x/WebGLRenderer-4CHIZDHY.js.map +7 -0
- package/x/WebGPURenderer-O6WXU2QR.js +2 -0
- package/x/WebGPURenderer-O6WXU2QR.js.map +7 -0
- package/x/browserAll-WNIOTNNP.js +2 -0
- package/x/browserAll-WNIOTNNP.js.map +7 -0
- package/x/chunk-23FOBGX6.js +2 -0
- package/x/chunk-23FOBGX6.js.map +7 -0
- package/x/chunk-73XOWA4F.js +393 -0
- package/x/chunk-73XOWA4F.js.map +7 -0
- package/x/chunk-IAWJKNX5.js +42 -0
- package/x/chunk-IAWJKNX5.js.map +7 -0
- package/x/chunk-KCMVLWBS.js +157 -0
- package/x/chunk-KCMVLWBS.js.map +7 -0
- package/x/chunk-MR32SQ27.js +2 -0
- package/x/chunk-MR32SQ27.js.map +7 -0
- package/x/chunk-VCUJJYRJ.js +327 -0
- package/x/chunk-VCUJJYRJ.js.map +7 -0
- package/x/chunk-XNIZ4L5W.js +269 -0
- package/x/chunk-XNIZ4L5W.js.map +7 -0
- package/x/chunk-ZLGNQFUI.js +15 -0
- package/x/chunk-ZLGNQFUI.js.map +7 -0
- package/x/context.d.ts +1 -4
- package/x/context.js +1 -5
- package/x/context.js.map +1 -1
- package/x/demo/WebGLRenderer-RSJAHIG2.js +2 -0
- package/x/demo/WebGLRenderer-RSJAHIG2.js.map +7 -0
- package/x/demo/WebGPURenderer-MGFAJCYY.js +2 -0
- package/x/demo/WebGPURenderer-MGFAJCYY.js.map +7 -0
- package/x/demo/browserAll-PTRBXBRP.js +2 -0
- package/x/demo/browserAll-PTRBXBRP.js.map +7 -0
- package/x/demo/chunk-CUYOGHWU.js +42 -0
- package/x/demo/chunk-CUYOGHWU.js.map +7 -0
- package/x/demo/chunk-DGTDNJ7W.js +2 -0
- package/x/demo/chunk-DGTDNJ7W.js.map +7 -0
- package/x/demo/chunk-FZ5BYF63.js +15 -0
- package/x/demo/chunk-FZ5BYF63.js.map +7 -0
- package/x/demo/chunk-KWN4NNES.js +269 -0
- package/x/demo/chunk-KWN4NNES.js.map +7 -0
- package/x/demo/chunk-OJ3FPXW7.js +157 -0
- package/x/demo/chunk-OJ3FPXW7.js.map +7 -0
- package/x/demo/chunk-PSLUOGTZ.js +393 -0
- package/x/demo/chunk-PSLUOGTZ.js.map +7 -0
- package/x/demo/chunk-VLCVEIFU.js +327 -0
- package/x/demo/chunk-VLCVEIFU.js.map +7 -0
- package/x/demo/chunk-ZWCPT5FR.js +2 -0
- package/x/demo/chunk-ZWCPT5FR.js.map +7 -0
- package/x/demo/demo.bundle.js +94 -40
- package/x/demo/demo.bundle.js.map +1 -1
- package/x/demo/demo.bundle.min.js +7340 -90
- package/x/demo/demo.bundle.min.js.map +4 -4
- package/x/demo/demo.css +244 -14
- package/x/demo/routines/export-test.d.ts +2 -0
- package/x/demo/routines/export-test.js +10 -0
- package/x/demo/routines/export-test.js.map +1 -0
- package/x/demo/routines/filmstrip-test.d.ts +1 -1
- package/x/demo/routines/filmstrip-test.js +20 -17
- package/x/demo/routines/filmstrip-test.js.map +1 -1
- package/x/demo/routines/load-video.d.ts +1 -1
- package/x/demo/routines/load-video.js +1 -2
- package/x/demo/routines/load-video.js.map +1 -1
- package/x/demo/routines/playback-test.d.ts +2 -0
- package/x/demo/routines/playback-test.js +51 -0
- package/x/demo/routines/playback-test.js.map +1 -0
- package/x/demo/routines/timeline-setup.d.ts +6 -0
- package/x/demo/routines/timeline-setup.js +32 -0
- package/x/demo/routines/timeline-setup.js.map +1 -0
- package/x/demo/routines/transcode-test.js +15 -5
- package/x/demo/routines/transcode-test.js.map +1 -1
- package/x/demo/routines/transitions-test.js +2 -2
- package/x/demo/routines/transitions-test.js.map +1 -1
- package/x/demo/routines/waveform-test.d.ts +2 -1
- package/x/demo/routines/waveform-test.js +29 -8
- package/x/demo/routines/waveform-test.js.map +1 -1
- package/x/demo/webworkerAll-JENRT6BT.js +2 -0
- package/x/demo/webworkerAll-JENRT6BT.js.map +7 -0
- package/x/driver/driver-worker.d.ts +1 -0
- package/x/driver/driver-worker.js +6 -0
- package/x/driver/driver-worker.js.map +1 -0
- package/x/driver/driver.d.ts +25 -8
- package/x/driver/driver.js +43 -39
- package/x/driver/driver.js.map +1 -1
- package/x/driver/driver.test.js +1 -1
- package/x/driver/driver.test.js.map +1 -1
- package/x/driver/driver.worker.bundle.min.js +117 -3506
- package/x/driver/driver.worker.bundle.min.js.map +4 -4
- package/x/driver/fns/host.d.ts +3 -2
- package/x/driver/fns/schematic.d.ts +25 -4
- package/x/driver/fns/work.d.ts +4 -4
- package/x/driver/fns/work.js +55 -155
- package/x/driver/fns/work.js.map +1 -1
- package/x/driver/parts/compositor.d.ts +39 -0
- package/x/driver/parts/compositor.js +261 -0
- package/x/driver/parts/compositor.js.map +1 -0
- package/x/driver/parts/machina.d.ts +0 -20
- package/x/driver/parts/machina.js +6 -10
- package/x/driver/parts/machina.js.map +1 -1
- package/x/driver/utils/find-pixi-filter.d.ts +5 -0
- package/x/driver/utils/find-pixi-filter.js +13 -0
- package/x/driver/utils/find-pixi-filter.js.map +1 -0
- package/x/features/speech/transcribe/parts/prep-audio.d.ts +1 -1
- package/x/features/speech/transcribe/worker.bundle.min.js +899 -899
- package/x/features/speech/transcribe/worker.bundle.min.js.map +4 -4
- package/x/index.d.ts +1 -0
- package/x/index.html +347 -38
- package/x/index.html.js +103 -24
- package/x/index.html.js.map +1 -1
- package/x/index.js +1 -0
- package/x/index.js.map +1 -1
- package/x/tests.bundle.js +8 -0
- package/x/tests.bundle.js.map +1 -0
- package/x/tests.bundle.min.js +7374 -0
- package/x/tests.bundle.min.js.map +7 -0
- package/x/tests.html +316 -0
- package/x/tests.html.d.ts +2 -0
- package/x/tests.html.js +22 -0
- package/x/tests.html.js.map +1 -0
- package/x/timeline/index.d.ts +7 -1
- package/x/timeline/index.js +7 -1
- package/x/timeline/index.js.map +1 -1
- package/x/timeline/parts/animations.d.ts +105 -0
- package/x/timeline/parts/animations.js +28 -0
- package/x/timeline/parts/animations.js.map +1 -0
- package/x/timeline/parts/filmstrip.d.ts +4 -3
- package/x/timeline/parts/filmstrip.js +29 -10
- package/x/timeline/parts/filmstrip.js.map +1 -1
- package/x/timeline/parts/filters.d.ts +108 -0
- package/x/timeline/parts/filters.js +340 -0
- package/x/timeline/parts/filters.js.map +1 -0
- package/x/timeline/parts/item.d.ts +56 -4
- package/x/timeline/parts/item.js +5 -1
- package/x/timeline/parts/item.js.map +1 -1
- package/x/timeline/parts/media.d.ts +2 -0
- package/x/timeline/parts/media.js +11 -2
- package/x/timeline/parts/media.js.map +1 -1
- package/x/timeline/parts/resource-pool.d.ts +3 -0
- package/x/timeline/parts/resource-pool.js +7 -4
- package/x/timeline/parts/resource-pool.js.map +1 -1
- package/x/timeline/parts/resource.d.ts +3 -0
- package/x/timeline/parts/waveform/parts/collect.d.ts +11 -0
- package/x/timeline/parts/waveform/parts/collect.js +56 -0
- package/x/timeline/parts/waveform/parts/collect.js.map +1 -0
- package/x/timeline/parts/waveform/parts/render.d.ts +5 -0
- package/x/timeline/parts/waveform/parts/render.js +29 -0
- package/x/timeline/parts/waveform/parts/render.js.map +1 -0
- package/x/timeline/parts/waveform/parts/types.d.ts +21 -0
- package/x/timeline/parts/waveform/parts/types.js.map +1 -0
- package/x/timeline/parts/waveform/waveform.d.ts +19 -0
- package/x/timeline/parts/waveform/waveform.js +133 -0
- package/x/timeline/parts/waveform/waveform.js.map +1 -0
- package/x/timeline/renderers/export/parts/audio-gain.d.ts +1 -0
- package/x/timeline/renderers/export/parts/audio-gain.js +13 -0
- package/x/timeline/renderers/export/parts/audio-gain.js.map +1 -0
- package/x/timeline/renderers/export/parts/audio-mix.d.ts +21 -0
- package/x/timeline/renderers/export/parts/audio-mix.js +89 -0
- package/x/timeline/renderers/export/parts/audio-mix.js.map +1 -0
- package/x/timeline/renderers/export/parts/cursor.d.ts +18 -0
- package/x/timeline/renderers/export/parts/cursor.js +99 -0
- package/x/timeline/renderers/export/parts/cursor.js.map +1 -0
- package/x/timeline/renderers/export/parts/produce-audio.d.ts +6 -0
- package/x/timeline/renderers/export/parts/produce-audio.js +41 -0
- package/x/timeline/renderers/export/parts/produce-audio.js.map +1 -0
- package/x/timeline/renderers/export/parts/produce-video.d.ts +10 -0
- package/x/timeline/renderers/export/parts/produce-video.js +26 -0
- package/x/timeline/renderers/export/parts/produce-video.js.map +1 -0
- package/x/timeline/renderers/export/parts/resamplers.d.ts +12 -0
- package/x/timeline/renderers/export/parts/resamplers.js +29 -0
- package/x/timeline/renderers/export/parts/resamplers.js.map +1 -0
- package/x/timeline/renderers/export/produce.d.ts +13 -0
- package/x/timeline/renderers/export/produce.js +15 -0
- package/x/timeline/renderers/export/produce.js.map +1 -0
- package/x/timeline/renderers/parts/handy.d.ts +36 -0
- package/x/timeline/renderers/parts/handy.js +221 -0
- package/x/timeline/renderers/parts/handy.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/find.d.ts +6 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/find.js +15 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/find.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/init.d.ts +5 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/init.js +40 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/init.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/sink.d.ts +8 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/sink.js +24 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/sink.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/types.d.ts +14 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/types.js +2 -0
- package/x/timeline/renderers/parts/samplers/audio/parts/types.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/audio/sampler.d.ts +11 -0
- package/x/timeline/renderers/parts/samplers/audio/sampler.js +22 -0
- package/x/timeline/renderers/parts/samplers/audio/sampler.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/defaults.d.ts +5 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/defaults.js +10 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/defaults.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sample.d.ts +6 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sample.js +48 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sample.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sequence.d.ts +6 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sequence.js +75 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sequence.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sink.d.ts +8 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sink.js +24 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/sink.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/transition.d.ts +3 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/transition.js +18 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/transition.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/types.d.ts +8 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/types.js +2 -0
- package/x/timeline/renderers/parts/samplers/visual/parts/types.js.map +1 -0
- package/x/timeline/renderers/parts/samplers/visual/sampler.d.ts +7 -0
- package/x/timeline/renderers/parts/samplers/visual/sampler.js +17 -0
- package/x/timeline/renderers/parts/samplers/visual/sampler.js.map +1 -0
- package/x/timeline/renderers/parts/schedulers.d.ts +17 -0
- package/x/timeline/renderers/parts/schedulers.js +64 -0
- package/x/timeline/renderers/parts/schedulers.js.map +1 -0
- package/x/timeline/renderers/player/parts/playback.d.ts +36 -0
- package/x/timeline/renderers/player/parts/playback.js +113 -0
- package/x/timeline/renderers/player/parts/playback.js.map +1 -0
- package/x/timeline/renderers/player/player.d.ts +25 -0
- package/x/timeline/renderers/player/player.js +56 -0
- package/x/timeline/renderers/player/player.js.map +1 -0
- package/x/timeline/renderers/renderers.test.d.ts +32 -0
- package/x/timeline/renderers/renderers.test.js +305 -0
- package/x/timeline/renderers/renderers.test.js.map +1 -0
- package/x/timeline/sugar/helpers.d.ts +56 -0
- package/x/timeline/sugar/helpers.js +104 -0
- package/x/timeline/sugar/helpers.js.map +1 -0
- package/x/timeline/sugar/o.d.ts +48 -14
- package/x/timeline/sugar/o.js +213 -54
- package/x/timeline/sugar/o.js.map +1 -1
- package/x/timeline/sugar/omni.d.ts +11 -6
- package/x/timeline/sugar/omni.js +28 -8
- package/x/timeline/sugar/omni.js.map +1 -1
- package/x/timeline/sugar/omni.test.d.ts +27 -0
- package/x/timeline/sugar/omni.test.js +128 -0
- package/x/timeline/sugar/omni.test.js.map +1 -0
- package/x/timeline/types.d.ts +12 -2
- package/x/timeline/utils/anim.d.ts +5 -0
- package/x/timeline/utils/anim.js +44 -0
- package/x/timeline/utils/anim.js.map +1 -0
- package/x/timeline/utils/checksum.d.ts +3 -2
- package/x/timeline/utils/checksum.js.map +1 -1
- package/x/timeline/utils/datafile.d.ts +5 -3
- package/x/timeline/utils/datafile.js +18 -5
- package/x/timeline/utils/datafile.js.map +1 -1
- package/x/timeline/utils/dummy-data.d.ts +1 -2
- package/x/timeline/utils/dummy-data.js +4 -2
- package/x/timeline/utils/dummy-data.js.map +1 -1
- package/x/timeline/utils/matrix.d.ts +9 -0
- package/x/timeline/utils/matrix.js +10 -0
- package/x/timeline/utils/matrix.js.map +1 -1
- package/x/timeline/utils/terps.d.ts +11 -0
- package/x/timeline/utils/terps.js +57 -0
- package/x/timeline/utils/terps.js.map +1 -0
- package/x/units/fps.d.ts +6 -0
- package/x/units/fps.js +2 -0
- package/x/units/fps.js.map +1 -0
- package/x/units/ms.d.ts +6 -0
- package/x/units/ms.js +2 -0
- package/x/units/ms.js.map +1 -0
- package/x/units/seconds.d.ts +6 -0
- package/x/units/seconds.js +2 -0
- package/x/units/seconds.js.map +1 -0
- package/x/webworkerAll-RLCTMSDD.js +2 -0
- package/x/webworkerAll-RLCTMSDD.js.map +7 -0
- package/s/tests.test.ts +0 -8
- package/s/timeline/parts/compositor/export.ts +0 -77
- package/s/timeline/parts/compositor/parts/html-tree.ts +0 -37
- package/s/timeline/parts/compositor/parts/schedulers.ts +0 -85
- package/s/timeline/parts/compositor/parts/tree-builder.ts +0 -184
- package/s/timeline/parts/compositor/parts/webcodecs-tree.ts +0 -30
- package/s/timeline/parts/compositor/playback.ts +0 -81
- package/s/timeline/parts/compositor/samplers/html.ts +0 -115
- package/s/timeline/parts/compositor/samplers/webcodecs.ts +0 -60
- package/s/timeline/parts/waveform.ts +0 -62
- package/s/timeline/sugar/builders.ts +0 -102
- package/s/timeline/sugar/omni-test.ts +0 -38
- package/s/timeline/timeline.ts +0 -22
- package/s/timeline/utils/audio-stream.ts +0 -15
- package/s/timeline/utils/video-cursor.ts +0 -40
- package/s/tools/common/loader.ts +0 -26
- package/s/tools/common/transformer-pipeline.ts +0 -26
- package/s/tools/speech-recognition/common/model.ts +0 -26
- package/s/tools/speech-recognition/whisper/fns/host.ts +0 -25
- package/s/tools/speech-recognition/whisper/fns/schematic.ts +0 -23
- package/s/tools/speech-recognition/whisper/fns/work.ts +0 -91
- package/s/tools/speech-recognition/whisper/parts/types.ts +0 -38
- package/s/tools/speech-recognition/whisper/parts/worker.bundle.ts +0 -7
- package/s/tools/speech-recognition/whisper/tool.ts +0 -70
- package/x/tests.test.js +0 -6
- package/x/tests.test.js.map +0 -1
- package/x/timeline/parts/compositor/export.d.ts +0 -9
- package/x/timeline/parts/compositor/export.js +0 -64
- package/x/timeline/parts/compositor/export.js.map +0 -1
- package/x/timeline/parts/compositor/parts/html-tree.d.ts +0 -3
- package/x/timeline/parts/compositor/parts/html-tree.js +0 -40
- package/x/timeline/parts/compositor/parts/html-tree.js.map +0 -1
- package/x/timeline/parts/compositor/parts/schedulers.d.ts +0 -15
- package/x/timeline/parts/compositor/parts/schedulers.js +0 -64
- package/x/timeline/parts/compositor/parts/schedulers.js.map +0 -1
- package/x/timeline/parts/compositor/parts/tree-builder.d.ts +0 -37
- package/x/timeline/parts/compositor/parts/tree-builder.js +0 -147
- package/x/timeline/parts/compositor/parts/tree-builder.js.map +0 -1
- package/x/timeline/parts/compositor/parts/webcodecs-tree.d.ts +0 -3
- package/x/timeline/parts/compositor/parts/webcodecs-tree.js +0 -28
- package/x/timeline/parts/compositor/parts/webcodecs-tree.js.map +0 -1
- package/x/timeline/parts/compositor/playback.d.ts +0 -19
- package/x/timeline/parts/compositor/playback.js +0 -71
- package/x/timeline/parts/compositor/playback.js.map +0 -1
- package/x/timeline/parts/compositor/samplers/html.d.ts +0 -3
- package/x/timeline/parts/compositor/samplers/html.js +0 -106
- package/x/timeline/parts/compositor/samplers/html.js.map +0 -1
- package/x/timeline/parts/compositor/samplers/webcodecs.d.ts +0 -2
- package/x/timeline/parts/compositor/samplers/webcodecs.js +0 -55
- package/x/timeline/parts/compositor/samplers/webcodecs.js.map +0 -1
- package/x/timeline/parts/waveform.d.ts +0 -8
- package/x/timeline/parts/waveform.js +0 -51
- package/x/timeline/parts/waveform.js.map +0 -1
- package/x/timeline/sugar/builders.d.ts +0 -96
- package/x/timeline/sugar/builders.js +0 -108
- package/x/timeline/sugar/builders.js.map +0 -1
- package/x/timeline/sugar/omni-test.d.ts +0 -1
- package/x/timeline/sugar/omni-test.js +0 -22
- package/x/timeline/sugar/omni-test.js.map +0 -1
- package/x/timeline/timeline.d.ts +0 -9
- package/x/timeline/timeline.js +0 -22
- package/x/timeline/timeline.js.map +0 -1
- package/x/timeline/utils/audio-stream.d.ts +0 -6
- package/x/timeline/utils/audio-stream.js +0 -17
- package/x/timeline/utils/audio-stream.js.map +0 -1
- package/x/timeline/utils/video-cursor.d.ts +0 -10
- package/x/timeline/utils/video-cursor.js +0 -36
- package/x/timeline/utils/video-cursor.js.map +0 -1
- package/x/tools/common/loader.d.ts +0 -19
- package/x/tools/common/loader.js +0 -18
- package/x/tools/common/loader.js.map +0 -1
- package/x/tools/common/transformer-pipeline.d.ts +0 -8
- package/x/tools/common/transformer-pipeline.js +0 -24
- package/x/tools/common/transformer-pipeline.js.map +0 -1
- package/x/tools/speech-recognition/common/model.d.ts +0 -14
- package/x/tools/speech-recognition/common/model.js +0 -16
- package/x/tools/speech-recognition/common/model.js.map +0 -1
- package/x/tools/speech-recognition/whisper/fns/host.d.ts +0 -13
- package/x/tools/speech-recognition/whisper/fns/host.js +0 -19
- package/x/tools/speech-recognition/whisper/fns/host.js.map +0 -1
- package/x/tools/speech-recognition/whisper/fns/schematic.d.ts +0 -19
- package/x/tools/speech-recognition/whisper/fns/schematic.js +0 -2
- package/x/tools/speech-recognition/whisper/fns/schematic.js.map +0 -1
- package/x/tools/speech-recognition/whisper/fns/work.d.ts +0 -12
- package/x/tools/speech-recognition/whisper/fns/work.js +0 -74
- package/x/tools/speech-recognition/whisper/fns/work.js.map +0 -1
- package/x/tools/speech-recognition/whisper/parts/types.d.ts +0 -31
- package/x/tools/speech-recognition/whisper/parts/types.js.map +0 -1
- package/x/tools/speech-recognition/whisper/parts/worker.bundle.d.ts +0 -1
- package/x/tools/speech-recognition/whisper/parts/worker.bundle.js +0 -4
- package/x/tools/speech-recognition/whisper/parts/worker.bundle.js.map +0 -1
- package/x/tools/speech-recognition/whisper/parts/worker.bundle.min.js +0 -8
- package/x/tools/speech-recognition/whisper/parts/worker.bundle.min.js.map +0 -7
- package/x/tools/speech-recognition/whisper/tool.d.ts +0 -12
- package/x/tools/speech-recognition/whisper/tool.js +0 -63
- package/x/tools/speech-recognition/whisper/tool.js.map +0 -1
- /package/x/{tests.test.d.ts → tests.bundle.d.ts} +0 -0
- /package/x/{tools/speech-recognition/whisper → timeline/parts/waveform}/parts/types.js +0 -0
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
|
|
2
|
+
import {AudioSample} from "mediabunny"
|
|
3
|
+
|
|
4
|
+
import {itemsFrom} from "../../handy.js"
|
|
5
|
+
import {initStreams} from "./parts/init.js"
|
|
6
|
+
import {AudioSinkPool} from "./parts/sink.js"
|
|
7
|
+
import {Ms} from "../../../../../units/ms.js"
|
|
8
|
+
import {findEarliestStream} from "./parts/find.js"
|
|
9
|
+
import {TimelineFile} from "../../../../parts/basics.js"
|
|
10
|
+
import {DecoderSource} from "../../../../../driver/fns/schematic.js"
|
|
11
|
+
|
|
12
|
+
export function createAudioSampler(resolveMedia: (hash: string) => DecoderSource) {
|
|
13
|
+
const sinkPool = new AudioSinkPool(resolveMedia)
|
|
14
|
+
|
|
15
|
+
return {
|
|
16
|
+
async *sampleAudio(timeline: TimelineFile, from: Ms) {
|
|
17
|
+
const items = itemsFrom({timeline, from})
|
|
18
|
+
const streams = await initStreams(sinkPool, items, from)
|
|
19
|
+
|
|
20
|
+
while (streams.length > 0) {
|
|
21
|
+
const {stream, index} = findEarliestStream(streams)
|
|
22
|
+
|
|
23
|
+
yield stream.output()
|
|
24
|
+
|
|
25
|
+
const advancing = await stream.advance()
|
|
26
|
+
|
|
27
|
+
if (!advancing) {
|
|
28
|
+
streams.splice(index, 1)
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
|
|
2
|
+
import {VideoSink} from "./sink.js"
|
|
3
|
+
import {Ms} from "../../../../../../units/ms.js"
|
|
4
|
+
import {Item} from "../../../../../parts/item.js"
|
|
5
|
+
|
|
6
|
+
export type VideoSampler = (item: Item.Video, time: Ms) => Promise<VideoFrame | undefined>
|
|
7
|
+
|
|
8
|
+
export function createDefaultVideoSampler(sink: VideoSink): VideoSampler {
|
|
9
|
+
return async (item, time) => {
|
|
10
|
+
const s = await sink.getSink(item.mediaHash)
|
|
11
|
+
const sample = await s?.getSample(time / 1000)
|
|
12
|
+
const frame = sample?.toVideoFrame()
|
|
13
|
+
sample?.close()
|
|
14
|
+
return frame ?? undefined
|
|
15
|
+
}
|
|
16
|
+
}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
|
|
2
|
+
import {SampleContext} from "./types.js"
|
|
3
|
+
import {sampleSequence} from "./sequence.js"
|
|
4
|
+
import {Ms} from "../../../../../../units/ms.js"
|
|
5
|
+
import {Item, Kind, SpatialItem} from "../../../../../parts/item.js"
|
|
6
|
+
import {FilterSpec, Layer} from "../../../../../../driver/fns/schematic.js"
|
|
7
|
+
import {AncestorAt, computeOpacity, computeWorldMatrix} from "../../../handy.js"
|
|
8
|
+
|
|
9
|
+
export async function sampleVisual(
|
|
10
|
+
ctx: SampleContext,
|
|
11
|
+
item: Item.Any,
|
|
12
|
+
time: Ms,
|
|
13
|
+
ancestors: AncestorAt[]
|
|
14
|
+
): Promise<Layer[]> {
|
|
15
|
+
const matrix = computeWorldMatrix(ctx.items, ancestors, item, time)
|
|
16
|
+
const alpha = computeOpacity(ctx, item, time)
|
|
17
|
+
const crop = "spatialId" in item && item.spatialId
|
|
18
|
+
? (ctx.items.get(item.spatialId) as SpatialItem | undefined)?.crop
|
|
19
|
+
: undefined
|
|
20
|
+
const filters = "filterIds" in item && item.filterIds
|
|
21
|
+
? item.filterIds
|
|
22
|
+
.map(id => ctx.items.get(id) as Item.Filter | undefined)
|
|
23
|
+
.filter((filter): filter is Item.Filter => !!filter?.enabled)
|
|
24
|
+
.map(filter => ({type: filter.type, params: filter.params}) as FilterSpec)
|
|
25
|
+
: undefined
|
|
26
|
+
|
|
27
|
+
switch (item.kind) {
|
|
28
|
+
case Kind.Stack: {
|
|
29
|
+
const nextAnc = [...ancestors, {item, localTime: time}]
|
|
30
|
+
|
|
31
|
+
const layers = await Promise.all(
|
|
32
|
+
item.childrenIds
|
|
33
|
+
.map(id => ctx.items.get(id))
|
|
34
|
+
.filter((child): child is Item.Any => !!child)
|
|
35
|
+
.map(child => sampleVisual(ctx, child, time, nextAnc))
|
|
36
|
+
)
|
|
37
|
+
|
|
38
|
+
return layers.flat()
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
case Kind.Sequence:
|
|
42
|
+
return sampleSequence(ctx, item, time, ancestors)
|
|
43
|
+
|
|
44
|
+
case Kind.Video: {
|
|
45
|
+
if (time < 0 || time >= item.duration) return []
|
|
46
|
+
|
|
47
|
+
const frame = await ctx.videoSampler(item, time)
|
|
48
|
+
return frame ? [{kind: "image", frame, matrix, alpha, crop, filters, id: item.id}] : []
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
case Kind.Text: {
|
|
52
|
+
if (time < 0 || time >= item.duration) return []
|
|
53
|
+
|
|
54
|
+
const style = item.styleId
|
|
55
|
+
? (ctx.items.get(item.styleId) as Item.TextStyle)?.style
|
|
56
|
+
: undefined
|
|
57
|
+
|
|
58
|
+
return [{id: item.id, kind: "text", content: item.content, style, matrix, alpha, crop, filters}]
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
case Kind.Gap: {
|
|
62
|
+
return [{id: item.id, kind: "gap"}]
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
default:
|
|
66
|
+
return []
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
|
|
2
|
+
import {sampleVisual} from "./sample.js"
|
|
3
|
+
import {SampleContext} from "./types.js"
|
|
4
|
+
import {sampleTransition} from "./transition.js"
|
|
5
|
+
import {ms, Ms} from "../../../../../../units/ms.js"
|
|
6
|
+
import {Item, Kind} from "../../../../../parts/item.js"
|
|
7
|
+
import {Layer} from "../../../../../../driver/fns/schematic.js"
|
|
8
|
+
import {AncestorAt, computeItemDuration} from "../../../handy.js"
|
|
9
|
+
|
|
10
|
+
export async function sampleSequence(
|
|
11
|
+
ctx: SampleContext,
|
|
12
|
+
seq: Item.Sequence,
|
|
13
|
+
time: Ms,
|
|
14
|
+
ancestors: AncestorAt[]
|
|
15
|
+
): Promise<Layer[]> {
|
|
16
|
+
const state = sampleSequenceAt(ctx, seq, time)
|
|
17
|
+
|
|
18
|
+
if (!state) return []
|
|
19
|
+
|
|
20
|
+
const nextAnc = [...ancestors, {item: seq, localTime: time}]
|
|
21
|
+
|
|
22
|
+
if (!state.isTransitioning) {
|
|
23
|
+
return sampleVisual(ctx, state.item, state.localTime, nextAnc)
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
return sampleTransition(
|
|
27
|
+
state.transition,
|
|
28
|
+
state.progress,
|
|
29
|
+
sampleVisual(ctx, state.outgoing, state.outgoingTime, nextAnc),
|
|
30
|
+
sampleVisual(ctx, state.incoming, state.incomingTime, nextAnc)
|
|
31
|
+
)
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
function sampleSequenceAt(
|
|
35
|
+
ctx: SampleContext,
|
|
36
|
+
seq: Item.Sequence,
|
|
37
|
+
time: Ms
|
|
38
|
+
) {
|
|
39
|
+
const children = seq.childrenIds
|
|
40
|
+
.map(id => ctx.items.get(id))
|
|
41
|
+
.filter((i): i is Item.Any => !!i)
|
|
42
|
+
|
|
43
|
+
let cursor = ms(0)
|
|
44
|
+
|
|
45
|
+
for (let i = 0; i < children.length; i++) {
|
|
46
|
+
const currentItem = children[i]
|
|
47
|
+
if (currentItem.kind === Kind.Transition)
|
|
48
|
+
continue
|
|
49
|
+
|
|
50
|
+
const currentItemStart = cursor
|
|
51
|
+
const currentItemDuration = computeItemDuration(currentItem.id, ctx.timeline)
|
|
52
|
+
const currentItemEnd = ms(currentItemStart + currentItemDuration)
|
|
53
|
+
|
|
54
|
+
const next = children[i + 1]
|
|
55
|
+
const hasTransition = next?.kind === Kind.Transition
|
|
56
|
+
|
|
57
|
+
if (!hasTransition) {
|
|
58
|
+
if (time < currentItemEnd) {
|
|
59
|
+
return {
|
|
60
|
+
isTransitioning: false,
|
|
61
|
+
item: currentItem,
|
|
62
|
+
localTime: ms(time - currentItemStart)
|
|
63
|
+
} as const
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
cursor = currentItemEnd
|
|
67
|
+
continue
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const transition = next as Item.Transition
|
|
71
|
+
const incoming = children[i + 2]
|
|
72
|
+
|
|
73
|
+
if (!incoming || incoming.kind === Kind.Transition) {
|
|
74
|
+
cursor = currentItemEnd
|
|
75
|
+
continue
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const incomingItemDuration = computeItemDuration(incoming.id, ctx.timeline)
|
|
79
|
+
const overlap = Math.max(0, Math.min(transition.duration, currentItemDuration, incomingItemDuration))
|
|
80
|
+
const currentItemSoloEnd = ms(currentItemEnd - overlap)
|
|
81
|
+
|
|
82
|
+
if (time < currentItemSoloEnd) {
|
|
83
|
+
return {
|
|
84
|
+
isTransitioning: false,
|
|
85
|
+
item: currentItem,
|
|
86
|
+
localTime: ms(time - currentItemStart)
|
|
87
|
+
} as const
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (time < currentItemEnd) {
|
|
91
|
+
const inLocal = ms(time - currentItemSoloEnd)
|
|
92
|
+
const outLocal = ms(time - currentItemStart)
|
|
93
|
+
|
|
94
|
+
return {
|
|
95
|
+
isTransitioning: true,
|
|
96
|
+
incoming,
|
|
97
|
+
outgoing: currentItem,
|
|
98
|
+
outgoingTime: outLocal,
|
|
99
|
+
incomingTime: inLocal,
|
|
100
|
+
progress: overlap > 0 ? inLocal / overlap : 1,
|
|
101
|
+
transition
|
|
102
|
+
} as const
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
cursor = currentItemSoloEnd
|
|
106
|
+
i++
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
return null
|
|
110
|
+
}
|
|
111
|
+
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
|
|
2
|
+
import {ALL_FORMATS, Input, VideoSampleSink} from "mediabunny"
|
|
3
|
+
|
|
4
|
+
import {DecoderSource} from "../../../../../../driver/fns/schematic.js"
|
|
5
|
+
import {loadDecoderSource} from "../../../../../../driver/utils/load-decoder-source.js"
|
|
6
|
+
|
|
7
|
+
type SinkState = {
|
|
8
|
+
input: Input
|
|
9
|
+
sink: VideoSampleSink | null
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export class VideoSink {
|
|
13
|
+
readonly #sinks = new Map<string, SinkState>()
|
|
14
|
+
|
|
15
|
+
constructor(
|
|
16
|
+
private resolveMedia: (hash: string) => DecoderSource
|
|
17
|
+
) {}
|
|
18
|
+
|
|
19
|
+
async getSink(hash: string) {
|
|
20
|
+
const existing = this.#sinks.get(hash)
|
|
21
|
+
|
|
22
|
+
if (existing)
|
|
23
|
+
return existing.sink
|
|
24
|
+
|
|
25
|
+
const input = new Input({
|
|
26
|
+
formats: ALL_FORMATS,
|
|
27
|
+
source: await loadDecoderSource(this.resolveMedia(hash)),
|
|
28
|
+
})
|
|
29
|
+
|
|
30
|
+
const videoTrack = await input.getPrimaryVideoTrack()
|
|
31
|
+
const canDecodeVideo = !!videoTrack && await videoTrack.canDecode()
|
|
32
|
+
const sink = canDecodeVideo && videoTrack ? new VideoSampleSink(videoTrack) : null
|
|
33
|
+
|
|
34
|
+
this.#sinks.set(hash, {input, sink})
|
|
35
|
+
|
|
36
|
+
return sink
|
|
37
|
+
}
|
|
38
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
|
|
2
|
+
import {Item} from "../../../../../parts/item.js"
|
|
3
|
+
import {Layer} from "../../../../../../driver/fns/schematic.js"
|
|
4
|
+
|
|
5
|
+
export async function sampleTransition(
|
|
6
|
+
item: Item.Transition,
|
|
7
|
+
progress: number,
|
|
8
|
+
p1: Promise<Layer[]>,
|
|
9
|
+
p2: Promise<Layer[]>
|
|
10
|
+
): Promise<Layer[]> {
|
|
11
|
+
const [l1, l2] = await Promise.all([p1, p2])
|
|
12
|
+
const f1 = l1.find(l => l.kind === "image")?.frame
|
|
13
|
+
const f2 = l2.find(l => l.kind === "image")?.frame
|
|
14
|
+
|
|
15
|
+
const rest = [
|
|
16
|
+
...l1.filter(l => l.kind !== "image"),
|
|
17
|
+
...l2.filter(l => l.kind !== "image")
|
|
18
|
+
]
|
|
19
|
+
|
|
20
|
+
return f1 && f2 ? [{
|
|
21
|
+
id: item.id,
|
|
22
|
+
kind: "transition",
|
|
23
|
+
name: "circle",
|
|
24
|
+
progress,
|
|
25
|
+
from: f1,
|
|
26
|
+
to: f2
|
|
27
|
+
}, ...rest] : rest
|
|
28
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
|
|
2
|
+
import {VideoSampler} from "./defaults.js"
|
|
3
|
+
import {Item} from "../../../../../parts/item.js"
|
|
4
|
+
import {TimelineFile} from "../../../../../parts/basics.js"
|
|
5
|
+
|
|
6
|
+
export type SampleContext = {
|
|
7
|
+
videoSampler: VideoSampler
|
|
8
|
+
timeline: TimelineFile
|
|
9
|
+
items: Map<number, Item.Any>
|
|
10
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
|
|
2
|
+
import {VideoSink} from "./parts/sink.js"
|
|
3
|
+
import {sampleVisual} from "./parts/sample.js"
|
|
4
|
+
import {Ms} from "../../../../../units/ms.js"
|
|
5
|
+
import {TimelineFile} from "../../../../parts/basics.js"
|
|
6
|
+
import {DecoderSource} from "../../../../../driver/fns/schematic.js"
|
|
7
|
+
import {createDefaultVideoSampler, VideoSampler} from "./parts/defaults.js"
|
|
8
|
+
|
|
9
|
+
export function createVisualSampler(
|
|
10
|
+
resolveMedia: (hash: string) => DecoderSource,
|
|
11
|
+
sampleVideo?: VideoSampler
|
|
12
|
+
) {
|
|
13
|
+
const sink = new VideoSink(resolveMedia)
|
|
14
|
+
const videoSampler = sampleVideo ?? createDefaultVideoSampler(sink)
|
|
15
|
+
|
|
16
|
+
return {
|
|
17
|
+
async sample(timeline: TimelineFile, timecode: Ms) {
|
|
18
|
+
const items = new Map(timeline.items.map(item => [item.id, item]))
|
|
19
|
+
const root = items.get(timeline.rootId)
|
|
20
|
+
|
|
21
|
+
if (!root)
|
|
22
|
+
return []
|
|
23
|
+
|
|
24
|
+
return sampleVisual({videoSampler, timeline, items}, root, timecode, [])
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
|
|
2
|
+
import {Pub, pub} from "@e280/stz"
|
|
3
|
+
|
|
4
|
+
import {ms, Ms} from "../../../units/ms.js"
|
|
5
|
+
import {fps, Fps} from "../../../units/fps.js"
|
|
6
|
+
|
|
7
|
+
export type RealtimeGenerator = {
|
|
8
|
+
play(): void
|
|
9
|
+
pause(): void
|
|
10
|
+
setFPS(v: Fps): void
|
|
11
|
+
isPlaying(): boolean
|
|
12
|
+
ticks(): AsyncGenerator<void>
|
|
13
|
+
onTick: Pub<[]>
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export const realtime = (): RealtimeGenerator => {
|
|
17
|
+
|
|
18
|
+
let playing = false
|
|
19
|
+
let frameRate = fps(60)
|
|
20
|
+
let frameDuration = 1000 / frameRate
|
|
21
|
+
|
|
22
|
+
let lastNow = 0
|
|
23
|
+
let lastComposite = 0
|
|
24
|
+
|
|
25
|
+
const onTick = pub()
|
|
26
|
+
let resolveTick: (() => void) | null = null
|
|
27
|
+
|
|
28
|
+
const loop = (now: number) => {
|
|
29
|
+
requestAnimationFrame(loop)
|
|
30
|
+
|
|
31
|
+
if (!playing) return
|
|
32
|
+
|
|
33
|
+
lastNow = now
|
|
34
|
+
|
|
35
|
+
while (now - lastComposite >= frameDuration) {
|
|
36
|
+
lastComposite += frameDuration
|
|
37
|
+
|
|
38
|
+
resolveTick?.()
|
|
39
|
+
resolveTick = null
|
|
40
|
+
onTick()
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
async function* ticks(): AsyncGenerator<void> {
|
|
45
|
+
lastNow = performance.now()
|
|
46
|
+
lastComposite = lastNow
|
|
47
|
+
requestAnimationFrame(loop)
|
|
48
|
+
|
|
49
|
+
while (true) {
|
|
50
|
+
await new Promise<void>(r => resolveTick = r)
|
|
51
|
+
yield
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
return {
|
|
57
|
+
play() {
|
|
58
|
+
if (playing) return
|
|
59
|
+
playing = true
|
|
60
|
+
lastNow = performance.now()
|
|
61
|
+
lastComposite = lastNow
|
|
62
|
+
},
|
|
63
|
+
pause() {
|
|
64
|
+
playing = false
|
|
65
|
+
},
|
|
66
|
+
setFPS(v: Fps) {
|
|
67
|
+
frameRate = v
|
|
68
|
+
frameDuration = 1000 / frameRate
|
|
69
|
+
},
|
|
70
|
+
isPlaying() {
|
|
71
|
+
return playing
|
|
72
|
+
},
|
|
73
|
+
ticks,
|
|
74
|
+
onTick
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
export type FixedStepOptions = {
|
|
79
|
+
fps: Fps
|
|
80
|
+
duration: Ms
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export const fixedStep = async (
|
|
84
|
+
opts: FixedStepOptions,
|
|
85
|
+
onFrame: (t: Ms, index: number) => Promise<void> | void
|
|
86
|
+
) => {
|
|
87
|
+
const dt = ms(1000 / opts.fps)
|
|
88
|
+
const durationInSeconds = opts.duration / 1000
|
|
89
|
+
const total = Math.ceil(durationInSeconds * opts.fps)
|
|
90
|
+
|
|
91
|
+
for (let i = 0; i < total; i++) {
|
|
92
|
+
const t = ms(i * dt)
|
|
93
|
+
await onFrame(t, i)
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
|
|
2
|
+
import {Fps} from '../../../../units/fps.js'
|
|
3
|
+
import {ms, Ms} from '../../../../units/ms.js'
|
|
4
|
+
import {Driver} from '../../../../driver/driver.js'
|
|
5
|
+
import {realtime} from '../../parts/schedulers.js'
|
|
6
|
+
import {TimelineFile} from '../../../parts/basics.js'
|
|
7
|
+
import {computeItemDuration} from '../../parts/handy.js'
|
|
8
|
+
import {seconds, Seconds} from '../../../../units/seconds.js'
|
|
9
|
+
import {CursorVisualSampler} from '../../export/parts/cursor.js'
|
|
10
|
+
import {DecoderSource} from '../../../../driver/fns/schematic.js'
|
|
11
|
+
import {createAudioSampler} from '../../parts/samplers/audio/sampler.js'
|
|
12
|
+
import {createVisualSampler} from '../../parts/samplers/visual/sampler.js'
|
|
13
|
+
|
|
14
|
+
export class Playback {
|
|
15
|
+
audioSampler
|
|
16
|
+
seekVisualSampler
|
|
17
|
+
playVisualSampler: CursorVisualSampler | null = null
|
|
18
|
+
|
|
19
|
+
#playbackStart = ms(0)
|
|
20
|
+
#audioStartSec: number | null = null
|
|
21
|
+
|
|
22
|
+
#controller = realtime()
|
|
23
|
+
onTick = this.#controller.onTick
|
|
24
|
+
|
|
25
|
+
audioContext = new AudioContext({sampleRate: 48000})
|
|
26
|
+
audioGain = this.audioContext.createGain()
|
|
27
|
+
audioNodes = new Set<AudioBufferSourceNode>()
|
|
28
|
+
#audioAbort: AbortController | null = null
|
|
29
|
+
|
|
30
|
+
constructor(
|
|
31
|
+
private driver: Driver,
|
|
32
|
+
private timeline: TimelineFile,
|
|
33
|
+
private resolveMedia: (hash: string) => DecoderSource
|
|
34
|
+
) {
|
|
35
|
+
this.audioGain.connect(this.audioContext.destination)
|
|
36
|
+
this.audioGain.gain.value = 0.7 ** 2
|
|
37
|
+
this.seekVisualSampler = createVisualSampler(this.resolveMedia)
|
|
38
|
+
this.audioSampler = createAudioSampler(this.resolveMedia)
|
|
39
|
+
this.#samples()
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
update(timeline: TimelineFile) {
|
|
43
|
+
this.timeline = timeline
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
get isPlaying() {
|
|
47
|
+
return this.#controller.isPlaying()
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
async #samples() {
|
|
51
|
+
for await (const _ of this.#controller.ticks()) {
|
|
52
|
+
const layers = await this.playVisualSampler?.next(this.currentTime) ?? []
|
|
53
|
+
|
|
54
|
+
const frame = await this.driver.composite(layers)
|
|
55
|
+
frame.close()
|
|
56
|
+
|
|
57
|
+
if (this.currentTime >= this.duration)
|
|
58
|
+
this.pause()
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
async seek(time: Ms) {
|
|
63
|
+
this.pause()
|
|
64
|
+
this.#playbackStart = time
|
|
65
|
+
return await this.seekVisualSampler.sample(this.timeline, time)
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
async start() {
|
|
69
|
+
if(this.#controller.isPlaying())
|
|
70
|
+
return
|
|
71
|
+
|
|
72
|
+
await this.audioContext.resume()
|
|
73
|
+
|
|
74
|
+
this.#playbackStart = this.currentTime
|
|
75
|
+
this.#audioStartSec = this.audioContext.currentTime
|
|
76
|
+
|
|
77
|
+
this.#audioAbort?.abort()
|
|
78
|
+
this.#audioAbort = new AbortController()
|
|
79
|
+
|
|
80
|
+
for (const node of this.audioNodes)
|
|
81
|
+
node.stop()
|
|
82
|
+
|
|
83
|
+
this.audioNodes.clear()
|
|
84
|
+
|
|
85
|
+
this.playVisualSampler = new CursorVisualSampler(this.driver, this.resolveMedia, this.timeline)
|
|
86
|
+
|
|
87
|
+
this.#controller.play()
|
|
88
|
+
this.#startAudio(this.#audioAbort.signal, seconds(this.#playbackStart / 1000))
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
pause() {
|
|
92
|
+
this.#playbackStart = this.currentTime
|
|
93
|
+
this.#controller.pause()
|
|
94
|
+
this.#audioAbort?.abort()
|
|
95
|
+
|
|
96
|
+
for (const node of this.audioNodes)
|
|
97
|
+
node.stop()
|
|
98
|
+
|
|
99
|
+
this.audioNodes.clear()
|
|
100
|
+
|
|
101
|
+
if (this.playVisualSampler) {
|
|
102
|
+
this.playVisualSampler.cancel()
|
|
103
|
+
this.playVisualSampler = null
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
get duration() {
|
|
109
|
+
return computeItemDuration(
|
|
110
|
+
this.timeline.rootId,
|
|
111
|
+
this.timeline
|
|
112
|
+
)
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
get currentTime() {
|
|
116
|
+
if (!this.#controller.isPlaying() || this.#audioStartSec === null)
|
|
117
|
+
return this.#playbackStart
|
|
118
|
+
|
|
119
|
+
const elapsedMs = (this.audioContext.currentTime - this.#audioStartSec) * 1000
|
|
120
|
+
return ms(this.#playbackStart + elapsedMs)
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
setFps(fps: Fps) {
|
|
124
|
+
this.#controller.setFPS(fps)
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
async #startAudio(signal: AbortSignal, from: Seconds) {
|
|
128
|
+
const ctx = this.audioContext
|
|
129
|
+
|
|
130
|
+
if (this.#audioStartSec === null)
|
|
131
|
+
return
|
|
132
|
+
|
|
133
|
+
for await (const {sample, timestamp} of this.audioSampler.sampleAudio(
|
|
134
|
+
this.timeline,
|
|
135
|
+
ms(from * 1000)
|
|
136
|
+
)) {
|
|
137
|
+
|
|
138
|
+
if (signal.aborted || !this.#controller.isPlaying())
|
|
139
|
+
return
|
|
140
|
+
|
|
141
|
+
while (timestamp - (ctx.currentTime - this.#audioStartSec + from) > 0.75)
|
|
142
|
+
await new Promise(r => setTimeout(r, 25))
|
|
143
|
+
|
|
144
|
+
const node = ctx.createBufferSource()
|
|
145
|
+
node.buffer = sample.toAudioBuffer()
|
|
146
|
+
node.connect(this.audioGain)
|
|
147
|
+
node.onended = () => this.audioNodes.delete(node)
|
|
148
|
+
this.audioNodes.add(node)
|
|
149
|
+
|
|
150
|
+
const startAt = this.#audioStartSec + timestamp - from
|
|
151
|
+
|
|
152
|
+
startAt >= ctx.currentTime
|
|
153
|
+
? node.start(startAt)
|
|
154
|
+
: node.start(ctx.currentTime, ctx.currentTime - startAt)
|
|
155
|
+
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
|
|
2
|
+
import {ms} from "../../../units/ms.js"
|
|
3
|
+
import {fps} from "../../../units/fps.js"
|
|
4
|
+
import {Playback} from "./parts/playback.js"
|
|
5
|
+
import {Driver} from "../../../driver/driver.js"
|
|
6
|
+
import {TimelineFile} from "../../parts/basics.js"
|
|
7
|
+
import {DecoderSource} from "../../../driver/fns/schematic.js"
|
|
8
|
+
|
|
9
|
+
type ResolveMedia = (hash: string) => DecoderSource
|
|
10
|
+
|
|
11
|
+
export class VideoPlayer {
|
|
12
|
+
canvas: HTMLCanvasElement
|
|
13
|
+
playback: Playback
|
|
14
|
+
|
|
15
|
+
#pendingSeek: number | null = null
|
|
16
|
+
#flushTask: Promise<void> | null = null
|
|
17
|
+
|
|
18
|
+
constructor(
|
|
19
|
+
private driver: Driver,
|
|
20
|
+
resolveMedia: ResolveMedia,
|
|
21
|
+
timeline: TimelineFile
|
|
22
|
+
) {
|
|
23
|
+
this.playback = new Playback(driver, timeline, resolveMedia)
|
|
24
|
+
this.canvas = driver.compositor.pixi.renderer.canvas
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
async play() {
|
|
28
|
+
await this.playback.start()
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
pause() {
|
|
32
|
+
this.playback.pause()
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
seek(timeMs: number) {
|
|
36
|
+
this.#pendingSeek = timeMs
|
|
37
|
+
return this.#flushTask ??= this.#flushSeeks().finally(() => this.#flushTask = null)
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
setFPS(value: number) {
|
|
41
|
+
this.playback.setFps(fps(value))
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
get isSeeking() {
|
|
45
|
+
return this.#flushTask !== null
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
get isPlaying() {
|
|
49
|
+
return this.playback.isPlaying
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
get duration() {
|
|
53
|
+
return this.playback.duration
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
get currentTime() {
|
|
57
|
+
return this.playback.currentTime
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
call this whenever your timeline state changes
|
|
62
|
+
*/
|
|
63
|
+
update(timeline: TimelineFile) {
|
|
64
|
+
this.playback.update(timeline)
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
async #flushSeeks() {
|
|
68
|
+
while (this.#pendingSeek !== null) {
|
|
69
|
+
const next = this.#pendingSeek
|
|
70
|
+
this.#pendingSeek = null
|
|
71
|
+
const layers = await this.playback.seek(ms(next))
|
|
72
|
+
const frame = await this.driver.composite(layers)
|
|
73
|
+
frame.close()
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|