speechrecorderng 3.0.0 → 3.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (142) hide show
  1. package/esm2020/lib/action/action.mjs +2 -1
  2. package/esm2020/lib/audio/array_audio_buffer.mjs +65 -2
  3. package/esm2020/lib/audio/array_audio_buffer_input_stream.mjs +2 -2
  4. package/esm2020/lib/audio/array_audio_buffer_random_access_stream.mjs +16 -0
  5. package/esm2020/lib/audio/audio_data_holder.mjs +203 -48
  6. package/esm2020/lib/audio/audio_display.mjs +10 -34
  7. package/esm2020/lib/audio/audio_player.mjs +18 -45
  8. package/esm2020/lib/audio/capture/capture.mjs +293 -69
  9. package/esm2020/lib/audio/dsp/level_measure.mjs +211 -88
  10. package/esm2020/lib/audio/impl/wavformat.mjs +1 -1
  11. package/esm2020/lib/audio/impl/wavreader.mjs +134 -0
  12. package/esm2020/lib/audio/impl/wavwriter.mjs +10 -9
  13. package/esm2020/lib/audio/inddb_audio_buffer.mjs +508 -0
  14. package/esm2020/lib/audio/net_audio_buffer.mjs +297 -0
  15. package/esm2020/lib/audio/persistor.mjs +8 -2
  16. package/esm2020/lib/audio/playback/array_audio_buffer_source_node.mjs +15 -154
  17. package/esm2020/lib/audio/playback/audio_source_node.mjs +18 -0
  18. package/esm2020/lib/audio/playback/audio_source_worklet_module_loader.mjs +167 -0
  19. package/esm2020/lib/audio/playback/inddb_audio_buffer_source_node.mjs +167 -0
  20. package/esm2020/lib/audio/playback/net_audio_buffer_source_node.mjs +218 -0
  21. package/esm2020/lib/audio/playback/player.mjs +190 -171
  22. package/esm2020/lib/audio/ui/audio_canvas_layer_comp.mjs +35 -76
  23. package/esm2020/lib/audio/ui/audio_display_control.mjs +12 -24
  24. package/esm2020/lib/audio/ui/audio_display_scroll_pane.mjs +14 -45
  25. package/esm2020/lib/audio/ui/audiosignal.mjs +333 -267
  26. package/esm2020/lib/audio/ui/container.mjs +40 -57
  27. package/esm2020/lib/audio/ui/livelevel.mjs +53 -52
  28. package/esm2020/lib/audio/ui/scroll_pane_horizontal.mjs +5 -19
  29. package/esm2020/lib/audio/ui/sonagram.mjs +386 -339
  30. package/esm2020/lib/db/inddb.mjs +120 -0
  31. package/esm2020/lib/io/BinaryReader.mjs +85 -0
  32. package/esm2020/lib/io/stream.mjs +101 -1
  33. package/esm2020/lib/net/uploader.mjs +111 -5
  34. package/esm2020/lib/recorder_component.mjs +59 -1
  35. package/esm2020/lib/speechrecorder/project/project.mjs +10 -1
  36. package/esm2020/lib/speechrecorder/project/project.service.mjs +3 -3
  37. package/esm2020/lib/speechrecorder/recording.mjs +30 -6
  38. package/esm2020/lib/speechrecorder/recordings/basic_recording.service.mjs +213 -0
  39. package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +732 -65
  40. package/esm2020/lib/speechrecorder/script/script.service.mjs +3 -3
  41. package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +358 -184
  42. package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +181 -28
  43. package/esm2020/lib/speechrecorder/session/controlpanel.mjs +47 -129
  44. package/esm2020/lib/speechrecorder/session/item.mjs +13 -1
  45. package/esm2020/lib/speechrecorder/session/progress.mjs +26 -55
  46. package/esm2020/lib/speechrecorder/session/prompting.mjs +33 -176
  47. package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +6 -6
  48. package/esm2020/lib/speechrecorder/session/recording_file_cache.mjs +28 -15
  49. package/esm2020/lib/speechrecorder/session/recording_list.mjs +92 -55
  50. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +9 -13
  51. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +9 -18
  52. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +10 -36
  53. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +10 -37
  54. package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +80 -56
  55. package/esm2020/lib/speechrecorder/session/session.service.mjs +3 -3
  56. package/esm2020/lib/speechrecorder/session/session_finished_dialog.mjs +4 -4
  57. package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +409 -165
  58. package/esm2020/lib/speechrecorder/session/warning_bar.mjs +6 -29
  59. package/esm2020/lib/speechrecorder/spruploader.mjs +3 -3
  60. package/esm2020/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +6 -41
  61. package/esm2020/lib/speechrecorderng.component.mjs +42 -31
  62. package/esm2020/lib/speechrecorderng.module.mjs +5 -5
  63. package/esm2020/lib/spr.config.mjs +3 -3
  64. package/esm2020/lib/spr.module.version.mjs +2 -2
  65. package/esm2020/lib/ui/canvas_layer_comp.mjs +3 -3
  66. package/esm2020/lib/ui/message_dialog.mjs +7 -7
  67. package/esm2020/lib/ui/recordingitem_display.mjs +26 -59
  68. package/esm2020/lib/utils/scrollIntoViewToBottom.mjs +3 -3
  69. package/esm2020/lib/utils/ua-parser.mjs +28 -4
  70. package/esm2020/lib/utils/utils.mjs +29 -1
  71. package/fesm2015/speechrecorderng.mjs +12755 -9195
  72. package/fesm2015/speechrecorderng.mjs.map +1 -1
  73. package/fesm2020/speechrecorderng.mjs +12652 -9101
  74. package/fesm2020/speechrecorderng.mjs.map +1 -1
  75. package/{speechrecorderng.d.ts → index.d.ts} +0 -0
  76. package/lib/audio/array_audio_buffer.d.ts +17 -1
  77. package/lib/audio/array_audio_buffer_random_access_stream.d.ts +9 -0
  78. package/lib/audio/audio_data_holder.d.ts +69 -14
  79. package/lib/audio/audio_display.d.ts +1 -1
  80. package/lib/audio/audio_player.d.ts +3 -7
  81. package/lib/audio/capture/capture.d.ts +24 -4
  82. package/lib/audio/dsp/level_measure.d.ts +2 -23
  83. package/lib/audio/impl/wavformat.d.ts +3 -3
  84. package/lib/audio/impl/wavreader.d.ts +16 -0
  85. package/lib/audio/impl/wavwriter.d.ts +1 -4
  86. package/lib/audio/inddb_audio_buffer.d.ts +68 -0
  87. package/lib/audio/net_audio_buffer.d.ts +59 -0
  88. package/lib/audio/persistor.d.ts +3 -9
  89. package/lib/audio/playback/array_audio_buffer_source_node.d.ts +2 -8
  90. package/lib/audio/playback/audio_source_node.d.ts +10 -0
  91. package/lib/audio/playback/audio_source_worklet_module_loader.d.ts +4 -0
  92. package/lib/audio/playback/inddb_audio_buffer_source_node.d.ts +21 -0
  93. package/lib/audio/playback/net_audio_buffer_source_node.d.ts +27 -0
  94. package/lib/audio/playback/player.d.ts +19 -16
  95. package/lib/audio/ui/audio_canvas_layer_comp.d.ts +3 -3
  96. package/lib/audio/ui/audio_display_control.d.ts +3 -6
  97. package/lib/audio/ui/audio_display_scroll_pane.d.ts +1 -1
  98. package/lib/audio/ui/audiosignal.d.ts +4 -3
  99. package/lib/audio/ui/container.d.ts +1 -1
  100. package/lib/audio/ui/livelevel.d.ts +11 -4
  101. package/lib/audio/ui/scroll_pane_horizontal.d.ts +1 -1
  102. package/lib/audio/ui/sonagram.d.ts +4 -3
  103. package/lib/db/inddb.d.ts +21 -0
  104. package/lib/io/BinaryReader.d.ts +18 -0
  105. package/lib/io/stream.d.ts +18 -0
  106. package/lib/net/uploader.d.ts +20 -2
  107. package/lib/recorder_component.d.ts +5 -0
  108. package/lib/speechrecorder/project/project.d.ts +9 -0
  109. package/lib/speechrecorder/recording.d.ts +8 -2
  110. package/lib/speechrecorder/recordings/basic_recording.service.d.ts +27 -0
  111. package/lib/speechrecorder/recordings/recordings.service.d.ts +21 -6
  112. package/lib/speechrecorder/session/audiorecorder.d.ts +7 -18
  113. package/lib/speechrecorder/session/basicrecorder.d.ts +28 -4
  114. package/lib/speechrecorder/session/controlpanel.d.ts +7 -7
  115. package/lib/speechrecorder/session/item.d.ts +1 -0
  116. package/lib/speechrecorder/session/progress.d.ts +2 -1
  117. package/lib/speechrecorder/session/prompting.d.ts +5 -5
  118. package/lib/speechrecorder/session/recorder_combi_pane.d.ts +1 -1
  119. package/lib/speechrecorder/session/recording_file_cache.d.ts +7 -4
  120. package/lib/speechrecorder/session/recording_list.d.ts +2 -1
  121. package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +1 -1
  122. package/lib/speechrecorder/session/recordingfile/recording-file-navi.component.d.ts +1 -1
  123. package/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.d.ts +1 -1
  124. package/lib/speechrecorder/session/recordingfile/recording-file-view.component.d.ts +1 -1
  125. package/lib/speechrecorder/session/recordingfile/recordingfile-service.d.ts +4 -5
  126. package/lib/speechrecorder/session/session_finished_dialog.d.ts +1 -1
  127. package/lib/speechrecorder/session/sessionmanager.d.ts +10 -10
  128. package/lib/speechrecorder/session/warning_bar.d.ts +1 -1
  129. package/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.d.ts +1 -1
  130. package/lib/speechrecorderng.component.d.ts +2 -1
  131. package/lib/spr.module.version.d.ts +1 -1
  132. package/lib/ui/canvas_layer_comp.d.ts +1 -1
  133. package/lib/ui/message_dialog.d.ts +1 -1
  134. package/lib/ui/recordingitem_display.d.ts +3 -2
  135. package/lib/utils/scrollIntoViewToBottom.d.ts +1 -1
  136. package/lib/utils/ua-parser.d.ts +4 -1
  137. package/lib/utils/utils.d.ts +6 -0
  138. package/package.json +5 -4
  139. package/esm2020/lib/math/utils.mjs +0 -14
  140. package/esm2020/lib/utils/css_utils.mjs +0 -7
  141. package/lib/math/utils.d.ts +0 -3
  142. package/lib/utils/css_utils.d.ts +0 -3
