stream-chat-angular 5.3.1 → 5.4.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 (91) hide show
  1. package/assets/i18n/en.d.ts +4 -0
  2. package/assets/version.d.ts +1 -1
  3. package/esm2020/assets/i18n/en.mjs +5 -1
  4. package/esm2020/assets/version.mjs +2 -2
  5. package/esm2020/lib/attachment-list/attachment-list.component.mjs +4 -4
  6. package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +5 -5
  7. package/esm2020/lib/attachment.service.mjs +44 -10
  8. package/esm2020/lib/channel-list/channel-list.component.mjs +5 -5
  9. package/esm2020/lib/channel-preview/channel-preview.component.mjs +1 -1
  10. package/esm2020/lib/file-utils.mjs +35 -0
  11. package/esm2020/lib/format-duration.mjs +16 -0
  12. package/esm2020/lib/icon/icon-placeholder/icon-placeholder.component.mjs +28 -0
  13. package/esm2020/lib/icon/icon.component.mjs +1 -1
  14. package/esm2020/lib/icon/icon.module.mjs +37 -0
  15. package/esm2020/lib/{loading-indicator → icon/loading-indicator}/loading-indicator.component.mjs +1 -1
  16. package/esm2020/lib/{loading-indicator-placeholder → icon/loading-indicator-placeholder}/loading-indicator-placeholder.component.mjs +2 -2
  17. package/esm2020/lib/is-safari.mjs +2 -0
  18. package/esm2020/lib/message/message.component.mjs +6 -6
  19. package/esm2020/lib/message-input/message-input-config.service.mjs +6 -1
  20. package/esm2020/lib/message-input/message-input.component.mjs +57 -14
  21. package/esm2020/lib/message-input/voice-recorder.service.mjs +27 -0
  22. package/esm2020/lib/message-list/message-list.component.mjs +9 -9
  23. package/esm2020/lib/modal/modal.component.mjs +1 -1
  24. package/esm2020/lib/paginated-list/paginated-list.component.mjs +1 -1
  25. package/esm2020/lib/stream-chat.module.mjs +21 -35
  26. package/esm2020/lib/thread/thread.component.mjs +1 -1
  27. package/esm2020/lib/types.mjs +1 -1
  28. package/esm2020/lib/voice-recorder/amplitude-recorder.service.mjs +119 -0
  29. package/esm2020/lib/voice-recorder/audio-recorder.service.mjs +79 -0
  30. package/esm2020/lib/voice-recorder/media-recorder.mjs +190 -0
  31. package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +61 -0
  32. package/esm2020/lib/voice-recorder/transcoder.service.mjs +121 -0
  33. package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +35 -0
  34. package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +80 -0
  35. package/esm2020/lib/voice-recorder/voice-recorder.module.mjs +34 -0
  36. package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +4 -75
  37. package/esm2020/lib/voice-recording/voice-recording.component.mjs +4 -15
  38. package/esm2020/lib/voice-recording/voice-recording.module.mjs +21 -0
  39. package/esm2020/lib/wave-form-sampler.mjs +72 -0
  40. package/esm2020/public-api.mjs +18 -5
  41. package/fesm2015/stream-chat-angular.mjs +1040 -146
  42. package/fesm2015/stream-chat-angular.mjs.map +1 -1
  43. package/fesm2020/stream-chat-angular.mjs +991 -141
  44. package/fesm2020/stream-chat-angular.mjs.map +1 -1
  45. package/lib/attachment.service.d.ts +7 -1
  46. package/lib/file-utils.d.ts +9 -0
  47. package/lib/format-duration.d.ts +1 -0
  48. package/lib/{icon-placeholder → icon/icon-placeholder}/icon-placeholder.component.d.ts +3 -3
  49. package/lib/icon/icon.component.d.ts +1 -1
  50. package/lib/icon/icon.module.d.ts +11 -0
  51. package/lib/{loading-indicator-placeholder → icon/loading-indicator-placeholder}/loading-indicator-placeholder.component.d.ts +1 -1
  52. package/lib/is-safari.d.ts +1 -0
  53. package/lib/message-input/message-input-config.service.d.ts +5 -0
  54. package/lib/message-input/message-input.component.d.ts +19 -5
  55. package/lib/message-input/voice-recorder.service.d.ts +19 -0
  56. package/lib/message-list/message-list.component.d.ts +0 -1
  57. package/lib/stream-chat.module.d.ts +20 -24
  58. package/lib/types.d.ts +11 -1
  59. package/lib/voice-recorder/amplitude-recorder.service.d.ts +71 -0
  60. package/lib/voice-recorder/audio-recorder.service.d.ts +46 -0
  61. package/lib/voice-recorder/media-recorder.d.ts +46 -0
  62. package/lib/voice-recorder/mp3-transcoder.d.ts +1 -0
  63. package/lib/voice-recorder/transcoder.service.d.ts +40 -0
  64. package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +21 -0
  65. package/lib/voice-recorder/voice-recorder.component.d.ts +30 -0
  66. package/lib/voice-recorder/voice-recorder.module.d.ts +12 -0
  67. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +0 -7
  68. package/lib/voice-recording/voice-recording.component.d.ts +0 -1
  69. package/lib/voice-recording/voice-recording.module.d.ts +11 -0
  70. package/lib/wave-form-sampler.d.ts +1 -0
  71. package/package.json +8 -1
  72. package/public-api.d.ts +17 -4
  73. package/src/assets/assets/icons/stream-chat-icons.eot +0 -0
  74. package/src/assets/assets/icons/stream-chat-icons.svg +4 -0
  75. package/src/assets/assets/icons/stream-chat-icons.ttf +0 -0
  76. package/src/assets/assets/icons/stream-chat-icons.woff +0 -0
  77. package/src/assets/assets/icons/stream-chat-icons.woff2 +0 -0
  78. package/src/assets/i18n/en.ts +6 -0
  79. package/src/assets/styles/css/index.css +1 -1
  80. package/src/assets/styles/css/index.layout.css +1 -1
  81. package/src/assets/styles/scss/AudioRecorder/AudioRecorder-layout.scss +64 -14
  82. package/src/assets/styles/scss/AudioRecorder/AudioRecorder-theme.scss +11 -1
  83. package/src/assets/styles/scss/ChannelList/ChannelList-layout.scss +4 -0
  84. package/src/assets/styles/scss/Icon/Icon-layout.scss +6 -1
  85. package/src/assets/styles/scss/MessageInput/MessageInput-layout.scss +1 -0
  86. package/src/assets/styles/scss/MessageInput/MessageInput-theme.scss +1 -0
  87. package/src/assets/version.ts +1 -1
  88. package/esm2020/lib/icon-placeholder/icon-placeholder.component.mjs +0 -28
  89. package/esm2020/lib/is-image-file.mjs +0 -5
  90. package/lib/is-image-file.d.ts +0 -1
  91. /package/lib/{loading-indicator → icon/loading-indicator}/loading-indicator.component.d.ts +0 -0
@@ -0,0 +1,34 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { VoiceRecorderComponent } from './voice-recorder.component';
4
+ import { VoiceRecordingModule } from '../voice-recording/voice-recording.module';
5
+ import { IconModule } from '../icon/icon.module';
6
+ import { TranslateModule } from '@ngx-translate/core';
7
+ import { AudioRecorderService } from './audio-recorder.service';
8
+ import { TranscoderService } from './transcoder.service';
9
+ import { AmplitudeRecorderService } from './amplitude-recorder.service';
10
+ import { VoiceRecorderWavebarComponent } from './voice-recorder-wavebar/voice-recorder-wavebar.component';
11
+ import * as i0 from "@angular/core";
12
+ export class VoiceRecorderModule {
13
+ }
14
+ VoiceRecorderModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
15
+ VoiceRecorderModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, declarations: [VoiceRecorderComponent, VoiceRecorderWavebarComponent], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule], exports: [VoiceRecorderComponent, VoiceRecorderWavebarComponent] });
16
+ VoiceRecorderModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, providers: [
17
+ AudioRecorderService,
18
+ TranscoderService,
19
+ AmplitudeRecorderService,
20
+ ], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule] });
21
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, decorators: [{
22
+ type: NgModule,
23
+ args: [{
24
+ declarations: [VoiceRecorderComponent, VoiceRecorderWavebarComponent],
25
+ imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule],
26
+ exports: [VoiceRecorderComponent, VoiceRecorderWavebarComponent],
27
+ providers: [
28
+ AudioRecorderService,
29
+ TranscoderService,
30
+ AmplitudeRecorderService,
31
+ ],
32
+ }]
33
+ }] });
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtcmVjb3JkZXIubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3ZvaWNlLXJlY29yZGVyL3ZvaWNlLXJlY29yZGVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwyQ0FBMkMsQ0FBQztBQUNqRixPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDakQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3RELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ2hFLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQ3pELE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQ3hFLE9BQU8sRUFBRSw2QkFBNkIsRUFBRSxNQUFNLDJEQUEyRCxDQUFDOztBQVkxRyxNQUFNLE9BQU8sbUJBQW1COztnSEFBbkIsbUJBQW1CO2lIQUFuQixtQkFBbUIsaUJBVGYsc0JBQXNCLEVBQUUsNkJBQTZCLGFBQzFELFlBQVksRUFBRSxvQkFBb0IsRUFBRSxVQUFVLEVBQUUsZUFBZSxhQUMvRCxzQkFBc0IsRUFBRSw2QkFBNkI7aUhBT3BELG1CQUFtQixhQU5uQjtRQUNULG9CQUFvQjtRQUNwQixpQkFBaUI7UUFDakIsd0JBQXdCO0tBQ3pCLFlBTlMsWUFBWSxFQUFFLG9CQUFvQixFQUFFLFVBQVUsRUFBRSxlQUFlOzJGQVE5RCxtQkFBbUI7a0JBVi9CLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFLENBQUMsc0JBQXNCLEVBQUUsNkJBQTZCLENBQUM7b0JBQ3JFLE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxvQkFBb0IsRUFBRSxVQUFVLEVBQUUsZUFBZSxDQUFDO29CQUMxRSxPQUFPLEVBQUUsQ0FBQyxzQkFBc0IsRUFBRSw2QkFBNkIsQ0FBQztvQkFDaEUsU0FBUyxFQUFFO3dCQUNULG9CQUFvQjt3QkFDcEIsaUJBQWlCO3dCQUNqQix3QkFBd0I7cUJBQ3pCO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBWb2ljZVJlY29yZGVyQ29tcG9uZW50IH0gZnJvbSAnLi92b2ljZS1yZWNvcmRlci5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VSZWNvcmRpbmdNb2R1bGUgfSBmcm9tICcuLi92b2ljZS1yZWNvcmRpbmcvdm9pY2UtcmVjb3JkaW5nLm1vZHVsZSc7XG5pbXBvcnQgeyBJY29uTW9kdWxlIH0gZnJvbSAnLi4vaWNvbi9pY29uLm1vZHVsZSc7XG5pbXBvcnQgeyBUcmFuc2xhdGVNb2R1bGUgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJztcbmltcG9ydCB7IEF1ZGlvUmVjb3JkZXJTZXJ2aWNlIH0gZnJvbSAnLi9hdWRpby1yZWNvcmRlci5zZXJ2aWNlJztcbmltcG9ydCB7IFRyYW5zY29kZXJTZXJ2aWNlIH0gZnJvbSAnLi90cmFuc2NvZGVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgQW1wbGl0dWRlUmVjb3JkZXJTZXJ2aWNlIH0gZnJvbSAnLi9hbXBsaXR1ZGUtcmVjb3JkZXIuc2VydmljZSc7XG5pbXBvcnQgeyBWb2ljZVJlY29yZGVyV2F2ZWJhckNvbXBvbmVudCB9IGZyb20gJy4vdm9pY2UtcmVjb3JkZXItd2F2ZWJhci92b2ljZS1yZWNvcmRlci13YXZlYmFyLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1ZvaWNlUmVjb3JkZXJDb21wb25lbnQsIFZvaWNlUmVjb3JkZXJXYXZlYmFyQ29tcG9uZW50XSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgVm9pY2VSZWNvcmRpbmdNb2R1bGUsIEljb25Nb2R1bGUsIFRyYW5zbGF0ZU1vZHVsZV0sXG4gIGV4cG9ydHM6IFtWb2ljZVJlY29yZGVyQ29tcG9uZW50LCBWb2ljZVJlY29yZGVyV2F2ZWJhckNvbXBvbmVudF0sXG4gIHByb3ZpZGVyczogW1xuICAgIEF1ZGlvUmVjb3JkZXJTZXJ2aWNlLFxuICAgIFRyYW5zY29kZXJTZXJ2aWNlLFxuICAgIEFtcGxpdHVkZVJlY29yZGVyU2VydmljZSxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgVm9pY2VSZWNvcmRlck1vZHVsZSB7fVxuIl19
@@ -1,4 +1,5 @@
1
1
  import { Component, Input, ViewChild, } from '@angular/core';
