speechrecorderng 3.10.13 → 3.11.0

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 (99) hide show
  1. package/README.md +1 -1
  2. package/fesm2022/speechrecorderng.mjs +204 -201
  3. package/fesm2022/speechrecorderng.mjs.map +1 -1
  4. package/lib/audio/ui/audio_canvas_layer_comp.d.ts +2 -2
  5. package/lib/spr.module.version.d.ts +1 -1
  6. package/lib/ui/canvas_layer_comp.d.ts +1 -1
  7. package/lib/ui/responsive_component.d.ts +1 -1
  8. package/package.json +13 -15
  9. package/esm2022/lib/action/action.mjs +0 -73
  10. package/esm2022/lib/audio/array_audio_buffer.mjs +0 -164
  11. package/esm2022/lib/audio/array_audio_buffer_input_stream.mjs +0 -86
  12. package/esm2022/lib/audio/array_audio_buffer_random_access_stream.mjs +0 -16
  13. package/esm2022/lib/audio/audio_data_holder.mjs +0 -264
  14. package/esm2022/lib/audio/audio_display.mjs +0 -118
  15. package/esm2022/lib/audio/audio_player.mjs +0 -238
  16. package/esm2022/lib/audio/capture/capture.mjs +0 -855
  17. package/esm2022/lib/audio/context.mjs +0 -79
  18. package/esm2022/lib/audio/dsp/level_measure.mjs +0 -517
  19. package/esm2022/lib/audio/format.mjs +0 -20
  20. package/esm2022/lib/audio/impl/wavformat.mjs +0 -7
  21. package/esm2022/lib/audio/impl/wavreader.mjs +0 -144
  22. package/esm2022/lib/audio/impl/wavwriter.mjs +0 -191
  23. package/esm2022/lib/audio/inddb_audio_buffer.mjs +0 -508
  24. package/esm2022/lib/audio/io/stream.mjs +0 -59
  25. package/esm2022/lib/audio/net_audio_buffer.mjs +0 -293
  26. package/esm2022/lib/audio/persistor.mjs +0 -81
  27. package/esm2022/lib/audio/playback/array_audio_buffer_source_node.mjs +0 -126
  28. package/esm2022/lib/audio/playback/audio_source_node.mjs +0 -18
  29. package/esm2022/lib/audio/playback/audio_source_worklet_module_loader.mjs +0 -167
  30. package/esm2022/lib/audio/playback/inddb_audio_buffer_source_node.mjs +0 -167
  31. package/esm2022/lib/audio/playback/net_audio_buffer_source_node.mjs +0 -218
  32. package/esm2022/lib/audio/playback/player.mjs +0 -402
  33. package/esm2022/lib/audio/ui/audio_canvas_layer_comp.mjs +0 -347
  34. package/esm2022/lib/audio/ui/audio_display_control.mjs +0 -150
  35. package/esm2022/lib/audio/ui/audio_display_scroll_pane.mjs +0 -146
  36. package/esm2022/lib/audio/ui/audiosignal.mjs +0 -533
  37. package/esm2022/lib/audio/ui/common.mjs +0 -19
  38. package/esm2022/lib/audio/ui/container.mjs +0 -414
  39. package/esm2022/lib/audio/ui/livelevel.mjs +0 -361
  40. package/esm2022/lib/audio/ui/scroll_pane_horizontal.mjs +0 -11
  41. package/esm2022/lib/audio/ui/sonagram.mjs +0 -900
  42. package/esm2022/lib/db/inddb.mjs +0 -120
  43. package/esm2022/lib/dsp/utils.mjs +0 -48
  44. package/esm2022/lib/environment/environment.defaults.mjs +0 -9
  45. package/esm2022/lib/io/BinaryReader.mjs +0 -93
  46. package/esm2022/lib/io/BinaryWriter.mjs +0 -80
  47. package/esm2022/lib/io/stream.mjs +0 -259
  48. package/esm2022/lib/math/2d/geometry.mjs +0 -28
  49. package/esm2022/lib/math/complex.mjs +0 -58
  50. package/esm2022/lib/math/dft.mjs +0 -196
  51. package/esm2022/lib/media/utils.mjs +0 -14
  52. package/esm2022/lib/net/uploader.mjs +0 -367
  53. package/esm2022/lib/recorder_component.mjs +0 -65
  54. package/esm2022/lib/speechrecorder/project/project.mjs +0 -54
  55. package/esm2022/lib/speechrecorder/project/project.service.mjs +0 -64
  56. package/esm2022/lib/speechrecorder/recording.mjs +0 -124
  57. package/esm2022/lib/speechrecorder/recordings/basic_recording.service.mjs +0 -221
  58. package/esm2022/lib/speechrecorder/recordings/recordings.service.mjs +0 -1014
  59. package/esm2022/lib/speechrecorder/script/script.mjs +0 -114
  60. package/esm2022/lib/speechrecorder/script/script.service.mjs +0 -47
  61. package/esm2022/lib/speechrecorder/session/audiorecorder.mjs +0 -1179
  62. package/esm2022/lib/speechrecorder/session/basicrecorder.mjs +0 -676
  63. package/esm2022/lib/speechrecorder/session/controlpanel.mjs +0 -416
  64. package/esm2022/lib/speechrecorder/session/item.mjs +0 -30
  65. package/esm2022/lib/speechrecorder/session/progress.mjs +0 -135
  66. package/esm2022/lib/speechrecorder/session/prompting.mjs +0 -639
  67. package/esm2022/lib/speechrecorder/session/recorder_combi_pane.mjs +0 -88
  68. package/esm2022/lib/speechrecorder/session/recording_file_cache.mjs +0 -195
  69. package/esm2022/lib/speechrecorder/session/recording_list.mjs +0 -188
  70. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +0 -128
  71. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +0 -114
  72. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +0 -146
  73. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +0 -424
  74. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file.mjs +0 -68
  75. package/esm2022/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +0 -288
  76. package/esm2022/lib/speechrecorder/session/session.mjs +0 -2
  77. package/esm2022/lib/speechrecorder/session/session.service.mjs +0 -69
  78. package/esm2022/lib/speechrecorder/session/session_finished_dialog.mjs +0 -46
  79. package/esm2022/lib/speechrecorder/session/sessionmanager.mjs +0 -1385
  80. package/esm2022/lib/speechrecorder/session/warning_bar.mjs +0 -33
  81. package/esm2022/lib/speechrecorder/spruploader.mjs +0 -22
  82. package/esm2022/lib/speechrecorder/startstopsignal/startstopsignal.mjs +0 -2
  83. package/esm2022/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +0 -57
  84. package/esm2022/lib/speechrecorderng.component.mjs +0 -392
  85. package/esm2022/lib/speechrecorderng.module.mjs +0 -97
  86. package/esm2022/lib/spr.config.mjs +0 -27
  87. package/esm2022/lib/spr.module.version.mjs +0 -2
  88. package/esm2022/lib/ui/canvas_layer_comp.mjs +0 -38
  89. package/esm2022/lib/ui/intersection-observer.directive.mjs +0 -32
  90. package/esm2022/lib/ui/message_dialog.mjs +0 -51
  91. package/esm2022/lib/ui/recordingitem_display.mjs +0 -253
  92. package/esm2022/lib/ui/responsive_component.mjs +0 -24
  93. package/esm2022/lib/utils/scrollIntoViewToBottom.mjs +0 -23
  94. package/esm2022/lib/utils/ua-parser.mjs +0 -190
  95. package/esm2022/lib/utils/utils.mjs +0 -132
  96. package/esm2022/lib/utils/wake_lock.mjs +0 -114
  97. package/esm2022/lib/utils/wake_lock_media.mjs +0 -2
  98. package/esm2022/public-api.mjs +0 -35
  99. package/esm2022/speechrecorderng.mjs +0 -5