@@ -0,0 +1,167 @@
1
+ import { AsyncEditFloat32ArrayInputStream } from "../../io/stream";
2
+ import { IndexedDbAudioInputStream } from "../inddb_audio_buffer";
3
+ import { ArrayAudioBufferSourceNode } from "./array_audio_buffer_source_node";
4
+ import { EMPTY, expand, Observable } from "rxjs";
5
+ import { AudioSourceNode } from "./audio_source_node";
6
+ export class IndexedDbAudioBufferSourceNode extends AudioSourceNode {
7
+ constructor(context) {
8
+ super(context, 'audio-source-worklet');
9
+ this._bufferFillSeconds = AudioSourceNode.DEFAULT_BUFFER_FILL_SECONDS;
10
+ this.bufferFillFrames = 0;
11
+ this._streamReadFrameLen = AudioSourceNode.DEFAULT_STREAM_READ_FRAME_LEN * 8; // Much overhead fetching small buffers from indexed db, use larger buffer (8192).
12
+ this._inddbAudioBuffer = null;
13
+ this._audioInputStream = null;
14
+ this._aisBufs = null;
15
+ this._active = false;
16
+ this._endOfStream = false;
17
+ this.filledFrames = 0;
18
+ this.channelCountMode = 'explicit';
19
+ this.port.onmessage = (msgEv) => {
20
+ if (msgEv.data) {
21
+ let evType = msgEv.data.eventType;
22
+ if (evType) {
23
+ if ('bufferNotification' === evType) {
24
+ this.filledFrames = msgEv.data.filledFrames;
25
+ if (!this._endOfStream) {
26
+ this.fillBufferObs().subscribe();
27
+ }
28
+ }
29
+ else if ('ended' === evType) {
30
+ //console.debug("Inddb audio source ended playback.");
31
+ let drainTime = 0;
32
+ if (this._inddbAudioBuffer?.sampleRate) {
33
+ drainTime = ArrayAudioBufferSourceNode.QUANTUM_FRAME_LEN / this._inddbAudioBuffer.sampleRate;
34
+ }
35
+ //let dstAny:any=this.context.destination;
36
+ //console.debug('Destination node tail-time: '+dstAny['tail-time']);
37
+ window.setTimeout(() => {
38
+ this.onended?.call(this);
39
+ }, drainTime * 1000);
40
+ }
41
+ }
42
+ }
43
+ };
44
+ }
45
+ fillBufferObs(frameOffset) {
46
+ let obs = new Observable(subscriber => {
47
+ if (frameOffset) {
48
+ subscriber.error(new Error("Starting playback from position not equal zero not supported yet."));
49
+ }
50
+ else {
51
+ let filled = this.filledFrames;
52
+ let bufLen = 0;
53
+ if (this._audioInputStream && this._aisBufs) {
54
+ this._audioInputStream.readObs(this._aisBufs).pipe(expand((read) => {
55
+ if (read && this._aisBufs) {
56
+ let trBuffers = new Array(this.channelCount);
57
+ for (let ch = 0; ch < this.channelCount; ch++) {
58
+ let adCh = this._aisBufs[ch];
59
+ let adChCopy = new Float32Array(read);
60
+ bufLen = adChCopy.length;
61
+ if (read === adCh.length) {
62
+ adChCopy.set(adCh);
63
+ }
64
+ else {
65
+ // Note: slice() does not work here, since it returns a shallow copy.
66
+ for (let i = 0; i < read; i++) {
67
+ adChCopy[i] = adCh[i];
68
+ }
69
+ }
70
+ trBuffers[ch] = adChCopy.buffer;
71
+ }
72
+ this.port.postMessage({
73
+ cmd: 'data',
74
+ chs: this.channelCount,
75
+ audioData: trBuffers
76
+ }, trBuffers);
77
+ filled += read;
78
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Sent "+read+" frames to audio source worklet. Filled: "+filled+", to fill: "+this.bufferFillFrames);
79
+ if (this._audioInputStream && filled < this.bufferFillFrames) {
80
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Next inddb audio input stream readObs");
81
+ return this._audioInputStream.readObs(this._aisBufs);
82
+ }
83
+ else {
84
+ //console.debug("Return EMPTY");
85
+ return EMPTY;
86
+ }
87
+ }
88
+ else {
89
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Return EMPTY (read: "+read+")");
90
+ this._endOfStream = true;
91
+ this.port.postMessage({
92
+ cmd: 'endOfStream'
93
+ });
94
+ return EMPTY;
95
+ }
96
+ })).subscribe(subscriber);
97
+ }
98
+ }
99
+ });
100
+ return obs;
101
+ }
102
+ get inddbAudioBuffer() {
103
+ return this._inddbAudioBuffer;
104
+ }
105
+ set inddbAudioBuffer(value) {
106
+ this._inddbAudioBuffer = value;
107
+ if (this._inddbAudioBuffer?.channelCount) {
108
+ this.channelCount = this._inddbAudioBuffer?.channelCount;
109
+ this.bufferFillFrames = this._inddbAudioBuffer.sampleRate * this._bufferFillSeconds;
110
+ }
111
+ }
112
+ get bufferFillSeconds() {
113
+ return this._bufferFillSeconds;
114
+ }
115
+ set bufferFillSeconds(value) {
116
+ this._bufferFillSeconds = value;
117
+ }
118
+ start(when, offset, duration) {
119
+ if (when) {
120
+ throw Error("when parameter currently not supported by IndexedDbAudioBufferSourceNode class");
121
+ }
122
+ if (this._inddbAudioBuffer) {
123
+ let arrAis = new IndexedDbAudioInputStream(this._inddbAudioBuffer);
124
+ if (offset === undefined && duration === undefined) {
125
+ this._audioInputStream = arrAis;
126
+ }
127
+ else {
128
+ let offsetFrames = 0;
129
+ let durationFrames = undefined;
130
+ if (offset !== undefined) {
131
+ offsetFrames = Math.floor(offset * this._inddbAudioBuffer.sampleRate);
132
+ }
133
+ if (duration !== undefined) {
134
+ durationFrames = Math.floor(duration * this._inddbAudioBuffer.sampleRate);
135
+ }
136
+ //console.debug("Create AsyncEditFloat32ArrayInputStream: offsetFrames:"+offsetFrames+", durationFrames:"+durationFrames);
137
+ this._audioInputStream = new AsyncEditFloat32ArrayInputStream(arrAis, offsetFrames, durationFrames);
138
+ }
139
+ let chs = this._inddbAudioBuffer.channelCount;
140
+ this._aisBufs = new Array(chs);
141
+ for (let ch = 0; ch < chs; ch++) {
142
+ this._aisBufs[ch] = new Float32Array(this._streamReadFrameLen);
143
+ }
144
+ this.fillBufferObs().subscribe({
145
+ complete: () => {
146
+ //console.debug("IndexedDbAudioBufferSourceNode::start: Async play buffer fill completed. Sending start command to audio worklet.");
147
+ if (this._active) {
148
+ this.port.postMessage({ cmd: 'start' });
149
+ if (offset) {
150
+ this._playStartTime = this.context.currentTime - offset;
151
+ }
152
+ else {
153
+ this._playStartTime = this.context.currentTime;
154
+ }
155
+ }
156
+ }
157
+ });
158
+ this._active = true;
159
+ }
160
+ }
161
+ stop() {
162
+ this._active = false;
163
+ this.port.postMessage({ cmd: 'stop' });
164
+ this.onended?.call(this);
165
+ }
166
+ }
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inddb_audio_buffer_source_node.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/audio/playback/inddb_audio_buffer_source_node.ts"],"names":[],"mappings":"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","sourcesContent":["import {AsyncEditFloat32ArrayInputStream, AsyncFloat32ArrayInputStream} from \"../../io/stream\";\r\nimport {IndexedDbAudioBuffer, IndexedDbAudioInputStream} from \"../inddb_audio_buffer\";\r\nimport {ArrayAudioBufferSourceNode} from \"./array_audio_buffer_source_node\";\r\nimport {EMPTY, expand, map, Observable} from \"rxjs\";\r\nimport {AudioSourceNode} from \"./audio_source_node\";\r\n\r\nexport class IndexedDbAudioBufferSourceNode extends AudioSourceNode {\r\n\r\n  private _bufferFillSeconds = AudioSourceNode.DEFAULT_BUFFER_FILL_SECONDS;\r\n  private bufferFillFrames = 0;\r\n  private _streamReadFrameLen=AudioSourceNode.DEFAULT_STREAM_READ_FRAME_LEN * 8;  // Much overhead fetching small buffers from indexed db, use larger buffer (8192).\r\n  private _inddbAudioBuffer:IndexedDbAudioBuffer|null=null;\r\n  private _audioInputStream:AsyncFloat32ArrayInputStream|null=null;\r\n  private _aisBufs:Float32Array[]|null=null;\r\n  private _active=false;\r\n  private _endOfStream=false;\r\n  private filledFrames = 0;\r\n\r\n  constructor(context: AudioContext) {\r\n\r\n    super(context, 'audio-source-worklet');\r\n    this.channelCountMode = 'explicit';\r\n    this.port.onmessage = (msgEv: MessageEvent) => {\r\n      if (msgEv.data) {\r\n        let evType = msgEv.data.eventType;\r\n        if (evType) {\r\n          if ('bufferNotification' === evType) {\r\n            this.filledFrames = msgEv.data.filledFrames;\r\n            if(!this._endOfStream) {\r\n              this.fillBufferObs().subscribe();\r\n            }\r\n          } else if ('ended' === evType) {\r\n            //console.debug(\"Inddb audio source ended playback.\");\r\n            let drainTime = 0;\r\n            if (this._inddbAudioBuffer?.sampleRate) {\r\n              drainTime = ArrayAudioBufferSourceNode.QUANTUM_FRAME_LEN / this._inddbAudioBuffer.sampleRate;\r\n            }\r\n            //let dstAny:any=this.context.destination;\r\n            //console.debug('Destination node tail-time: '+dstAny['tail-time']);\r\n            window.setTimeout(() => {\r\n              this.onended?.call(this);\r\n            }, drainTime * 1000);\r\n\r\n          }\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  private fillBufferObs(frameOffset?:number):Observable<number|null> {\r\n\r\n      let obs = new Observable<number | null>(subscriber => {\r\n        if(frameOffset){\r\n          subscriber.error(new Error(\"Starting playback from position not equal zero not supported yet.\"));\r\n        }else {\r\n          let filled = this.filledFrames;\r\n          let bufLen = 0;\r\n          if (this._audioInputStream && this._aisBufs) {\r\n            this._audioInputStream.readObs(this._aisBufs).pipe(\r\n              expand((read) => {\r\n                  if (read && this._aisBufs) {\r\n                    let trBuffers = new Array<any>(this.channelCount);\r\n                    for (let ch = 0; ch < this.channelCount; ch++) {\r\n                      let adCh = this._aisBufs[ch];\r\n                      let adChCopy = new Float32Array(read);\r\n                      bufLen = adChCopy.length;\r\n                      if(read===adCh.length) {\r\n                        adChCopy.set(adCh);\r\n                      }else{\r\n                        // Note: slice() does not work here, since it returns a shallow copy.\r\n                        for(let i=0;i<read;i++){\r\n                          adChCopy[i]=adCh[i];\r\n                        }\r\n                      }\r\n                      trBuffers[ch] = adChCopy.buffer;\r\n                    }\r\n                    this.port.postMessage({\r\n                      cmd: 'data',\r\n                      chs: this.channelCount,\r\n                      audioData: trBuffers\r\n                    }, trBuffers);\r\n                    filled += read;\r\n                    //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Sent \"+read+\" frames to audio source worklet. Filled: \"+filled+\", to fill: \"+this.bufferFillFrames);\r\n                    if (this._audioInputStream && filled < this.bufferFillFrames) {\r\n                      //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Next inddb audio input stream readObs\");\r\n                      return this._audioInputStream.readObs(this._aisBufs);\r\n                    } else {\r\n                      //console.debug(\"Return EMPTY\");\r\n                      return EMPTY;\r\n                    }\r\n                  } else {\r\n                    //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Return EMPTY (read: \"+read+\")\");\r\n                    this._endOfStream=true;\r\n                    this.port.postMessage({\r\n                      cmd: 'endOfStream'\r\n                    });\r\n                    return EMPTY;\r\n                  }\r\n                }\r\n              )\r\n            ).subscribe(subscriber);\r\n          }\r\n        }\r\n      });\r\n      return obs;\r\n  }\r\n\r\n\r\n  get inddbAudioBuffer(): IndexedDbAudioBuffer | null {\r\n    return this._inddbAudioBuffer\r\n  }\r\n\r\n  set inddbAudioBuffer(value: IndexedDbAudioBuffer | null) {\r\n    this._inddbAudioBuffer = value;\r\n    if (this._inddbAudioBuffer?.channelCount) {\r\n      this.channelCount = this._inddbAudioBuffer?.channelCount;\r\n      this.bufferFillFrames = this._inddbAudioBuffer.sampleRate * this._bufferFillSeconds;\r\n    }\r\n  }\r\n\r\n  get bufferFillSeconds(): number {\r\n    return this._bufferFillSeconds;\r\n  }\r\n\r\n  set bufferFillSeconds(value: number) {\r\n    this._bufferFillSeconds = value;\r\n  }\r\n\r\n  start(when?: number | undefined,offset?: number | undefined,duration?: number | undefined): void {\r\n    if (when) {\r\n      throw Error(\"when parameter currently not supported by IndexedDbAudioBufferSourceNode class\")\r\n    }\r\n\r\n    if (this._inddbAudioBuffer) {\r\n      let arrAis=new IndexedDbAudioInputStream(this._inddbAudioBuffer);\r\n      if(offset===undefined && duration===undefined){\r\n        this._audioInputStream = arrAis;\r\n      }else{\r\n        let offsetFrames=0;\r\n        let durationFrames=undefined;\r\n        if(offset!==undefined) {\r\n          offsetFrames=Math.floor(offset * this._inddbAudioBuffer.sampleRate);\r\n        }\r\n        if(duration!==undefined){\r\n          durationFrames=Math.floor(duration * this._inddbAudioBuffer.sampleRate);\r\n        }\r\n        //console.debug(\"Create AsyncEditFloat32ArrayInputStream: offsetFrames:\"+offsetFrames+\", durationFrames:\"+durationFrames);\r\n        this._audioInputStream=new AsyncEditFloat32ArrayInputStream(arrAis,offsetFrames,durationFrames);\r\n      }\r\n\r\n      let chs=this._inddbAudioBuffer.channelCount;\r\n      this._aisBufs=new Array<Float32Array>(chs);\r\n      for(let ch=0;ch<chs;ch++){\r\n        this._aisBufs[ch]=new Float32Array(this._streamReadFrameLen);\r\n      }\r\n\r\n      this.fillBufferObs().subscribe({\r\n        complete: ()=>{\r\n          //console.debug(\"IndexedDbAudioBufferSourceNode::start: Async play buffer fill completed. Sending start command to audio worklet.\");\r\n          if(this._active) {\r\n            this.port.postMessage({cmd: 'start'});\r\n            if (offset) {\r\n              this._playStartTime = this.context.currentTime - offset;\r\n            } else {\r\n              this._playStartTime = this.context.currentTime;\r\n            }\r\n          }\r\n        }\r\n      })\r\n    this._active=true;\r\n    }\r\n  }\r\n\r\n  stop() {\r\n    this._active=false;\r\n    this.port.postMessage({cmd: 'stop'});\r\n    this.onended?.call(this);\r\n  }\r\n\r\n}\r\n\r\n"]}
@@ -0,0 +1,218 @@
1
+ import { AsyncEditFloat32ArrayInputStream } from "../../io/stream";
2
+ import { ArrayAudioBufferSourceNode } from "./array_audio_buffer_source_node";
3
+ import { EMPTY, expand, Observable } from "rxjs";
4
+ import { AudioSourceNode } from "./audio_source_node";
5
+ import { NetAudioInputStream } from "../net_audio_buffer";
6
+ export class NetAudioBufferSourceNode extends AudioSourceNode {
7
+ constructor(context) {
8
+ super(context, 'audio-source-worklet');
9
+ this._bufferFillSeconds = AudioSourceNode.DEFAULT_BUFFER_FILL_SECONDS;
10
+ this.bufferFillFrames = 0;
11
+ this._streamReadFrameLen = AudioSourceNode.DEFAULT_STREAM_READ_FRAME_LEN * 8; // Much overhead fetching small buffers from indexed db, use larger buffer (8192).
12
+ this._netAudioBuffer = null;
13
+ this._audioInputStream = null;
14
+ this._aisBufs = null;
15
+ this._active = false;
16
+ this.stalled = false;
17
+ this._endOfStream = false;
18
+ this.readDataSubscription = null;
19
+ this.filledFrames = 0;
20
+ this.stalledStartTime = null;
21
+ this.stalledTime = 0;
22
+ this.stopEndTime = null;
23
+ this.channelCountMode = 'explicit';
24
+ this.port.onmessage = (msgEv) => {
25
+ if (msgEv.data) {
26
+ let evType = msgEv.data.eventType;
27
+ if (evType) {
28
+ if ('bufferNotification' === evType) {
29
+ this.filledFrames = msgEv.data.filledFrames;
30
+ //console.debug("IndexedDbAudioBufferSourceNode: Buffer notification: filled frames: " + this.filledFrames);
31
+ if (!this._endOfStream) {
32
+ this.fillBufferObs().subscribe();
33
+ }
34
+ }
35
+ else if ('ended' === evType) {
36
+ //console.debug("Inddb audio source ended playback.");
37
+ let drainTime = 0;
38
+ if (this._netAudioBuffer?.sampleRate) {
39
+ drainTime = ArrayAudioBufferSourceNode.QUANTUM_FRAME_LEN / this._netAudioBuffer.sampleRate;
40
+ }
41
+ //let dstAny:any=this.context.destination;
42
+ //console.debug('Destination node tail-time: '+dstAny['tail-time']);
43
+ window.setTimeout(() => {
44
+ this.stopEndTime = this.context.currentTime;
45
+ this._active = false;
46
+ this.onended?.call(this);
47
+ }, drainTime * 1000);
48
+ }
49
+ else if ('stalled' === evType) {
50
+ console.debug('Playback stalled...');
51
+ this.stalled = true;
52
+ this.stalledStartTime = this.context.currentTime;
53
+ }
54
+ else if ('resumed' === evType) {
55
+ console.debug('Playback resumed after stall.');
56
+ this.stalled = false;
57
+ if (this.stalledStartTime != null) {
58
+ this.stalledTime += this.context.currentTime - this.stalledStartTime;
59
+ this.stalledStartTime = null;
60
+ }
61
+ }
62
+ }
63
+ }
64
+ };
65
+ }
66
+ fillBufferObs(frameOffset) {
67
+ return new Observable(subscriber => {
68
+ if (frameOffset) {
69
+ subscriber.error(new Error("Starting playback from position not equal zero not supported yet."));
70
+ }
71
+ else {
72
+ let filled = this.filledFrames;
73
+ let bufLen = 0;
74
+ if (this._audioInputStream && this._aisBufs && (this.readDataSubscription == null || this.readDataSubscription?.closed)) {
75
+ this.readDataSubscription = this._audioInputStream.readObs(this._aisBufs).pipe(expand((read) => {
76
+ if (read && this._aisBufs) {
77
+ let trBuffers = new Array(this.channelCount);
78
+ for (let ch = 0; ch < this.channelCount; ch++) {
79
+ let adCh = this._aisBufs[ch];
80
+ let adChCopy = new Float32Array(read);
81
+ bufLen = adChCopy.length;
82
+ if (read === adCh.length) {
83
+ adChCopy.set(adCh);
84
+ }
85
+ else {
86
+ // Note: slice() does not work here, since it returns a shallow copy.
87
+ for (let i = 0; i < read; i++) {
88
+ adChCopy[i] = adCh[i];
89
+ }
90
+ }
91
+ trBuffers[ch] = adChCopy.buffer;
92
+ }
93
+ this.port.postMessage({
94
+ cmd: 'data',
95
+ chs: this.channelCount,
96
+ audioData: trBuffers
97
+ }, trBuffers);
98
+ filled += read;
99
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Sent "+read+" frames to audio source worklet. Filled: "+filled+", to fill: "+this.bufferFillFrames);
100
+ if (this._audioInputStream && filled < this.bufferFillFrames) {
101
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Next inddb audio input stream readObs");
102
+ return this._audioInputStream.readObs(this._aisBufs);
103
+ }
104
+ else {
105
+ if (this.stalled) {
106
+ this.port.postMessage({
107
+ cmd: 'continue'
108
+ });
109
+ }
110
+ return EMPTY;
111
+ }
112
+ }
113
+ else {
114
+ //console.debug("IndexedDbAudioBufferSourceNode::fillBufferObs: Return EMPTY (read: "+read+")");
115
+ this._endOfStream = true;
116
+ this.port.postMessage({
117
+ cmd: 'endOfStream'
118
+ });
119
+ return EMPTY;
120
+ }
121
+ })).subscribe(subscriber);
122
+ }
123
+ }
124
+ });
125
+ }
126
+ get netAudioBuffer() {
127
+ return this._netAudioBuffer;
128
+ }
129
+ set netAudioBuffer(value) {
130
+ this._netAudioBuffer = value;
131
+ if (this._netAudioBuffer?.channelCount) {
132
+ this.channelCount = this._netAudioBuffer?.channelCount;
133
+ this.bufferFillFrames = this._netAudioBuffer.sampleRate * this._bufferFillSeconds;
134
+ }
135
+ }
136
+ get bufferFillSeconds() {
137
+ return this._bufferFillSeconds;
138
+ }
139
+ set bufferFillSeconds(value) {
140
+ this._bufferFillSeconds = value;
141
+ }
142
+ start(when, offset, duration) {
143
+ if (when) {
144
+ throw Error("when parameter currently not supported by IndexedDbAudioBufferSourceNode class");
145
+ }
146
+ this._playStartTime = null;
147
+ this.stopEndTime = null;
148
+ this.stalledTime = 0;
149
+ this.stalledStartTime = null;
150
+ if (this._netAudioBuffer) {
151
+ let arrAis = new NetAudioInputStream(this._netAudioBuffer);
152
+ if (offset === undefined && duration === undefined) {
153
+ this._audioInputStream = arrAis;
154
+ }
155
+ else {
156
+ let offsetFrames = 0;
157
+ let durationFrames = undefined;
158
+ if (offset !== undefined) {
159
+ offsetFrames = Math.floor(offset * this._netAudioBuffer.sampleRate);
160
+ }
161
+ if (duration !== undefined) {
162
+ durationFrames = Math.floor(duration * this._netAudioBuffer.sampleRate);
163
+ }
164
+ //console.error("Playback selection not supported yet!");
165
+ this._audioInputStream = new AsyncEditFloat32ArrayInputStream(arrAis, offsetFrames, durationFrames);
166
+ }
167
+ let chs = this._netAudioBuffer.channelCount;
168
+ this._aisBufs = new Array(chs);
169
+ for (let ch = 0; ch < chs; ch++) {
170
+ this._aisBufs[ch] = new Float32Array(this._streamReadFrameLen);
171
+ }
172
+ this.fillBufferObs().subscribe({
173
+ complete: () => {
174
+ //console.debug("IndexedDbAudioBufferSourceNode::start: Async play buffer fill completed. Sending start command to audio worklet.");
175
+ if (this._active) {
176
+ this.port.postMessage({ cmd: 'start' });
177
+ if (offset) {
178
+ this._playStartTime = this.context.currentTime - offset;
179
+ }
180
+ else {
181
+ this._playStartTime = this.context.currentTime;
182
+ }
183
+ }
184
+ }
185
+ });
186
+ this._active = true;
187
+ }
188
+ }
189
+ stop() {
190
+ this._active = false;
191
+ this.port.postMessage({ cmd: 'stop' });
192
+ this.onended?.call(this);
193
+ }
194
+ get playPositionTime() {
195
+ let ppt = null;
196
+ if (this._playStartTime !== null) {
197
+ if (this._active) {
198
+ if (this.stalledStartTime == null) {
199
+ ppt = this.context.currentTime - this._playStartTime - this.stalledTime;
200
+ }
201
+ else {
202
+ ppt = this.stalledStartTime - this._playStartTime - this.stalledTime;
203
+ }
204
+ }
205
+ else {
206
+ if (this.stopEndTime != null) {
207
+ // if(this._netAudioBuffer?.frameLen !=null && this._netAudioBuffer.sampleRate) {
208
+ // ppt = this._netAudioBuffer?.frameLen / this._netAudioBuffer?.sampleRate;
209
+ // }else{
210
+ ppt = this.stopEndTime - this._playStartTime - this.stalledTime;
211
+ //}
212
+ }
213
+ }
214
+ }
215
+ return ppt;
216
+ }
217
+ }
218
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"net_audio_buffer_source_node.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/audio/playback/net_audio_buffer_source_node.ts"],"names":[],"mappings":"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","sourcesContent":["import {AsyncEditFloat32ArrayInputStream, AsyncFloat32ArrayInputStream} from \"../../io/stream\";\r\nimport {ArrayAudioBufferSourceNode} from \"./array_audio_buffer_source_node\";\r\nimport {EMPTY, expand, Observable, Subscription} from \"rxjs\";\r\nimport {AudioSourceNode} from \"./audio_source_node\";\r\nimport {NetAudioBuffer, NetAudioInputStream} from \"../net_audio_buffer\";\r\n\r\nexport class NetAudioBufferSourceNode extends AudioSourceNode {\r\n\r\n  private _bufferFillSeconds = AudioSourceNode.DEFAULT_BUFFER_FILL_SECONDS;\r\n  private bufferFillFrames = 0;\r\n  private _streamReadFrameLen=AudioSourceNode.DEFAULT_STREAM_READ_FRAME_LEN * 8;  // Much overhead fetching small buffers from indexed db, use larger buffer (8192).\r\n  private _netAudioBuffer:NetAudioBuffer|null=null;\r\n  private _audioInputStream:AsyncFloat32ArrayInputStream|null=null;\r\n  private _aisBufs:Float32Array[]|null=null;\r\n  private _active=false;\r\n  private stalled=false;\r\n  private _endOfStream=false;\r\n  private readDataSubscription:Subscription|null=null;\r\n  private filledFrames = 0;\r\n\r\n  private stalledStartTime:number|null=null;\r\n  private stalledTime:number=0;\r\n\r\n  private stopEndTime:number|null=null;\r\n\r\n  constructor(context: AudioContext) {\r\n\r\n    super(context, 'audio-source-worklet');\r\n    this.channelCountMode = 'explicit';\r\n    this.port.onmessage = (msgEv: MessageEvent) => {\r\n      if (msgEv.data) {\r\n        let evType = msgEv.data.eventType;\r\n        if (evType) {\r\n          if ('bufferNotification' === evType) {\r\n            this.filledFrames = msgEv.data.filledFrames;\r\n            //console.debug(\"IndexedDbAudioBufferSourceNode: Buffer notification: filled frames: \" + this.filledFrames);\r\n            if(!this._endOfStream) {\r\n              this.fillBufferObs().subscribe();\r\n            }\r\n          } else if ('ended' === evType) {\r\n            //console.debug(\"Inddb audio source ended playback.\");\r\n            let drainTime = 0;\r\n            if (this._netAudioBuffer?.sampleRate) {\r\n              drainTime = ArrayAudioBufferSourceNode.QUANTUM_FRAME_LEN / this._netAudioBuffer.sampleRate;\r\n            }\r\n            //let dstAny:any=this.context.destination;\r\n            //console.debug('Destination node tail-time: '+dstAny['tail-time']);\r\n            window.setTimeout(() => {\r\n              this.stopEndTime=this.context.currentTime;\r\n              this._active=false;\r\n              this.onended?.call(this);\r\n            }, drainTime * 1000);\r\n\r\n          }else if ('stalled' === evType) {\r\n            console.debug('Playback stalled...');\r\n            this.stalled=true;\r\n            this.stalledStartTime=this.context.currentTime;\r\n          }else if ('resumed' === evType) {\r\n            console.debug('Playback resumed after stall.');\r\n            this.stalled=false;\r\n            if(this.stalledStartTime!=null) {\r\n              this.stalledTime += this.context.currentTime - this.stalledStartTime;\r\n              this.stalledStartTime = null;\r\n            }\r\n          }\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  private fillBufferObs(frameOffset?:number):Observable<number|null> {\r\n\r\n      return new Observable<number | null>(subscriber => {\r\n        if(frameOffset){\r\n          subscriber.error(new Error(\"Starting playback from position not equal zero not supported yet.\"));\r\n        }else {\r\n          let filled = this.filledFrames;\r\n          let bufLen = 0;\r\n          if (this._audioInputStream && this._aisBufs && (this.readDataSubscription==null || this.readDataSubscription?.closed)) {\r\n            this.readDataSubscription=this._audioInputStream.readObs(this._aisBufs).pipe(\r\n              expand((read) => {\r\n                  if (read && this._aisBufs) {\r\n                    let trBuffers = new Array<any>(this.channelCount);\r\n                    for (let ch = 0; ch < this.channelCount; ch++) {\r\n                      let adCh = this._aisBufs[ch];\r\n                      let adChCopy = new Float32Array(read);\r\n                      bufLen = adChCopy.length;\r\n                      if(read===adCh.length) {\r\n                        adChCopy.set(adCh);\r\n                      }else{\r\n                        // Note: slice() does not work here, since it returns a shallow copy.\r\n                        for(let i=0;i<read;i++){\r\n                          adChCopy[i]=adCh[i];\r\n                        }\r\n                      }\r\n                      trBuffers[ch] = adChCopy.buffer;\r\n                    }\r\n                    this.port.postMessage({\r\n                      cmd: 'data',\r\n                      chs: this.channelCount,\r\n                      audioData: trBuffers\r\n                    }, trBuffers);\r\n                    filled += read;\r\n                    //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Sent \"+read+\" frames to audio source worklet. Filled: \"+filled+\", to fill: \"+this.bufferFillFrames);\r\n                    if (this._audioInputStream && filled < this.bufferFillFrames) {\r\n                      //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Next inddb audio input stream readObs\");\r\n                      return this._audioInputStream.readObs(this._aisBufs);\r\n                    } else {\r\n                      if(this.stalled){\r\n                        this.port.postMessage({\r\n                          cmd: 'continue'\r\n                        });\r\n\r\n                      }\r\n                      return EMPTY;\r\n                    }\r\n                  } else {\r\n                    //console.debug(\"IndexedDbAudioBufferSourceNode::fillBufferObs: Return EMPTY (read: \"+read+\")\");\r\n                    this._endOfStream=true;\r\n                    this.port.postMessage({\r\n                      cmd: 'endOfStream'\r\n                    });\r\n                    return EMPTY;\r\n                  }\r\n                }\r\n              )\r\n            ).subscribe(subscriber);\r\n          }\r\n        }\r\n      });\r\n  }\r\n\r\n\r\n  get netAudioBuffer(): NetAudioBuffer | null {\r\n    return this._netAudioBuffer\r\n  }\r\n\r\n  set netAudioBuffer(value: NetAudioBuffer | null) {\r\n    this._netAudioBuffer = value;\r\n    if (this._netAudioBuffer?.channelCount) {\r\n      this.channelCount = this._netAudioBuffer?.channelCount;\r\n      this.bufferFillFrames = this._netAudioBuffer.sampleRate * this._bufferFillSeconds;\r\n    }\r\n  }\r\n\r\n  get bufferFillSeconds(): number {\r\n    return this._bufferFillSeconds;\r\n  }\r\n\r\n  set bufferFillSeconds(value: number) {\r\n    this._bufferFillSeconds = value;\r\n  }\r\n\r\n  start(when?: number | undefined,offset?: number | undefined,duration?: number | undefined): void {\r\n    if (when) {\r\n      throw Error(\"when parameter currently not supported by IndexedDbAudioBufferSourceNode class\")\r\n    }\r\n    this._playStartTime=null;\r\n    this.stopEndTime=null;\r\n    this.stalledTime=0;\r\n    this.stalledStartTime=null;\r\n\r\n    if (this._netAudioBuffer) {\r\n      let arrAis=new NetAudioInputStream(this._netAudioBuffer);\r\n      if(offset===undefined && duration===undefined){\r\n        this._audioInputStream = arrAis;\r\n      }else{\r\n        let offsetFrames=0;\r\n        let durationFrames=undefined;\r\n        if(offset!==undefined) {\r\n          offsetFrames=Math.floor(offset * this._netAudioBuffer.sampleRate);\r\n        }\r\n        if(duration!==undefined){\r\n          durationFrames=Math.floor(duration * this._netAudioBuffer.sampleRate);\r\n        }\r\n        //console.error(\"Playback selection not supported yet!\");\r\n        this._audioInputStream=new AsyncEditFloat32ArrayInputStream(arrAis,offsetFrames,durationFrames);\r\n      }\r\n\r\n      let chs=this._netAudioBuffer.channelCount;\r\n      this._aisBufs=new Array<Float32Array>(chs);\r\n      for(let ch=0;ch<chs;ch++){\r\n        this._aisBufs[ch]=new Float32Array(this._streamReadFrameLen);\r\n      }\r\n\r\n      this.fillBufferObs().subscribe({\r\n        complete: ()=>{\r\n          //console.debug(\"IndexedDbAudioBufferSourceNode::start: Async play buffer fill completed. Sending start command to audio worklet.\");\r\n          if(this._active) {\r\n            this.port.postMessage({cmd: 'start'});\r\n            if(offset) {\r\n              this._playStartTime = this.context.currentTime - offset;\r\n            }else{\r\n              this._playStartTime = this.context.currentTime;\r\n            }\r\n          }\r\n        }\r\n      })\r\n      this._active=true;\r\n\r\n    }\r\n  }\r\n\r\n  stop() {\r\n    this._active=false;\r\n    this.port.postMessage({cmd: 'stop'});\r\n    this.onended?.call(this);\r\n  }\r\n\r\n  get playPositionTime():number|null {\r\n    let ppt:number|null=null;\r\n    if(this._playStartTime!==null) {\r\n      if(this._active) {\r\n        if (this.stalledStartTime == null) {\r\n          ppt = this.context.currentTime - this._playStartTime - this.stalledTime;\r\n        } else {\r\n          ppt = this.stalledStartTime - this._playStartTime - this.stalledTime;\r\n        }\r\n      }else{\r\n        if(this.stopEndTime!=null){\r\n          // if(this._netAudioBuffer?.frameLen !=null && this._netAudioBuffer.sampleRate) {\r\n          //   ppt = this._netAudioBuffer?.frameLen / this._netAudioBuffer?.sampleRate;\r\n          // }else{\r\n            ppt = this.stopEndTime - this._playStartTime - this.stalledTime;\r\n          //}\r\n        }\r\n      }\r\n    }\r\n    return ppt;\r\n  }\r\n\r\n}\r\n\r\n"]}