@omnimedia/omnitool 1.1.0-4 → 1.1.0-7

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 (133) hide show
  1. package/package.json +4 -2
  2. package/s/demo/demo.css +5 -0
  3. package/s/demo/routines/transcode-test.ts +4 -2
  4. package/s/demo/routines/transitions-test.ts +43 -0
  5. package/s/driver/driver.ts +17 -9
  6. package/s/driver/fns/schematic.ts +44 -21
  7. package/s/driver/fns/work.ts +112 -97
  8. package/s/features/transition/parts/fragment.ts +24 -0
  9. package/s/features/transition/parts/types.ts +94 -0
  10. package/s/features/transition/parts/uniforms.ts +29 -0
  11. package/s/features/transition/parts/vertex.ts +31 -0
  12. package/s/features/transition/transition.ts +60 -0
  13. package/s/index.html.ts +6 -1
  14. package/s/timeline/parts/basics.ts +1 -1
  15. package/s/timeline/parts/compositor/export.ts +77 -0
  16. package/s/timeline/parts/compositor/parts/html-tree.ts +37 -0
  17. package/s/timeline/parts/compositor/parts/schedulers.ts +85 -0
  18. package/s/timeline/parts/compositor/parts/tree-builder.ts +184 -0
  19. package/s/timeline/parts/compositor/parts/webcodecs-tree.ts +30 -0
  20. package/s/timeline/parts/compositor/playback.ts +81 -0
  21. package/s/timeline/parts/compositor/samplers/html.ts +115 -0
  22. package/s/timeline/parts/compositor/samplers/webcodecs.ts +60 -0
  23. package/s/timeline/parts/item.ts +38 -6
  24. package/s/timeline/parts/media.ts +21 -0
  25. package/s/timeline/parts/waveform.ts +1 -1
  26. package/s/timeline/sugar/builders.ts +102 -0
  27. package/s/timeline/sugar/o.ts +75 -16
  28. package/s/timeline/sugar/omni-test.ts +2 -2
  29. package/s/timeline/sugar/omni.ts +14 -11
  30. package/s/timeline/timeline.ts +22 -0
  31. package/s/timeline/types.ts +29 -0
  32. package/s/timeline/utils/audio-stream.ts +15 -0
  33. package/s/timeline/utils/matrix.ts +33 -0
  34. package/s/timeline/utils/video-cursor.ts +40 -0
  35. package/x/demo/demo.bundle.min.js +39 -37
  36. package/x/demo/demo.bundle.min.js.map +4 -4
  37. package/x/demo/demo.css +5 -0
  38. package/x/demo/routines/transcode-test.js +4 -2
  39. package/x/demo/routines/transcode-test.js.map +1 -1
  40. package/x/demo/routines/transitions-test.d.ts +5 -0
  41. package/x/demo/routines/transitions-test.js +35 -0
  42. package/x/demo/routines/transitions-test.js.map +1 -0
  43. package/x/driver/driver.d.ts +3 -5
  44. package/x/driver/driver.js +16 -9
  45. package/x/driver/driver.js.map +1 -1
  46. package/x/driver/driver.worker.bundle.min.js +2537 -148
  47. package/x/driver/driver.worker.bundle.min.js.map +4 -4
  48. package/x/driver/fns/host.d.ts +9 -2
  49. package/x/driver/fns/schematic.d.ts +38 -20
  50. package/x/driver/fns/work.d.ts +11 -4
  51. package/x/driver/fns/work.js +105 -96
  52. package/x/driver/fns/work.js.map +1 -1
  53. package/x/features/speech/transcribe/worker.bundle.min.js +541 -541
  54. package/x/features/speech/transcribe/worker.bundle.min.js.map +4 -4
  55. package/x/features/transition/parts/fragment.d.ts +1 -0
  56. package/x/features/transition/parts/fragment.js +25 -0
  57. package/x/features/transition/parts/fragment.js.map +1 -0
  58. package/x/features/transition/parts/types.d.ts +23 -0
  59. package/x/features/transition/parts/types.js +2 -0
  60. package/x/features/transition/parts/types.js.map +1 -0
  61. package/x/features/transition/parts/uniforms.d.ts +31 -0
  62. package/x/features/transition/parts/uniforms.js +27 -0
  63. package/x/features/transition/parts/uniforms.js.map +1 -0
  64. package/x/features/transition/parts/vertex.d.ts +1 -0
  65. package/x/features/transition/parts/vertex.js +32 -0
  66. package/x/features/transition/parts/vertex.js.map +1 -0
  67. package/x/features/transition/transition.d.ts +5 -0
  68. package/x/features/transition/transition.js +50 -0
  69. package/x/features/transition/transition.js.map +1 -0
  70. package/x/index.html +13 -3
  71. package/x/index.html.js +6 -1
  72. package/x/index.html.js.map +1 -1
  73. package/x/timeline/parts/basics.d.ts +1 -1
  74. package/x/timeline/parts/compositor/export.d.ts +9 -0
  75. package/x/timeline/parts/compositor/export.js +64 -0
  76. package/x/timeline/parts/compositor/export.js.map +1 -0
  77. package/x/timeline/parts/compositor/parts/html-tree.d.ts +3 -0
  78. package/x/timeline/parts/compositor/parts/html-tree.js +40 -0
  79. package/x/timeline/parts/compositor/parts/html-tree.js.map +1 -0
  80. package/x/timeline/parts/compositor/parts/schedulers.d.ts +15 -0
  81. package/x/timeline/parts/compositor/parts/schedulers.js +64 -0
  82. package/x/timeline/parts/compositor/parts/schedulers.js.map +1 -0
  83. package/x/timeline/parts/compositor/parts/tree-builder.d.ts +37 -0
  84. package/x/timeline/parts/compositor/parts/tree-builder.js +147 -0
  85. package/x/timeline/parts/compositor/parts/tree-builder.js.map +1 -0
  86. package/x/timeline/parts/compositor/parts/webcodecs-tree.d.ts +3 -0
  87. package/x/timeline/parts/compositor/parts/webcodecs-tree.js +28 -0
  88. package/x/timeline/parts/compositor/parts/webcodecs-tree.js.map +1 -0
  89. package/x/timeline/parts/compositor/playback.d.ts +19 -0
  90. package/x/timeline/parts/compositor/playback.js +71 -0
  91. package/x/timeline/parts/compositor/playback.js.map +1 -0
  92. package/x/timeline/parts/compositor/samplers/html.d.ts +3 -0
  93. package/x/timeline/parts/compositor/samplers/html.js +106 -0
  94. package/x/timeline/parts/compositor/samplers/html.js.map +1 -0
  95. package/x/timeline/parts/compositor/samplers/webcodecs.d.ts +2 -0
  96. package/x/timeline/parts/compositor/samplers/webcodecs.js +55 -0
  97. package/x/timeline/parts/compositor/samplers/webcodecs.js.map +1 -0
  98. package/x/timeline/parts/item.d.ts +34 -8
  99. package/x/timeline/parts/item.js +6 -3
  100. package/x/timeline/parts/item.js.map +1 -1
  101. package/x/timeline/parts/media.d.ts +3 -0
  102. package/x/timeline/parts/media.js +17 -0
  103. package/x/timeline/parts/media.js.map +1 -1
  104. package/x/timeline/parts/waveform.js +1 -1
  105. package/x/timeline/parts/waveform.js.map +1 -1
  106. package/x/timeline/sugar/builders.d.ts +96 -0
  107. package/x/timeline/sugar/builders.js +108 -0
  108. package/x/timeline/sugar/builders.js.map +1 -0
  109. package/x/timeline/sugar/o.d.ts +21 -8
  110. package/x/timeline/sugar/o.js +63 -14
  111. package/x/timeline/sugar/o.js.map +1 -1
  112. package/x/timeline/sugar/omni-test.js +1 -1
  113. package/x/timeline/sugar/omni-test.js.map +1 -1
  114. package/x/timeline/sugar/omni.d.ts +7 -3
  115. package/x/timeline/sugar/omni.js +9 -8
  116. package/x/timeline/sugar/omni.js.map +1 -1
  117. package/x/timeline/timeline.d.ts +9 -0
  118. package/x/timeline/timeline.js +22 -0
  119. package/x/timeline/timeline.js.map +1 -0
  120. package/x/timeline/types.d.ts +24 -0
  121. package/x/timeline/types.js +2 -0
  122. package/x/timeline/types.js.map +1 -0
  123. package/x/timeline/utils/audio-stream.d.ts +6 -0
  124. package/x/timeline/utils/audio-stream.js +17 -0
  125. package/x/timeline/utils/audio-stream.js.map +1 -0
  126. package/x/timeline/utils/matrix.d.ts +8 -0
  127. package/x/timeline/utils/matrix.js +26 -0
  128. package/x/timeline/utils/matrix.js.map +1 -0
  129. package/x/timeline/utils/video-cursor.d.ts +10 -0
  130. package/x/timeline/utils/video-cursor.js +36 -0
  131. package/x/timeline/utils/video-cursor.js.map +1 -0
  132. package/x/tools/speech-recognition/whisper/parts/worker.bundle.min.js +6 -6
  133. package/x/tools/speech-recognition/whisper/parts/worker.bundle.min.js.map +4 -4
