@remotion/studio 4.0.460 → 4.0.462

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 (86) hide show
  1. package/dist/Studio.d.ts +0 -1
  2. package/dist/Studio.js +4 -4
  3. package/dist/components/AudioWaveform.js +17 -9
  4. package/dist/components/ExpandedTracksProvider.js +8 -3
  5. package/dist/components/MenuBuildIndicator.js +1 -1
  6. package/dist/components/MenuCompositionName.js +1 -0
  7. package/dist/components/RenderButton.js +1 -1
  8. package/dist/components/Timeline/SequencePropsObserver.js +1 -1
  9. package/dist/components/Timeline/SubscribeToNodePaths.d.ts +4 -3
  10. package/dist/components/Timeline/SubscribeToNodePaths.js +11 -1
  11. package/dist/components/Timeline/Timeline.js +11 -10
  12. package/dist/components/Timeline/TimelineBooleanField.d.ts +4 -4
  13. package/dist/components/Timeline/TimelineBooleanField.js +5 -5
  14. package/dist/components/Timeline/TimelineColorField.d.ts +11 -0
  15. package/dist/components/Timeline/TimelineColorField.js +181 -0
  16. package/dist/components/Timeline/TimelineDragHandler.js +37 -3
  17. package/dist/components/Timeline/TimelineEffectFieldRow.d.ts +11 -0
  18. package/dist/components/Timeline/TimelineEffectFieldRow.js +177 -0
  19. package/dist/components/Timeline/TimelineEnumField.d.ts +5 -5
  20. package/dist/components/Timeline/TimelineEnumField.js +7 -7
  21. package/dist/components/Timeline/TimelineExpandArrowButton.js +1 -1
  22. package/dist/components/Timeline/TimelineExpandedRow.d.ts +3 -3
  23. package/dist/components/Timeline/TimelineExpandedRow.js +8 -1
  24. package/dist/components/Timeline/TimelineExpandedSection.d.ts +2 -2
  25. package/dist/components/Timeline/TimelineExpandedSection.js +8 -21
  26. package/dist/components/Timeline/TimelineFieldRow.d.ts +3 -3
  27. package/dist/components/Timeline/TimelineFieldRow.js +65 -26
  28. package/dist/components/Timeline/TimelineHeightContainer.d.ts +1 -1
  29. package/dist/components/Timeline/TimelineHeightContainer.js +36 -3
  30. package/dist/components/Timeline/TimelineListItem.js +20 -18
  31. package/dist/components/Timeline/TimelineNumberField.d.ts +5 -5
  32. package/dist/components/Timeline/TimelineNumberField.js +12 -10
  33. package/dist/components/Timeline/TimelineRotationField.d.ts +5 -5
  34. package/dist/components/Timeline/TimelineRotationField.js +10 -10
  35. package/dist/components/Timeline/TimelineSchemaField.d.ts +11 -7
  36. package/dist/components/Timeline/TimelineSchemaField.js +27 -21
  37. package/dist/components/Timeline/TimelineSequence.js +2 -2
  38. package/dist/components/Timeline/TimelineSlider.js +2 -2
  39. package/dist/components/Timeline/TimelineStack/get-stack.js +17 -31
  40. package/dist/components/Timeline/TimelineStack/index.js +0 -10
  41. package/dist/components/Timeline/TimelineTimeIndicators.js +2 -2
  42. package/dist/components/Timeline/TimelineTracks.d.ts +1 -1
  43. package/dist/components/Timeline/TimelineTracks.js +42 -12
  44. package/dist/components/Timeline/TimelineTranslateField.d.ts +5 -5
  45. package/dist/components/Timeline/TimelineTranslateField.js +21 -37
  46. package/dist/components/Timeline/sequence-props-subscription-store.d.ts +2 -1
  47. package/dist/components/Timeline/sequence-props-subscription-store.js +11 -5
  48. package/dist/components/Timeline/use-sequence-props-subscription.d.ts +2 -1
  49. package/dist/components/Timeline/use-sequence-props-subscription.js +6 -4
  50. package/dist/components/Timeline/use-timeline-height.js +5 -14
  51. package/dist/error-overlay/react-overlay/utils/get-source-map.d.ts +3 -3
  52. package/dist/error-overlay/react-overlay/utils/get-source-map.js +5 -5
  53. package/dist/error-overlay/react-overlay/utils/unmapper.d.ts +0 -6
  54. package/dist/error-overlay/react-overlay/utils/unmapper.js +8 -1
  55. package/dist/esm/{chunk-nrgs0ad5.js → chunk-yvg1f56k.js} +4168 -3948
  56. package/dist/esm/index.mjs +20 -8
  57. package/dist/esm/internals.mjs +4168 -3948
  58. package/dist/esm/previewEntry.mjs +2913 -2705
  59. package/dist/esm/renderEntry.mjs +2 -5
  60. package/dist/helpers/calculate-timeline.js +34 -4
  61. package/dist/helpers/get-timeline-sequence-sort-key.d.ts +4 -2
  62. package/dist/helpers/timeline-layout.d.ts +11 -15
  63. package/dist/helpers/timeline-layout.js +36 -28
  64. package/dist/icons/eyedropper.d.ts +4 -0
  65. package/dist/icons/eyedropper.js +6 -0
  66. package/dist/internals.d.ts +0 -1
  67. package/dist/previewEntry.js +1 -1
  68. package/dist/renderEntry.js +3 -3
  69. package/package.json +14 -24
  70. package/dist/audio-waveform-worker.d.ts +0 -1
  71. package/dist/audio-waveform-worker.js +0 -102
  72. package/dist/components/audio-waveform-worker-types.d.ts +0 -28
  73. package/dist/components/audio-waveform-worker-types.js +0 -2
  74. package/dist/components/draw-peaks.d.ts +0 -1
  75. package/dist/components/draw-peaks.js +0 -68
  76. package/dist/components/load-waveform-peaks.d.ts +0 -13
  77. package/dist/components/load-waveform-peaks.js +0 -76
  78. package/dist/components/parse-color.d.ts +0 -1
  79. package/dist/components/parse-color.js +0 -17
  80. package/dist/components/slice-waveform-peaks.d.ts +0 -7
  81. package/dist/components/slice-waveform-peaks.js +0 -15
  82. package/dist/components/waveform-peak-processor.d.ts +0 -23
  83. package/dist/components/waveform-peak-processor.js +0 -77
  84. package/dist/esm/audio-waveform-worker.mjs +0 -346
  85. package/dist/make-audio-waveform-worker.d.ts +0 -1
  86. package/dist/make-audio-waveform-worker.js +0 -10
