@omnimedia/omnitool 1.1.0-53 → 1.1.0-56

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 (71) hide show
  1. package/package.json +1 -1
  2. package/s/demo/demo.bundle.ts +5 -3
  3. package/s/demo/routines/transcode-test.ts +2 -2
  4. package/s/demo/routines/transitions-test.ts +2 -2
  5. package/s/demo/routines/waveform-test.ts +30 -5
  6. package/s/driver/driver.ts +26 -5
  7. package/s/driver/fns/schematic.ts +2 -0
  8. package/s/driver/fns/work.ts +41 -20
  9. package/s/index.html.ts +1 -2
  10. package/s/timeline/parts/waveform/parts/collect.ts +72 -0
  11. package/s/timeline/parts/waveform/parts/render.ts +45 -0
  12. package/s/timeline/parts/waveform/parts/types.ts +24 -0
  13. package/s/timeline/parts/waveform/waveform.ts +152 -0
  14. package/s/timeline/parts/waveform.ts +0 -61
  15. package/s/timeline/renderers/export/parts/cursor.ts +75 -56
  16. package/s/timeline/renderers/export/parts/produce-video.ts +2 -3
  17. package/s/timeline/renderers/player/parts/playback.ts +33 -6
  18. package/s/timeline/renderers/player/player.ts +2 -14
  19. package/s/timeline/renderers/renderers.test.ts +1 -3
  20. package/x/demo/demo.bundle.js +2 -0
  21. package/x/demo/demo.bundle.js.map +1 -1
  22. package/x/demo/demo.bundle.min.js +11 -84
  23. package/x/demo/demo.bundle.min.js.map +4 -4
  24. package/x/demo/routines/transcode-test.js +2 -2
  25. package/x/demo/routines/transcode-test.js.map +1 -1
  26. package/x/demo/routines/transitions-test.js +2 -2
  27. package/x/demo/routines/transitions-test.js.map +1 -1
  28. package/x/demo/routines/waveform-test.js +24 -4
  29. package/x/demo/routines/waveform-test.js.map +1 -1
  30. package/x/driver/driver.d.ts +14 -2
  31. package/x/driver/driver.js +26 -5
  32. package/x/driver/driver.js.map +1 -1
  33. package/x/driver/driver.worker.bundle.min.js +1 -1
  34. package/x/driver/driver.worker.bundle.min.js.map +3 -3
  35. package/x/driver/fns/host.d.ts +2 -0
  36. package/x/driver/fns/schematic.d.ts +2 -0
  37. package/x/driver/fns/work.d.ts +2 -0
  38. package/x/driver/fns/work.js +33 -20
  39. package/x/driver/fns/work.js.map +1 -1
  40. package/x/index.html +3 -3
  41. package/x/index.html.js +1 -1
  42. package/x/tests.bundle.min.js +16 -16
  43. package/x/tests.bundle.min.js.map +4 -4
  44. package/x/tests.html +1 -1
  45. package/x/timeline/parts/waveform/parts/collect.d.ts +11 -0
  46. package/x/timeline/parts/waveform/parts/collect.js +56 -0
  47. package/x/timeline/parts/waveform/parts/collect.js.map +1 -0
  48. package/x/timeline/parts/waveform/parts/render.d.ts +5 -0
  49. package/x/timeline/parts/waveform/parts/render.js +29 -0
  50. package/x/timeline/parts/waveform/parts/render.js.map +1 -0
  51. package/x/timeline/parts/waveform/parts/types.d.ts +21 -0
  52. package/x/timeline/parts/waveform/parts/types.js +2 -0
  53. package/x/timeline/parts/waveform/parts/types.js.map +1 -0
  54. package/x/timeline/parts/waveform/waveform.d.ts +17 -0
  55. package/x/timeline/parts/waveform/waveform.js +125 -0
  56. package/x/timeline/parts/waveform/waveform.js.map +1 -0
  57. package/x/timeline/parts/waveform.d.ts +1 -9
  58. package/x/timeline/parts/waveform.js +1 -48
  59. package/x/timeline/parts/waveform.js.map +1 -1
  60. package/x/timeline/renderers/export/parts/cursor.d.ts +4 -5
  61. package/x/timeline/renderers/export/parts/cursor.js +68 -50
  62. package/x/timeline/renderers/export/parts/cursor.js.map +1 -1
  63. package/x/timeline/renderers/export/parts/produce-video.js +2 -3
  64. package/x/timeline/renderers/export/parts/produce-video.js.map +1 -1
  65. package/x/timeline/renderers/player/parts/playback.d.ts +9 -5
  66. package/x/timeline/renderers/player/parts/playback.js +25 -6
  67. package/x/timeline/renderers/player/parts/playback.js.map +1 -1
  68. package/x/timeline/renderers/player/player.js +2 -9
  69. package/x/timeline/renderers/player/player.js.map +1 -1
  70. package/x/timeline/renderers/renderers.test.js +1 -2
  71. package/x/timeline/renderers/renderers.test.js.map +1 -1
package/x/tests.html CHANGED
@@ -295,7 +295,7 @@ canvas {
295
295
  }
296
296
  }
297
297
  </style>
298
- <script type=module src="tests.bundle.min.js?v=74c60c4e029b"></script>
298
+ <script type=module src="tests.bundle.min.js?v=e7a2a7297b62"></script>
299
299
 
300
300
 
301
301
  <meta name="theme-color" content="#3cff9c">
