@omnimedia/omnitool 1.1.0-55 → 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 (32) hide show
  1. package/package.json +1 -1
  2. package/s/demo/demo.bundle.ts +5 -3
  3. package/s/demo/routines/waveform-test.ts +30 -5
  4. package/s/index.html.ts +1 -2
  5. package/s/timeline/parts/waveform/parts/collect.ts +72 -0
  6. package/s/timeline/parts/waveform/parts/render.ts +45 -0
  7. package/s/timeline/parts/waveform/parts/types.ts +24 -0
  8. package/s/timeline/parts/waveform/waveform.ts +152 -0
  9. package/s/timeline/parts/waveform.ts +0 -61
  10. package/x/demo/demo.bundle.js +2 -0
  11. package/x/demo/demo.bundle.js.map +1 -1
  12. package/x/demo/demo.bundle.min.js +11 -84
  13. package/x/demo/demo.bundle.min.js.map +4 -4
  14. package/x/demo/routines/waveform-test.js +24 -4
  15. package/x/demo/routines/waveform-test.js.map +1 -1
  16. package/x/index.html +3 -3
  17. package/x/index.html.js +1 -1
  18. package/x/timeline/parts/waveform/parts/collect.d.ts +11 -0
  19. package/x/timeline/parts/waveform/parts/collect.js +56 -0
  20. package/x/timeline/parts/waveform/parts/collect.js.map +1 -0
  21. package/x/timeline/parts/waveform/parts/render.d.ts +5 -0
  22. package/x/timeline/parts/waveform/parts/render.js +29 -0
  23. package/x/timeline/parts/waveform/parts/render.js.map +1 -0
  24. package/x/timeline/parts/waveform/parts/types.d.ts +21 -0
  25. package/x/timeline/parts/waveform/parts/types.js +2 -0
  26. package/x/timeline/parts/waveform/parts/types.js.map +1 -0
  27. package/x/timeline/parts/waveform/waveform.d.ts +17 -0
  28. package/x/timeline/parts/waveform/waveform.js +125 -0
  29. package/x/timeline/parts/waveform/waveform.js.map +1 -0
  30. package/x/timeline/parts/waveform.d.ts +1 -9
  31. package/x/timeline/parts/waveform.js +1 -48
  32. package/x/timeline/parts/waveform.js.map +1 -1
