speechrecorderng 2.25.2 → 3.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/audio/array_audio_buffer.mjs +101 -0
- package/esm2020/lib/audio/array_audio_buffer_input_stream.mjs +86 -0
- package/esm2020/lib/audio/audio_data_holder.mjs +109 -0
- package/esm2020/lib/audio/audio_display.mjs +5 -5
- package/esm2020/lib/audio/audio_player.mjs +9 -7
- package/esm2020/lib/audio/capture/capture.mjs +6 -1
- package/esm2020/lib/audio/dsp/level_measure.mjs +97 -55
- package/esm2020/lib/audio/io/stream.mjs +3 -3
- package/esm2020/lib/audio/persistor.mjs +12 -6
- package/esm2020/lib/audio/playback/array_audio_buffer_source_node.mjs +265 -0
- package/esm2020/lib/audio/playback/player.mjs +184 -50
- package/esm2020/lib/audio/ui/audio_canvas_layer_comp.mjs +25 -13
- package/esm2020/lib/audio/ui/audio_display_scroll_pane.mjs +2 -2
- package/esm2020/lib/audio/ui/audiosignal.mjs +206 -77
- package/esm2020/lib/audio/ui/container.mjs +14 -14
- package/esm2020/lib/audio/ui/livelevel.mjs +17 -3
- package/esm2020/lib/audio/ui/sonagram.mjs +343 -160
- package/esm2020/lib/io/stream.mjs +77 -1
- package/esm2020/lib/net/uploader.mjs +15 -3
- package/esm2020/lib/speechrecorder/recording.mjs +63 -7
- package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +96 -7
- package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +119 -106
- package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +34 -7
- package/esm2020/lib/speechrecorder/session/progress.mjs +2 -2
- package/esm2020/lib/speechrecorder/session/prompting.mjs +2 -2
- package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +6 -3
- package/esm2020/lib/speechrecorder/session/recording_file_cache.mjs +195 -0
- package/esm2020/lib/speechrecorder/session/recording_list.mjs +23 -10
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +12 -6
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +5 -5
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +13 -9
- package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +8 -5
- package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +123 -101
- package/esm2020/lib/speechrecorderng.component.mjs +2 -2
- package/esm2020/lib/speechrecorderng.module.mjs +5 -4
- package/esm2020/lib/spr.module.version.mjs +2 -2
- package/esm2020/lib/ui/recordingitem_display.mjs +2 -2
- package/fesm2015/speechrecorderng.mjs +2355 -753
- package/fesm2015/speechrecorderng.mjs.map +1 -1
- package/fesm2020/speechrecorderng.mjs +2338 -746
- package/fesm2020/speechrecorderng.mjs.map +1 -1
- package/lib/audio/array_audio_buffer.d.ts +15 -0
- package/lib/audio/array_audio_buffer_input_stream.d.ts +14 -0
- package/lib/audio/audio_data_holder.d.ts +29 -0
- package/lib/audio/audio_display.d.ts +2 -1
- package/lib/audio/audio_player.d.ts +2 -1
- package/lib/audio/capture/capture.d.ts +2 -0
- package/lib/audio/dsp/level_measure.d.ts +2 -1
- package/lib/audio/persistor.d.ts +8 -3
- package/lib/audio/playback/array_audio_buffer_source_node.d.ts +24 -0
- package/lib/audio/playback/player.d.ts +6 -0
- package/lib/audio/ui/audio_canvas_layer_comp.d.ts +4 -1
- package/lib/audio/ui/audio_display_scroll_pane.d.ts +2 -1
- package/lib/audio/ui/audiosignal.d.ts +3 -2
- package/lib/audio/ui/container.d.ts +3 -2
- package/lib/audio/ui/livelevel.d.ts +4 -2
- package/lib/audio/ui/sonagram.d.ts +3 -2
- package/lib/io/stream.d.ts +17 -0
- package/lib/net/uploader.d.ts +6 -1
- package/lib/speechrecorder/recording.d.ts +16 -3
- package/lib/speechrecorder/recordings/recordings.service.d.ts +2 -0
- package/lib/speechrecorder/session/audiorecorder.d.ts +2 -7
- package/lib/speechrecorder/session/basicrecorder.d.ts +10 -5
- package/lib/speechrecorder/session/progress.d.ts +1 -1
- package/lib/speechrecorder/session/prompting.d.ts +1 -1
- package/lib/speechrecorder/session/recorder_combi_pane.d.ts +3 -1
- package/lib/speechrecorder/session/recording_file_cache.d.ts +33 -0
- package/lib/speechrecorder/session/recording_list.d.ts +3 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +2 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-view.component.d.ts +1 -0
- package/lib/speechrecorder/session/sessionmanager.d.ts +5 -11
- package/lib/speechrecorderng.module.d.ts +2 -1
- package/lib/spr.module.version.d.ts +1 -1
- package/package.json +1 -1
|
@@ -14,7 +14,7 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
14
14
|
this._playFramePosition = null;
|
|
15
15
|
this.dftSize = DEFAULT_DFT_SIZE;
|
|
16
16
|
this.worker = null;
|
|
17
|
-
this.
|
|
17
|
+
this._audioDataHolder = null;
|
|
18
18
|
this.markers = new Array();
|
|
19
19
|
this.dft = new DFTFloat32(this.dftSize);
|
|
20
20
|
this.workerURL = WorkerHelper.buildWorkerBlobURL(this.workerFunction);
|
|
@@ -67,7 +67,7 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
67
67
|
g.lineTo(xViewPortPixelpos, h);
|
|
68
68
|
g.closePath();
|
|
69
69
|
g.stroke();
|
|
70
|
-
if (this.
|
|
70
|
+
if (this._audioDataHolder) {
|
|
71
71
|
let framePosRound = this.viewPortXPixelToFramePosition(xViewPortPixelpos);
|
|
72
72
|
if (framePosRound != null) {
|
|
73
73
|
g.font = '14px sans-serif';
|
|
@@ -331,11 +331,21 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
331
331
|
}
|
|
332
332
|
GaussianWindow.DEFAULT_SIGMA = 0.3;
|
|
333
333
|
self.onmessage = function (msg) {
|
|
334
|
+
//console.debug("Sonagram render thread");
|
|
334
335
|
let l = msg.data.l;
|
|
335
336
|
let w = msg.data.w;
|
|
336
337
|
let h = msg.data.h;
|
|
337
338
|
let vw = msg.data.vw;
|
|
338
339
|
let chs = msg.data.chs;
|
|
340
|
+
let audioDataOffset = 0;
|
|
341
|
+
let adOffset = msg.data.audioDataOffset;
|
|
342
|
+
if (adOffset) {
|
|
343
|
+
audioDataOffset = adOffset;
|
|
344
|
+
}
|
|
345
|
+
let maxPsd = null;
|
|
346
|
+
if (msg.data.maxPsd !== undefined) {
|
|
347
|
+
maxPsd = msg.data.maxPsd;
|
|
348
|
+
}
|
|
339
349
|
let audioData = new Array(chs);
|
|
340
350
|
for (let ch = 0; ch < chs; ch++) {
|
|
341
351
|
audioData[ch] = new Float32Array(msg.data['audioData'][ch]);
|
|
@@ -351,16 +361,17 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
351
361
|
}
|
|
352
362
|
let imgData = new Uint8ClampedArray(arrSize);
|
|
353
363
|
//console.log("Render method:");
|
|
354
|
-
|
|
364
|
+
//console.debug("Created imgData arrSize: "+arrSize+" ", w, "x", h);
|
|
365
|
+
let calcMaxPsd = -Infinity;
|
|
366
|
+
if (arrSize > 0) {
|
|
355
367
|
let chH = Math.round(h / chs);
|
|
356
368
|
let framesPerPixel = frameLength / vw;
|
|
357
|
-
//console.
|
|
369
|
+
//console.debug("Render: ", w, "x", h);
|
|
358
370
|
let b = new Float32Array(dftSize);
|
|
359
371
|
let sona = new Array(chs);
|
|
360
|
-
let
|
|
361
|
-
let p = 0;
|
|
372
|
+
//let p = 0;
|
|
362
373
|
for (let ch = 0; ch < chs; ch++) {
|
|
363
|
-
p = ch * frameLength;
|
|
374
|
+
//p = ch * frameLength;
|
|
364
375
|
let chDataLen = audioData[ch].length;
|
|
365
376
|
let x = 0;
|
|
366
377
|
// initialize DFT array buffer
|
|
@@ -377,8 +388,13 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
377
388
|
// initialize for negative sample positions and out of bounds positions
|
|
378
389
|
let chDat = 0;
|
|
379
390
|
// Set audio sample if available
|
|
380
|
-
|
|
381
|
-
|
|
391
|
+
let adp = samplePos - audioDataOffset;
|
|
392
|
+
if (adp >= 0 && adp < chDataLen) {
|
|
393
|
+
chDat = audioData[ch][adp];
|
|
394
|
+
//console.debug("Audio data: "+chDat);
|
|
395
|
+
}
|
|
396
|
+
else {
|
|
397
|
+
//console.debug("Sample buf pos oob: adp: "+adp+", chDataLen: "+chDataLen+", samplePos: "+samplePos+", i: "+i);
|
|
382
398
|
}
|
|
383
399
|
// apply Window
|
|
384
400
|
b[i] = chDat * wf.getScale(i);
|
|
@@ -388,63 +404,78 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
388
404
|
// Get maximum value of spectral energy
|
|
389
405
|
for (let s = 0; s < dftBands; s++) {
|
|
390
406
|
let psd = (2 * Math.pow(spectr[s], 2)) / dftBands;
|
|
391
|
-
if (psd >
|
|
392
|
-
|
|
407
|
+
if (psd > calcMaxPsd) {
|
|
408
|
+
calcMaxPsd = psd;
|
|
393
409
|
}
|
|
394
410
|
}
|
|
395
411
|
// Set render model data for this pixel
|
|
396
412
|
sona[ch][pii] = spectr;
|
|
397
413
|
}
|
|
398
414
|
}
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
scaledVal =
|
|
419
|
-
|
|
420
|
-
scaledVal
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
rgbVal =
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
415
|
+
if (!msg.data.norender) {
|
|
416
|
+
if (!maxPsd) {
|
|
417
|
+
maxPsd = calcMaxPsd;
|
|
418
|
+
}
|
|
419
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
420
|
+
for (let pii = 0; pii < w; pii++) {
|
|
421
|
+
let allBlack = true;
|
|
422
|
+
for (let y = 0; y < chH; y++) {
|
|
423
|
+
let freqIdx = Math.round(y * dftBands / chH);
|
|
424
|
+
// calculate the one sided power spectral density PSD (f, t) in Pa2/Hz
|
|
425
|
+
// PSD(f) proportional to 2|X(f)|2 / (t2 - t1)
|
|
426
|
+
let val = sona[ch][pii][freqIdx];
|
|
427
|
+
let psd = (2 * Math.pow(val, 2)) / dftBands;
|
|
428
|
+
// Calculate logarithmic value
|
|
429
|
+
//let psdLog = ips.dsp.DSPUtils.toLevelInDB(psd / maxPsd);
|
|
430
|
+
let linearLevel = psd / maxPsd;
|
|
431
|
+
let psdLog = 10 * Math.log(linearLevel) / Math.log(10);
|
|
432
|
+
// Fixed dynamic Range value of 70dB for now
|
|
433
|
+
let dynRangeInDb = 70;
|
|
434
|
+
let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;
|
|
435
|
+
// are the following checks necessary for clamped array ?
|
|
436
|
+
if (scaledVal > 1.0)
|
|
437
|
+
scaledVal = 1;
|
|
438
|
+
if (scaledVal < 0.0) {
|
|
439
|
+
scaledVal = 0;
|
|
440
|
+
}
|
|
441
|
+
let rgbVal = Math.round(255 * scaledVal);
|
|
442
|
+
if (rgbVal < 0) {
|
|
443
|
+
// System.out.println("Neg RGB val: "+rgbVal);
|
|
444
|
+
rgbVal = 0;
|
|
445
|
+
}
|
|
446
|
+
if (rgbVal > 255) {
|
|
447
|
+
rgbVal = 255;
|
|
448
|
+
}
|
|
449
|
+
rgbVal = 255 - rgbVal;
|
|
450
|
+
if (rgbVal > 0) {
|
|
451
|
+
allBlack = false;
|
|
452
|
+
}
|
|
453
|
+
let py = chH - y;
|
|
454
|
+
let dataPos = ((((ch * chH) + py) * w) + pii) * 4;
|
|
455
|
+
imgData[dataPos + 0] = rgbVal; //R
|
|
456
|
+
imgData[dataPos + 1] = rgbVal; //G
|
|
457
|
+
imgData[dataPos + 2] = rgbVal; //B
|
|
458
|
+
imgData[dataPos + 3] = 255; //A (alpha: fully opaque)
|
|
459
|
+
//console.debug("Rendered: py: "+py+", rgbval: "+rgbVal);
|
|
460
|
+
// example 1x1, 2chs
|
|
461
|
+
// ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,
|
|
462
|
+
// ch0x1y0R,ch0x1y0G,ch0x1y0B,ch0x1y0A,
|
|
463
|
+
// ch0x0y0R,ch0x0y0G,ch0x0y0B,ch0x0y0A,
|
|
464
|
+
// ch0x1y1R,ch0x1y1G,ch0x1y1B,ch0x1y1A,
|
|
465
|
+
// ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,
|
|
466
|
+
// ch1x1y0R,ch1x1y0G,ch1x1y0B,ch1x1y0A,
|
|
467
|
+
// ch1x0y0R,ch1x0y0G,ch1x0y0B,ch1x0y0A,
|
|
468
|
+
// ch1x1y1R,ch1x1y1G,ch1x1y1B,ch1x1y1A
|
|
433
469
|
}
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
imgData[dataPos + 1] = rgbVal; //G
|
|
438
|
-
imgData[dataPos + 2] = rgbVal; //B
|
|
439
|
-
imgData[dataPos + 3] = 255; //A (alpha: fully opaque)
|
|
470
|
+
// if (allBlack) {
|
|
471
|
+
// console.log("Black: ", pii, " ", ch);
|
|
472
|
+
// }
|
|
440
473
|
}
|
|
441
|
-
// if (allBlack) {
|
|
442
|
-
// console.log("Black: ", pii, " ", ch);
|
|
443
|
-
// }
|
|
444
474
|
}
|
|
445
475
|
}
|
|
446
476
|
}
|
|
447
|
-
|
|
477
|
+
//console.debug("Render thread post message imgData: "+imgData.length)
|
|
478
|
+
postMessage({ imgData: imgData, l: l, w: msg.data.w, h: msg.data.h, vw: vw, maxPsd: calcMaxPsd, terminate: msg.data.terminate }, [imgData.buffer]);
|
|
448
479
|
};
|
|
449
480
|
}
|
|
450
481
|
startDraw(clear = true) {
|
|
@@ -474,42 +505,190 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
474
505
|
if (this.bounds) {
|
|
475
506
|
let w = Math.round(this.bounds.dimension.width);
|
|
476
507
|
let h = Math.round(this.bounds.dimension.height);
|
|
477
|
-
if (this.
|
|
508
|
+
if (this._audioDataHolder && w > 0 && h > 0) {
|
|
478
509
|
this.worker = new Worker(this.workerURL);
|
|
479
510
|
//this.wo = new Worker('./worker/sonagram.worker', { type: `module` });
|
|
480
|
-
let chs = this.
|
|
481
|
-
let
|
|
482
|
-
let
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
let
|
|
511
|
+
let chs = this._audioDataHolder.numberOfChannels;
|
|
512
|
+
let vw = Math.round(this.virtualDimension.width);
|
|
513
|
+
let frameLength = this._audioDataHolder.frameLen;
|
|
514
|
+
let framesPerPixel = Math.ceil(frameLength / vw);
|
|
515
|
+
let leftPos = Math.round(this.bounds.position.left);
|
|
516
|
+
let renderPos = leftPos;
|
|
517
|
+
let audioBuffer = this._audioDataHolder.buffer;
|
|
518
|
+
//let arrayAudioBuffer=this._audioDataHolder.arrayBuffer;
|
|
519
|
+
let arrAbBuf;
|
|
520
|
+
let imgData;
|
|
521
|
+
let maxPsd = -Infinity;
|
|
522
|
+
let norender = true;
|
|
488
523
|
if (this.worker) {
|
|
489
524
|
this.worker.onmessage = (me) => {
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
525
|
+
if (true === me.data.terminate) {
|
|
526
|
+
let drawImgData;
|
|
527
|
+
if (imgData) {
|
|
528
|
+
drawImgData = imgData;
|
|
529
|
+
}
|
|
530
|
+
else {
|
|
531
|
+
drawImgData = me.data.imgData;
|
|
532
|
+
}
|
|
533
|
+
this.drawRendered(w, h, drawImgData);
|
|
534
|
+
if (this.worker) {
|
|
535
|
+
this.worker.terminate();
|
|
536
|
+
}
|
|
537
|
+
this.worker = null;
|
|
538
|
+
}
|
|
539
|
+
else {
|
|
540
|
+
// set rendered vertical values of one pixel of timescale
|
|
541
|
+
//let dataPos = renderPos * h * 4;
|
|
542
|
+
if (norender) {
|
|
543
|
+
if (me.data.maxPsd > maxPsd) {
|
|
544
|
+
maxPsd = me.data.maxPsd;
|
|
545
|
+
//console.debug("new maxPsd: "+maxPsd);
|
|
546
|
+
}
|
|
547
|
+
}
|
|
548
|
+
else {
|
|
549
|
+
let chH = Math.round(h / chs);
|
|
550
|
+
let idp = me.data.l - leftPos;
|
|
551
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
552
|
+
for (let y = 0; y < chH; y++) {
|
|
553
|
+
let py = chH - y;
|
|
554
|
+
let dataPos = ((((ch * chH) + py) * w) + idp) * 4;
|
|
555
|
+
let mePos = ((ch * chH) + py) * 4;
|
|
556
|
+
//let lastPos = dataPos + me.data.imgData.length;
|
|
557
|
+
//if (lastPos < imgData.length) {
|
|
558
|
+
// set one pixel
|
|
559
|
+
imgData[dataPos] = me.data.imgData[mePos];
|
|
560
|
+
imgData[dataPos + 1] = me.data.imgData[mePos + 1];
|
|
561
|
+
imgData[dataPos + 2] = me.data.imgData[mePos + 2];
|
|
562
|
+
imgData[dataPos + 3] = me.data.imgData[mePos + 3];
|
|
563
|
+
//} else {
|
|
564
|
+
//console.error("Out of range: " + dataPos + "+" + me.data.imgData.length + ">=" + imgData.length);
|
|
565
|
+
// }
|
|
566
|
+
}
|
|
567
|
+
}
|
|
568
|
+
}
|
|
569
|
+
if (this._audioDataHolder && arrAbBuf && this.worker) {
|
|
570
|
+
// proceed with next pixel
|
|
571
|
+
renderPos++;
|
|
572
|
+
//console.debug("Render pos: "+renderPos);
|
|
573
|
+
let terminate = false;
|
|
574
|
+
let windowEnd = renderPos >= leftPos + w;
|
|
575
|
+
if (windowEnd) {
|
|
576
|
+
if (norender) {
|
|
577
|
+
// phase two: rendering
|
|
578
|
+
norender = false;
|
|
579
|
+
// start from beginning
|
|
580
|
+
renderPos = leftPos;
|
|
581
|
+
//console.debug("now rendering: maxPsd: "+maxPsd);
|
|
582
|
+
}
|
|
583
|
+
else {
|
|
584
|
+
// terminate render phase
|
|
585
|
+
terminate = true;
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;
|
|
589
|
+
if (leftFramePos < 0) {
|
|
590
|
+
leftFramePos = 0;
|
|
591
|
+
}
|
|
592
|
+
let ada = new Array(chs);
|
|
593
|
+
//console.debug("Render pos: "+renderPos+" leftFramePos: "+leftFramePos);
|
|
594
|
+
if (!terminate) {
|
|
595
|
+
if (this._audioDataHolder) {
|
|
596
|
+
let read = this._audioDataHolder.frames(leftFramePos, this.dftSize, arrAbBuf);
|
|
597
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
598
|
+
// Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker
|
|
599
|
+
ada[ch] = arrAbBuf[ch].buffer.slice(0);
|
|
600
|
+
}
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
else {
|
|
604
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
605
|
+
ada[ch] = new ArrayBuffer(0);
|
|
606
|
+
}
|
|
607
|
+
}
|
|
608
|
+
this.worker.postMessage({
|
|
609
|
+
audioData: ada,
|
|
610
|
+
audioDataOffset: leftFramePos,
|
|
611
|
+
l: renderPos,
|
|
612
|
+
w: me.data.w,
|
|
613
|
+
h: h,
|
|
614
|
+
vw: vw,
|
|
615
|
+
chs: chs,
|
|
616
|
+
frameLength: frameLength,
|
|
617
|
+
dftSize: this.dftSize,
|
|
618
|
+
maxPsd: maxPsd,
|
|
619
|
+
norender: norender,
|
|
620
|
+
terminate: terminate
|
|
621
|
+
}, ada);
|
|
622
|
+
}
|
|
493
623
|
}
|
|
494
|
-
this.worker = null;
|
|
495
624
|
};
|
|
496
625
|
}
|
|
497
|
-
if (
|
|
498
|
-
let
|
|
499
|
-
|
|
500
|
-
|
|
626
|
+
if (audioBuffer && audioBuffer.length * audioBuffer.numberOfChannels < AudioCanvasLayerComponent.ENABLE_STREAMING_NUMBER_OF_SAMPLES_THRESHOLD) {
|
|
627
|
+
let ada = new Array(chs);
|
|
628
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
629
|
+
// Need a copy here for the worker, otherwise this.audioData is not accessible after posting to the worker
|
|
630
|
+
ada[ch] = audioBuffer.getChannelData(ch).buffer.slice(0);
|
|
631
|
+
}
|
|
632
|
+
let start = Date.now();
|
|
633
|
+
if (this.markerCanvas) {
|
|
634
|
+
let g = this.markerCanvas.getContext("2d");
|
|
635
|
+
if (g) {
|
|
636
|
+
g.fillText("Rendering...", 10, 20);
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
this.worker.postMessage({
|
|
640
|
+
audioData: ada,
|
|
641
|
+
l: leftPos,
|
|
642
|
+
w: w,
|
|
643
|
+
h: h,
|
|
644
|
+
vw: Math.round(this.virtualDimension.width),
|
|
645
|
+
chs: chs,
|
|
646
|
+
frameLength: frameLength,
|
|
647
|
+
dftSize: this.dftSize,
|
|
648
|
+
terminate: true
|
|
649
|
+
}, ada);
|
|
650
|
+
}
|
|
651
|
+
else {
|
|
652
|
+
if (w > 0) {
|
|
653
|
+
if (framesPerPixel > 0) {
|
|
654
|
+
let arrSize = w * h * 4;
|
|
655
|
+
if (arrSize < 0) {
|
|
656
|
+
arrSize = 0;
|
|
657
|
+
}
|
|
658
|
+
imgData = new Uint8ClampedArray(arrSize);
|
|
659
|
+
let rw = 1;
|
|
660
|
+
//ais = new ArrayAudioBufferInputStream(arrayAudioBuffer);
|
|
661
|
+
arrAbBuf = new Array(chs);
|
|
662
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
663
|
+
arrAbBuf[ch] = new Float32Array(this.dftSize);
|
|
664
|
+
}
|
|
665
|
+
let leftFramePos = Math.floor(frameLength * renderPos / vw) - this.dftSize / 2;
|
|
666
|
+
let framesToRead = this.dftSize;
|
|
667
|
+
if (leftFramePos < 0) {
|
|
668
|
+
//framesToRead=this.dftSize+leftFramePos;
|
|
669
|
+
leftFramePos = 0;
|
|
670
|
+
}
|
|
671
|
+
let read = this._audioDataHolder.frames(leftFramePos, framesToRead, arrAbBuf);
|
|
672
|
+
let ad = new Float32Array(chs * framesToRead);
|
|
673
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
674
|
+
ad.set(arrAbBuf[ch], ch * framesToRead);
|
|
675
|
+
}
|
|
676
|
+
this.worker.postMessage({
|
|
677
|
+
l: renderPos,
|
|
678
|
+
w: rw,
|
|
679
|
+
h: h,
|
|
680
|
+
vw: vw,
|
|
681
|
+
chs: chs,
|
|
682
|
+
frameLength: frameLength,
|
|
683
|
+
audioData: ad,
|
|
684
|
+
audioDataOffset: leftFramePos,
|
|
685
|
+
dftSize: this.dftSize,
|
|
686
|
+
norender: norender,
|
|
687
|
+
terminate: false
|
|
688
|
+
}, [ad.buffer]);
|
|
689
|
+
}
|
|
501
690
|
}
|
|
502
691
|
}
|
|
503
|
-
this.worker.postMessage({
|
|
504
|
-
audioData: ada,
|
|
505
|
-
l: Math.round(this.bounds.position.left),
|
|
506
|
-
w: w,
|
|
507
|
-
h: h,
|
|
508
|
-
vw: Math.round(this.virtualDimension.width),
|
|
509
|
-
chs: chs,
|
|
510
|
-
frameLength: frameLength,
|
|
511
|
-
dftSize: this.dftSize
|
|
512
|
-
}, ada);
|
|
513
692
|
}
|
|
514
693
|
else {
|
|
515
694
|
let g = this.sonagramCanvas.getContext("2d");
|
|
@@ -519,17 +698,17 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
519
698
|
}
|
|
520
699
|
}
|
|
521
700
|
}
|
|
522
|
-
drawRendered(
|
|
701
|
+
drawRendered(w, h, imgData) {
|
|
523
702
|
if (this.sonagramCanvas) {
|
|
524
|
-
this.sonagramCanvas.width =
|
|
525
|
-
this.sonagramCanvas.height =
|
|
703
|
+
this.sonagramCanvas.width = w;
|
|
704
|
+
this.sonagramCanvas.height = h;
|
|
526
705
|
let g = this.sonagramCanvas.getContext("2d");
|
|
527
706
|
if (g) {
|
|
528
|
-
let imgDataArr =
|
|
529
|
-
if (
|
|
530
|
-
let
|
|
531
|
-
|
|
532
|
-
g.putImageData(
|
|
707
|
+
let imgDataArr = imgData;
|
|
708
|
+
if (w > 0 && h > 0) {
|
|
709
|
+
let gImgData = g.createImageData(w, h);
|
|
710
|
+
gImgData.data.set(imgDataArr);
|
|
711
|
+
g.putImageData(gImgData, 0, 0);
|
|
533
712
|
}
|
|
534
713
|
}
|
|
535
714
|
}
|
|
@@ -545,91 +724,95 @@ export class Sonagram extends AudioCanvasLayerComponent {
|
|
|
545
724
|
g.clearRect(0, 0, w, h);
|
|
546
725
|
g.fillStyle = "white";
|
|
547
726
|
g.fillRect(0, 0, w, h);
|
|
548
|
-
if (this.
|
|
727
|
+
if (this._audioDataHolder) {
|
|
549
728
|
let spectSize = Math.floor(this.dftSize / 2);
|
|
550
|
-
let chs = this.
|
|
729
|
+
let chs = this._audioDataHolder.numberOfChannels;
|
|
551
730
|
let chH = h / chs;
|
|
552
|
-
let frameLength = this.
|
|
731
|
+
let frameLength = this._audioDataHolder.frameLen;
|
|
553
732
|
let framesPerPixel = frameLength / w;
|
|
554
733
|
let y = 0;
|
|
555
|
-
|
|
556
|
-
let
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
let
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
framePos =
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
sona[ch][pii] = spectr;
|
|
575
|
-
// @ts-ignore
|
|
576
|
-
let pMax = Math.max.apply(null, spectr);
|
|
577
|
-
if (pMax > max) {
|
|
578
|
-
max = pMax;
|
|
579
|
-
}
|
|
580
|
-
for (let s = 0; s < spectSize; s++) {
|
|
581
|
-
let psd = (2 * Math.pow(spectr[s], 2)) / spectSize;
|
|
582
|
-
if (psd > maxPsd) {
|
|
583
|
-
maxPsd = psd;
|
|
734
|
+
let audioBuffer = this._audioDataHolder.buffer;
|
|
735
|
+
let arrayAudioBuffer = this._audioDataHolder.arrayBuffer;
|
|
736
|
+
if (audioBuffer) {
|
|
737
|
+
let b = new Float32Array(this.dftSize);
|
|
738
|
+
let sona = new Array(chs);
|
|
739
|
+
let max = 0;
|
|
740
|
+
let maxPsd = -Infinity;
|
|
741
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
742
|
+
let x = 0;
|
|
743
|
+
sona[ch] = new Array(w);
|
|
744
|
+
let chData = audioBuffer.getChannelData(ch);
|
|
745
|
+
// TODO center buffer
|
|
746
|
+
let framePos = 0;
|
|
747
|
+
for (let pii = 0; pii < w; pii++) {
|
|
748
|
+
framePos = Math.round(pii * framesPerPixel);
|
|
749
|
+
// calculate DFT at pixel position
|
|
750
|
+
for (let i = 0; i < this.dftSize; i++) {
|
|
751
|
+
let chDat = chData[framePos + i];
|
|
752
|
+
b[i] = chDat;
|
|
584
753
|
}
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
let framePos = 0;
|
|
592
|
-
for (let pii = 0; pii < w; pii++) {
|
|
593
|
-
framePos = pii * framesPerPixel;
|
|
594
|
-
for (let y = 0; y < h; y++) {
|
|
595
|
-
let freqIdx = Math.round(y * spectSize / h);
|
|
596
|
-
// calculate the one sided power spectral density PSD (f, t) in Pa2/Hz
|
|
597
|
-
// PSD(f) proportional to 2|X(f)|2 / (t2 - t1)
|
|
598
|
-
let val = sona[ch][pii][freqIdx];
|
|
599
|
-
let psd = (2 * Math.pow(val, 2)) / spectSize;
|
|
600
|
-
// Calculate logarithmic
|
|
601
|
-
let psdLog = DSPUtils.toLevelInDB(psd / maxPsd);
|
|
602
|
-
let dynRangeInDb = 70;
|
|
603
|
-
let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;
|
|
604
|
-
if (scaledVal > 1)
|
|
605
|
-
scaledVal = 1;
|
|
606
|
-
if (scaledVal < 0) {
|
|
607
|
-
scaledVal = 0;
|
|
754
|
+
let spectr = this.dft.processRealMagnitude(b);
|
|
755
|
+
sona[ch][pii] = spectr;
|
|
756
|
+
// @ts-ignore
|
|
757
|
+
let pMax = Math.max.apply(null, spectr);
|
|
758
|
+
if (pMax > max) {
|
|
759
|
+
max = pMax;
|
|
608
760
|
}
|
|
609
|
-
let
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
761
|
+
for (let s = 0; s < spectSize; s++) {
|
|
762
|
+
let psd = (2 * Math.pow(spectr[s], 2)) / spectSize;
|
|
763
|
+
if (psd > maxPsd) {
|
|
764
|
+
maxPsd = psd;
|
|
765
|
+
}
|
|
613
766
|
}
|
|
614
|
-
|
|
615
|
-
|
|
767
|
+
}
|
|
768
|
+
}
|
|
769
|
+
//console.log("max: ", max);
|
|
770
|
+
maxPsd = (2 * Math.pow(max, 2)) / spectSize;
|
|
771
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
772
|
+
let framePos = 0;
|
|
773
|
+
for (let pii = 0; pii < w; pii++) {
|
|
774
|
+
framePos = pii * framesPerPixel;
|
|
775
|
+
for (let y = 0; y < h; y++) {
|
|
776
|
+
let freqIdx = Math.round(y * spectSize / h);
|
|
777
|
+
// calculate the one sided power spectral density PSD (f, t) in Pa2/Hz
|
|
778
|
+
// PSD(f) proportional to 2|X(f)|2 / (t2 - t1)
|
|
779
|
+
let val = sona[ch][pii][freqIdx];
|
|
780
|
+
let psd = (2 * Math.pow(val, 2)) / spectSize;
|
|
781
|
+
// Calculate logarithmic
|
|
782
|
+
let psdLog = DSPUtils.toLevelInDB(psd / maxPsd);
|
|
783
|
+
let dynRangeInDb = 70;
|
|
784
|
+
let scaledVal = (psdLog + dynRangeInDb) / dynRangeInDb;
|
|
785
|
+
if (scaledVal > 1)
|
|
786
|
+
scaledVal = 1;
|
|
787
|
+
if (scaledVal < 0) {
|
|
788
|
+
scaledVal = 0;
|
|
789
|
+
}
|
|
790
|
+
let rgbVal = (255 * scaledVal);
|
|
791
|
+
if (rgbVal < 0) {
|
|
792
|
+
// System.out.println("Neg RGB val: "+rgbVal);
|
|
793
|
+
rgbVal = 0;
|
|
794
|
+
}
|
|
795
|
+
if (rgbVal > 255) {
|
|
796
|
+
rgbVal = 255;
|
|
797
|
+
}
|
|
798
|
+
rgbVal = 255 - rgbVal;
|
|
799
|
+
let colorStr = CSSUtils.toColorString(rgbVal, rgbVal, rgbVal);
|
|
800
|
+
g.fillStyle = colorStr;
|
|
801
|
+
g.fillRect(pii, chH - y, 1, 1);
|
|
616
802
|
}
|
|
617
|
-
rgbVal = 255 - rgbVal;
|
|
618
|
-
let colorStr = CSSUtils.toColorString(rgbVal, rgbVal, rgbVal);
|
|
619
|
-
g.fillStyle = colorStr;
|
|
620
|
-
g.fillRect(pii, chH - y, 1, 1);
|
|
621
803
|
}
|
|
622
804
|
}
|
|
805
|
+
this.drawPlayPosition();
|
|
806
|
+
}
|
|
807
|
+
else if (arrayAudioBuffer) {
|
|
808
|
+
throw Error("Redraw with array audio buffer not supported.");
|
|
623
809
|
}
|
|
624
|
-
this.drawPlayPosition();
|
|
625
810
|
}
|
|
626
811
|
}
|
|
627
812
|
}
|
|
628
813
|
setData(audioData) {
|
|
629
|
-
this.
|
|
814
|
+
this._audioDataHolder = audioData;
|
|
630
815
|
this.playFramePosition = 0;
|
|
631
|
-
//this.redraw();
|
|
632
|
-
//this.startRender();
|
|
633
816
|
}
|
|
634
817
|
}
|
|
635
818
|
Sonagram.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: Sonagram, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
@@ -667,4 +850,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
667
850
|
type: ViewChild,
|
|
668
851
|
args: ['marker', { static: true }]
|
|
669
852
|
}] } });
|
|
670
|
-
//# 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 {DSPUtils} from '../../dsp/utils'\r\nimport {CSSUtils} from '../../utils/css_utils'\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\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 workerURL: string;\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._audioData = 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    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._audioData) {\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            var w = this.markerCanvas.width;\r\n            var h = this.markerCanvas.height;\r\n            var g = this.markerCanvas.getContext(\"2d\");\r\n            if (g) {\r\n\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    // layout() {\r\n    //\r\n    //\r\n    //   var offW = this.sonagramCanvas.offsetWidth;\r\n    //   var offH = this.sonagramCanvas.offsetHeight;\r\n    //   this.layoutBounds(0, 0, offW, offH, true);\r\n    // }\r\n\r\n\r\n    // layoutBounds(left: number, top: number, offW: number, offH: number, virtualWidth:number,redraw: boolean) {\r\n    //\r\n    //   const leftStr = left.toString() + 'px';\r\n    //   this.sonagramCanvas.style.left = leftStr;\r\n    //   const topStr = top.toString() + 'px';\r\n    //   this.sonagramCanvas.style.top = topStr;\r\n    //   this.cursorCanvas.style.top = topStr;\r\n    //   this.markerCanvas.style.top = topStr;\r\n    //\r\n    //   if (offW) {\r\n    //     const wStr = offW.toString() + 'px';\r\n    //     if (redraw) {\r\n    //\r\n    //       this.cursorCanvas.width = offW;\r\n    //       this.markerCanvas.width = offW;\r\n    //     }\r\n    //     this.sonagramCanvas.style.width = wStr;\r\n    //     this.cursorCanvas.style.width = wStr;\r\n    //     this.markerCanvas.style.width = wStr;\r\n    //   }\r\n    //   if (offH) {\r\n    //     const hStr = offH.toString() + 'px';\r\n    //     if (redraw) {\r\n    //       this.cursorCanvas.height = offH;\r\n    //       this.markerCanvas.height = offH;\r\n    //     }\r\n    //     this.sonagramCanvas.style.height = hStr;\r\n    //     this.cursorCanvas.style.height = hStr;\r\n    //     this.markerCanvas.style.height = hStr;\r\n    //\r\n    //   }\r\n    //   if (redraw) {\r\n    //     //this.redraw();\r\n    //     if (offW > 0) {\r\n    //       if (offH > 0) {\r\n    //         this.startDraw(left,top,offW, offH,virtualWidth);\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 n: number;\r\n            private m: number;\r\n\r\n            private cosLookup: Float32Array;\r\n            private 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 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\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 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            if (audioData && arrSize>0) {\r\n                let chH = Math.round(h / chs);\r\n                let framesPerPixel = frameLength / vw;\r\n                //console.log(\"Render: \", w, \"x\", h);\r\n\r\n                let b = new Float32Array(dftSize);\r\n                let sona = new Array(chs);\r\n\r\n                let maxPsd = -Infinity;\r\n                let p = 0;\r\n                for (let ch = 0; ch < chs; ch++) {\r\n                    p = ch * frameLength;\r\n                    let chDataLen=audioData[ch].length;\r\n                    let x = 0;\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                            if(samplePos>=0 && samplePos < chDataLen) {\r\n                              chDat = audioData[ch][samplePos];\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 > maxPsd) {\r\n                                maxPsd = 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                //maxPsd = (2 * Math.pow(max, 2)) / dftBands;\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                        }\r\n                        // if (allBlack) {\r\n                        //   console.log(\"Black: \", pii, \" \", ch);\r\n                        // }\r\n                    }\r\n                }\r\n            }\r\n            postMessage({imgData: imgData, l: l, w: msg.data.w, h: msg.data.h, vw: vw}, [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\r\n        if (this.worker) {\r\n            this.worker.terminate();\r\n            this.worker = null;\r\n\r\n        }\r\n        if (this.bounds) {\r\n            let w = Math.round(this.bounds.dimension.width);\r\n            let h = Math.round(this.bounds.dimension.height);\r\n\r\n\r\n            if (this._audioData && w>0 && h>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._audioData.numberOfChannels;\r\n\r\n                let frameLength = this._audioData.getChannelData(0).length;\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] = this._audioData.getChannelData(ch).buffer.slice(0);\r\n                }\r\n                let start = Date.now();\r\n                if (this.worker) {\r\n                    this.worker.onmessage = (me) => {\r\n                        this.drawRendered(me);\r\n                        if (this.worker) {\r\n                            this.worker.terminate();\r\n                        }\r\n                        this.worker = null;\r\n                    }\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: Math.round(this.bounds.position.left),\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                }, ada);\r\n            } else {\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(me: MessageEvent) {\r\n        if (this.sonagramCanvas) {\r\n\r\n            this.sonagramCanvas.width = me.data.w;\r\n            this.sonagramCanvas.height = me.data.h;\r\n            let g = this.sonagramCanvas.getContext(\"2d\");\r\n            if (g) {\r\n                let imgDataArr: Uint8ClampedArray = me.data.imgData;\r\n                if (me.data.w > 0 && me.data.h > 0) {\r\n                    let imgData = g.createImageData(me.data.w, me.data.h);\r\n                    imgData.data.set(imgDataArr);\r\n                    g.putImageData(imgData, 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._audioData) {\r\n                let spectSize = Math.floor(this.dftSize / 2)\r\n                let chs = this._audioData.numberOfChannels;\r\n                let chH = h / chs;\r\n\r\n                let frameLength = this._audioData.getChannelData(0).length;\r\n\r\n                let framesPerPixel = frameLength / w;\r\n                let y = 0;\r\n                // TODO\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 = this._audioData.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            }\r\n        }\r\n    }\r\n\r\n\r\n    setData(audioData: AudioBuffer | null) {\r\n        this._audioData = audioData;\r\n        this.playFramePosition = 0;\r\n        //this.redraw();\r\n        //this.startRender();\r\n    }\r\n\r\n\r\n}\r\n\r\n"]}
|
|
853
|
+
//# 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 {DSPUtils} from '../../dsp/utils'\r\nimport {CSSUtils} from '../../utils/css_utils'\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 {AudioDataHolder} from \"../audio_data_holder\";\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 workerURL: string;\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  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            var w = this.markerCanvas.width;\r\n            var h = this.markerCanvas.height;\r\n            var g = this.markerCanvas.getContext(\"2d\");\r\n            if (g) {\r\n\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    // layout() {\r\n    //\r\n    //\r\n    //   var offW = this.sonagramCanvas.offsetWidth;\r\n    //   var offH = this.sonagramCanvas.offsetHeight;\r\n    //   this.layoutBounds(0, 0, offW, offH, true);\r\n    // }\r\n\r\n\r\n    // layoutBounds(left: number, top: number, offW: number, offH: number, virtualWidth:number,redraw: boolean) {\r\n    //\r\n    //   const leftStr = left.toString() + 'px';\r\n    //   this.sonagramCanvas.style.left = leftStr;\r\n    //   const topStr = top.toString() + 'px';\r\n    //   this.sonagramCanvas.style.top = topStr;\r\n    //   this.cursorCanvas.style.top = topStr;\r\n    //   this.markerCanvas.style.top = topStr;\r\n    //\r\n    //   if (offW) {\r\n    //     const wStr = offW.toString() + 'px';\r\n    //     if (redraw) {\r\n    //\r\n    //       this.cursorCanvas.width = offW;\r\n    //       this.markerCanvas.width = offW;\r\n    //     }\r\n    //     this.sonagramCanvas.style.width = wStr;\r\n    //     this.cursorCanvas.style.width = wStr;\r\n    //     this.markerCanvas.style.width = wStr;\r\n    //   }\r\n    //   if (offH) {\r\n    //     const hStr = offH.toString() + 'px';\r\n    //     if (redraw) {\r\n    //       this.cursorCanvas.height = offH;\r\n    //       this.markerCanvas.height = offH;\r\n    //     }\r\n    //     this.sonagramCanvas.style.height = hStr;\r\n    //     this.cursorCanvas.style.height = hStr;\r\n    //     this.markerCanvas.style.height = hStr;\r\n    //\r\n    //   }\r\n    //   if (redraw) {\r\n    //     //this.redraw();\r\n    //     if (offW > 0) {\r\n    //       if (offH > 0) {\r\n    //         this.startDraw(left,top,offW, offH,virtualWidth);\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 n: number;\r\n            private m: number;\r\n\r\n            private cosLookup: Float32Array;\r\n            private 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 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\r\n                //let p = 0;\r\n                for (let ch = 0; ch < chs; ch++) {\r\n                    //p = ch * frameLength;\r\n                    let chDataLen=audioData[ch].length;\r\n                    let x = 0;\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\r\n        if (this.worker) {\r\n            this.worker.terminate();\r\n            this.worker = null;\r\n\r\n        }\r\n        if (this.bounds) {\r\n            let w = Math.round(this.bounds.dimension.width);\r\n            let h = Math.round(this.bounds.dimension.height);\r\n\r\n\r\n            if (this._audioDataHolder && w>0 && h>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 audioBuffer=this._audioDataHolder.buffer;\r\n                //let arrayAudioBuffer=this._audioDataHolder.arrayBuffer;\r\n              let arrAbBuf:Float32Array[]|null;\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                  }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\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                          }\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\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                  }\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\r\n\r\n                      let rw=1;\r\n                      //ais = new ArrayAudioBufferInputStream(arrayAudioBuffer);\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                        //framesToRead=this.dftSize+leftFramePos;\r\n                        leftFramePos=0;\r\n                      }\r\n                      let read=this._audioDataHolder.frames(leftFramePos,framesToRead,arrAbBuf);\r\n                      let ad=new Float32Array(chs*framesToRead);\r\n                      for (let ch = 0; ch < chs; ch++) {\r\n                        ad.set(arrAbBuf[ch],ch*framesToRead);\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: ad,\r\n                        audioDataOffset:leftFramePos,\r\n                        dftSize: this.dftSize,\r\n                        norender:norender,\r\n                        terminate:false\r\n                      }, [ad.buffer]);\r\n                    }\r\n                  }\r\n              }\r\n\r\n            } else {\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\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                let imgDataArr: Uint8ClampedArray = imgData;\r\n                if (w > 0 && h > 0) {\r\n                    let gImgData = g.createImageData(w, h);\r\n                    gImgData.data.set(imgDataArr);\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=this._audioDataHolder.buffer;\r\n        let arrayAudioBuffer=this._audioDataHolder.arrayBuffer;\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 if(arrayAudioBuffer){\r\n            throw Error(\"Redraw with array audio buffer not supported.\")\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\r\n"]}
|