@@ -0,0 +1,11 @@
1
+ import { Driver } from "../../../../driver/driver.js";
2
+ import { DecoderSource } from "../../../../driver/fns/schematic.js";
3
+ export declare const PEAK_LEVELS: readonly [2048, 1024, 512, 256, 128, 64, 32];
4
+ export declare function collectPeakLevels(driver: Driver, source: DecoderSource): Promise<{
5
+ duration: number;
6
+ levels: {
7
+ samplesPerPeak: 1024 | 32 | 2048 | 512 | 256 | 128 | 64;
8
+ peaks: Float32Array<ArrayBuffer>;
9
+ peaksPerSecond: number;
10
+ }[];
11
+ }>;
@@ -0,0 +1,56 @@
1
+ export const PEAK_LEVELS = [2048, 1024, 512, 256, 128, 64, 32];
2
+ export async function collectPeakLevels(driver, source) {
3
+ const duration = (await driver.getAudioDuration(source)) ?? 0;
4
+ const readable = driver.decodeAudio({ source }).readable;
5
+ const finestSamplesPerPeak = PEAK_LEVELS[PEAK_LEVELS.length - 1];
6
+ const finestPeaks = [];
7
+ let currentMax = 0;
8
+ let sampleCount = 0;
9
+ let sampleRate = 0;
10
+ for await (const audioData of readable) {
11
+ sampleRate ||= audioData.sampleRate;
12
+ const frames = audioData.numberOfFrames;
13
+ const plane = new Float32Array(frames);
14
+ audioData.copyTo(plane, { planeIndex: 0 });
15
+ for (let i = 0; i < plane.length; i++) {
16
+ const amplitude = Math.abs(plane[i]);
17
+ if (amplitude > currentMax)
18
+ currentMax = amplitude;
19
+ sampleCount++;
20
+ if (sampleCount >= finestSamplesPerPeak) {
21
+ finestPeaks.push(currentMax);
22
+ currentMax = 0;
23
+ sampleCount = 0;
24
+ }
25
+ }
26
+ audioData.close();
27
+ }
28
+ if (sampleCount > 0)
29
+ finestPeaks.push(currentMax);
30
+ const base = new Float32Array(finestPeaks);
31
+ const levels = PEAK_LEVELS.map(samplesPerPeak => {
32
+ const factor = Math.max(1, Math.round(samplesPerPeak / finestSamplesPerPeak));
33
+ const peaks = factor === 1 ? base : downsampleMax(base, factor);
34
+ return {
35
+ samplesPerPeak,
36
+ peaks,
37
+ peaksPerSecond: sampleRate > 0 ? sampleRate / samplesPerPeak : 0,
38
+ };
39
+ });
40
+ return { duration, levels };
41
+ }
42
+ function downsampleMax(peaks, factor) {
43
+ const downsampled = new Float32Array(Math.ceil(peaks.length / factor));
44
+ for (let i = 0; i < downsampled.length; i++) {
45
+ let maxPeak = 0;
46
+ const start = i * factor;
47
+ const end = Math.min(start + factor, peaks.length);
48
+ for (let j = start; j < end; j++) {
49
+ if (peaks[j] > maxPeak)
50
+ maxPeak = peaks[j];
51
+ }
52
+ downsampled[i] = maxPeak;
53
+ }
54
+ return downsampled;
55
+ }
56
+ //# sourceMappingURL=collect.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"collect.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/waveform/parts/collect.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export declare function renderTile(peaks: Float32Array, opts: {
2
+ width: number;
3
+ height: number;
4
+ color: string;
5
+ }): HTMLCanvasElement;
@@ -0,0 +1,29 @@
1
+ export function renderTile(peaks, opts) {
2
+ const dpr = typeof window !== "undefined" ? window.devicePixelRatio || 1 : 1;
3
+ const canvas = document.createElement("canvas");
4
+ canvas.width = opts.width * dpr;
5
+ canvas.height = opts.height * dpr;
6
+ canvas.style.width = `${opts.width}px`;
7
+ canvas.style.height = `${opts.height}px`;
8
+ const ctx = canvas.getContext("2d");
9
+ if (!ctx)
10
+ return canvas;
11
+ ctx.scale(dpr, dpr);
12
+ ctx.fillStyle = opts.color;
13
+ const centerY = opts.height / 2;
14
+ const columns = Math.max(1, opts.width);
15
+ const peaksPerPixel = peaks.length / columns;
16
+ for (let px = 0; px < columns; px++) {
17
+ const startIndex = Math.floor(px * peaksPerPixel);
18
+ const endIndex = Math.max(startIndex + 1, Math.floor((px + 1) * peaksPerPixel));
19
+ let maxPeak = 0;
20
+ for (let i = startIndex; i < endIndex && i < peaks.length; i++) {
21
+ if (peaks[i] > maxPeak)
22
+ maxPeak = peaks[i];
23
+ }
24
+ const barHeight = maxPeak * opts.height;
25
+ ctx.fillRect(px, centerY - barHeight / 2, 1, barHeight);
26
+ }
27
+ return canvas;
28
+ }
29
+ //# sourceMappingURL=render.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"render.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/waveform/parts/render.ts"],"names":[],"mappings":"AACA,MAAM,UAAU,UAAU,CACzB,KAAmB,EACnB,IAIC;IAED,MAAM,GAAG,GAAG,OAAO,MAAM,KAAK,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,gBAAgB,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;IAC5E,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAA;IAC/C,MAAM,CAAC,KAAK,GAAG,IAAI,CAAC,KAAK,GAAG,GAAG,CAAA;IAC/B,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,GAAG,GAAG,CAAA;IACjC,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,IAAI,CAAC,KAAK,IAAI,CAAA;IACtC,MAAM,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,IAAI,CAAA;IAExC,MAAM,GAAG,GAAG,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAA;IACnC,IAAI,CAAC,GAAG;QAAE,OAAO,MAAM,CAAA;IAEvB,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,GAAG,CAAC,CAAA;IACnB,GAAG,CAAC,SAAS,GAAG,IAAI,CAAC,KAAK,CAAA;IAE1B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAA;IAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,CAAA;IACvC,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,GAAG,OAAO,CAAA;IAE5C,KAAK,IAAI,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,OAAO,EAAE,EAAE,EAAE,EAAE,CAAC;QACrC,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,GAAG,aAAa,CAAC,CAAA;QACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CACxB,UAAU,GAAG,CAAC,EACd,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,GAAG,aAAa,CAAC,CACpC,CAAA;QAED,IAAI,OAAO,GAAG,CAAC,CAAA;QACf,KAAK,IAAI,CAAC,GAAG,UAAU,EAAE,CAAC,GAAG,QAAQ,IAAI,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAChE,IAAI,KAAK,CAAC,CAAC,CAAE,GAAG,OAAO;gBAAE,OAAO,GAAG,KAAK,CAAC,CAAC,CAAE,CAAA;QAC7C,CAAC;QAED,MAAM,SAAS,GAAG,OAAO,GAAG,IAAI,CAAC,MAAM,CAAA;QACvC,GAAG,CAAC,QAAQ,CAAC,EAAE,EAAE,OAAO,GAAG,SAAS,GAAG,CAAC,EAAE,CAAC,EAAE,SAAS,CAAC,CAAA;IACxD,CAAC;IAED,OAAO,MAAM,CAAA;AACd,CAAC"}
@@ -0,0 +1,21 @@
1
+ export interface WaveformTileData {
2
+ startTime: number;
3
+ endTime: number;
4
+ peaks: Float32Array;
5
+ canvas: HTMLCanvasElement;
6
+ }
7
+ export interface WaveformOptions {
8
+ tileSize?: number;
9
+ zoom?: number;
10
+ tileWidth?: number;
11
+ tileHeight?: number;
12
+ preloadMargin?: number;
13
+ color?: string;
14
+ onChange?: (tiles: WaveformTileData[]) => void;
15
+ }
16
+ export type WaveformTimeRange = [start: number, end: number];
17
+ export type WaveformPeakLevel = {
18
+ samplesPerPeak: number;
19
+ peaks: Float32Array;
20
+ peaksPerSecond: number;
21
+ };
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/waveform/parts/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,17 @@
1
+ import { Driver } from "../../../driver/driver.js";
2
+ import { DecoderSource } from "../../../driver/fns/schematic.js";
3
+ import { WaveformOptions, WaveformTileData, WaveformTimeRange } from "./parts/types.js";
4
+ export declare class Waveform {
5
+ #private;
6
+ readonly color: string;
7
+ readonly duration: number;
8
+ readonly tileSize: number;
9
+ readonly tileHeight: number;
10
+ readonly preloadMargin: number;
11
+ private constructor();
12
+ static init(driver: Driver, source: DecoderSource, options?: WaveformOptions): Promise<Waveform>;
13
+ set zoom(value: number);
14
+ get zoom(): number;
15
+ set range(visibleRange: WaveformTimeRange);
16
+ getTiles(): Map<number, WaveformTileData>;
17
+ }
@@ -0,0 +1,125 @@
1
+ import { renderTile } from "./parts/render.js";
2
+ import { collectPeakLevels } from "./parts/collect.js";
3
+ const MAX_TILE_WIDTH = 4096;
4
+ export class Waveform {
5
+ #tiles = new Map();
6
+ #activeRange = [0, 0];
7
+ #zoom;
8
+ #levels;
9
+ #onChange;
10
+ #updateQueued = false;
11
+ color;
12
+ duration;
13
+ tileSize;
14
+ tileHeight;
15
+ preloadMargin;
16
+ constructor(levels, duration, options) {
17
+ this.#levels = levels;
18
+ this.duration = duration;
19
+ this.tileSize = options.tileSize ?? 1;
20
+ this.#zoom = options.zoom ?? ((options.tileWidth ?? 256) / this.tileSize);
21
+ this.tileHeight = options.tileHeight ?? 96;
22
+ this.preloadMargin = options.preloadMargin ?? 2;
23
+ this.color = options.color ?? "rgb(3, 148, 129)";
24
+ this.#onChange = options.onChange;
25
+ }
26
+ static async init(driver, source, options = {}) {
27
+ const { duration, levels } = await collectPeakLevels(driver, source);
28
+ return new Waveform(levels, duration, options);
29
+ }
30
+ set zoom(value) {
31
+ const next = Math.max(1, value);
32
+ if (next === this.#zoom)
33
+ return;
34
+ this.#zoom = next;
35
+ this.#tiles.clear();
36
+ this.#queueUpdate();
37
+ }
38
+ get zoom() {
39
+ return this.#zoom;
40
+ }
41
+ #computeActiveRange([start, end], margin = 1) {
42
+ const visibleSize = end - start;
43
+ return [
44
+ Math.max(0, start - visibleSize * margin),
45
+ Math.min(this.duration, end + visibleSize * margin),
46
+ ];
47
+ }
48
+ set range(visibleRange) {
49
+ const [visibleStart, visibleEnd] = visibleRange;
50
+ const visibleSize = visibleEnd - visibleStart;
51
+ const [activeStart, activeEnd] = this.#activeRange;
52
+ const leftTrigger = activeStart + visibleSize;
53
+ const rightTrigger = activeEnd - visibleSize;
54
+ if (visibleStart >= leftTrigger && visibleEnd <= rightTrigger)
55
+ return;
56
+ this.#activeRange = this.#computeActiveRange(visibleRange, this.preloadMargin);
57
+ this.#queueUpdate();
58
+ }
59
+ #queueUpdate() {
60
+ if (this.#updateQueued)
61
+ return;
62
+ this.#updateQueued = true;
63
+ queueMicrotask(() => {
64
+ this.#updateQueued = false;
65
+ this.#generateTiles();
66
+ });
67
+ }
68
+ #generateTiles() {
69
+ const [rangeStart, rangeEnd] = this.#activeRange;
70
+ const neededStarts = new Set();
71
+ const level = this.#levelForZoom();
72
+ const firstStart = Math.max(0, Math.floor(rangeStart / this.tileSize) * this.tileSize);
73
+ const lastStart = Math.min(this.duration, rangeEnd);
74
+ for (let startTime = firstStart; startTime <= lastStart; startTime += this.tileSize) {
75
+ neededStarts.add(startTime);
76
+ }
77
+ for (const startTime of neededStarts) {
78
+ if (!this.#tiles.has(startTime)) {
79
+ const endTime = Math.min(startTime + this.tileSize, this.duration);
80
+ this.#tiles.set(startTime, this.#buildTileData(startTime, endTime, level));
81
+ }
82
+ }
83
+ for (const startTime of this.#tiles.keys()) {
84
+ if (!neededStarts.has(startTime))
85
+ this.#tiles.delete(startTime);
86
+ }
87
+ this.#emit();
88
+ }
89
+ #buildTileData(startTime, endTime, level) {
90
+ const peaks = this.#slicePeaks(level, startTime, endTime);
91
+ return {
92
+ startTime,
93
+ endTime,
94
+ peaks,
95
+ canvas: renderTile(peaks, {
96
+ width: this.#tilePixelWidth(startTime, endTime),
97
+ height: this.tileHeight,
98
+ color: this.color,
99
+ }),
100
+ };
101
+ }
102
+ #levelForZoom() {
103
+ return this.#levels.find(level => level.peaksPerSecond >= this.#zoom)
104
+ ?? this.#levels[this.#levels.length - 1];
105
+ }
106
+ #slicePeaks(level, startTime, endTime) {
107
+ if (!level.peaksPerSecond)
108
+ return new Float32Array();
109
+ const from = Math.max(0, Math.floor(startTime * level.peaksPerSecond));
110
+ const to = Math.max(from + 1, Math.min(level.peaks.length, Math.ceil(endTime * level.peaksPerSecond)));
111
+ return level.peaks.slice(from, to);
112
+ }
113
+ #tilePixelWidth(startTime, endTime) {
114
+ return Math.min(MAX_TILE_WIDTH, Math.max(1, Math.ceil((endTime - startTime) * this.#zoom)));
115
+ }
116
+ #emit() {
117
+ if (!this.#onChange)
118
+ return;
119
+ this.#onChange([...this.#tiles.values()].sort((a, b) => a.startTime - b.startTime));
120
+ }
121
+ getTiles() {
122
+ return this.#tiles;
123
+ }
124
+ }
125
+ //# sourceMappingURL=waveform.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"waveform.js","sourceRoot":"","sources":["../../../../s/timeline/parts/waveform/waveform.ts"],"names":[],"mappings":"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"}
@@ -1,9 +1 @@
1
- import WaveSurfer from "wavesurfer.js";
2
- import { Driver } from "../../driver/driver.js";
3
- import { DecoderSource } from "../../driver/fns/schematic.js";
4
- export declare class Waveform {
5
- wavesurfer: WaveSurfer;
6
- constructor(peaks: number[], container: HTMLElement, duration: number);
7
- static init(driver: Driver, source: DecoderSource, container: HTMLElement): Promise<Waveform>;
8
- set width(value: number);
9
- }
1
+ export {};
@@ -1,49 +1,2 @@
1
- import WaveSurfer from "wavesurfer.js";
2
- export class Waveform {
3
- wavesurfer;
4
- constructor(peaks, container, duration) {
5
- this.wavesurfer = WaveSurfer.create({
6
- container,
7
- waveColor: 'rgb(200, 0, 200)',
8
- progressColor: 'rgb(100, 0, 100)',
9
- barWidth: 10,
10
- barRadius: 10,
11
- barGap: 2,
12
- peaks: [peaks],
13
- duration
14
- });
15
- }
16
- static async init(driver, source, container) {
17
- const reader = driver.decodeAudio({ source }).getReader();
18
- const peaks = [];
19
- let buffer = [];
20
- const samplesPerPeak = 1024;
21
- const duration = await driver.getAudioDuration(source);
22
- while (true) {
23
- const { done, value: audioData } = await reader.read();
24
- if (done)
25
- break;
26
- const frames = audioData.numberOfFrames;
27
- const plane = new Float32Array(frames);
28
- audioData.copyTo(plane, { planeIndex: 0 }); // Use left channel only
29
- for (let i = 0; i < plane.length; i++) {
30
- buffer.push(plane[i]);
31
- if (buffer.length >= samplesPerPeak) {
32
- const chunk = buffer.splice(0, samplesPerPeak);
33
- const min = Math.min(...chunk);
34
- const max = Math.max(...chunk);
35
- peaks.push(min, max);
36
- }
37
- }
38
- audioData.close();
39
- }
40
- return new Waveform(peaks, container, duration ?? 0);
41
- }
42
- // set zoom(value: number) {
43
- // this.wavesurfer.zoom(value)
44
- // }
45
- set width(value) {
46
- this.wavesurfer.setOptions({ width: value });
47
- }
48
- }
1
+ export {};
49
2
  //# sourceMappingURL=waveform.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"waveform.js","sourceRoot":"","sources":["../../../s/timeline/parts/waveform.ts"],"names":[],"mappings":"AAAA,OAAO,UAAU,MAAM,eAAe,CAAA;AAKtC,MAAM,OAAO,QAAQ;IACpB,UAAU,CAAY;IAEtB,YAAY,KAAe,EAAE,SAAsB,EAAE,QAAgB;QACpE,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC,MAAM,CAAC;YAClC,SAAS;YACT,SAAS,EAAE,kBAAkB;YAC7B,aAAa,EAAE,kBAAkB;YACjC,QAAQ,EAAE,EAAE;YACZ,SAAS,EAAE,EAAE;YACb,MAAM,EAAE,CAAC;YACT,KAAK,EAAE,CAAC,KAAK,CAAC;YACd,QAAQ;SACT,CAAC,CAAA;IACH,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAc,EAAE,MAAqB,EAAE,SAAsB;QAC9E,MAAM,MAAM,GAAG,MAAM,CAAC,WAAW,CAAC,EAAC,MAAM,EAAC,CAAC,CAAC,SAAS,EAAE,CAAA;QAEvD,MAAM,KAAK,GAAa,EAAE,CAAA;QAC1B,IAAI,MAAM,GAAa,EAAE,CAAA;QACzB,MAAM,cAAc,GAAG,IAAI,CAAA;QAC3B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAA;QAEtD,OAAO,IAAI,EAAE,CAAC;YACb,MAAM,EAAC,IAAI,EAAE,KAAK,EAAE,SAAS,EAAC,GAAG,MAAM,MAAM,CAAC,IAAI,EAAE,CAAA;YACpD,IAAI,IAAI;gBAAE,MAAK;YAEf,MAAM,MAAM,GAAG,SAAS,CAAC,cAAc,CAAA;YACvC,MAAM,KAAK,GAAG,IAAI,YAAY,CAAC,MAAM,CAAC,CAAA;YACtC,SAAS,CAAC,MAAM,CAAC,KAAK,EAAE,EAAC,UAAU,EAAE,CAAC,EAAC,CAAC,CAAA,CAAC,wBAAwB;YAEjE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACvC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAA;gBACrB,IAAI,MAAM,CAAC,MAAM,IAAI,cAAc,EAAE,CAAC;oBACrC,MAAM,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,EAAE,cAAc,CAAC,CAAA;oBAC9C,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,CAAA;oBAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,CAAA;oBAC9B,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,CAAA;gBACrB,CAAC;YACF,CAAC;YAED,SAAS,CAAC,KAAK,EAAE,CAAA;QAClB,CAAC;QAED,OAAO,IAAI,QAAQ,CAAC,KAAK,EAAE,SAAS,EAAE,QAAQ,IAAI,CAAC,CAAC,CAAA;IACrD,CAAC;IAED,4BAA4B;IAC5B,+BAA+B;IAC/B,IAAI;IAEJ,IAAI,KAAK,CAAC,KAAa;QACtB,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAC,KAAK,EAAE,KAAK,EAAC,CAAC,CAAA;IAC3C,CAAC;CACD"}
1
+ {"version":3,"file":"waveform.js","sourceRoot":"","sources":["../../../s/timeline/parts/waveform.ts"],"names":[],"mappings":""}
@@ -11,9 +11,8 @@ export declare class CursorVisualSampler {
11
11
  #private;
12
12
  private driver;
13
13
  private resolveMedia;
14
- constructor(driver: Driver, resolveMedia: (hash: string) => DecoderSource);
15
- cursor(timeline: TimelineFile): {
16
- next: (timecode: Ms) => Promise<import("../../../../driver/fns/schematic.js").Layer[]>;
17
- cancel: () => Promise<void>;
18
- };
14
+ private timeline;
15
+ constructor(driver: Driver, resolveMedia: (hash: string) => DecoderSource, timeline: TimelineFile);
16
+ next(timecode: Ms): Promise<import("../../../../driver/fns/schematic.js").Layer[]>;
17
+ cancel(): Promise<void>;
19
18
  }
@@ -8,71 +8,89 @@ import { createVisualSampler } from "../../parts/samplers/visual/sampler.js";
8
8
  export class CursorVisualSampler {
9
9
  driver;
10
10
  resolveMedia;
11
- #sampler;
11
+ timeline;
12
+ #lastTimecode = -Infinity;
12
13
  #videoCursors = new Map();
13
- constructor(driver, resolveMedia) {
14
+ #sampler;
15
+ constructor(driver, resolveMedia, timeline) {
14
16
  this.driver = driver;
15
17
  this.resolveMedia = resolveMedia;
16
- this.#sampler = createVisualSampler(resolveMedia, (item, time) => {
17
- const mediaTime = toUs(ms(item.start + time));
18
- const cursor = this.#getCursorForVideo(item);
19
- return cursor.next(mediaTime);
18
+ this.timeline = timeline;
19
+ this.#sampler = createVisualSampler(this.resolveMedia, (item, time) => {
20
+ const targetUs = toUs(time);
21
+ let cursor = this.#videoCursors.get(item.id);
22
+ if (!cursor) {
23
+ const source = this.resolveMedia(item.mediaHash);
24
+ const endUs = toUs(ms(item.start + item.duration));
25
+ cursor = this.#createVideoCursor(source, targetUs, endUs);
26
+ this.#videoCursors.set(item.id, cursor);
27
+ }
28
+ return cursor.next(targetUs);
20
29
  });
21
30
  }
22
- cursor(timeline) {
23
- let lastTimecode = Number.NEGATIVE_INFINITY;
24
- return {
25
- next: (timecode) => {
26
- if (timecode < lastTimecode)
27
- throw new Error(`CursorVisualSampler is forward-only: requested ${timecode}ms after ${lastTimecode}ms`);
28
- lastTimecode = timecode;
29
- return this.#sampler.sample(timeline, timecode);
30
- },
31
- cancel: () => this.#cancel(),
32
- };
33
- }
34
- #getCursorForVideo(videoItem) {
35
- const existing = this.#videoCursors.get(videoItem.id);
36
- if (existing)
37
- return existing;
38
- const source = this.resolveMedia(videoItem.mediaHash);
39
- const video = this.driver.decodeVideo({ source });
40
- const cursor = this.#cursor(video.getReader());
41
- this.#videoCursors.set(videoItem.id, cursor);
42
- return cursor;
31
+ next(timecode) {
32
+ if (timecode < this.#lastTimecode)
33
+ throw new Error(`Forward-only cursor regression: ${timecode}ms < ${this.#lastTimecode}ms`);
34
+ this.#lastTimecode = timecode;
35
+ return this.#sampler.sample(this.timeline, timecode);
43
36
  }
44
- async #cancel() {
45
- await Promise.all([...this.#videoCursors.values()].map(cursor => cursor.cancel()));
37
+ async cancel() {
38
+ await Promise.all([...this.#videoCursors.values()].map(c => c.cancel()));
46
39
  this.#videoCursors.clear();
47
40
  }
48
- // forward only
49
- #cursor(reader) {
41
+ #createVideoCursor(source, startUs, endUs) {
42
+ const video = this.driver.decodeVideo({ source, start: startUs / 1_000_000, end: endUs / 1_000_000 });
43
+ const reader = video.readable.getReader();
44
+ let current = null;
45
+ let nextPromise = null;
46
+ let ended = false;
47
+ const readNext = async () => {
48
+ if (ended)
49
+ return null;
50
+ const { done, value } = await reader.read();
51
+ if (done)
52
+ return (ended = true, null);
53
+ const frame = new VideoFrame(value);
54
+ value.close();
55
+ return frame;
56
+ };
50
57
  return {
51
58
  async next(targetUs) {
52
- let prev = null;
59
+ current ??= await readNext();
60
+ if (!current)
61
+ return undefined;
53
62
  while (true) {
54
- const { done, value: hit } = await reader.read();
55
- if (done) {
56
- const out = prev ? new VideoFrame(prev) : undefined;
57
- prev?.close();
58
- return out;
63
+ nextPromise ??= readNext();
64
+ const nextFrame = await nextPromise;
65
+ if (!nextFrame)
66
+ return new VideoFrame(current);
67
+ const currentUs = current.timestamp ?? -Infinity;
68
+ const nextUs = nextFrame.timestamp ?? currentUs;
69
+ if (nextUs < targetUs) {
70
+ current.close();
71
+ current = nextFrame;
72
+ nextPromise = null;
73
+ continue;
59
74
  }
60
- const hitUs = hit.timestamp ?? 0;
61
- if (hitUs >= targetUs) {
62
- const prevUs = prev?.timestamp ?? Number.NEGATIVE_INFINITY;
63
- const usePrev = !!prev && Math.abs(prevUs - targetUs) < Math.abs(hitUs - targetUs);
64
- const chosen = usePrev ? prev : hit;
65
- const other = usePrev ? hit : prev;
66
- const copy = new VideoFrame(chosen);
67
- chosen.close();
68
- other?.close();
69
- return copy;
75
+ const useNext = Math.abs(nextUs - targetUs) < Math.abs(currentUs - targetUs);
76
+ if (useNext) {
77
+ current.close();
78
+ current = nextFrame;
79
+ nextPromise = null;
80
+ continue;
70
81
  }
71
- prev?.close();
72
- prev = hit;
82
+ return new VideoFrame(current);
73
83
  }
74
84
  },
75
- cancel: async () => await reader.cancel()
85
+ async cancel() {
86
+ const pending = nextPromise;
87
+ nextPromise = null;
88
+ current?.close();
89
+ current = null;
90
+ video.cancel();
91
+ const buffered = await pending?.catch(() => null);
92
+ buffered?.close();
93
+ }
76
94
  };
77
95
  }
78
96
  }
@@ -1 +1 @@
1
- {"version":3,"file":"cursor.js","sourceRoot":"","sources":["../../../../../s/timeline/renderers/export/parts/cursor.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,EAAE,EAAK,MAAM,yBAAyB,CAAA;AAK9C,OAAO,EAAC,mBAAmB,EAAC,MAAM,wCAAwC,CAAA;AAE1E;;;;GAIG;AAEH,MAAM,OAAO,mBAAmB;IAKtB;IACA;IALT,QAAQ,CAAA;IACR,aAAa,GAAG,IAAI,GAAG,EAA4B,CAAA;IAEnD,YACS,MAAc,EACd,YAA6C;QAD7C,WAAM,GAAN,MAAM,CAAQ;QACd,iBAAY,GAAZ,YAAY,CAAiC;QAErD,IAAI,CAAC,QAAQ,GAAG,mBAAmB,CAAC,YAAY,EAAE,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE;YAChE,MAAM,SAAS,GAAG,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAA;YAC7C,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAA;YAC5C,OAAO,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;QAC9B,CAAC,CAAC,CAAA;IACH,CAAC;IAED,MAAM,CAAC,QAAsB;QAC5B,IAAI,YAAY,GAAG,MAAM,CAAC,iBAAiB,CAAA;QAC3C,OAAO;YACN,IAAI,EAAE,CAAC,QAAY,EAAE,EAAE;gBACtB,IAAI,QAAQ,GAAG,YAAY;oBAC1B,MAAM,IAAI,KAAK,CAAC,kDAAkD,QAAQ,YAAY,YAAY,IAAI,CAAC,CAAA;gBACxG,YAAY,GAAG,QAAQ,CAAA;gBACvB,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAA;YAChD,CAAC;YACD,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE;SAC5B,CAAA;IACF,CAAC;IAED,kBAAkB,CAAC,SAAqB;QACvC,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAA;QACrD,IAAI,QAAQ;YACX,OAAO,QAAQ,CAAA;QAEhB,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,SAAS,CAAC,CAAA;QACrD,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC,CAAA;QACjD,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAA;QAE9C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,EAAE,MAAM,CAAC,CAAA;QAC5C,OAAO,MAAM,CAAA;IACd,CAAC;IAED,KAAK,CAAC,OAAO;QACZ,MAAM,OAAO,CAAC,GAAG,CAChB,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAC/D,CAAA;QACD,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAA;IAC3B,CAAC;IAED,eAAe;IACf,OAAO,CAAC,MAA+C;QACtD,OAAO;YACN,KAAK,CAAC,IAAI,CAAC,QAAgB;gBAC1B,IAAI,IAAI,GAAsB,IAAI,CAAA;gBAClC,OAAO,IAAI,EAAE,CAAC;oBACb,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,MAAM,MAAM,CAAC,IAAI,EAAE,CAAA;oBAEhD,IAAI,IAAI,EAAE,CAAC;wBACV,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;wBACnD,IAAI,EAAE,KAAK,EAAE,CAAA;wBACb,OAAO,GAAG,CAAA;oBACX,CAAC;oBAED,MAAM,KAAK,GAAG,GAAG,CAAC,SAAS,IAAI,CAAC,CAAA;oBAChC,IAAI,KAAK,IAAI,QAAQ,EAAE,CAAC;wBACvB,MAAM,MAAM,GAAG,IAAI,EAAE,SAAS,IAAI,MAAM,CAAC,iBAAiB,CAAA;wBAC1D,MAAM,OAAO,GAAG,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,GAAG,CAAC,MAAM,GAAG,QAAQ,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,GAAG,QAAQ,CAAC,CAAA;wBAElF,MAAM,MAAM,GAAG,OAAO,CAAC,CAAC,CAAC,IAAK,CAAC,CAAC,CAAC,GAAG,CAAA;wBACpC,MAAM,KAAK,GAAG,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAA;wBAElC,MAAM,IAAI,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC,CAAA;wBACnC,MAAM,CAAC,KAAK,EAAE,CAAA;wBACd,KAAK,EAAE,KAAK,EAAE,CAAA;wBACd,OAAO,IAAI,CAAA;oBACZ,CAAC;oBAED,IAAI,EAAE,KAAK,EAAE,CAAA;oBACb,IAAI,GAAG,GAAG,CAAA;gBACX,CAAC;YACF,CAAC;YAED,MAAM,EAAE,KAAK,IAAI,EAAE,CAAC,MAAM,MAAM,CAAC,MAAM,EAAE;SACzC,CAAA;IACF,CAAC;CACD;AAED,MAAM,IAAI,GAAG,CAAC,EAAM,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,GAAG,KAAK,CAAC,CAAA"}
1
+ {"version":3,"file":"cursor.js","sourceRoot":"","sources":["../../../../../s/timeline/renderers/export/parts/cursor.ts"],"names":[],"mappings":"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"}
@@ -4,13 +4,12 @@ import { computeItemDuration } from "../../parts/handy.js";
4
4
  export function produceVideo({ timeline, fps, driver, resolveMedia, }) {
5
5
  const stream = new TransformStream();
6
6
  const writer = stream.writable.getWriter();
7
- const sampler = new CursorVisualSampler(driver, resolveMedia);
8
- const cursor = sampler.cursor(timeline);
7
+ const sampler = new CursorVisualSampler(driver, resolveMedia, timeline);
9
8
  const dt = 1 / fps;
10
9
  const duration = computeItemDuration(timeline.rootId, timeline);
11
10
  async function produce() {
12
11
  await fixedStep({ fps, duration }, async (timecode, i) => {
13
- const layers = await cursor.next(timecode);
12
+ const layers = await sampler.next(timecode);
14
13
  const composed = await driver.composite(layers);
15
14
  const frame = new VideoFrame(composed, {
16
15
  timestamp: Math.round(i * dt * 1_000_000),
@@ -1 +1 @@
1
- {"version":3,"file":"produce-video.js","sourceRoot":"","sources":["../../../../../s/timeline/renderers/export/parts/produce-video.ts"],"names":[],"mappings":"AAEA,OAAO,EAAC,mBAAmB,EAAC,MAAM,aAAa,CAAA;AAE/C,OAAO,EAAC,SAAS,EAAC,MAAM,2BAA2B,CAAA;AAEnD,OAAO,EAAC,mBAAmB,EAAC,MAAM,sBAAsB,CAAA;AAGxD,MAAM,UAAU,YAAY,CAAC,EAC5B,QAAQ,EACR,GAAG,EACH,MAAM,EACN,YAAY,GAMZ;IAEA,MAAM,MAAM,GAAG,IAAI,eAAe,EAA0B,CAAA;IAC5D,MAAM,MAAM,GAAG,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAA;IAC1C,MAAM,OAAO,GAAG,IAAI,mBAAmB,CAAC,MAAM,EAAE,YAAY,CAAC,CAAA;IAC7D,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAA;IACvC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,CAAA;IAClB,MAAM,QAAQ,GAAG,mBAAmB,CAAC,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;IAE/D,KAAK,UAAU,OAAO;QACrB,MAAM,SAAS,CAAC,EAAC,GAAG,EAAE,QAAQ,EAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,EAAE,EAAE;YACtD,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;YAC1C,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAA;YAE/C,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,QAAQ,EAAE;gBACtC,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;gBACzC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,GAAG,SAAS,CAAC;aACpC,CAAC,CAAA;YAEF,MAAM,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;YACzB,QAAQ,CAAC,KAAK,EAAE,CAAA;QACjB,CAAC,CAAC,CAAA;QAEF,MAAM,MAAM,CAAC,KAAK,EAAE,CAAA;IACrB,CAAC;IAED,OAAO,EAAE,CAAA;IAET,OAAO,MAAM,CAAC,QAAQ,CAAA;AACvB,CAAC"}
1
+ {"version":3,"file":"produce-video.js","sourceRoot":"","sources":["../../../../../s/timeline/renderers/export/parts/produce-video.ts"],"names":[],"mappings":"AAEA,OAAO,EAAC,mBAAmB,EAAC,MAAM,aAAa,CAAA;AAE/C,OAAO,EAAC,SAAS,EAAC,MAAM,2BAA2B,CAAA;AAEnD,OAAO,EAAC,mBAAmB,EAAC,MAAM,sBAAsB,CAAA;AAGxD,MAAM,UAAU,YAAY,CAAC,EAC5B,QAAQ,EACR,GAAG,EACH,MAAM,EACN,YAAY,GAMZ;IAEA,MAAM,MAAM,GAAG,IAAI,eAAe,EAA0B,CAAA;IAC5D,MAAM,MAAM,GAAG,MAAM,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAA;IAC1C,MAAM,OAAO,GAAG,IAAI,mBAAmB,CAAC,MAAM,EAAE,YAAY,EAAE,QAAQ,CAAC,CAAA;IACvE,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,CAAA;IAClB,MAAM,QAAQ,GAAG,mBAAmB,CAAC,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAA;IAE/D,KAAK,UAAU,OAAO;QACrB,MAAM,SAAS,CAAC,EAAC,GAAG,EAAE,QAAQ,EAAC,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,EAAE,EAAE;YACtD,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;YAC3C,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAA;YAE/C,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,QAAQ,EAAE;gBACtC,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;gBACzC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,GAAG,SAAS,CAAC;aACpC,CAAC,CAAA;YAEF,MAAM,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;YACzB,QAAQ,CAAC,KAAK,EAAE,CAAA;QACjB,CAAC,CAAC,CAAA;QAEF,MAAM,MAAM,CAAC,KAAK,EAAE,CAAA;IACrB,CAAC;IAED,OAAO,EAAE,CAAA;IAET,OAAO,MAAM,CAAC,QAAQ,CAAA;AACvB,CAAC"}
@@ -1,14 +1,14 @@
1
1
  import { Fps } from '../../../../units/fps.js';
2
2
  import { Ms } from '../../../../units/ms.js';
3
+ import { Driver } from '../../../../driver/driver.js';
3
4
  import { TimelineFile } from '../../../parts/basics.js';
5
+ import { CursorVisualSampler } from '../../export/parts/cursor.js';
4
6
  import { DecoderSource } from '../../../../driver/fns/schematic.js';
5
7
  export declare class Playback {
6
8
  #private;
9
+ private driver;
7
10
  private timeline;
8
11
  private resolveMedia;
9
- visualSampler: {
10
- sample(timeline: TimelineFile, timecode: Ms): Promise<import("../../../../driver/fns/schematic.js").Layer[]>;
11
- };
12
12
  audioSampler: {
13
13
  sampleAudio(timeline: TimelineFile, from: Ms): AsyncGenerator<{
14
14
  sample: import("mediabunny").AudioSample;
@@ -16,15 +16,19 @@ export declare class Playback {
16
16
  gain: number;
17
17
  }, void, unknown>;
18
18
  };
19
+ seekVisualSampler: {
20
+ sample(timeline: TimelineFile, timecode: Ms): Promise<import("../../../../driver/fns/schematic.js").Layer[]>;
21
+ };
22
+ playVisualSampler: CursorVisualSampler | null;
19
23
  onTick: import("@e280/stz").Pub<[]>;
20
24
  audioContext: AudioContext;
21
25
  audioGain: GainNode;
22
26
  audioNodes: Set<AudioBufferSourceNode>;
23
- constructor(timeline: TimelineFile, resolveMedia: (hash: string) => DecoderSource);
24
- samples(): AsyncGenerator<import("../../../../driver/fns/schematic.js").Layer[], void, unknown>;
27
+ constructor(driver: Driver, timeline: TimelineFile, resolveMedia: (hash: string) => DecoderSource);
25
28
  seek(time: Ms): Promise<import("../../../../driver/fns/schematic.js").Layer[]>;
26
29
  start(timeline: TimelineFile): Promise<void>;
27
30
  pause(): void;
31
+ get duration(): Ms;
28
32
  get currentTime(): Ms;
29
33
  setFps(fps: Fps): void;
30
34
  }