@@ -1,12 +1,32 @@
1
- import { Waveform } from "../../timeline/parts/waveform.js";
1
+ import { Waveform } from "../../timeline/parts/waveform/waveform.js";
2
2
  export async function waveformTest(driver, source, root) {
3
3
  const container = root.querySelector(".waveform-canvas");
4
4
  const widthSlider = root.querySelector(".width");
5
5
  container.replaceChildren();
6
- const waveform = await Waveform.init(driver, source, container);
7
- widthSlider.oninput = () => {
6
+ container.style.position = "relative";
7
+ container.style.height = "96px";
8
+ container.style.overflow = "hidden";
9
+ const waveform = await Waveform.init(driver, source, {
10
+ tileHeight: 96,
11
+ onChange: () => renderTiles()
12
+ });
13
+ const renderTiles = () => {
8
14
  const width = +widthSlider.value;
9
- waveform.width = width;
15
+ const zoom = pixelsPerSecond(width, waveform.duration);
16
+ waveform.zoom = zoom;
17
+ container.style.width = `${width}px`;
18
+ container.replaceChildren(...[...waveform.getTiles().values()].map(tile => {
19
+ tile.canvas.style.position = "absolute";
20
+ tile.canvas.style.top = "0";
21
+ tile.canvas.style.left = `${tile.startTime * pixelsPerSecond(width, waveform.duration)}px`;
22
+ tile.canvas.style.height = "100%";
23
+ return tile.canvas;
24
+ }));
10
25
  };
26
+ widthSlider.oninput = renderTiles;
27
+ waveform.range = [0, waveform.duration];
28
+ }
29
+ function pixelsPerSecond(width, duration) {
30
+ return duration > 0 ? width / duration : 0;
11
31
  }
12
32
  //# sourceMappingURL=waveform-test.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"waveform-test.js","sourceRoot":"","sources":["../../../s/demo/routines/waveform-test.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,QAAQ,EAAC,MAAM,kCAAkC,CAAA;AAEzD,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,MAAc,EAAE,MAAY,EAAE,IAAiB;IACjF,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,kBAAkB,CAAgB,CAAA;IACvE,MAAM,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAqB,CAAA;IACpE,SAAS,CAAC,eAAe,EAAE,CAAA;IAC3B,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,CAAC,CAAA;IAE/D,WAAW,CAAC,OAAO,GAAG,GAAG,EAAE;QAC1B,MAAM,KAAK,GAAG,CAAC,WAAW,CAAC,KAAK,CAAA;QAChC,QAAQ,CAAC,KAAK,GAAG,KAAK,CAAA;IACvB,CAAC,CAAA;AACF,CAAC"}
1
+ {"version":3,"file":"waveform-test.js","sourceRoot":"","sources":["../../../s/demo/routines/waveform-test.ts"],"names":[],"mappings":"AAEA,OAAO,EAAC,QAAQ,EAAC,MAAM,2CAA2C,CAAA;AAElE,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,MAAc,EAAE,MAAY,EAAE,IAAiB;IACjF,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,kBAAkB,CAAmB,CAAA;IAC1E,MAAM,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAqB,CAAA;IAEpE,SAAS,CAAC,eAAe,EAAE,CAAA;IAC3B,SAAS,CAAC,KAAK,CAAC,QAAQ,GAAG,UAAU,CAAA;IACrC,SAAS,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAA;IAC/B,SAAS,CAAC,KAAK,CAAC,QAAQ,GAAG,QAAQ,CAAA;IAEnC,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE;QACpD,UAAU,EAAE,EAAE;QACd,QAAQ,EAAE,GAAG,EAAE,CAAC,WAAW,EAAE;KAC7B,CAAC,CAAA;IAEF,MAAM,WAAW,GAAG,GAAG,EAAE;QACxB,MAAM,KAAK,GAAG,CAAC,WAAW,CAAC,KAAK,CAAA;QAChC,MAAM,IAAI,GAAG,eAAe,CAAC,KAAK,EAAE,QAAQ,CAAC,QAAQ,CAAC,CAAA;QACtD,QAAQ,CAAC,IAAI,GAAG,IAAI,CAAA;QACpB,SAAS,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,KAAK,IAAI,CAAA;QACpC,SAAS,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YACzE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,QAAQ,GAAG,UAAU,CAAA;YACvC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,CAAA;YAC3B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,IAAI,CAAC,SAAS,GAAG,eAAe,CAAC,KAAK,EAAE,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAA;YAC1F,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAA;YACjC,OAAO,IAAI,CAAC,MAAM,CAAA;QACnB,CAAC,CAAC,CAAC,CAAA;IACJ,CAAC,CAAA;IAED,WAAW,CAAC,OAAO,GAAG,WAAW,CAAA;IACjC,QAAQ,CAAC,KAAK,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,QAAQ,CAAC,CAAA;AACxC,CAAC;AAED,SAAS,eAAe,CAAC,KAAa,EAAE,QAAgB;IACvD,OAAO,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAA;AAC3C,CAAC"}
package/x/index.html CHANGED
@@ -295,7 +295,7 @@ canvas {
295
295
  }
296
296
  }
297
297
  </style>
298
- <script type=module src="demo/demo.bundle.min.js?v=e5ad0d42200e"></script>
298
+ <script type=module src="demo/demo.bundle.min.js?v=94749ea134e7"></script>
299
299
 
300
300
 
301
301
  <meta name="theme-color" content="#3cff9c">
@@ -317,7 +317,7 @@ canvas {
317
317
  <section>
318
318
  <header class="hero">
319
319
  <div>
320
- <h1>Omnitool <small>v1.1.0-55</small></h1>
320
+ <h1>Omnitool <small>v1.1.0-56</small></h1>
321
321
  <p>Developer demos for decoding, preview, waveform, playback, and export.</p>
322
322
  </div>
323
323
  </header>
@@ -372,7 +372,7 @@ canvas {
372
372
  </div>
373
373
  <div class="waveform-controls">
374
374
  <label>width</label>
375
- <input class="width" type="range" min="100" max="1000000" value="1000" />
375
+ <input class="width" type="range" min="100" max="1000000" value="1000" step="100" />
376
376
  </div>
377
377
  <div class="waveform-canvas"></div>
378
378
  </article>
package/x/index.html.js CHANGED
@@ -79,7 +79,7 @@ export default ssg.page(import.meta.url, async (orb) => ({
79
79
  </div>
80
80
  <div class="waveform-controls">
81
81
  <label>width</label>
82
- <input class="width" type="range" min="100" max="1000000" value="1000" />
82
+ <input class="width" type="range" min="100" max="1000000" value="1000" step="100" />
83
83
  </div>
84
84
  <div class="waveform-canvas"></div>
85
85
  </article>
@@ -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 }).readable.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,QAAQ,CAAC,SAAS,EAAE,CAAA;QAEhE,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":""}