stream-chat-angular 4.32.2 → 4.34.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/version.d.ts +1 -1
- package/bundles/stream-chat-angular.umd.js +67 -9
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/custom-templates.service.js +5 -1
- package/esm2015/lib/message/message.component.js +2 -2
- package/esm2015/lib/message-list/message-list.component.js +62 -8
- package/esm2015/lib/types.js +1 -1
- package/fesm2015/stream-chat-angular.js +67 -9
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/custom-templates.service.d.ts +4 -0
- package/lib/message-list/message-list.component.d.ts +11 -2
- package/lib/types.d.ts +1 -0
- package/package.json +2 -2
- package/src/assets/version.ts +1 -1
|
@@ -47,27 +47,47 @@ export class MessageListComponent {
|
|
|
47
47
|
this.displayDateSeparator = true;
|
|
48
48
|
/**
|
|
49
49
|
* If date separators are displayed, you can set the horizontal position of the date text.
|
|
50
|
+
* If `openMessageListAt` is `last-unread-message` it will also set the text position of the new messages indicator.
|
|
50
51
|
*/
|
|
51
52
|
this.dateSeparatorTextPos = 'center';
|
|
53
|
+
/**
|
|
54
|
+
* `last-message` option will open the message list at the last message, `last-unread-message` will open the list at the last unread message. This option only works if mode is `main`.
|
|
55
|
+
*/
|
|
56
|
+
this.openMessageListAt = 'last-message';
|
|
52
57
|
this.enabledMessageActions = [];
|
|
53
58
|
this.class = 'str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner';
|
|
54
59
|
this.unreadMessageCount = 0;
|
|
55
60
|
this.groupStyles = [];
|
|
56
61
|
this.isLoading = false;
|
|
57
62
|
this.isScrollInProgress = false;
|
|
63
|
+
this.isJumpingToLatestUnreadMessage = false;
|
|
58
64
|
this.subscriptions = [];
|
|
59
65
|
this.isLatestMessageInList = true;
|
|
60
66
|
this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
|
|
61
|
-
var _a, _b;
|
|
67
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
68
|
+
(_b = (_a = this.chatClientService.chatClient) === null || _a === void 0 ? void 0 : _a.logger) === null || _b === void 0 ? void 0 : _b.call(_a, 'info', `${(channel === null || channel === void 0 ? void 0 : channel.cid) || 'undefined'} selected`, { tags: `message list ${this.mode}` });
|
|
62
69
|
if (this.channelId !== (channel === null || channel === void 0 ? void 0 : channel.id)) {
|
|
70
|
+
(_e = (_d = (_c = this.chatClientService) === null || _c === void 0 ? void 0 : _c.chatClient) === null || _d === void 0 ? void 0 : _d.logger) === null || _e === void 0 ? void 0 : _e.call(_d, 'info', `new channel is different from prev channel, reseting scroll state`, { tags: `message list ${this.mode}` });
|
|
63
71
|
this.resetScrollState();
|
|
64
72
|
this.channelId = channel === null || channel === void 0 ? void 0 : channel.id;
|
|
73
|
+
if (this.openMessageListAt === 'last-unread-message' &&
|
|
74
|
+
this.mode === 'main') {
|
|
75
|
+
this.lastReadMessageId =
|
|
76
|
+
(_g = channel === null || channel === void 0 ? void 0 : channel.state.read[((_f = this.chatClientService.chatClient.user) === null || _f === void 0 ? void 0 : _f.id) || '']) === null || _g === void 0 ? void 0 : _g.last_read_message_id;
|
|
77
|
+
if (this.lastReadMessageId) {
|
|
78
|
+
this.isJumpingToLatestUnreadMessage = true;
|
|
79
|
+
void this.channelService.jumpToMessage(this.lastReadMessageId);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
this.lastReadMessageId = undefined;
|
|
84
|
+
}
|
|
65
85
|
}
|
|
66
|
-
const capabilites = (
|
|
86
|
+
const capabilites = (_h = channel === null || channel === void 0 ? void 0 : channel.data) === null || _h === void 0 ? void 0 : _h.own_capabilities;
|
|
67
87
|
if (capabilites) {
|
|
68
88
|
this.enabledMessageActions = capabilites;
|
|
69
89
|
}
|
|
70
|
-
(
|
|
90
|
+
(_j = this.newMessageSubscription) === null || _j === void 0 ? void 0 : _j.unsubscribe();
|
|
71
91
|
if (channel) {
|
|
72
92
|
this.newMessageSubscription = channel.on('message.new', (event) => {
|
|
73
93
|
// If we display main channel messages and we're switched to an older message set -> use message.new event to update unread count and detect new messages sent by current user
|
|
@@ -95,6 +115,7 @@ export class MessageListComponent {
|
|
|
95
115
|
}));
|
|
96
116
|
this.subscriptions.push(this.customTemplatesService.messageTemplate$.subscribe((template) => (this.messageTemplate = template)));
|
|
97
117
|
this.subscriptions.push(this.customTemplatesService.dateSeparatorTemplate$.subscribe((template) => (this.customDateSeparatorTemplate = template)));
|
|
118
|
+
this.subscriptions.push(this.customTemplatesService.newMessagesIndicatorTemplate$.subscribe((template) => (this.customnewMessagesIndicatorTemplate = template)));
|
|
98
119
|
this.subscriptions.push(this.customTemplatesService.typingIndicatorTemplate$.subscribe((template) => (this.typingIndicatorTemplate = template)));
|
|
99
120
|
this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;
|
|
100
121
|
this.usersTypingInThread$ = this.channelService.usersTypingInThread$;
|
|
@@ -117,6 +138,7 @@ export class MessageListComponent {
|
|
|
117
138
|
this.subscriptions.push(this.channelService.jumpToMessage$
|
|
118
139
|
.pipe(filter((config) => !!config.id))
|
|
119
140
|
.subscribe((config) => {
|
|
141
|
+
var _a, _b;
|
|
120
142
|
let messageId = undefined;
|
|
121
143
|
if (this.mode === 'main') {
|
|
122
144
|
messageId = config.parentId || config.id;
|
|
@@ -124,6 +146,7 @@ export class MessageListComponent {
|
|
|
124
146
|
else if (config.parentId) {
|
|
125
147
|
messageId = config.id;
|
|
126
148
|
}
|
|
149
|
+
(_b = (_a = this.chatClientService.chatClient) === null || _a === void 0 ? void 0 : _a.logger) === null || _b === void 0 ? void 0 : _b.call(_a, 'info', `Jumping to ${messageId || ''}`, { tags: `message list ${this.mode}` });
|
|
127
150
|
if (messageId) {
|
|
128
151
|
if (messageId === 'latest') {
|
|
129
152
|
this.scrollToLatestMessage();
|
|
@@ -136,6 +159,7 @@ export class MessageListComponent {
|
|
|
136
159
|
}));
|
|
137
160
|
}
|
|
138
161
|
ngAfterViewChecked() {
|
|
162
|
+
var _a, _b, _c, _d, _e, _f;
|
|
139
163
|
if (this.highlightedMessageId) {
|
|
140
164
|
// Turn off programatic scroll adjustments while jump to message is in progress
|
|
141
165
|
this.hasNewMessages = false;
|
|
@@ -154,6 +178,7 @@ export class MessageListComponent {
|
|
|
154
178
|
else {
|
|
155
179
|
if (this.hasNewMessages) {
|
|
156
180
|
if (!this.isUserScrolled || this.isNewMessageSentByUser) {
|
|
181
|
+
(_b = (_a = this.chatClientService.chatClient) === null || _a === void 0 ? void 0 : _a.logger) === null || _b === void 0 ? void 0 : _b.call(_a, 'info', `User has new messages, and not scrolled or sent new messages, therefore we ${this.isLatestMessageInList ? 'scroll' : 'jump'} to latest message`, { tags: `message list ${this.mode}` });
|
|
157
182
|
this.isLatestMessageInList
|
|
158
183
|
? this.scrollToBottom()
|
|
159
184
|
: this.jumpToLatestMessage();
|
|
@@ -162,6 +187,7 @@ export class MessageListComponent {
|
|
|
162
187
|
this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;
|
|
163
188
|
}
|
|
164
189
|
else if (this.olderMassagesLoaded) {
|
|
190
|
+
(_d = (_c = this.chatClientService.chatClient) === null || _c === void 0 ? void 0 : _c.logger) === null || _d === void 0 ? void 0 : _d.call(_c, 'info', `Older messages are loaded, we preserve the scroll position`, { tags: `message list ${this.mode}` });
|
|
165
191
|
this.preserveScrollbarPosition();
|
|
166
192
|
this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;
|
|
167
193
|
this.olderMassagesLoaded = false;
|
|
@@ -169,6 +195,7 @@ export class MessageListComponent {
|
|
|
169
195
|
else if (this.getScrollPosition() !== 'bottom' &&
|
|
170
196
|
!this.isUserScrolled &&
|
|
171
197
|
!this.highlightedMessageId) {
|
|
198
|
+
(_f = (_e = this.chatClientService.chatClient) === null || _e === void 0 ? void 0 : _e.logger) === null || _f === void 0 ? void 0 : _f.call(_e, 'info', `Container grew and user didn't scroll therefore we ${this.isLatestMessageInList ? 'scroll' : 'jump'} to latest message`, { tags: `message list ${this.mode}` });
|
|
172
199
|
this.isLatestMessageInList
|
|
173
200
|
? this.scrollToBottom()
|
|
174
201
|
: this.jumpToLatestMessage();
|
|
@@ -202,6 +229,7 @@ export class MessageListComponent {
|
|
|
202
229
|
this.scrollContainer.nativeElement.scrollTop = 0;
|
|
203
230
|
}
|
|
204
231
|
scrolled() {
|
|
232
|
+
var _a, _b;
|
|
205
233
|
this.isScrollInProgress = true;
|
|
206
234
|
if (this.scrollEndTimeout) {
|
|
207
235
|
clearTimeout(this.scrollEndTimeout);
|
|
@@ -214,6 +242,7 @@ export class MessageListComponent {
|
|
|
214
242
|
return;
|
|
215
243
|
}
|
|
216
244
|
const scrollPosition = this.getScrollPosition();
|
|
245
|
+
(_b = (_a = this.chatClientService.chatClient) === null || _a === void 0 ? void 0 : _a.logger) === null || _b === void 0 ? void 0 : _b.call(_a, 'info', `Scrolled - scroll position: ${scrollPosition}`, { tags: `message list ${this.mode}` });
|
|
217
246
|
this.isUserScrolled =
|
|
218
247
|
(this.direction === 'bottom-to-top'
|
|
219
248
|
? scrollPosition !== 'bottom'
|
|
@@ -247,7 +276,7 @@ export class MessageListComponent {
|
|
|
247
276
|
return text;
|
|
248
277
|
}
|
|
249
278
|
areOnSeparateDates(message, nextMessage) {
|
|
250
|
-
if (!nextMessage) {
|
|
279
|
+
if (!message || !nextMessage) {
|
|
251
280
|
return false;
|
|
252
281
|
}
|
|
253
282
|
if (message.created_at.getDate() !== nextMessage.created_at.getDate()) {
|
|
@@ -260,6 +289,13 @@ export class MessageListComponent {
|
|
|
260
289
|
}
|
|
261
290
|
return false;
|
|
262
291
|
}
|
|
292
|
+
isSentByCurrentUser(message) {
|
|
293
|
+
var _a, _b;
|
|
294
|
+
if (!message) {
|
|
295
|
+
return false;
|
|
296
|
+
}
|
|
297
|
+
return ((_a = message.user) === null || _a === void 0 ? void 0 : _a.id) === ((_b = this.chatClientService.chatClient.user) === null || _b === void 0 ? void 0 : _b.id);
|
|
298
|
+
}
|
|
263
299
|
parseDate(date) {
|
|
264
300
|
return this.dateParser.parseDate(date);
|
|
265
301
|
}
|
|
@@ -296,11 +332,18 @@ export class MessageListComponent {
|
|
|
296
332
|
this.messages$ = (this.mode === 'main'
|
|
297
333
|
? this.channelService.activeChannelMessages$
|
|
298
334
|
: this.channelService.activeThreadMessages$).pipe(tap((messages) => {
|
|
335
|
+
var _a, _b, _c, _d;
|
|
299
336
|
this.isLoading = false;
|
|
300
337
|
if (messages.length === 0) {
|
|
338
|
+
(_b = (_a = this.chatClientService.chatClient) === null || _a === void 0 ? void 0 : _a.logger) === null || _b === void 0 ? void 0 : _b.call(_a, 'info', `Empty messages array, reseting scroll state`, {
|
|
339
|
+
tags: `message list ${this.mode}`,
|
|
340
|
+
});
|
|
301
341
|
this.resetScrollState();
|
|
302
342
|
return;
|
|
303
343
|
}
|
|
344
|
+
(_d = (_c = this.chatClientService.chatClient) === null || _c === void 0 ? void 0 : _c.logger) === null || _d === void 0 ? void 0 : _d.call(_c, 'info', `Received one or more messages`, {
|
|
345
|
+
tags: `message list ${this.mode}`,
|
|
346
|
+
});
|
|
304
347
|
const currentLatestMessage = messages[messages.length - 1];
|
|
305
348
|
this.newMessageReceived(currentLatestMessage);
|
|
306
349
|
const currentOldestMessage = messages[0];
|
|
@@ -330,6 +373,13 @@ export class MessageListComponent {
|
|
|
330
373
|
if (!this.isLatestMessageInList) {
|
|
331
374
|
this.isUserScrolled = true;
|
|
332
375
|
}
|
|
376
|
+
}), tap(() => {
|
|
377
|
+
var _a;
|
|
378
|
+
if (this.isJumpingToLatestUnreadMessage &&
|
|
379
|
+
this.lastReadMessageId &&
|
|
380
|
+
this.lastReadMessageId === ((_a = this.latestMessage) === null || _a === void 0 ? void 0 : _a.id)) {
|
|
381
|
+
this.lastReadMessageId = undefined;
|
|
382
|
+
}
|
|
333
383
|
}), map((messages) => this.direction === 'bottom-to-top' ? messages : [...messages].reverse()), tap((messages) => {
|
|
334
384
|
this.groupStyles = messages.map((m, i) => getGroupStyles(m, messages[i - 1], messages[i + 1]));
|
|
335
385
|
}));
|
|
@@ -361,6 +411,7 @@ export class MessageListComponent {
|
|
|
361
411
|
element.scrollIntoView({ block: 'center' });
|
|
362
412
|
setTimeout(() => {
|
|
363
413
|
this.highlightedMessageId = undefined;
|
|
414
|
+
this.isJumpingToLatestUnreadMessage = false;
|
|
364
415
|
}, 1000);
|
|
365
416
|
}
|
|
366
417
|
}
|
|
@@ -376,13 +427,14 @@ export class MessageListComponent {
|
|
|
376
427
|
}
|
|
377
428
|
}
|
|
378
429
|
newMessageReceived(message) {
|
|
379
|
-
var _a, _b, _c, _d;
|
|
430
|
+
var _a, _b, _c, _d, _e, _f;
|
|
380
431
|
if (!this.latestMessage ||
|
|
381
432
|
((_a = this.latestMessage.created_at) === null || _a === void 0 ? void 0 : _a.getTime()) < message.created_at.getTime()) {
|
|
433
|
+
(_c = (_b = this.chatClientService.chatClient) === null || _b === void 0 ? void 0 : _b.logger) === null || _c === void 0 ? void 0 : _c.call(_b, 'info', `Received new message`, { tags: `message list ${this.mode}` });
|
|
382
434
|
this.latestMessage = message;
|
|
383
435
|
this.hasNewMessages = true;
|
|
384
436
|
this.isNewMessageSentByUser =
|
|
385
|
-
((
|
|
437
|
+
((_d = message.user) === null || _d === void 0 ? void 0 : _d.id) === ((_f = (_e = this.chatClientService.chatClient) === null || _e === void 0 ? void 0 : _e.user) === null || _f === void 0 ? void 0 : _f.id);
|
|
386
438
|
if (this.isUserScrolled) {
|
|
387
439
|
this.unreadMessageCount++;
|
|
388
440
|
}
|
|
@@ -390,7 +442,7 @@ export class MessageListComponent {
|
|
|
390
442
|
}
|
|
391
443
|
}
|
|
392
444
|
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.CustomTemplatesService }, { token: i4.DateParserService }], target: i0.ɵɵFactoryTarget.Component });
|
|
393
|
-
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { mode: "mode", direction: "direction", messageOptionsTrigger: "messageOptionsTrigger", hideJumpToLatestButtonDuringScroll: "hideJumpToLatestButtonDuringScroll", customMessageActions: "customMessageActions", displayDateSeparator: "displayDateSeparator", dateSeparatorTextPos: "dateSeparatorTextPos" }, 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 str-chat__message-list-scroll\">\n <ul\n class=\"str-chat__ul\"\n [class.str-chat__message-options-in-bubble]=\"\n messageOptionsTrigger === 'message-bubble'\n \"\n >\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread' && parentMessage\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage, index: 'parent' }\n \"\n ></ng-container>\n <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n {{parentMessage?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </div>\n </li>\n <stream-loading-indicator\n data-testid=\"top-loading-indicator\"\n *ngIf=\"isLoading && direction === 'bottom-to-top'\"\n ></stream-loading-indicator>\n <ng-container *ngIf=\"messages$ | async as messages\">\n <ng-container\n *ngFor=\"\n let message of messages;\n let i = index;\n let isFirst = first;\n trackBy: trackByMessageId\n \"\n >\n <ng-container *ngIf=\"isFirst\">\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: message.created_at,\n parsedDate: parseDate(message.created_at)\n }\n \"\n ></ng-container>\n </ng-container>\n <li\n tabindex=\"0\"\n data-testclass=\"message\"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message, index: i }\n \"\n ></ng-container>\n </li>\n <ng-container *ngIf=\"areOnSeparateDates(message, messages[i + 1])\">\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: messages[i + 1].created_at,\n parsedDate: parseDate(messages[i + 1].created_at)\n }\n \"\n ></ng-container>\n </ng-container>\n </ng-container>\n </ng-container>\n <stream-loading-indicator\n data-testid=\"bottom-loading-indicator\"\n *ngIf=\"isLoading && direction === 'top-to-bottom'\"\n ></stream-loading-indicator>\n </ul>\n <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n <ng-container *ngIf=\"$any(usersTyping$ | async) as users\">\n <div\n *ngIf=\"users.length > 0\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n <div\n data-testid=\"typing-users\"\n class=\"str-chat__typing-indicator__users\"\n >\n {{\n users.length === 1\n ? (\"streamChat.user is typing\"\n | translate: { user: getTypingIndicatorText(users) })\n : (\"streamChat.users are typing\"\n | translate: { users: getTypingIndicatorText(users) })\n }}\n </div>\n </div>\n </ng-container>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate || defaultTypingIndicator;\n context: getTypingIndicatorContext()\n \"\n ></ng-container>\n </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n <button\n data-testid=\"scroll-to-latest\"\n *ngIf=\"\n isUserScrolled &&\n (!isScrollInProgress || !hideJumpToLatestButtonDuringScroll)\n \"\n class=\"\n str-chat__message-notification-scroll-to-latest\n str-chat__message-notification-scroll-to-latest-right\n str-chat__circle-fab\n \"\n (keyup.enter)=\"jumpToLatestMessage()\"\n (click)=\"jumpToLatestMessage()\"\n >\n <stream-icon\n class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n ></stream-icon>\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-to-latest-unread-count\n str-chat__jump-to-latest-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n <ng-template\n #defaultMessageTemplate\n let-messageInput=\"message\"\n let-isLastSentMessage=\"isLastSentMessage\"\n let-enabledMessageActions=\"enabledMessageActions\"\n let-mode=\"mode\"\n let-isHighlighted=\"isHighlighted\"\n let-customActions=\"customActions\"\n >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n [customActions]=\"customActions\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: {\n message: message,\n isLastSentMessage: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n enabledMessageActions: enabledMessageActions,\n mode: mode,\n isHighlighted: message?.id === highlightedMessageId,\n customActions: customMessageActions\n }\n \"\n ></ng-container>\n</ng-template>\n\n<ng-template #dateSeparator let-date=\"date\" let-parsedDate=\"parsedDate\">\n <ng-container *ngIf=\"displayDateSeparator\">\n <ng-container\n *ngTemplateOutlet=\"\n customDateSeparatorTemplate || defaultDateSeparator;\n context: { date: date, parsedDate: parsedDate }\n \"\n ></ng-container>\n </ng-container>\n\n <ng-template\n #defaultDateSeparator\n let-date=\"date\"\n let-parsedDate=\"parsedDate\"\n >\n <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n <div class=\"str-chat__date-separator-date\">\n {{ parsedDate }}\n </div>\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n </div>\n </ng-template>\n</ng-template>\n", components: [{ type: i5.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: i6.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i7.MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted", "customActions"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i9.TranslatePipe, "async": i8.AsyncPipe } });
|
|
445
|
+
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { mode: "mode", direction: "direction", messageOptionsTrigger: "messageOptionsTrigger", hideJumpToLatestButtonDuringScroll: "hideJumpToLatestButtonDuringScroll", customMessageActions: "customMessageActions", displayDateSeparator: "displayDateSeparator", dateSeparatorTextPos: "dateSeparatorTextPos", openMessageListAt: "openMessageListAt" }, 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 str-chat__message-list-scroll\">\n <ul\n class=\"str-chat__ul\"\n [class.str-chat__message-options-in-bubble]=\"\n messageOptionsTrigger === 'message-bubble'\n \"\n >\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread' && parentMessage\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage, index: 'parent' }\n \"\n ></ng-container>\n <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n {{parentMessage?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </div>\n </li>\n <stream-loading-indicator\n data-testid=\"top-loading-indicator\"\n *ngIf=\"isLoading && direction === 'bottom-to-top'\"\n ></stream-loading-indicator>\n <ng-container *ngIf=\"messages$ | async as messages\">\n <ng-container\n *ngFor=\"\n let message of messages;\n let i = index;\n let isFirst = first;\n let isLast = last;\n trackBy: trackByMessageId\n \"\n >\n <ng-container *ngIf=\"isFirst\">\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: message.created_at,\n parsedDate: parseDate(message.created_at),\n isNewMessage: false\n }\n \"\n ></ng-container>\n </ng-container>\n <li\n tabindex=\"0\"\n data-testclass=\"message\"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message, index: i }\n \"\n ></ng-container>\n </li>\n <ng-container\n *ngIf=\"\n (lastReadMessageId === message.id &&\n !isLast &&\n direction === 'bottom-to-top' &&\n !isSentByCurrentUser(messages[i + 1]) &&\n (!displayDateSeparator ||\n !areOnSeparateDates(message, messages[i + 1]))) ||\n (direction === 'top-to-bottom' &&\n !isLast &&\n !isSentByCurrentUser(message) &&\n lastReadMessageId === messages[i + 1].id)\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n customnewMessagesIndicatorTemplate ||\n defaultNewMessagesIndicator\n \"\n ></ng-container>\n </ng-container>\n <ng-container *ngIf=\"areOnSeparateDates(message, messages[i + 1])\">\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: messages[i + 1].created_at,\n parsedDate: parseDate(messages[i + 1].created_at),\n isNewMessage:\n (direction === 'bottom-to-top' &&\n message.id === lastReadMessageId &&\n !isSentByCurrentUser(messages[i + 1])) ||\n (direction === 'top-to-bottom' && false)\n }\n \"\n ></ng-container>\n </ng-container>\n </ng-container>\n </ng-container>\n <stream-loading-indicator\n data-testid=\"bottom-loading-indicator\"\n *ngIf=\"isLoading && direction === 'top-to-bottom'\"\n ></stream-loading-indicator>\n </ul>\n <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n <ng-container *ngIf=\"$any(usersTyping$ | async) as users\">\n <div\n *ngIf=\"users.length > 0\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n <div\n data-testid=\"typing-users\"\n class=\"str-chat__typing-indicator__users\"\n >\n {{\n users.length === 1\n ? (\"streamChat.user is typing\"\n | translate: { user: getTypingIndicatorText(users) })\n : (\"streamChat.users are typing\"\n | translate: { users: getTypingIndicatorText(users) })\n }}\n </div>\n </div>\n </ng-container>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate || defaultTypingIndicator;\n context: getTypingIndicatorContext()\n \"\n ></ng-container>\n </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n <button\n data-testid=\"scroll-to-latest\"\n *ngIf=\"\n isUserScrolled &&\n (!isScrollInProgress || !hideJumpToLatestButtonDuringScroll)\n \"\n class=\"\n str-chat__message-notification-scroll-to-latest\n str-chat__message-notification-scroll-to-latest-right\n str-chat__circle-fab\n \"\n (keyup.enter)=\"jumpToLatestMessage()\"\n (click)=\"jumpToLatestMessage()\"\n >\n <stream-icon\n class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n ></stream-icon>\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-to-latest-unread-count\n str-chat__jump-to-latest-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n <ng-template\n #defaultMessageTemplate\n let-messageInput=\"message\"\n let-isLastSentMessage=\"isLastSentMessage\"\n let-enabledMessageActions=\"enabledMessageActions\"\n let-mode=\"mode\"\n let-isHighlighted=\"isHighlighted\"\n let-customActions=\"customActions\"\n >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n [customActions]=\"customActions\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: {\n message: message,\n isLastSentMessage: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n enabledMessageActions: enabledMessageActions,\n mode: mode,\n isHighlighted:\n message?.id === highlightedMessageId &&\n !isJumpingToLatestUnreadMessage,\n customActions: customMessageActions\n }\n \"\n ></ng-container>\n</ng-template>\n\n<ng-template\n #dateSeparator\n let-date=\"date\"\n let-parsedDate=\"parsedDate\"\n let-isNewMessage=\"isNewMessage\"\n>\n <ng-container *ngIf=\"displayDateSeparator\">\n <ng-container\n *ngTemplateOutlet=\"\n customDateSeparatorTemplate || defaultDateSeparator;\n context: {\n date: date,\n parsedDate: parsedDate,\n isNewMessage: isNewMessage\n }\n \"\n ></ng-container>\n </ng-container>\n\n <ng-template\n #defaultDateSeparator\n let-date=\"date\"\n let-parsedDate=\"parsedDate\"\n let-isNewMessage=\"isNewMessage\"\n >\n <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n <div class=\"str-chat__date-separator-date\">\n {{ parsedDate }}\n <span\n *ngIf=\"isNewMessage\"\n data-testid=\"new-messages-indicator-date-separator\"\n >\u2022 {{ \"streamChat.New\" | translate }}</span\n >\n </div>\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n </div>\n </ng-template>\n</ng-template>\n\n<ng-template #defaultNewMessagesIndicator>\n <div data-testid=\"new-messages-indicator\" class=\"str-chat__date-separator\">\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n <div class=\"str-chat__date-separator-date\" translate>streamChat.New</div>\n <hr\n *ngIf=\"\n dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n \"\n class=\"str-chat__date-separator-line\"\n />\n </div>\n</ng-template>\n", components: [{ type: i5.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: i6.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i7.MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted", "customActions"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i9.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], pipes: { "translate": i9.TranslatePipe, "async": i8.AsyncPipe } });
|
|
394
446
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
|
|
395
447
|
type: Component,
|
|
396
448
|
args: [{
|
|
@@ -412,6 +464,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
412
464
|
type: Input
|
|
413
465
|
}], dateSeparatorTextPos: [{
|
|
414
466
|
type: Input
|
|
467
|
+
}], openMessageListAt: [{
|
|
468
|
+
type: Input
|
|
415
469
|
}], class: [{
|
|
416
470
|
type: HostBinding,
|
|
417
471
|
args: ['class']
|
|
@@ -422,4 +476,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
422
476
|
type: ViewChild,
|
|
423
477
|
args: ['parentMessageElement']
|
|
424
478
|
}] } });
|
|
425
|
-
//# 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  AfterViewInit,\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 { filter, map, tap } from 'rxjs/operators';\nimport {\n  MessageContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n  TypingIndicatorContext,\n  CustomMessageActionItem,\n  DateSeparatorContext,\n} from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { UserResponse } from 'stream-chat';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\nimport { DateParserService } from '../date-parser.service';\n\n/**\n * The `MessageList` component renders a scrollable list of messages.\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, AfterViewInit\n{\n  /**\n   * Determines if the message list should display channel messages or [thread messages](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'main' | 'thread' = 'main';\n  /**\n   * The direction of the messages in the list, `bottom-to-top` means newest message is at the bottom of the message list and users scroll upwards to load older messages\n   */\n  @Input() direction: 'bottom-to-top' | 'top-to-bottom' = 'bottom-to-top';\n  /**\n   * Determines what triggers the appearance of the message options: by default you can hover (click on mobile) anywhere in the row of the message (`message-row` option), or you can set `message-bubble`, in that case only a hover (click on mobile) in the message bubble will trigger the options to appear.\n   */\n  @Input() messageOptionsTrigger: 'message-row' | 'message-bubble' =\n    'message-row';\n  typingIndicatorTemplate: TemplateRef<TypingIndicatorContext> | undefined;\n  /**\n   * You can hide the \"jump to latest\" button while scrolling. A potential use-case for this input would be to [workaround a known issue on iOS Safar](https://github.com/GetStream/stream-chat-angular/issues/418)\n   */\n  @Input() hideJumpToLatestButtonDuringScroll = false;\n  /**\n   * A list of custom message actions to be displayed in the message action box\n   */\n  @Input() customMessageActions: CustomMessageActionItem<any>[] = [];\n  /**\n   * If `true` date separators will be displayed\n   */\n  @Input() displayDateSeparator = true;\n  /**\n   * If date separators are displayed, you can set the horizontal position of the date text.\n   */\n  @Input() dateSeparatorTextPos: 'center' | 'right' | 'left' = 'center';\n  messageTemplate: TemplateRef<MessageContext> | undefined;\n  customDateSeparatorTemplate: TemplateRef<DateSeparatorContext> | undefined;\n  messages$!: Observable<StreamMessage[]>;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner';\n  unreadMessageCount = 0;\n  isUserScrolled: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  parentMessage: StreamMessage | undefined;\n  highlightedMessageId: string | undefined;\n  isLoading = false;\n  isScrollInProgress = false;\n  scrollEndTimeout: any;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  @ViewChild('parentMessageElement')\n  private parentMessageElement!: ElementRef<HTMLElement>;\n  private latestMessage: { id: string; created_at: Date } | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessage: { id: string; created_at: Date } | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private subscriptions: Subscription[] = [];\n  private newMessageSubscription: { unsubscribe: () => void } | undefined;\n  private prevScrollTop: number | undefined;\n  private usersTypingInChannel$!: Observable<\n    UserResponse<DefaultStreamChatGenerics>[]\n  >;\n  private usersTypingInThread$!: Observable<\n    UserResponse<DefaultStreamChatGenerics>[]\n  >;\n  private isLatestMessageInList = true;\n  private channelId?: string;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private customTemplatesService: CustomTemplatesService,\n    private dateParser: DateParserService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        if (this.channelId !== channel?.id) {\n          this.resetScrollState();\n          this.channelId = channel?.id;\n        }\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.enabledMessageActions = capabilites;\n        }\n        this.newMessageSubscription?.unsubscribe();\n        if (channel) {\n          this.newMessageSubscription = channel.on('message.new', (event) => {\n            // If we display main channel messages and we're switched to an older message set -> use message.new event to update unread count and detect new messages sent by current user\n            if (\n              !event.message ||\n              channel.state.messages === channel.state.latestMessages ||\n              this.mode === 'thread'\n            ) {\n              return;\n            }\n            this.newMessageReceived({\n              id: event.message.id,\n              user: event.message.user,\n              created_at: new Date(event.message.created_at || ''),\n            });\n          });\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    this.subscriptions.push(\n      this.customTemplatesService.messageTemplate$.subscribe(\n        (template) => (this.messageTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.dateSeparatorTemplate$.subscribe(\n        (template) => (this.customDateSeparatorTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.typingIndicatorTemplate$.subscribe(\n        (template) => (this.typingIndicatorTemplate = template)\n      )\n    );\n    this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;\n    this.usersTypingInThread$ = this.channelService.usersTypingInThread$;\n  }\n\n  ngOnInit(): void {\n    this.setMessages$();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.mode || changes.direction) {\n      this.setMessages$();\n    }\n    if (changes.direction) {\n      if (this.scrollContainer?.nativeElement) {\n        this.jumpToLatestMessage();\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.subscriptions.push(\n      this.channelService.jumpToMessage$\n        .pipe(filter((config) => !!config.id))\n        .subscribe((config) => {\n          let messageId: string | undefined = undefined;\n          if (this.mode === 'main') {\n            messageId = config.parentId || config.id;\n          } else if (config.parentId) {\n            messageId = config.id;\n          }\n          if (messageId) {\n            if (messageId === 'latest') {\n              this.scrollToLatestMessage();\n            } else {\n              this.scrollMessageIntoView(messageId);\n              this.highlightedMessageId = messageId;\n            }\n          }\n        })\n    );\n  }\n\n  ngAfterViewChecked() {\n    if (this.highlightedMessageId) {\n      // Turn off programatic scroll adjustments while jump to message is in progress\n      this.hasNewMessages = false;\n      this.olderMassagesLoaded = false;\n    }\n    if (this.direction === 'top-to-bottom') {\n      if (\n        this.hasNewMessages &&\n        (this.isNewMessageSentByUser || !this.isUserScrolled)\n      ) {\n        this.isLatestMessageInList\n          ? this.scrollToTop()\n          : this.jumpToLatestMessage();\n        this.hasNewMessages = false;\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      }\n    } else {\n      if (this.hasNewMessages) {\n        if (!this.isUserScrolled || this.isNewMessageSentByUser) {\n          this.isLatestMessageInList\n            ? this.scrollToBottom()\n            : this.jumpToLatestMessage();\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.getScrollPosition() !== 'bottom' &&\n        !this.isUserScrolled &&\n        !this.highlightedMessageId\n      ) {\n        this.isLatestMessageInList\n          ? this.scrollToBottom()\n          : this.jumpToLatestMessage();\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n    this.newMessageSubscription?.unsubscribe();\n    if (this.scrollEndTimeout) {\n      clearTimeout(this.scrollEndTimeout);\n    }\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  trackByUserId(index: number, user: UserResponse) {\n    return user.id;\n  }\n\n  jumpToLatestMessage() {\n    void this.channelService.jumpToMessage(\n      'latest',\n      this.mode === 'thread' ? this.parentMessage?.id : undefined\n    );\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrollToTop() {\n    this.scrollContainer.nativeElement.scrollTop = 0;\n  }\n\n  scrolled() {\n    this.isScrollInProgress = true;\n    if (this.scrollEndTimeout) {\n      clearTimeout(this.scrollEndTimeout);\n    }\n    this.scrollEndTimeout = setTimeout(() => {\n      this.isScrollInProgress = false;\n    }, 100);\n    if (\n      this.scrollContainer.nativeElement.scrollHeight ===\n      this.scrollContainer.nativeElement.clientHeight\n    ) {\n      return;\n    }\n    const scrollPosition = this.getScrollPosition();\n\n    this.isUserScrolled =\n      (this.direction === 'bottom-to-top'\n        ? scrollPosition !== 'bottom'\n        : scrollPosition !== 'top') || !this.isLatestMessageInList;\n    if (!this.isUserScrolled) {\n      this.unreadMessageCount = 0;\n    }\n    if (this.shouldLoadMoreMessages(scrollPosition)) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      let direction: 'newer' | 'older';\n      if (this.direction === 'top-to-bottom') {\n        direction = scrollPosition === 'top' ? 'newer' : 'older';\n      } else {\n        direction = scrollPosition === 'top' ? 'older' : 'newer';\n      }\n      this.mode === 'main'\n        ? void this.channelService.loadMoreMessages(direction)\n        : void this.channelService.loadMoreThreadReplies(direction);\n      this.isLoading = true;\n    }\n    this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;\n  }\n\n  getTypingIndicatorContext(): TypingIndicatorContext {\n    return {\n      usersTyping$: this.usersTyping$,\n    };\n  }\n\n  getTypingIndicatorText(users: UserResponse[]) {\n    const text = listUsers(users);\n\n    return text;\n  }\n\n  areOnSeparateDates(message: StreamMessage, nextMessage?: StreamMessage) {\n    if (!nextMessage) {\n      return false;\n    }\n    if (message.created_at.getDate() !== nextMessage.created_at.getDate()) {\n      return true;\n    } else if (\n      message.created_at.getFullYear() !==\n        nextMessage.created_at.getFullYear() ||\n      message.created_at.getMonth() !== nextMessage.created_at.getMonth()\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  parseDate(date: Date) {\n    return this.dateParser.parseDate(date);\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.parentMessage?.reply_count };\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 getScrollPosition(): 'top' | 'bottom' | 'middle' {\n    let position: 'top' | 'bottom' | 'middle' = 'middle';\n    if (\n      Math.floor(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      position = 'top';\n    } else if (\n      Math.ceil(this.scrollContainer.nativeElement.scrollTop) +\n        this.scrollContainer.nativeElement.clientHeight >=\n      this.scrollContainer.nativeElement.scrollHeight\n    ) {\n      position = 'bottom';\n    }\n\n    return position;\n  }\n\n  private shouldLoadMoreMessages(scrollPosition: 'top' | 'bottom' | 'middle') {\n    return scrollPosition !== 'middle' && !this.highlightedMessageId;\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        this.isLoading = false;\n        if (messages.length === 0) {\n          this.resetScrollState();\n          return;\n        }\n        const currentLatestMessage = messages[messages.length - 1];\n        this.newMessageReceived(currentLatestMessage);\n        const currentOldestMessage = messages[0];\n        if (\n          !this.oldestMessage ||\n          !messages.find((m) => m.id === this.oldestMessage!.id)\n        ) {\n          this.oldestMessage = currentOldestMessage;\n        } else if (\n          this.oldestMessage.created_at.getTime() >\n          currentOldestMessage.created_at.getTime()\n        ) {\n          this.oldestMessage = currentOldestMessage;\n          this.olderMassagesLoaded = true;\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      tap((messages) => {\n        this.isLatestMessageInList =\n          !this.latestMessage ||\n          messages.length === 0 ||\n          messages[messages.length - 1].id === this.latestMessage.id;\n        if (!this.isLatestMessageInList) {\n          this.isUserScrolled = true;\n        }\n      }),\n      map((messages) =>\n        this.direction === 'bottom-to-top' ? messages : [...messages].reverse()\n      ),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      })\n    );\n  }\n\n  private resetScrollState() {\n    this.latestMessage = undefined;\n    this.hasNewMessages = true;\n    this.isUserScrolled = false;\n    this.containerHeight = undefined;\n    this.olderMassagesLoaded = false;\n    this.oldestMessage = undefined;\n    this.unreadMessageCount = 0;\n    this.prevScrollTop = undefined;\n    this.isNewMessageSentByUser = undefined;\n    this.isLatestMessageInList = true;\n  }\n\n  private get usersTyping$() {\n    return this.mode === 'thread'\n      ? this.usersTypingInThread$\n      : this.usersTypingInChannel$;\n  }\n\n  private scrollMessageIntoView(messageId: string, withRetry = true) {\n    const element = document.getElementById(messageId);\n    if (!element && withRetry) {\n      // If the message was newly inserted into activeChannelMessages$, the message will be rendered after the current change detection cycle -> wait for this cycle to complete\n      setTimeout(() => this.scrollMessageIntoView(messageId, false));\n    } else if (element) {\n      element.scrollIntoView({ block: 'center' });\n      setTimeout(() => {\n        this.highlightedMessageId = undefined;\n      }, 1000);\n    }\n  }\n\n  private scrollToLatestMessage(withRetry = true) {\n    if (document.getElementById(this.latestMessage!.id)) {\n      this.direction === 'bottom-to-top'\n        ? this.scrollToBottom()\n        : this.scrollToTop();\n    } else if (withRetry) {\n      // If the message was newly inserted into activeChannelMessages$, the message will be rendered after the current change detection cycle -> wait for this cycle to complete\n      setTimeout(() => this.scrollToLatestMessage(false), 0);\n    }\n  }\n\n  private newMessageReceived(message: {\n    id: string;\n    created_at: Date;\n    user?: { id: string } | null;\n  }) {\n    if (\n      !this.latestMessage ||\n      this.latestMessage.created_at?.getTime() < message.created_at.getTime()\n    ) {\n      this.latestMessage = message;\n      this.hasNewMessages = true;\n      this.isNewMessageSentByUser =\n        message.user?.id === this.chatClientService.chatClient?.user?.id;\n      if (this.isUserScrolled) {\n        this.unreadMessageCount++;\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 str-chat__message-list-scroll\">\n    <ul\n      class=\"str-chat__ul\"\n      [class.str-chat__message-options-in-bubble]=\"\n        messageOptionsTrigger === 'message-bubble'\n      \"\n    >\n      <li\n        #parentMessageElement\n        *ngIf=\"mode === 'thread' && parentMessage\"\n        data-testid=\"parent-message\"\n        class=\"str-chat__parent-message-li\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage, index: 'parent' }\n          \"\n        ></ng-container>\n        <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n          {{parentMessage?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n          replies' | translate:replyCountParam)}}\n        </div>\n      </li>\n      <stream-loading-indicator\n        data-testid=\"top-loading-indicator\"\n        *ngIf=\"isLoading && direction === 'bottom-to-top'\"\n      ></stream-loading-indicator>\n      <ng-container *ngIf=\"messages$ | async as messages\">\n        <ng-container\n          *ngFor=\"\n            let message of messages;\n            let i = index;\n            let isFirst = first;\n            trackBy: trackByMessageId\n          \"\n        >\n          <ng-container *ngIf=\"isFirst\">\n            <ng-container\n              *ngTemplateOutlet=\"\n                dateSeparator;\n                context: {\n                  date: message.created_at,\n                  parsedDate: parseDate(message.created_at)\n                }\n              \"\n            ></ng-container>\n          </ng-container>\n          <li\n            tabindex=\"0\"\n            data-testclass=\"message\"\n            class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n            id=\"{{ message.id }}\"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"\n                messageTemplateContainer;\n                context: { message: message, index: i }\n              \"\n            ></ng-container>\n          </li>\n          <ng-container *ngIf=\"areOnSeparateDates(message, messages[i + 1])\">\n            <ng-container\n              *ngTemplateOutlet=\"\n                dateSeparator;\n                context: {\n                  date: messages[i + 1].created_at,\n                  parsedDate: parseDate(messages[i + 1].created_at)\n                }\n              \"\n            ></ng-container>\n          </ng-container>\n        </ng-container>\n      </ng-container>\n      <stream-loading-indicator\n        data-testid=\"bottom-loading-indicator\"\n        *ngIf=\"isLoading && direction === 'top-to-bottom'\"\n      ></stream-loading-indicator>\n    </ul>\n    <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n      <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n      <ng-container *ngIf=\"$any(usersTyping$ | async) as users\">\n        <div\n          *ngIf=\"users.length > 0\"\n          data-testid=\"typing-indicator\"\n          class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n        >\n          <div class=\"str-chat__typing-indicator__dots\">\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n          </div>\n          <div\n            data-testid=\"typing-users\"\n            class=\"str-chat__typing-indicator__users\"\n          >\n            {{\n              users.length === 1\n                ? (\"streamChat.user is typing\"\n                  | translate: { user: getTypingIndicatorText(users) })\n                : (\"streamChat.users are typing\"\n                  | translate: { users: getTypingIndicatorText(users) })\n            }}\n          </div>\n        </div>\n      </ng-container>\n    </ng-template>\n    <ng-container\n      *ngTemplateOutlet=\"\n        typingIndicatorTemplate || defaultTypingIndicator;\n        context: getTypingIndicatorContext()\n      \"\n    ></ng-container>\n  </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n  <button\n    data-testid=\"scroll-to-latest\"\n    *ngIf=\"\n      isUserScrolled &&\n      (!isScrollInProgress || !hideJumpToLatestButtonDuringScroll)\n    \"\n    class=\"\n      str-chat__message-notification-scroll-to-latest\n      str-chat__message-notification-scroll-to-latest-right\n      str-chat__circle-fab\n    \"\n    (keyup.enter)=\"jumpToLatestMessage()\"\n    (click)=\"jumpToLatestMessage()\"\n  >\n    <stream-icon\n      class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n      [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n    ></stream-icon>\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-to-latest-unread-count\n        str-chat__jump-to-latest-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n  <ng-template\n    #defaultMessageTemplate\n    let-messageInput=\"message\"\n    let-isLastSentMessage=\"isLastSentMessage\"\n    let-enabledMessageActions=\"enabledMessageActions\"\n    let-mode=\"mode\"\n    let-isHighlighted=\"isHighlighted\"\n    let-customActions=\"customActions\"\n  >\n    <stream-message\n      [message]=\"messageInput\"\n      [isLastSentMessage]=\"isLastSentMessage\"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [mode]=\"mode\"\n      [isHighlighted]=\"isHighlighted\"\n      [customActions]=\"customActions\"\n    ></stream-message>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      messageTemplate || defaultMessageTemplate;\n      context: {\n        message: message,\n        isLastSentMessage: !!(\n          lastSentMessageId && message?.id === lastSentMessageId\n        ),\n        enabledMessageActions: enabledMessageActions,\n        mode: mode,\n        isHighlighted: message?.id === highlightedMessageId,\n        customActions: customMessageActions\n      }\n    \"\n  ></ng-container>\n</ng-template>\n\n<ng-template #dateSeparator let-date=\"date\" let-parsedDate=\"parsedDate\">\n  <ng-container *ngIf=\"displayDateSeparator\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        customDateSeparatorTemplate || defaultDateSeparator;\n        context: { date: date, parsedDate: parsedDate }\n      \"\n    ></ng-container>\n  </ng-container>\n\n  <ng-template\n    #defaultDateSeparator\n    let-date=\"date\"\n    let-parsedDate=\"parsedDate\"\n  >\n    <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n      <hr\n        *ngIf=\"\n          dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n        \"\n        class=\"str-chat__date-separator-line\"\n      />\n      <div class=\"str-chat__date-separator-date\">\n        {{ parsedDate }}\n      </div>\n      <hr\n        *ngIf=\"\n          dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n        \"\n        class=\"str-chat__date-separator-line\"\n      />\n    </div>\n  </ng-template>\n</ng-template>\n"]}
|
|
479
|
+
//# 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  AfterViewInit,\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 { filter, map, tap } from 'rxjs/operators';\nimport {\n  MessageContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n  TypingIndicatorContext,\n  CustomMessageActionItem,\n  DateSeparatorContext,\n} from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { UserResponse } from 'stream-chat';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\nimport { DateParserService } from '../date-parser.service';\n\n/**\n * The `MessageList` component renders a scrollable list of messages.\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, AfterViewInit\n{\n  /**\n   * Determines if the message list should display channel messages or [thread messages](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'main' | 'thread' = 'main';\n  /**\n   * The direction of the messages in the list, `bottom-to-top` means newest message is at the bottom of the message list and users scroll upwards to load older messages\n   */\n  @Input() direction: 'bottom-to-top' | 'top-to-bottom' = 'bottom-to-top';\n  /**\n   * Determines what triggers the appearance of the message options: by default you can hover (click on mobile) anywhere in the row of the message (`message-row` option), or you can set `message-bubble`, in that case only a hover (click on mobile) in the message bubble will trigger the options to appear.\n   */\n  @Input() messageOptionsTrigger: 'message-row' | 'message-bubble' =\n    'message-row';\n  typingIndicatorTemplate: TemplateRef<TypingIndicatorContext> | undefined;\n  /**\n   * You can hide the \"jump to latest\" button while scrolling. A potential use-case for this input would be to [workaround a known issue on iOS Safar](https://github.com/GetStream/stream-chat-angular/issues/418)\n   */\n  @Input() hideJumpToLatestButtonDuringScroll = false;\n  /**\n   * A list of custom message actions to be displayed in the message action box\n   */\n  @Input() customMessageActions: CustomMessageActionItem<any>[] = [];\n  /**\n   * If `true` date separators will be displayed\n   */\n  @Input() displayDateSeparator = true;\n  /**\n   * If date separators are displayed, you can set the horizontal position of the date text.\n   * If `openMessageListAt` is `last-unread-message` it will also set the text position of the new messages indicator.\n   */\n  @Input() dateSeparatorTextPos: 'center' | 'right' | 'left' = 'center';\n  /**\n   * `last-message` option will open the message list at the last message, `last-unread-message` will open the list at the last unread message. This option only works if mode is `main`.\n   */\n  @Input() openMessageListAt: 'last-message' | 'last-unread-message' =\n    'last-message';\n  messageTemplate: TemplateRef<MessageContext> | undefined;\n  customDateSeparatorTemplate: TemplateRef<DateSeparatorContext> | undefined;\n  customnewMessagesIndicatorTemplate: TemplateRef<void> | undefined;\n  messages$!: Observable<StreamMessage[]>;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner';\n  unreadMessageCount = 0;\n  isUserScrolled: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  parentMessage: StreamMessage | undefined;\n  highlightedMessageId: string | undefined;\n  isLoading = false;\n  isScrollInProgress = false;\n  scrollEndTimeout: any;\n  lastReadMessageId?: string;\n  isJumpingToLatestUnreadMessage = false;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  @ViewChild('parentMessageElement')\n  private parentMessageElement!: ElementRef<HTMLElement>;\n  private latestMessage: { id: string; created_at: Date } | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessage: { id: string; created_at: Date } | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private subscriptions: Subscription[] = [];\n  private newMessageSubscription: { unsubscribe: () => void } | undefined;\n  private prevScrollTop: number | undefined;\n  private usersTypingInChannel$!: Observable<\n    UserResponse<DefaultStreamChatGenerics>[]\n  >;\n  private usersTypingInThread$!: Observable<\n    UserResponse<DefaultStreamChatGenerics>[]\n  >;\n  private isLatestMessageInList = true;\n  private channelId?: string;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private customTemplatesService: CustomTemplatesService,\n    private dateParser: DateParserService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.chatClientService.chatClient?.logger?.(\n          'info',\n          `${channel?.cid || 'undefined'} selected`,\n          { tags: `message list ${this.mode}` }\n        );\n        if (this.channelId !== channel?.id) {\n          this.chatClientService?.chatClient?.logger?.(\n            'info',\n            `new channel is different from prev channel, reseting scroll state`,\n            { tags: `message list ${this.mode}` }\n          );\n          this.resetScrollState();\n          this.channelId = channel?.id;\n          if (\n            this.openMessageListAt === 'last-unread-message' &&\n            this.mode === 'main'\n          ) {\n            this.lastReadMessageId =\n              channel?.state.read[\n                this.chatClientService.chatClient.user?.id || ''\n              ]?.last_read_message_id;\n            if (this.lastReadMessageId) {\n              this.isJumpingToLatestUnreadMessage = true;\n              void this.channelService.jumpToMessage(this.lastReadMessageId);\n            }\n          } else {\n            this.lastReadMessageId = undefined;\n          }\n        }\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.enabledMessageActions = capabilites;\n        }\n        this.newMessageSubscription?.unsubscribe();\n        if (channel) {\n          this.newMessageSubscription = channel.on('message.new', (event) => {\n            // If we display main channel messages and we're switched to an older message set -> use message.new event to update unread count and detect new messages sent by current user\n            if (\n              !event.message ||\n              channel.state.messages === channel.state.latestMessages ||\n              this.mode === 'thread'\n            ) {\n              return;\n            }\n            this.newMessageReceived({\n              id: event.message.id,\n              user: event.message.user,\n              created_at: new Date(event.message.created_at || ''),\n            });\n          });\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    this.subscriptions.push(\n      this.customTemplatesService.messageTemplate$.subscribe(\n        (template) => (this.messageTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.dateSeparatorTemplate$.subscribe(\n        (template) => (this.customDateSeparatorTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.newMessagesIndicatorTemplate$.subscribe(\n        (template) => (this.customnewMessagesIndicatorTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.typingIndicatorTemplate$.subscribe(\n        (template) => (this.typingIndicatorTemplate = template)\n      )\n    );\n    this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;\n    this.usersTypingInThread$ = this.channelService.usersTypingInThread$;\n  }\n\n  ngOnInit(): void {\n    this.setMessages$();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.mode || changes.direction) {\n      this.setMessages$();\n    }\n    if (changes.direction) {\n      if (this.scrollContainer?.nativeElement) {\n        this.jumpToLatestMessage();\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.subscriptions.push(\n      this.channelService.jumpToMessage$\n        .pipe(filter((config) => !!config.id))\n        .subscribe((config) => {\n          let messageId: string | undefined = undefined;\n          if (this.mode === 'main') {\n            messageId = config.parentId || config.id;\n          } else if (config.parentId) {\n            messageId = config.id;\n          }\n          this.chatClientService.chatClient?.logger?.(\n            'info',\n            `Jumping to ${messageId || ''}`,\n            { tags: `message list ${this.mode}` }\n          );\n          if (messageId) {\n            if (messageId === 'latest') {\n              this.scrollToLatestMessage();\n            } else {\n              this.scrollMessageIntoView(messageId);\n              this.highlightedMessageId = messageId;\n            }\n          }\n        })\n    );\n  }\n\n  ngAfterViewChecked() {\n    if (this.highlightedMessageId) {\n      // Turn off programatic scroll adjustments while jump to message is in progress\n      this.hasNewMessages = false;\n      this.olderMassagesLoaded = false;\n    }\n    if (this.direction === 'top-to-bottom') {\n      if (\n        this.hasNewMessages &&\n        (this.isNewMessageSentByUser || !this.isUserScrolled)\n      ) {\n        this.isLatestMessageInList\n          ? this.scrollToTop()\n          : this.jumpToLatestMessage();\n        this.hasNewMessages = false;\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      }\n    } else {\n      if (this.hasNewMessages) {\n        if (!this.isUserScrolled || this.isNewMessageSentByUser) {\n          this.chatClientService.chatClient?.logger?.(\n            'info',\n            `User has new messages, and not scrolled or sent new messages, therefore we ${\n              this.isLatestMessageInList ? 'scroll' : 'jump'\n            } to latest message`,\n            { tags: `message list ${this.mode}` }\n          );\n          this.isLatestMessageInList\n            ? this.scrollToBottom()\n            : this.jumpToLatestMessage();\n        }\n        this.hasNewMessages = false;\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      } else if (this.olderMassagesLoaded) {\n        this.chatClientService.chatClient?.logger?.(\n          'info',\n          `Older messages are loaded, we preserve the scroll position`,\n          { tags: `message list ${this.mode}` }\n        );\n        this.preserveScrollbarPosition();\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n        this.olderMassagesLoaded = false;\n      } else if (\n        this.getScrollPosition() !== 'bottom' &&\n        !this.isUserScrolled &&\n        !this.highlightedMessageId\n      ) {\n        this.chatClientService.chatClient?.logger?.(\n          'info',\n          `Container grew and user didn't scroll therefore we ${\n            this.isLatestMessageInList ? 'scroll' : 'jump'\n          } to latest message`,\n          { tags: `message list ${this.mode}` }\n        );\n        this.isLatestMessageInList\n          ? this.scrollToBottom()\n          : this.jumpToLatestMessage();\n        this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n    this.newMessageSubscription?.unsubscribe();\n    if (this.scrollEndTimeout) {\n      clearTimeout(this.scrollEndTimeout);\n    }\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  trackByUserId(index: number, user: UserResponse) {\n    return user.id;\n  }\n\n  jumpToLatestMessage() {\n    void this.channelService.jumpToMessage(\n      'latest',\n      this.mode === 'thread' ? this.parentMessage?.id : undefined\n    );\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrollToTop() {\n    this.scrollContainer.nativeElement.scrollTop = 0;\n  }\n\n  scrolled() {\n    this.isScrollInProgress = true;\n    if (this.scrollEndTimeout) {\n      clearTimeout(this.scrollEndTimeout);\n    }\n    this.scrollEndTimeout = setTimeout(() => {\n      this.isScrollInProgress = false;\n    }, 100);\n    if (\n      this.scrollContainer.nativeElement.scrollHeight ===\n      this.scrollContainer.nativeElement.clientHeight\n    ) {\n      return;\n    }\n    const scrollPosition = this.getScrollPosition();\n    this.chatClientService.chatClient?.logger?.(\n      'info',\n      `Scrolled - scroll position: ${scrollPosition}`,\n      { tags: `message list ${this.mode}` }\n    );\n\n    this.isUserScrolled =\n      (this.direction === 'bottom-to-top'\n        ? scrollPosition !== 'bottom'\n        : scrollPosition !== 'top') || !this.isLatestMessageInList;\n    if (!this.isUserScrolled) {\n      this.unreadMessageCount = 0;\n    }\n    if (this.shouldLoadMoreMessages(scrollPosition)) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      let direction: 'newer' | 'older';\n      if (this.direction === 'top-to-bottom') {\n        direction = scrollPosition === 'top' ? 'newer' : 'older';\n      } else {\n        direction = scrollPosition === 'top' ? 'older' : 'newer';\n      }\n      this.mode === 'main'\n        ? void this.channelService.loadMoreMessages(direction)\n        : void this.channelService.loadMoreThreadReplies(direction);\n      this.isLoading = true;\n    }\n    this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;\n  }\n\n  getTypingIndicatorContext(): TypingIndicatorContext {\n    return {\n      usersTyping$: this.usersTyping$,\n    };\n  }\n\n  getTypingIndicatorText(users: UserResponse[]) {\n    const text = listUsers(users);\n\n    return text;\n  }\n\n  areOnSeparateDates(message?: StreamMessage, nextMessage?: StreamMessage) {\n    if (!message || !nextMessage) {\n      return false;\n    }\n    if (message.created_at.getDate() !== nextMessage.created_at.getDate()) {\n      return true;\n    } else if (\n      message.created_at.getFullYear() !==\n        nextMessage.created_at.getFullYear() ||\n      message.created_at.getMonth() !== nextMessage.created_at.getMonth()\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  isSentByCurrentUser(message?: StreamMessage) {\n    if (!message) {\n      return false;\n    }\n    return message.user?.id === this.chatClientService.chatClient.user?.id;\n  }\n\n  parseDate(date: Date) {\n    return this.dateParser.parseDate(date);\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.parentMessage?.reply_count };\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 getScrollPosition(): 'top' | 'bottom' | 'middle' {\n    let position: 'top' | 'bottom' | 'middle' = 'middle';\n    if (\n      Math.floor(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      position = 'top';\n    } else if (\n      Math.ceil(this.scrollContainer.nativeElement.scrollTop) +\n        this.scrollContainer.nativeElement.clientHeight >=\n      this.scrollContainer.nativeElement.scrollHeight\n    ) {\n      position = 'bottom';\n    }\n\n    return position;\n  }\n\n  private shouldLoadMoreMessages(scrollPosition: 'top' | 'bottom' | 'middle') {\n    return scrollPosition !== 'middle' && !this.highlightedMessageId;\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        this.isLoading = false;\n        if (messages.length === 0) {\n          this.chatClientService.chatClient?.logger?.(\n            'info',\n            `Empty messages array, reseting scroll state`,\n            {\n              tags: `message list ${this.mode}`,\n            }\n          );\n          this.resetScrollState();\n          return;\n        }\n        this.chatClientService.chatClient?.logger?.(\n          'info',\n          `Received one or more messages`,\n          {\n            tags: `message list ${this.mode}`,\n          }\n        );\n        const currentLatestMessage = messages[messages.length - 1];\n        this.newMessageReceived(currentLatestMessage);\n        const currentOldestMessage = messages[0];\n        if (\n          !this.oldestMessage ||\n          !messages.find((m) => m.id === this.oldestMessage!.id)\n        ) {\n          this.oldestMessage = currentOldestMessage;\n        } else if (\n          this.oldestMessage.created_at.getTime() >\n          currentOldestMessage.created_at.getTime()\n        ) {\n          this.oldestMessage = currentOldestMessage;\n          this.olderMassagesLoaded = true;\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      tap((messages) => {\n        this.isLatestMessageInList =\n          !this.latestMessage ||\n          messages.length === 0 ||\n          messages[messages.length - 1].id === this.latestMessage.id;\n        if (!this.isLatestMessageInList) {\n          this.isUserScrolled = true;\n        }\n      }),\n      tap(() => {\n        if (\n          this.isJumpingToLatestUnreadMessage &&\n          this.lastReadMessageId &&\n          this.lastReadMessageId === this.latestMessage?.id\n        ) {\n          this.lastReadMessageId = undefined;\n        }\n      }),\n      map((messages) =>\n        this.direction === 'bottom-to-top' ? messages : [...messages].reverse()\n      ),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      })\n    );\n  }\n\n  private resetScrollState() {\n    this.latestMessage = undefined;\n    this.hasNewMessages = true;\n    this.isUserScrolled = false;\n    this.containerHeight = undefined;\n    this.olderMassagesLoaded = false;\n    this.oldestMessage = undefined;\n    this.unreadMessageCount = 0;\n    this.prevScrollTop = undefined;\n    this.isNewMessageSentByUser = undefined;\n    this.isLatestMessageInList = true;\n  }\n\n  private get usersTyping$() {\n    return this.mode === 'thread'\n      ? this.usersTypingInThread$\n      : this.usersTypingInChannel$;\n  }\n\n  private scrollMessageIntoView(messageId: string, withRetry = true) {\n    const element = document.getElementById(messageId);\n    if (!element && withRetry) {\n      // If the message was newly inserted into activeChannelMessages$, the message will be rendered after the current change detection cycle -> wait for this cycle to complete\n      setTimeout(() => this.scrollMessageIntoView(messageId, false));\n    } else if (element) {\n      element.scrollIntoView({ block: 'center' });\n      setTimeout(() => {\n        this.highlightedMessageId = undefined;\n        this.isJumpingToLatestUnreadMessage = false;\n      }, 1000);\n    }\n  }\n\n  private scrollToLatestMessage(withRetry = true) {\n    if (document.getElementById(this.latestMessage!.id)) {\n      this.direction === 'bottom-to-top'\n        ? this.scrollToBottom()\n        : this.scrollToTop();\n    } else if (withRetry) {\n      // If the message was newly inserted into activeChannelMessages$, the message will be rendered after the current change detection cycle -> wait for this cycle to complete\n      setTimeout(() => this.scrollToLatestMessage(false), 0);\n    }\n  }\n\n  private newMessageReceived(message: {\n    id: string;\n    created_at: Date;\n    user?: { id: string } | null;\n  }) {\n    if (\n      !this.latestMessage ||\n      this.latestMessage.created_at?.getTime() < message.created_at.getTime()\n    ) {\n      this.chatClientService.chatClient?.logger?.(\n        'info',\n        `Received new message`,\n        { tags: `message list ${this.mode}` }\n      );\n      this.latestMessage = message;\n      this.hasNewMessages = true;\n      this.isNewMessageSentByUser =\n        message.user?.id === this.chatClientService.chatClient?.user?.id;\n      if (this.isUserScrolled) {\n        this.unreadMessageCount++;\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 str-chat__message-list-scroll\">\n    <ul\n      class=\"str-chat__ul\"\n      [class.str-chat__message-options-in-bubble]=\"\n        messageOptionsTrigger === 'message-bubble'\n      \"\n    >\n      <li\n        #parentMessageElement\n        *ngIf=\"mode === 'thread' && parentMessage\"\n        data-testid=\"parent-message\"\n        class=\"str-chat__parent-message-li\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage, index: 'parent' }\n          \"\n        ></ng-container>\n        <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n          {{parentMessage?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n          replies' | translate:replyCountParam)}}\n        </div>\n      </li>\n      <stream-loading-indicator\n        data-testid=\"top-loading-indicator\"\n        *ngIf=\"isLoading && direction === 'bottom-to-top'\"\n      ></stream-loading-indicator>\n      <ng-container *ngIf=\"messages$ | async as messages\">\n        <ng-container\n          *ngFor=\"\n            let message of messages;\n            let i = index;\n            let isFirst = first;\n            let isLast = last;\n            trackBy: trackByMessageId\n          \"\n        >\n          <ng-container *ngIf=\"isFirst\">\n            <ng-container\n              *ngTemplateOutlet=\"\n                dateSeparator;\n                context: {\n                  date: message.created_at,\n                  parsedDate: parseDate(message.created_at),\n                  isNewMessage: false\n                }\n              \"\n            ></ng-container>\n          </ng-container>\n          <li\n            tabindex=\"0\"\n            data-testclass=\"message\"\n            class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n            id=\"{{ message.id }}\"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"\n                messageTemplateContainer;\n                context: { message: message, index: i }\n              \"\n            ></ng-container>\n          </li>\n          <ng-container\n            *ngIf=\"\n              (lastReadMessageId === message.id &&\n                !isLast &&\n                direction === 'bottom-to-top' &&\n                !isSentByCurrentUser(messages[i + 1]) &&\n                (!displayDateSeparator ||\n                  !areOnSeparateDates(message, messages[i + 1]))) ||\n              (direction === 'top-to-bottom' &&\n                !isLast &&\n                !isSentByCurrentUser(message) &&\n                lastReadMessageId === messages[i + 1].id)\n            \"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"\n                customnewMessagesIndicatorTemplate ||\n                defaultNewMessagesIndicator\n              \"\n            ></ng-container>\n          </ng-container>\n          <ng-container *ngIf=\"areOnSeparateDates(message, messages[i + 1])\">\n            <ng-container\n              *ngTemplateOutlet=\"\n                dateSeparator;\n                context: {\n                  date: messages[i + 1].created_at,\n                  parsedDate: parseDate(messages[i + 1].created_at),\n                  isNewMessage:\n                    (direction === 'bottom-to-top' &&\n                      message.id === lastReadMessageId &&\n                      !isSentByCurrentUser(messages[i + 1])) ||\n                    (direction === 'top-to-bottom' && false)\n                }\n              \"\n            ></ng-container>\n          </ng-container>\n        </ng-container>\n      </ng-container>\n      <stream-loading-indicator\n        data-testid=\"bottom-loading-indicator\"\n        *ngIf=\"isLoading && direction === 'top-to-bottom'\"\n      ></stream-loading-indicator>\n    </ul>\n    <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n      <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n      <ng-container *ngIf=\"$any(usersTyping$ | async) as users\">\n        <div\n          *ngIf=\"users.length > 0\"\n          data-testid=\"typing-indicator\"\n          class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n        >\n          <div class=\"str-chat__typing-indicator__dots\">\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n            <span class=\"str-chat__typing-indicator__dot\"></span>\n          </div>\n          <div\n            data-testid=\"typing-users\"\n            class=\"str-chat__typing-indicator__users\"\n          >\n            {{\n              users.length === 1\n                ? (\"streamChat.user is typing\"\n                  | translate: { user: getTypingIndicatorText(users) })\n                : (\"streamChat.users are typing\"\n                  | translate: { users: getTypingIndicatorText(users) })\n            }}\n          </div>\n        </div>\n      </ng-container>\n    </ng-template>\n    <ng-container\n      *ngTemplateOutlet=\"\n        typingIndicatorTemplate || defaultTypingIndicator;\n        context: getTypingIndicatorContext()\n      \"\n    ></ng-container>\n  </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n  <button\n    data-testid=\"scroll-to-latest\"\n    *ngIf=\"\n      isUserScrolled &&\n      (!isScrollInProgress || !hideJumpToLatestButtonDuringScroll)\n    \"\n    class=\"\n      str-chat__message-notification-scroll-to-latest\n      str-chat__message-notification-scroll-to-latest-right\n      str-chat__circle-fab\n    \"\n    (keyup.enter)=\"jumpToLatestMessage()\"\n    (click)=\"jumpToLatestMessage()\"\n  >\n    <stream-icon\n      class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n      [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n    ></stream-icon>\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-to-latest-unread-count\n        str-chat__jump-to-latest-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n  <ng-template\n    #defaultMessageTemplate\n    let-messageInput=\"message\"\n    let-isLastSentMessage=\"isLastSentMessage\"\n    let-enabledMessageActions=\"enabledMessageActions\"\n    let-mode=\"mode\"\n    let-isHighlighted=\"isHighlighted\"\n    let-customActions=\"customActions\"\n  >\n    <stream-message\n      [message]=\"messageInput\"\n      [isLastSentMessage]=\"isLastSentMessage\"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [mode]=\"mode\"\n      [isHighlighted]=\"isHighlighted\"\n      [customActions]=\"customActions\"\n    ></stream-message>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      messageTemplate || defaultMessageTemplate;\n      context: {\n        message: message,\n        isLastSentMessage: !!(\n          lastSentMessageId && message?.id === lastSentMessageId\n        ),\n        enabledMessageActions: enabledMessageActions,\n        mode: mode,\n        isHighlighted:\n          message?.id === highlightedMessageId &&\n          !isJumpingToLatestUnreadMessage,\n        customActions: customMessageActions\n      }\n    \"\n  ></ng-container>\n</ng-template>\n\n<ng-template\n  #dateSeparator\n  let-date=\"date\"\n  let-parsedDate=\"parsedDate\"\n  let-isNewMessage=\"isNewMessage\"\n>\n  <ng-container *ngIf=\"displayDateSeparator\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        customDateSeparatorTemplate || defaultDateSeparator;\n        context: {\n          date: date,\n          parsedDate: parsedDate,\n          isNewMessage: isNewMessage\n        }\n      \"\n    ></ng-container>\n  </ng-container>\n\n  <ng-template\n    #defaultDateSeparator\n    let-date=\"date\"\n    let-parsedDate=\"parsedDate\"\n    let-isNewMessage=\"isNewMessage\"\n  >\n    <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n      <hr\n        *ngIf=\"\n          dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n        \"\n        class=\"str-chat__date-separator-line\"\n      />\n      <div class=\"str-chat__date-separator-date\">\n        {{ parsedDate }}\n        <span\n          *ngIf=\"isNewMessage\"\n          data-testid=\"new-messages-indicator-date-separator\"\n          >• {{ \"streamChat.New\" | translate }}</span\n        >\n      </div>\n      <hr\n        *ngIf=\"\n          dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n        \"\n        class=\"str-chat__date-separator-line\"\n      />\n    </div>\n  </ng-template>\n</ng-template>\n\n<ng-template #defaultNewMessagesIndicator>\n  <div data-testid=\"new-messages-indicator\" class=\"str-chat__date-separator\">\n    <hr\n      *ngIf=\"\n        dateSeparatorTextPos === 'right' || dateSeparatorTextPos === 'center'\n      \"\n      class=\"str-chat__date-separator-line\"\n    />\n    <div class=\"str-chat__date-separator-date\" translate>streamChat.New</div>\n    <hr\n      *ngIf=\"\n        dateSeparatorTextPos === 'left' || dateSeparatorTextPos === 'center'\n      \"\n      class=\"str-chat__date-separator-line\"\n    />\n  </div>\n</ng-template>\n"]}
|
package/esm2015/lib/types.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
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  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 AttachmentUpload<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  file: File;\n  state: 'error' | 'success' | 'uploading';\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 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};\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  customActions: CustomMessageActionItem[];\n  /**\n   * @deprecated because the name contains typos, use the `displayedActionsCountChangeHandler` instead\n   */\n  displayedActionsCountChaneHanler: (count: number) => any;\n  displayedActionsCountChangeHandler: (count: number) => any;\n  isEditingChangeHandler: (isEditing: boolean) => 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';\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 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 ChannelQueryState = {\n  state: 'in-progress' | 'success' | 'error';\n  // No type def from stream-chat\n  error?: unknown;\n};\n"]}
|
|
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  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 AttachmentUpload<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  file: File;\n  state: 'error' | 'success' | 'uploading';\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 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};\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  customActions: CustomMessageActionItem[];\n  /**\n   * @deprecated because the name contains typos, use the `displayedActionsCountChangeHandler` instead\n   */\n  displayedActionsCountChaneHanler: (count: number) => any;\n  displayedActionsCountChangeHandler: (count: number) => any;\n  isEditingChangeHandler: (isEditing: boolean) => 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';\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 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  isNewMessage: boolean;\n};\n\nexport type ChannelQueryState = {\n  state: 'in-progress' | 'success' | 'error';\n  // No type def from stream-chat\n  error?: unknown;\n};\n"]}
|