stream-chat-angular 2.10.0 → 2.12.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.
Files changed (33) hide show
  1. package/assets/i18n/en.d.ts +2 -0
  2. package/assets/version.d.ts +1 -1
  3. package/bundles/stream-chat-angular.umd.js +670 -337
  4. package/bundles/stream-chat-angular.umd.js.map +1 -1
  5. package/esm2015/assets/i18n/en.js +3 -1
  6. package/esm2015/assets/version.js +2 -2
  7. package/esm2015/lib/channel/channel.component.js +3 -2
  8. package/esm2015/lib/channel.service.js +170 -40
  9. package/esm2015/lib/icon/icon.component.js +2 -2
  10. package/esm2015/lib/message/message.component.js +67 -42
  11. package/esm2015/lib/message-actions-box/message-actions-box.component.js +7 -4
  12. package/esm2015/lib/message-input/message-input.component.js +61 -13
  13. package/esm2015/lib/message-list/message-list.component.js +104 -53
  14. package/esm2015/lib/message-preview.js +4 -2
  15. package/esm2015/lib/message-reactions/message-reactions.component.js +2 -2
  16. package/esm2015/lib/stream-chat.module.js +8 -3
  17. package/esm2015/lib/thread/thread.component.js +37 -0
  18. package/esm2015/public-api.js +2 -1
  19. package/fesm2015/stream-chat-angular.js +541 -246
  20. package/fesm2015/stream-chat-angular.js.map +1 -1
  21. package/lib/channel/channel.component.d.ts +2 -1
  22. package/lib/channel.service.d.ts +21 -7
  23. package/lib/icon/icon.component.d.ts +1 -1
  24. package/lib/message/message.component.d.ts +16 -6
  25. package/lib/message-input/message-input.component.d.ts +7 -1
  26. package/lib/message-list/message-list.component.d.ts +12 -3
  27. package/lib/message-preview.d.ts +1 -1
  28. package/lib/stream-chat.module.d.ts +5 -4
  29. package/lib/thread/thread.component.d.ts +18 -0
  30. package/package.json +2 -11
  31. package/public-api.d.ts +1 -0
  32. package/src/assets/i18n/en.ts +4 -0
  33. package/src/assets/version.ts +1 -1
@@ -7,6 +7,7 @@ import * as i2 from "../chat-client.service";
7
7
  import * as i3 from "./image-load.service";
8
8
  import * as i4 from "../message/message.component";
9
9
  import * as i5 from "@angular/common";
