stream-chat-angular 4.66.1 → 5.0.0-v5.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/version.d.ts +1 -1
- package/{esm2015/assets/version.js → esm2020/assets/version.mjs} +2 -2
- package/esm2020/lib/attachment-configuration.service.mjs +185 -0
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +205 -0
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +45 -0
- package/esm2020/lib/attachment.service.mjs +262 -0
- package/esm2020/lib/avatar/avatar.component.mjs +163 -0
- package/esm2020/lib/avatar-placeholder/avatar-placeholder.component.mjs +74 -0
- package/esm2020/lib/channel/channel.component.mjs +46 -0
- package/esm2020/lib/channel-header/channel-header.component.mjs +79 -0
- package/esm2020/lib/channel-list/channel-list-toggle.service.mjs +72 -0
- package/esm2020/lib/channel-list/channel-list.component.mjs +60 -0
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +155 -0
- package/esm2020/lib/channel.service.mjs +1460 -0
- package/esm2020/lib/chat-client.service.mjs +206 -0
- package/{esm2015/lib/custom-templates.service.js → esm2020/lib/custom-templates.service.mjs} +3 -3
- package/{esm2015/lib/date-parser.service.js → esm2020/lib/date-parser.service.mjs} +3 -3
- package/esm2020/lib/edit-message-form/edit-message-form.component.mjs +83 -0
- package/esm2020/lib/get-channel-display-text.mjs +14 -0
- package/esm2020/lib/get-message-translation.mjs +12 -0
- package/esm2020/lib/icon/icon.component.mjs +21 -0
- package/esm2020/lib/icon-placeholder/icon-placeholder.component.mjs +31 -0
- package/esm2020/lib/loading-indicator/loading-indicator.component.mjs +31 -0
- package/esm2020/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +38 -0
- package/esm2020/lib/message/message.component.mjs +418 -0
- package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +130 -0
- package/esm2020/lib/message-actions.service.mjs +119 -0
- package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
- package/esm2020/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +262 -0
- package/{esm2015/lib/message-input/emoji-input.service.js → esm2020/lib/message-input/emoji-input.service.mjs} +3 -3
- package/{esm2015/lib/message-input/message-input-config.service.js → esm2020/lib/message-input/message-input-config.service.mjs} +3 -3
- package/esm2020/lib/message-input/message-input.component.mjs +443 -0
- package/{esm2015/lib/message-input/textarea/textarea.component.js → esm2020/lib/message-input/textarea/textarea.component.mjs} +5 -9
- package/esm2020/lib/message-input/textarea.directive.mjs +89 -0
- package/esm2020/lib/message-list/group-styles.mjs +52 -0
- package/{esm2015/lib/message-list/image-load.service.js → esm2020/lib/message-list/image-load.service.mjs} +3 -3
- package/esm2020/lib/message-list/message-list.component.mjs +699 -0
- package/esm2020/lib/message-preview.mjs +21 -0
- package/esm2020/lib/message-reactions/message-reactions.component.mjs +252 -0
- package/{esm2015/lib/message-reactions.service.js → esm2020/lib/message-reactions.service.mjs} +3 -3
- package/{esm2015/lib/message.service.js → esm2020/lib/message.service.mjs} +4 -4
- package/esm2020/lib/modal/modal.component.mjs +69 -0
- package/esm2020/lib/notification/notification.component.mjs +20 -0
- package/esm2020/lib/notification-list/notification-list.component.mjs +37 -0
- package/esm2020/lib/notification.service.mjs +79 -0
- package/esm2020/lib/read-by.mjs +12 -0
- package/{esm2015/lib/stream-autocomplete-textarea.module.js → esm2020/lib/stream-autocomplete-textarea.module.mjs} +6 -6
- package/{esm2015/lib/stream-avatar.module.js → esm2020/lib/stream-avatar.module.mjs} +5 -5
- package/{esm2015/lib/stream-chat.module.js → esm2020/lib/stream-chat.module.mjs} +14 -16
- package/{esm2015/lib/stream-i18n.service.js → esm2020/lib/stream-i18n.service.mjs} +5 -5
- package/{esm2015/lib/stream-textarea.module.js → esm2020/lib/stream-textarea.module.mjs} +6 -6
- package/esm2020/lib/theme.service.mjs +123 -0
- package/esm2020/lib/thread/thread.component.mjs +51 -0
- package/{esm2015/lib/transliteration.service.js → esm2020/lib/transliteration.service.mjs} +3 -3
- package/esm2020/lib/types.mjs +2 -0
- package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +183 -0
- package/esm2020/lib/voice-recording/voice-recording.component.mjs +102 -0
- package/fesm2015/{stream-chat-angular.js → stream-chat-angular.mjs} +313 -426
- package/fesm2015/stream-chat-angular.mjs.map +1 -0
- package/fesm2020/stream-chat-angular.mjs +7123 -0
- package/fesm2020/stream-chat-angular.mjs.map +1 -0
- package/lib/attachment-list/attachment-list.component.d.ts +3 -3
- package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
- package/lib/attachment.service.d.ts +0 -1
- package/lib/avatar/avatar.component.d.ts +1 -1
- package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
- package/lib/channel/channel.component.d.ts +1 -1
- package/lib/channel-header/channel-header.component.d.ts +1 -1
- package/lib/channel-list/channel-list-toggle.service.d.ts +0 -1
- package/lib/channel-list/channel-list.component.d.ts +1 -1
- package/lib/channel-preview/channel-preview.component.d.ts +1 -1
- package/lib/channel.service.d.ts +7 -7
- package/lib/chat-client.service.d.ts +1 -1
- package/lib/edit-message-form/edit-message-form.component.d.ts +1 -1
- package/lib/get-message-translation.d.ts +1 -1
- package/lib/icon/icon.component.d.ts +1 -1
- package/lib/icon-placeholder/icon-placeholder.component.d.ts +1 -1
- package/lib/loading-indicator/loading-indicator.component.d.ts +1 -1
- package/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.d.ts +1 -1
- package/lib/message/message.component.d.ts +3 -7
- package/lib/message-actions-box/message-actions-box.component.d.ts +2 -4
- package/lib/message-actions.service.d.ts +0 -1
- package/lib/message-bounce-prompt/message-bounce-prompt.component.d.ts +1 -1
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +1 -1
- package/lib/message-input/message-input.component.d.ts +2 -2
- package/lib/message-input/textarea/textarea.component.d.ts +1 -1
- package/lib/message-input/textarea.directive.d.ts +1 -1
- package/lib/message-list/group-styles.d.ts +1 -1
- package/lib/message-list/message-list.component.d.ts +4 -5
- package/lib/message-reactions/message-reactions.component.d.ts +1 -4
- package/lib/message.service.d.ts +0 -1
- package/lib/modal/modal.component.d.ts +1 -1
- package/lib/notification/notification.component.d.ts +1 -1
- package/lib/notification-list/notification-list.component.d.ts +2 -2
- package/lib/notification.service.d.ts +2 -5
- package/lib/stream-chat.module.d.ts +3 -3
- package/lib/theme.service.d.ts +1 -2
- package/lib/thread/thread.component.d.ts +1 -1
- package/lib/types.d.ts +18 -18
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +2 -2
- package/lib/voice-recording/voice-recording.component.d.ts +1 -1
- package/package.json +31 -17
- package/src/assets/styles/v2/css/index.css +1 -1
- package/src/assets/styles/v2/css/index.layout.css +1 -1
- package/src/assets/styles/v2/scss/_base.scss +2 -2
- package/src/assets/version.ts +1 -1
- package/bundles/stream-chat-angular.umd.js +0 -8445
- package/bundles/stream-chat-angular.umd.js.map +0 -1
- package/esm2015/lib/attachment-configuration.service.js +0 -186
- package/esm2015/lib/attachment-list/attachment-list.component.js +0 -209
- package/esm2015/lib/attachment-preview-list/attachment-preview-list.component.js +0 -49
- package/esm2015/lib/attachment.service.js +0 -276
- package/esm2015/lib/avatar/avatar.component.js +0 -172
- package/esm2015/lib/avatar-placeholder/avatar-placeholder.component.js +0 -78
- package/esm2015/lib/channel/channel.component.js +0 -50
- package/esm2015/lib/channel-header/channel-header.component.js +0 -86
- package/esm2015/lib/channel-list/channel-list-toggle.service.js +0 -73
- package/esm2015/lib/channel-list/channel-list.component.js +0 -67
- package/esm2015/lib/channel-preview/channel-preview.component.js +0 -167
- package/esm2015/lib/channel.service.js +0 -1487
- package/esm2015/lib/chat-client.service.js +0 -211
- package/esm2015/lib/edit-message-form/edit-message-form.component.js +0 -87
- package/esm2015/lib/get-channel-display-text.js +0 -15
- package/esm2015/lib/get-message-translation.js +0 -13
- package/esm2015/lib/icon/icon.component.js +0 -25
- package/esm2015/lib/icon-placeholder/icon-placeholder.component.js +0 -35
- package/esm2015/lib/loading-indicator/loading-indicator.component.js +0 -35
- package/esm2015/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.js +0 -42
- package/esm2015/lib/message/message.component.js +0 -436
- package/esm2015/lib/message-actions-box/message-actions-box.component.js +0 -137
- package/esm2015/lib/message-actions.service.js +0 -114
- package/esm2015/lib/message-bounce-prompt/message-bounce-prompt.component.js +0 -80
- package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +0 -262
- package/esm2015/lib/message-input/message-input.component.js +0 -455
- package/esm2015/lib/message-input/textarea.directive.js +0 -90
- package/esm2015/lib/message-list/group-styles.js +0 -53
- package/esm2015/lib/message-list/message-list.component.js +0 -726
- package/esm2015/lib/message-preview.js +0 -7
- package/esm2015/lib/message-reactions/message-reactions.component.js +0 -266
- package/esm2015/lib/modal/modal.component.js +0 -74
- package/esm2015/lib/notification/notification.component.js +0 -24
- package/esm2015/lib/notification-list/notification-list.component.js +0 -38
- package/esm2015/lib/notification.service.js +0 -79
- package/esm2015/lib/read-by.js +0 -13
- package/esm2015/lib/theme.service.js +0 -122
- package/esm2015/lib/thread/thread.component.js +0 -55
- package/esm2015/lib/types.js +0 -2
- package/esm2015/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.js +0 -192
- package/esm2015/lib/voice-recording/voice-recording.component.js +0 -115
- package/fesm2015/stream-chat-angular.js.map +0 -1
- /package/{esm2015/assets/i18n/en.js → esm2020/assets/i18n/en.mjs} +0 -0
- /package/{esm2015/lib/injection-tokens.js → esm2020/lib/injection-tokens.mjs} +0 -0
- /package/{esm2015/lib/is-image-attachment.js → esm2020/lib/is-image-attachment.mjs} +0 -0
- /package/{esm2015/lib/is-image-file.js → esm2020/lib/is-image-file.mjs} +0 -0
- /package/{esm2015/lib/is-on-separate-date.js → esm2020/lib/is-on-separate-date.mjs} +0 -0
- /package/{esm2015/lib/list-users.js → esm2020/lib/list-users.mjs} +0 -0
- /package/{esm2015/lib/message-input/textarea.interface.js → esm2020/lib/message-input/textarea.interface.mjs} +0 -0
- /package/{esm2015/lib/parse-date.js → esm2020/lib/parse-date.mjs} +0 -0
- /package/{esm2015/public-api.js → esm2020/public-api.mjs} +0 -0
- /package/{esm2015/stream-chat-angular.js → esm2020/stream-chat-angular.mjs} +0 -0
- /package/{stream-chat-angular.d.ts → index.d.ts} +0 -0
|
@@ -1,79 +0,0 @@
|
|
|
1
|
-
import { Injectable } from '@angular/core';
|
|
2
|
-
import { BehaviorSubject } from 'rxjs';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
/**
|
|
5
|
-
* The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
|
|
6
|
-
*/
|
|
7
|
-
export class NotificationService {
|
|
8
|
-
constructor() {
|
|
9
|
-
this.notificationsSubject = new BehaviorSubject([]);
|
|
10
|
-
this.notifications$ = this.notificationsSubject.asObservable();
|
|
11
|
-
}
|
|
12
|
-
/**
|
|
13
|
-
* Displays a notification for the given amount of time.
|
|
14
|
-
* @param content The text of the notification or the HTML template for the notification
|
|
15
|
-
* @param type The type of the notification
|
|
16
|
-
* @param timeout The number of milliseconds while the notification should be visible
|
|
17
|
-
* @param translateParams Translation parameters for the `content` (for text notifications)
|
|
18
|
-
* @param templateContext The input of the notification template (for HTML notifications)
|
|
19
|
-
* @returns A method to clear the notification (before the timeout).
|
|
20
|
-
*/
|
|
21
|
-
addTemporaryNotification(content, type = 'error', timeout = 5000, translateParams, templateContext) {
|
|
22
|
-
const notification = this.createNotification(content, type, translateParams, templateContext);
|
|
23
|
-
const id = setTimeout(() => this.removeNotification(notification.id), timeout);
|
|
24
|
-
notification.dismissFn = () => {
|
|
25
|
-
clearTimeout(id);
|
|
26
|
-
this.removeNotification(notification.id);
|
|
27
|
-
};
|
|
28
|
-
this.notificationsSubject.next([
|
|
29
|
-
...this.notificationsSubject.getValue(),
|
|
30
|
-
notification,
|
|
31
|
-
]);
|
|
32
|
-
return notification.dismissFn;
|
|
33
|
-
}
|
|
34
|
-
/**
|
|
35
|
-
* Displays a notification, that will be visible until it's removed.
|
|
36
|
-
* @param content The text of the notification or the HTML template for the notification
|
|
37
|
-
* @param type The type of the notification
|
|
38
|
-
* @param translateParams Translation parameters for the `content` (for text notifications)
|
|
39
|
-
* @param templateContext The input of the notification template (for HTML notifications)
|
|
40
|
-
* @returns A method to clear the notification.
|
|
41
|
-
*/
|
|
42
|
-
addPermanentNotification(content, type = 'error', translateParams, templateContext) {
|
|
43
|
-
const notification = this.createNotification(content, type, translateParams, templateContext);
|
|
44
|
-
this.notificationsSubject.next([
|
|
45
|
-
...this.notificationsSubject.getValue(),
|
|
46
|
-
notification,
|
|
47
|
-
]);
|
|
48
|
-
return notification.dismissFn;
|
|
49
|
-
}
|
|
50
|
-
createNotification(content, type, translateParams, templateContext) {
|
|
51
|
-
const id = new Date().getTime().toString() + Math.random().toString();
|
|
52
|
-
return {
|
|
53
|
-
id,
|
|
54
|
-
[typeof content === 'string' ? 'text' : 'template']: content,
|
|
55
|
-
type,
|
|
56
|
-
translateParams,
|
|
57
|
-
templateContext,
|
|
58
|
-
dismissFn: () => this.removeNotification(id),
|
|
59
|
-
};
|
|
60
|
-
}
|
|
61
|
-
removeNotification(id) {
|
|
62
|
-
const notifications = this.notificationsSubject.getValue();
|
|
63
|
-
const index = notifications.findIndex((n) => n.id === id);
|
|
64
|
-
if (index === -1) {
|
|
65
|
-
return;
|
|
66
|
-
}
|
|
67
|
-
notifications.splice(index, 1);
|
|
68
|
-
this.notificationsSubject.next([...notifications]);
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
NotificationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: NotificationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
72
|
-
NotificationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: NotificationService, providedIn: 'root' });
|
|
73
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: NotificationService, decorators: [{
|
|
74
|
-
type: Injectable,
|
|
75
|
-
args: [{
|
|
76
|
-
providedIn: 'root',
|
|
77
|
-
}]
|
|
78
|
-
}], ctorParameters: function () { return []; } });
|
|
79
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notification.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/notification.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAe,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;;AAGnD;;GAEG;AAIH,MAAM,OAAO,mBAAmB;IAO9B;QAFQ,yBAAoB,GAAG,IAAI,eAAe,CAAwB,EAAE,CAAC,CAAC;QAG5E,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;IACjE,CAAC;IAED;;;;;;;;OAQG;IACH,wBAAwB,CACtB,OAAgC,EAChC,OAAyB,OAAO,EAChC,UAAkB,IAAI,EACtB,eAAwB,EACxB,eAAmB;QAEnB,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,CAC1C,OAAO,EACP,IAAI,EACJ,eAAe,EACf,eAAe,CAChB,CAAC;QACF,MAAM,EAAE,GAAG,UAAU,CACnB,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,EAAE,CAAC,EAC9C,OAAO,CACR,CAAC;QACF,YAAY,CAAC,SAAS,GAAG,GAAG,EAAE;YAC5B,YAAY,CAAC,EAAE,CAAC,CAAC;YACjB,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;QAC3C,CAAC,CAAC;QACF,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC;YAC7B,GAAG,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAAE;YACvC,YAAY;SACb,CAAC,CAAC;QAEH,OAAO,YAAY,CAAC,SAAS,CAAC;IAChC,CAAC;IAED;;;;;;;OAOG;IACH,wBAAwB,CAMtB,OAAgC,EAChC,OAAyB,OAAO,EAChC,eAAwB,EACxB,eAAmB;QAEnB,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,CAC1C,OAAO,EACP,IAAI,EACJ,eAAe,EACf,eAAe,CAChB,CAAC;QACF,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC;YAC7B,GAAG,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAAE;YACvC,YAAY;SACb,CAAC,CAAC;QAEH,OAAO,YAAY,CAAC,SAAS,CAAC;IAChC,CAAC;IAEO,kBAAkB,CACxB,OAAgC,EAChC,IAAsB,EACtB,eAAwB,EACxB,eAAmB;QAEnB,MAAM,EAAE,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC,QAAQ,EAAE,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC;QACtE,OAAO;YACL,EAAE;YACF,CAAC,OAAO,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,EAAE,OAAO;YAC5D,IAAI;YACJ,eAAe;YACf,eAAe;YACf,SAAS,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,EAAE,CAAC;SAC7C,CAAC;IACJ,CAAC;IAEO,kBAAkB,CAAC,EAAU;QACnC,MAAM,aAAa,GAAG,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAAE,CAAC;QAC3D,MAAM,KAAK,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;QAC1D,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE;YAChB,OAAO;SACR;QACD,aAAa,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAC/B,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC,GAAG,aAAa,CAAC,CAAC,CAAC;IACrD,CAAC;;gHA3GU,mBAAmB;oHAAnB,mBAAmB,cAFlB,MAAM;2FAEP,mBAAmB;kBAH/B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable, TemplateRef } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { NotificationPayload, NotificationType } from './types';\n\n/**\n * The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class NotificationService {\n  /**\n   * Emits the currently active [notifications](https://github.com/GetStream/stream-chat-angular/blob/master/projects/stream-chat-angular/src/lib/notification.service.ts).\n   */\n  notifications$: Observable<NotificationPayload[]>;\n  private notificationsSubject = new BehaviorSubject<NotificationPayload[]>([]);\n\n  constructor() {\n    this.notifications$ = this.notificationsSubject.asObservable();\n  }\n\n  /**\n   * Displays a notification for the given amount of time.\n   * @param content The text of the notification or the HTML template for the notification\n   * @param type The type of the notification\n   * @param timeout The number of milliseconds while the notification should be visible\n   * @param translateParams Translation parameters for the `content` (for text notifications)\n   * @param templateContext The input of the notification template (for HTML notifications)\n   * @returns A method to clear the notification (before the timeout).\n   */\n  addTemporaryNotification<T>(\n    content: string | TemplateRef<T>,\n    type: NotificationType = 'error',\n    timeout: number = 5000,\n    translateParams?: Object,\n    templateContext?: T\n  ) {\n    const notification = this.createNotification<T>(\n      content,\n      type,\n      translateParams,\n      templateContext\n    );\n    const id = setTimeout(\n      () => this.removeNotification(notification.id),\n      timeout\n    );\n    notification.dismissFn = () => {\n      clearTimeout(id);\n      this.removeNotification(notification.id);\n    };\n    this.notificationsSubject.next([\n      ...this.notificationsSubject.getValue(),\n      notification,\n    ]);\n\n    return notification.dismissFn;\n  }\n\n  /**\n   * Displays a notification, that will be visible until it's removed.\n   * @param content The text of the notification or the HTML template for the notification\n   * @param type The type of the notification\n   * @param translateParams Translation parameters for the `content` (for text notifications)\n   * @param templateContext The input of the notification template (for HTML notifications)\n   * @returns A method to clear the notification.\n   */\n  addPermanentNotification<\n    T = {\n      [key: string]: any;\n      dismissFn: () => {};\n    }\n  >(\n    content: string | TemplateRef<T>,\n    type: NotificationType = 'error',\n    translateParams?: Object,\n    templateContext?: T\n  ) {\n    const notification = this.createNotification<T>(\n      content,\n      type,\n      translateParams,\n      templateContext\n    );\n    this.notificationsSubject.next([\n      ...this.notificationsSubject.getValue(),\n      notification,\n    ]);\n\n    return notification.dismissFn;\n  }\n\n  private createNotification<T>(\n    content: string | TemplateRef<T>,\n    type: NotificationType,\n    translateParams?: Object,\n    templateContext?: T\n  ) {\n    const id = new Date().getTime().toString() + Math.random().toString();\n    return {\n      id,\n      [typeof content === 'string' ? 'text' : 'template']: content,\n      type,\n      translateParams,\n      templateContext,\n      dismissFn: () => this.removeNotification(id),\n    };\n  }\n\n  private removeNotification(id: string) {\n    const notifications = this.notificationsSubject.getValue();\n    const index = notifications.findIndex((n) => n.id === id);\n    if (index === -1) {\n      return;\n    }\n    notifications.splice(index, 1);\n    this.notificationsSubject.next([...notifications]);\n  }\n}\n"]}
|
package/esm2015/lib/read-by.js
DELETED
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
export const getReadBy = (message, channel) => {
|
|
2
|
-
const readBy = [];
|
|
3
|
-
Object.keys(channel.state.read).forEach((key) => {
|
|
4
|
-
var _a;
|
|
5
|
-
if (channel.state.read[key].last_read.getTime() >=
|
|
6
|
-
message.created_at.getTime() &&
|
|
7
|
-
((_a = message.user) === null || _a === void 0 ? void 0 : _a.id) !== key) {
|
|
8
|
-
readBy.push(channel.state.read[key].user);
|
|
9
|
-
}
|
|
10
|
-
});
|
|
11
|
-
return readBy;
|
|
12
|
-
};
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVhZC1ieS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9yZWFkLWJ5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUdBLE1BQU0sQ0FBQyxNQUFNLFNBQVMsR0FBRyxDQUd2QixPQUFpQyxFQUNqQyxPQUFtQixFQUNuQixFQUFFO0lBQ0YsTUFBTSxNQUFNLEdBQW1CLEVBQUUsQ0FBQztJQUNsQyxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsR0FBRyxFQUFFLEVBQUU7O1FBQzlDLElBQ0UsT0FBTyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsU0FBUyxDQUFDLE9BQU8sRUFBRTtZQUN6QyxPQUFPLENBQUMsVUFBVSxDQUFDLE9BQU8sRUFBRTtZQUM5QixDQUFBLE1BQUEsT0FBTyxDQUFDLElBQUksMENBQUUsRUFBRSxNQUFLLEdBQUcsRUFDeEI7WUFDQSxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQzNDO0lBQ0gsQ0FBQyxDQUFDLENBQUM7SUFFSCxPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFubmVsLCBGb3JtYXRNZXNzYWdlUmVzcG9uc2UsIFVzZXJSZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3MgfSBmcm9tICcuL3R5cGVzJztcblxuZXhwb3J0IGNvbnN0IGdldFJlYWRCeSA9IDxcbiAgVCBleHRlbmRzIERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3MgPSBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzXG4+KFxuICBtZXNzYWdlOiBGb3JtYXRNZXNzYWdlUmVzcG9uc2U8VD4sXG4gIGNoYW5uZWw6IENoYW5uZWw8VD5cbikgPT4ge1xuICBjb25zdCByZWFkQnk6IFVzZXJSZXNwb25zZVtdID0gW107XG4gIE9iamVjdC5rZXlzKGNoYW5uZWwuc3RhdGUucmVhZCkuZm9yRWFjaCgoa2V5KSA9PiB7XG4gICAgaWYgKFxuICAgICAgY2hhbm5lbC5zdGF0ZS5yZWFkW2tleV0ubGFzdF9yZWFkLmdldFRpbWUoKSA+PVxuICAgICAgICBtZXNzYWdlLmNyZWF0ZWRfYXQuZ2V0VGltZSgpICYmXG4gICAgICBtZXNzYWdlLnVzZXI/LmlkICE9PSBrZXlcbiAgICApIHtcbiAgICAgIHJlYWRCeS5wdXNoKGNoYW5uZWwuc3RhdGUucmVhZFtrZXldLnVzZXIpO1xuICAgIH1cbiAgfSk7XG5cbiAgcmV0dXJuIHJlYWRCeTtcbn07XG4iXX0=
|
|
@@ -1,122 +0,0 @@
|
|
|
1
|
-
import { Injectable } from '@angular/core';
|
|
2
|
-
import { BehaviorSubject } from 'rxjs';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
/**
|
|
5
|
-
* The `ThemeService` can be used to change the theme of the chat UI and to customize the theme. Our [theming guide](../theming/introduction.mdx) gives a complete overview about the topic.
|
|
6
|
-
*/
|
|
7
|
-
export class ThemeService {
|
|
8
|
-
constructor() {
|
|
9
|
-
/**
|
|
10
|
-
* A Subject that can be used to get or set the currently active theme.
|
|
11
|
-
*/
|
|
12
|
-
this.theme$ = new BehaviorSubject('light');
|
|
13
|
-
this.defaultDarkModeVariables = {
|
|
14
|
-
'--bg-gradient-end': '#101214',
|
|
15
|
-
'--bg-gradient-start': '#070a0d',
|
|
16
|
-
'--black': '#ffffff',
|
|
17
|
-
'--blue-alice': '#00193d',
|
|
18
|
-
'--border': '#141924',
|
|
19
|
-
'--button-background': '#ffffff',
|
|
20
|
-
'--button-text': '#005fff',
|
|
21
|
-
'--grey': '#7a7a7a',
|
|
22
|
-
'--grey-gainsboro': '#2d2f2f',
|
|
23
|
-
'--grey-whisper': '#1c1e22',
|
|
24
|
-
'--modal-shadow': '#000000',
|
|
25
|
-
'--overlay': '#00000066',
|
|
26
|
-
'--overlay-dark': '#ffffffcc',
|
|
27
|
-
'--shadow-icon': '#00000080',
|
|
28
|
-
'--targetedMessageBackground': '#302d22',
|
|
29
|
-
'--transparent': 'transparent',
|
|
30
|
-
'--white': '#101418',
|
|
31
|
-
'--white-smoke': '#13151b',
|
|
32
|
-
'--white-snow': '#070a0d',
|
|
33
|
-
};
|
|
34
|
-
this.variablesToDelete = [];
|
|
35
|
-
this.theme$.subscribe((theme) => {
|
|
36
|
-
const darkVariables = this.customDarkThemeVariables
|
|
37
|
-
? Object.assign(Object.assign({}, this.defaultDarkModeVariables), this.customDarkThemeVariables) : this.defaultDarkModeVariables;
|
|
38
|
-
const lightVariables = this.customLightThemeVariables
|
|
39
|
-
? this.customLightThemeVariables
|
|
40
|
-
: {};
|
|
41
|
-
this.variablesToDelete.forEach((variables) => this.deleteVariables(variables));
|
|
42
|
-
if (theme === 'dark') {
|
|
43
|
-
this.deleteVariables(lightVariables);
|
|
44
|
-
this.setVariables(darkVariables);
|
|
45
|
-
}
|
|
46
|
-
else {
|
|
47
|
-
this.deleteVariables(darkVariables);
|
|
48
|
-
this.setVariables(lightVariables);
|
|
49
|
-
}
|
|
50
|
-
});
|
|
51
|
-
this.themeVersion = (getComputedStyle(document.documentElement)
|
|
52
|
-
.getPropertyValue('--str-chat__theme-version')
|
|
53
|
-
.replace(' ', '') || '1');
|
|
54
|
-
}
|
|
55
|
-
/**
|
|
56
|
-
* A getter that returns the currently set custom light theme variables.
|
|
57
|
-
* @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)
|
|
58
|
-
* @returns An object where the keys are theme variables, and the values are the currently set CSS values.
|
|
59
|
-
*/
|
|
60
|
-
get customLightThemeVariables() {
|
|
61
|
-
return this._customLightThemeVariables;
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* A setter that can be used to overwrite the values of the CSS theme variables of the light theme.
|
|
65
|
-
* @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)
|
|
66
|
-
* @param variables An object where the keys are theme variables, and the values are CSS values.
|
|
67
|
-
*/
|
|
68
|
-
set customLightThemeVariables(variables) {
|
|
69
|
-
const prevVariables = this.customLightThemeVariables;
|
|
70
|
-
if (prevVariables) {
|
|
71
|
-
this.variablesToDelete.push(prevVariables);
|
|
72
|
-
}
|
|
73
|
-
this._customLightThemeVariables = variables;
|
|
74
|
-
if (this.theme$.getValue() === 'light') {
|
|
75
|
-
this.theme$.next('light');
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
/**
|
|
79
|
-
* A getter that returns the currently set custom dark theme variables.
|
|
80
|
-
* @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)
|
|
81
|
-
* @returns An object where the keys are theme variables, and the values are the currently set CSS values.
|
|
82
|
-
*/
|
|
83
|
-
get customDarkThemeVariables() {
|
|
84
|
-
return this._customDarkThemeVariables;
|
|
85
|
-
}
|
|
86
|
-
/**
|
|
87
|
-
* A setter that can be used to overwrite the values of the CSS theme variables of the dark theme.
|
|
88
|
-
* @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)
|
|
89
|
-
* @param variables An object where the keys are theme variables, and the values are CSS values.
|
|
90
|
-
*/
|
|
91
|
-
set customDarkThemeVariables(variables) {
|
|
92
|
-
const prevVariables = this.customDarkThemeVariables;
|
|
93
|
-
if (prevVariables) {
|
|
94
|
-
this.variablesToDelete.push(prevVariables);
|
|
95
|
-
}
|
|
96
|
-
this._customDarkThemeVariables = variables;
|
|
97
|
-
if (this.theme$.getValue() === 'dark') {
|
|
98
|
-
this.theme$.next('dark');
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
deleteVariables(variables) {
|
|
102
|
-
if (!variables) {
|
|
103
|
-
return;
|
|
104
|
-
}
|
|
105
|
-
Object.keys(variables).forEach((key) => document.documentElement.style.setProperty(key, null));
|
|
106
|
-
}
|
|
107
|
-
setVariables(variables) {
|
|
108
|
-
if (!variables) {
|
|
109
|
-
return;
|
|
110
|
-
}
|
|
111
|
-
Object.keys(variables).forEach((key) => document.documentElement.style.setProperty(key, variables[key]));
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
ThemeService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ThemeService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
115
|
-
ThemeService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ThemeService, providedIn: 'root' });
|
|
116
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ThemeService, decorators: [{
|
|
117
|
-
type: Injectable,
|
|
118
|
-
args: [{
|
|
119
|
-
providedIn: 'root',
|
|
120
|
-
}]
|
|
121
|
-
}], ctorParameters: function () { return []; } });
|
|
122
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/theme.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\n\nexport type Theme = 'light' | 'dark';\n\n/**\n * The `ThemeService` can be used to change the theme of the chat UI and to customize the theme. Our [theming guide](../theming/introduction.mdx) gives a complete overview about the topic.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class ThemeService {\n  /**\n   * A Subject that can be used to get or set the currently active theme.\n   */\n  theme$ = new BehaviorSubject<Theme | string>('light');\n  /**\n   * Stream chat theme version - this is used internally by some UI components of the SDK, integrators shouldn't need to use this variable\n   */\n  themeVersion: '1' | '2';\n  private _customLightThemeVariables: { [key: string]: string } | undefined;\n  private _customDarkThemeVariables: { [key: string]: string } | undefined;\n  private defaultDarkModeVariables = {\n    '--bg-gradient-end': '#101214',\n    '--bg-gradient-start': '#070a0d',\n    '--black': '#ffffff',\n    '--blue-alice': '#00193d',\n    '--border': '#141924',\n    '--button-background': '#ffffff',\n    '--button-text': '#005fff',\n    '--grey': '#7a7a7a',\n    '--grey-gainsboro': '#2d2f2f',\n    '--grey-whisper': '#1c1e22',\n    '--modal-shadow': '#000000',\n    '--overlay': '#00000066', // 66 = 40% opacity\n    '--overlay-dark': '#ffffffcc', // CC = 80% opacity\n    '--shadow-icon': '#00000080', // 80 = 50% opacity\n    '--targetedMessageBackground': '#302d22',\n    '--transparent': 'transparent',\n    '--white': '#101418',\n    '--white-smoke': '#13151b',\n    '--white-snow': '#070a0d',\n  };\n  private variablesToDelete: { [key: string]: string }[] = [];\n\n  constructor() {\n    this.theme$.subscribe((theme) => {\n      const darkVariables = this.customDarkThemeVariables\n        ? { ...this.defaultDarkModeVariables, ...this.customDarkThemeVariables }\n        : this.defaultDarkModeVariables;\n      const lightVariables = this.customLightThemeVariables\n        ? this.customLightThemeVariables\n        : {};\n      this.variablesToDelete.forEach((variables) =>\n        this.deleteVariables(variables)\n      );\n      if (theme === 'dark') {\n        this.deleteVariables(lightVariables);\n        this.setVariables(darkVariables);\n      } else {\n        this.deleteVariables(darkVariables);\n        this.setVariables(lightVariables);\n      }\n    });\n\n    this.themeVersion = (getComputedStyle(document.documentElement)\n      .getPropertyValue('--str-chat__theme-version')\n      .replace(' ', '') || '1') as '1' | '2';\n  }\n\n  /**\n   * A getter that returns the currently set custom light theme variables.\n   * @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)\n   * @returns An object where the keys are theme variables, and the values are the currently set CSS values.\n   */\n  get customLightThemeVariables() {\n    return this._customLightThemeVariables;\n  }\n\n  /**\n   * A setter that can be used to overwrite the values of the CSS theme variables of the light theme.\n   * @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)\n   * @param variables An object where the keys are theme variables, and the values are CSS values.\n   */\n  set customLightThemeVariables(\n    variables: { [key: string]: string } | undefined\n  ) {\n    const prevVariables = this.customLightThemeVariables;\n    if (prevVariables) {\n      this.variablesToDelete.push(prevVariables);\n    }\n    this._customLightThemeVariables = variables;\n    if (this.theme$.getValue() === 'light') {\n      this.theme$.next('light');\n    }\n  }\n\n  /**\n   * A getter that returns the currently set custom dark theme variables.\n   * @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)\n   * @returns An object where the keys are theme variables, and the values are the currently set CSS values.\n   */\n  get customDarkThemeVariables() {\n    return this._customDarkThemeVariables;\n  }\n\n  /**\n   * A setter that can be used to overwrite the values of the CSS theme variables of the dark theme.\n   * @deprecated Only use with [theme v1](../concepts/theming-and-css.mdx)\n   * @param variables An object where the keys are theme variables, and the values are CSS values.\n   */\n  set customDarkThemeVariables(\n    variables: { [key: string]: string } | undefined\n  ) {\n    const prevVariables = this.customDarkThemeVariables;\n    if (prevVariables) {\n      this.variablesToDelete.push(prevVariables);\n    }\n    this._customDarkThemeVariables = variables;\n    if (this.theme$.getValue() === 'dark') {\n      this.theme$.next('dark');\n    }\n  }\n\n  private deleteVariables(variables: { [key: string]: string } | undefined) {\n    if (!variables) {\n      return;\n    }\n    Object.keys(variables).forEach((key) =>\n      document.documentElement.style.setProperty(key, null)\n    );\n  }\n\n  private setVariables(variables: { [key: string]: string } | undefined) {\n    if (!variables) {\n      return;\n    }\n    Object.keys(variables).forEach((key) =>\n      document.documentElement.style.setProperty(key, variables[key])\n    );\n  }\n}\n"]}
|
|
@@ -1,55 +0,0 @@
|
|
|
1
|
-
import { Component, HostBinding } from '@angular/core';
|
|
2
|
-
import { getChannelDisplayText } from '../get-channel-display-text';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
import * as i1 from "../custom-templates.service";
|
|
5
|
-
import * as i2 from "../channel.service";
|
|
6
|
-
import * as i3 from "../chat-client.service";
|
|
7
|
-
import * as i4 from "../icon-placeholder/icon-placeholder.component";
|
|
8
|
-
import * as i5 from "@angular/common";
|
|
9
|
-
import * as i6 from "@ngx-translate/core";
|
|
10
|
-
/**
|
|
11
|
-
* The `Thread` component represents a [message thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript), it is a container component that displays a thread with a header, [`MessageList`](./MessageListComponent.mdx) and [`MessageInput`](./MessageInputComponent.mdx) components.
|
|
12
|
-
*/
|
|
13
|
-
export class ThreadComponent {
|
|
14
|
-
constructor(customTemplatesService, channelService, chatClientService) {
|
|
15
|
-
this.customTemplatesService = customTemplatesService;
|
|
16
|
-
this.channelService = channelService;
|
|
17
|
-
this.chatClientService = chatClientService;
|
|
18
|
-
this.class = 'str-chat__thread';
|
|
19
|
-
this.subscriptions = [];
|
|
20
|
-
this.subscriptions.push(this.channelService.activeParentMessage$.subscribe((parentMessage) => (this.parentMessage = parentMessage)));
|
|
21
|
-
this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => (this.channel = channel)));
|
|
22
|
-
}
|
|
23
|
-
ngOnDestroy() {
|
|
24
|
-
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
25
|
-
}
|
|
26
|
-
getThreadHeaderContext() {
|
|
27
|
-
return {
|
|
28
|
-
parentMessage: this.parentMessage,
|
|
29
|
-
closeThreadHandler: () => this.closeThread(),
|
|
30
|
-
};
|
|
31
|
-
}
|
|
32
|
-
closeThread() {
|
|
33
|
-
void this.channelService.setAsActiveParentMessage(undefined);
|
|
34
|
-
}
|
|
35
|
-
get channelName() {
|
|
36
|
-
if (!this.channel || !this.chatClientService.chatClient.user) {
|
|
37
|
-
return '';
|
|
38
|
-
}
|
|
39
|
-
return getChannelDisplayText(this.channel, this.chatClientService.chatClient.user);
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
ThreadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ThreadComponent, deps: [{ token: i1.CustomTemplatesService }, { token: i2.ChannelService }, { token: i3.ChatClientService }], target: i0.ɵɵFactoryTarget.Component });
|
|
43
|
-
ThreadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: ThreadComponent, selector: "stream-thread", host: { properties: { "class": "this.class" } }, ngImport: i0, template: "<div class=\"str-chat__thread-container\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.threadHeaderTemplate$ | async) ||\n defaultThreadHeader;\n context: getThreadHeaderContext()\n \"\n ></ng-container>\n <ng-content></ng-content>\n</div>\n\n<ng-template\n #defaultThreadHeader\n let-parentMessage=\"parentMessage\"\n let-closeThreadHandler=\"closeThreadHandler\"\n>\n <div class=\"str-chat__thread-header\">\n <div class=\"str-chat__thread-header-details\">\n <div class=\"str-chat__thread-header-name\" translate>\n streamChat.Thread\n </div>\n <div\n class=\"str-chat__thread-header-channel-name\"\n data-testid=\"channel-name\"\n >\n {{ channelName }}\n </div>\n </div>\n <button\n class=\"str-chat__square-button str-chat__close-thread-button\"\n data-testid=\"close-button\"\n (click)=\"closeThreadHandler()\"\n >\n <stream-icon-placeholder\n icon=\"close-no-outline\"\n ></stream-icon-placeholder>\n </button>\n </div>\n</ng-template>\n", components: [{ type: i4.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon", "size"] }], directives: [{ type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i6.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], pipes: { "async": i5.AsyncPipe } });
|
|
44
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ThreadComponent, decorators: [{
|
|
45
|
-
type: Component,
|
|
46
|
-
args: [{
|
|
47
|
-
selector: 'stream-thread',
|
|
48
|
-
templateUrl: './thread.component.html',
|
|
49
|
-
styles: [],
|
|
50
|
-
}]
|
|
51
|
-
}], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ChannelService }, { type: i3.ChatClientService }]; }, propDecorators: { class: [{
|
|
52
|
-
type: HostBinding,
|
|
53
|
-
args: ['class']
|
|
54
|
-
}] } });
|
|
55
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGhyZWFkLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi90aHJlYWQvdGhyZWFkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi90aHJlYWQvdGhyZWFkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsV0FBVyxFQUFhLE1BQU0sZUFBZSxDQUFDO0FBTWxFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDOzs7Ozs7OztBQU9wRTs7R0FFRztBQU1ILE1BQU0sT0FBTyxlQUFlO0lBTTFCLFlBQ1Msc0JBQThDLEVBQzdDLGNBQThCLEVBQzlCLGlCQUFvQztRQUZyQywyQkFBc0IsR0FBdEIsc0JBQXNCLENBQXdCO1FBQzdDLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtRQUM5QixzQkFBaUIsR0FBakIsaUJBQWlCLENBQW1CO1FBUmhCLFVBQUssR0FBRyxrQkFBa0IsQ0FBQztRQUdqRCxrQkFBYSxHQUFtQixFQUFFLENBQUM7UUFPekMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQ3JCLElBQUksQ0FBQyxjQUFjLENBQUMsb0JBQW9CLENBQUMsU0FBUyxDQUNoRCxDQUFDLGFBQWEsRUFBRSxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsYUFBYSxHQUFHLGFBQWEsQ0FBQyxDQUN4RCxDQUNGLENBQUM7UUFDRixJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FDckIsSUFBSSxDQUFDLGNBQWMsQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUMxQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTyxHQUFHLE9BQU8sQ0FBQyxDQUN0QyxDQUNGLENBQUM7SUFDSixDQUFDO0lBRUQsV0FBVztRQUNULElBQUksQ0FBQyxhQUFhLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBRUQsc0JBQXNCO1FBQ3BCLE9BQU87WUFDTCxhQUFhLEVBQUUsSUFBSSxDQUFDLGFBQWE7WUFDakMsa0JBQWtCLEVBQUUsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRTtTQUM3QyxDQUFDO0lBQ0osQ0FBQztJQUVELFdBQVc7UUFDVCxLQUFLLElBQUksQ0FBQyxjQUFjLENBQUMsd0JBQXdCLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDL0QsQ0FBQztJQUVELElBQUksV0FBVztRQUNiLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLFVBQVUsQ0FBQyxJQUFJLEVBQUU7WUFDNUQsT0FBTyxFQUFFLENBQUM7U0FDWDtRQUNELE9BQU8scUJBQXFCLENBQzFCLElBQUksQ0FBQyxPQUFPLEVBQ1osSUFBSSxDQUFDLGlCQUFpQixDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQ3ZDLENBQUM7SUFDSixDQUFDOzs0R0E5Q1UsZUFBZTtnR0FBZixlQUFlLHNHQ3JCNUIsMGtDQXVDQTsyRkRsQmEsZUFBZTtrQkFMM0IsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsZUFBZTtvQkFDekIsV0FBVyxFQUFFLHlCQUF5QjtvQkFDdEMsTUFBTSxFQUFFLEVBQUU7aUJBQ1g7MEtBRStCLEtBQUs7c0JBQWxDLFdBQVc7dUJBQUMsT0FBTyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSG9zdEJpbmRpbmcsIE9uRGVzdHJveSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgU3Vic2NyaXB0aW9uIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBDaGFubmVsIH0gZnJvbSAnc3RyZWFtLWNoYXQnO1xuaW1wb3J0IHsgQ2hhdENsaWVudFNlcnZpY2UgfSBmcm9tICcuLi9jaGF0LWNsaWVudC5zZXJ2aWNlJztcbmltcG9ydCB7IENoYW5uZWxTZXJ2aWNlIH0gZnJvbSAnLi4vY2hhbm5lbC5zZXJ2aWNlJztcbmltcG9ydCB7IEN1c3RvbVRlbXBsYXRlc1NlcnZpY2UgfSBmcm9tICcuLi9jdXN0b20tdGVtcGxhdGVzLnNlcnZpY2UnO1xuaW1wb3J0IHsgZ2V0Q2hhbm5lbERpc3BsYXlUZXh0IH0gZnJvbSAnLi4vZ2V0LWNoYW5uZWwtZGlzcGxheS10ZXh0JztcbmltcG9ydCB7XG4gIERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3MsXG4gIFN0cmVhbU1lc3NhZ2UsXG4gIFRocmVhZEhlYWRlckNvbnRleHQsXG59IGZyb20gJy4uL3R5cGVzJztcblxuLyoqXG4gKiBUaGUgYFRocmVhZGAgY29tcG9uZW50IHJlcHJlc2VudHMgYSBbbWVzc2FnZSB0aHJlYWRdKGh0dHBzOi8vZ2V0c3RyZWFtLmlvL2NoYXQvZG9jcy9qYXZhc2NyaXB0L3RocmVhZHMvP2xhbmd1YWdlPWphdmFzY3JpcHQpLCBpdCBpcyBhIGNvbnRhaW5lciBjb21wb25lbnQgdGhhdCBkaXNwbGF5cyBhIHRocmVhZCB3aXRoIGEgaGVhZGVyLCBbYE1lc3NhZ2VMaXN0YF0oLi9NZXNzYWdlTGlzdENvbXBvbmVudC5tZHgpIGFuZCBbYE1lc3NhZ2VJbnB1dGBdKC4vTWVzc2FnZUlucHV0Q29tcG9uZW50Lm1keCkgY29tcG9uZW50cy5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc3RyZWFtLXRocmVhZCcsXG4gIHRlbXBsYXRlVXJsOiAnLi90aHJlYWQuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxufSlcbmV4cG9ydCBjbGFzcyBUaHJlYWRDb21wb25lbnQgaW1wbGVtZW50cyBPbkRlc3Ryb3kge1xuICBASG9zdEJpbmRpbmcoJ2NsYXNzJykgcHJpdmF0ZSBjbGFzcyA9ICdzdHItY2hhdF9fdGhyZWFkJztcbiAgcGFyZW50TWVzc2FnZTogU3RyZWFtTWVzc2FnZSB8IHVuZGVmaW5lZDtcbiAgY2hhbm5lbDogQ2hhbm5lbDxEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzPiB8IHVuZGVmaW5lZDtcbiAgcHJpdmF0ZSBzdWJzY3JpcHRpb25zOiBTdWJzY3JpcHRpb25bXSA9IFtdO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBjdXN0b21UZW1wbGF0ZXNTZXJ2aWNlOiBDdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLFxuICAgIHByaXZhdGUgY2hhbm5lbFNlcnZpY2U6IENoYW5uZWxTZXJ2aWNlLFxuICAgIHByaXZhdGUgY2hhdENsaWVudFNlcnZpY2U6IENoYXRDbGllbnRTZXJ2aWNlXG4gICkge1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jaGFubmVsU2VydmljZS5hY3RpdmVQYXJlbnRNZXNzYWdlJC5zdWJzY3JpYmUoXG4gICAgICAgIChwYXJlbnRNZXNzYWdlKSA9PiAodGhpcy5wYXJlbnRNZXNzYWdlID0gcGFyZW50TWVzc2FnZSlcbiAgICAgIClcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jaGFubmVsU2VydmljZS5hY3RpdmVDaGFubmVsJC5zdWJzY3JpYmUoXG4gICAgICAgIChjaGFubmVsKSA9PiAodGhpcy5jaGFubmVsID0gY2hhbm5lbClcbiAgICAgIClcbiAgICApO1xuICB9XG5cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLmZvckVhY2goKHMpID0+IHMudW5zdWJzY3JpYmUoKSk7XG4gIH1cblxuICBnZXRUaHJlYWRIZWFkZXJDb250ZXh0KCk6IFRocmVhZEhlYWRlckNvbnRleHQge1xuICAgIHJldHVybiB7XG4gICAgICBwYXJlbnRNZXNzYWdlOiB0aGlzLnBhcmVudE1lc3NhZ2UsXG4gICAgICBjbG9zZVRocmVhZEhhbmRsZXI6ICgpID0+IHRoaXMuY2xvc2VUaHJlYWQoKSxcbiAgICB9O1xuICB9XG5cbiAgY2xvc2VUaHJlYWQoKSB7XG4gICAgdm9pZCB0aGlzLmNoYW5uZWxTZXJ2aWNlLnNldEFzQWN0aXZlUGFyZW50TWVzc2FnZSh1bmRlZmluZWQpO1xuICB9XG5cbiAgZ2V0IGNoYW5uZWxOYW1lKCkge1xuICAgIGlmICghdGhpcy5jaGFubmVsIHx8ICF0aGlzLmNoYXRDbGllbnRTZXJ2aWNlLmNoYXRDbGllbnQudXNlcikge1xuICAgICAgcmV0dXJuICcnO1xuICAgIH1cbiAgICByZXR1cm4gZ2V0Q2hhbm5lbERpc3BsYXlUZXh0KFxuICAgICAgdGhpcy5jaGFubmVsLFxuICAgICAgdGhpcy5jaGF0Q2xpZW50U2VydmljZS5jaGF0Q2xpZW50LnVzZXJcbiAgICApO1xuICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX3RocmVhZC1jb250YWluZXJcIj5cbiAgPG5nLWNvbnRhaW5lclxuICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwiXG4gICAgICAoY3VzdG9tVGVtcGxhdGVzU2VydmljZS50aHJlYWRIZWFkZXJUZW1wbGF0ZSQgfCBhc3luYykgfHxcbiAgICAgICAgZGVmYXVsdFRocmVhZEhlYWRlcjtcbiAgICAgIGNvbnRleHQ6IGdldFRocmVhZEhlYWRlckNvbnRleHQoKVxuICAgIFwiXG4gID48L25nLWNvbnRhaW5lcj5cbiAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuPC9kaXY+XG5cbjxuZy10ZW1wbGF0ZVxuICAjZGVmYXVsdFRocmVhZEhlYWRlclxuICBsZXQtcGFyZW50TWVzc2FnZT1cInBhcmVudE1lc3NhZ2VcIlxuICBsZXQtY2xvc2VUaHJlYWRIYW5kbGVyPVwiY2xvc2VUaHJlYWRIYW5kbGVyXCJcbj5cbiAgPGRpdiBjbGFzcz1cInN0ci1jaGF0X190aHJlYWQtaGVhZGVyXCI+XG4gICAgPGRpdiBjbGFzcz1cInN0ci1jaGF0X190aHJlYWQtaGVhZGVyLWRldGFpbHNcIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJzdHItY2hhdF9fdGhyZWFkLWhlYWRlci1uYW1lXCIgdHJhbnNsYXRlPlxuICAgICAgICBzdHJlYW1DaGF0LlRocmVhZFxuICAgICAgPC9kaXY+XG4gICAgICA8ZGl2XG4gICAgICAgIGNsYXNzPVwic3RyLWNoYXRfX3RocmVhZC1oZWFkZXItY2hhbm5lbC1uYW1lXCJcbiAgICAgICAgZGF0YS10ZXN0aWQ9XCJjaGFubmVsLW5hbWVcIlxuICAgICAgPlxuICAgICAgICB7eyBjaGFubmVsTmFtZSB9fVxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gICAgPGJ1dHRvblxuICAgICAgY2xhc3M9XCJzdHItY2hhdF9fc3F1YXJlLWJ1dHRvbiBzdHItY2hhdF9fY2xvc2UtdGhyZWFkLWJ1dHRvblwiXG4gICAgICBkYXRhLXRlc3RpZD1cImNsb3NlLWJ1dHRvblwiXG4gICAgICAoY2xpY2spPVwiY2xvc2VUaHJlYWRIYW5kbGVyKClcIlxuICAgID5cbiAgICAgIDxzdHJlYW0taWNvbi1wbGFjZWhvbGRlclxuICAgICAgICBpY29uPVwiY2xvc2Utbm8tb3V0bGluZVwiXG4gICAgICA+PC9zdHJlYW0taWNvbi1wbGFjZWhvbGRlcj5cbiAgICA8L2J1dHRvbj5cbiAgPC9kaXY+XG48L25nLXRlbXBsYXRlPlxuIl19
|
package/esm2015/lib/types.js
DELETED
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TemplateRef } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport type {\n  Attachment,\n  Channel,\n  ChannelFilters,\n  ChannelMemberResponse,\n  CommandResponse,\n  Event,\n  ExtendableGenerics,\n  FormatMessageResponse,\n  LiteralStringForUnion,\n  MessageResponseBase,\n  Mute,\n  ReactionResponse,\n  User,\n  UserResponse,\n} from 'stream-chat';\nimport { AttachmentService } from './attachment.service';\nimport { Icon } from './icon/icon.component';\n\nexport type UnknownType = Record<string, unknown>;\n\nexport type CustomTrigger = {\n  [key: string]: {\n    componentProps: UnknownType;\n    data: UnknownType;\n  };\n};\n\nexport type DefaultStreamChatGenerics = ExtendableGenerics & {\n  attachmentType: DefaultAttachmentType;\n  channelType: DefaultChannelType;\n  commandType: LiteralStringForUnion;\n  eventType: UnknownType;\n  messageType: DefaultMessageType;\n  reactionType: UnknownType;\n  userType: DefaultUserType;\n};\n\nexport type DefaultAttachmentType = UnknownType & {\n  asset_url?: string;\n  id?: string;\n  images?: Array<Attachment<DefaultStreamChatGenerics>>;\n  mime_type?: string;\n  isCustomAttachment?: boolean;\n};\n\nexport type DefaultChannelType = UnknownType & {\n  image?: string;\n  member_count?: number;\n  subtitle?: string;\n};\n\nexport type DefaultCommandType = LiteralStringForUnion;\n\nexport type DefaultMessageType = UnknownType & {\n  customType?: 'channel.intro' | 'message.date';\n  date?: string | Date;\n  errorStatusCode?: number;\n  event?: Event<DefaultStreamChatGenerics>;\n  unread?: boolean;\n  readBy: UserResponse<DefaultStreamChatGenerics>[];\n  translation?: string;\n  quoted_message?: MessageResponseBase<DefaultStreamChatGenerics>;\n};\n\nexport type DefaultUserTypeInternal = {\n  image?: string;\n  status?: string;\n};\n\nexport type DefaultUserType = UnknownType &\n  DefaultUserTypeInternal & {\n    mutes?: Array<Mute<DefaultStreamChatGenerics>>;\n  };\n\nexport type StreamMessage<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = FormatMessageResponse<T>;\n\nexport type AttachmentUploadErrorReason =\n  | 'file-size'\n  | 'file-extension'\n  | 'unknown';\n\nexport type AttachmentUpload<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  file: File;\n  state: 'error' | 'success' | 'uploading';\n  errorReason?: AttachmentUploadErrorReason;\n  errorExtraInfo?: { param: string }[];\n  url?: string;\n  type: 'image' | 'file' | 'video';\n  previewUri?: string | ArrayBuffer;\n  thumb_url?: string;\n  fromAttachment?: Attachment<T>;\n};\n\nexport type MentionAutcompleteListItemContext = {\n  item: MentionAutcompleteListItem;\n};\n\nexport type CommandAutocompleteListItemContext = {\n  item: ComandAutocompleteListItem;\n};\n\nexport type MentionAutcompleteListItem = (\n  | ChannelMemberResponse\n  | UserResponse\n) & {\n  autocompleteLabel: string;\n};\n\nexport type ComandAutocompleteListItem = CommandResponse & {\n  autocompleteLabel: string;\n};\n\nexport type NotificationType = 'success' | 'error' | 'info';\n\nexport type NotificationPayload<T = {}> = {\n  id: string;\n  type: NotificationType;\n  text?: string;\n  translateParams?: Object;\n  template?: TemplateRef<T>;\n  templateContext?: T;\n  dismissFn: Function;\n};\n\nexport type ChannelPreviewContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  channel: Channel<T>;\n};\n\nexport type ChannelPreviewInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = ChannelPreviewContext & {\n  latestMessage?: StreamMessage<T>;\n  /**\n   * The text of the latest message, or some meta information (for example: \"Nothing yet\")\n   */\n  latestMessageText: string;\n  /**\n   * The title of the channel, or the name of the channel members\n   */\n  channelDisplayTitle: string;\n  /**\n   * The status of the last message (only available if the last message was sent by the current user)\n   */\n  latestMessageStatus?: 'delivered' | 'read';\n  /**\n   * The time of the last message (formatted to a user-friendly string)\n   */\n  latestMessageTime?: string;\n  unreadCount: number;\n};\n\nexport type MessageInputContext = {\n  isFileUploadEnabled: boolean | undefined;\n  areMentionsEnabled: boolean | undefined;\n  mentionScope: 'channel' | 'application' | undefined;\n  mode: 'thread' | 'main' | undefined;\n  isMultipleFileUploadEnabled: boolean | undefined;\n  message: StreamMessage | undefined;\n  messageUpdateHandler: Function | undefined;\n  sendMessage$: Observable<void>;\n};\n\nexport type MentionTemplateContext = {\n  content: string;\n  user: UserResponse;\n};\n\nexport type EmojiPickerContext = {\n  emojiInput$: Subject<string>;\n};\n\nexport type TypingIndicatorContext = {\n  usersTyping$: Observable<UserResponse<DefaultStreamChatGenerics>[]>;\n};\n\nexport type MessageContext = {\n  message: StreamMessage | undefined;\n  enabledMessageActions: string[];\n  isLastSentMessage: boolean | undefined;\n  mode: 'thread' | 'main';\n  isHighlighted: boolean;\n  customActions: CustomMessageActionItem[];\n};\n\nexport type ChannelActionsContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type AttachmentListContext = {\n  messageId: string;\n  attachments: Attachment<DefaultStreamChatGenerics>[];\n  parentMessageId?: string;\n  imageModalStateChangeHandler?: (state: 'opened' | 'closed') => {};\n};\n\nexport type AvatarType = 'channel' | 'user';\n\nexport type AvatarLocation =\n  | 'channel-preview'\n  | 'channel-header'\n  | 'message-sender'\n  | 'message-reader'\n  | 'quoted-message-sender'\n  | 'autocomplete-item'\n  | 'typing-indicator'\n  | 'reaction';\n\nexport type AvatarContext = {\n  name: string | undefined;\n  imageUrl: string | undefined;\n  size: number | undefined;\n  type: AvatarType | undefined;\n  location: AvatarLocation | undefined;\n  channel?: Channel<DefaultStreamChatGenerics>;\n  user?: User<DefaultStreamChatGenerics>;\n  initialsType?: 'first-letter-of-first-word' | 'first-letter-of-each-word';\n  showOnlineIndicator?: boolean;\n};\n\nexport type AttachmentPreviewListContext = {\n  attachmentUploads$: Observable<AttachmentUpload[]> | undefined;\n  retryUploadHandler: (f: File) => any;\n  deleteUploadHandler: (u: AttachmentUpload) => any;\n};\n\nexport type IconContext = {\n  icon: Icon | undefined;\n  size: number | undefined;\n};\n\nexport type LoadingIndicatorContext = {\n  size: number | undefined;\n  color: string | undefined;\n};\n\nexport type MessageActionsBoxContext = {\n  isOpen: boolean;\n  isMine: boolean;\n  message: StreamMessage | undefined;\n  enabledActions: string[];\n  /**\n   * @deprecated please use `messageReactionsService.customActions$`\n   *\n   * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService\n   */\n  customActions: CustomMessageActionItem[];\n  /**\n   * @deprecated because the name contains typos, use the `displayedActionsCountChangeHandler` instead\n   */\n  displayedActionsCountChaneHanler: (count: number) => any;\n  /**\n   * @deprecated components should use `messageReactionsService.getAuthorizedMessageActionsCount` method\n   *\n   * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService\n   */\n  displayedActionsCountChangeHandler: (count: number) => any;\n};\n\nexport type MessageActionBoxItemContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionName: 'quote' | 'pin' | 'flag' | 'edit' | 'delete' | string;\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  message: StreamMessage<T>;\n  isMine: boolean;\n  actionHandler: (message: StreamMessage<T>, isMine: boolean) => any;\n};\n\ntype MessageActionItemBase<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  isVisible: (\n    enabledActions: string[],\n    isMine: boolean,\n    message: StreamMessage<T>\n  ) => boolean;\n  actionHandler: (message: StreamMessage<T>, isMine: boolean) => any;\n};\n\nexport type MessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName: 'quote' | 'pin' | 'flag' | 'edit' | 'delete' | 'mark-unread';\n};\n\nexport type CustomMessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName: string;\n};\n\nexport type MessageReactionsContext = {\n  messageId: string | undefined;\n  messageReactionCounts: { [key in MessageReactionType]?: number };\n  isSelectorOpen: boolean;\n  latestReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  ownReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  isSelectorOpenChangeHandler: (isOpen: boolean) => any;\n};\n\nexport type ModalContext = {\n  isOpen: boolean;\n  isOpenChangeHandler: (isOpen: boolean) => any;\n  content: TemplateRef<void>;\n};\n\nexport type NotificationContext = {\n  type: NotificationType | undefined;\n  content: TemplateRef<void> | undefined;\n};\n\nexport type ThreadHeaderContext = {\n  parentMessage: StreamMessage | undefined;\n  closeThreadHandler: Function;\n};\n\nexport type MessageReactionType =\n  | 'angry'\n  | 'haha'\n  | 'like'\n  | 'love'\n  | 'sad'\n  | 'wow'\n  | string;\n\nexport type AttachmentConfigration = {\n  url: string;\n  height: string;\n  width: string;\n};\n\nexport type ImageAttachmentConfiguration = AttachmentConfigration & {\n  originalHeight: number;\n  originalWidth: number;\n};\n\nexport type VideoAttachmentConfiguration = ImageAttachmentConfiguration & {\n  thumbUrl?: string;\n};\n\nexport type DeliveredStatusContext = {\n  message: StreamMessage;\n};\n\nexport type SendingStatusContext = {\n  message: StreamMessage;\n};\n\nexport type CustomMetadataContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  message: StreamMessage<T>;\n};\n\nexport type ReadStatusContext = {\n  message: StreamMessage;\n  readByText: string;\n};\n\nexport type ChannelHeaderInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type CustomAttachmentUploadContext = {\n  isMultipleFileUploadEnabled: boolean | undefined;\n  attachmentService: AttachmentService;\n};\n\nexport type AttachmentContext = {\n  attachment: Attachment<DefaultStreamChatGenerics>;\n};\n\nexport type SystemMessageContext = MessageContext & {\n  parsedDate: string | undefined;\n};\n\nexport type DateSeparatorContext = {\n  date: Date;\n  parsedDate: string;\n};\n\nexport type UnreadMessagesIndicatorContext = {\n  unreadCount: number;\n};\n\nexport type UnreadMessagesNotificationContext =\n  UnreadMessagesIndicatorContext & {\n    onJump: Function;\n    onDismiss: Function;\n  };\n\nexport type ChannelQueryState = {\n  state: 'in-progress' | 'success' | 'error';\n  // No type def from stream-chat\n  error?: unknown;\n};\n\nexport type MessageInput<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  text: string;\n  attachments: Attachment<T>[];\n  mentionedUsers: UserResponse<T>[];\n  parentId: string | undefined;\n  quotedMessageId: string | undefined;\n  customData: undefined | Partial<T['messageType']>;\n};\n\nexport type OffsetNextPageConfiguration = {\n  type: 'offset';\n  offset: number;\n};\n\nexport type FiltertNextPageConfiguration<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  type: 'filter';\n  paginationFilter: ChannelFilters<T>;\n};\n\nexport type NextPageConfiguration =\n  | OffsetNextPageConfiguration\n  | FiltertNextPageConfiguration;\n\nexport type MessageReactionClickDetails = {\n  messageId: string;\n  reactionType: string;\n};\n\nexport type MessageActionsClickDetails<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  message: StreamMessage<T>;\n  enabledActions: string[];\n  isMine: boolean;\n  customActions: CustomMessageActionItem[];\n};\n\nexport type GroupStyleOptions = {\n  noGroupByUser?: boolean;\n  lastReadMessageId?: string;\n  noGroupByReadState?: boolean;\n};\n"]}
|
package/esm2015/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.js
DELETED
|
@@ -1,192 +0,0 @@
|
|
|
1
|
-
import { Component, Input, ViewChild, } from '@angular/core';
|
|
2
|
-
import * as i0 from "@angular/core";
|
|
3
|
-
import * as i1 from "@angular/common";
|
|
4
|
-
/**
|
|
5
|
-
* This component can be used to visualize the wave bar of a voice recording
|
|
6
|
-
*/
|
|
7
|
-
export class VoiceRecordingWavebarComponent {
|
|
8
|
-
constructor(ngZone, cdRef) {
|
|
9
|
-
this.ngZone = ngZone;
|
|
10
|
-
this.cdRef = cdRef;
|
|
11
|
-
/**
|
|
12
|
-
* The waveform data to visualize
|
|
13
|
-
*/
|
|
14
|
-
this.waveFormData = [];
|
|
15
|
-
this.resampledWaveFormData = [];
|
|
16
|
-
this.progress = 0;
|
|
17
|
-
this.isDragging = false;
|
|
18
|
-
this.sampleSize = 40;
|
|
19
|
-
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
|
-
}
|
|
51
|
-
ngOnInit() {
|
|
52
|
-
var _a;
|
|
53
|
-
this.containerSizeChanged();
|
|
54
|
-
if ((_a = this.container) === null || _a === void 0 ? void 0 : _a.nativeElement) {
|
|
55
|
-
this.ngZone.runOutsideAngular(() => {
|
|
56
|
-
new ResizeObserver(() => {
|
|
57
|
-
this.containerSizeChanged();
|
|
58
|
-
}).observe(this.container.nativeElement);
|
|
59
|
-
});
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
ngOnChanges(changes) {
|
|
63
|
-
if (changes.waveFormData) {
|
|
64
|
-
this.resampledWaveFormData =
|
|
65
|
-
this.waveFormData.length > this.sampleSize
|
|
66
|
-
? this.downsample()
|
|
67
|
-
: this.upsample();
|
|
68
|
-
}
|
|
69
|
-
if (changes.audioElement) {
|
|
70
|
-
this.ngZone.runOutsideAngular(() => {
|
|
71
|
-
var _a;
|
|
72
|
-
(_a = this.audioElement) === null || _a === void 0 ? void 0 : _a.addEventListener('timeupdate', () => {
|
|
73
|
-
var _a;
|
|
74
|
-
const progress = (((_a = this.audioElement) === null || _a === void 0 ? void 0 : _a.currentTime) || 0) / (this.duration || 0) || 0;
|
|
75
|
-
if (Math.abs(progress - this.progress) >= 0.02) {
|
|
76
|
-
this.ngZone.run(() => {
|
|
77
|
-
this.progress = progress;
|
|
78
|
-
this.cdRef.detectChanges();
|
|
79
|
-
});
|
|
80
|
-
}
|
|
81
|
-
});
|
|
82
|
-
});
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
ngAfterViewInit() {
|
|
86
|
-
this.isViewInited = true;
|
|
87
|
-
}
|
|
88
|
-
seek(event) {
|
|
89
|
-
var _a, _b, _c, _d, _e;
|
|
90
|
-
const containerWidth = ((_b = (_a = this.container) === null || _a === void 0 ? void 0 : _a.nativeElement) === null || _b === void 0 ? void 0 : _b.getBoundingClientRect().width) || 0;
|
|
91
|
-
const containerStart = ((_e = (_d = (_c = this.container) === null || _c === void 0 ? void 0 : _c.nativeElement) === null || _d === void 0 ? void 0 : _d.getBoundingClientRect()) === null || _e === void 0 ? void 0 : _e.x) || 0;
|
|
92
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
|
93
|
-
const progress = (event.x - containerStart) / containerWidth;
|
|
94
|
-
if (!isNaN(progress) && this.audioElement) {
|
|
95
|
-
const duration = this.duration || 0;
|
|
96
|
-
const time = duration * progress;
|
|
97
|
-
this.audioElement.currentTime = time;
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
trackByIndex(index) {
|
|
101
|
-
return index;
|
|
102
|
-
}
|
|
103
|
-
containerSizeChanged() {
|
|
104
|
-
var _a;
|
|
105
|
-
if (!((_a = this.container) === null || _a === void 0 ? void 0 : _a.nativeElement)) {
|
|
106
|
-
return;
|
|
107
|
-
}
|
|
108
|
-
const containerWidth = this.container.nativeElement.clientWidth;
|
|
109
|
-
if (containerWidth === 0) {
|
|
110
|
-
return;
|
|
111
|
-
}
|
|
112
|
-
const barWidth = +getComputedStyle(this.container.nativeElement)
|
|
113
|
-
.getPropertyValue('--str-chat__voice-recording-amplitude-bar-width')
|
|
114
|
-
.replace('px', '');
|
|
115
|
-
const barGap = +getComputedStyle(this.container.nativeElement)
|
|
116
|
-
.getPropertyValue('--str-chat__voice-recording-amplitude-bar-gap-width')
|
|
117
|
-
.replace('px', '');
|
|
118
|
-
if (!isNaN(barWidth) && !isNaN(barGap)) {
|
|
119
|
-
const sampleSize = Math.floor(containerWidth / (barWidth + barGap));
|
|
120
|
-
if (sampleSize !== this.sampleSize &&
|
|
121
|
-
!isNaN(sampleSize) &&
|
|
122
|
-
sampleSize !== Infinity) {
|
|
123
|
-
this.ngZone.run(() => {
|
|
124
|
-
this.sampleSize = sampleSize;
|
|
125
|
-
this.resampledWaveFormData =
|
|
126
|
-
this.waveFormData.length > this.sampleSize
|
|
127
|
-
? this.downsample()
|
|
128
|
-
: this.upsample();
|
|
129
|
-
if (this.isViewInited) {
|
|
130
|
-
this.cdRef.detectChanges();
|
|
131
|
-
}
|
|
132
|
-
});
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
}
|
|
136
|
-
downsample() {
|
|
137
|
-
if (this.waveFormData.length <= this.sampleSize) {
|
|
138
|
-
return this.waveFormData;
|
|
139
|
-
}
|
|
140
|
-
if (this.sampleSize === 1)
|
|
141
|
-
return [this.mean(this.waveFormData)];
|
|
142
|
-
const result = [];
|
|
143
|
-
// bucket size adjusted due to the fact that the first and the last item in the original data array is kept in target output
|
|
144
|
-
const bucketSize = (this.waveFormData.length - 2) / (this.sampleSize - 2);
|
|
145
|
-
let lastSelectedPointIndex = 0;
|
|
146
|
-
result.push(this.waveFormData[lastSelectedPointIndex]); // Always add the first point
|
|
147
|
-
let maxAreaPoint, maxArea, triangleArea;
|
|
148
|
-
for (let bucketIndex = 1; bucketIndex < this.sampleSize - 1; bucketIndex++) {
|
|
149
|
-
const previousBucketRefPoint = this.waveFormData[lastSelectedPointIndex];
|
|
150
|
-
const nextBucketMean = this.getNextBucketMean(this.waveFormData, bucketIndex, bucketSize);
|
|
151
|
-
const currentBucketStartIndex = Math.floor((bucketIndex - 1) * bucketSize) + 1;
|
|
152
|
-
const nextBucketStartIndex = Math.floor(bucketIndex * bucketSize) + 1;
|
|
153
|
-
const countUnitsBetweenAtoC = 1 + nextBucketStartIndex - currentBucketStartIndex;
|
|
154
|
-
maxArea = triangleArea = -1;
|
|
155
|
-
for (let currentPointIndex = currentBucketStartIndex; currentPointIndex < nextBucketStartIndex; currentPointIndex++) {
|
|
156
|
-
const countUnitsBetweenAtoB = Math.abs(currentPointIndex - currentBucketStartIndex) + 1;
|
|
157
|
-
const countUnitsBetweenBtoC = countUnitsBetweenAtoC - countUnitsBetweenAtoB;
|
|
158
|
-
const currentPointValue = this.waveFormData[currentPointIndex];
|
|
159
|
-
triangleArea = this.triangleAreaHeron(this.triangleBase(Math.abs(previousBucketRefPoint - currentPointValue), countUnitsBetweenAtoB), this.triangleBase(Math.abs(currentPointValue - nextBucketMean), countUnitsBetweenBtoC), this.triangleBase(Math.abs(previousBucketRefPoint - nextBucketMean), countUnitsBetweenAtoC));
|
|
160
|
-
if (triangleArea > maxArea) {
|
|
161
|
-
maxArea = triangleArea;
|
|
162
|
-
maxAreaPoint = this.waveFormData[currentPointIndex];
|
|
163
|
-
lastSelectedPointIndex = currentPointIndex;
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
if (typeof maxAreaPoint !== 'undefined')
|
|
167
|
-
result.push(maxAreaPoint);
|
|
168
|
-
}
|
|
169
|
-
result.push(this.waveFormData[this.waveFormData.length - 1]); // Always add the last point
|
|
170
|
-
return result;
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
VoiceRecordingWavebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: VoiceRecordingWavebarComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
174
|
-
VoiceRecordingWavebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", 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: "<div\n class=\"str-chat__wave-progress-bar__track\"\n #container\n data-testid=\"wave-progress-bar-track\"\n (mousedown)=\"isDragging = true\"\n (mouseup)=\"isDragging = false\"\n (mouseleave)=\"isDragging = false\"\n (mousemove)=\"isDragging ? seek($event) : null\"\n (click)=\"seek($event)\"\n (keyup.enter)=\"seek($event)\"\n role=\"progressbar\"\n>\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", directives: [{ type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
|
|
175
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: VoiceRecordingWavebarComponent, decorators: [{
|
|
176
|
-
type: Component,
|
|
177
|
-
args: [{
|
|
178
|
-
selector: 'stream-voice-recording-wavebar',
|
|
179
|
-
templateUrl: './voice-recording-wavebar.component.html',
|
|
180
|
-
styles: [],
|
|
181
|
-
}]
|
|
182
|
-
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { audioElement: [{
|
|
183
|
-
type: Input
|
|
184
|
-
}], waveFormData: [{
|
|
185
|
-
type: Input
|
|
186
|
-
}], duration: [{
|
|
187
|
-
type: Input
|
|
188
|
-
}], container: [{
|
|
189
|
-
type: ViewChild,
|
|
190
|
-
args: ['container', { static: true }]
|
|
191
|
-
}] } });
|
|
192
|
-
//# 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: any) {\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(...Array(bucketSize + extra).fill(this.waveFormData[i]));\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","<div\n  class=\"str-chat__wave-progress-bar__track\"\n  #container\n  data-testid=\"wave-progress-bar-track\"\n  (mousedown)=\"isDragging = true\"\n  (mouseup)=\"isDragging = false\"\n  (mouseleave)=\"isDragging = false\"\n  (mousemove)=\"isDragging ? seek($event) : null\"\n  (click)=\"seek($event)\"\n  (keyup.enter)=\"seek($event)\"\n  role=\"progressbar\"\n>\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"]}
|