@@ -0,0 +1,147 @@
1
+ import { Kind } from "../../item.js";
2
+ import { I6, mul6, transformToMat6 } from "../../../utils/matrix.js";
3
+ const requireItem = (items, id) => items.get(id);
4
+ export const getWorldMat6 = (items, item, parent) => {
5
+ let world = parent ?? I6;
6
+ if (item.spatialId) {
7
+ const spatial = requireItem(items, item.spatialId);
8
+ const local = transformToMat6(spatial.transform);
9
+ world = mul6(local, world);
10
+ }
11
+ return world;
12
+ };
13
+ export class TreeBuilder {
14
+ items;
15
+ sampler;
16
+ constructor(items, sampler) {
17
+ this.items = items;
18
+ this.sampler = sampler;
19
+ }
20
+ async build(root, parentMatrix) {
21
+ switch (root.kind) {
22
+ case Kind.Video: return this.sampler.video(root, getWorldMat6(this.items, root, parentMatrix));
23
+ case Kind.Audio: return this.sampler.audio(root);
24
+ case Kind.Text: {
25
+ const matrix = getWorldMat6(this.items, root, parentMatrix);
26
+ return {
27
+ duration: Infinity,
28
+ visuals: {
29
+ sampleAt: async () => [{ kind: "text", content: root.content, color: "white", fontSize: 48, matrix }]
30
+ }
31
+ };
32
+ }
33
+ case Kind.Gap: return {
34
+ duration: root.duration,
35
+ visuals: {
36
+ sampleAt: async () => []
37
+ }
38
+ };
39
+ case Kind.Stack: {
40
+ const matrix = getWorldMat6(this.items, root, parentMatrix);
41
+ const children = await Promise.all(root.childrenIds.map(id => this.build(requireItem(this.items, id), matrix)));
42
+ return this.#composeStack(children);
43
+ }
44
+ case Kind.Sequence: {
45
+ const matrix = getWorldMat6(this.items, root, parentMatrix);
46
+ return this.#composeSequence(root, matrix);
47
+ }
48
+ default: return { duration: 0 };
49
+ }
50
+ }
51
+ // Visual composition is the same for both builders, so it lives here.
52
+ #composeVisuals_Stack(children) {
53
+ return {
54
+ sampleAt: async (time) => {
55
+ const layers = await Promise.all(children.map(c => c.visuals ? c.visuals.sampleAt(time) : Promise.resolve([])));
56
+ return layers.flat();
57
+ }
58
+ };
59
+ }
60
+ #composeVisuals_Sequence(children) {
61
+ return {
62
+ sampleAt: async (time) => {
63
+ let localTime = time;
64
+ for (const child of children) {
65
+ if (localTime < child.duration)
66
+ return child.visuals ? child.visuals.sampleAt(localTime) : [];
67
+ localTime -= child.duration;
68
+ }
69
+ return [];
70
+ }
71
+ };
72
+ }
73
+ #composeStack(children) {
74
+ const duration = Math.max(0, ...children.map(k => (Number.isFinite(k.duration) ? k.duration : 0)));
75
+ return {
76
+ duration,
77
+ visuals: this.#composeVisuals_Stack(children),
78
+ audio: this.composeAudio_Stack(children),
79
+ };
80
+ }
81
+ async #composeSequence(sequence, parentMatrix) {
82
+ const childItems = sequence.childrenIds.map(id => requireItem(this.items, id));
83
+ const children = await this.#processChildren(childItems, parentMatrix);
84
+ const duration = children.reduce((a, k) => a + k.duration, 0);
85
+ return {
86
+ duration,
87
+ visuals: this.#composeVisuals_Sequence(children),
88
+ audio: this.composeAudio_Sequence(children),
89
+ };
90
+ }
91
+ async #processChildren(childItems, parentMatrix) {
92
+ const processedNodes = [];
93
+ for (let i = 0; i < childItems.length; i++) {
94
+ const item = childItems[i];
95
+ if (item.kind !== Kind.Transition) {
96
+ processedNodes.push(await this.build(item, parentMatrix));
97
+ continue;
98
+ }
99
+ const outgoingNode = processedNodes.pop();
100
+ const incomingItem = childItems[i + 1];
101
+ if (!outgoingNode || !incomingItem || incomingItem.kind === Kind.Transition) {
102
+ if (outgoingNode)
103
+ processedNodes.push(outgoingNode);
104
+ continue;
105
+ }
106
+ const incomingNode = await this.build(incomingItem, parentMatrix);
107
+ const transitionNode = await this.#createTransitionNode(item, outgoingNode, incomingNode);
108
+ processedNodes.push(transitionNode);
109
+ i++;
110
+ }
111
+ return processedNodes;
112
+ }
113
+ async #createTransitionNode(transitionItem, outgoingNode, incomingNode) {
114
+ const overlap = Math.max(0, Math.min(transitionItem.duration, outgoingNode.duration, incomingNode.duration));
115
+ const start = Math.max(0, outgoingNode.duration - overlap);
116
+ const combinedDuration = outgoingNode.duration + incomingNode.duration - overlap;
117
+ return {
118
+ duration: combinedDuration,
119
+ visuals: {
120
+ sampleAt: async (t) => {
121
+ if (!outgoingNode.visuals || !incomingNode.visuals)
122
+ return [];
123
+ if (t < start)
124
+ return outgoingNode.visuals.sampleAt(t);
125
+ if (t < outgoingNode.duration) {
126
+ const localTime = t - start;
127
+ const progress = overlap > 0 ? (localTime / overlap) : 1;
128
+ const from = await outgoingNode.visuals.sampleAt(t);
129
+ const to = await incomingNode.visuals.sampleAt(localTime);
130
+ if (!from[0]?.frame || !to[0]?.frame)
131
+ return [];
132
+ return [{
133
+ kind: "transition",
134
+ name: "circle",
135
+ progress,
136
+ from: from[0].frame,
137
+ to: to[0].frame,
138
+ }];
139
+ }
140
+ return incomingNode.visuals.sampleAt(t - outgoingNode.duration + overlap);
141
+ }
142
+ },
143
+ audio: this.composeAudio_Sequence([outgoingNode, incomingNode])
144
+ };
145
+ }
146
+ }
147
+ //# sourceMappingURL=tree-builder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tree-builder.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/compositor/parts/tree-builder.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Item } from "../../item.js";
2
+ import { AudioStreamComponent, Node, WebcodecsSampler } from "./tree-builder.js";
3
+ export declare function buildWebCodecsNodeTree(root: Item.Any, items: Map<number, Item.Any>, sampler: WebcodecsSampler): Promise<Node<AudioStreamComponent>>;
@@ -0,0 +1,28 @@
1
+ import { TreeBuilder } from "./tree-builder.js";
2
+ class WebCodecsNodeBuilder extends TreeBuilder {
3
+ composeAudio_Stack(children) {
4
+ return {
5
+ getStream: async function* () {
6
+ for (const child of children) {
7
+ if (child.audio)
8
+ yield* child.audio.getStream();
9
+ }
10
+ }
11
+ };
12
+ }
13
+ composeAudio_Sequence(children) {
14
+ return {
15
+ getStream: async function* () {
16
+ for (const child of children) {
17
+ if (child.audio)
18
+ yield* child.audio.getStream();
19
+ }
20
+ }
21
+ };
22
+ }
23
+ }
24
+ export function buildWebCodecsNodeTree(root, items, sampler) {
25
+ const builder = new WebCodecsNodeBuilder(items, sampler);
26
+ return builder.build(root);
27
+ }
28
+ //# sourceMappingURL=webcodecs-tree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webcodecs-tree.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/compositor/parts/webcodecs-tree.ts"],"names":[],"mappings":"AACA,OAAO,EAA+C,WAAW,EAAC,MAAM,mBAAmB,CAAA;AAE3F,MAAM,oBAAqB,SAAQ,WAAiC;IACnE,kBAAkB,CAAC,QAAsC;QACxD,OAAO;YACN,SAAS,EAAE,KAAK,SAAS,CAAC;gBACzB,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE,CAAC;oBAC9B,IAAI,KAAK,CAAC,KAAK;wBACd,KAAK,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAA;gBAChC,CAAC;YACF,CAAC;SACD,CAAA;IACF,CAAC;IACD,qBAAqB,CAAC,QAAsC;QAC3D,OAAO;YACN,SAAS,EAAE,KAAK,SAAS,CAAC;gBACzB,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE,CAAC;oBAC9B,IAAI,KAAK,CAAC,KAAK;wBACd,KAAK,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAA;gBAChC,CAAC;YACF,CAAC;SACD,CAAA;IACF,CAAC;CACD;AAED,MAAM,UAAU,sBAAsB,CAAC,IAAc,EAAE,KAA4B,EAAE,OAAyB;IAC7G,MAAM,OAAO,GAAG,IAAI,oBAAoB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAA;IACxD,OAAO,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;AAC3B,CAAC"}
@@ -0,0 +1,19 @@
1
+ import { TimelineFile } from "../basics.js";
2
+ import { DecoderSource } from "../../../driver/fns/schematic.js";
3
+ import { AudioPlaybackComponent, HTMLSampler, Node } from "./parts/tree-builder.js";
4
+ type ResolveMedia = (hash: string) => DecoderSource;
5
+ export declare class VideoPlayer {
6
+ #private;
7
+ canvas: HTMLCanvasElement;
8
+ private root;
9
+ private sampler;
10
+ private resolveMedia;
11
+ constructor(canvas: HTMLCanvasElement, root: Node<AudioPlaybackComponent>, sampler: HTMLSampler, resolveMedia?: ResolveMedia);
12
+ get context(): CanvasRenderingContext2D;
13
+ static create(timeline: TimelineFile): Promise<VideoPlayer>;
14
+ play(): Promise<void>;
15
+ pause(): void;
16
+ seek(time: number): Promise<void>;
17
+ setFPS(value: number): void;
18
+ }
19
+ export {};
@@ -0,0 +1,71 @@
1
+ import { context } from "../../../context.js";
2
+ import { realtime } from "./parts/schedulers.js";
3
+ import { makeHtmlSampler } from "./samplers/html.js";
4
+ import { buildHTMLNodeTree } from "./parts/html-tree.js";
5
+ export class VideoPlayer {
6
+ canvas;
7
+ root;
8
+ sampler;
9
+ resolveMedia;
10
+ #controller = realtime(t => this.#tick(t));
11
+ constructor(canvas, root, sampler, resolveMedia = _hash => "/assets/temp/gl.mp4") {
12
+ this.canvas = canvas;
13
+ this.root = root;
14
+ this.sampler = sampler;
15
+ this.resolveMedia = resolveMedia;
16
+ this.#controller.setFPS(30);
17
+ }
18
+ get context() {
19
+ return this.canvas.getContext("2d");
20
+ }
21
+ static async create(timeline) {
22
+ const rootItem = new Map(timeline.items.map(i => [i.id, i])).get(timeline.rootId);
23
+ const items = new Map(timeline.items.map(i => [i.id, i]));
24
+ const sampler = makeHtmlSampler(() => "/assets/temp/gl.mp4");
25
+ const root = await buildHTMLNodeTree(rootItem, items, sampler);
26
+ const canvas = document.createElement("canvas");
27
+ canvas.width = 1920;
28
+ canvas.height = 1080;
29
+ return new this(canvas, root, sampler);
30
+ }
31
+ async #tick(t) {
32
+ const driver = await context.driver;
33
+ const dur = this.root.duration;
34
+ const tt = t > dur ? dur : t;
35
+ this.root.audio?.onTimeUpdate(tt);
36
+ for (const layer of await this.root.visuals?.sampleAt(tt) ?? []) {
37
+ const frame = await driver.composite(layer);
38
+ this.context.drawImage(frame, 0, 0);
39
+ frame.close();
40
+ }
41
+ if (t >= dur)
42
+ this.pause();
43
+ }
44
+ async play() {
45
+ if (!this.#controller.isPlaying()) {
46
+ this.sampler.setPaused(false);
47
+ this.#controller.play();
48
+ }
49
+ }
50
+ pause() {
51
+ if (this.#controller.isPlaying()) {
52
+ this.#controller.pause();
53
+ this.sampler.setPaused(true);
54
+ }
55
+ }
56
+ async seek(time) {
57
+ const driver = await context.driver;
58
+ this.pause();
59
+ this.#controller.seek(time);
60
+ this.root.audio?.onTimeUpdate(time);
61
+ for (const draw of await this.root.visuals?.sampleAt(time) ?? []) {
62
+ const frame = await driver.composite(draw);
63
+ this.context.drawImage(frame, 0, 0);
64
+ frame.close();
65
+ }
66
+ }
67
+ setFPS(value) {
68
+ this.#controller.setFPS(value);
69
+ }
70
+ }
71
+ //# sourceMappingURL=playback.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"playback.js","sourceRoot":"","sources":["../../../../s/timeline/parts/compositor/playback.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { HTMLSampler } from "../parts/tree-builder.js";
2
+ import { DecoderSource } from "../../../../driver/fns/schematic.js";
3
+ export declare function makeHtmlSampler(resolveMedia: (hash: string) => DecoderSource): HTMLSampler;
@@ -0,0 +1,106 @@
1
+ const toUrl = (src) => (src instanceof Blob ? URL.createObjectURL(src) : String(src));
2
+ export function makeHtmlSampler(resolveMedia) {
3
+ const videoElements = new Map();
4
+ const audioElements = new Map();
5
+ function getOrCreateVideoElement(clip) {
6
+ let video = videoElements.get(clip.id);
7
+ if (!video) {
8
+ video = document.createElement("video");
9
+ video.playsInline = true;
10
+ video.muted = true;
11
+ video.preload = "auto";
12
+ video.crossOrigin = "anonymous";
13
+ video.src = toUrl(resolveMedia(clip.mediaHash));
14
+ videoElements.set(clip.id, video);
15
+ }
16
+ return video;
17
+ }
18
+ function getOrCreateAudioElement(clip) {
19
+ let audio = audioElements.get(clip.id);
20
+ if (!audio) {
21
+ audio = document.createElement("audio");
22
+ audio.preload = "auto";
23
+ audio.crossOrigin = "anonymous";
24
+ audio.src = toUrl(resolveMedia(clip.mediaHash));
25
+ audio.volume = 0.2;
26
+ audioElements.set(clip.id, audio);
27
+ }
28
+ return audio;
29
+ }
30
+ let paused = true;
31
+ return {
32
+ async video(item, matrix) {
33
+ const video = getOrCreateVideoElement(item);
34
+ return {
35
+ duration: item.duration,
36
+ // if paused seek otherwise play
37
+ visuals: {
38
+ sampleAt: async (t) => {
39
+ if (t < 0 || t >= item.duration)
40
+ return [];
41
+ if (video.paused && paused) {
42
+ await seek(video, t);
43
+ }
44
+ if (video.paused && !paused) {
45
+ await video.play();
46
+ }
47
+ const frame = new VideoFrame(video);
48
+ return frame ? [{ kind: "image", frame, matrix }] : [];
49
+ }
50
+ }
51
+ };
52
+ },
53
+ async audio(item) {
54
+ const audio = getOrCreateAudioElement(item);
55
+ return {
56
+ duration: item.duration,
57
+ audio: {
58
+ onTimeUpdate: async (time) => {
59
+ const localTime = item.start + time;
60
+ if (audio.paused && paused) {
61
+ await seek(audio, localTime);
62
+ }
63
+ if (audio.paused && !paused) {
64
+ await audio.play();
65
+ }
66
+ return [];
67
+ }
68
+ }
69
+ };
70
+ },
71
+ async dispose() {
72
+ const elements = [...videoElements.values(), ...audioElements.values()];
73
+ for (const element of elements) {
74
+ element.pause();
75
+ if (element.src.startsWith("blob:"))
76
+ URL.revokeObjectURL(element.src);
77
+ element.remove();
78
+ }
79
+ videoElements.clear();
80
+ audioElements.clear();
81
+ },
82
+ async setPaused(p) {
83
+ paused = p;
84
+ const elements = [...videoElements.values(), ...audioElements.values()];
85
+ for (const element of elements) {
86
+ if (p)
87
+ element.pause();
88
+ }
89
+ },
90
+ };
91
+ }
92
+ function seek(media, time) {
93
+ return new Promise((resolve) => {
94
+ const onSeeked = () => {
95
+ media.removeEventListener("seeked", onSeeked);
96
+ resolve();
97
+ };
98
+ media.addEventListener("seeked", onSeeked);
99
+ if (media.fastSeek) {
100
+ media.fastSeek(time);
101
+ }
102
+ else
103
+ media.currentTime = time;
104
+ });
105
+ }
106
+ //# sourceMappingURL=html.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"html.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/compositor/samplers/html.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { WebcodecsSampler } from "../parts/tree-builder.js";
2
+ export declare function makeWebCodecsSampler(resolveMedia: (hash: string) => any): WebcodecsSampler;
@@ -0,0 +1,55 @@
1
+ import { context } from "../../../../context.js";
2
+ import { VideoCursor } from "../../../utils/video-cursor.js";
3
+ import { AudioStream } from "../../../utils/audio-stream.js";
4
+ const toUs = (seconds) => Math.round(seconds * 1_000_000);
5
+ export function makeWebCodecsSampler(resolveMedia) {
6
+ const videoCursors = new Map();
7
+ async function getCursorForVideo(videoItem) {
8
+ const existing = videoCursors.get(videoItem.id);
9
+ if (existing)
10
+ return existing;
11
+ const driver = await context.driver;
12
+ const source = resolveMedia(videoItem.mediaHash);
13
+ const video = driver.decodeVideo({ source });
14
+ const cursor = new VideoCursor(video.getReader());
15
+ videoCursors.set(videoItem.id, cursor);
16
+ return cursor;
17
+ }
18
+ return {
19
+ async video(item, matrix) {
20
+ const cursor = await getCursorForVideo(item);
21
+ const baseUs = toUs(item.start ?? 0);
22
+ return {
23
+ duration: item.duration,
24
+ visuals: {
25
+ sampleAt: async (time) => {
26
+ const frame = await cursor.atOrNear(baseUs + toUs(time));
27
+ return frame ? [{ kind: "image", frame, matrix }] : [];
28
+ }
29
+ }
30
+ };
31
+ },
32
+ async audio(item) {
33
+ return {
34
+ duration: item.duration,
35
+ audio: {
36
+ getStream: async function* () {
37
+ const driver = await context.driver;
38
+ const source = resolveMedia(item.mediaHash);
39
+ const startUs = item.start;
40
+ const endUs = (item.start + item.duration);
41
+ const audio = driver.decodeAudio({ source, start: startUs, end: endUs });
42
+ const audioStream = new AudioStream(audio.getReader());
43
+ yield* audioStream.stream();
44
+ },
45
+ }
46
+ };
47
+ },
48
+ async dispose() {
49
+ const tasks = Array.from([...videoCursors.values()], c => c.cancel());
50
+ videoCursors.clear();
51
+ await Promise.all(tasks);
52
+ }
53
+ };
54
+ }
55
+ //# sourceMappingURL=webcodecs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webcodecs.js","sourceRoot":"","sources":["../../../../../s/timeline/parts/compositor/samplers/webcodecs.ts"],"names":[],"mappings":"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"}
@@ -1,28 +1,52 @@
1
1
  import { Id, Hash } from "./basics.js";