@@ -1,76 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.TARGET_SAMPLE_RATE = void 0;
4
- exports.loadWaveformPeaks = loadWaveformPeaks;
5
- const mediabunny_1 = require("mediabunny");
6
- const waveform_peak_processor_1 = require("./waveform-peak-processor");
7
- const TARGET_SAMPLE_RATE = 100;
8
- exports.TARGET_SAMPLE_RATE = TARGET_SAMPLE_RATE;
9
- const DEFAULT_PROGRESS_INTERVAL_IN_MS = 50;
10
- const peaksCache = new Map();
11
- async function loadWaveformPeaks(url, signal, options) {
12
- var _a, _b;
13
- const cached = peaksCache.get(url);
14
- if (cached) {
15
- (0, waveform_peak_processor_1.emitWaveformProgress)({
16
- peaks: cached,
17
- completedPeaks: cached.length,
18
- totalPeaks: cached.length,
19
- final: true,
20
- onProgress: options === null || options === void 0 ? void 0 : options.onProgress,
21
- });
22
- return cached;
23
- }
24
- const input = new mediabunny_1.Input({
25
- formats: mediabunny_1.ALL_FORMATS,
26
- source: new mediabunny_1.UrlSource(url),
27
- });
28
- try {
29
- const audioTrack = await input.getPrimaryAudioTrack();
30
- if (!audioTrack) {
31
- return new Float32Array(0);
32
- }
33
- if (await audioTrack.isLive()) {
34
- throw new Error('Live streams are not currently supported by Remotion. Sorry! Source: ' +
35
- url);
36
- }
37
- if (await audioTrack.isRelativeToUnixEpoch()) {
38
- throw new Error('Streams with UNIX timestamps are not currently supported by Remotion. Sorry! Source: ' +
39
- url);
40
- }
41
- const sampleRate = await audioTrack.getSampleRate();
42
- const durationInSeconds = (_a = (await audioTrack.getDurationFromMetadata({ skipLiveWait: true }))) !== null && _a !== void 0 ? _a : (await audioTrack.computeDuration({ skipLiveWait: true }));
43
- const totalPeaks = Math.ceil(durationInSeconds * TARGET_SAMPLE_RATE);
44
- const samplesPerPeak = Math.max(1, Math.floor(sampleRate / TARGET_SAMPLE_RATE));
45
- const sink = new mediabunny_1.AudioSampleSink(audioTrack);
46
- const processor = (0, waveform_peak_processor_1.createWaveformPeakProcessor)({
47
- totalPeaks,
48
- samplesPerPeak,
49
- onProgress: options === null || options === void 0 ? void 0 : options.onProgress,
50
- progressIntervalInMs: (_b = options === null || options === void 0 ? void 0 : options.progressIntervalInMs) !== null && _b !== void 0 ? _b : DEFAULT_PROGRESS_INTERVAL_IN_MS,
51
- now: () => Date.now(),
52
- });
53
- for await (const sample of sink.samples()) {
54
- if (signal.aborted) {
55
- sample.close();
56
- return new Float32Array(0);
57
- }
58
- const bytesNeeded = sample.allocationSize({
59
- format: 'f32',
60
- planeIndex: 0,
61
- });
62
- const floats = new Float32Array(bytesNeeded / 4);
63
- sample.copyTo(floats, { format: 'f32', planeIndex: 0 });
64
- const channels = Math.max(1, sample.numberOfChannels);
65
- sample.close();
66
- processor.processSampleChunk(floats, channels);
67
- }
68
- processor.finalize();
69
- const { peaks } = processor;
70
- peaksCache.set(url, peaks);
71
- return peaks;
72
- }
73
- finally {
74
- input.dispose();
75
- }
76
- }
@@ -1 +0,0 @@
1
- export declare const parseColor: (color: string) => [number, number, number, number];
@@ -1,17 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.parseColor = void 0;
4
- const colorCache = new Map();
5
- const parseColor = (color) => {
6
- const cached = colorCache.get(color);
7
- if (cached)
8
- return cached;
9
- const ctx = new OffscreenCanvas(1, 1).getContext('2d');
10
- ctx.fillStyle = color;
11
- ctx.fillRect(0, 0, 1, 1);
12
- const [r, g, b, a] = ctx.getImageData(0, 0, 1, 1).data;
13
- const result = [r, g, b, a];
14
- colorCache.set(color, result);
15
- return result;
16
- };
17
- exports.parseColor = parseColor;
@@ -1,7 +0,0 @@
1
- export declare const sliceWaveformPeaks: ({ durationInFrames, fps, peaks, playbackRate, startFrom, }: {
2
- readonly peaks: Float32Array<ArrayBufferLike>;
3
- readonly startFrom: number;
4
- readonly durationInFrames: number;
5
- readonly fps: number;
6
- readonly playbackRate: number;
7
- }) => Float32Array<ArrayBufferLike>;
@@ -1,15 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.sliceWaveformPeaks = void 0;
4
- const load_waveform_peaks_1 = require("./load-waveform-peaks");
5
- const sliceWaveformPeaks = ({ durationInFrames, fps, peaks, playbackRate, startFrom, }) => {
6
- if (peaks.length === 0) {
7
- return peaks;
8
- }
9
- const startTimeInSeconds = startFrom / fps;
10
- const durationInSeconds = (durationInFrames / fps) * playbackRate;
11
- const startPeakIndex = Math.floor(startTimeInSeconds * load_waveform_peaks_1.TARGET_SAMPLE_RATE);
12
- const endPeakIndex = Math.ceil((startTimeInSeconds + durationInSeconds) * load_waveform_peaks_1.TARGET_SAMPLE_RATE);
13
- return peaks.subarray(Math.max(0, startPeakIndex), Math.min(peaks.length, endPeakIndex));
14
- };
15
- exports.sliceWaveformPeaks = sliceWaveformPeaks;
@@ -1,23 +0,0 @@
1
- type Progress = {
2
- readonly peaks: Float32Array;
3
- readonly completedPeaks: number;
4
- readonly totalPeaks: number;
5
- readonly final: boolean;
6
- };
7
- type WaveformPeakProcessorOptions = {
8
- readonly totalPeaks: number;
9
- readonly samplesPerPeak: number;
10
- readonly onProgress?: (progress: Progress) => void;
11
- readonly progressIntervalInMs: number;
12
- readonly now: () => number;
13
- };
14
- type WaveformPeakProcessor = {
15
- readonly peaks: Float32Array;
16
- processSampleChunk: (floats: Float32Array, channels: number) => void;
17
- finalize: () => void;
18
- };
19
- export declare const emitWaveformProgress: ({ completedPeaks, final, onProgress, peaks, totalPeaks, }: Progress & {
20
- readonly onProgress?: ((progress: Progress) => void) | undefined;
21
- }) => void;
22
- export declare const createWaveformPeakProcessor: ({ totalPeaks, samplesPerPeak, onProgress, progressIntervalInMs, now, }: WaveformPeakProcessorOptions) => WaveformPeakProcessor;
23
- export {};
@@ -1,77 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.createWaveformPeakProcessor = exports.emitWaveformProgress = void 0;
4
- const emitWaveformProgress = ({ completedPeaks, final, onProgress, peaks, totalPeaks, }) => {
5
- onProgress === null || onProgress === void 0 ? void 0 : onProgress({
6
- peaks,
7
- completedPeaks,
8
- totalPeaks,
9
- final,
10
- });
11
- };
12
- exports.emitWaveformProgress = emitWaveformProgress;
13
- const createWaveformPeakProcessor = ({ totalPeaks, samplesPerPeak, onProgress, progressIntervalInMs, now, }) => {
14
- const peaks = new Float32Array(totalPeaks);
15
- let peakIndex = 0;
16
- let peakMax = 0;
17
- let sampleInPeak = 0;
18
- let lastProgressAt = 0;
19
- let lastProgressPeak = 0;
20
- const emitProgress = (force) => {
21
- const timestamp = now();
22
- if (!force && peakIndex === lastProgressPeak && sampleInPeak === 0) {
23
- return;
24
- }
25
- if (!force && timestamp - lastProgressAt < progressIntervalInMs) {
26
- return;
27
- }
28
- lastProgressAt = timestamp;
29
- lastProgressPeak = peakIndex;
30
- (0, exports.emitWaveformProgress)({
31
- peaks,
32
- completedPeaks: peakIndex,
33
- totalPeaks,
34
- final: force,
35
- onProgress,
36
- });
37
- };
38
- return {
39
- peaks,
40
- processSampleChunk: (floats, channels) => {
41
- var _a;
42
- const frameCount = Math.floor(floats.length / Math.max(1, channels));
43
- for (let frame = 0; frame < frameCount; frame++) {
44
- // `f32` copies are interleaved, so timing advances per frame.
45
- let framePeak = 0;
46
- for (let channel = 0; channel < channels; channel++) {
47
- const sampleIndex = frame * channels + channel;
48
- const abs = Math.abs((_a = floats[sampleIndex]) !== null && _a !== void 0 ? _a : 0);
49
- if (abs > framePeak) {
50
- framePeak = abs;
51
- }
52
- }
53
- if (framePeak > peakMax) {
54
- peakMax = framePeak;
55
- }
56
- sampleInPeak++;
57
- if (sampleInPeak >= samplesPerPeak) {
58
- if (peakIndex < totalPeaks) {
59
- peaks[peakIndex] = peakMax;
60
- }
61
- peakIndex++;
62
- peakMax = 0;
63
- sampleInPeak = 0;
64
- }
65
- }
66
- emitProgress(false);
67
- },
68
- finalize: () => {
69
- if (sampleInPeak > 0 && peakIndex < totalPeaks) {
70
- peaks[peakIndex] = peakMax;
71
- peakIndex++;
72
- }
73
- emitProgress(true);
74
- },
75
- };
76
- };
77
- exports.createWaveformPeakProcessor = createWaveformPeakProcessor;
@@ -1,346 +0,0 @@
1
- // src/audio-waveform-worker.ts
2
- import {
3
- getLoopDisplayWidth,
4
- shouldTileLoopDisplay
5
- } from "@remotion/timeline-utils";
6
-
7
- // src/components/parse-color.ts
8
- var colorCache = new Map;
9
- var parseColor = (color) => {
10
- const cached = colorCache.get(color);
11
- if (cached)
12
- return cached;
13
- const ctx = new OffscreenCanvas(1, 1).getContext("2d");
14
- ctx.fillStyle = color;
15
- ctx.fillRect(0, 0, 1, 1);
16
- const [r, g, b, a] = ctx.getImageData(0, 0, 1, 1).data;
17
- const result = [r, g, b, a];
18
- colorCache.set(color, result);
19
- return result;
20
- };
21
-
22
- // src/components/draw-peaks.ts
23
- var CLIPPING_COLOR = "#FF7F50";
24
- var drawBars = (canvas, peaks, color, volume, width) => {
25
- const ctx = canvas.getContext("2d");
26
- if (!ctx) {
27
- throw new Error("Failed to get canvas context");
28
- }
29
- const { height } = canvas;
30
- const w = canvas.width;
31
- if (w === 0 || height === 0) {
32
- return;
33
- }
34
- ctx.clearRect(0, 0, w, height);
35
- if (volume === 0)
36
- return;
37
- const [r, g, b, a] = parseColor(color);
38
- const [cr, cg, cb, ca] = parseColor(CLIPPING_COLOR);
39
- const imageData = ctx.createImageData(w, height);
40
- const { data } = imageData;
41
- const numBars = width;
42
- for (let barIndex = 0;barIndex < numBars; barIndex++) {
43
- const x = barIndex;
44
- if (x >= w)
45
- break;
46
- const peakIndex = Math.floor(barIndex / numBars * peaks.length);
47
- const peak = peaks[peakIndex] || 0;
48
- const scaledPeak = peak * volume;
49
- const halfBar = Math.max(0, Math.min(height / 2, scaledPeak * height / 2));
50
- if (halfBar === 0)
51
- continue;
52
- const mid = height / 2;
53
- const barY = Math.round(mid - halfBar);
54
- const barEnd = Math.round(mid + halfBar);
55
- const isClipping = scaledPeak > 1;
56
- const clipTopEnd = isClipping ? Math.min(barY + 2, barEnd) : barY;
57
- const clipBotStart = isClipping ? Math.max(barEnd - 2, barY) : barEnd;
58
- for (let y = barY;y < clipTopEnd; y++) {
59
- const idx = (y * w + x) * 4;
60
- data[idx] = cr;
61
- data[idx + 1] = cg;
62
- data[idx + 2] = cb;
63
- data[idx + 3] = ca;
64
- }
65
- for (let y = clipTopEnd;y < clipBotStart; y++) {
66
- const idx = (y * w + x) * 4;
67
- data[idx] = r;
68
- data[idx + 1] = g;
69
- data[idx + 2] = b;
70
- data[idx + 3] = a;
71
- }
72
- for (let y = clipBotStart;y < barEnd; y++) {
73
- const idx = (y * w + x) * 4;
74
- data[idx] = cr;
75
- data[idx + 1] = cg;
76
- data[idx + 2] = cb;
77
- data[idx + 3] = ca;
78
- }
79
- }
80
- ctx.putImageData(imageData, 0, 0);
81
- };
82
-
83
- // src/components/load-waveform-peaks.ts
84
- import { ALL_FORMATS, AudioSampleSink, Input, UrlSource } from "mediabunny";
85
-
86
- // src/components/waveform-peak-processor.ts
87
- var emitWaveformProgress = ({
88
- completedPeaks,
89
- final,
90
- onProgress,
91
- peaks,
92
- totalPeaks
93
- }) => {
94
- onProgress?.({
95
- peaks,
96
- completedPeaks,
97
- totalPeaks,
98
- final
99
- });
100
- };
101
- var createWaveformPeakProcessor = ({
102
- totalPeaks,
103
- samplesPerPeak,
104
- onProgress,
105
- progressIntervalInMs,
106
- now
107
- }) => {
108
- const peaks = new Float32Array(totalPeaks);
109
- let peakIndex = 0;
110
- let peakMax = 0;
111
- let sampleInPeak = 0;
112
- let lastProgressAt = 0;
113
- let lastProgressPeak = 0;
114
- const emitProgress = (force) => {
115
- const timestamp = now();
116
- if (!force && peakIndex === lastProgressPeak && sampleInPeak === 0) {
117
- return;
118
- }
119
- if (!force && timestamp - lastProgressAt < progressIntervalInMs) {
120
- return;
121
- }
122
- lastProgressAt = timestamp;
123
- lastProgressPeak = peakIndex;
124
- emitWaveformProgress({
125
- peaks,
126
- completedPeaks: peakIndex,
127
- totalPeaks,
128
- final: force,
129
- onProgress
130
- });
131
- };
132
- return {
133
- peaks,
134
- processSampleChunk: (floats, channels) => {
135
- const frameCount = Math.floor(floats.length / Math.max(1, channels));
136
- for (let frame = 0;frame < frameCount; frame++) {
137
- let framePeak = 0;
138
- for (let channel = 0;channel < channels; channel++) {
139
- const sampleIndex = frame * channels + channel;
140
- const abs = Math.abs(floats[sampleIndex] ?? 0);
141
- if (abs > framePeak) {
142
- framePeak = abs;
143
- }
144
- }
145
- if (framePeak > peakMax) {
146
- peakMax = framePeak;
147
- }
148
- sampleInPeak++;
149
- if (sampleInPeak >= samplesPerPeak) {
150
- if (peakIndex < totalPeaks) {
151
- peaks[peakIndex] = peakMax;
152
- }
153
- peakIndex++;
154
- peakMax = 0;
155
- sampleInPeak = 0;
156
- }
157
- }
158
- emitProgress(false);
159
- },
160
- finalize: () => {
161
- if (sampleInPeak > 0 && peakIndex < totalPeaks) {
162
- peaks[peakIndex] = peakMax;
163
- peakIndex++;
164
- }
165
- emitProgress(true);
166
- }
167
- };
168
- };
169
-
170
- // src/components/load-waveform-peaks.ts
171
- var TARGET_SAMPLE_RATE = 100;
172
- var DEFAULT_PROGRESS_INTERVAL_IN_MS = 50;
173
- var peaksCache = new Map;
174
- async function loadWaveformPeaks(url, signal, options) {
175
- const cached = peaksCache.get(url);
176
- if (cached) {
177
- emitWaveformProgress({
178
- peaks: cached,
179
- completedPeaks: cached.length,
180
- totalPeaks: cached.length,
181
- final: true,
182
- onProgress: options?.onProgress
183
- });
184
- return cached;
185
- }
186
- const input = new Input({
187
- formats: ALL_FORMATS,
188
- source: new UrlSource(url)
189
- });
190
- try {
191
- const audioTrack = await input.getPrimaryAudioTrack();
192
- if (!audioTrack) {
193
- return new Float32Array(0);
194
- }
195
- if (await audioTrack.isLive()) {
196
- throw new Error("Live streams are not currently supported by Remotion. Sorry! Source: " + url);
197
- }
198
- if (await audioTrack.isRelativeToUnixEpoch()) {
199
- throw new Error("Streams with UNIX timestamps are not currently supported by Remotion. Sorry! Source: " + url);
200
- }
201
- const sampleRate = await audioTrack.getSampleRate();
202
- const durationInSeconds = await audioTrack.getDurationFromMetadata({ skipLiveWait: true }) ?? await audioTrack.computeDuration({ skipLiveWait: true });
203
- const totalPeaks = Math.ceil(durationInSeconds * TARGET_SAMPLE_RATE);
204
- const samplesPerPeak = Math.max(1, Math.floor(sampleRate / TARGET_SAMPLE_RATE));
205
- const sink = new AudioSampleSink(audioTrack);
206
- const processor = createWaveformPeakProcessor({
207
- totalPeaks,
208
- samplesPerPeak,
209
- onProgress: options?.onProgress,
210
- progressIntervalInMs: options?.progressIntervalInMs ?? DEFAULT_PROGRESS_INTERVAL_IN_MS,
211
- now: () => Date.now()
212
- });
213
- for await (const sample of sink.samples()) {
214
- if (signal.aborted) {
215
- sample.close();
216
- return new Float32Array(0);
217
- }
218
- const bytesNeeded = sample.allocationSize({
219
- format: "f32",
220
- planeIndex: 0
221
- });
222
- const floats = new Float32Array(bytesNeeded / 4);
223
- sample.copyTo(floats, { format: "f32", planeIndex: 0 });
224
- const channels = Math.max(1, sample.numberOfChannels);
225
- sample.close();
226
- processor.processSampleChunk(floats, channels);
227
- }
228
- processor.finalize();
229
- const { peaks } = processor;
230
- peaksCache.set(url, peaks);
231
- return peaks;
232
- } finally {
233
- input.dispose();
234
- }
235
- }
236
-
237
- // src/components/slice-waveform-peaks.ts
238
- var sliceWaveformPeaks = ({
239
- durationInFrames,
240
- fps,
241
- peaks,
242
- playbackRate,
243
- startFrom
244
- }) => {
245
- if (peaks.length === 0) {
246
- return peaks;
247
- }
248
- const startTimeInSeconds = startFrom / fps;
249
- const durationInSeconds = durationInFrames / fps * playbackRate;
250
- const startPeakIndex = Math.floor(startTimeInSeconds * TARGET_SAMPLE_RATE);
251
- const endPeakIndex = Math.ceil((startTimeInSeconds + durationInSeconds) * TARGET_SAMPLE_RATE);
252
- return peaks.subarray(Math.max(0, startPeakIndex), Math.min(peaks.length, endPeakIndex));
253
- };
254
-
255
- // src/audio-waveform-worker.ts
256
- var canvas = null;
257
- var currentController = null;
258
- var latestRequestId = 0;
259
- var postError = (requestId, error) => {
260
- const message = error instanceof Error ? error.message : "Failed to render waveform";
261
- const payload = {
262
- type: "error",
263
- requestId,
264
- message
265
- };
266
- self.postMessage(payload);
267
- };
268
- var drawPartialWaveform = (message, peaks) => {
269
- if (!canvas) {
270
- return;
271
- }
272
- const portionPeaks = sliceWaveformPeaks({
273
- durationInFrames: shouldTileLoopDisplay(message.loopDisplay) ? message.loopDisplay.durationInFrames : message.durationInFrames,
274
- fps: message.fps,
275
- peaks,
276
- playbackRate: message.playbackRate,
277
- startFrom: message.startFrom
278
- });
279
- if (!shouldTileLoopDisplay(message.loopDisplay)) {
280
- drawBars(canvas, portionPeaks, "rgba(255, 255, 255, 0.6)", message.volume, message.width);
281
- return;
282
- }
283
- const loopWidth = getLoopDisplayWidth({
284
- visualizationWidth: message.width,
285
- loopDisplay: message.loopDisplay
286
- });
287
- const targetCanvas = new OffscreenCanvas(Math.max(1, Math.ceil(loopWidth)), message.height);
288
- drawBars(targetCanvas, portionPeaks, "rgba(255, 255, 255, 0.6)", message.volume, targetCanvas.width);
289
- const ctx = canvas.getContext("2d");
290
- if (!ctx) {
291
- throw new Error("Failed to get canvas context");
292
- }
293
- const pattern = ctx.createPattern(targetCanvas, "repeat-x");
294
- if (!pattern) {
295
- return;
296
- }
297
- pattern.setTransform(new DOMMatrix().scaleSelf(loopWidth / targetCanvas.width, 1));
298
- ctx.clearRect(0, 0, message.width, message.height);
299
- ctx.fillStyle = pattern;
300
- ctx.fillRect(0, 0, message.width, message.height);
301
- };
302
- var renderWaveform = async (message) => {
303
- if (!canvas) {
304
- postError(message.requestId, new Error("Waveform canvas not initialized"));
305
- return;
306
- }
307
- const controller = new AbortController;
308
- currentController?.abort();
309
- currentController = controller;
310
- latestRequestId = message.requestId;
311
- try {
312
- canvas.width = message.width;
313
- canvas.height = message.height;
314
- const peaks = await loadWaveformPeaks(message.src, controller.signal, {
315
- onProgress: ({ peaks: nextPeaks }) => {
316
- if (controller.signal.aborted || latestRequestId !== message.requestId) {
317
- return;
318
- }
319
- drawPartialWaveform(message, nextPeaks);
320
- }
321
- });
322
- if (controller.signal.aborted || latestRequestId !== message.requestId) {
323
- return;
324
- }
325
- drawPartialWaveform(message, peaks);
326
- } catch (error) {
327
- if (controller.signal.aborted || latestRequestId !== message.requestId) {
328
- return;
329
- }
330
- postError(message.requestId, error);
331
- }
332
- };
333
- self.addEventListener("message", (event) => {
334
- const message = event.data;
335
- if (message.type === "init") {
336
- canvas = message.canvas;
337
- return;
338
- }
339
- if (message.type === "dispose") {
340
- currentController?.abort();
341
- currentController = null;
342
- canvas = null;
343
- return;
344
- }
345
- renderWaveform(message);
346
- });
@@ -1 +0,0 @@
1
- export declare const makeAudioWaveformWorker: () => Worker;
@@ -1,10 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.makeAudioWaveformWorker = void 0;
4
- const makeAudioWaveformWorker = () => {
5
- // @ts-expect-error `import.meta.url` is required for bundling the worker entry.
6
- return new Worker(new URL('./audio-waveform-worker.mjs', import.meta.url), {
7
- type: 'module',
8
- });
9
- };
10
- exports.makeAudioWaveformWorker = makeAudioWaveformWorker;