10
+ import * as i6 from "@ngx-translate/core";
10
11
  export class MessageListComponent {
11
12
  constructor(channelService, chatClientService, imageLoadService) {
12
13
  this.channelService = channelService;
@@ -21,22 +22,17 @@ export class MessageListComponent {
21
22
  */
22
23
  /* eslint-disable-next-line @angular-eslint/no-input-rename */
23
24
  this.enabledMessageActionsInput = undefined;
25
+ this.mode = 'main';
24
26
  this.enabledMessageActions = [];
25
27
  this.class = 'str-chat-angular__main-panel-inner str-chat-angular__message-list-host';
26
28
  this.unreadMessageCount = 0;
27
29
  this.groupStyles = [];
28
30
  this.authorizedMessageActions = ['flag'];
29
31
  this.isUserScrolledUpThreshold = 300;
30
- this.channelService.activeChannel$.subscribe((channel) => {
32
+ this.subscriptions = [];
33
+ this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
31
34
  var _a;
32
- this.latestMessageDate = undefined;
33
- this.hasNewMessages = true;
34
- this.isUserScrolledUp = false;
35
- this.containerHeight = undefined;
36
- this.olderMassagesLoaded = false;
37
- this.oldestMessageDate = undefined;
38
- this.unreadMessageCount = 0;
39
- this.isNewMessageSentByUser = undefined;
35
+ this.resetScrollState();
40
36
  const capabilites = (_a = channel === null || channel === void 0 ? void 0 : channel.data) === null || _a === void 0 ? void 0 : _a.own_capabilities;
41
37
  if (capabilites) {
42
38
  this.canReactToMessage = capabilites.indexOf('send-reaction') !== -1;
@@ -65,47 +61,16 @@ export class MessageListComponent {
65
61
  this.authorizedMessageActions.push('delete');
66
62
  this.authorizedMessageActions.push('delete-any');
67
63
  }
68
- this.setEnabledActions();
69
- }
70
- });
71
- this.messages$ = this.channelService.activeChannelMessages$.pipe(tap((messages) => {
72
- var _a, _b, _c, _d, _e;
73
- if (messages.length === 0) {
74
- return;
75
- }
76
- const currentLatestMessageDate = messages[messages.length - 1].created_at;
77
- if (!this.latestMessageDate ||
78
- ((_a = this.latestMessageDate) === null || _a === void 0 ? void 0 : _a.getTime()) < currentLatestMessageDate.getTime()) {
79
- this.latestMessageDate = currentLatestMessageDate;
80
- this.hasNewMessages = true;
81
- this.isNewMessageSentByUser =
82
- ((_b = messages[messages.length - 1].user) === null || _b === void 0 ? void 0 : _b.id) ===
83
- ((_d = (_c = this.chatClientService.chatClient) === null || _c === void 0 ? void 0 : _c.user) === null || _d === void 0 ? void 0 : _d.id);
84
- if (this.isUserScrolledUp) {
85
- this.unreadMessageCount++;
64
+ if (capabilites.indexOf('send-reply') !== -1) {
65
+ this.authorizedMessageActions.push('send-reply');
86
66
  }
67
+ if (capabilites.indexOf('quote-message') !== -1) {
68
+ this.authorizedMessageActions.push('quote-message');
69
+ }
70
+ this.setEnabledActions();
87
71
  }
88
- const currentOldestMessageDate = messages[0].created_at;
89
- if (!this.oldestMessageDate) {
90
- this.oldestMessageDate = currentOldestMessageDate;
91
- }
92
- else if (((_e = this.oldestMessageDate) === null || _e === void 0 ? void 0 : _e.getTime()) > currentOldestMessageDate.getTime()) {
93
- this.oldestMessageDate = currentOldestMessageDate;
94
- this.olderMassagesLoaded = true;
95
- }
96
- }), tap((messages) => {
97
- this.groupStyles = messages.map((m, i) => getGroupStyles(m, messages[i - 1], messages[i + 1]));
98
- }), tap((messages) => {
99
- var _a;
100
- return (this.lastSentMessageId = (_a = [...messages]
101
- .reverse()
102
- .find((m) => {
103
- var _a, _b, _c;
104
- return ((_a = m.user) === null || _a === void 0 ? void 0 : _a.id) === ((_c = (_b = this.chatClientService.chatClient) === null || _b === void 0 ? void 0 : _b.user) === null || _c === void 0 ? void 0 : _c.id) &&
105
- m.status !== 'sending';
106
- })) === null || _a === void 0 ? void 0 : _a.id);
107
72
  }));
108
- this.imageLoadService.imageLoad$.subscribe(() => {
73
+ this.subscriptions.push(this.imageLoadService.imageLoad$.subscribe(() => {
109
74
  if (!this.isUserScrolledUp) {
110
75
  this.scrollToBottom();
111
76
  // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar
@@ -113,12 +78,27 @@ export class MessageListComponent {
113
78
  this.scrollToBottom();
114
79
  }, 300);
115
80
  }
116
- });
81
+ }));
82
+ this.subscriptions.push(this.channelService.activeParentMessage$.subscribe((message) => {
83
+ if (message &&
84
+ this.parentMessage &&
85
+ message.id !== this.parentMessage.id &&
86
+ this.mode === 'thread') {
87
+ this.resetScrollState();
88
+ }
89
+ this.parentMessage = message;
90
+ }));
91
+ }
92
+ ngOnInit() {
93
+ this.setMessages$();
117
94
  }
118
95
  ngOnChanges(changes) {
119
96
  if (changes.enabledMessageActionsInput) {
120
97
  this.setEnabledActions();
121
98
  }
99
+ if (changes.mode) {
100
+ this.setMessages$();
101
+ }
122
102
  }
123
103
  ngAfterViewChecked() {
124
104
  if (this.hasNewMessages) {
@@ -144,6 +124,9 @@ export class MessageListComponent {
144
124
  this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;
145
125
  }
146
126
  }
127
+ ngOnDestroy() {
128
+ this.subscriptions.forEach((s) => s.unsubscribe());
129
+ }
147
130
  trackByMessageId(index, item) {
148
131
  return item.id;
149
132
  }
@@ -152,6 +135,7 @@ export class MessageListComponent {
152
135
  this.scrollContainer.nativeElement.scrollHeight;
153
136
  }
154
137
  scrolled() {
138
+ var _a, _b;
155
139
  this.isUserScrolledUp =
156
140
  this.scrollContainer.nativeElement.scrollHeight -
157
141
  (this.scrollContainer.nativeElement.scrollTop +
@@ -160,14 +144,22 @@ export class MessageListComponent {
160
144
  if (!this.isUserScrolledUp) {
161
145
  this.unreadMessageCount = 0;
162
146
  }
163
- if (this.scrollContainer.nativeElement.scrollTop === 0) {
147
+ if (this.scrollContainer.nativeElement.scrollTop <=
148
+ (((_a = this.parentMessageElement) === null || _a === void 0 ? void 0 : _a.nativeElement.clientHeight) || 0) &&
149
+ (this.prevScrollTop === undefined ||
150
+ this.prevScrollTop >
151
+ (((_b = this.parentMessageElement) === null || _b === void 0 ? void 0 : _b.nativeElement.clientHeight) || 0))) {
164
152
  this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;
165
- void this.channelService.loadMoreMessages();
153
+ this.mode === 'main'
154
+ ? void this.channelService.loadMoreMessages()
155
+ : void this.channelService.loadMoreThreadReplies();
166
156
  }
157
+ this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;
167
158
  }
168
159
  preserveScrollbarPosition() {
169
160
  this.scrollContainer.nativeElement.scrollTop =
170
- this.scrollContainer.nativeElement.scrollHeight - this.containerHeight;
161
+ (this.prevScrollTop || 0) +
162
+ (this.scrollContainer.nativeElement.scrollHeight - this.containerHeight);
171
163
  }
172
164
  setEnabledActions() {
173
165
  this.enabledMessageActions = [];
@@ -179,6 +171,8 @@ export class MessageListComponent {
179
171
  ...this.enabledMessageActionsInput,
180
172
  'send-reaction',
181
173
  'read-events',
174
+ 'send-reply',
175
+ 'quote-message',
182
176
  ];
183
177
  this.enabledMessageActionsInput.forEach((action) => {
184
178
  const isAuthorized = this.authorizedMessageActions.indexOf(action) !== -1;
@@ -187,9 +181,61 @@ export class MessageListComponent {
187
181
  }
188
182
  });
189
183
  }
184
+ setMessages$() {
185
+ this.messages$ = (this.mode === 'main'
186
+ ? this.channelService.activeChannelMessages$
187
+ : this.channelService.activeThreadMessages$).pipe(tap((messages) => {
188
+ var _a, _b, _c, _d, _e;
189
+ if (messages.length === 0) {
190
+ return;
191
+ }
192
+ const currentLatestMessageDate = messages[messages.length - 1].created_at;
193
+ if (!this.latestMessageDate ||
194
+ ((_a = this.latestMessageDate) === null || _a === void 0 ? void 0 : _a.getTime()) < currentLatestMessageDate.getTime()) {
195
+ this.latestMessageDate = currentLatestMessageDate;
196
+ this.hasNewMessages = true;
197
+ this.isNewMessageSentByUser =
198
+ ((_b = messages[messages.length - 1].user) === null || _b === void 0 ? void 0 : _b.id) ===
199
+ ((_d = (_c = this.chatClientService.chatClient) === null || _c === void 0 ? void 0 : _c.user) === null || _d === void 0 ? void 0 : _d.id);
200
+ if (this.isUserScrolledUp) {
201
+ this.unreadMessageCount++;
202
+ }
203
+ }
204
+ const currentOldestMessageDate = messages[0].created_at;
205
+ if (!this.oldestMessageDate) {
206
+ this.oldestMessageDate = currentOldestMessageDate;
207
+ }
208
+ else if (((_e = this.oldestMessageDate) === null || _e === void 0 ? void 0 : _e.getTime()) > currentOldestMessageDate.getTime()) {
209
+ this.oldestMessageDate = currentOldestMessageDate;
210
+ this.olderMassagesLoaded = true;
211
+ }
212
+ }), tap((messages) => {
213
+ this.groupStyles = messages.map((m, i) => getGroupStyles(m, messages[i - 1], messages[i + 1]));
214
+ }), tap((messages) => {
215
+ var _a;
216
+ return (this.lastSentMessageId = (_a = [...messages]
217
+ .reverse()
218
+ .find((m) => {
219
+ var _a, _b, _c;
220
+ return ((_a = m.user) === null || _a === void 0 ? void 0 : _a.id) === ((_c = (_b = this.chatClientService.chatClient) === null || _b === void 0 ? void 0 : _b.user) === null || _c === void 0 ? void 0 : _c.id) &&
221
+ m.status !== 'sending';
222
+ })) === null || _a === void 0 ? void 0 : _a.id);
223
+ }));
224
+ }
225
+ resetScrollState() {
226
+ this.latestMessageDate = undefined;
227
+ this.hasNewMessages = true;
228
+ this.isUserScrolledUp = false;
229
+ this.containerHeight = undefined;
230
+ this.olderMassagesLoaded = false;
231
+ this.oldestMessageDate = undefined;
232
+ this.unreadMessageCount = 0;
233
+ this.prevScrollTop = undefined;
234
+ this.isNewMessageSentByUser = undefined;
235
+ }
190
236
  }
191
237
  MessageListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, deps: [{ token: i1.ChannelService }, { token: i2.ChatClientService }, { token: i3.ImageLoadService }], target: i0.ɵɵFactoryTarget.Component });
192
- MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"] }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n ></stream-message>\n </ng-template>\n </li>\n </ul>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n", components: [{ type: i4.MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents"] }], directives: [{ type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "async": i5.AsyncPipe } });
238
+ MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"], mode: "mode" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread'\"\n data-testid=\"parent-message\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\n \"\n ></ng-container>\n <div class=\"str-chat__thread-start\" translate>\n streamChat.Start of a new thread\n </div>\n </li>\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\n </ul>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate,\n mode: mode\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n [mode]=\"mode\"\n ></stream-message>\n </ng-template>\n</ng-template>\n", components: [{ type: i4.MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents", "mode"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i6.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i5.AsyncPipe } });
193
239
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
194
240
  type: Component,
195
241
  args: [{
@@ -208,11 +254,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
208
254
  }], enabledMessageActionsInput: [{
209
255
  type: Input,
210
256
  args: ['enabledMessageActions']
257
+ }], mode: [{
258
+ type: Input
211
259
  }], class: [{
212
260
  type: HostBinding,
213
261
  args: ['class']
214
262
  }], scrollContainer: [{
215
263
  type: ViewChild,
216
264
  args: ['scrollContainer']
265
+ }], parentMessageElement: [{
266
+ type: ViewChild,
267
+ args: ['parentMessageElement']
217
268
  }] } });
218
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { Observable } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { StreamMessage } from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { ImageLoadService } from './image-load.service';\n\n@Component({\n  selector: 'stream-message-list',\n  templateUrl: './message-list.component.html',\n  styles: [],\n})\nexport class MessageListComponent implements AfterViewChecked, OnChanges {\n  @Input() messageTemplate: TemplateRef<any> | undefined;\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\n  /**\n   * @deprecated https://getstream.io/chat/docs/sdk/angular/components/message_list/#caution-arereactionsenabled-deprecated\n   */\n  @Input() areReactionsEnabled: boolean | undefined = undefined;\n  /**\n   * @deprecated https://getstream.io/chat/docs/sdk/angular/components/message_list/#caution-enabledmessageactions-deprecated\n   */\n  /* eslint-disable-next-line @angular-eslint/no-input-rename */\n  @Input('enabledMessageActions') enabledMessageActionsInput:\n    | string[]\n    | undefined = undefined;\n  messages$!: Observable<StreamMessage[]>;\n  canReactToMessage: boolean | undefined;\n  canReceiveReadEvents: boolean | undefined;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host';\n  unreadMessageCount = 0;\n  isUserScrolledUp: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  private latestMessageDate: Date | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessageDate: Date | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private authorizedMessageActions: string[] = ['flag'];\n  private readonly isUserScrolledUpThreshold = 300;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private imageLoadService: ImageLoadService\n  ) {\n    this.channelService.activeChannel$.subscribe((channel) => {\n      this.latestMessageDate = undefined;\n      this.hasNewMessages = true;\n      this.isUserScrolledUp = false;\n      this.containerHeight = undefined;\n      this.olderMassagesLoaded = false;\n      this.oldestMessageDate = undefined;\n      this.unreadMessageCount = 0;\n      this.isNewMessageSentByUser = undefined;\n      const capabilites = channel?.data?.own_capabilities as string[];\n      if (capabilites) {\n        this.canReactToMessage = capabilites.indexOf('send-reaction') !== -1;\n        this.canReceiveReadEvents = capabilites.indexOf('read-events') !== -1;\n        this.authorizedMessageActions = [];\n        if (this.canReactToMessage) {\n          this.authorizedMessageActions.push('send-reaction');\n        }\n        if (this.canReceiveReadEvents) {\n          this.authorizedMessageActions.push('read-events');\n        }\n        if (capabilites.indexOf('flag-message') !== -1) {\n          this.authorizedMessageActions.push('flag');\n        }\n        if (capabilites.indexOf('update-own-message') !== -1) {\n          this.authorizedMessageActions.push('edit');\n        }\n        if (capabilites.indexOf('update-any-message') !== -1) {\n          this.authorizedMessageActions.push('edit');\n          this.authorizedMessageActions.push('edit-any');\n        }\n        if (capabilites.indexOf('delete-own-message') !== -1) {\n          this.authorizedMessageActions.push('delete');\n        }\n        if (capabilites.indexOf('delete-any-message') !== -1) {\n          this.authorizedMessageActions.push('delete');\n          this.authorizedMessageActions.push('delete-any');\n        }\n        this.setEnabledActions();\n      }\n    });\n    this.messages$ = this.channelService.activeChannelMessages$.pipe(\n      tap((messages) => {\n        if (messages.length === 0) {\n          return;\n        }\n        const currentLatestMessageDate =\n          messages[messages.length - 1].created_at;\n        if (\n          !this.latestMessageDate ||\n          this.latestMessageDate?.getTime() < currentLatestMessageDate.getTime()\n        ) {\n          this.latestMessageDate = currentLatestMessageDate;\n          this.hasNewMessages = true;\n          this.isNewMessageSentByUser =\n            messages[messages.length - 1].user?.id ===\n            this.chatClientService.chatClient?.user?.id;\n          if (this.isUserScrolledUp) {\n            this.unreadMessageCount++;\n          }\n        }\n        const currentOldestMessageDate = messages[0].created_at;\n        if (!this.oldestMessageDate) {\n          this.oldestMessageDate = currentOldestMessageDate;\n        } else if (\n          this.oldestMessageDate?.getTime() > currentOldestMessageDate.getTime()\n        ) {\n          this.oldestMessageDate = currentOldestMessageDate;\n          this.olderMassagesLoaded = true;\n        }\n      }),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      }),\n      tap(\n        (messages) =>\n          (this.lastSentMessageId = [...messages]\n            .reverse()\n            .find(\n              (m) =>\n                m.user?.id === this.chatClientService.chatClient?.user?.id &&\n                m.status !== 'sending'\n            )?.id)\n      )\n    );\n    this.imageLoadService.imageLoad$.subscribe(() => {\n      if (!this.isUserScrolledUp) {\n        this.scrollToBottom();\n        // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n        setTimeout(() => {\n          this.scrollToBottom();\n        }, 300);\n      }\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.enabledMessageActionsInput) {\n      this.setEnabledActions();\n    }\n  }\n\n  ngAfterViewChecked() {\n    if (this.hasNewMessages) {\n      if (!this.isUserScrolledUp || this.isNewMessageSentByUser) {\n        this.scrollToBottom();\n        // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n        setTimeout(() => {\n          this.scrollToBottom();\n        }, 300);\n      }\n      this.hasNewMessages = false;\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    } else if (this.olderMassagesLoaded) {\n      this.preserveScrollbarPosition();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.olderMassagesLoaded = false;\n    } else if (\n      this.containerHeight !== undefined &&\n      this.containerHeight < this.scrollContainer.nativeElement.scrollHeight &&\n      !this.isUserScrolledUp\n    ) {\n      this.scrollToBottom();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    }\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrolled() {\n    this.isUserScrolledUp =\n      this.scrollContainer.nativeElement.scrollHeight -\n        (this.scrollContainer.nativeElement.scrollTop +\n          this.scrollContainer.nativeElement.clientHeight) >\n      this.isUserScrolledUpThreshold;\n    if (!this.isUserScrolledUp) {\n      this.unreadMessageCount = 0;\n    }\n    if (this.scrollContainer.nativeElement.scrollTop === 0) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      void this.channelService.loadMoreMessages();\n    }\n  }\n\n  private preserveScrollbarPosition() {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight - this.containerHeight!;\n  }\n\n  private setEnabledActions() {\n    this.enabledMessageActions = [];\n    if (!this.enabledMessageActionsInput) {\n      this.enabledMessageActions = this.authorizedMessageActions;\n      return;\n    }\n    this.enabledMessageActionsInput = [\n      ...this.enabledMessageActionsInput,\n      'send-reaction',\n      'read-events',\n    ];\n    this.enabledMessageActionsInput.forEach((action) => {\n      const isAuthorized = this.authorizedMessageActions.indexOf(action) !== -1;\n      if (isAuthorized) {\n        this.enabledMessageActions.push(action);\n      }\n    });\n  }\n}\n","<div\n  #scrollContainer\n  data-testid=\"scroll-container\"\n  class=\"str-chat__list\"\n  (scroll)=\"scrolled()\"\n>\n  <div class=\"str-chat__reverse-infinite-scroll\">\n    <ul class=\"str-chat__ul\">\n      <li\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n      >\n        <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n          <ng-container\n            *ngTemplateOutlet=\"\n              messageTemplate;\n              context: {\n                message: message,\n                areReactionsEnabled: areReactionsEnabled,\n                canReactToMessage: canReactToMessage,\n                lastSentMessageId: !!(\n                  lastSentMessageId && message?.id === lastSentMessageId\n                ),\n                canReceiveReadEvents: canReceiveReadEvents,\n                messageInputTemplate: messageInputTemplate,\n                mentionTemplate: mentionTemplate\n              }\n            \"\n          ></ng-container>\n        </ng-container>\n        <ng-template #defaultMessageTemplate>\n          <stream-message\n            [message]=\"message\"\n            [areReactionsEnabled]=\"areReactionsEnabled\"\n            [canReactToMessage]=\"canReactToMessage\"\n            [isLastSentMessage]=\"\n              !!(lastSentMessageId && message?.id === lastSentMessageId)\n            \"\n            [enabledMessageActions]=\"enabledMessageActions\"\n            [canReceiveReadEvents]=\"canReceiveReadEvents\"\n            [messageInputTemplate]=\"messageInputTemplate\"\n            [mentionTemplate]=\"mentionTemplate\"\n          ></stream-message>\n        </ng-template>\n      </li>\n    </ul>\n  </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n  <button\n    data-testid=\"scroll-to-bottom\"\n    *ngIf=\"isUserScrolledUp\"\n    class=\"\n      str-chat__message-notification\n      str-chat__message-notification-right\n      str-chat__message-notification-scroll-down\n    \"\n    (keyup.enter)=\"scrollToBottom()\"\n    (click)=\"scrollToBottom()\"\n  >\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-down-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n"]}
269
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { Observable, Subscription } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { StreamMessage } from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { ImageLoadService } from './image-load.service';\n\n@Component({\n  selector: 'stream-message-list',\n  templateUrl: './message-list.component.html',\n  styles: [],\n})\nexport class MessageListComponent\n  implements AfterViewChecked, OnChanges, OnInit, OnDestroy\n{\n  @Input() messageTemplate: TemplateRef<any> | undefined;\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\n  /**\n   * @deprecated https://getstream.io/chat/docs/sdk/angular/components/message_list/#caution-arereactionsenabled-deprecated\n   */\n  @Input() areReactionsEnabled: boolean | undefined = undefined;\n  /**\n   * @deprecated https://getstream.io/chat/docs/sdk/angular/components/message_list/#caution-enabledmessageactions-deprecated\n   */\n  /* eslint-disable-next-line @angular-eslint/no-input-rename */\n  @Input('enabledMessageActions') enabledMessageActionsInput:\n    | string[]\n    | undefined = undefined;\n  @Input() mode: 'main' | 'thread' = 'main';\n  messages$!: Observable<StreamMessage[]>;\n  canReactToMessage: boolean | undefined;\n  canReceiveReadEvents: boolean | undefined;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host';\n  unreadMessageCount = 0;\n  isUserScrolledUp: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  parentMessage: StreamMessage | undefined;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  @ViewChild('parentMessageElement')\n  private parentMessageElement!: ElementRef<HTMLElement>;\n  private latestMessageDate: Date | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessageDate: Date | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private authorizedMessageActions: string[] = ['flag'];\n  private readonly isUserScrolledUpThreshold = 300;\n  private subscriptions: Subscription[] = [];\n  private prevScrollTop: number | undefined;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private imageLoadService: ImageLoadService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.resetScrollState();\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.canReactToMessage = capabilites.indexOf('send-reaction') !== -1;\n          this.canReceiveReadEvents = capabilites.indexOf('read-events') !== -1;\n          this.authorizedMessageActions = [];\n          if (this.canReactToMessage) {\n            this.authorizedMessageActions.push('send-reaction');\n          }\n          if (this.canReceiveReadEvents) {\n            this.authorizedMessageActions.push('read-events');\n          }\n          if (capabilites.indexOf('flag-message') !== -1) {\n            this.authorizedMessageActions.push('flag');\n          }\n          if (capabilites.indexOf('update-own-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n          }\n          if (capabilites.indexOf('update-any-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n            this.authorizedMessageActions.push('edit-any');\n          }\n          if (capabilites.indexOf('delete-own-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n          }\n          if (capabilites.indexOf('delete-any-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n            this.authorizedMessageActions.push('delete-any');\n          }\n          if (capabilites.indexOf('send-reply') !== -1) {\n            this.authorizedMessageActions.push('send-reply');\n          }\n          if (capabilites.indexOf('quote-message') !== -1) {\n            this.authorizedMessageActions.push('quote-message');\n          }\n          this.setEnabledActions();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.imageLoadService.imageLoad$.subscribe(() => {\n        if (!this.isUserScrolledUp) {\n          this.scrollToBottom();\n          // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n          setTimeout(() => {\n            this.scrollToBottom();\n          }, 300);\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.channelService.activeParentMessage$.subscribe((message) => {\n        if (\n          message &&\n          this.parentMessage &&\n          message.id !== this.parentMessage.id &&\n          this.mode === 'thread'\n        ) {\n          this.resetScrollState();\n        }\n        this.parentMessage = message;\n      })\n    );\n  }\n\n  ngOnInit(): void {\n    this.setMessages$();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.enabledMessageActionsInput) {\n      this.setEnabledActions();\n    }\n    if (changes.mode) {\n      this.setMessages$();\n    }\n  }\n\n  ngAfterViewChecked() {\n    if (this.hasNewMessages) {\n      if (!this.isUserScrolledUp || this.isNewMessageSentByUser) {\n        this.scrollToBottom();\n        // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n        setTimeout(() => {\n          this.scrollToBottom();\n        }, 300);\n      }\n      this.hasNewMessages = false;\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    } else if (this.olderMassagesLoaded) {\n      this.preserveScrollbarPosition();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.olderMassagesLoaded = false;\n    } else if (\n      this.containerHeight !== undefined &&\n      this.containerHeight < this.scrollContainer.nativeElement.scrollHeight &&\n      !this.isUserScrolledUp\n    ) {\n      this.scrollToBottom();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrolled() {\n    this.isUserScrolledUp =\n      this.scrollContainer.nativeElement.scrollHeight -\n        (this.scrollContainer.nativeElement.scrollTop +\n          this.scrollContainer.nativeElement.clientHeight) >\n      this.isUserScrolledUpThreshold;\n    if (!this.isUserScrolledUp) {\n      this.unreadMessageCount = 0;\n    }\n    if (\n      this.scrollContainer.nativeElement.scrollTop <=\n        (this.parentMessageElement?.nativeElement.clientHeight || 0) &&\n      (this.prevScrollTop === undefined ||\n        this.prevScrollTop >\n          (this.parentMessageElement?.nativeElement.clientHeight || 0))\n    ) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.mode === 'main'\n        ? void this.channelService.loadMoreMessages()\n        : void this.channelService.loadMoreThreadReplies();\n    }\n    this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;\n  }\n\n  private preserveScrollbarPosition() {\n    this.scrollContainer.nativeElement.scrollTop =\n      (this.prevScrollTop || 0) +\n      (this.scrollContainer.nativeElement.scrollHeight - this.containerHeight!);\n  }\n\n  private setEnabledActions() {\n    this.enabledMessageActions = [];\n    if (!this.enabledMessageActionsInput) {\n      this.enabledMessageActions = this.authorizedMessageActions;\n      return;\n    }\n    this.enabledMessageActionsInput = [\n      ...this.enabledMessageActionsInput,\n      'send-reaction',\n      'read-events',\n      'send-reply',\n      'quote-message',\n    ];\n    this.enabledMessageActionsInput.forEach((action) => {\n      const isAuthorized = this.authorizedMessageActions.indexOf(action) !== -1;\n      if (isAuthorized) {\n        this.enabledMessageActions.push(action);\n      }\n    });\n  }\n\n  private setMessages$() {\n    this.messages$ = (\n      this.mode === 'main'\n        ? this.channelService.activeChannelMessages$\n        : this.channelService.activeThreadMessages$\n    ).pipe(\n      tap((messages) => {\n        if (messages.length === 0) {\n          return;\n        }\n        const currentLatestMessageDate =\n          messages[messages.length - 1].created_at;\n        if (\n          !this.latestMessageDate ||\n          this.latestMessageDate?.getTime() < currentLatestMessageDate.getTime()\n        ) {\n          this.latestMessageDate = currentLatestMessageDate;\n          this.hasNewMessages = true;\n          this.isNewMessageSentByUser =\n            messages[messages.length - 1].user?.id ===\n            this.chatClientService.chatClient?.user?.id;\n          if (this.isUserScrolledUp) {\n            this.unreadMessageCount++;\n          }\n        }\n        const currentOldestMessageDate = messages[0].created_at;\n        if (!this.oldestMessageDate) {\n          this.oldestMessageDate = currentOldestMessageDate;\n        } else if (\n          this.oldestMessageDate?.getTime() > currentOldestMessageDate.getTime()\n        ) {\n          this.oldestMessageDate = currentOldestMessageDate;\n          this.olderMassagesLoaded = true;\n        }\n      }),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      }),\n      tap(\n        (messages) =>\n          (this.lastSentMessageId = [...messages]\n            .reverse()\n            .find(\n              (m) =>\n                m.user?.id === this.chatClientService.chatClient?.user?.id &&\n                m.status !== 'sending'\n            )?.id)\n      )\n    );\n  }\n\n  private resetScrollState() {\n    this.latestMessageDate = undefined;\n    this.hasNewMessages = true;\n    this.isUserScrolledUp = false;\n    this.containerHeight = undefined;\n    this.olderMassagesLoaded = false;\n    this.oldestMessageDate = undefined;\n    this.unreadMessageCount = 0;\n    this.prevScrollTop = undefined;\n    this.isNewMessageSentByUser = undefined;\n  }\n}\n","<div\n  #scrollContainer\n  data-testid=\"scroll-container\"\n  class=\"str-chat__list\"\n  (scroll)=\"scrolled()\"\n>\n  <div class=\"str-chat__reverse-infinite-scroll\">\n    <ul class=\"str-chat__ul\">\n      <li\n        #parentMessageElement\n        *ngIf=\"mode === 'thread'\"\n        data-testid=\"parent-message\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage }\n          \"\n        ></ng-container>\n        <div class=\"str-chat__thread-start\" translate>\n          streamChat.Start of a new thread\n        </div>\n      </li>\n      <li\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: message }\n          \"\n        ></ng-container>\n      </li>\n    </ul>\n  </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n  <button\n    data-testid=\"scroll-to-bottom\"\n    *ngIf=\"isUserScrolledUp\"\n    class=\"\n      str-chat__message-notification\n      str-chat__message-notification-right\n      str-chat__message-notification-scroll-down\n    \"\n    (keyup.enter)=\"scrollToBottom()\"\n    (click)=\"scrollToBottom()\"\n  >\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-down-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n  <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        messageTemplate;\n        context: {\n          message: message,\n          areReactionsEnabled: areReactionsEnabled,\n          canReactToMessage: canReactToMessage,\n          lastSentMessageId: !!(\n            lastSentMessageId && message?.id === lastSentMessageId\n          ),\n          canReceiveReadEvents: canReceiveReadEvents,\n          messageInputTemplate: messageInputTemplate,\n          mentionTemplate: mentionTemplate,\n          mode: mode\n        }\n      \"\n    ></ng-container>\n  </ng-container>\n  <ng-template #defaultMessageTemplate>\n    <stream-message\n      [message]=\"message\"\n      [areReactionsEnabled]=\"areReactionsEnabled\"\n      [canReactToMessage]=\"canReactToMessage\"\n      [isLastSentMessage]=\"\n        !!(lastSentMessageId && message?.id === lastSentMessageId)\n      \"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [canReceiveReadEvents]=\"canReceiveReadEvents\"\n      [messageInputTemplate]=\"messageInputTemplate\"\n      [mentionTemplate]=\"mentionTemplate\"\n      [mode]=\"mode\"\n    ></stream-message>\n  </ng-template>\n</ng-template>\n"]}
@@ -1,5 +1,5 @@
1
1
  import { v4 as uuidv4 } from 'uuid';
2
- export const createMessagePreview = (user, text, attachments, mentionedUsers) => {
2
+ export const createMessagePreview = (user, text, attachments = [], mentionedUsers = [], parentId = undefined, quotedMessageId = undefined) => {
3
3
  const clientSideId = `${user.id}-${uuidv4()}`;
4
4
  return {
5
5
  __html: text,
@@ -13,6 +13,8 @@ export const createMessagePreview = (user, text, attachments, mentionedUsers) =>
13
13
  user,
14
14
  attachments,
15
15
  mentioned_users: mentionedUsers,
16
+ parent_id: parentId,
17
+ quoted_message_id: quotedMessageId,
16
18
  };
17
19
  };
18
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1wcmV2aWV3LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtcHJldmlldy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsRUFBRSxJQUFJLE1BQU0sRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUVwQyxNQUFNLENBQUMsTUFBTSxvQkFBb0IsR0FBRyxDQUNsQyxJQUFrQixFQUNsQixJQUFZLEVBQ1osV0FBeUIsRUFDekIsY0FBOEIsRUFDOUIsRUFBRTtJQUNGLE1BQU0sWUFBWSxHQUFHLEdBQUcsSUFBSSxDQUFDLEVBQUUsSUFBSSxNQUFNLEVBQUUsRUFBRSxDQUFDO0lBRTlDLE9BQU87UUFDTCxNQUFNLEVBQUUsSUFBSTtRQUNaLFVBQVUsRUFBRSxJQUFJLElBQUksRUFBRTtRQUN0QixJQUFJLEVBQUUsSUFBSTtRQUNWLEVBQUUsRUFBRSxZQUFZO1FBQ2hCLFNBQVMsRUFBRSxFQUFFO1FBQ2IsTUFBTSxFQUFFLFNBQVM7UUFDakIsSUFBSTtRQUNKLElBQUksRUFBRSxTQUFTO1FBQ2YsSUFBSTtRQUNKLFdBQVc7UUFDWCxlQUFlLEVBQUUsY0FBYztLQUNGLENBQUM7QUFDbEMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXR0YWNobWVudCwgTWVzc2FnZVJlc3BvbnNlLCBVc2VyUmVzcG9uc2UgfSBmcm9tICdzdHJlYW0tY2hhdCc7XG5pbXBvcnQgeyB2NCBhcyB1dWlkdjQgfSBmcm9tICd1dWlkJztcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZU1lc3NhZ2VQcmV2aWV3ID0gKFxuICB1c2VyOiBVc2VyUmVzcG9uc2UsXG4gIHRleHQ6IHN0cmluZyxcbiAgYXR0YWNobWVudHM6IEF0dGFjaG1lbnRbXSxcbiAgbWVudGlvbmVkVXNlcnM6IFVzZXJSZXNwb25zZVtdXG4pID0+IHtcbiAgY29uc3QgY2xpZW50U2lkZUlkID0gYCR7dXNlci5pZH0tJHt1dWlkdjQoKX1gO1xuXG4gIHJldHVybiB7XG4gICAgX19odG1sOiB0ZXh0LFxuICAgIGNyZWF0ZWRfYXQ6IG5ldyBEYXRlKCksXG4gICAgaHRtbDogdGV4dCxcbiAgICBpZDogY2xpZW50U2lkZUlkLFxuICAgIHJlYWN0aW9uczogW10sXG4gICAgc3RhdHVzOiAnc2VuZGluZycsXG4gICAgdGV4dCxcbiAgICB0eXBlOiAncmVndWxhcicsXG4gICAgdXNlcixcbiAgICBhdHRhY2htZW50cyxcbiAgICBtZW50aW9uZWRfdXNlcnM6IG1lbnRpb25lZFVzZXJzLFxuICB9IGFzIHVua25vd24gYXMgTWVzc2FnZVJlc3BvbnNlO1xufTtcbiJdfQ==
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1wcmV2aWV3LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtcHJldmlldy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsRUFBRSxJQUFJLE1BQU0sRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUVwQyxNQUFNLENBQUMsTUFBTSxvQkFBb0IsR0FBRyxDQUNsQyxJQUFrQixFQUNsQixJQUFZLEVBQ1osY0FBNEIsRUFBRSxFQUM5QixpQkFBaUMsRUFBRSxFQUNuQyxXQUErQixTQUFTLEVBQ3hDLGtCQUFzQyxTQUFTLEVBQy9DLEVBQUU7SUFDRixNQUFNLFlBQVksR0FBRyxHQUFHLElBQUksQ0FBQyxFQUFFLElBQUksTUFBTSxFQUFFLEVBQUUsQ0FBQztJQUU5QyxPQUFPO1FBQ0wsTUFBTSxFQUFFLElBQUk7UUFDWixVQUFVLEVBQUUsSUFBSSxJQUFJLEVBQUU7UUFDdEIsSUFBSSxFQUFFLElBQUk7UUFDVixFQUFFLEVBQUUsWUFBWTtRQUNoQixTQUFTLEVBQUUsRUFBRTtRQUNiLE1BQU0sRUFBRSxTQUFTO1FBQ2pCLElBQUk7UUFDSixJQUFJLEVBQUUsU0FBUztRQUNmLElBQUk7UUFDSixXQUFXO1FBQ1gsZUFBZSxFQUFFLGNBQWM7UUFDL0IsU0FBUyxFQUFFLFFBQVE7UUFDbkIsaUJBQWlCLEVBQUUsZUFBZTtLQUNMLENBQUM7QUFDbEMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXR0YWNobWVudCwgTWVzc2FnZVJlc3BvbnNlLCBVc2VyUmVzcG9uc2UgfSBmcm9tICdzdHJlYW0tY2hhdCc7XG5pbXBvcnQgeyB2NCBhcyB1dWlkdjQgfSBmcm9tICd1dWlkJztcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZU1lc3NhZ2VQcmV2aWV3ID0gKFxuICB1c2VyOiBVc2VyUmVzcG9uc2UsXG4gIHRleHQ6IHN0cmluZyxcbiAgYXR0YWNobWVudHM6IEF0dGFjaG1lbnRbXSA9IFtdLFxuICBtZW50aW9uZWRVc2VyczogVXNlclJlc3BvbnNlW10gPSBbXSxcbiAgcGFyZW50SWQ6IHVuZGVmaW5lZCB8IHN0cmluZyA9IHVuZGVmaW5lZCxcbiAgcXVvdGVkTWVzc2FnZUlkOiB1bmRlZmluZWQgfCBzdHJpbmcgPSB1bmRlZmluZWRcbikgPT4ge1xuICBjb25zdCBjbGllbnRTaWRlSWQgPSBgJHt1c2VyLmlkfS0ke3V1aWR2NCgpfWA7XG5cbiAgcmV0dXJuIHtcbiAgICBfX2h0bWw6IHRleHQsXG4gICAgY3JlYXRlZF9hdDogbmV3IERhdGUoKSxcbiAgICBodG1sOiB0ZXh0LFxuICAgIGlkOiBjbGllbnRTaWRlSWQsXG4gICAgcmVhY3Rpb25zOiBbXSxcbiAgICBzdGF0dXM6ICdzZW5kaW5nJyxcbiAgICB0ZXh0LFxuICAgIHR5cGU6ICdyZWd1bGFyJyxcbiAgICB1c2VyLFxuICAgIGF0dGFjaG1lbnRzLFxuICAgIG1lbnRpb25lZF91c2VyczogbWVudGlvbmVkVXNlcnMsXG4gICAgcGFyZW50X2lkOiBwYXJlbnRJZCxcbiAgICBxdW90ZWRfbWVzc2FnZV9pZDogcXVvdGVkTWVzc2FnZUlkLFxuICB9IGFzIHVua25vd24gYXMgTWVzc2FnZVJlc3BvbnNlO1xufTtcbiJdfQ==
@@ -104,7 +104,7 @@ export class MessageReactionsComponent {
104
104
  }
105
105
  }
106
106
  MessageReactionsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageReactionsComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ChannelService }], target: i0.ɵɵFactoryTarget.Component });
107
- MessageReactionsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageReactionsComponent, selector: "stream-message-reactions", inputs: { messageId: "messageId", messageReactionCounts: "messageReactionCounts", isSelectorOpen: "isSelectorOpen", latestReactions: "latestReactions", ownReactions: "ownReactions" }, outputs: { isSelectorOpenChange: "isSelectorOpenChange" }, viewQueries: [{ propertyName: "selectorContainer", first: true, predicate: ["selectorContainer"], descendants: true }, { propertyName: "selectorTooltip", first: true, predicate: ["selectorTooltip"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n *ngIf=\"existingReactions.length > 0 && !isSelectorOpen\"\n class=\"str-chat__reaction-list\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n data-testid=\"reaction-list\"\n>\n <ul>\n <li\n *ngFor=\"\n let reactionType of existingReactions;\n trackBy: trackByMessageReaction\n \"\n data-testclass=\"emoji\"\n >\n <span class=\"emoji\">\n {{ getEmojiByReaction(reactionType) }}\n </span>\n &nbsp;\n </li>\n <li>\n <span\n data-testid=\"reactions-count\"\n class=\"str-chat__reaction-list--counter\"\n >{{ reactionsCount }}</span\n >\n </li>\n </ul>\n</div>\n\n<div\n #selectorContainer\n class=\"str-chat__reaction-selector\"\n *ngIf=\"isSelectorOpen\"\n data-testid=\"reaction-selector\"\n>\n <div\n *ngIf=\"tooltipText\"\n data-testid=\"tooltip\"\n #selectorTooltip\n class=\"str-chat__reaction-selector-tooltip\"\n [ngStyle]=\"{\n left: tooltipPositions?.tooltip + 'px',\n visibility: tooltipPositions ? 'visible' : 'hidden'\n }\"\n >\n <div\n class=\"arrow\"\n [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n ></div>\n <span class=\"latest-user-username\">\n {{ tooltipText }}\n </span>\n </div>\n <ul class=\"str-chat__message-reactions-list\">\n <li\n class=\"str-chat__message-reactions-list-item\"\n *ngFor=\"\n let reactionType of reactionOptions;\n trackBy: trackByMessageReaction\n \"\n data-testclass=\"emoji-option\"\n (click)=\"react(reactionType)\"\n (keyup.enter)=\"react(reactionType)\"\n >\n <div\n *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n class=\"latest-user\"\n (click)=\"hideTooltip()\"\n (keyup.enter)=\"hideTooltip()\"\n (mouseenter)=\"showTooltip($event, reactionType)\"\n (mouseleave)=\"hideTooltip()\"\n attr.data-testid=\"{{ reactionType }}-last-user\"\n >\n <stream-avatar\n attr.data-testid=\"{{ reactionType }}-avatar\"\n [imageUrl]=\"user.image\"\n [name]=\"user.name || user.id\"\n [size]=\"20\"\n ></stream-avatar>\n </div>\n <span class=\"emoji\" style=\"width: 20px; height: 20px; top: 10px\">\n {{ getEmojiByReaction(reactionType) }}\n </span>\n <span\n *ngIf=\"\n messageReactionCounts[reactionType] &&\n messageReactionCounts[reactionType]! > 0\n \"\n class=\"str-chat__message-reactions-list-item__count\"\n attr.data-testid=\"{{ reactionType }}-reaction-count\"\n >\n {{ messageReactionCounts[reactionType] }}\n </span>\n </li>\n </ul>\n</div>\n", styles: [".emoji {position: relative; display: inline-block; }"], components: [{ type: i2.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
107
+ MessageReactionsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageReactionsComponent, selector: "stream-message-reactions", inputs: { messageId: "messageId", messageReactionCounts: "messageReactionCounts", isSelectorOpen: "isSelectorOpen", latestReactions: "latestReactions", ownReactions: "ownReactions" }, outputs: { isSelectorOpenChange: "isSelectorOpenChange" }, viewQueries: [{ propertyName: "selectorContainer", first: true, predicate: ["selectorContainer"], descendants: true }, { propertyName: "selectorTooltip", first: true, predicate: ["selectorTooltip"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n *ngIf=\"existingReactions.length > 0 && !isSelectorOpen\"\n class=\"str-chat__reaction-list\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n data-testid=\"reaction-list\"\n>\n <ul>\n <li\n *ngFor=\"\n let reactionType of existingReactions;\n trackBy: trackByMessageReaction\n \"\n data-testclass=\"emoji\"\n >\n <span class=\"emoji\">\n {{ getEmojiByReaction(reactionType) }}\n </span>\n &nbsp;\n </li>\n <li>\n <span\n data-testid=\"reactions-count\"\n class=\"str-chat__reaction-list--counter\"\n >{{ reactionsCount }}</span\n >\n </li>\n </ul>\n</div>\n\n<div\n #selectorContainer\n class=\"str-chat__reaction-selector\"\n *ngIf=\"isSelectorOpen\"\n data-testid=\"reaction-selector\"\n>\n <div\n *ngIf=\"tooltipText\"\n data-testid=\"tooltip\"\n #selectorTooltip\n class=\"str-chat__reaction-selector-tooltip\"\n [ngStyle]=\"{\n left: tooltipPositions?.tooltip + 'px',\n visibility: tooltipPositions ? 'visible' : 'hidden'\n }\"\n >\n <div\n class=\"arrow\"\n [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n ></div>\n <span class=\"latest-user-username\">\n {{ tooltipText }}\n </span>\n </div>\n <ul class=\"str-chat__message-reactions-list\">\n <li\n class=\"str-chat__message-reactions-list-item emoji-mart-emoji\"\n *ngFor=\"\n let reactionType of reactionOptions;\n trackBy: trackByMessageReaction\n \"\n data-testclass=\"emoji-option\"\n (click)=\"react(reactionType)\"\n (keyup.enter)=\"react(reactionType)\"\n >\n <div\n *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n class=\"latest-user\"\n (click)=\"hideTooltip()\"\n (keyup.enter)=\"hideTooltip()\"\n (mouseenter)=\"showTooltip($event, reactionType)\"\n (mouseleave)=\"hideTooltip()\"\n attr.data-testid=\"{{ reactionType }}-last-user\"\n >\n <stream-avatar\n attr.data-testid=\"{{ reactionType }}-avatar\"\n [imageUrl]=\"user.image\"\n [name]=\"user.name || user.id\"\n [size]=\"20\"\n ></stream-avatar>\n </div>\n <span class=\"emoji\" style=\"width: 20px; height: 20px; top: 10px\">\n {{ getEmojiByReaction(reactionType) }}\n </span>\n <span\n *ngIf=\"\n messageReactionCounts[reactionType] &&\n messageReactionCounts[reactionType]! > 0\n \"\n class=\"str-chat__message-reactions-list-item__count\"\n attr.data-testid=\"{{ reactionType }}-reaction-count\"\n >\n {{ messageReactionCounts[reactionType] }}\n </span>\n </li>\n </ul>\n</div>\n", styles: [".emoji {position: relative; display: inline-block; }"], components: [{ type: i2.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
108
108
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageReactionsComponent, decorators: [{
109
109
  type: Component,
110
110
  args: [{
@@ -131,4 +131,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
131
131
  type: ViewChild,
132
132
  args: ['selectorTooltip']
133
133
  }] } });
134
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-reactions.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ReactionResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { DefaultReactionType, DefaultUserType } from '../types';\n\nexport type MessageReactionType =\n  | 'angry'\n  | 'haha'\n  | 'like'\n  | 'love'\n  | 'sad'\n  | 'wow';\n\nconst emojiReactionsMapping: { [key in MessageReactionType]: string } = {\n  like: '👍',\n  angry: '😠',\n  love: '❤️',\n  haha: '😂',\n  wow: '😮',\n  sad: '😞',\n};\n@Component({\n  selector: 'stream-message-reactions',\n  templateUrl: './message-reactions.component.html',\n  styles: ['.emoji {position: relative; display: inline-block; }'],\n})\nexport class MessageReactionsComponent implements AfterViewChecked, OnChanges {\n  @Input() messageId: string | undefined;\n  @Input() messageReactionCounts: { [key in MessageReactionType]?: number } =\n    {};\n  @Input() isSelectorOpen: boolean = false;\n  @Output() readonly isSelectorOpenChange = new EventEmitter<boolean>();\n  @Input() latestReactions: ReactionResponse<\n    DefaultReactionType,\n    DefaultUserType\n  >[] = [];\n  @Input() ownReactions: ReactionResponse<\n    DefaultReactionType,\n    DefaultUserType\n  >[] = [];\n  tooltipPositions: { arrow: number; tooltip: number } | undefined;\n  tooltipText: string | undefined;\n  @ViewChild('selectorContainer') private selectorContainer:\n    | ElementRef<HTMLElement>\n    | undefined;\n  @ViewChild('selectorTooltip') private selectorTooltip:\n    | ElementRef<HTMLElement>\n    | undefined;\n  currentTooltipTarget: HTMLElement | undefined;\n\n  constructor(\n    private cdRef: ChangeDetectorRef,\n    private channelService: ChannelService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.isSelectorOpen) {\n      this.isSelectorOpen\n        ? setTimeout(() => this.watchForOutsideClicks()) // setTimeout: wait for current click to bubble up, and only watch for clicks after that\n        : this.stopWatchForOutsideClicks();\n    }\n  }\n\n  ngAfterViewChecked(): void {\n    if (this.tooltipText && !this.tooltipPositions) {\n      this.setTooltipPosition();\n      this.cdRef.detectChanges();\n    }\n  }\n\n  get existingReactions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionCounts).filter(\n      (k) => this.messageReactionCounts[k as MessageReactionType]! > 0\n    ) as MessageReactionType[];\n  }\n\n  get reactionsCount() {\n    return Object.values(this.messageReactionCounts).reduce(\n      (total, count) => total + count,\n      0\n    );\n  }\n\n  get reactionOptions(): MessageReactionType[] {\n    return ['like', 'love', 'haha', 'wow', 'sad', 'angry'];\n  }\n\n  getLatestUserByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions.find((r) => r.type === reactionType && r.user)\n      ?.user;\n  }\n\n  getEmojiByReaction(reactionType: MessageReactionType) {\n    return emojiReactionsMapping[reactionType];\n  }\n\n  getUsersByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions\n      .filter((r) => r.type === reactionType)\n      .map((r) => r.user?.name || r.user?.id)\n      .filter((i) => !!i)\n      .join(', ');\n  }\n\n  showTooltip(event: Event, reactionType: MessageReactionType) {\n    this.currentTooltipTarget = event.target as HTMLElement;\n    this.tooltipText = this.getUsersByReaction(reactionType);\n  }\n\n  hideTooltip() {\n    this.tooltipText = undefined;\n    this.currentTooltipTarget = undefined;\n    this.tooltipPositions = undefined;\n  }\n\n  trackByMessageReaction(index: number, item: MessageReactionType) {\n    return item;\n  }\n\n  react(type: MessageReactionType) {\n    this.ownReactions.find((r) => r.type === type)\n      ? void this.channelService.removeReaction(this.messageId!, type)\n      : void this.channelService.addReaction(this.messageId!, type);\n  }\n\n  private eventHandler = (event: Event) => {\n    if (!this.selectorContainer?.nativeElement.contains(event.target as Node)) {\n      this.isSelectorOpenChange.emit(false);\n    }\n  };\n\n  private watchForOutsideClicks() {\n    window.addEventListener('click', this.eventHandler);\n  }\n\n  private stopWatchForOutsideClicks() {\n    window.removeEventListener('click', this.eventHandler);\n  }\n\n  private setTooltipPosition() {\n    const tooltip = this.selectorTooltip?.nativeElement.getBoundingClientRect();\n    const target = this.currentTooltipTarget?.getBoundingClientRect();\n\n    const container =\n      this.selectorContainer?.nativeElement.getBoundingClientRect();\n\n    if (!tooltip || !target || !container) return;\n\n    const tooltipPosition =\n      tooltip.width === container.width || tooltip.x < container.x\n        ? 0\n        : target.left + target.width / 2 - container.left - tooltip.width / 2;\n\n    const arrowPosition =\n      target.x - tooltip.x + target.width / 2 - tooltipPosition;\n\n    this.tooltipPositions = {\n      tooltip: tooltipPosition,\n      arrow: arrowPosition,\n    };\n  }\n}\n","<div\n  *ngIf=\"existingReactions.length > 0 && !isSelectorOpen\"\n  class=\"str-chat__reaction-list\"\n  [class.str-chat__reaction-list--reverse]=\"true\"\n  data-testid=\"reaction-list\"\n>\n  <ul>\n    <li\n      *ngFor=\"\n        let reactionType of existingReactions;\n        trackBy: trackByMessageReaction\n      \"\n      data-testclass=\"emoji\"\n    >\n      <span class=\"emoji\">\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      &nbsp;\n    </li>\n    <li>\n      <span\n        data-testid=\"reactions-count\"\n        class=\"str-chat__reaction-list--counter\"\n        >{{ reactionsCount }}</span\n      >\n    </li>\n  </ul>\n</div>\n\n<div\n  #selectorContainer\n  class=\"str-chat__reaction-selector\"\n  *ngIf=\"isSelectorOpen\"\n  data-testid=\"reaction-selector\"\n>\n  <div\n    *ngIf=\"tooltipText\"\n    data-testid=\"tooltip\"\n    #selectorTooltip\n    class=\"str-chat__reaction-selector-tooltip\"\n    [ngStyle]=\"{\n      left: tooltipPositions?.tooltip + 'px',\n      visibility: tooltipPositions ? 'visible' : 'hidden'\n    }\"\n  >\n    <div\n      class=\"arrow\"\n      [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n    ></div>\n    <span class=\"latest-user-username\">\n      {{ tooltipText }}\n    </span>\n  </div>\n  <ul class=\"str-chat__message-reactions-list\">\n    <li\n      class=\"str-chat__message-reactions-list-item\"\n      *ngFor=\"\n        let reactionType of reactionOptions;\n        trackBy: trackByMessageReaction\n      \"\n      data-testclass=\"emoji-option\"\n      (click)=\"react(reactionType)\"\n      (keyup.enter)=\"react(reactionType)\"\n    >\n      <div\n        *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n        class=\"latest-user\"\n        (click)=\"hideTooltip()\"\n        (keyup.enter)=\"hideTooltip()\"\n        (mouseenter)=\"showTooltip($event, reactionType)\"\n        (mouseleave)=\"hideTooltip()\"\n        attr.data-testid=\"{{ reactionType }}-last-user\"\n      >\n        <stream-avatar\n          attr.data-testid=\"{{ reactionType }}-avatar\"\n          [imageUrl]=\"user.image\"\n          [name]=\"user.name || user.id\"\n          [size]=\"20\"\n        ></stream-avatar>\n      </div>\n      <span class=\"emoji\" style=\"width: 20px; height: 20px; top: 10px\">\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      <span\n        *ngIf=\"\n          messageReactionCounts[reactionType] &&\n          messageReactionCounts[reactionType]! > 0\n        \"\n        class=\"str-chat__message-reactions-list-item__count\"\n        attr.data-testid=\"{{ reactionType }}-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n  </ul>\n</div>\n"]}
134
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-reactions.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ReactionResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { DefaultReactionType, DefaultUserType } from '../types';\n\nexport type MessageReactionType =\n  | 'angry'\n  | 'haha'\n  | 'like'\n  | 'love'\n  | 'sad'\n  | 'wow';\n\nconst emojiReactionsMapping: { [key in MessageReactionType]: string } = {\n  like: '👍',\n  angry: '😠',\n  love: '❤️',\n  haha: '😂',\n  wow: '😮',\n  sad: '😞',\n};\n@Component({\n  selector: 'stream-message-reactions',\n  templateUrl: './message-reactions.component.html',\n  styles: ['.emoji {position: relative; display: inline-block; }'],\n})\nexport class MessageReactionsComponent implements AfterViewChecked, OnChanges {\n  @Input() messageId: string | undefined;\n  @Input() messageReactionCounts: { [key in MessageReactionType]?: number } =\n    {};\n  @Input() isSelectorOpen: boolean = false;\n  @Output() readonly isSelectorOpenChange = new EventEmitter<boolean>();\n  @Input() latestReactions: ReactionResponse<\n    DefaultReactionType,\n    DefaultUserType\n  >[] = [];\n  @Input() ownReactions: ReactionResponse<\n    DefaultReactionType,\n    DefaultUserType\n  >[] = [];\n  tooltipPositions: { arrow: number; tooltip: number } | undefined;\n  tooltipText: string | undefined;\n  @ViewChild('selectorContainer') private selectorContainer:\n    | ElementRef<HTMLElement>\n    | undefined;\n  @ViewChild('selectorTooltip') private selectorTooltip:\n    | ElementRef<HTMLElement>\n    | undefined;\n  currentTooltipTarget: HTMLElement | undefined;\n\n  constructor(\n    private cdRef: ChangeDetectorRef,\n    private channelService: ChannelService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.isSelectorOpen) {\n      this.isSelectorOpen\n        ? setTimeout(() => this.watchForOutsideClicks()) // setTimeout: wait for current click to bubble up, and only watch for clicks after that\n        : this.stopWatchForOutsideClicks();\n    }\n  }\n\n  ngAfterViewChecked(): void {\n    if (this.tooltipText && !this.tooltipPositions) {\n      this.setTooltipPosition();\n      this.cdRef.detectChanges();\n    }\n  }\n\n  get existingReactions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionCounts).filter(\n      (k) => this.messageReactionCounts[k as MessageReactionType]! > 0\n    ) as MessageReactionType[];\n  }\n\n  get reactionsCount() {\n    return Object.values(this.messageReactionCounts).reduce(\n      (total, count) => total + count,\n      0\n    );\n  }\n\n  get reactionOptions(): MessageReactionType[] {\n    return ['like', 'love', 'haha', 'wow', 'sad', 'angry'];\n  }\n\n  getLatestUserByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions.find((r) => r.type === reactionType && r.user)\n      ?.user;\n  }\n\n  getEmojiByReaction(reactionType: MessageReactionType) {\n    return emojiReactionsMapping[reactionType];\n  }\n\n  getUsersByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions\n      .filter((r) => r.type === reactionType)\n      .map((r) => r.user?.name || r.user?.id)\n      .filter((i) => !!i)\n      .join(', ');\n  }\n\n  showTooltip(event: Event, reactionType: MessageReactionType) {\n    this.currentTooltipTarget = event.target as HTMLElement;\n    this.tooltipText = this.getUsersByReaction(reactionType);\n  }\n\n  hideTooltip() {\n    this.tooltipText = undefined;\n    this.currentTooltipTarget = undefined;\n    this.tooltipPositions = undefined;\n  }\n\n  trackByMessageReaction(index: number, item: MessageReactionType) {\n    return item;\n  }\n\n  react(type: MessageReactionType) {\n    this.ownReactions.find((r) => r.type === type)\n      ? void this.channelService.removeReaction(this.messageId!, type)\n      : void this.channelService.addReaction(this.messageId!, type);\n  }\n\n  private eventHandler = (event: Event) => {\n    if (!this.selectorContainer?.nativeElement.contains(event.target as Node)) {\n      this.isSelectorOpenChange.emit(false);\n    }\n  };\n\n  private watchForOutsideClicks() {\n    window.addEventListener('click', this.eventHandler);\n  }\n\n  private stopWatchForOutsideClicks() {\n    window.removeEventListener('click', this.eventHandler);\n  }\n\n  private setTooltipPosition() {\n    const tooltip = this.selectorTooltip?.nativeElement.getBoundingClientRect();\n    const target = this.currentTooltipTarget?.getBoundingClientRect();\n\n    const container =\n      this.selectorContainer?.nativeElement.getBoundingClientRect();\n\n    if (!tooltip || !target || !container) return;\n\n    const tooltipPosition =\n      tooltip.width === container.width || tooltip.x < container.x\n        ? 0\n        : target.left + target.width / 2 - container.left - tooltip.width / 2;\n\n    const arrowPosition =\n      target.x - tooltip.x + target.width / 2 - tooltipPosition;\n\n    this.tooltipPositions = {\n      tooltip: tooltipPosition,\n      arrow: arrowPosition,\n    };\n  }\n}\n","<div\n  *ngIf=\"existingReactions.length > 0 && !isSelectorOpen\"\n  class=\"str-chat__reaction-list\"\n  [class.str-chat__reaction-list--reverse]=\"true\"\n  data-testid=\"reaction-list\"\n>\n  <ul>\n    <li\n      *ngFor=\"\n        let reactionType of existingReactions;\n        trackBy: trackByMessageReaction\n      \"\n      data-testclass=\"emoji\"\n    >\n      <span class=\"emoji\">\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      &nbsp;\n    </li>\n    <li>\n      <span\n        data-testid=\"reactions-count\"\n        class=\"str-chat__reaction-list--counter\"\n        >{{ reactionsCount }}</span\n      >\n    </li>\n  </ul>\n</div>\n\n<div\n  #selectorContainer\n  class=\"str-chat__reaction-selector\"\n  *ngIf=\"isSelectorOpen\"\n  data-testid=\"reaction-selector\"\n>\n  <div\n    *ngIf=\"tooltipText\"\n    data-testid=\"tooltip\"\n    #selectorTooltip\n    class=\"str-chat__reaction-selector-tooltip\"\n    [ngStyle]=\"{\n      left: tooltipPositions?.tooltip + 'px',\n      visibility: tooltipPositions ? 'visible' : 'hidden'\n    }\"\n  >\n    <div\n      class=\"arrow\"\n      [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n    ></div>\n    <span class=\"latest-user-username\">\n      {{ tooltipText }}\n    </span>\n  </div>\n  <ul class=\"str-chat__message-reactions-list\">\n    <li\n      class=\"str-chat__message-reactions-list-item emoji-mart-emoji\"\n      *ngFor=\"\n        let reactionType of reactionOptions;\n        trackBy: trackByMessageReaction\n      \"\n      data-testclass=\"emoji-option\"\n      (click)=\"react(reactionType)\"\n      (keyup.enter)=\"react(reactionType)\"\n    >\n      <div\n        *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n        class=\"latest-user\"\n        (click)=\"hideTooltip()\"\n        (keyup.enter)=\"hideTooltip()\"\n        (mouseenter)=\"showTooltip($event, reactionType)\"\n        (mouseleave)=\"hideTooltip()\"\n        attr.data-testid=\"{{ reactionType }}-last-user\"\n      >\n        <stream-avatar\n          attr.data-testid=\"{{ reactionType }}-avatar\"\n          [imageUrl]=\"user.image\"\n          [name]=\"user.name || user.id\"\n          [size]=\"20\"\n        ></stream-avatar>\n      </div>\n      <span class=\"emoji\" style=\"width: 20px; height: 20px; top: 10px\">\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      <span\n        *ngIf=\"\n          messageReactionCounts[reactionType] &&\n          messageReactionCounts[reactionType]! > 0\n        \"\n        class=\"str-chat__message-reactions-list-item__count\"\n        attr.data-testid=\"{{ reactionType }}-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n  </ul>\n</div>\n"]}
@@ -19,6 +19,7 @@ import { AttachmentPreviewListComponent } from './attachment-preview-list/attach
19
19
  import { ModalComponent } from './modal/modal.component';
20
20
  import { TextareaDirective } from './message-input/textarea.directive';
21
21
  import { StreamAvatarModule } from './stream-avatar.module';
22
+ import { ThreadComponent } from './thread/thread.component';
22
23
  import * as i0 from "@angular/core";
23
24
  export class StreamChatModule {
24
25
  }
@@ -39,7 +40,8 @@ StreamChatModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", versio
39
40
  NotificationListComponent,
40
41
  AttachmentPreviewListComponent,
41
42
  ModalComponent,
42
- TextareaDirective], imports: [CommonModule, TranslateModule, StreamAvatarModule], exports: [ChannelComponent,
43
+ TextareaDirective,
44
+ ThreadComponent], imports: [CommonModule, TranslateModule, StreamAvatarModule], exports: [ChannelComponent,
43
45
  ChannelHeaderComponent,
44
46
  ChannelListComponent,
45
47
  ChannelPreviewComponent,
@@ -55,7 +57,8 @@ StreamChatModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", versio
55
57
  NotificationListComponent,
56
58
  AttachmentPreviewListComponent,
57
59
  ModalComponent,
58
- StreamAvatarModule] });
60
+ StreamAvatarModule,
61
+ ThreadComponent] });
59
62
  StreamChatModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: StreamChatModule, imports: [[CommonModule, TranslateModule, StreamAvatarModule], StreamAvatarModule] });
60
63
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: StreamChatModule, decorators: [{
61
64
  type: NgModule,
@@ -78,6 +81,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
78
81
  AttachmentPreviewListComponent,
79
82
  ModalComponent,
80
83
  TextareaDirective,
84
+ ThreadComponent,
81
85
  ],
82
86
  imports: [CommonModule, TranslateModule, StreamAvatarModule],
83
87
  exports: [
@@ -98,7 +102,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
98
102
  AttachmentPreviewListComponent,
99
103
  ModalComponent,
100
104
  StreamAvatarModule,
105
+ ThreadComponent,
101
106
  ],
102
107
  }]
103
108
  }] });
104
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyZWFtLWNoYXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3N0cmVhbS1jaGF0Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQy9ELE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLDJDQUEyQyxDQUFDO0FBQ25GLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBQzdFLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDZDQUE2QyxDQUFDO0FBQ3RGLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQy9ELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLHlDQUF5QyxDQUFDO0FBQ2hGLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxpREFBaUQsQ0FBQztBQUM1RixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDdEQsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0scURBQXFELENBQUM7QUFDakcsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0saURBQWlELENBQUM7QUFDNUYsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDOUUsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0saURBQWlELENBQUM7QUFDNUYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3RELE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDZEQUE2RCxDQUFDO0FBQzdHLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUN2RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7QUEyQzVELE1BQU0sT0FBTyxnQkFBZ0I7OzZHQUFoQixnQkFBZ0I7OEdBQWhCLGdCQUFnQixpQkF2Q3pCLGdCQUFnQjtRQUNoQixzQkFBc0I7UUFDdEIsb0JBQW9CO1FBQ3BCLHVCQUF1QjtRQUN2QixnQkFBZ0I7UUFDaEIscUJBQXFCO1FBQ3JCLG9CQUFvQjtRQUNwQix5QkFBeUI7UUFDekIsYUFBYTtRQUNiLDBCQUEwQjtRQUMxQix1QkFBdUI7UUFDdkIseUJBQXlCO1FBQ3pCLHFCQUFxQjtRQUNyQix5QkFBeUI7UUFDekIsOEJBQThCO1FBQzlCLGNBQWM7UUFDZCxpQkFBaUIsYUFFVCxZQUFZLEVBQUUsZUFBZSxFQUFFLGtCQUFrQixhQUV6RCxnQkFBZ0I7UUFDaEIsc0JBQXNCO1FBQ3RCLG9CQUFvQjtRQUNwQix1QkFBdUI7UUFDdkIsZ0JBQWdCO1FBQ2hCLHFCQUFxQjtRQUNyQixvQkFBb0I7UUFDcEIseUJBQXlCO1FBQ3pCLGFBQWE7UUFDYiwwQkFBMEI7UUFDMUIsdUJBQXVCO1FBQ3ZCLHlCQUF5QjtRQUN6QixxQkFBcUI7UUFDckIseUJBQXlCO1FBQ3pCLDhCQUE4QjtRQUM5QixjQUFjO1FBQ2Qsa0JBQWtCOzhHQUdULGdCQUFnQixZQXJCbEIsQ0FBQyxZQUFZLEVBQUUsZUFBZSxFQUFFLGtCQUFrQixDQUFDLEVBa0IxRCxrQkFBa0I7MkZBR1QsZ0JBQWdCO2tCQXpDNUIsUUFBUTttQkFBQztvQkFDUixZQUFZLEVBQUU7d0JBQ1osZ0JBQWdCO3dCQUNoQixzQkFBc0I7d0JBQ3RCLG9CQUFvQjt3QkFDcEIsdUJBQXVCO3dCQUN2QixnQkFBZ0I7d0JBQ2hCLHFCQUFxQjt3QkFDckIsb0JBQW9CO3dCQUNwQix5QkFBeUI7d0JBQ3pCLGFBQWE7d0JBQ2IsMEJBQTBCO3dCQUMxQix1QkFBdUI7d0JBQ3ZCLHlCQUF5Qjt3QkFDekIscUJBQXFCO3dCQUNyQix5QkFBeUI7d0JBQ3pCLDhCQUE4Qjt3QkFDOUIsY0FBYzt3QkFDZCxpQkFBaUI7cUJBQ2xCO29CQUNELE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxlQUFlLEVBQUUsa0JBQWtCLENBQUM7b0JBQzVELE9BQU8sRUFBRTt3QkFDUCxnQkFBZ0I7d0JBQ2hCLHNCQUFzQjt3QkFDdEIsb0JBQW9CO3dCQUNwQix1QkFBdUI7d0JBQ3ZCLGdCQUFnQjt3QkFDaEIscUJBQXFCO3dCQUNyQixvQkFBb0I7d0JBQ3BCLHlCQUF5Qjt3QkFDekIsYUFBYTt3QkFDYiwwQkFBMEI7d0JBQzFCLHVCQUF1Qjt3QkFDdkIseUJBQXlCO3dCQUN6QixxQkFBcUI7d0JBQ3JCLHlCQUF5Qjt3QkFDekIsOEJBQThCO3dCQUM5QixjQUFjO3dCQUNkLGtCQUFrQjtxQkFDbkI7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ2hhbm5lbENvbXBvbmVudCB9IGZyb20gJy4vY2hhbm5lbC9jaGFubmVsLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDaGFubmVsSGVhZGVyQ29tcG9uZW50IH0gZnJvbSAnLi9jaGFubmVsLWhlYWRlci9jaGFubmVsLWhlYWRlci5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ2hhbm5lbExpc3RDb21wb25lbnQgfSBmcm9tICcuL2NoYW5uZWwtbGlzdC9jaGFubmVsLWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IENoYW5uZWxQcmV2aWV3Q29tcG9uZW50IH0gZnJvbSAnLi9jaGFubmVsLXByZXZpZXcvY2hhbm5lbC1wcmV2aWV3LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBNZXNzYWdlQ29tcG9uZW50IH0gZnJvbSAnLi9tZXNzYWdlL21lc3NhZ2UuY29tcG9uZW50JztcbmltcG9ydCB7IE1lc3NhZ2VJbnB1dENvbXBvbmVudCB9IGZyb20gJy4vbWVzc2FnZS1pbnB1dC9tZXNzYWdlLWlucHV0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBNZXNzYWdlTGlzdENvbXBvbmVudCB9IGZyb20gJy4vbWVzc2FnZS1saXN0L21lc3NhZ2UtbGlzdC5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IExvYWRpbmdJbmRpY2F0b3JDb21wb25lbnQgfSBmcm9tICcuL2xvYWRpbmctaW5kaWNhdG9yL2xvYWRpbmctaW5kaWNhdG9yLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBJY29uQ29tcG9uZW50IH0gZnJvbSAnLi9pY29uL2ljb24uY29tcG9uZW50JztcbmltcG9ydCB7IE1lc3NhZ2VBY3Rpb25zQm94Q29tcG9uZW50IH0gZnJvbSAnLi9tZXNzYWdlLWFjdGlvbnMtYm94L21lc3NhZ2UtYWN0aW9ucy1ib3guY29tcG9uZW50JztcbmltcG9ydCB7IEF0dGFjaG1lbnRMaXN0Q29tcG9uZW50IH0gZnJvbSAnLi9hdHRhY2htZW50LWxpc3QvYXR0YWNobWVudC1saXN0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBNZXNzYWdlUmVhY3Rpb25zQ29tcG9uZW50IH0gZnJvbSAnLi9tZXNzYWdlLXJlYWN0aW9ucy9tZXNzYWdlLXJlYWN0aW9ucy5jb21wb25lbnQnO1xuaW1wb3J0IHsgTm90aWZpY2F0aW9uQ29tcG9uZW50IH0gZnJvbSAnLi9ub3RpZmljYXRpb24vbm90aWZpY2F0aW9uLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBOb3RpZmljYXRpb25MaXN0Q29tcG9uZW50IH0gZnJvbSAnLi9ub3RpZmljYXRpb24tbGlzdC9ub3RpZmljYXRpb24tbGlzdC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVHJhbnNsYXRlTW9kdWxlIH0gZnJvbSAnQG5neC10cmFuc2xhdGUvY29yZSc7XG5pbXBvcnQgeyBBdHRhY2htZW50UHJldmlld0xpc3RDb21wb25lbnQgfSBmcm9tICcuL2F0dGFjaG1lbnQtcHJldmlldy1saXN0L2F0dGFjaG1lbnQtcHJldmlldy1saXN0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vbW9kYWwvbW9kYWwuY29tcG9uZW50JztcbmltcG9ydCB7IFRleHRhcmVhRGlyZWN0aXZlIH0gZnJvbSAnLi9tZXNzYWdlLWlucHV0L3RleHRhcmVhLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBTdHJlYW1BdmF0YXJNb2R1bGUgfSBmcm9tICcuL3N0cmVhbS1hdmF0YXIubW9kdWxlJztcblxuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgQ2hhbm5lbENvbXBvbmVudCxcbiAgICBDaGFubmVsSGVhZGVyQ29tcG9uZW50LFxuICAgIENoYW5uZWxMaXN0Q29tcG9uZW50LFxuICAgIENoYW5uZWxQcmV2aWV3Q29tcG9uZW50LFxuICAgIE1lc3NhZ2VDb21wb25lbnQsXG4gICAgTWVzc2FnZUlucHV0Q29tcG9uZW50LFxuICAgIE1lc3NhZ2VMaXN0Q29tcG9uZW50LFxuICAgIExvYWRpbmdJbmRpY2F0b3JDb21wb25lbnQsXG4gICAgSWNvbkNvbXBvbmVudCxcbiAgICBNZXNzYWdlQWN0aW9uc0JveENvbXBvbmVudCxcbiAgICBBdHRhY2htZW50TGlzdENvbXBvbmVudCxcbiAgICBNZXNzYWdlUmVhY3Rpb25zQ29tcG9uZW50LFxuICAgIE5vdGlmaWNhdGlvbkNvbXBvbmVudCxcbiAgICBOb3RpZmljYXRpb25MaXN0Q29tcG9uZW50LFxuICAgIEF0dGFjaG1lbnRQcmV2aWV3TGlzdENvbXBvbmVudCxcbiAgICBNb2RhbENvbXBvbmVudCxcbiAgICBUZXh0YXJlYURpcmVjdGl2ZSxcbiAgXSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgVHJhbnNsYXRlTW9kdWxlLCBTdHJlYW1BdmF0YXJNb2R1bGVdLFxuICBleHBvcnRzOiBbXG4gICAgQ2hhbm5lbENvbXBvbmVudCxcbiAgICBDaGFubmVsSGVhZGVyQ29tcG9uZW50LFxuICAgIENoYW5uZWxMaXN0Q29tcG9uZW50LFxuICAgIENoYW5uZWxQcmV2aWV3Q29tcG9uZW50LFxuICAgIE1lc3NhZ2VDb21wb25lbnQsXG4gICAgTWVzc2FnZUlucHV0Q29tcG9uZW50LFxuICAgIE1lc3NhZ2VMaXN0Q29tcG9uZW50LFxuICAgIExvYWRpbmdJbmRpY2F0b3JDb21wb25lbnQsXG4gICAgSWNvbkNvbXBvbmVudCxcbiAgICBNZXNzYWdlQWN0aW9uc0JveENvbXBvbmVudCxcbiAgICBBdHRhY2htZW50TGlzdENvbXBvbmVudCxcbiAgICBNZXNzYWdlUmVhY3Rpb25zQ29tcG9uZW50LFxuICAgIE5vdGlmaWNhdGlvbkNvbXBvbmVudCxcbiAgICBOb3RpZmljYXRpb25MaXN0Q29tcG9uZW50LFxuICAgIEF0dGFjaG1lbnRQcmV2aWV3TGlzdENvbXBvbmVudCxcbiAgICBNb2RhbENvbXBvbmVudCxcbiAgICBTdHJlYW1BdmF0YXJNb2R1bGUsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIFN0cmVhbUNoYXRNb2R1bGUge31cbiJdfQ==
109
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyZWFtLWNoYXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3N0cmVhbS1jaGF0Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQy9ELE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLDJDQUEyQyxDQUFDO0FBQ25GLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBQzdFLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDZDQUE2QyxDQUFDO0FBQ3RGLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQy9ELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLHlDQUF5QyxDQUFDO0FBQ2hGLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxpREFBaUQsQ0FBQztBQUM1RixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDdEQsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0scURBQXFELENBQUM7QUFDakcsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0saURBQWlELENBQUM7QUFDNUYsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDOUUsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0saURBQWlELENBQUM7QUFDNUYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3RELE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDZEQUE2RCxDQUFDO0FBQzdHLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUN2RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUM1RCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7O0FBNkM1RCxNQUFNLE9BQU8sZ0JBQWdCOzs2R0FBaEIsZ0JBQWdCOzhHQUFoQixnQkFBZ0IsaUJBekN6QixnQkFBZ0I7UUFDaEIsc0JBQXNCO1FBQ3RCLG9CQUFvQjtRQUNwQix1QkFBdUI7UUFDdkIsZ0JBQWdCO1FBQ2hCLHFCQUFxQjtRQUNyQixvQkFBb0I7UUFDcEIseUJBQXlCO1FBQ3pCLGFBQWE7UUFDYiwwQkFBMEI7UUFDMUIsdUJBQXVCO1FBQ3ZCLHlCQUF5QjtRQUN6QixxQkFBcUI7UUFDckIseUJBQXlCO1FBQ3pCLDhCQUE4QjtRQUM5QixjQUFjO1FBQ2QsaUJBQWlCO1FBQ2pCLGVBQWUsYUFFUCxZQUFZLEVBQUUsZUFBZSxFQUFFLGtCQUFrQixhQUV6RCxnQkFBZ0I7UUFDaEIsc0JBQXNCO1FBQ3RCLG9CQUFvQjtRQUNwQix1QkFBdUI7UUFDdkIsZ0JBQWdCO1FBQ2hCLHFCQUFxQjtRQUNyQixvQkFBb0I7UUFDcEIseUJBQXlCO1FBQ3pCLGFBQWE7UUFDYiwwQkFBMEI7UUFDMUIsdUJBQXVCO1FBQ3ZCLHlCQUF5QjtRQUN6QixxQkFBcUI7UUFDckIseUJBQXlCO1FBQ3pCLDhCQUE4QjtRQUM5QixjQUFjO1FBQ2Qsa0JBQWtCO1FBQ2xCLGVBQWU7OEdBR04sZ0JBQWdCLFlBdEJsQixDQUFDLFlBQVksRUFBRSxlQUFlLEVBQUUsa0JBQWtCLENBQUMsRUFrQjFELGtCQUFrQjsyRkFJVCxnQkFBZ0I7a0JBM0M1QixRQUFRO21CQUFDO29CQUNSLFlBQVksRUFBRTt3QkFDWixnQkFBZ0I7d0JBQ2hCLHNCQUFzQjt3QkFDdEIsb0JBQW9CO3dCQUNwQix1QkFBdUI7d0JBQ3ZCLGdCQUFnQjt3QkFDaEIscUJBQXFCO3dCQUNyQixvQkFBb0I7d0JBQ3BCLHlCQUF5Qjt3QkFDekIsYUFBYTt3QkFDYiwwQkFBMEI7d0JBQzFCLHVCQUF1Qjt3QkFDdkIseUJBQXlCO3dCQUN6QixxQkFBcUI7d0JBQ3JCLHlCQUF5Qjt3QkFDekIsOEJBQThCO3dCQUM5QixjQUFjO3dCQUNkLGlCQUFpQjt3QkFDakIsZUFBZTtxQkFDaEI7b0JBQ0QsT0FBTyxFQUFFLENBQUMsWUFBWSxFQUFFLGVBQWUsRUFBRSxrQkFBa0IsQ0FBQztvQkFDNUQsT0FBTyxFQUFFO3dCQUNQLGdCQUFnQjt3QkFDaEIsc0JBQXNCO3dCQUN0QixvQkFBb0I7d0JBQ3BCLHVCQUF1Qjt3QkFDdkIsZ0JBQWdCO3dCQUNoQixxQkFBcUI7d0JBQ3JCLG9CQUFvQjt3QkFDcEIseUJBQXlCO3dCQUN6QixhQUFhO3dCQUNiLDBCQUEwQjt3QkFDMUIsdUJBQXVCO3dCQUN2Qix5QkFBeUI7d0JBQ3pCLHFCQUFxQjt3QkFDckIseUJBQXlCO3dCQUN6Qiw4QkFBOEI7d0JBQzlCLGNBQWM7d0JBQ2Qsa0JBQWtCO3dCQUNsQixlQUFlO3FCQUNoQjtpQkFDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDaGFubmVsQ29tcG9uZW50IH0gZnJvbSAnLi9jaGFubmVsL2NoYW5uZWwuY29tcG9uZW50JztcbmltcG9ydCB7IENoYW5uZWxIZWFkZXJDb21wb25lbnQgfSBmcm9tICcuL2NoYW5uZWwtaGVhZGVyL2NoYW5uZWwtaGVhZGVyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDaGFubmVsTGlzdENvbXBvbmVudCB9IGZyb20gJy4vY2hhbm5lbC1saXN0L2NoYW5uZWwtbGlzdC5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ2hhbm5lbFByZXZpZXdDb21wb25lbnQgfSBmcm9tICcuL2NoYW5uZWwtcHJldmlldy9jaGFubmVsLXByZXZpZXcuY29tcG9uZW50JztcbmltcG9ydCB7IE1lc3NhZ2VDb21wb25lbnQgfSBmcm9tICcuL21lc3NhZ2UvbWVzc2FnZS5jb21wb25lbnQnO1xuaW1wb3J0IHsgTWVzc2FnZUlucHV0Q29tcG9uZW50IH0gZnJvbSAnLi9tZXNzYWdlLWlucHV0L21lc3NhZ2UtaW5wdXQuY29tcG9uZW50JztcbmltcG9ydCB7IE1lc3NhZ2VMaXN0Q29tcG9uZW50IH0gZnJvbSAnLi9tZXNzYWdlLWxpc3QvbWVzc2FnZS1saXN0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgTG9hZGluZ0luZGljYXRvckNvbXBvbmVudCB9IGZyb20gJy4vbG9hZGluZy1pbmRpY2F0b3IvbG9hZGluZy1pbmRpY2F0b3IuY29tcG9uZW50JztcbmltcG9ydCB7IEljb25Db21wb25lbnQgfSBmcm9tICcuL2ljb24vaWNvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgTWVzc2FnZUFjdGlvbnNCb3hDb21wb25lbnQgfSBmcm9tICcuL21lc3NhZ2UtYWN0aW9ucy1ib3gvbWVzc2FnZS1hY3Rpb25zLWJveC5jb21wb25lbnQnO1xuaW1wb3J0IHsgQXR0YWNobWVudExpc3RDb21wb25lbnQgfSBmcm9tICcuL2F0dGFjaG1lbnQtbGlzdC9hdHRhY2htZW50LWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IE1lc3NhZ2VSZWFjdGlvbnNDb21wb25lbnQgfSBmcm9tICcuL21lc3NhZ2UtcmVhY3Rpb25zL21lc3NhZ2UtcmVhY3Rpb25zLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBOb3RpZmljYXRpb25Db21wb25lbnQgfSBmcm9tICcuL25vdGlmaWNhdGlvbi9ub3RpZmljYXRpb24uY29tcG9uZW50JztcbmltcG9ydCB7IE5vdGlmaWNhdGlvbkxpc3RDb21wb25lbnQgfSBmcm9tICcuL25vdGlmaWNhdGlvbi1saXN0L25vdGlmaWNhdGlvbi1saXN0LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBUcmFuc2xhdGVNb2R1bGUgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJztcbmltcG9ydCB7IEF0dGFjaG1lbnRQcmV2aWV3TGlzdENvbXBvbmVudCB9IGZyb20gJy4vYXR0YWNobWVudC1wcmV2aWV3LWxpc3QvYXR0YWNobWVudC1wcmV2aWV3LWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IE1vZGFsQ29tcG9uZW50IH0gZnJvbSAnLi9tb2RhbC9tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVGV4dGFyZWFEaXJlY3RpdmUgfSBmcm9tICcuL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEuZGlyZWN0aXZlJztcbmltcG9ydCB7IFN0cmVhbUF2YXRhck1vZHVsZSB9IGZyb20gJy4vc3RyZWFtLWF2YXRhci5tb2R1bGUnO1xuaW1wb3J0IHsgVGhyZWFkQ29tcG9uZW50IH0gZnJvbSAnLi90aHJlYWQvdGhyZWFkLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1xuICAgIENoYW5uZWxDb21wb25lbnQsXG4gICAgQ2hhbm5lbEhlYWRlckNvbXBvbmVudCxcbiAgICBDaGFubmVsTGlzdENvbXBvbmVudCxcbiAgICBDaGFubmVsUHJldmlld0NvbXBvbmVudCxcbiAgICBNZXNzYWdlQ29tcG9uZW50LFxuICAgIE1lc3NhZ2VJbnB1dENvbXBvbmVudCxcbiAgICBNZXNzYWdlTGlzdENvbXBvbmVudCxcbiAgICBMb2FkaW5nSW5kaWNhdG9yQ29tcG9uZW50LFxuICAgIEljb25Db21wb25lbnQsXG4gICAgTWVzc2FnZUFjdGlvbnNCb3hDb21wb25lbnQsXG4gICAgQXR0YWNobWVudExpc3RDb21wb25lbnQsXG4gICAgTWVzc2FnZVJlYWN0aW9uc0NvbXBvbmVudCxcbiAgICBOb3RpZmljYXRpb25Db21wb25lbnQsXG4gICAgTm90aWZpY2F0aW9uTGlzdENvbXBvbmVudCxcbiAgICBBdHRhY2htZW50UHJldmlld0xpc3RDb21wb25lbnQsXG4gICAgTW9kYWxDb21wb25lbnQsXG4gICAgVGV4dGFyZWFEaXJlY3RpdmUsXG4gICAgVGhyZWFkQ29tcG9uZW50LFxuICBdLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBUcmFuc2xhdGVNb2R1bGUsIFN0cmVhbUF2YXRhck1vZHVsZV0sXG4gIGV4cG9ydHM6IFtcbiAgICBDaGFubmVsQ29tcG9uZW50LFxuICAgIENoYW5uZWxIZWFkZXJDb21wb25lbnQsXG4gICAgQ2hhbm5lbExpc3RDb21wb25lbnQsXG4gICAgQ2hhbm5lbFByZXZpZXdDb21wb25lbnQsXG4gICAgTWVzc2FnZUNvbXBvbmVudCxcbiAgICBNZXNzYWdlSW5wdXRDb21wb25lbnQsXG4gICAgTWVzc2FnZUxpc3RDb21wb25lbnQsXG4gICAgTG9hZGluZ0luZGljYXRvckNvbXBvbmVudCxcbiAgICBJY29uQ29tcG9uZW50LFxuICAgIE1lc3NhZ2VBY3Rpb25zQm94Q29tcG9uZW50LFxuICAgIEF0dGFjaG1lbnRMaXN0Q29tcG9uZW50LFxuICAgIE1lc3NhZ2VSZWFjdGlvbnNDb21wb25lbnQsXG4gICAgTm90aWZpY2F0aW9uQ29tcG9uZW50LFxuICAgIE5vdGlmaWNhdGlvbkxpc3RDb21wb25lbnQsXG4gICAgQXR0YWNobWVudFByZXZpZXdMaXN0Q29tcG9uZW50LFxuICAgIE1vZGFsQ29tcG9uZW50LFxuICAgIFN0cmVhbUF2YXRhck1vZHVsZSxcbiAgICBUaHJlYWRDb21wb25lbnQsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIFN0cmVhbUNoYXRNb2R1bGUge31cbiJdfQ==