@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.
- package/package.json +1 -1
- package/s/demo/demo.bundle.ts +5 -3
- package/s/demo/routines/waveform-test.ts +30 -5
- package/s/index.html.ts +1 -2
- 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 +152 -0
- package/s/timeline/parts/waveform.ts +0 -61
- package/x/demo/demo.bundle.js +2 -0
- package/x/demo/demo.bundle.js.map +1 -1
- package/x/demo/demo.bundle.min.js +11 -84
- package/x/demo/demo.bundle.min.js.map +4 -4
- package/x/demo/routines/waveform-test.js +24 -4
- package/x/demo/routines/waveform-test.js.map +1 -1
- package/x/index.html +3 -3
- package/x/index.html.js +1 -1
- 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 +2 -0
- package/x/timeline/parts/waveform/parts/types.js.map +1 -0
- package/x/timeline/parts/waveform/waveform.d.ts +17 -0
- package/x/timeline/parts/waveform/waveform.js +125 -0
- package/x/timeline/parts/waveform/waveform.js.map +1 -0
- package/x/timeline/parts/waveform.d.ts +1 -9
- package/x/timeline/parts/waveform.js +1 -48
- 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
|
-
|
|
7
|
-
|
|
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
|
-
|
|
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":"
|
|
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=
|
|
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-
|
|
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,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 @@
|
|
|
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
|
-
|
|
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
|
-
|
|
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":"
|
|
1
|
+
{"version":3,"file":"waveform.js","sourceRoot":"","sources":["../../../s/timeline/parts/waveform.ts"],"names":[],"mappings":""}
|