2
+ import { Transform } from "../types.js";
2
3
  export declare enum Kind {
3
4
  Sequence = 0,
4
5
  Stack = 1,
5
- Clip = 2,
6
- Text = 3,
7
- Transition = 4
6
+ Video = 2,
7
+ Audio = 3,
8
+ Text = 4,
9
+ Gap = 5,
10
+ Spatial = 6,
11
+ Transition = 7
8
12
  }
9
13
  export declare enum Effect {
10
14
  Crossfade = 0
11
15
  }
12
16
  export declare namespace Item {
17
+ type Spatial = {
18
+ id: Id;
19
+ kind: Kind.Spatial;
20
+ transform: Transform;
21
+ };
22
+ type Gap = {
23
+ id: Id;
24
+ kind: Kind.Gap;
25
+ duration: number;
26
+ };
13
27
  type Sequence = {
14
28
  id: Id;
15
29
  kind: Kind.Sequence;
16
- children: Id[];
30
+ childrenIds: Id[];
31
+ spatialId?: Id;
17
32
  };
18
33
  type Stack = {
19
34
  id: Id;
20
35
  kind: Kind.Stack;
21
- children: Id[];
36
+ childrenIds: Id[];
37
+ spatialId?: Id;
38
+ };
39
+ type Video = {
40
+ id: Id;
41
+ kind: Kind.Video;
42
+ mediaHash: Hash;
43
+ start: number;
44
+ duration: number;
45
+ spatialId?: Id;
22
46
  };
23
- type Clip = {
47
+ type Audio = {
24
48
  id: Id;
25
- kind: Kind.Clip;
49
+ kind: Kind.Audio;
26
50
  mediaHash: Hash;
27
51
  start: number;
28
52
  duration: number;
@@ -31,6 +55,8 @@ export declare namespace Item {
31
55
  id: Id;
32
56
  kind: Kind.Text;
33
57
  content: string;
58
+ spatialId?: Id;
59
+ color: string;
34
60
  };
35
61
  type Transition = {
36
62
  id: Id;
@@ -38,5 +64,5 @@ export declare namespace Item {
38
64
  effect: Effect.Crossfade;
39
65
  duration: number;
40
66
  };
41
- type Any = (Sequence | Stack | Clip | Text | Transition);
67
+ type Any = (Sequence | Stack | Video | Audio | Text | Gap | Transition | Spatial);
42
68
  }
@@ -2,9 +2,12 @@ export var Kind;
2
2
  (function (Kind) {
3
3
  Kind[Kind["Sequence"] = 0] = "Sequence";
4
4
  Kind[Kind["Stack"] = 1] = "Stack";
5
- Kind[Kind["Clip"] = 2] = "Clip";
6
- Kind[Kind["Text"] = 3] = "Text";
7
- Kind[Kind["Transition"] = 4] = "Transition";
5
+ Kind[Kind["Video"] = 2] = "Video";
6
+ Kind[Kind["Audio"] = 3] = "Audio";
7
+ Kind[Kind["Text"] = 4] = "Text";
8
+ Kind[Kind["Gap"] = 5] = "Gap";
9
+ Kind[Kind["Spatial"] = 6] = "Spatial";
10
+ Kind[Kind["Transition"] = 7] = "Transition";
8
11
  })(Kind || (Kind = {}));
9
12
  export var Effect;
10
13
  (function (Effect) {
@@ -1 +1 @@
1
- {"version":3,"file":"item.js","sourceRoot":"","sources":["../../../s/timeline/parts/item.ts"],"names":[],"mappings":"AAGA,MAAM,CAAN,IAAY,IAMX;AAND,WAAY,IAAI;IACf,uCAAQ,CAAA;IACR,iCAAK,CAAA;IACL,+BAAI,CAAA;IACJ,+BAAI,CAAA;IACJ,2CAAU,CAAA;AACX,CAAC,EANW,IAAI,KAAJ,IAAI,QAMf;AAED,MAAM,CAAN,IAAY,MAEX;AAFD,WAAY,MAAM;IACjB,6CAAS,CAAA;AACV,CAAC,EAFW,MAAM,KAAN,MAAM,QAEjB"}
1
+ {"version":3,"file":"item.js","sourceRoot":"","sources":["../../../s/timeline/parts/item.ts"],"names":[],"mappings":"AAIA,MAAM,CAAN,IAAY,IASX;AATD,WAAY,IAAI;IACf,uCAAQ,CAAA;IACR,iCAAK,CAAA;IACL,iCAAK,CAAA;IACL,iCAAK,CAAA;IACL,+BAAI,CAAA;IACJ,6BAAG,CAAA;IACH,qCAAO,CAAA;IACP,2CAAU,CAAA;AACX,CAAC,EATW,IAAI,KAAJ,IAAI,QASf;AAED,MAAM,CAAN,IAAY,MAEX;AAFD,WAAY,MAAM;IACjB,6CAAS,CAAA;AACV,CAAC,EAFW,MAAM,KAAN,MAAM,QAEjB"}
@@ -1,7 +1,10 @@
1
1
  import { Datafile } from "../utils/datafile.js";
2
2
  export declare class Media {
3
+ #private;
3
4
  datafile: Datafile;
4
5
  duration: number;
6
+ hasVideo: boolean;
7
+ hasAudio: boolean;
5
8
  constructor(datafile: Datafile);
6
9
  static analyze(datafile: Datafile): Promise<Media>;
7
10
  }
@@ -1,13 +1,30 @@
1
+ import { ALL_FORMATS, Input } from "mediabunny";
2
+ import { loadDecoderSource } from "../../driver/utils/load-decoder-source.js";
1
3
  export class Media {
2
4
  datafile;
3
5
  duration = 0;
6
+ hasVideo = false;
7
+ hasAudio = false;
4
8
  constructor(datafile) {
5
9
  this.datafile = datafile;
6
10
  }
7
11
  static async analyze(datafile) {
8
12
  const media = new this(datafile);
9
13
  media.duration = 10;
14
+ const { video, audio } = await this.#has("/assets/temp/gl.mp4");
15
+ media.hasAudio = audio;
16
+ media.hasVideo = video;
10
17
  return media;
11
18
  }
19
+ static async #has(source) {
20
+ const input = new Input({
21
+ formats: ALL_FORMATS,
22
+ source: await loadDecoderSource(source)
23
+ });
24
+ return {
25
+ audio: !!(await input.getPrimaryAudioTrack()),
26
+ video: !!(await input.getPrimaryVideoTrack())
27
+ };
28
+ }
12
29
  }
13
30
  //# sourceMappingURL=media.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"media.js","sourceRoot":"","sources":["../../../s/timeline/parts/media.ts"],"names":[],"mappings":"AAGA,MAAM,OAAO,KAAK;IAEE;IADnB,QAAQ,GAAG,CAAC,CAAA;IACZ,YAAmB,QAAkB;QAAlB,aAAQ,GAAR,QAAQ,CAAU;IAAG,CAAC;IAEzC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,QAAkB;QACtC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,CAAA;QAChC,KAAK,CAAC,QAAQ,GAAG,EAAE,CAAA;QACnB,OAAO,KAAK,CAAA;IACb,CAAC;CACD"}
1
+ {"version":3,"file":"media.js","sourceRoot":"","sources":["../../../s/timeline/parts/media.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,WAAW,EAAE,KAAK,EAAC,MAAM,YAAY,CAAA;AAI7C,OAAO,EAAC,iBAAiB,EAAC,MAAM,2CAA2C,CAAA;AAE3E,MAAM,OAAO,KAAK;IAKE;IAJnB,QAAQ,GAAG,CAAC,CAAA;IACZ,QAAQ,GAAG,KAAK,CAAA;IAChB,QAAQ,GAAG,KAAK,CAAA;IAEhB,YAAmB,QAAkB;QAAlB,aAAQ,GAAR,QAAQ,CAAU;IAAG,CAAC;IAEzC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,QAAkB;QACtC,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,CAAA;QAChC,KAAK,CAAC,QAAQ,GAAG,EAAE,CAAA;QACnB,MAAM,EAAC,KAAK,EAAE,KAAK,EAAC,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAA;QAC7D,KAAK,CAAC,QAAQ,GAAG,KAAK,CAAA;QACtB,KAAK,CAAC,QAAQ,GAAG,KAAK,CAAA;QACtB,OAAO,KAAK,CAAA;IACb,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,MAAqB;QACtC,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC;YACvB,OAAO,EAAE,WAAW;YACpB,MAAM,EAAE,MAAM,iBAAiB,CAAC,MAAM,CAAC;SACvC,CAAC,CAAA;QACF,OAAO;YACN,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,oBAAoB,EAAE,CAAC;YAC7C,KAAK,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,oBAAoB,EAAE,CAAC;SAC7C,CAAA;IACF,CAAC;CACD"}
@@ -16,7 +16,7 @@ export class Waveform {
16
16
  }
17
17
  static async init(source, container) {
18
18
  const driver = await context.driver;
19
- const reader = driver.decode({ source }).audio.getReader();
19
+ const reader = driver.decodeAudio({ source }).getReader();
20
20
  const peaks = [];
21
21
  let buffer = [];
22
22
  const samplesPerPeak = 1024;