@@ -1,900 +0,0 @@
1
- import { DFTFloat32 } from '../../math/dft';
2
- import { Point } from './common';
3
- import { Component, ViewChild } from "@angular/core";
4
- import { AudioCanvasLayerComponent } from "./audio_canvas_layer_comp";
5
- import { WorkerHelper } from "../../utils/utils";
6
- import { AudioBufferSource } from "../audio_data_holder";
7
- import * as i0 from "@angular/core";
8
- const DEFAULT_DFT_SIZE = 1024;
9
- export class Sonagram extends AudioCanvasLayerComponent {
10
- constructor(ref) {
11
- super();
12
- this.ref = ref;
13
- this._playFramePosition = null;
14
- this.raAsSubsc = null;
15
- this.dftSize = DEFAULT_DFT_SIZE;
16
- this.worker = null;
17
- this._audioDataHolder = null;
18
- this.markers = new Array();
19
- this.dft = new DFTFloat32(this.dftSize);
20
- this.workerURL = WorkerHelper.buildWorkerBlobURL(this.workerFunction);
21
- this._bgColor = null;
22
- this._selectColor = 'rgba(255,255,0,0.1)';
23
- }
24
- ngAfterViewInit() {
25
- this.ce = this.ref.nativeElement;
26
- this.sonagramCanvas = this.sonagramCanvasRef.nativeElement;
27
- this.sonagramCanvas.style.zIndex = '1';
28
- this.bgCanvas = this.bgCanvasRef.nativeElement;
29
- this.bgCanvas.style.zIndex = '2';
30
- this.cursorCanvas = this.cursorCanvasRef.nativeElement;
31
- this.cursorCanvas.style.zIndex = '4';
32
- this.markerCanvas = this.markerCanvasRef.nativeElement;
33
- this.markerCanvas.style.zIndex = '3';
34
- this.canvasLayers[0] = this.sonagramCanvas;
35
- this.canvasLayers[1] = this.bgCanvas;
36
- this.canvasLayers[2] = this.cursorCanvas;
37
- this.canvasLayers[3] = this.markerCanvas;
38
- }
39
- get playFramePosition() {
40
- return this._playFramePosition;
41
- }
42
- set playFramePosition(playFramePosition) {
43
- this._playFramePosition = playFramePosition;
44
- // this.redraw();
45
- this.drawPlayPosition();
46
- }
47
- canvasMousePos(c, e) {
48
- const cr = c.getBoundingClientRect();
49
- const x = e.x - cr.left;
50
- const y = e.y - cr.top;
51
- return new Point(x, y);
52
- }
53
- drawStateText(stateText) {
54
- if (stateText) {
55
- const g = this.sonagramCanvas.getContext("2d");
56
- const w = this.sonagramCanvas.width;
57
- const h = this.sonagramCanvas.height;
58
- if (g && w && h) {
59
- g.strokeStyle = 'black';
60
- g.fillStyle = 'black';
61
- g.font = '20px sans-serif';
62
- g.fillText(stateText, 10, 25);
63
- }
64
- }
65
- }
66
- drawCursorPosition(e, show) {
67
- if (this.cursorCanvas) {
68
- const w = this.cursorCanvas.width;
69
- const h = this.cursorCanvas.height;
70
- const g = this.cursorCanvas.getContext('2d');
71
- if (g) {
72
- g.clearRect(0, 0, w, h);
73
- if (show) {
74
- //const pp = this.canvasMousePos(this.cursorCanvas, e);
75
- let xViewPortPixelpos = e.offsetX;
76
- g.fillStyle = 'yellow';
77
- g.strokeStyle = 'yellow';
78
- g.beginPath();
79
- g.moveTo(xViewPortPixelpos, 0);
80
- g.lineTo(xViewPortPixelpos, h);
81
- g.closePath();
82
- g.stroke();
83
- if (this._audioDataHolder) {
84
- let framePosRound = this.viewPortXPixelToFramePosition(xViewPortPixelpos);
85
- if (framePosRound != null) {
86
- g.font = '14px sans-serif';
87
- g.fillStyle = 'yellow';
88
- g.fillText(framePosRound.toString(), xViewPortPixelpos + 2, 50);
89
- }
90
- }
91
- }
92
- }
93
- }
94
- }
95
- drawPlayPosition() {
96
- if (this.markerCanvas) {
97
- const w = this.markerCanvas.width;
98
- const h = this.markerCanvas.height;
99
- const g = this.markerCanvas.getContext("2d");
100
- if (g) {
101
- g.clearRect(0, 0, w, h);
102
- if (this._playFramePosition != null) {
103
- let pixelPos = this.frameToViewPortXPixelPosition(this._playFramePosition);
104
- if (pixelPos) {
105
- g.fillStyle = 'red';
106
- g.strokeStyle = 'red';
107
- g.beginPath();
108
- g.moveTo(pixelPos, 0);
109
- g.lineTo(pixelPos, h);
110
- g.closePath();
111
- g.stroke();
112
- }
113
- }
114
- }
115
- }
116
- }
117
- /*
118
- * Method used as worker code.
119
- */
120
- workerFunction() {
121
- // Redefine some DSP classes for worker function
122
- // See e.g. audio.math.Complex
123
- class Complex {
124
- static fromPolarForm(magnitude, argument) {
125
- const r = Math.cos(argument) * magnitude;
126
- const i = Math.sin(argument) * magnitude;
127
- return new Complex(r, i);
128
- }
129
- constructor(real, img) {
130
- this.real = real;
131
- this.img = img;
132
- }
133
- magnitude() {
134
- return Math.sqrt((this.real * this.real) + (this.img * this.img));
135
- }
136
- argument() {
137
- return Math.atan2(this.img, this.real);
138
- }
139
- add(addC) {
140
- return new Complex(this.real + addC.real, this.img + addC.img);
141
- }
142
- sub(subC) {
143
- return new Complex(this.real - subC.real, this.img - subC.img);
144
- }
145
- mult(multC) {
146
- const multR = (this.real * multC.real) - (this.img * multC.img);
147
- const multI = (this.real * multC.img) + (multC.real * this.img);
148
- return new Complex(multR, multI);
149
- }
150
- multReal(multF) {
151
- return new Complex(this.real * multF, this.img * multF);
152
- }
153
- div(divisor) {
154
- const divReal = divisor.real;
155
- const divImg = divisor.img;
156
- const div = (divReal * divReal) + (divImg * divImg);
157
- const divisionReal = ((this.real * divReal) + (this.img * divImg)) / div;
158
- const divisionImg = ((divReal * this.img) - (this.real * divImg)) / div;
159
- return new Complex(divisionReal, divisionImg);
160
- }
161
- divReal(divisor) {
162
- const div = divisor * divisor;
163
- const divsionReal = (this.real * divisor) / div;
164
- const divsionImg = (divisor * this.img) / div;
165
- return new Complex(divsionReal, divsionImg);
166
- }
167
- conjugate() {
168
- return new Complex(this.real, -this.img);
169
- }
170
- equals(c) {
171
- if (c === null) {
172
- return false;
173
- }
174
- return (this.real === c.real && this.img === c.img);
175
- }
176
- toString() {
177
- return 'Real: ' + this.real + ', Img: ' + this.img;
178
- }
179
- }
180
- class DFTFloat32 {
181
- constructor(n) {
182
- this.n = n;
183
- this.m = Math.log(n) / Math.log(2);
184
- // if(n != (1 << m))throw new RuntimeException("length N must be power of 2");
185
- // lookup tables
186
- this.cosLookup = new Float32Array(n / 2);
187
- this.sinLookup = new Float32Array(n / 2);
188
- for (let i = 0; i < n / 2; i++) {
189
- const arc = (-2 * Math.PI * i) / n;
190
- this.cosLookup[i] = Math.cos(arc);
191
- this.sinLookup[i] = Math.sin(arc);
192
- }
193
- }
194
- processReal(srcBuf) {
195
- const x = srcBuf.slice();
196
- const y = new Float32Array(srcBuf.length);
197
- for (let yi = 0; yi < y.length; yi++) {
198
- y[yi] = 0.0;
199
- }
200
- this.fftCooleyTukey(x, y);
201
- const rc = new Array(x.length);
202
- for (let i = 0; i < x.length; i++) {
203
- rc[i] = new Complex(x[i], y[i]);
204
- }
205
- return rc;
206
- }
207
- processRealMagnitude(srcBuf) {
208
- const x = srcBuf.slice();
209
- const y = new Float32Array(srcBuf.length);
210
- for (let yi = 0; yi < y.length; yi++) {
211
- y[yi] = 0.0;
212
- }
213
- this.fftCooleyTukey(x, y);
214
- const rc = new Float32Array(x.length);
215
- for (let i = 0; i < x.length; i++) {
216
- const rcc = new Complex(x[i], y[i]);
217
- rc[i] = rcc.magnitude();
218
- }
219
- return rc;
220
- }
221
- fftCooleyTukey(real, img) {
222
- let i;
223
- let j = 0;
224
- let k;
225
- let n1;
226
- let n2 = this.n / 2;
227
- let a;
228
- let c;
229
- let s;
230
- let t1;
231
- let t2;
232
- for (i = 1; i < this.n - 1; i++) {
233
- n1 = n2;
234
- while (j >= n1) {
235
- j = j - n1;
236
- n1 = n1 / 2;
237
- }
238
- j = j + n1;
239
- if (i < j) {
240
- t1 = real[i];
241
- real[i] = real[j];
242
- real[j] = t1;
243
- t1 = img[i];
244
- img[i] = img[j];
245
- img[j] = t1;
246
- }
247
- }
248
- n1 = 0;
249
- n2 = 1;
250
- for (i = 0; i < this.m; i++) {
251
- n1 = n2;
252
- n2 = n2 + n2;
253
- a = 0;
254
- for (j = 0; j < n1; j++) {
255
- c = this.cosLookup[a];
256
- s = this.sinLookup[a];
257
- a += (1 << (this.m - i - 1));
258
- for (k = j; k < this.n; k = k + n2) {
259
- t1 = c * real[k + n1] - s * img[k + n1];
260
- t2 = s * real[k + n1] + c * img[k + n1];
261
- real[k + n1] = real[k] - t1;
262
- img[k + n1] = img[k] - t2;
263
- real[k] = real[k] + t1;
264
- img[k] = img[k] + t2;
265
- }
266
- }
267
- }
268
- }
269
- process(t) {
270
- const reals = new Float32Array(this.n);
271
- const imgs = new Float32Array(this.n);
272
- const trans = new Array(this.n);
273
- for (let i = 0; i < this.n; i++) {
274
- reals[i] = t[i].real;
275
- imgs[i] = t[i].img;
276
- }
277
- this.fftCooleyTukey(reals, imgs);
278
- for (let i = 0; i < this.n; i++) {
279
- trans[i] = new Complex(reals[i], imgs[i]);
280
- }
281
- return trans;
282
- }
283
- }
284
- class GaussianWindow {
285
- static { this.DEFAULT_SIGMA = 0.3; }
286
- constructor(size, sigma = GaussianWindow.DEFAULT_SIGMA) {
287
- this.buf = new Float32Array(size);
288
- const center = (size - 1) / 2;
289
- for (let i = 0; i < size; i++) {
290
- const quot = (i - center) / (sigma * center);
291
- const exp = -0.5 * quot * quot;
292
- this.buf[i] = Math.exp(exp);
293
- }
294
- }
295
- getScale(i) {
296
- return this.buf[i];
297
- }
298
- }
299
- self.onmessage = function (msg) {
300
- //console.debug("Sonagram render thread");
301
- let l = msg.data.l;
302
- let w = msg.data.w;
303
- let h = msg.data.h;
304
- let vw = msg.data.vw;
305
- let chs = msg.data.chs;
306
- let audioDataOffset = 0;
307
- let adOffset = msg.data.audioDataOffset;
308
- if (adOffset) {
309
- audioDataOffset = adOffset;
310
- }
311
- let maxPsd = null;
312
- if (msg.data.maxPsd !== undefined) {
313
- maxPsd = msg.data.maxPsd;
314
- }
315
- let audioData = new Array(chs);
316
- for (let ch = 0; ch < chs; ch++) {
317
- audioData[ch] = new Float32Array(msg.data['audioData'][ch]);
318
- }
319
- let frameLength = msg.data.frameLength;
320
- let dftSize = msg.data.dftSize;
321
- let dftBands = dftSize / 2;
322
- let dft = new DFTFloat32(dftSize);
323
- let wf = new GaussianWindow(dftSize);
324
- let arrSize = w * h * 4;
325
- if (arrSize < 0) {
326
- arrSize = 0;
327
- }
328
- let imgData = new Uint8ClampedArray(arrSize);
329
- //console.log("Render method:");
330
- //console.debug("Created imgData arrSize: "+arrSize+" ", w, "x", h);
331
- let calcMaxPsd = -Infinity;
332
- if (arrSize > 0) {
333
- let chH = Math.round(h / chs);
334
- let framesPerPixel = frameLength / vw;
335
- //console.debug("Render: ", w, "x", h);
336
- let b = new Float32Array(dftSize);
337
- let sona = new Array(chs);
338
- for (let ch = 0; ch < chs; ch++) {
339
- let chDataLen = audioData[ch].length;
340
- // initialize DFT array buffer
341
- sona[ch] = new Array(w);
342
- let framePos = 0;
343
- for (let pii = 0; pii < w; pii++) {
344
- let virtPii = l + pii;
345
- // Position of sample data frame is pixel position mapped to audio frame position.
346
- // Then "center" the frame by shifting left by half the DFT size (=dftBands)
347
- framePos = Math.round((virtPii * framesPerPixel) - dftBands);
348
- // fill DFT buffer with windowed sample values
349
- for (let i = 0; i < dftSize; i++) {
350
- let samplePos = framePos + i;
351
- // initialize for negative sample positions and out of bounds positions
352
- let chDat = 0;
353
- // Set audio sample if available
354
- let adp = samplePos - audioDataOffset;
355
- if (adp >= 0 && adp < chDataLen) {
356
- chDat = audioData[ch][adp];
357
- //console.debug("Audio data: "+chDat);
358
- }
359
- else {
360
- //console.debug("Sample buf pos oob: adp: "+adp+", chDataLen: "+chDataLen+", samplePos: "+samplePos+", i: "+i);
361
- }
362
- // apply Window
363
- b[i] = chDat * wf.getScale(i);
364
- }
365
- // Calc DFT magnitudes
366
- let spectr = dft.processRealMagnitude(b);
367
- // Get maximum value of spectral energy
368
- for (let s = 0; s < dftBands; s++) {
369
- let psd = (2 * Math.pow(spectr[s], 2)) / dftBands;
370
- if (psd > calcMaxPsd) {
371
- calcMaxPsd = psd;
372
- }
373
- }
374
- // Set render model data for this pixel
375
- sona[ch][pii] = spectr;
376
- }
377
- }
378
- if (!msg.data.norender) {
379
- if (!maxPsd) {
380
- maxPsd = calcMaxPsd;
381
- }
382
- for (let ch = 0; ch < chs; ch++) {
383
- for (let pii = 0; pii < w; pii++) {
384
- let allBlack = true;
385
- for (let y = 0; y < chH; y++) {
386
- let freqIdx = Math.round(y * dftBands / chH);
387
- // calculate the one-sided power spectral density PSD (f, t) in Pa2/Hz
388
- // PSD(f) proportional to 2|X(f)|2 / (t2 - t1)
389
- let val = sona[ch][pii][freqIdx];
390
- let psd = (2 * Math.pow(val, 2)) / dftBands;
391
- // Calculate logarithmic value
392
- //let psdLog = ips.dsp.DSPUtils.toLevelInDB(psd / maxPsd);
393
- let linearLevel = psd / maxPsd;
394
- let psdLog = 10 * Math.log(linearLevel) / Math.log(10);
395
- // Fixed dynamic Range value of 70dB for now
396
- let dynRangeInDb = 70;
397
- let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;
398
- // are the following checks necessary for clamped array ?
399
- if (scaledVal > 1.0)
400
- scaledVal = 1;
401
- if (scaledVal < 0.0) {
402
- scaledVal = 0;
403
- }
404
- let rgbVal = Math.round(255 * scaledVal);
405
- if (rgbVal < 0) {
406
- // System.out.println("Neg RGB val: "+rgbVal);
407
- rgbVal = 0;
408
- }
409
- if (rgbVal > 255) {
410
- rgbVal = 255;
411
- }
412
- rgbVal = 255 - rgbVal;
413
- if (rgbVal > 0) {
414
- allBlack = false;
415
- }
416
- let py = chH - y;
417
- let dataPos = ((((ch * chH) + py) * w) + pii) * 4;
418
- imgData[dataPos + 0] = rgbVal; //R
419
- imgData[dataPos + 1] = rgbVal; //G
420
- imgData[dataPos + 2] = rgbVal; //B
421
- imgData[dataPos + 3] = 255; //A (alpha: fully opaque)
422
- //console.debug("Rendered: py: "+py+", rgbval: "+rgbVal);
423
- // example 1x1, 2chs
424
- // ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,
425
- // ch0x1y0R,ch0x1y0G,ch0x1y0B,ch0x1y0A,
426
- // ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,
427
- // ch0x1y1R,ch0x1y1G,ch0x1y1B,ch0x1y1A,
428
- // ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,
429
- // ch1x1y0R,ch1x1y0G,ch1x1y0B,ch1x1y0A,
430
- // ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,
431
- // ch1x1y1R,ch1x1y1G,ch1x1y1B,ch1x1y1A
432
- }
433
- // if (allBlack) {
434
- // console.log("Black: ", pii, " ", ch);
435
- // }
436
- }
437
- }
438
- }
439
- }
440
- //console.debug("Render thread post message imgData: "+imgData.length)
441
- postMessage({ imgData: imgData, l: l, w: msg.data.w, h: msg.data.h, vw: vw, maxPsd: calcMaxPsd, terminate: msg.data.terminate }, [imgData.buffer]);
442
- };
443
- }
444
- startDraw(clear = true) {
445
- if (clear) {
446
- if (this.bounds) {
447
- this.sonagramCanvas.style.left = Math.round(this.bounds.position.left).toString() + 'px';
448
- let intW = Math.round(this.bounds.dimension.width);
449
- let intH = Math.round(this.bounds.dimension.height);
450
- this.sonagramCanvas.width = intW;
451
- this.sonagramCanvas.height = intH;
452
- let g = this.sonagramCanvas.getContext("2d");
453
- if (g) {
454
- //g.clearRect(0, 0,w, h);
455
- g.fillStyle = "white";
456
- g.fillRect(0, 0, intW, intH);
457
- }
458
- }
459
- }
460
- this.startRender();
461
- this.drawCursorLayer();
462
- }
463
- startRender() {
464
- //console.debug("Sonagram start render...");
465
- if (this.worker) {
466
- this.worker.terminate();
467
- this.worker = null;
468
- }
469
- if (this.raAsSubsc) {
470
- this.raAsSubsc.unsubscribe();
471
- }
472
- if (this._audioDataHolder) {
473
- this._audioDataHolder.addOnReadyListener(() => {
474
- if (this._audioDataHolder && this.bounds && this.bounds.dimension) {
475
- let w = Math.round(this.bounds.dimension.width);
476
- let h = Math.round(this.bounds.dimension.height);
477
- let vw = Math.round(this.virtualDimension.width);
478
- if (w > 0 && h > 0 && vw > 0) {
479
- this.worker = new Worker(this.workerURL);
480
- //this.wo = new Worker('./worker/sonagram.worker', { type: `module` });
481
- let chs = this._audioDataHolder.numberOfChannels;
482
- let vw = Math.round(this.virtualDimension.width);
483
- let frameLength = this._audioDataHolder.frameLen;
484
- let framesPerPixel = Math.ceil(frameLength / vw);
485
- let leftPos = Math.round(this.bounds.position.left);
486
- let renderPos = leftPos;
487
- let raAs = this._audioDataHolder.randomAccessAudioStream();
488
- let audioBuffer = null;
489
- let audioSource = this._audioDataHolder.audioSource;
490
- if (audioSource instanceof AudioBufferSource) {
491
- audioBuffer = audioSource.audioBuffer;
492
- }
493
- //let arrayAudioBuffer=this._audioDataHolder.arrayBuffer;
494
- let arrAbBuf;
495
- let ada = new Array(chs);
496
- // for(let ch=0;ch<chs;ch++) {
497
- // ada[ch] = new Float32Array(this.dftSize);
498
- // }
499
- let imgData;
500
- let maxPsd = -Infinity;
501
- let norender = true;
502
- if (this.worker) {
503
- this.worker.onmessage = (me) => {
504
- if (true === me.data.terminate) {
505
- let drawImgData;
506
- if (imgData) {
507
- drawImgData = imgData;
508
- }
509
- else {
510
- drawImgData = me.data.imgData;
511
- }
512
- this.drawRendered(w, h, drawImgData);
513
- if (this.worker) {
514
- this.worker.terminate();
515
- }
516
- this.worker = null;
517
- raAs.close();
518
- }
519
- else {
520
- // set rendered vertical values of one pixel of timescale
521
- //let dataPos = renderPos * h * 4;
522
- if (norender) {
523
- if (me.data.maxPsd > maxPsd) {
524
- maxPsd = me.data.maxPsd;
525
- //console.debug("new maxPsd: "+maxPsd);
526
- }
527
- }
528
- else {
529
- let chH = Math.round(h / chs);
530
- let idp = me.data.l - leftPos;
531
- for (let ch = 0; ch < chs; ch++) {
532
- for (let y = 0; y < chH; y++) {
533
- let py = chH - y;
534
- let dataPos = ((((ch * chH) + py) * w) + idp) * 4;
535
- let mePos = ((ch * chH) + py) * 4;
536
- //let lastPos = dataPos + me.data.imgData.length;
537
- //if (lastPos < imgData.length) {
538
- // set one pixel
539
- imgData[dataPos] = me.data.imgData[mePos];
540
- imgData[dataPos + 1] = me.data.imgData[mePos + 1];
541
- imgData[dataPos + 2] = me.data.imgData[mePos + 2];
542
- imgData[dataPos + 3] = me.data.imgData[mePos + 3];
543
- //} else {
544
- //console.error("Out of range: " + dataPos + "+" + me.data.imgData.length + ">=" + imgData.length);
545
- // }
546
- }
547
- }
548
- }
549
- if (this._audioDataHolder && arrAbBuf && this.worker) {
550
- // proceed with next pixel
551
- renderPos++;
552
- //console.debug("Render pos: "+renderPos);
553
- let terminate = false;
554
- let windowEnd = renderPos >= leftPos + w;
555
- if (windowEnd) {
556
- if (norender) {
557
- // phase two: rendering
558
- norender = false;
559
- // start from beginning
560
- renderPos = leftPos;
561
- //console.debug("now rendering: maxPsd: "+maxPsd);
562
- }
563
- else {
564
- // terminate render phase
565
- terminate = true;
566
- }
567
- }
568
- let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;
569
- if (leftFramePos < 0) {
570
- leftFramePos = 0;
571
- }
572
- //let ada = new Array<ArrayBuffer>(chs);
573
- //console.debug("Render pos: "+renderPos+" leftFramePos: "+leftFramePos);
574
- if (!terminate) {
575
- if (this._audioDataHolder) {
576
- //let read = this._audioDataHolder.frames(leftFramePos, this.dftSize, arrAbBuf);
577
- this.raAsSubsc = raAs.framesObs(leftFramePos, this.dftSize, arrAbBuf).subscribe({
578
- next: (read) => {
579
- if (arrAbBuf && this.worker) {
580
- if (read < this.dftSize) {
581
- // zero padding
582
- for (let ch = 0; ch < chs; ch++) {
583
- for (let zp = read; zp < this.dftSize; zp++) {
584
- arrAbBuf[ch][zp] = 0;
585
- }
586
- }
587
- }
588
- for (let ch = 0; ch < chs; ch++) {
589
- // Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker
590
- //ada[ch] = arrAbBuf[ch].buffer.slice(0);
591
- const arrAbBufCh = arrAbBuf[ch];
592
- const adLen = arrAbBufCh.buffer.byteLength;
593
- if (!ada[ch] || ada[ch].byteLength !== adLen) {
594
- ada[ch] = new ArrayBuffer(adLen);
595
- }
596
- let fAdaCh = new Float32Array(ada[ch]);
597
- fAdaCh.set(arrAbBuf[ch]);
598
- }
599
- this.worker.postMessage({
600
- audioData: ada,
601
- audioDataOffset: leftFramePos,
602
- l: renderPos,
603
- w: me.data.w,
604
- h: h,
605
- vw: vw,
606
- chs: chs,
607
- frameLength: frameLength,
608
- dftSize: this.dftSize,
609
- maxPsd: maxPsd,
610
- norender: norender,
611
- terminate: terminate
612
- }, ada);
613
- }
614
- },
615
- error: (err) => {
616
- console.error("Sonagram: Error reading audio data: " + err);
617
- }
618
- });
619
- }
620
- }
621
- else {
622
- for (let ch = 0; ch < chs; ch++) {
623
- ada[ch] = new ArrayBuffer(0);
624
- }
625
- this.worker.postMessage({
626
- audioData: ada,
627
- audioDataOffset: leftFramePos,
628
- l: renderPos,
629
- w: me.data.w,
630
- h: h,
631
- vw: vw,
632
- chs: chs,
633
- frameLength: frameLength,
634
- dftSize: this.dftSize,
635
- maxPsd: maxPsd,
636
- norender: norender,
637
- terminate: terminate
638
- }, ada);
639
- }
640
- }
641
- }
642
- };
643
- }
644
- if (audioBuffer && audioBuffer.length * audioBuffer.numberOfChannels < AudioCanvasLayerComponent.ENABLE_STREAMING_NUMBER_OF_SAMPLES_THRESHOLD) {
645
- let ada = new Array(chs);
646
- for (let ch = 0; ch < chs; ch++) {
647
- // Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker
648
- ada[ch] = audioBuffer.getChannelData(ch).buffer.slice(0);
649
- }
650
- //let start = Date.now();
651
- if (this.markerCanvas) {
652
- let g = this.markerCanvas.getContext("2d");
653
- if (g) {
654
- g.fillText("Rendering...", 10, 20);
655
- }
656
- }
657
- this.worker.postMessage({
658
- audioData: ada,
659
- l: leftPos,
660
- w: w,
661
- h: h,
662
- vw: Math.round(this.virtualDimension.width),
663
- chs: chs,
664
- frameLength: frameLength,
665
- dftSize: this.dftSize,
666
- terminate: true
667
- }, ada);
668
- }
669
- else {
670
- if (w > 0) {
671
- if (framesPerPixel > 0) {
672
- let arrSize = w * h * 4;
673
- if (arrSize < 0) {
674
- arrSize = 0;
675
- }
676
- imgData = new Uint8ClampedArray(arrSize);
677
- let rw = 1;
678
- arrAbBuf = new Array(chs);
679
- for (let ch = 0; ch < chs; ch++) {
680
- arrAbBuf[ch] = new Float32Array(this.dftSize);
681
- }
682
- let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;
683
- let framesToRead = this.dftSize;
684
- if (leftFramePos < 0) {
685
- leftFramePos = 0;
686
- }
687
- this.drawStateText('Loading/Rendering...');
688
- this.raAsSubsc = raAs.framesObs(leftFramePos, framesToRead, arrAbBuf).subscribe({
689
- next: (read) => {
690
- if (arrAbBuf && this.worker) {
691
- if (read < this.dftSize) {
692
- // zero padding
693
- for (let ch = 0; ch < chs; ch++) {
694
- for (let zp = read; zp < this.dftSize; zp++) {
695
- arrAbBuf[ch][zp] = 0;
696
- }
697
- }
698
- }
699
- for (let ch = 0; ch < chs; ch++) {
700
- const arrAbBufCh = arrAbBuf[ch];
701
- const adLen = arrAbBufCh.buffer.byteLength;
702
- if (!ada[ch] || ada[ch].byteLength !== adLen) {
703
- ada[ch] = new ArrayBuffer(adLen);
704
- }
705
- let fAdaCh = new Float32Array(ada[ch]);
706
- fAdaCh.set(arrAbBuf[ch]);
707
- }
708
- this.worker.postMessage({
709
- l: renderPos,
710
- w: rw,
711
- h: h,
712
- vw: vw,
713
- chs: chs,
714
- frameLength: frameLength,
715
- audioData: ada,
716
- audioDataOffset: leftFramePos,
717
- dftSize: this.dftSize,
718
- norender: norender,
719
- terminate: false
720
- }, ada);
721
- }
722
- },
723
- error: (err) => {
724
- console.error("Sonagram: Error reading audio data: " + err);
725
- }
726
- });
727
- }
728
- }
729
- }
730
- }
731
- }
732
- });
733
- }
734
- else {
735
- if (this.bounds && this.bounds.dimension) {
736
- let w = Math.round(this.bounds.dimension.width);
737
- let h = Math.round(this.bounds.dimension.height);
738
- let g = this.sonagramCanvas.getContext("2d");
739
- if (g) {
740
- g.clearRect(0, 0, w, h);
741
- }
742
- }
743
- }
744
- }
745
- drawRendered(w, h, imgData) {
746
- if (this.sonagramCanvas) {
747
- this.sonagramCanvas.width = w;
748
- this.sonagramCanvas.height = h;
749
- let g = this.sonagramCanvas.getContext("2d");
750
- if (g) {
751
- if (w > 0 && h > 0) {
752
- let gImgData = g.createImageData(w, h);
753
- gImgData.data.set(imgData);
754
- g.putImageData(gImgData, 0, 0);
755
- }
756
- }
757
- }
758
- this.drawBg();
759
- this.drawPlayPosition();
760
- }
761
- // // synchronous draw (not used anymore)
762
- // redraw() {
763
- //
764
- // let g = this.sonagramCanvas.getContext("2d");
765
- //
766
- // let w = this.sonagramCanvas.width;
767
- // let h = this.sonagramCanvas.height;
768
- // if (g) {
769
- // g.clearRect(0, 0, w, h);
770
- // g.fillStyle = "white";
771
- // g.fillRect(0, 0, w, h);
772
- // if (this._audioDataHolder) {
773
- // let spectSize = Math.floor(this.dftSize / 2)
774
- // let chs = this._audioDataHolder.numberOfChannels;
775
- // let chH = h / chs;
776
- //
777
- // let frameLength = this._audioDataHolder.frameLen;
778
- //
779
- // let framesPerPixel = frameLength / w;
780
- // let y = 0;
781
- // let audioBuffer:AudioBuffer|null=null;
782
- // let audioSource=this._audioDataHolder.audioSource;
783
- // if(audioSource instanceof AudioBufferSource){
784
- // audioBuffer=audioSource.audioBuffer;
785
- // }
786
- //
787
- // if(audioBuffer) {
788
- // let b = new Float32Array(this.dftSize)
789
- //
790
- // let sona = new Array<Array<Float32Array>>(chs);
791
- // let max = 0;
792
- // let maxPsd = -Infinity;
793
- // for (let ch = 0; ch < chs; ch++) {
794
- // let x = 0;
795
- // sona[ch] = new Array<Float32Array>(w);
796
- //
797
- // let chData = audioBuffer.getChannelData(ch);
798
- // // TODO center buffer
799
- //
800
- // let framePos = 0;
801
- // for (let pii = 0; pii < w; pii++) {
802
- // framePos = Math.round(pii * framesPerPixel);
803
- // // calculate DFT at pixel position
804
- // for (let i = 0; i < this.dftSize; i++) {
805
- // let chDat = chData[framePos + i];
806
- // b[i] = chDat;
807
- // }
808
- // let spectr = this.dft.processRealMagnitude(b);
809
- // sona[ch][pii] = spectr;
810
- // // @ts-ignore
811
- // let pMax = Math.max.apply(null, spectr);
812
- // if (pMax > max) {
813
- // max = pMax;
814
- // }
815
- //
816
- // for (let s = 0; s < spectSize; s++) {
817
- // let psd = (2 * Math.pow(spectr[s], 2)) / spectSize;
818
- // if (psd > maxPsd) {
819
- // maxPsd = psd;
820
- // }
821
- // }
822
- // }
823
- // }
824
- // //console.log("max: ", max);
825
- // maxPsd = (2 * Math.pow(max, 2)) / spectSize;
826
- // for (let ch = 0; ch < chs; ch++) {
827
- //
828
- // let framePos = 0;
829
- // for (let pii = 0; pii < w; pii++) {
830
- // framePos = pii * framesPerPixel;
831
- //
832
- // for (let y = 0; y < h; y++) {
833
- // let freqIdx = Math.round(y * spectSize / h);
834
- //
835
- // // calculate the one sided power spectral density PSD (f, t) in Pa2/Hz
836
- // // PSD(f) proportional to 2|X(f)|2 / (t2 - t1)
837
- // let val = sona[ch][pii][freqIdx];
838
- // let psd = (2 * Math.pow(val, 2)) / spectSize;
839
- //
840
- // // Calculate logarithmic
841
- // let psdLog = DSPUtils.toLevelInDB(psd / maxPsd);
842
- // let dynRangeInDb = 70;
843
- // let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;
844
- //
845
- // if (scaledVal > 1)
846
- // scaledVal = 1;
847
- // if (scaledVal < 0) {
848
- // scaledVal = 0;
849
- // }
850
- // let rgbVal = (255 * scaledVal);
851
- // if (rgbVal < 0) {
852
- // // System.out.println("Neg RGB val: "+rgbVal);
853
- // rgbVal = 0;
854
- // }
855
- // if (rgbVal > 255) {
856
- // rgbVal = 255;
857
- // }
858
- // rgbVal = 255 - rgbVal;
859
- // let colorStr = CSSUtils.toColorString(rgbVal, rgbVal, rgbVal);
860
- // g.fillStyle = colorStr;
861
- // g.fillRect(pii, chH - y, 1, 1);
862
- // }
863
- // }
864
- // }
865
- // this.drawPlayPosition();
866
- // }else{
867
- // throw Error("Redraw only supported with audio buffer.")
868
- // }
869
- // }
870
- // }
871
- // }
872
- //
873
- setData(audioData) {
874
- this._audioDataHolder = audioData;
875
- this.playFramePosition = 0;
876
- }
877
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: Sonagram, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
878
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.12", type: Sonagram, selector: "audio-sonagram", viewQueries: [{ propertyName: "sonagramCanvasRef", first: true, predicate: ["sonagram"], descendants: true, static: true }, { propertyName: "markerCanvasRef", first: true, predicate: ["marker"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: `
879
- <canvas #sonagram height="10"></canvas>
880
- <canvas #bg height="10"></canvas>
881
- <canvas #cursor height="10" (mousedown)="selectionStart($event)" (mouseover)="updateCursorCanvas($event)" (mousemove)="updateCursorCanvas($event)"
882
- (mouseleave)="updateCursorCanvas($event, false)"></canvas>
883
- <canvas #marker height="10"></canvas>`, isInline: true, styles: [":host{min-height:0px}\n", "canvas{top:0;left:0;width:0;height:0;min-height:0px;position:absolute}\n"] }); }
884
- }
885
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: Sonagram, decorators: [{
886
- type: Component,
887
- args: [{ selector: 'audio-sonagram', template: `
888
- <canvas #sonagram height="10"></canvas>
889
- <canvas #bg height="10"></canvas>
890
- <canvas #cursor height="10" (mousedown)="selectionStart($event)" (mouseover)="updateCursorCanvas($event)" (mousemove)="updateCursorCanvas($event)"
891
- (mouseleave)="updateCursorCanvas($event, false)"></canvas>
892
- <canvas #marker height="10"></canvas>`, styles: [":host{min-height:0px}\n", "canvas{top:0;left:0;width:0;height:0;min-height:0px;position:absolute}\n"] }]
893
- }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { sonagramCanvasRef: [{
894
- type: ViewChild,
895
- args: ['sonagram', { static: true }]
896
- }], markerCanvasRef: [{
897
- type: ViewChild,
898
- args: ['marker', { static: true }]
899
- }] } });
900
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sonagram.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/audio/ui/sonagram.ts"],"names":[],"mappings":"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","sourcesContent":["import {DFTFloat32} from '../../math/dft';\r\nimport {Marker, Point} from './common';\r\nimport {Component, ElementRef, ViewChild} from \"@angular/core\";\r\nimport {AudioCanvasLayerComponent} from \"./audio_canvas_layer_comp\";\r\nimport {WorkerHelper} from \"../../utils/utils\";\r\nimport {AudioBufferSource, AudioDataHolder} from \"../audio_data_holder\";\r\nimport {Subscription} from \"rxjs\";\r\n\r\ndeclare function postMessage(message: any, transfer: Array<any>): void;\r\n\r\nconst DEFAULT_DFT_SIZE = 1024;\r\n\r\n@Component({\r\n\r\n    selector: 'audio-sonagram',\r\n    template: `\r\n        <canvas #sonagram height=\"10\"></canvas>\r\n        <canvas #bg height=\"10\"></canvas>\r\n        <canvas #cursor height=\"10\" (mousedown)=\"selectionStart($event)\" (mouseover)=\"updateCursorCanvas($event)\" (mousemove)=\"updateCursorCanvas($event)\"\r\n                (mouseleave)=\"updateCursorCanvas($event, false)\"></canvas>\r\n        <canvas #marker height=\"10\"></canvas>`,\r\n\r\n    styles: [`:host{\r\n      min-height: 0px;\r\n    }`,`canvas {\r\n        top: 0;\r\n        left: 0;\r\n        width: 0;\r\n        height: 0;\r\n      min-height: 0px;\r\n        position: absolute;\r\n    }`]\r\n\r\n})\r\nexport class Sonagram extends AudioCanvasLayerComponent {\r\n\r\n    dft: DFTFloat32;\r\n    n: any;\r\n    ce!: HTMLDivElement;\r\n    sonagramCanvas!: HTMLCanvasElement;\r\n    //cursorCanvas: HTMLCanvasElement;\r\n    markerCanvas!: HTMLCanvasElement;\r\n    @ViewChild('sonagram', { static: true }) sonagramCanvasRef!: ElementRef;\r\n\r\n    @ViewChild('marker', { static: true }) markerCanvasRef!: ElementRef;\r\n    markers: Array<Marker>;\r\n    private _playFramePosition: number|null=null;\r\n\r\n    private worker: Worker | null;\r\n    private readonly workerURL: string;\r\n    private raAsSubsc:Subscription|null=null;\r\n\r\n    private dftSize = DEFAULT_DFT_SIZE;\r\n\r\n    constructor(private ref: ElementRef) {\r\n        super();\r\n        this.worker = null;\r\n        this._audioDataHolder = null;\r\n        this.markers = new Array<Marker>();\r\n        this.dft = new DFTFloat32(this.dftSize);\r\n\r\n        this.workerURL = WorkerHelper.buildWorkerBlobURL(this.workerFunction)\r\n       this._bgColor=null;\r\n       this._selectColor='rgba(255,255,0,0.1)'\r\n    }\r\n\r\n    ngAfterViewInit() {\r\n\r\n        this.ce = this.ref.nativeElement;\r\n        this.sonagramCanvas = this.sonagramCanvasRef.nativeElement;\r\n        this.sonagramCanvas.style.zIndex = '1';\r\n      this.bgCanvas = this.bgCanvasRef.nativeElement;\r\n      this.bgCanvas.style.zIndex = '2';\r\n        this.cursorCanvas = this.cursorCanvasRef.nativeElement;\r\n        this.cursorCanvas.style.zIndex = '4';\r\n        this.markerCanvas = this.markerCanvasRef.nativeElement;\r\n        this.markerCanvas.style.zIndex = '3';\r\n\r\n        this.canvasLayers[0] = this.sonagramCanvas;\r\n      this.canvasLayers[1] = this.bgCanvas;\r\n      this.canvasLayers[2] = this.cursorCanvas;\r\n        this.canvasLayers[3] = this.markerCanvas;\r\n\r\n    }\r\n\r\n    get playFramePosition(): number|null {\r\n        return this._playFramePosition;\r\n    }\r\n\r\n    set playFramePosition(playFramePosition: number|null) {\r\n        this._playFramePosition = playFramePosition;\r\n        // this.redraw();\r\n        this.drawPlayPosition();\r\n    }\r\n\r\n    private canvasMousePos(c: HTMLCanvasElement, e: MouseEvent): Point {\r\n        const cr = c.getBoundingClientRect();\r\n\r\n        const x = e.x - cr.left;\r\n        const y = e.y - cr.top;\r\n        return new Point(x,y);\r\n    }\r\n\r\n  drawStateText(stateText:string) {\r\n        if(stateText) {\r\n            const g = this.sonagramCanvas.getContext(\"2d\");\r\n            const w = this.sonagramCanvas.width;\r\n            const h = this.sonagramCanvas.height;\r\n            if (g && w && h) {\r\n                g.strokeStyle = 'black';\r\n                g.fillStyle = 'black';\r\n                g.font = '20px sans-serif';\r\n                g.fillText(stateText, 10, 25);\r\n            }\r\n        }\r\n  }\r\n\r\n  drawCursorPosition(e: MouseEvent, show: boolean) {\r\n\r\n    if (this.cursorCanvas) {\r\n      const w = this.cursorCanvas.width;\r\n      const h = this.cursorCanvas.height;\r\n      const g = this.cursorCanvas.getContext('2d');\r\n      if (g) {\r\n        g.clearRect(0, 0, w, h);\r\n        if (show) {\r\n          //const pp = this.canvasMousePos(this.cursorCanvas, e);\r\n          let xViewPortPixelpos = e.offsetX;\r\n\r\n          g.fillStyle = 'yellow';\r\n          g.strokeStyle = 'yellow';\r\n          g.beginPath();\r\n          g.moveTo(xViewPortPixelpos, 0);\r\n          g.lineTo(xViewPortPixelpos, h);\r\n          g.closePath();\r\n\r\n          g.stroke();\r\n\r\n          if (this._audioDataHolder) {\r\n\r\n            let framePosRound = this.viewPortXPixelToFramePosition(xViewPortPixelpos);\r\n            if(framePosRound!=null) {\r\n              g.font = '14px sans-serif';\r\n              g.fillStyle = 'yellow';\r\n              g.fillText(framePosRound.toString(), xViewPortPixelpos + 2, 50);\r\n            }\r\n          }\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n    drawPlayPosition() {\r\n        if (this.markerCanvas) {\r\n            const w = this.markerCanvas.width;\r\n            const h = this.markerCanvas.height;\r\n            const g = this.markerCanvas.getContext(\"2d\");\r\n            if (g) {\r\n                g.clearRect(0, 0, w, h);\r\n                if(this._playFramePosition!=null) {\r\n                    let pixelPos = this.frameToViewPortXPixelPosition(this._playFramePosition);\r\n                    if (pixelPos) {\r\n                        g.fillStyle = 'red';\r\n                        g.strokeStyle = 'red';\r\n                        g.beginPath();\r\n                        g.moveTo(pixelPos, 0);\r\n                        g.lineTo(pixelPos, h);\r\n                        g.closePath();\r\n                        g.stroke();\r\n                    }\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    /*\r\n     *  Method used as worker code.\r\n     */\r\n    workerFunction() {\r\n\r\n        // Redefine some DSP classes for worker function\r\n        // See e.g. audio.math.Complex\r\n        class Complex {\r\n\r\n            real: number;\r\n            img: number;\r\n\r\n            public static fromPolarForm(magnitude: number, argument: number): Complex {\r\n                const r = Math.cos(argument) * magnitude;\r\n                const i = Math.sin(argument) * magnitude;\r\n                return new Complex(r, i);\r\n            }\r\n\r\n            constructor(real: number, img: number) {\r\n                this.real = real;\r\n                this.img = img;\r\n            }\r\n\r\n            public magnitude(): number {\r\n                return Math.sqrt((this.real * this.real) + (this.img * this.img));\r\n            }\r\n\r\n            public argument(): number {\r\n                return Math.atan2(this.img, this.real);\r\n            }\r\n\r\n            public add(addC: Complex): Complex {\r\n                return new Complex(this.real + addC.real, this.img + addC.img);\r\n            }\r\n\r\n            public sub(subC: Complex): Complex {\r\n                return new Complex(this.real - subC.real, this.img - subC.img);\r\n            }\r\n\r\n            public mult(multC: Complex): Complex {\r\n                const multR = (this.real * multC.real) - (this.img * multC.img);\r\n                const multI = (this.real * multC.img) + (multC.real * this.img);\r\n                return new Complex(multR, multI);\r\n            }\r\n\r\n            public multReal(multF: number): Complex {\r\n                return new Complex(this.real * multF, this.img * multF);\r\n            }\r\n\r\n            public div(divisor: Complex): Complex {\r\n                const divReal = divisor.real;\r\n                const divImg = divisor.img;\r\n                const div = (divReal * divReal) + (divImg * divImg);\r\n                const divisionReal = ((this.real * divReal) + (this.img * divImg)) / div;\r\n                const divisionImg = ((divReal * this.img) - (this.real * divImg)) / div;\r\n\r\n                return new Complex(divisionReal, divisionImg);\r\n            }\r\n\r\n            public divReal(divisor: number): Complex {\r\n                const div = divisor * divisor;\r\n                const divsionReal = (this.real * divisor) / div;\r\n                const divsionImg = (divisor * this.img) / div;\r\n\r\n                return new Complex(divsionReal, divsionImg);\r\n            }\r\n\r\n            public conjugate(): Complex {\r\n                return new Complex(this.real, -this.img);\r\n            }\r\n\r\n            public equals(c: Complex): boolean {\r\n                if (c === null) {\r\n                    return false;\r\n                }\r\n                return (this.real === c.real && this.img === c.img);\r\n            }\r\n\r\n            public toString(): string {\r\n                return 'Real: ' + this.real + ', Img: ' + this.img;\r\n            }\r\n        }\r\n\r\n        class DFTFloat32 {\r\n\r\n            private readonly n: number;\r\n            private readonly m: number;\r\n\r\n            private readonly cosLookup: Float32Array;\r\n            private readonly sinLookup: Float32Array;\r\n\r\n            constructor(n: number) {\r\n                this.n = n;\r\n                this.m = Math.log(n) / Math.log(2);\r\n\r\n                // if(n != (1 << m))throw new RuntimeException(\"length N must be power of 2\");\r\n\r\n                // lookup tables\r\n                this.cosLookup = new Float32Array(n / 2);\r\n                this.sinLookup = new Float32Array(n / 2);\r\n\r\n                for (let i = 0; i < n / 2; i++) {\r\n                    const arc = (-2 * Math.PI * i) / n;\r\n                    this.cosLookup[i] = Math.cos(arc);\r\n                    this.sinLookup[i] = Math.sin(arc);\r\n                }\r\n            }\r\n\r\n            public processReal(srcBuf: Float32Array): Array<Complex> {\r\n                const x = srcBuf.slice();\r\n                const y = new Float32Array(srcBuf.length);\r\n                for (let yi = 0; yi < y.length; yi++) {\r\n                    y[yi] = 0.0;\r\n                }\r\n                this.fftCooleyTukey(x, y);\r\n                const rc = new Array<Complex>(x.length);\r\n                for (let i = 0; i < x.length; i++) {\r\n                    rc[i] = new Complex(x[i], y[i]);\r\n                }\r\n                return rc;\r\n            }\r\n\r\n            public processRealMagnitude(srcBuf: Float32Array): Float32Array {\r\n                const x = srcBuf.slice();\r\n                const y = new Float32Array(srcBuf.length);\r\n                for (let yi = 0; yi < y.length; yi++) {\r\n                    y[yi] = 0.0;\r\n                }\r\n                this.fftCooleyTukey(x, y);\r\n                const rc = new Float32Array(x.length);\r\n                for (let i = 0; i < x.length; i++) {\r\n                    const rcc = new Complex(x[i], y[i]);\r\n                    rc[i] = rcc.magnitude();\r\n                }\r\n                return rc;\r\n            }\r\n\r\n            public fftCooleyTukey(real: Float32Array, img: Float32Array): void {\r\n                let i: number;\r\n                let j = 0;\r\n                let k: number;\r\n                let n1: number;\r\n                let n2: number = this.n / 2;\r\n                let a: number;\r\n                let c: number;\r\n                let s: number;\r\n                let t1: number;\r\n                let t2: number;\r\n\r\n                for (i = 1; i < this.n - 1; i++) {\r\n                    n1 = n2;\r\n                    while (j >= n1) {\r\n                        j = j - n1;\r\n                        n1 = n1 / 2;\r\n                    }\r\n                    j = j + n1;\r\n\r\n                    if (i < j) {\r\n                        t1 = real[i];\r\n                        real[i] = real[j];\r\n                        real[j] = t1;\r\n                        t1 = img[i];\r\n                        img[i] = img[j];\r\n                        img[j] = t1;\r\n                    }\r\n                }\r\n\r\n                n1 = 0;\r\n                n2 = 1;\r\n                for (i = 0; i < this.m; i++) {\r\n                    n1 = n2;\r\n                    n2 = n2 + n2;\r\n                    a = 0;\r\n                    for (j = 0; j < n1; j++) {\r\n                        c = this.cosLookup[a];\r\n                        s = this.sinLookup[a];\r\n                        a += (1 << (this.m - i - 1));\r\n\r\n                        for (k = j; k < this.n; k = k + n2) {\r\n                            t1 = c * real[k + n1] - s * img[k + n1];\r\n                            t2 = s * real[k + n1] + c * img[k + n1];\r\n                            real[k + n1] = real[k] - t1;\r\n                            img[k + n1] = img[k] - t2;\r\n                            real[k] = real[k] + t1;\r\n                            img[k] = img[k] + t2;\r\n                        }\r\n                    }\r\n                }\r\n            }\r\n\r\n\r\n            public process(t: Array<Complex>): Array<Complex> {\r\n                const reals: Float32Array = new Float32Array(this.n);\r\n                const imgs: Float32Array = new Float32Array(this.n);\r\n                const trans: Array<Complex> = new Array<Complex>(this.n);\r\n                for (let i = 0; i < this.n; i++) {\r\n                    reals[i] = t[i].real;\r\n                    imgs[i] = t[i].img;\r\n                }\r\n                this.fftCooleyTukey(reals, imgs);\r\n                for (let i = 0; i < this.n; i++) {\r\n                    trans[i] = new Complex(reals[i], imgs[i]);\r\n                }\r\n                return trans;\r\n\r\n\r\n            }\r\n\r\n        }\r\n\r\n        interface WindowFunction {\r\n            getScale(i: number): number;\r\n        }\r\n\r\n        class GaussianWindow implements WindowFunction {\r\n\r\n            public static DEFAULT_SIGMA = 0.3;\r\n            // Gaussian window function,\r\n            // http://reference.wolfram.com/language/ref/GaussianWindow.html\r\n            // val=exp(-50*x*x/9) => sigma=0.3\r\n\r\n            private readonly buf: Float32Array;\r\n\r\n            constructor(size: number, sigma: number = GaussianWindow.DEFAULT_SIGMA) {\r\n                this.buf = new Float32Array(size);\r\n                const center = (size - 1) / 2;\r\n                for (let i = 0; i < size; i++) {\r\n                    const quot = (i - center) / (sigma * center);\r\n                    const exp = -0.5 * quot * quot;\r\n                    this.buf[i] = Math.exp(exp);\r\n                }\r\n            }\r\n\r\n            getScale(i: number): number {\r\n                return this.buf[i];\r\n            }\r\n\r\n        }\r\n\r\n        self.onmessage = function (msg:MessageEvent) {\r\n            //console.debug(\"Sonagram render thread\");\r\n            let l = msg.data.l;\r\n            let w = msg.data.w;\r\n            let h = msg.data.h;\r\n            let vw = msg.data.vw;\r\n            let chs = msg.data.chs;\r\n            let audioDataOffset=0;\r\n            let adOffset=msg.data.audioDataOffset;\r\n            if(adOffset){\r\n              audioDataOffset=adOffset;\r\n            }\r\n           let maxPsd = null;\r\n            if(msg.data.maxPsd!==undefined){\r\n              maxPsd=msg.data.maxPsd;\r\n            }\r\n\r\n            let audioData = new Array(chs);\r\n            for (let ch = 0; ch < chs; ch++) {\r\n                audioData[ch] = new Float32Array(msg.data['audioData'][ch]);\r\n            }\r\n\r\n            let frameLength = msg.data.frameLength;\r\n            let dftSize = msg.data.dftSize;\r\n\r\n            let dftBands = dftSize / 2;\r\n            let dft = new DFTFloat32(dftSize);\r\n            let wf = new GaussianWindow(dftSize);\r\n\r\n            let arrSize=w*h*4;\r\n            if(arrSize<0){\r\n                arrSize=0\r\n            }\r\n            let imgData = new Uint8ClampedArray(arrSize);\r\n            //console.log(\"Render method:\");\r\n          //console.debug(\"Created imgData arrSize: \"+arrSize+\" \", w, \"x\", h);\r\n          let calcMaxPsd=-Infinity;\r\n            if (arrSize>0) {\r\n\r\n                let chH = Math.round(h / chs);\r\n                let framesPerPixel = frameLength / vw;\r\n                //console.debug(\"Render: \", w, \"x\", h);\r\n\r\n                let b = new Float32Array(dftSize);\r\n                let sona = new Array(chs);\r\n\r\n                for (let ch = 0; ch < chs; ch++) {\r\n\r\n                    let chDataLen=audioData[ch].length;\r\n                    // initialize DFT array buffer\r\n                    sona[ch] = new Array(w);\r\n                    let framePos = 0;\r\n                    for (let pii = 0; pii < w; pii++) {\r\n                        let virtPii = l + pii;\r\n                        // Position of sample data frame is pixel position mapped to audio frame position.\r\n                       // Then \"center\" the frame by shifting left by half the DFT size (=dftBands)\r\n                        framePos = Math.round((virtPii * framesPerPixel)-dftBands);\r\n                        // fill DFT buffer with windowed sample values\r\n                        for (let i = 0; i < dftSize; i++) {\r\n                            let samplePos=framePos + i;\r\n                            // initialize for negative sample positions and out of bounds positions\r\n                            let chDat=0;\r\n\r\n                            // Set audio sample if available\r\n                            let adp=samplePos-audioDataOffset;\r\n                            if(adp>=0 && adp < chDataLen) {\r\n                              chDat = audioData[ch][adp];\r\n                              //console.debug(\"Audio data: \"+chDat);\r\n                            }else{\r\n                              //console.debug(\"Sample buf pos oob: adp: \"+adp+\", chDataLen: \"+chDataLen+\", samplePos: \"+samplePos+\", i: \"+i);\r\n                            }\r\n\r\n                            // apply Window\r\n                            b[i] = chDat * wf.getScale(i);\r\n                        }\r\n                        // Calc DFT magnitudes\r\n                        let spectr = dft.processRealMagnitude(b);\r\n\r\n                        // Get maximum value of spectral energy\r\n                        for (let s = 0; s < dftBands; s++) {\r\n                            let psd = (2 * Math.pow(spectr[s], 2)) / dftBands;\r\n                            if (psd > calcMaxPsd) {\r\n                                calcMaxPsd = psd;\r\n                            }\r\n                        }\r\n                        // Set render model data for this pixel\r\n                        sona[ch][pii] = spectr;\r\n                    }\r\n                }\r\n\r\n                if(!msg.data.norender) {\r\n                  if(!maxPsd){\r\n                    maxPsd=calcMaxPsd;\r\n                  }\r\n                for (let ch = 0; ch < chs; ch++) {\r\n\r\n                  for (let pii = 0; pii < w; pii++) {\r\n                    let allBlack = true;\r\n                    for (let y = 0; y < chH; y++) {\r\n                      let freqIdx = Math.round(y * dftBands / chH);\r\n                      // calculate the one-sided power spectral density PSD (f, t) in Pa2/Hz\r\n                      // PSD(f) proportional to 2|X(f)|2 / (t2 - t1)\r\n                      let val = sona[ch][pii][freqIdx];\r\n                      let psd = (2 * Math.pow(val, 2)) / dftBands;\r\n                      // Calculate logarithmic value\r\n                      //let psdLog = ips.dsp.DSPUtils.toLevelInDB(psd / maxPsd);\r\n                      let linearLevel = psd / maxPsd;\r\n                      let psdLog = 10 * Math.log(linearLevel) / Math.log(10);\r\n                      // Fixed dynamic Range value of 70dB for now\r\n                      let dynRangeInDb = 70;\r\n                      let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;\r\n\r\n                      // are the following checks necessary for clamped array ?\r\n                      if (scaledVal > 1.0)\r\n                        scaledVal = 1;\r\n                      if (scaledVal < 0.0) {\r\n                        scaledVal = 0;\r\n                      }\r\n                      let rgbVal = Math.round(255 * scaledVal);\r\n                      if (rgbVal < 0) {\r\n                        //\t\t\t\t\t\t\tSystem.out.println(\"Neg RGB val: \"+rgbVal);\r\n                        rgbVal = 0;\r\n                      }\r\n                      if (rgbVal > 255) {\r\n                        rgbVal = 255;\r\n                      }\r\n                      rgbVal = 255 - rgbVal;\r\n                      if (rgbVal > 0) {\r\n                        allBlack = false;\r\n                      }\r\n                      let py = chH - y;\r\n                      let dataPos = ((((ch * chH) + py) * w) + pii) * 4;\r\n                      imgData[dataPos + 0] = rgbVal; //R\r\n                      imgData[dataPos + 1] = rgbVal; //G\r\n                      imgData[dataPos + 2] = rgbVal; //B\r\n                      imgData[dataPos + 3] = 255; //A (alpha: fully opaque)\r\n                      //console.debug(\"Rendered: py: \"+py+\", rgbval: \"+rgbVal);\r\n                      // example 1x1, 2chs\r\n                      //  ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,\r\n                      //  ch0x1y0R,ch0x1y0G,ch0x1y0B,ch0x1y0A,\r\n                      //  ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,\r\n                      //  ch0x1y1R,ch0x1y1G,ch0x1y1B,ch0x1y1A,\r\n\r\n                      //  ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,\r\n                      //  ch1x1y0R,ch1x1y0G,ch1x1y0B,ch1x1y0A,\r\n                      //  ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,\r\n                      //  ch1x1y1R,ch1x1y1G,ch1x1y1B,ch1x1y1A\r\n                    }\r\n                    // if (allBlack) {\r\n                    //   console.log(\"Black: \", pii, \" \", ch);\r\n                    // }\r\n                  }\r\n                }\r\n                }\r\n            }\r\n            //console.debug(\"Render thread post message imgData: \"+imgData.length)\r\n            postMessage({imgData: imgData, l: l, w: msg.data.w, h: msg.data.h, vw: vw,maxPsd:calcMaxPsd,terminate:msg.data.terminate}, [imgData.buffer]);\r\n        }\r\n    }\r\n\r\n    startDraw(clear = true) {\r\n        if (clear) {\r\n            if(this.bounds) {\r\n                this.sonagramCanvas.style.left = Math.round(this.bounds.position.left).toString() + 'px';\r\n                let intW = Math.round(this.bounds.dimension.width)\r\n                let intH = Math.round(this.bounds.dimension.height)\r\n                this.sonagramCanvas.width = intW;\r\n                this.sonagramCanvas.height = intH;\r\n\r\n                let g = this.sonagramCanvas.getContext(\"2d\");\r\n                if (g) {\r\n                    //g.clearRect(0, 0,w, h);\r\n                    g.fillStyle = \"white\";\r\n                    g.fillRect(0, 0, intW, intH);\r\n                }\r\n            }\r\n        }\r\n        this.startRender();\r\n        this.drawCursorLayer()\r\n    }\r\n\r\n    private startRender() {\r\n      //console.debug(\"Sonagram start render...\");\r\n        if (this.worker) {\r\n            this.worker.terminate();\r\n            this.worker = null;\r\n        }\r\n      if(this.raAsSubsc){\r\n        this.raAsSubsc.unsubscribe();\r\n      }\r\n      if (this._audioDataHolder) {\r\n        this._audioDataHolder.addOnReadyListener(()=> {\r\n          if (this._audioDataHolder && this.bounds && this.bounds.dimension) {\r\n            let w = Math.round(this.bounds.dimension.width);\r\n            let h = Math.round(this.bounds.dimension.height);\r\n            let vw = Math.round(this.virtualDimension.width);\r\n\r\n            if (w > 0 && h > 0 && vw > 0) {\r\n\r\n              this.worker = new Worker(this.workerURL);\r\n              //this.wo = new Worker('./worker/sonagram.worker', { type: `module` });\r\n\r\n              let chs = this._audioDataHolder.numberOfChannels;\r\n              let vw = Math.round(this.virtualDimension.width);\r\n\r\n              let frameLength = this._audioDataHolder.frameLen;\r\n              let framesPerPixel = Math.ceil(frameLength / vw);\r\n              let leftPos = Math.round(this.bounds.position.left);\r\n              let renderPos = leftPos;\r\n              let raAs = this._audioDataHolder.randomAccessAudioStream();\r\n\r\n              let audioBuffer: AudioBuffer | null = null;\r\n              let audioSource = this._audioDataHolder.audioSource;\r\n              if (audioSource instanceof AudioBufferSource) {\r\n                audioBuffer = audioSource.audioBuffer;\r\n              }\r\n              //let arrayAudioBuffer=this._audioDataHolder.arrayBuffer;\r\n              let arrAbBuf: Float32Array[] | null;\r\n              let ada = new Array<ArrayBuffer>(chs);\r\n              // for(let ch=0;ch<chs;ch++) {\r\n              //   ada[ch] = new Float32Array(this.dftSize);\r\n              // }\r\n              let imgData: Uint8ClampedArray;\r\n              let maxPsd = -Infinity;\r\n              let norender = true;\r\n\r\n              if (this.worker) {\r\n                this.worker.onmessage = (me) => {\r\n                  if (true === me.data.terminate) {\r\n                    let drawImgData;\r\n                    if (imgData) {\r\n                      drawImgData = imgData;\r\n                    } else {\r\n                      drawImgData = me.data.imgData;\r\n                    }\r\n                    this.drawRendered(w, h, drawImgData);\r\n                    if (this.worker) {\r\n                      this.worker.terminate();\r\n                    }\r\n                    this.worker = null;\r\n                    raAs.close();\r\n                  } else {\r\n\r\n                    // set rendered vertical values of one pixel of timescale\r\n                    //let dataPos = renderPos * h * 4;\r\n                    if (norender) {\r\n                      if (me.data.maxPsd > maxPsd) {\r\n                        maxPsd = me.data.maxPsd;\r\n                        //console.debug(\"new maxPsd: \"+maxPsd);\r\n                      }\r\n                    } else {\r\n                      let chH = Math.round(h / chs);\r\n                      let idp = me.data.l - leftPos;\r\n                      for (let ch = 0; ch < chs; ch++) {\r\n                        for (let y = 0; y < chH; y++) {\r\n                          let py = chH - y;\r\n                          let dataPos = ((((ch * chH) + py) * w) + idp) * 4;\r\n                          let mePos = ((ch * chH) + py) * 4;\r\n                          //let lastPos = dataPos + me.data.imgData.length;\r\n                          //if (lastPos < imgData.length) {\r\n                          // set one pixel\r\n\r\n                          imgData[dataPos] = me.data.imgData[mePos];\r\n                          imgData[dataPos + 1] = me.data.imgData[mePos + 1];\r\n                          imgData[dataPos + 2] = me.data.imgData[mePos + 2];\r\n                          imgData[dataPos + 3] = me.data.imgData[mePos + 3];\r\n                          //} else {\r\n                          //console.error(\"Out of range: \" + dataPos + \"+\" + me.data.imgData.length + \">=\" + imgData.length);\r\n                          // }\r\n                        }\r\n                      }\r\n                    }\r\n                    if (this._audioDataHolder && arrAbBuf && this.worker) {\r\n                      // proceed with next pixel\r\n                      renderPos++;\r\n                      //console.debug(\"Render pos: \"+renderPos);\r\n\r\n                      let terminate = false;\r\n                      let windowEnd = renderPos >= leftPos + w;\r\n\r\n                      if (windowEnd) {\r\n                        if (norender) {\r\n                          // phase two: rendering\r\n                          norender = false;\r\n                          // start from beginning\r\n                          renderPos = leftPos;\r\n                          //console.debug(\"now rendering: maxPsd: \"+maxPsd);\r\n                        } else {\r\n                          // terminate render phase\r\n                          terminate = true;\r\n                        }\r\n                      }\r\n\r\n                      let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;\r\n                      if (leftFramePos < 0) {\r\n                        leftFramePos = 0;\r\n                      }\r\n                      //let ada = new Array<ArrayBuffer>(chs);\r\n\r\n                      //console.debug(\"Render pos: \"+renderPos+\" leftFramePos: \"+leftFramePos);\r\n\r\n                      if (!terminate) {\r\n                        if (this._audioDataHolder) {\r\n                          //let read = this._audioDataHolder.frames(leftFramePos, this.dftSize, arrAbBuf);\r\n                          this.raAsSubsc = raAs.framesObs(leftFramePos, this.dftSize, arrAbBuf).subscribe(\r\n                            {\r\n                              next: (read) => {\r\n                                if (arrAbBuf && this.worker) {\r\n                                  if (read < this.dftSize) {\r\n                                    // zero padding\r\n                                    for (let ch = 0; ch < chs; ch++) {\r\n                                      for (let zp = read; zp < this.dftSize; zp++) {\r\n                                        arrAbBuf[ch][zp] = 0;\r\n                                      }\r\n                                    }\r\n                                  }\r\n                                  for (let ch = 0; ch < chs; ch++) {\r\n                                    // Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker\r\n                                    //ada[ch] = arrAbBuf[ch].buffer.slice(0);\r\n                                    const arrAbBufCh = arrAbBuf[ch];\r\n                                    const adLen = arrAbBufCh.buffer.byteLength;\r\n                                    if (!ada[ch] || ada[ch].byteLength !== adLen) {\r\n                                      ada[ch] = new ArrayBuffer(adLen);\r\n                                    }\r\n                                    let fAdaCh = new Float32Array(ada[ch]);\r\n                                    fAdaCh.set(arrAbBuf[ch]);\r\n                                  }\r\n\r\n                                  this.worker.postMessage({\r\n                                    audioData: ada,\r\n                                    audioDataOffset: leftFramePos,\r\n                                    l: renderPos,\r\n                                    w: me.data.w,\r\n                                    h: h,\r\n                                    vw: vw,\r\n                                    chs: chs,\r\n                                    frameLength: frameLength,\r\n                                    dftSize: this.dftSize,\r\n                                    maxPsd: maxPsd,\r\n                                    norender: norender,\r\n                                    terminate: terminate\r\n                                  }, ada);\r\n                                }\r\n                              },\r\n                              error: (err) => {\r\n                                console.error(\"Sonagram: Error reading audio data: \" + err);\r\n                              }\r\n                            }\r\n                          )\r\n\r\n\r\n                        }\r\n                      } else {\r\n                        for (let ch = 0; ch < chs; ch++) {\r\n                          ada[ch] = new ArrayBuffer(0);\r\n                        }\r\n                        this.worker.postMessage({\r\n                          audioData: ada,\r\n                          audioDataOffset: leftFramePos,\r\n                          l: renderPos,\r\n                          w: me.data.w,\r\n                          h: h,\r\n                          vw: vw,\r\n                          chs: chs,\r\n                          frameLength: frameLength,\r\n                          dftSize: this.dftSize,\r\n                          maxPsd: maxPsd,\r\n                          norender: norender,\r\n                          terminate: terminate\r\n                        }, ada);\r\n                      }\r\n\r\n\r\n                    }\r\n\r\n                  }\r\n                }\r\n              }\r\n              if (audioBuffer && audioBuffer.length * audioBuffer.numberOfChannels < AudioCanvasLayerComponent.ENABLE_STREAMING_NUMBER_OF_SAMPLES_THRESHOLD) {\r\n                let ada = new Array<ArrayBuffer>(chs);\r\n                for (let ch = 0; ch < chs; ch++) {\r\n                  // Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker\r\n                  ada[ch] = audioBuffer.getChannelData(ch).buffer.slice(0);\r\n                }\r\n                //let start = Date.now();\r\n\r\n                if (this.markerCanvas) {\r\n                  let g = this.markerCanvas.getContext(\"2d\");\r\n                  if (g) {\r\n                    g.fillText(\"Rendering...\", 10, 20);\r\n                  }\r\n\r\n                }\r\n                this.worker.postMessage({\r\n                  audioData: ada,\r\n                  l: leftPos,\r\n                  w: w,\r\n                  h: h,\r\n                  vw: Math.round(this.virtualDimension.width),\r\n                  chs: chs,\r\n                  frameLength: frameLength,\r\n                  dftSize: this.dftSize,\r\n                  terminate: true\r\n                }, ada);\r\n              } else {\r\n                if (w > 0) {\r\n\r\n                  if (framesPerPixel > 0) {\r\n                    let arrSize = w * h * 4;\r\n                    if (arrSize < 0) {\r\n                      arrSize = 0\r\n                    }\r\n                    imgData = new Uint8ClampedArray(arrSize);\r\n                    let rw = 1;\r\n                    arrAbBuf = new Array<Float32Array>(chs);\r\n\r\n                    for (let ch = 0; ch < chs; ch++) {\r\n                      arrAbBuf[ch] = new Float32Array(this.dftSize);\r\n                    }\r\n\r\n                    let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;\r\n                    let framesToRead = this.dftSize;\r\n                    if (leftFramePos < 0) {\r\n                      leftFramePos = 0;\r\n                    }\r\n                    this.drawStateText('Loading/Rendering...');\r\n                    this.raAsSubsc = raAs.framesObs(leftFramePos, framesToRead, arrAbBuf).subscribe(\r\n                      {\r\n                        next: (read) => {\r\n                          if (arrAbBuf && this.worker) {\r\n                            if (read < this.dftSize) {\r\n                              // zero padding\r\n                              for (let ch = 0; ch < chs; ch++) {\r\n                                for (let zp = read; zp < this.dftSize; zp++) {\r\n                                  arrAbBuf[ch][zp] = 0;\r\n                                }\r\n                              }\r\n                            }\r\n                            for (let ch = 0; ch < chs; ch++) {\r\n                              const arrAbBufCh = arrAbBuf[ch];\r\n                              const adLen = arrAbBufCh.buffer.byteLength;\r\n                              if (!ada[ch] || ada[ch].byteLength !== adLen) {\r\n                                ada[ch] = new ArrayBuffer(adLen);\r\n                              }\r\n                              let fAdaCh = new Float32Array(ada[ch]);\r\n                              fAdaCh.set(arrAbBuf[ch]);\r\n                            }\r\n\r\n                            this.worker.postMessage({\r\n                              l: renderPos,\r\n                              w: rw,\r\n                              h: h,\r\n                              vw: vw,\r\n                              chs: chs,\r\n                              frameLength: frameLength,\r\n                              audioData: ada,\r\n                              audioDataOffset: leftFramePos,\r\n                              dftSize: this.dftSize,\r\n                              norender: norender,\r\n                              terminate: false\r\n                            }, ada);\r\n                          }\r\n                        },\r\n                        error: (err) => {\r\n                          console.error(\"Sonagram: Error reading audio data: \" + err);\r\n                        }\r\n                      }\r\n                    );\r\n                  }\r\n                }\r\n              }\r\n            }\r\n          }\r\n        });\r\n            } else {\r\n        if (this.bounds && this.bounds.dimension) {\r\n          let w = Math.round(this.bounds.dimension.width);\r\n          let h = Math.round(this.bounds.dimension.height);\r\n          let g = this.sonagramCanvas.getContext(\"2d\");\r\n          if (g) {\r\n            g.clearRect(0, 0, w, h);\r\n          }\r\n        }\r\n      }\r\n    }\r\n\r\n    drawRendered(w:number,h:number,imgData:Uint8ClampedArray) {\r\n        if (this.sonagramCanvas) {\r\n            this.sonagramCanvas.width = w;\r\n            this.sonagramCanvas.height = h;\r\n            let g = this.sonagramCanvas.getContext(\"2d\");\r\n            if (g) {\r\n                if (w > 0 && h > 0) {\r\n                    let gImgData = g.createImageData(w, h);\r\n                    gImgData.data.set(imgData);\r\n                    g.putImageData(gImgData, 0, 0);\r\n                }\r\n            }\r\n        }\r\n        this.drawBg();\r\n        this.drawPlayPosition();\r\n    }\r\n\r\n//     // synchronous draw (not used anymore)\r\n//   redraw() {\r\n//\r\n//     let g = this.sonagramCanvas.getContext(\"2d\");\r\n//\r\n//     let w = this.sonagramCanvas.width;\r\n//     let h = this.sonagramCanvas.height;\r\n//     if (g) {\r\n//       g.clearRect(0, 0, w, h);\r\n//       g.fillStyle = \"white\";\r\n//       g.fillRect(0, 0, w, h);\r\n//       if (this._audioDataHolder) {\r\n//         let spectSize = Math.floor(this.dftSize / 2)\r\n//         let chs = this._audioDataHolder.numberOfChannels;\r\n//         let chH = h / chs;\r\n//\r\n//         let frameLength = this._audioDataHolder.frameLen;\r\n//\r\n//         let framesPerPixel = frameLength / w;\r\n//         let y = 0;\r\n//         let audioBuffer:AudioBuffer|null=null;\r\n//         let audioSource=this._audioDataHolder.audioSource;\r\n//         if(audioSource instanceof AudioBufferSource){\r\n//           audioBuffer=audioSource.audioBuffer;\r\n//         }\r\n//\r\n//         if(audioBuffer) {\r\n//           let b = new Float32Array(this.dftSize)\r\n//\r\n//           let sona = new Array<Array<Float32Array>>(chs);\r\n//           let max = 0;\r\n//           let maxPsd = -Infinity;\r\n//           for (let ch = 0; ch < chs; ch++) {\r\n//             let x = 0;\r\n//             sona[ch] = new Array<Float32Array>(w);\r\n//\r\n//             let chData = audioBuffer.getChannelData(ch);\r\n//             // TODO center buffer\r\n//\r\n//             let framePos = 0;\r\n//             for (let pii = 0; pii < w; pii++) {\r\n//               framePos = Math.round(pii * framesPerPixel);\r\n//               // calculate DFT at pixel position\r\n//               for (let i = 0; i < this.dftSize; i++) {\r\n//                 let chDat = chData[framePos + i];\r\n//                 b[i] = chDat;\r\n//               }\r\n//               let spectr = this.dft.processRealMagnitude(b);\r\n//               sona[ch][pii] = spectr;\r\n//               // @ts-ignore\r\n//               let pMax = Math.max.apply(null, spectr);\r\n//               if (pMax > max) {\r\n//                 max = pMax;\r\n//               }\r\n//\r\n//               for (let s = 0; s < spectSize; s++) {\r\n//                 let psd = (2 * Math.pow(spectr[s], 2)) / spectSize;\r\n//                 if (psd > maxPsd) {\r\n//                   maxPsd = psd;\r\n//                 }\r\n//               }\r\n//             }\r\n//           }\r\n//           //console.log(\"max: \", max);\r\n//           maxPsd = (2 * Math.pow(max, 2)) / spectSize;\r\n//           for (let ch = 0; ch < chs; ch++) {\r\n//\r\n//             let framePos = 0;\r\n//             for (let pii = 0; pii < w; pii++) {\r\n//               framePos = pii * framesPerPixel;\r\n//\r\n//               for (let y = 0; y < h; y++) {\r\n//                 let freqIdx = Math.round(y * spectSize / h);\r\n//\r\n//                 // calculate the one sided power spectral density PSD (f, t) in Pa2/Hz\r\n//                 // PSD(f) proportional to 2|X(f)|2 / (t2 - t1)\r\n//                 let val = sona[ch][pii][freqIdx];\r\n//                 let psd = (2 * Math.pow(val, 2)) / spectSize;\r\n//\r\n//                 // Calculate logarithmic\r\n//                 let psdLog = DSPUtils.toLevelInDB(psd / maxPsd);\r\n//                 let dynRangeInDb = 70;\r\n//                 let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;\r\n//\r\n//                 if (scaledVal > 1)\r\n//                   scaledVal = 1;\r\n//                 if (scaledVal < 0) {\r\n//                   scaledVal = 0;\r\n//                 }\r\n//                 let rgbVal = (255 * scaledVal);\r\n//                 if (rgbVal < 0) {\r\n// //\t\t\t\t\t\t\tSystem.out.println(\"Neg RGB val: \"+rgbVal);\r\n//                   rgbVal = 0;\r\n//                 }\r\n//                 if (rgbVal > 255) {\r\n//                   rgbVal = 255;\r\n//                 }\r\n//                 rgbVal = 255 - rgbVal;\r\n//                 let colorStr = CSSUtils.toColorString(rgbVal, rgbVal, rgbVal);\r\n//                 g.fillStyle = colorStr;\r\n//                 g.fillRect(pii, chH - y, 1, 1);\r\n//               }\r\n//             }\r\n//           }\r\n//           this.drawPlayPosition();\r\n//         }else{\r\n//           throw Error(\"Redraw only supported with audio buffer.\")\r\n//         }\r\n//       }\r\n//     }\r\n//   }\r\n//\r\n\r\n    setData(audioData: AudioDataHolder | null) {\r\n        this._audioDataHolder = audioData;\r\n        this.playFramePosition = 0;\r\n    }\r\n\r\n}\r\n\r\n"]}