2
+ import { resampleWaveForm } from '../../wave-form-sampler';
2
3
  import * as i0 from "@angular/core";
3
4
  import * as i1 from "@angular/common";
4
5
  /**
@@ -17,36 +18,6 @@ export class VoiceRecordingWavebarComponent {
17
18
  this.isDragging = false;
18
19
  this.sampleSize = 40;
19
20
  this.isViewInited = false;
20
- this.upsample = () => {
21
- if (this.sampleSize === this.waveFormData.length)
22
- return this.waveFormData;
23
- // eslint-disable-next-line prefer-const
24
- let [bucketSize, remainder] = this.divMod(this.sampleSize, this.waveFormData.length);
25
- const result = [];
26
- for (let i = 0; i < this.waveFormData.length; i++) {
27
- const extra = remainder && remainder-- ? 1 : 0;
28
- result.push(...Array(bucketSize + extra).fill(this.waveFormData[i]));
29
- }
30
- return result;
31
- };
32
- this.getNextBucketMean = (data, currentBucketIndex, bucketSize) => {
33
- const nextBucketStartIndex = Math.floor(currentBucketIndex * bucketSize) + 1;
34
- let nextNextBucketStartIndex = Math.floor((currentBucketIndex + 1) * bucketSize) + 1;
35
- nextNextBucketStartIndex =
36
- nextNextBucketStartIndex < data.length
37
- ? nextNextBucketStartIndex
38
- : data.length;
39
- return this.mean(data.slice(nextBucketStartIndex, nextNextBucketStartIndex));
40
- };
41
- this.mean = (values) => values.reduce((acc, value) => acc + value, 0) / values.length;
42
- this.triangleAreaHeron = (a, b, c) => {
43
- const s = (a + b + c) / 2;
44
- return Math.sqrt(s * (s - a) * (s - b) * (s - c));
45
- };
46
- this.triangleBase = (a, b) => Math.sqrt(Math.pow(a, 2) + Math.pow(b, 2));
47
- this.divMod = (num, divisor) => {
48
- return [Math.floor(num / divisor), num % divisor];
49
- };
50
21
  }
51
22
  ngOnInit() {
52
23
  this.containerSizeChanged();
@@ -60,10 +31,7 @@ export class VoiceRecordingWavebarComponent {
60
31
  }
61
32
  ngOnChanges(changes) {
62
33
  if (changes.waveFormData) {
63
- this.resampledWaveFormData =
64
- this.waveFormData.length > this.sampleSize
65
- ? this.downsample()
66
- : this.upsample();
34
+ this.resampledWaveFormData = resampleWaveForm(this.waveFormData, this.sampleSize);
67
35
  }
68
36
  if (changes.audioElement) {
69
37
  this.ngZone.runOutsideAngular(() => {
@@ -117,10 +85,7 @@ export class VoiceRecordingWavebarComponent {
117
85
  sampleSize !== Infinity) {
118
86
  this.ngZone.run(() => {
119
87
  this.sampleSize = sampleSize;
120
- this.resampledWaveFormData =
121
- this.waveFormData.length > this.sampleSize
122
- ? this.downsample()
123
- : this.upsample();
88
+ this.resampledWaveFormData = resampleWaveForm(this.waveFormData, this.sampleSize);
124
89
  if (this.isViewInited) {
125
90
  this.cdRef.detectChanges();
126
91
  }
@@ -128,42 +93,6 @@ export class VoiceRecordingWavebarComponent {
128
93
  }
129
94
  }
130
95
  }
131
- downsample() {
132
- if (this.waveFormData.length <= this.sampleSize) {
133
- return this.waveFormData;
134
- }
135
- if (this.sampleSize === 1)
136
- return [this.mean(this.waveFormData)];
137
- const result = [];
138
- // bucket size adjusted due to the fact that the first and the last item in the original data array is kept in target output
139
- const bucketSize = (this.waveFormData.length - 2) / (this.sampleSize - 2);
140
- let lastSelectedPointIndex = 0;
141
- result.push(this.waveFormData[lastSelectedPointIndex]); // Always add the first point
142
- let maxAreaPoint, maxArea, triangleArea;
143
- for (let bucketIndex = 1; bucketIndex < this.sampleSize - 1; bucketIndex++) {
144
- const previousBucketRefPoint = this.waveFormData[lastSelectedPointIndex];
145
- const nextBucketMean = this.getNextBucketMean(this.waveFormData, bucketIndex, bucketSize);
146
- const currentBucketStartIndex = Math.floor((bucketIndex - 1) * bucketSize) + 1;
147
- const nextBucketStartIndex = Math.floor(bucketIndex * bucketSize) + 1;
148
- const countUnitsBetweenAtoC = 1 + nextBucketStartIndex - currentBucketStartIndex;
149
- maxArea = triangleArea = -1;
150
- for (let currentPointIndex = currentBucketStartIndex; currentPointIndex < nextBucketStartIndex; currentPointIndex++) {
151
- const countUnitsBetweenAtoB = Math.abs(currentPointIndex - currentBucketStartIndex) + 1;
152
- const countUnitsBetweenBtoC = countUnitsBetweenAtoC - countUnitsBetweenAtoB;
153
- const currentPointValue = this.waveFormData[currentPointIndex];
154
- triangleArea = this.triangleAreaHeron(this.triangleBase(Math.abs(previousBucketRefPoint - currentPointValue), countUnitsBetweenAtoB), this.triangleBase(Math.abs(currentPointValue - nextBucketMean), countUnitsBetweenBtoC), this.triangleBase(Math.abs(previousBucketRefPoint - nextBucketMean), countUnitsBetweenAtoC));
155
- if (triangleArea > maxArea) {
156
- maxArea = triangleArea;
157
- maxAreaPoint = this.waveFormData[currentPointIndex];
158
- lastSelectedPointIndex = currentPointIndex;
159
- }
160
- }
161
- if (typeof maxAreaPoint !== 'undefined')
162
- result.push(maxAreaPoint);
163
- }
164
- result.push(this.waveFormData[this.waveFormData.length - 1]); // Always add the last point
165
- return result;
166
- }
167
96
  }
168
97
  VoiceRecordingWavebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingWavebarComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
169
98
  VoiceRecordingWavebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: VoiceRecordingWavebarComponent, selector: "stream-voice-recording-wavebar", inputs: { audioElement: "audioElement", waveFormData: "waveFormData", duration: "duration" }, viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n #container\n class=\"str-chat__wave-progress-bar__track\"\n data-testid=\"wave-progress-bar-track\"\n role=\"progressbar\"\n (mousedown)=\"isDragging = true\"\n (mouseup)=\"isDragging = false\"\n (mouseleave)=\"isDragging = false\"\n (mousemove)=\"isDragging ? seek($event) : null\"\n (click)=\"seek($event)\"\n>\n <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n <div\n *ngFor=\"\n let dataPoint of resampledWaveFormData;\n let i = index;\n trackBy: trackByIndex\n \"\n class=\"str-chat__wave-progress-bar__amplitude-bar\"\n [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n progress > i / resampledWaveFormData.length\n \"\n [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n dataPoint ? dataPoint * 100 + '%' : '0%'\n \"\n ></div>\n <div\n class=\"str-chat__wave-progress-bar__progress-indicator\"\n data-testid=\"wave-progress-bar-progress-indicator\"\n [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n ></div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
@@ -180,4 +109,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
180
109
  type: ViewChild,
181
110
  args: ['container', { static: true }]
182
111
  }] } });
183
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recording-wavebar.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.ts","../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\n\n/**\n * This component can be used to visualize the wave bar of a voice recording\n */\n@Component({\n  selector: 'stream-voice-recording-wavebar',\n  templateUrl: './voice-recording-wavebar.component.html',\n  styles: [],\n})\nexport class VoiceRecordingWavebarComponent\n  implements OnInit, OnChanges, AfterViewInit\n{\n  /**\n   * The audio element that plays the voice recording\n   */\n  @Input() audioElement?: HTMLAudioElement;\n  /**\n   * The waveform data to visualize\n   */\n  @Input() waveFormData: number[] = [];\n  /**\n   * The duration of the voice recording in seconds\n   */\n  @Input() duration?: number;\n  resampledWaveFormData: number[] = [];\n  progress: number = 0;\n  isDragging = false;\n  private sampleSize: number = 40;\n  @ViewChild('container', { static: true })\n  private container?: ElementRef<HTMLElement>;\n  private isViewInited = false;\n\n  constructor(private ngZone: NgZone, private cdRef: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.containerSizeChanged();\n    if (this.container?.nativeElement) {\n      this.ngZone.runOutsideAngular(() => {\n        new ResizeObserver(() => {\n          this.containerSizeChanged();\n        }).observe(this.container!.nativeElement);\n      });\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.waveFormData) {\n      this.resampledWaveFormData =\n        this.waveFormData.length > this.sampleSize\n          ? this.downsample()\n          : this.upsample();\n    }\n    if (changes.audioElement) {\n      this.ngZone.runOutsideAngular(() => {\n        this.audioElement?.addEventListener('timeupdate', () => {\n          const progress =\n            (this.audioElement?.currentTime || 0) / (this.duration || 0) || 0;\n          if (Math.abs(progress - this.progress) >= 0.02) {\n            this.ngZone.run(() => {\n              this.progress = progress;\n              this.cdRef.detectChanges();\n            });\n          }\n        });\n      });\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n  }\n\n  seek(event: MouseEvent) {\n    const containerWidth =\n      this.container?.nativeElement?.getBoundingClientRect().width || 0;\n    const containerStart =\n      this.container?.nativeElement?.getBoundingClientRect()?.x || 0;\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n    const progress = (event.x - containerStart) / containerWidth;\n\n    if (!isNaN(progress) && this.audioElement) {\n      const duration = this.duration || 0;\n      const time = duration * progress;\n      this.audioElement.currentTime = time;\n    }\n  }\n\n  trackByIndex(index: number) {\n    return index;\n  }\n\n  private containerSizeChanged() {\n    if (!this.container?.nativeElement) {\n      return;\n    }\n    const containerWidth = this.container.nativeElement.clientWidth;\n    if (containerWidth === 0) {\n      return;\n    }\n    const barWidth = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-width')\n      .replace('px', '');\n    const barGap = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-gap-width')\n      .replace('px', '');\n    if (!isNaN(barWidth) && !isNaN(barGap)) {\n      const sampleSize = Math.floor(containerWidth / (barWidth + barGap));\n      if (\n        sampleSize !== this.sampleSize &&\n        !isNaN(sampleSize) &&\n        sampleSize !== Infinity\n      ) {\n        this.ngZone.run(() => {\n          this.sampleSize = sampleSize;\n          this.resampledWaveFormData =\n            this.waveFormData.length > this.sampleSize\n              ? this.downsample()\n              : this.upsample();\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        });\n      }\n    }\n  }\n\n  private downsample() {\n    if (this.waveFormData.length <= this.sampleSize) {\n      return this.waveFormData;\n    }\n\n    if (this.sampleSize === 1) return [this.mean(this.waveFormData)];\n\n    const result: number[] = [];\n    // bucket size adjusted due to the fact that the first and the last item in the original data array is kept in target output\n    const bucketSize = (this.waveFormData.length - 2) / (this.sampleSize - 2);\n    let lastSelectedPointIndex = 0;\n    result.push(this.waveFormData[lastSelectedPointIndex]); // Always add the first point\n    let maxAreaPoint, maxArea, triangleArea;\n\n    for (\n      let bucketIndex = 1;\n      bucketIndex < this.sampleSize - 1;\n      bucketIndex++\n    ) {\n      const previousBucketRefPoint = this.waveFormData[lastSelectedPointIndex];\n      const nextBucketMean = this.getNextBucketMean(\n        this.waveFormData,\n        bucketIndex,\n        bucketSize\n      );\n\n      const currentBucketStartIndex =\n        Math.floor((bucketIndex - 1) * bucketSize) + 1;\n      const nextBucketStartIndex = Math.floor(bucketIndex * bucketSize) + 1;\n      const countUnitsBetweenAtoC =\n        1 + nextBucketStartIndex - currentBucketStartIndex;\n\n      maxArea = triangleArea = -1;\n\n      for (\n        let currentPointIndex = currentBucketStartIndex;\n        currentPointIndex < nextBucketStartIndex;\n        currentPointIndex++\n      ) {\n        const countUnitsBetweenAtoB =\n          Math.abs(currentPointIndex - currentBucketStartIndex) + 1;\n        const countUnitsBetweenBtoC =\n          countUnitsBetweenAtoC - countUnitsBetweenAtoB;\n        const currentPointValue = this.waveFormData[currentPointIndex];\n\n        triangleArea = this.triangleAreaHeron(\n          this.triangleBase(\n            Math.abs(previousBucketRefPoint - currentPointValue),\n            countUnitsBetweenAtoB\n          ),\n          this.triangleBase(\n            Math.abs(currentPointValue - nextBucketMean),\n            countUnitsBetweenBtoC\n          ),\n          this.triangleBase(\n            Math.abs(previousBucketRefPoint - nextBucketMean),\n            countUnitsBetweenAtoC\n          )\n        );\n\n        if (triangleArea > maxArea) {\n          maxArea = triangleArea;\n          maxAreaPoint = this.waveFormData[currentPointIndex];\n          lastSelectedPointIndex = currentPointIndex;\n        }\n      }\n\n      if (typeof maxAreaPoint !== 'undefined') result.push(maxAreaPoint);\n    }\n\n    result.push(this.waveFormData[this.waveFormData.length - 1]); // Always add the last point\n\n    return result;\n  }\n\n  private upsample = () => {\n    if (this.sampleSize === this.waveFormData.length) return this.waveFormData;\n\n    // eslint-disable-next-line  prefer-const\n    let [bucketSize, remainder] = this.divMod(\n      this.sampleSize,\n      this.waveFormData.length\n    );\n    const result: number[] = [];\n\n    for (let i = 0; i < this.waveFormData.length; i++) {\n      const extra = remainder && remainder-- ? 1 : 0;\n      result.push(\n        ...Array<number>(bucketSize + extra).fill(this.waveFormData[i])\n      );\n    }\n    return result;\n  };\n\n  private getNextBucketMean = (\n    data: number[],\n    currentBucketIndex: number,\n    bucketSize: number\n  ) => {\n    const nextBucketStartIndex =\n      Math.floor(currentBucketIndex * bucketSize) + 1;\n    let nextNextBucketStartIndex =\n      Math.floor((currentBucketIndex + 1) * bucketSize) + 1;\n    nextNextBucketStartIndex =\n      nextNextBucketStartIndex < data.length\n        ? nextNextBucketStartIndex\n        : data.length;\n\n    return this.mean(\n      data.slice(nextBucketStartIndex, nextNextBucketStartIndex)\n    );\n  };\n\n  private mean = (values: number[]) =>\n    values.reduce((acc, value) => acc + value, 0) / values.length;\n\n  private triangleAreaHeron = (a: number, b: number, c: number) => {\n    const s = (a + b + c) / 2;\n    return Math.sqrt(s * (s - a) * (s - b) * (s - c));\n  };\n\n  private triangleBase = (a: number, b: number) =>\n    Math.sqrt(Math.pow(a, 2) + Math.pow(b, 2));\n\n  private divMod = (num: number, divisor: number) => {\n    return [Math.floor(num / divisor), num % divisor];\n  };\n}\n","<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n  #container\n  class=\"str-chat__wave-progress-bar__track\"\n  data-testid=\"wave-progress-bar-track\"\n  role=\"progressbar\"\n  (mousedown)=\"isDragging = true\"\n  (mouseup)=\"isDragging = false\"\n  (mouseleave)=\"isDragging = false\"\n  (mousemove)=\"isDragging ? seek($event) : null\"\n  (click)=\"seek($event)\"\n>\n  <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n  <div\n    *ngFor=\"\n      let dataPoint of resampledWaveFormData;\n      let i = index;\n      trackBy: trackByIndex\n    \"\n    class=\"str-chat__wave-progress-bar__amplitude-bar\"\n    [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n      progress > i / resampledWaveFormData.length\n    \"\n    [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n      dataPoint ? dataPoint * 100 + '%' : '0%'\n    \"\n  ></div>\n  <div\n    class=\"str-chat__wave-progress-bar__progress-indicator\"\n    data-testid=\"wave-progress-bar-progress-indicator\"\n    [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n  ></div>\n</div>\n"]}
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recording-wavebar.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.ts","../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { resampleWaveForm } from '../../wave-form-sampler';\n\n/**\n * This component can be used to visualize the wave bar of a voice recording\n */\n@Component({\n  selector: 'stream-voice-recording-wavebar',\n  templateUrl: './voice-recording-wavebar.component.html',\n  styles: [],\n})\nexport class VoiceRecordingWavebarComponent\n  implements OnInit, OnChanges, AfterViewInit\n{\n  /**\n   * The audio element that plays the voice recording\n   */\n  @Input() audioElement?: HTMLAudioElement;\n  /**\n   * The waveform data to visualize\n   */\n  @Input() waveFormData: number[] = [];\n  /**\n   * The duration of the voice recording in seconds\n   */\n  @Input() duration?: number;\n  resampledWaveFormData: number[] = [];\n  progress: number = 0;\n  isDragging = false;\n  private sampleSize: number = 40;\n  @ViewChild('container', { static: true })\n  private container?: ElementRef<HTMLElement>;\n  private isViewInited = false;\n\n  constructor(private ngZone: NgZone, private cdRef: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.containerSizeChanged();\n    if (this.container?.nativeElement) {\n      this.ngZone.runOutsideAngular(() => {\n        new ResizeObserver(() => {\n          this.containerSizeChanged();\n        }).observe(this.container!.nativeElement);\n      });\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.waveFormData) {\n      this.resampledWaveFormData = resampleWaveForm(\n        this.waveFormData,\n        this.sampleSize\n      );\n    }\n    if (changes.audioElement) {\n      this.ngZone.runOutsideAngular(() => {\n        this.audioElement?.addEventListener('timeupdate', () => {\n          const progress =\n            (this.audioElement?.currentTime || 0) / (this.duration || 0) || 0;\n          if (Math.abs(progress - this.progress) >= 0.02) {\n            this.ngZone.run(() => {\n              this.progress = progress;\n              this.cdRef.detectChanges();\n            });\n          }\n        });\n      });\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n  }\n\n  seek(event: MouseEvent) {\n    const containerWidth =\n      this.container?.nativeElement?.getBoundingClientRect().width || 0;\n    const containerStart =\n      this.container?.nativeElement?.getBoundingClientRect()?.x || 0;\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n    const progress = (event.x - containerStart) / containerWidth;\n\n    if (!isNaN(progress) && this.audioElement) {\n      const duration = this.duration || 0;\n      const time = duration * progress;\n      this.audioElement.currentTime = time;\n    }\n  }\n\n  trackByIndex(index: number) {\n    return index;\n  }\n\n  private containerSizeChanged() {\n    if (!this.container?.nativeElement) {\n      return;\n    }\n    const containerWidth = this.container.nativeElement.clientWidth;\n    if (containerWidth === 0) {\n      return;\n    }\n    const barWidth = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-width')\n      .replace('px', '');\n    const barGap = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-gap-width')\n      .replace('px', '');\n    if (!isNaN(barWidth) && !isNaN(barGap)) {\n      const sampleSize = Math.floor(containerWidth / (barWidth + barGap));\n      if (\n        sampleSize !== this.sampleSize &&\n        !isNaN(sampleSize) &&\n        sampleSize !== Infinity\n      ) {\n        this.ngZone.run(() => {\n          this.sampleSize = sampleSize;\n          this.resampledWaveFormData = resampleWaveForm(\n            this.waveFormData,\n            this.sampleSize\n          );\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        });\n      }\n    }\n  }\n}\n","<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n  #container\n  class=\"str-chat__wave-progress-bar__track\"\n  data-testid=\"wave-progress-bar-track\"\n  role=\"progressbar\"\n  (mousedown)=\"isDragging = true\"\n  (mouseup)=\"isDragging = false\"\n  (mouseleave)=\"isDragging = false\"\n  (mousemove)=\"isDragging ? seek($event) : null\"\n  (click)=\"seek($event)\"\n>\n  <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n  <div\n    *ngFor=\"\n      let dataPoint of resampledWaveFormData;\n      let i = index;\n      trackBy: trackByIndex\n    \"\n    class=\"str-chat__wave-progress-bar__amplitude-bar\"\n    [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n      progress > i / resampledWaveFormData.length\n    \"\n    [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n      dataPoint ? dataPoint * 100 + '%' : '0%'\n    \"\n  ></div>\n  <div\n    class=\"str-chat__wave-progress-bar__progress-indicator\"\n    data-testid=\"wave-progress-bar-progress-indicator\"\n    [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n  ></div>\n</div>\n"]}
@@ -1,8 +1,9 @@
1
1
  import { Component, Input, ViewChild, } from '@angular/core';
2
2
  import prettybytes from 'pretty-bytes';
3
+ import { formatDuration } from '../format-duration';
3
4
  import * as i0 from "@angular/core";
4
5
  import * as i1 from "@angular/common";
5
- import * as i2 from "../icon-placeholder/icon-placeholder.component";
6
+ import * as i2 from "../icon/icon-placeholder/icon-placeholder.component";
6
7
  import * as i3 from "./voice-recording-wavebar/voice-recording-wavebar.component";
7
8
  import * as i4 from "@ngx-translate/core";
8
9
  /**
@@ -66,16 +67,7 @@ export class VoiceRecordingComponent {
66
67
  this.audioElement.nativeElement.playbackRate = playbackRate;
67
68
  }
68
69
  getFormattedDuration(duration) {
69
- if (duration === undefined || duration <= 0)
70
- return '00:00';
71
- const [hours, hoursLeftover] = this.divMod(duration, 3600);
72
- const [minutes, seconds] = this.divMod(hoursLeftover, 60);
73
- const roundedSeconds = Math.ceil(seconds);
74
- const prependHrsZero = hours.toString().length === 1 ? '0' : '';
75
- const prependMinZero = minutes.toString().length === 1 ? '0' : '';
76
- const prependSecZero = roundedSeconds.toString().length === 1 ? '0' : '';
77
- const minSec = `${prependMinZero}${minutes}:${prependSecZero}${roundedSeconds}`;
78
- return hours ? `${prependHrsZero}${hours}:` + minSec : minSec;
70
+ return formatDuration(duration);
79
71
  }
80
72
  getFileSize() {
81
73
  if (this.attachment?.file_size === undefined ||
@@ -84,9 +76,6 @@ export class VoiceRecordingComponent {
84
76
  }
85
77
  return prettybytes(Number(this.attachment.file_size || 0));
86
78
  }
87
- divMod(num, divisor) {
88
- return [Math.floor(num / divisor), num % divisor];
89
- }
90
79
  }
91
80
  VoiceRecordingComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
92
81
  VoiceRecordingComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: VoiceRecordingComponent, selector: "stream-voice-recording", inputs: { attachment: "attachment" }, viewQueries: [{ propertyName: "audioElement", first: true, predicate: ["audioElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"str-chat__message-attachment__voice-recording-widget\"\n data-testid=\"voice-recording-widget\"\n [class.str-chat__message-attachment__voice-recording-widget--error]=\"isError\"\n>\n <!-- Empty event handlers to trigger change detection -->\n <audio\n #audioElement\n (play)=\"(null)\"\n (pause)=\"(null)\"\n (ended)=\"(null)\"\n (error)=\"isError = true\"\n (abort)=\"isError = true\"\n >\n <source\n data-testid=\"audio-source\"\n [src]=\"attachment?.asset_url\"\n [type]=\"attachment?.mime_type\"\n />\n </audio>\n <button\n class=\"str-chat__message-attachment-audio-widget--play-button\"\n data-testid=\"play-button\"\n (click)=\"togglePlay()\"\n >\n <stream-icon-placeholder\n [icon]=\"audioElement?.paused ? 'play' : 'pause'\"\n ></stream-icon-placeholder>\n </button>\n <div class=\"str-chat__message-attachment__voice-recording-widget__metadata\">\n <div class=\"str-chat__message-attachment-voice-recording-widget--first-row\">\n <div\n class=\"str-chat__message-attachment__voice-recording-widget__title\"\n data-testid=\"voice-recording-title\"\n [title]=\"attachment?.title\"\n >\n {{ attachment?.title }}\n </div>\n </div>\n\n <ng-container *ngIf=\"isError; else state\">\n <div\n class=\"str-chat__message-attachment__voice-recording-widget__error-message\"\n >\n <stream-icon-placeholder icon=\"error\"></stream-icon-placeholder>\n <span data-testid=\"error-message\">{{\n \"streamChat.Error playing audio\" | translate\n }}</span>\n </div>\n </ng-container>\n <ng-template #state>\n <div\n class=\"str-chat__message-attachment__voice-recording-widget__audio-state\"\n >\n <div\n class=\"str-chat__message-attachment__voice-recording-widget__timer\"\n >\n <span\n *ngIf=\"!!attachment?.duration; else fileSizeTemplate\"\n data-testid=\"duration\"\n >\n {{\n secondsElapsed > 0 || !audioElement.paused\n ? secondsElapsedFormatted\n : durationFormatted\n }}</span\n >\n <ng-template #fileSizeTemplate>\n <span\n class=\"str-chat__message-attachment-file--item-size\"\n data-testid=\"file-size-indicator\"\n >\n {{ fileSize }}\n </span>\n </ng-template>\n </div>\n <stream-voice-recording-wavebar\n *ngIf=\"attachment?.waveform_data && attachment?.duration\"\n [waveFormData]=\"attachment?.waveform_data || []\"\n [duration]=\"attachment?.duration\"\n [audioElement]=\"audioElement\"\n ></stream-voice-recording-wavebar>\n </div>\n </ng-template>\n </div>\n <div\n class=\"str-chat__message-attachment__voice-recording-widget__right-section\"\n >\n <button\n *ngIf=\"!audioElement?.paused; else fileIcon\"\n class=\"str-chat__message_attachment__playback-rate-button\"\n data-testid=\"playback-rate-button\"\n (click)=\"setPlaybackRate()\"\n >\n {{ audioElement?.playbackRate | number : \"1.1-1\" }}x\n </button>\n <ng-template #fileIcon>\n <stream-icon-placeholder\n class=\"str-chat__attachment-type-icon\"\n icon=\"audio-file\"\n ></stream-icon-placeholder>\n </ng-template>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }, { kind: "component", type: i3.VoiceRecordingWavebarComponent, selector: "stream-voice-recording-wavebar", inputs: ["audioElement", "waveFormData", "duration"] }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] });
@@ -99,4 +88,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
99
88
  type: ViewChild,
100
89
  args: ['audioElement']
101
90
  }] } });
102
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recording.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording.component.ts","../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording.component.html"],"names":[],"mappings":"AAAA,OAAO,EAGL,SAAS,EAET,KAAK,EAIL,SAAS,GACV,MAAM,eAAe,CAAC;AAGvB,OAAO,WAAW,MAAM,cAAc,CAAC;;;;;;AAEvC;;GAEG;AAMH,MAAM,OAAO,uBAAuB;IAalC,YAAoB,MAAc,EAAU,KAAwB;QAAhD,WAAM,GAAN,MAAM,CAAQ;QAAU,UAAK,GAAL,KAAK,CAAmB;QARpE,aAAQ,GAAW,EAAE,CAAC;QAEtB,sBAAiB,GAAW,EAAE,CAAC;QAC/B,mBAAc,GAAG,CAAC,CAAC;QACnB,YAAO,GAAG,KAAK,CAAC;QAKd,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,oBAAoB,CACtD,IAAI,CAAC,cAAc,CACpB,CAAC;IACJ,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,UAAU,EAAE;YACtB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YACnC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAChD,IAAI,CAAC,UAAU,EAAE,QAAQ,CAC1B,CAAC;SACH;IACH,CAAC;IAED,eAAe;QACb,8DAA8D;QAC9D,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;YACjC,IAAI,CAAC,YAAY,EAAE,aAAa,CAAC,gBAAgB,CAAC,YAAY,EAAE,GAAG,EAAE;gBACnE,MAAM,cAAc,GAAG,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,KAAK;oBAC5D,CAAC,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,IAAI,CAAC;oBAChC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,WAAW,IAAI,CAAC,CAAC,CAAC;gBACnE,IAAI,IAAI,CAAC,cAAc,KAAK,cAAc,EAAE;oBAC1C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;wBACnB,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC;wBACrC,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,oBAAoB,CACtD,IAAI,CAAC,cAAc,CACpB,CAAC;wBACF,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;oBAC7B,CAAC,CAAC,CAAC;iBACJ;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,UAAU;QACd,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,SAAS,EAAE;YACrD,OAAO;SACR;QACD,IAAI;YACF,IAAI,CAAC,YAAY,EAAE,aAAa,CAAC,MAAM;gBACrC,CAAC,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,IAAI,EAAE;gBAC9C,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;YAC5C,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;SACtB;QAAC,OAAO,KAAK,EAAE;YACd,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;SACrB;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE;YACrC,OAAO;SACR;QACD,IAAI,YAAY,GAAG,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,YAAY,GAAG,GAAG,CAAC;QACxE,IAAI,YAAY,GAAG,CAAC,EAAE;YACpB,YAAY,GAAG,CAAC,CAAC;SAClB;QACD,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,YAAY,GAAG,YAAY,CAAC;IAC9D,CAAC;IAEO,oBAAoB,CAAC,QAAiB;QAC5C,IAAI,QAAQ,KAAK,SAAS,IAAI,QAAQ,IAAI,CAAC;YAAE,OAAO,OAAO,CAAC;QAE5D,MAAM,CAAC,KAAK,EAAE,aAAa,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;QAC3D,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC;QAC1D,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAE1C,MAAM,cAAc,GAAG,KAAK,CAAC,QAAQ,EAAE,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QAChE,MAAM,cAAc,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QAClE,MAAM,cAAc,GAAG,cAAc,CAAC,QAAQ,EAAE,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QACzE,MAAM,MAAM,GAAG,GAAG,cAAc,GAAG,OAAO,IAAI,cAAc,GAAG,cAAc,EAAE,CAAC;QAEhF,OAAO,KAAK,CAAC,CAAC,CAAC,GAAG,cAAc,GAAG,KAAK,GAAG,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;IAChE,CAAC;IAEO,WAAW;QACjB,IACE,IAAI,CAAC,UAAU,EAAE,SAAS,KAAK,SAAS;YACxC,IAAI,CAAC,UAAU,EAAE,SAAS,KAAK,IAAI,EACnC;YACA,OAAO,EAAE,CAAC;SACX;QACD,OAAO,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,CAAC;IAC7D,CAAC;IAEO,MAAM,CAAC,GAAW,EAAE,OAAe;QACzC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,OAAO,CAAC,EAAE,GAAG,GAAG,OAAO,CAAC,CAAC;IACpD,CAAC;;oHApGU,uBAAuB;wGAAvB,uBAAuB,uOCvBpC,85GAwGA;2FDjFa,uBAAuB;kBALnC,SAAS;+BACE,wBAAwB;6HAQzB,UAAU;sBAAlB,KAAK;gBAOE,YAAY;sBADnB,SAAS;uBAAC,cAAc","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport { DefaultStreamChatGenerics } from '../types';\nimport prettybytes from 'pretty-bytes';\n\n/**\n * This component can be used to display an attachment with type `voiceRecording`. The component allows playing the attachment inside the browser.\n */\n@Component({\n  selector: 'stream-voice-recording',\n  templateUrl: './voice-recording.component.html',\n  styles: [],\n})\nexport class VoiceRecordingComponent implements OnChanges, AfterViewInit {\n  /**\n   * The voice recording attachment\n   */\n  @Input() attachment?: Attachment<DefaultStreamChatGenerics>;\n  fileSize: string = '';\n  secondsElapsedFormatted: string;\n  durationFormatted: string = '';\n  secondsElapsed = 0;\n  isError = false;\n  @ViewChild('audioElement')\n  private audioElement?: ElementRef<HTMLAudioElement>;\n\n  constructor(private ngZone: NgZone, private cdRef: ChangeDetectorRef) {\n    this.secondsElapsedFormatted = this.getFormattedDuration(\n      this.secondsElapsed\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.attachment) {\n      this.fileSize = this.getFileSize();\n      this.durationFormatted = this.getFormattedDuration(\n        this.attachment?.duration\n      );\n    }\n  }\n\n  ngAfterViewInit(): void {\n    // timeupdate fired frequntly so we optimize change detections\n    this.ngZone.runOutsideAngular(() => {\n      this.audioElement?.nativeElement.addEventListener('timeupdate', () => {\n        const secondsElapsed = this.audioElement?.nativeElement?.ended\n          ? this.attachment?.duration || 0\n          : Math.round(this.audioElement?.nativeElement?.currentTime || 0);\n        if (this.secondsElapsed !== secondsElapsed) {\n          this.ngZone.run(() => {\n            this.secondsElapsed = secondsElapsed;\n            this.secondsElapsedFormatted = this.getFormattedDuration(\n              this.secondsElapsed\n            );\n            this.cdRef.detectChanges();\n          });\n        }\n      });\n    });\n  }\n\n  async togglePlay() {\n    if (!this.audioElement || !this.attachment?.asset_url) {\n      return;\n    }\n    try {\n      this.audioElement?.nativeElement.paused\n        ? await this.audioElement.nativeElement.play()\n        : this.audioElement.nativeElement.pause();\n      this.isError = false;\n    } catch (error) {\n      this.isError = true;\n    }\n  }\n\n  setPlaybackRate() {\n    if (!this.audioElement?.nativeElement) {\n      return;\n    }\n    let playbackRate = this.audioElement?.nativeElement?.playbackRate + 0.5;\n    if (playbackRate > 2) {\n      playbackRate = 1;\n    }\n    this.audioElement.nativeElement.playbackRate = playbackRate;\n  }\n\n  private getFormattedDuration(duration?: number) {\n    if (duration === undefined || duration <= 0) return '00:00';\n\n    const [hours, hoursLeftover] = this.divMod(duration, 3600);\n    const [minutes, seconds] = this.divMod(hoursLeftover, 60);\n    const roundedSeconds = Math.ceil(seconds);\n\n    const prependHrsZero = hours.toString().length === 1 ? '0' : '';\n    const prependMinZero = minutes.toString().length === 1 ? '0' : '';\n    const prependSecZero = roundedSeconds.toString().length === 1 ? '0' : '';\n    const minSec = `${prependMinZero}${minutes}:${prependSecZero}${roundedSeconds}`;\n\n    return hours ? `${prependHrsZero}${hours}:` + minSec : minSec;\n  }\n\n  private getFileSize() {\n    if (\n      this.attachment?.file_size === undefined ||\n      this.attachment?.file_size === null\n    ) {\n      return '';\n    }\n    return prettybytes(Number(this.attachment.file_size || 0));\n  }\n\n  private divMod(num: number, divisor: number) {\n    return [Math.floor(num / divisor), num % divisor];\n  }\n}\n","<div\n  class=\"str-chat__message-attachment__voice-recording-widget\"\n  data-testid=\"voice-recording-widget\"\n  [class.str-chat__message-attachment__voice-recording-widget--error]=\"isError\"\n>\n  <!-- Empty event handlers to trigger change detection -->\n  <audio\n    #audioElement\n    (play)=\"(null)\"\n    (pause)=\"(null)\"\n    (ended)=\"(null)\"\n    (error)=\"isError = true\"\n    (abort)=\"isError = true\"\n  >\n    <source\n      data-testid=\"audio-source\"\n      [src]=\"attachment?.asset_url\"\n      [type]=\"attachment?.mime_type\"\n    />\n  </audio>\n  <button\n    class=\"str-chat__message-attachment-audio-widget--play-button\"\n    data-testid=\"play-button\"\n    (click)=\"togglePlay()\"\n  >\n    <stream-icon-placeholder\n      [icon]=\"audioElement?.paused ? 'play' : 'pause'\"\n    ></stream-icon-placeholder>\n  </button>\n  <div class=\"str-chat__message-attachment__voice-recording-widget__metadata\">\n    <div class=\"str-chat__message-attachment-voice-recording-widget--first-row\">\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__title\"\n        data-testid=\"voice-recording-title\"\n        [title]=\"attachment?.title\"\n      >\n        {{ attachment?.title }}\n      </div>\n    </div>\n\n    <ng-container *ngIf=\"isError; else state\">\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__error-message\"\n      >\n        <stream-icon-placeholder icon=\"error\"></stream-icon-placeholder>\n        <span data-testid=\"error-message\">{{\n          \"streamChat.Error playing audio\" | translate\n        }}</span>\n      </div>\n    </ng-container>\n    <ng-template #state>\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__audio-state\"\n      >\n        <div\n          class=\"str-chat__message-attachment__voice-recording-widget__timer\"\n        >\n          <span\n            *ngIf=\"!!attachment?.duration; else fileSizeTemplate\"\n            data-testid=\"duration\"\n          >\n            {{\n              secondsElapsed > 0 || !audioElement.paused\n                ? secondsElapsedFormatted\n                : durationFormatted\n            }}</span\n          >\n          <ng-template #fileSizeTemplate>\n            <span\n              class=\"str-chat__message-attachment-file--item-size\"\n              data-testid=\"file-size-indicator\"\n            >\n              {{ fileSize }}\n            </span>\n          </ng-template>\n        </div>\n        <stream-voice-recording-wavebar\n          *ngIf=\"attachment?.waveform_data && attachment?.duration\"\n          [waveFormData]=\"attachment?.waveform_data || []\"\n          [duration]=\"attachment?.duration\"\n          [audioElement]=\"audioElement\"\n        ></stream-voice-recording-wavebar>\n      </div>\n    </ng-template>\n  </div>\n  <div\n    class=\"str-chat__message-attachment__voice-recording-widget__right-section\"\n  >\n    <button\n      *ngIf=\"!audioElement?.paused; else fileIcon\"\n      class=\"str-chat__message_attachment__playback-rate-button\"\n      data-testid=\"playback-rate-button\"\n      (click)=\"setPlaybackRate()\"\n    >\n      {{ audioElement?.playbackRate | number : \"1.1-1\" }}x\n    </button>\n    <ng-template #fileIcon>\n      <stream-icon-placeholder\n        class=\"str-chat__attachment-type-icon\"\n        icon=\"audio-file\"\n      ></stream-icon-placeholder>\n    </ng-template>\n  </div>\n</div>\n"]}
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recording.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording.component.ts","../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport { DefaultStreamChatGenerics } from '../types';\nimport prettybytes from 'pretty-bytes';\nimport { formatDuration } from '../format-duration';\n\n/**\n * This component can be used to display an attachment with type `voiceRecording`. The component allows playing the attachment inside the browser.\n */\n@Component({\n  selector: 'stream-voice-recording',\n  templateUrl: './voice-recording.component.html',\n  styles: [],\n})\nexport class VoiceRecordingComponent implements OnChanges, AfterViewInit {\n  /**\n   * The voice recording attachment\n   */\n  @Input() attachment?: Attachment<DefaultStreamChatGenerics>;\n  fileSize: string = '';\n  secondsElapsedFormatted: string;\n  durationFormatted: string = '';\n  secondsElapsed = 0;\n  isError = false;\n  @ViewChild('audioElement')\n  private audioElement?: ElementRef<HTMLAudioElement>;\n\n  constructor(private ngZone: NgZone, private cdRef: ChangeDetectorRef) {\n    this.secondsElapsedFormatted = this.getFormattedDuration(\n      this.secondsElapsed\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.attachment) {\n      this.fileSize = this.getFileSize();\n      this.durationFormatted = this.getFormattedDuration(\n        this.attachment?.duration\n      );\n    }\n  }\n\n  ngAfterViewInit(): void {\n    // timeupdate fired frequntly so we optimize change detections\n    this.ngZone.runOutsideAngular(() => {\n      this.audioElement?.nativeElement.addEventListener('timeupdate', () => {\n        const secondsElapsed = this.audioElement?.nativeElement?.ended\n          ? this.attachment?.duration || 0\n          : Math.round(this.audioElement?.nativeElement?.currentTime || 0);\n        if (this.secondsElapsed !== secondsElapsed) {\n          this.ngZone.run(() => {\n            this.secondsElapsed = secondsElapsed;\n            this.secondsElapsedFormatted = this.getFormattedDuration(\n              this.secondsElapsed\n            );\n            this.cdRef.detectChanges();\n          });\n        }\n      });\n    });\n  }\n\n  async togglePlay() {\n    if (!this.audioElement || !this.attachment?.asset_url) {\n      return;\n    }\n    try {\n      this.audioElement?.nativeElement.paused\n        ? await this.audioElement.nativeElement.play()\n        : this.audioElement.nativeElement.pause();\n      this.isError = false;\n    } catch (error) {\n      this.isError = true;\n    }\n  }\n\n  setPlaybackRate() {\n    if (!this.audioElement?.nativeElement) {\n      return;\n    }\n    let playbackRate = this.audioElement?.nativeElement?.playbackRate + 0.5;\n    if (playbackRate > 2) {\n      playbackRate = 1;\n    }\n    this.audioElement.nativeElement.playbackRate = playbackRate;\n  }\n\n  private getFormattedDuration(duration?: number) {\n    return formatDuration(duration);\n  }\n\n  private getFileSize() {\n    if (\n      this.attachment?.file_size === undefined ||\n      this.attachment?.file_size === null\n    ) {\n      return '';\n    }\n    return prettybytes(Number(this.attachment.file_size || 0));\n  }\n}\n","<div\n  class=\"str-chat__message-attachment__voice-recording-widget\"\n  data-testid=\"voice-recording-widget\"\n  [class.str-chat__message-attachment__voice-recording-widget--error]=\"isError\"\n>\n  <!-- Empty event handlers to trigger change detection -->\n  <audio\n    #audioElement\n    (play)=\"(null)\"\n    (pause)=\"(null)\"\n    (ended)=\"(null)\"\n    (error)=\"isError = true\"\n    (abort)=\"isError = true\"\n  >\n    <source\n      data-testid=\"audio-source\"\n      [src]=\"attachment?.asset_url\"\n      [type]=\"attachment?.mime_type\"\n    />\n  </audio>\n  <button\n    class=\"str-chat__message-attachment-audio-widget--play-button\"\n    data-testid=\"play-button\"\n    (click)=\"togglePlay()\"\n  >\n    <stream-icon-placeholder\n      [icon]=\"audioElement?.paused ? 'play' : 'pause'\"\n    ></stream-icon-placeholder>\n  </button>\n  <div class=\"str-chat__message-attachment__voice-recording-widget__metadata\">\n    <div class=\"str-chat__message-attachment-voice-recording-widget--first-row\">\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__title\"\n        data-testid=\"voice-recording-title\"\n        [title]=\"attachment?.title\"\n      >\n        {{ attachment?.title }}\n      </div>\n    </div>\n\n    <ng-container *ngIf=\"isError; else state\">\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__error-message\"\n      >\n        <stream-icon-placeholder icon=\"error\"></stream-icon-placeholder>\n        <span data-testid=\"error-message\">{{\n          \"streamChat.Error playing audio\" | translate\n        }}</span>\n      </div>\n    </ng-container>\n    <ng-template #state>\n      <div\n        class=\"str-chat__message-attachment__voice-recording-widget__audio-state\"\n      >\n        <div\n          class=\"str-chat__message-attachment__voice-recording-widget__timer\"\n        >\n          <span\n            *ngIf=\"!!attachment?.duration; else fileSizeTemplate\"\n            data-testid=\"duration\"\n          >\n            {{\n              secondsElapsed > 0 || !audioElement.paused\n                ? secondsElapsedFormatted\n                : durationFormatted\n            }}</span\n          >\n          <ng-template #fileSizeTemplate>\n            <span\n              class=\"str-chat__message-attachment-file--item-size\"\n              data-testid=\"file-size-indicator\"\n            >\n              {{ fileSize }}\n            </span>\n          </ng-template>\n        </div>\n        <stream-voice-recording-wavebar\n          *ngIf=\"attachment?.waveform_data && attachment?.duration\"\n          [waveFormData]=\"attachment?.waveform_data || []\"\n          [duration]=\"attachment?.duration\"\n          [audioElement]=\"audioElement\"\n        ></stream-voice-recording-wavebar>\n      </div>\n    </ng-template>\n  </div>\n  <div\n    class=\"str-chat__message-attachment__voice-recording-widget__right-section\"\n  >\n    <button\n      *ngIf=\"!audioElement?.paused; else fileIcon\"\n      class=\"str-chat__message_attachment__playback-rate-button\"\n      data-testid=\"playback-rate-button\"\n      (click)=\"setPlaybackRate()\"\n    >\n      {{ audioElement?.playbackRate | number : \"1.1-1\" }}x\n    </button>\n    <ng-template #fileIcon>\n      <stream-icon-placeholder\n        class=\"str-chat__attachment-type-icon\"\n        icon=\"audio-file\"\n      ></stream-icon-placeholder>\n    </ng-template>\n  </div>\n</div>\n"]}
@@ -0,0 +1,21 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { VoiceRecordingComponent } from './voice-recording.component';
4
+ import { VoiceRecordingWavebarComponent } from './voice-recording-wavebar/voice-recording-wavebar.component';
5
+ import { IconModule } from '../icon/icon.module';
6
+ import { TranslateModule } from '@ngx-translate/core';
7
+ import * as i0 from "@angular/core";
8
+ export class VoiceRecordingModule {
9
+ }
10
+ VoiceRecordingModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
11
+ VoiceRecordingModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingModule, declarations: [VoiceRecordingComponent, VoiceRecordingWavebarComponent], imports: [CommonModule, IconModule, TranslateModule], exports: [VoiceRecordingComponent, VoiceRecordingWavebarComponent] });
12
+ VoiceRecordingModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingModule, imports: [CommonModule, IconModule, TranslateModule] });
13
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingModule, decorators: [{
14
+ type: NgModule,
15
+ args: [{
16
+ declarations: [VoiceRecordingComponent, VoiceRecordingWavebarComponent],
17
+ imports: [CommonModule, IconModule, TranslateModule],
18
+ exports: [VoiceRecordingComponent, VoiceRecordingWavebarComponent],
19
+ }]
20
+ }] });
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtcmVjb3JkaW5nLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi92b2ljZS1yZWNvcmRpbmcvdm9pY2UtcmVjb3JkaW5nLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUN0RSxPQUFPLEVBQUUsOEJBQThCLEVBQUUsTUFBTSw2REFBNkQsQ0FBQztBQUM3RyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDakQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDOztBQU90RCxNQUFNLE9BQU8sb0JBQW9COztpSEFBcEIsb0JBQW9CO2tIQUFwQixvQkFBb0IsaUJBSmhCLHVCQUF1QixFQUFFLDhCQUE4QixhQUM1RCxZQUFZLEVBQUUsVUFBVSxFQUFFLGVBQWUsYUFDekMsdUJBQXVCLEVBQUUsOEJBQThCO2tIQUV0RCxvQkFBb0IsWUFIckIsWUFBWSxFQUFFLFVBQVUsRUFBRSxlQUFlOzJGQUd4QyxvQkFBb0I7a0JBTGhDLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFLENBQUMsdUJBQXVCLEVBQUUsOEJBQThCLENBQUM7b0JBQ3ZFLE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxVQUFVLEVBQUUsZUFBZSxDQUFDO29CQUNwRCxPQUFPLEVBQUUsQ0FBQyx1QkFBdUIsRUFBRSw4QkFBOEIsQ0FBQztpQkFDbkUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFZvaWNlUmVjb3JkaW5nQ29tcG9uZW50IH0gZnJvbSAnLi92b2ljZS1yZWNvcmRpbmcuY29tcG9uZW50JztcbmltcG9ydCB7IFZvaWNlUmVjb3JkaW5nV2F2ZWJhckNvbXBvbmVudCB9IGZyb20gJy4vdm9pY2UtcmVjb3JkaW5nLXdhdmViYXIvdm9pY2UtcmVjb3JkaW5nLXdhdmViYXIuY29tcG9uZW50JztcbmltcG9ydCB7IEljb25Nb2R1bGUgfSBmcm9tICcuLi9pY29uL2ljb24ubW9kdWxlJztcbmltcG9ydCB7IFRyYW5zbGF0ZU1vZHVsZSB9IGZyb20gJ0BuZ3gtdHJhbnNsYXRlL2NvcmUnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtWb2ljZVJlY29yZGluZ0NvbXBvbmVudCwgVm9pY2VSZWNvcmRpbmdXYXZlYmFyQ29tcG9uZW50XSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSWNvbk1vZHVsZSwgVHJhbnNsYXRlTW9kdWxlXSxcbiAgZXhwb3J0czogW1ZvaWNlUmVjb3JkaW5nQ29tcG9uZW50LCBWb2ljZVJlY29yZGluZ1dhdmViYXJDb21wb25lbnRdLFxufSlcbmV4cG9ydCBjbGFzcyBWb2ljZVJlY29yZGluZ01vZHVsZSB7fVxuIl19
@@ -0,0 +1,72 @@
1
+ export const resampleWaveForm = (waveFormData, sampleSize) => {
2
+ return waveFormData.length > sampleSize
3
+ ? downsample(waveFormData, sampleSize)
4
+ : upsample(waveFormData, sampleSize);
5
+ };
6
+ const downsample = (waveFormData, sampleSize) => {
7
+ if (waveFormData.length <= sampleSize) {
8
+ return waveFormData;
9
+ }
10
+ if (sampleSize === 1)
11
+ return [mean(waveFormData)];
12
+ const result = [];
13
+ // bucket size adjusted due to the fact that the first and the last item in the original data array is kept in target output
14
+ const bucketSize = (waveFormData.length - 2) / (sampleSize - 2);
15
+ let lastSelectedPointIndex = 0;
16
+ result.push(waveFormData[lastSelectedPointIndex]); // Always add the first point
17
+ let maxAreaPoint, maxArea, triangleArea;
18
+ for (let bucketIndex = 1; bucketIndex < sampleSize - 1; bucketIndex++) {
19
+ const previousBucketRefPoint = waveFormData[lastSelectedPointIndex];
20
+ const nextBucketMean = getNextBucketMean(waveFormData, bucketIndex, bucketSize);
21
+ const currentBucketStartIndex = Math.floor((bucketIndex - 1) * bucketSize) + 1;
22
+ const nextBucketStartIndex = Math.floor(bucketIndex * bucketSize) + 1;
23
+ const countUnitsBetweenAtoC = 1 + nextBucketStartIndex - currentBucketStartIndex;
24
+ maxArea = triangleArea = -1;
25
+ for (let currentPointIndex = currentBucketStartIndex; currentPointIndex < nextBucketStartIndex; currentPointIndex++) {
26
+ const countUnitsBetweenAtoB = Math.abs(currentPointIndex - currentBucketStartIndex) + 1;
27
+ const countUnitsBetweenBtoC = countUnitsBetweenAtoC - countUnitsBetweenAtoB;
28
+ const currentPointValue = waveFormData[currentPointIndex];
29
+ triangleArea = triangleAreaHeron(triangleBase(Math.abs(previousBucketRefPoint - currentPointValue), countUnitsBetweenAtoB), triangleBase(Math.abs(currentPointValue - nextBucketMean), countUnitsBetweenBtoC), triangleBase(Math.abs(previousBucketRefPoint - nextBucketMean), countUnitsBetweenAtoC));
30
+ if (triangleArea > maxArea) {
31
+ maxArea = triangleArea;
32
+ maxAreaPoint = waveFormData[currentPointIndex];
33
+ lastSelectedPointIndex = currentPointIndex;
34
+ }
35
+ }
36
+ if (typeof maxAreaPoint !== 'undefined')
37
+ result.push(maxAreaPoint);
38
+ }
39
+ result.push(waveFormData[waveFormData.length - 1]); // Always add the last point
40
+ return result;
41
+ };
42
+ const upsample = (waveFormData, sampleSize) => {
43
+ if (sampleSize === waveFormData.length)
44
+ return waveFormData;
45
+ // eslint-disable-next-line prefer-const
46
+ let [bucketSize, remainder] = divMod(sampleSize, waveFormData.length);
47
+ const result = [];
48
+ for (let i = 0; i < waveFormData.length; i++) {
49
+ const extra = remainder && remainder-- ? 1 : 0;
50
+ result.push(...Array(bucketSize + extra).fill(waveFormData[i]));
51
+ }
52
+ return result;
53
+ };
54
+ const getNextBucketMean = (data, currentBucketIndex, bucketSize) => {
55
+ const nextBucketStartIndex = Math.floor(currentBucketIndex * bucketSize) + 1;
56
+ let nextNextBucketStartIndex = Math.floor((currentBucketIndex + 1) * bucketSize) + 1;
57
+ nextNextBucketStartIndex =
58
+ nextNextBucketStartIndex < data.length
59
+ ? nextNextBucketStartIndex
60
+ : data.length;
61
+ return mean(data.slice(nextBucketStartIndex, nextNextBucketStartIndex));
62
+ };
63
+ const mean = (values) => values.reduce((acc, value) => acc + value, 0) / values.length;
64
+ const triangleAreaHeron = (a, b, c) => {
65
+ const s = (a + b + c) / 2;
66
+ return Math.sqrt(s * (s - a) * (s - b) * (s - c));
67
+ };
68
+ const triangleBase = (a, b) => Math.sqrt(Math.pow(a, 2) + Math.pow(b, 2));
69
+ const divMod = (num, divisor) => {
70
+ return [Math.floor(num / divisor), num % divisor];
71
+ };
72
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wave-form-sampler.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/wave-form-sampler.ts"],"names":[],"mappings":"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","sourcesContent":["export const resampleWaveForm = (\n  waveFormData: number[],\n  sampleSize: number\n) => {\n  return waveFormData.length > sampleSize\n    ? downsample(waveFormData, sampleSize)\n    : upsample(waveFormData, sampleSize);\n};\n\nconst downsample = (waveFormData: number[], sampleSize: number) => {\n  if (waveFormData.length <= sampleSize) {\n    return waveFormData;\n  }\n\n  if (sampleSize === 1) return [mean(waveFormData)];\n\n  const result: number[] = [];\n  // bucket size adjusted due to the fact that the first and the last item in the original data array is kept in target output\n  const bucketSize = (waveFormData.length - 2) / (sampleSize - 2);\n  let lastSelectedPointIndex = 0;\n  result.push(waveFormData[lastSelectedPointIndex]); // Always add the first point\n  let maxAreaPoint, maxArea, triangleArea;\n\n  for (let bucketIndex = 1; bucketIndex < sampleSize - 1; bucketIndex++) {\n    const previousBucketRefPoint = waveFormData[lastSelectedPointIndex];\n    const nextBucketMean = getNextBucketMean(\n      waveFormData,\n      bucketIndex,\n      bucketSize\n    );\n\n    const currentBucketStartIndex =\n      Math.floor((bucketIndex - 1) * bucketSize) + 1;\n    const nextBucketStartIndex = Math.floor(bucketIndex * bucketSize) + 1;\n    const countUnitsBetweenAtoC =\n      1 + nextBucketStartIndex - currentBucketStartIndex;\n\n    maxArea = triangleArea = -1;\n\n    for (\n      let currentPointIndex = currentBucketStartIndex;\n      currentPointIndex < nextBucketStartIndex;\n      currentPointIndex++\n    ) {\n      const countUnitsBetweenAtoB =\n        Math.abs(currentPointIndex - currentBucketStartIndex) + 1;\n      const countUnitsBetweenBtoC =\n        countUnitsBetweenAtoC - countUnitsBetweenAtoB;\n      const currentPointValue = waveFormData[currentPointIndex];\n\n      triangleArea = triangleAreaHeron(\n        triangleBase(\n          Math.abs(previousBucketRefPoint - currentPointValue),\n          countUnitsBetweenAtoB\n        ),\n        triangleBase(\n          Math.abs(currentPointValue - nextBucketMean),\n          countUnitsBetweenBtoC\n        ),\n        triangleBase(\n          Math.abs(previousBucketRefPoint - nextBucketMean),\n          countUnitsBetweenAtoC\n        )\n      );\n\n      if (triangleArea > maxArea) {\n        maxArea = triangleArea;\n        maxAreaPoint = waveFormData[currentPointIndex];\n        lastSelectedPointIndex = currentPointIndex;\n      }\n    }\n\n    if (typeof maxAreaPoint !== 'undefined') result.push(maxAreaPoint);\n  }\n\n  result.push(waveFormData[waveFormData.length - 1]); // Always add the last point\n\n  return result;\n};\n\nconst upsample = (waveFormData: number[], sampleSize: number) => {\n  if (sampleSize === waveFormData.length) return waveFormData;\n\n  // eslint-disable-next-line  prefer-const\n  let [bucketSize, remainder] = divMod(sampleSize, waveFormData.length);\n  const result: number[] = [];\n\n  for (let i = 0; i < waveFormData.length; i++) {\n    const extra = remainder && remainder-- ? 1 : 0;\n    result.push(...Array<number>(bucketSize + extra).fill(waveFormData[i]));\n  }\n  return result;\n};\n\nconst getNextBucketMean = (\n  data: number[],\n  currentBucketIndex: number,\n  bucketSize: number\n) => {\n  const nextBucketStartIndex = Math.floor(currentBucketIndex * bucketSize) + 1;\n  let nextNextBucketStartIndex =\n    Math.floor((currentBucketIndex + 1) * bucketSize) + 1;\n  nextNextBucketStartIndex =\n    nextNextBucketStartIndex < data.length\n      ? nextNextBucketStartIndex\n      : data.length;\n\n  return mean(data.slice(nextBucketStartIndex, nextNextBucketStartIndex));\n};\n\nconst mean = (values: number[]) =>\n  values.reduce((acc, value) => acc + value, 0) / values.length;\n\nconst triangleAreaHeron = (a: number, b: number, c: number) => {\n  const s = (a + b + c) / 2;\n  return Math.sqrt(s * (s - a) * (s - b) * (s - c));\n};\n\nconst triangleBase = (a: number, b: number) =>\n  Math.sqrt(Math.pow(a, 2) + Math.pow(b, 2));\n\nconst divMod = (num: number, divisor: number) => {\n  return [Math.floor(num / divisor), num % divisor];\n};\n"]}
@@ -10,9 +10,9 @@ export * from './lib/stream-i18n.service';
10
10
  export * from './lib/avatar/avatar.component';
11
11
  export * from './lib/avatar-placeholder/avatar-placeholder.component';
12
12
  export * from './lib/icon/icon.component';
13
- export * from './lib/icon-placeholder/icon-placeholder.component';
14
- export * from './lib/loading-indicator/loading-indicator.component';
15
- export * from './lib/loading-indicator-placeholder/loading-indicator-placeholder.component';
13
+ export * from './lib/icon/icon-placeholder/icon-placeholder.component';
14
+ export * from './lib/icon/loading-indicator/loading-indicator.component';
15
+ export * from './lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component';
16
16
  export * from './lib/message-actions-box/message-actions-box.component';
17
17
  export * from './lib/channel/channel.component';
18
18
  export * from './lib/channel-header/channel-header.component';
@@ -42,7 +42,7 @@ export * from './lib/read-by';
42
42
  export * from './lib/get-message-translation';
43
43
  export * from './lib/get-channel-display-text';
44
44
  export * from './lib/is-image-attachment';
45
- export * from './lib/is-image-file';
45
+ export * from './lib/file-utils';
46
46
  export * from './lib/message-preview';
47
47
  export * from './lib/notification.service';
48
48
  export * from './lib/transliteration.service';
@@ -66,4 +66,17 @@ export * from './lib/virtualized-list.service';
66
66
  export * from './lib/virtualized-message-list.service';
67
67
  export * from './lib/user-list/user-list.component';
68
68
  export * from './lib/paginated-list/paginated-list.component';
69
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL3B1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYyxxQkFBcUIsQ0FBQztBQUNwQyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsd0NBQXdDLENBQUM7QUFDdkQsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLCtCQUErQixDQUFDO0FBQzlDLGNBQWMsdURBQXVELENBQUM7QUFDdEUsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLG1EQUFtRCxDQUFDO0FBQ2xFLGNBQWMscURBQXFELENBQUM7QUFDcEUsY0FBYyw2RUFBNkUsQ0FBQztBQUM1RixjQUFjLHlEQUF5RCxDQUFDO0FBQ3hFLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYywrQ0FBK0MsQ0FBQztBQUM5RCxjQUFjLGlEQUFpRCxDQUFDO0FBQ2hFLGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyw2Q0FBNkMsQ0FBQztBQUM1RCxjQUFjLDZEQUE2RCxDQUFDO0FBQzVFLGNBQWMsaURBQWlELENBQUM7QUFDaEUsY0FBYywyRUFBMkUsQ0FBQztBQUMxRixjQUFjLGtEQUFrRCxDQUFDO0FBQ2pFLGNBQWMsd0NBQXdDLENBQUM7QUFDdkQsY0FBYyx3Q0FBd0MsQ0FBQztBQUN2RCxjQUFjLHlDQUF5QyxDQUFDO0FBQ3hELGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLGlEQUFpRCxDQUFDO0FBQ2hFLGNBQWMsaUVBQWlFLENBQUM7QUFDaEYsY0FBYyxxREFBcUQsQ0FBQztBQUNwRSxjQUFjLDJDQUEyQyxDQUFDO0FBQzFELGNBQWMscURBQXFELENBQUM7QUFDcEUsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyxnQ0FBZ0MsQ0FBQztBQUMvQyxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLDRCQUE0QixDQUFDO0FBQzNDLGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsYUFBYSxDQUFDO0FBQzVCLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLGlEQUFpRCxDQUFDO0FBQ2hFLGNBQWMsaUZBQWlGLENBQUM7QUFDaEcsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLHVFQUF1RSxDQUFDO0FBQ3RGLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyxnQ0FBZ0MsQ0FBQztBQUMvQyxjQUFjLHdDQUF3QyxDQUFDO0FBQ3ZELGNBQWMscUNBQXFDLENBQUM7QUFDcEQsY0FBYywrQ0FBK0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2Ygc3RyZWFtLWNoYXQtYW5ndWxhclxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vbGliL2NoYXQtY2xpZW50LnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY2hhbm5lbC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3RoZW1lLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYXR0YWNobWVudC5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2F0dGFjaG1lbnQtY29uZmlndXJhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3N0cmVhbS1pMThuLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYXZhdGFyL2F2YXRhci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYXZhdGFyLXBsYWNlaG9sZGVyL2F2YXRhci1wbGFjZWhvbGRlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvaWNvbi9pY29uLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9pY29uLXBsYWNlaG9sZGVyL2ljb24tcGxhY2Vob2xkZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2xvYWRpbmctaW5kaWNhdG9yL2xvYWRpbmctaW5kaWNhdG9yLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9sb2FkaW5nLWluZGljYXRvci1wbGFjZWhvbGRlci9sb2FkaW5nLWluZGljYXRvci1wbGFjZWhvbGRlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbWVzc2FnZS1hY3Rpb25zLWJveC9tZXNzYWdlLWFjdGlvbnMtYm94LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9jaGFubmVsL2NoYW5uZWwuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NoYW5uZWwtaGVhZGVyL2NoYW5uZWwtaGVhZGVyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9jaGFubmVsLXByZXZpZXcvY2hhbm5lbC1wcmV2aWV3LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9jaGFubmVsLWxpc3QvY2hhbm5lbC1saXN0LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlL21lc3NhZ2UuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3BhcnNlLWRhdGUnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbGlzdC11c2Vycyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlLWlucHV0L21lc3NhZ2UtaW5wdXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtYm91bmNlLXByb21wdC9tZXNzYWdlLWJvdW5jZS1wcm9tcHQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEvdGV4dGFyZWEuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtaW5wdXQvYXV0b2NvbXBsZXRlLXRleHRhcmVhL2F1dG9jb21wbGV0ZS10ZXh0YXJlYS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbWVzc2FnZS1pbnB1dC9tZXNzYWdlLWlucHV0LWNvbmZpZy5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEuaW50ZXJmYWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtaW5wdXQvZW1vamktaW5wdXQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlLWxpc3QvbWVzc2FnZS1saXN0LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlLWxpc3QvZ3JvdXAtc3R5bGVzJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2F0dGFjaG1lbnQtbGlzdC9hdHRhY2htZW50LWxpc3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2F0dGFjaG1lbnQtcHJldmlldy1saXN0L2F0dGFjaG1lbnQtcHJldmlldy1saXN0LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlLXJlYWN0aW9ucy9tZXNzYWdlLXJlYWN0aW9ucy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbm90aWZpY2F0aW9uL25vdGlmaWNhdGlvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbm90aWZpY2F0aW9uLWxpc3Qvbm90aWZpY2F0aW9uLWxpc3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3RocmVhZC90aHJlYWQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGFsL21vZGFsLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9yZWFkLWJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2dldC1tZXNzYWdlLXRyYW5zbGF0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2dldC1jaGFubmVsLWRpc3BsYXktdGV4dCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9pcy1pbWFnZS1hdHRhY2htZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2lzLWltYWdlLWZpbGUnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbWVzc2FnZS1wcmV2aWV3JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL25vdGlmaWNhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3RyYW5zbGl0ZXJhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3N0cmVhbS1jaGF0Lm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9zdHJlYW0tYXZhdGFyLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9zdHJlYW0tYXV0b2NvbXBsZXRlLXRleHRhcmVhLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9zdHJlYW0tdGV4dGFyZWEubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2luamVjdGlvbi10b2tlbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY3VzdG9tLXRlbXBsYXRlcy5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtcmVhY3Rpb25zLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0ZS1wYXJzZXIuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi90eXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9tZXNzYWdlLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbWVzc2FnZS1hY3Rpb25zLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdm9pY2UtcmVjb3JkaW5nL3ZvaWNlLXJlY29yZGluZy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdm9pY2UtcmVjb3JkaW5nL3ZvaWNlLXJlY29yZGluZy13YXZlYmFyL3ZvaWNlLXJlY29yZGluZy13YXZlYmFyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9pcy1vbi1zZXBhcmF0ZS1kYXRlJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL21lc3NhZ2UtcmVhY3Rpb25zLXNlbGVjdG9yL21lc3NhZ2UtcmVhY3Rpb25zLXNlbGVjdG9yLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9jaGFubmVsLXF1ZXJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3ZpcnR1YWxpemVkLWxpc3Quc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi92aXJ0dWFsaXplZC1tZXNzYWdlLWxpc3Quc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi91c2VyLWxpc3QvdXNlci1saXN0LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9wYWdpbmF0ZWQtbGlzdC9wYWdpbmF0ZWQtbGlzdC5jb21wb25lbnQnO1xuIl19
69
+ export * from './lib/is-safari';
70
+ export * from './lib/voice-recorder/voice-recorder.module';
71
+ export * from './lib/voice-recorder/amplitude-recorder.service';
72
+ export * from './lib/voice-recorder/audio-recorder.service';
73
+ export * from './lib/voice-recorder/media-recorder';
74
+ export * from './lib/voice-recorder/transcoder.service';
75
+ export * from './lib/voice-recorder/voice-recorder.component';
76
+ export * from './lib/voice-recording/voice-recording.module';
77
+ export * from './lib/icon/icon.module';
78
+ export * from './lib/voice-recorder//voice-recorder-wavebar/voice-recorder-wavebar.component';
79
+ export * from './lib/format-duration';
80
+ export * from './lib/message-input/voice-recorder.service';
81
+ export * from './lib/voice-recorder/mp3-transcoder';
82
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"public-api.js","sourceRoot":"","sources":["../../../projects/stream-chat-angular/src/public-api.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,cAAc,2BAA2B,CAAC;AAC1C,cAAc,uBAAuB,CAAC;AACtC,cAAc,qBAAqB,CAAC;AACpC,cAAc,0BAA0B,CAAC;AACzC,cAAc,wCAAwC,CAAC;AACvD,cAAc,2BAA2B,CAAC;AAC1C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,uDAAuD,CAAC;AACtE,cAAc,2BAA2B,CAAC;AAC1C,cAAc,wDAAwD,CAAC;AACvE,cAAc,0DAA0D,CAAC;AACzE,cAAc,kFAAkF,CAAC;AACjG,cAAc,yDAAyD,CAAC;AACxE,cAAc,iCAAiC,CAAC;AAChD,cAAc,+CAA+C,CAAC;AAC9D,cAAc,iDAAiD,CAAC;AAChE,cAAc,2CAA2C,CAAC;AAC1D,cAAc,iCAAiC,CAAC;AAChD,cAAc,kBAAkB,CAAC;AACjC,cAAc,kBAAkB,CAAC;AACjC,cAAc,6CAA6C,CAAC;AAC5D,cAAc,6DAA6D,CAAC;AAC5E,cAAc,iDAAiD,CAAC;AAChE,cAAc,2EAA2E,CAAC;AAC1F,cAAc,kDAAkD,CAAC;AACjE,cAAc,wCAAwC,CAAC;AACvD,cAAc,wCAAwC,CAAC;AACvD,cAAc,yCAAyC,CAAC;AACxD,cAAc,2CAA2C,CAAC;AAC1D,cAAc,iCAAiC,CAAC;AAChD,cAAc,iDAAiD,CAAC;AAChE,cAAc,iEAAiE,CAAC;AAChF,cAAc,qDAAqD,CAAC;AACpE,cAAc,2CAA2C,CAAC;AAC1D,cAAc,qDAAqD,CAAC;AACpE,cAAc,+BAA+B,CAAC;AAC9C,cAAc,6BAA6B,CAAC;AAC5C,cAAc,eAAe,CAAC;AAC9B,cAAc,+BAA+B,CAAC;AAC9C,cAAc,gCAAgC,CAAC;AAC/C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,kBAAkB,CAAC;AACjC,cAAc,uBAAuB,CAAC;AACtC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,+BAA+B,CAAC;AAC9C,cAAc,0BAA0B,CAAC;AACzC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,2CAA2C,CAAC;AAC1D,cAAc,8BAA8B,CAAC;AAC7C,cAAc,wBAAwB,CAAC;AACvC,cAAc,gCAAgC,CAAC;AAC/C,cAAc,iCAAiC,CAAC;AAChD,cAAc,2BAA2B,CAAC;AAC1C,cAAc,aAAa,CAAC;AAC5B,cAAc,uBAAuB,CAAC;AACtC,cAAc,+BAA+B,CAAC;AAC9C,cAAc,iDAAiD,CAAC;AAChE,cAAc,iFAAiF,CAAC;AAChG,cAAc,2BAA2B,CAAC;AAC1C,cAAc,uEAAuE,CAAC;AACtF,cAAc,qBAAqB,CAAC;AACpC,cAAc,gCAAgC,CAAC;AAC/C,cAAc,wCAAwC,CAAC;AACvD,cAAc,qCAAqC,CAAC;AACpD,cAAc,+CAA+C,CAAC;AAC9D,cAAc,iBAAiB,CAAC;AAChC,cAAc,4CAA4C,CAAC;AAC3D,cAAc,iDAAiD,CAAC;AAChE,cAAc,6CAA6C,CAAC;AAC5D,cAAc,qCAAqC,CAAC;AACpD,cAAc,yCAAyC,CAAC;AACxD,cAAc,+CAA+C,CAAC;AAC9D,cAAc,8CAA8C,CAAC;AAC7D,cAAc,wBAAwB,CAAC;AACvC,cAAc,+EAA+E,CAAC;AAC9F,cAAc,uBAAuB,CAAC;AACtC,cAAc,4CAA4C,CAAC;AAC3D,cAAc,qCAAqC,CAAC","sourcesContent":["/*\n * Public API Surface of stream-chat-angular\n */\n\nexport * from './lib/chat-client.service';\nexport * from './lib/channel.service';\nexport * from './lib/theme.service';\nexport * from './lib/attachment.service';\nexport * from './lib/attachment-configuration.service';\nexport * from './lib/stream-i18n.service';\nexport * from './lib/avatar/avatar.component';\nexport * from './lib/avatar-placeholder/avatar-placeholder.component';\nexport * from './lib/icon/icon.component';\nexport * from './lib/icon/icon-placeholder/icon-placeholder.component';\nexport * from './lib/icon/loading-indicator/loading-indicator.component';\nexport * from './lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component';\nexport * from './lib/message-actions-box/message-actions-box.component';\nexport * from './lib/channel/channel.component';\nexport * from './lib/channel-header/channel-header.component';\nexport * from './lib/channel-preview/channel-preview.component';\nexport * from './lib/channel-list/channel-list.component';\nexport * from './lib/message/message.component';\nexport * from './lib/parse-date';\nexport * from './lib/list-users';\nexport * from './lib/message-input/message-input.component';\nexport * from './lib/message-bounce-prompt/message-bounce-prompt.component';\nexport * from './lib/message-input/textarea/textarea.component';\nexport * from './lib/message-input/autocomplete-textarea/autocomplete-textarea.component';\nexport * from './lib/message-input/message-input-config.service';\nexport * from './lib/message-input/textarea.directive';\nexport * from './lib/message-input/textarea.interface';\nexport * from './lib/message-input/emoji-input.service';\nexport * from './lib/message-list/message-list.component';\nexport * from './lib/message-list/group-styles';\nexport * from './lib/attachment-list/attachment-list.component';\nexport * from './lib/attachment-preview-list/attachment-preview-list.component';\nexport * from './lib/message-reactions/message-reactions.component';\nexport * from './lib/notification/notification.component';\nexport * from './lib/notification-list/notification-list.component';\nexport * from './lib/thread/thread.component';\nexport * from './lib/modal/modal.component';\nexport * from './lib/read-by';\nexport * from './lib/get-message-translation';\nexport * from './lib/get-channel-display-text';\nexport * from './lib/is-image-attachment';\nexport * from './lib/file-utils';\nexport * from './lib/message-preview';\nexport * from './lib/notification.service';\nexport * from './lib/transliteration.service';\nexport * from './lib/stream-chat.module';\nexport * from './lib/stream-avatar.module';\nexport * from './lib/stream-autocomplete-textarea.module';\nexport * from './lib/stream-textarea.module';\nexport * from './lib/injection-tokens';\nexport * from './lib/custom-templates.service';\nexport * from './lib/message-reactions.service';\nexport * from './lib/date-parser.service';\nexport * from './lib/types';\nexport * from './lib/message.service';\nexport * from './lib/message-actions.service';\nexport * from './lib/voice-recording/voice-recording.component';\nexport * from './lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component';\nexport * from './lib/is-on-separate-date';\nexport * from './lib/message-reactions-selector/message-reactions-selector.component';\nexport * from './lib/channel-query';\nexport * from './lib/virtualized-list.service';\nexport * from './lib/virtualized-message-list.service';\nexport * from './lib/user-list/user-list.component';\nexport * from './lib/paginated-list/paginated-list.component';\nexport * from './lib/is-safari';\nexport * from './lib/voice-recorder/voice-recorder.module';\nexport * from './lib/voice-recorder/amplitude-recorder.service';\nexport * from './lib/voice-recorder/audio-recorder.service';\nexport * from './lib/voice-recorder/media-recorder';\nexport * from './lib/voice-recorder/transcoder.service';\nexport * from './lib/voice-recorder/voice-recorder.component';\nexport * from './lib/voice-recording/voice-recording.module';\nexport * from './lib/icon/icon.module';\nexport * from './lib/voice-recorder//voice-recorder-wavebar/voice-recorder-wavebar.component';\nexport * from './lib/format-duration';\nexport * from './lib/message-input/voice-recorder.service';\nexport * from './lib/voice-recorder/mp3-transcoder';\n"]}