stream-chat-angular 5.13.0 → 6.0.0-beta.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/i18n/en.d.ts +1 -0
- package/assets/version.d.ts +1 -1
- package/{esm2020 → esm2022}/assets/i18n/en.mjs +2 -1
- package/{esm2020 → esm2022}/assets/version.mjs +2 -2
- package/{esm2020 → esm2022}/lib/attachment-configuration.service.mjs +4 -4
- package/esm2022/lib/attachment-list/attachment-list.component.mjs +212 -0
- package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
- package/{esm2020 → esm2022}/lib/attachment.service.mjs +5 -5
- package/esm2022/lib/avatar/avatar.component.mjs +157 -0
- package/{esm2020 → esm2022}/lib/avatar-placeholder/avatar-placeholder.component.mjs +6 -6
- package/esm2022/lib/channel/channel.component.mjs +45 -0
- package/esm2022/lib/channel-header/channel-header.component.mjs +72 -0
- package/esm2022/lib/channel-list/channel-list.component.mjs +50 -0
- package/esm2022/lib/channel-preview/channel-preview.component.mjs +150 -0
- package/esm2022/lib/channel.service.mjs +1389 -0
- package/esm2022/lib/chat-client.service.mjs +227 -0
- package/{esm2020 → esm2022}/lib/custom-templates.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/date-parser.service.mjs +5 -5
- package/esm2022/lib/file-utils.mjs +35 -0
- package/{esm2020 → esm2022}/lib/get-channel-display-text.mjs +1 -1
- package/{esm2020 → esm2022}/lib/get-message-translation.mjs +1 -1
- package/{esm2020 → esm2022}/lib/icon/icon-placeholder/icon-placeholder.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/icon/icon.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/icon/icon.module.mjs +11 -11
- package/{esm2020 → esm2022}/lib/icon/loading-indicator/loading-indicator.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/list-users.mjs +1 -1
- package/esm2022/lib/message/message.component.mjs +486 -0
- package/esm2022/lib/message-actions-box/message-actions-box.component.mjs +120 -0
- package/{esm2020 → esm2022}/lib/message-actions.service.mjs +5 -5
- package/esm2022/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
- package/{esm2020 → esm2022}/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message-input/emoji-input.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/message-input-config.service.mjs +5 -5
- package/esm2022/lib/message-input/message-input.component.mjs +507 -0
- package/{esm2020 → esm2022}/lib/message-input/textarea/textarea.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/textarea.directive.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/voice-recorder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-list/group-styles.mjs +1 -1
- package/esm2022/lib/message-list/message-list.component.mjs +715 -0
- package/{esm2020 → esm2022}/lib/message-preview.mjs +1 -1
- package/esm2022/lib/message-reactions/message-reactions.component.mjs +165 -0
- package/esm2022/lib/message-reactions-selector/message-reactions-selector.component.mjs +57 -0
- package/{esm2020 → esm2022}/lib/message-reactions.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-text/message-text.component.mjs +6 -6
- package/esm2022/lib/message.service.mjs +43 -0
- package/{esm2020 → esm2022}/lib/modal/modal.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/notification/notification.component.mjs +6 -6
- package/esm2022/lib/notification-list/notification-list.component.mjs +33 -0
- package/{esm2020 → esm2022}/lib/notification.service.mjs +5 -5
- package/esm2022/lib/paginated-list/paginated-list.component.mjs +94 -0
- package/{esm2020 → esm2022}/lib/parse-date.mjs +1 -1
- package/{esm2020 → esm2022}/lib/read-by.mjs +1 -1
- package/esm2022/lib/stream-autocomplete-textarea.module.mjs +33 -0
- package/{esm2020 → esm2022}/lib/stream-avatar.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/stream-chat.module.mjs +59 -59
- package/{esm2020 → esm2022}/lib/stream-i18n.service.mjs +5 -5
- package/esm2022/lib/stream-textarea.module.mjs +31 -0
- package/{esm2020 → esm2022}/lib/theme.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/thread/thread.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/transliteration.service.mjs +5 -5
- package/esm2022/lib/types.mjs +2 -0
- package/{esm2020 → esm2022}/lib/user-list/user-list.component.mjs +5 -5
- package/esm2022/lib/virtualized-list.service.mjs +273 -0
- package/{esm2020 → esm2022}/lib/virtualized-message-list.service.mjs +1 -1
- package/{esm2020 → esm2022}/lib/voice-recorder/amplitude-recorder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/audio-recorder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/media-recorder.mjs +1 -1
- package/esm2022/lib/voice-recorder/mp3-transcoder.mjs +61 -0
- package/esm2022/lib/voice-recorder/transcoder.service.mjs +121 -0
- package/esm2022/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +32 -0
- package/esm2022/lib/voice-recorder/voice-recorder.component.mjs +80 -0
- package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
- package/esm2022/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +112 -0
- package/esm2022/lib/voice-recording/voice-recording.component.mjs +91 -0
- package/{esm2020 → esm2022}/lib/voice-recording/voice-recording.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/wave-form-sampler.mjs +1 -1
- package/esm2022/public-api.mjs +82 -0
- package/{fesm2020 → fesm2022}/stream-chat-angular.mjs +865 -1140
- package/fesm2022/stream-chat-angular.mjs.map +1 -0
- package/lib/attachment-list/attachment-list.component.d.ts +2 -5
- package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +2 -2
- package/lib/attachment.service.d.ts +1 -1
- package/lib/avatar/avatar.component.d.ts +4 -4
- package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
- package/lib/channel-list/channel-list.component.d.ts +1 -0
- package/lib/channel-preview/channel-preview.component.d.ts +3 -4
- package/lib/channel.service.d.ts +40 -106
- package/lib/chat-client.service.d.ts +1 -4
- package/lib/custom-templates.service.d.ts +10 -10
- package/lib/icon/icon-placeholder/icon-placeholder.component.d.ts +1 -1
- package/lib/icon/icon.component.d.ts +2 -2
- package/lib/message/message.component.d.ts +2 -2
- package/lib/message-actions-box/message-actions-box.component.d.ts +2 -3
- package/lib/message-actions.service.d.ts +1 -1
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +2 -2
- package/lib/message-input/message-input.component.d.ts +2 -2
- package/lib/message-input/textarea/textarea.component.d.ts +1 -1
- package/lib/message-input/textarea.directive.d.ts +2 -2
- package/lib/message-list/group-styles.d.ts +1 -1
- package/lib/message-list/message-list.component.d.ts +2 -3
- package/lib/message-reactions/message-reactions.component.d.ts +2 -3
- package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +1 -2
- package/lib/message-text/message-text.component.d.ts +2 -2
- package/lib/modal/modal.component.d.ts +1 -1
- package/lib/notification/notification.component.d.ts +1 -1
- package/lib/notification-list/notification-list.component.d.ts +0 -1
- package/lib/paginated-list/paginated-list.component.d.ts +5 -2
- package/lib/read-by.d.ts +1 -1
- package/lib/types.d.ts +98 -84
- package/lib/user-list/user-list.component.d.ts +1 -1
- package/lib/voice-recorder/amplitude-recorder.service.d.ts +2 -2
- package/lib/voice-recorder/media-recorder.d.ts +2 -2
- package/lib/voice-recorder/transcoder.service.d.ts +4 -4
- package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +0 -1
- package/lib/voice-recorder/voice-recorder.component.d.ts +2 -2
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +3 -3
- package/lib/voice-recording/voice-recording.component.d.ts +1 -1
- package/package.json +15 -21
- package/public-api.d.ts +0 -1
- package/src/assets/i18n/en.ts +1 -0
- package/src/assets/version.ts +1 -1
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +0 -224
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
- package/esm2020/lib/avatar/avatar.component.mjs +0 -160
- package/esm2020/lib/channel/channel.component.mjs +0 -45
- package/esm2020/lib/channel-header/channel-header.component.mjs +0 -72
- package/esm2020/lib/channel-list/channel-list.component.mjs +0 -47
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +0 -155
- package/esm2020/lib/channel-query.mjs +0 -77
- package/esm2020/lib/channel.service.mjs +0 -1546
- package/esm2020/lib/chat-client.service.mjs +0 -238
- package/esm2020/lib/file-utils.mjs +0 -35
- package/esm2020/lib/message/message.component.mjs +0 -486
- package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +0 -123
- package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +0 -71
- package/esm2020/lib/message-input/message-input.component.mjs +0 -507
- package/esm2020/lib/message-list/message-list.component.mjs +0 -717
- package/esm2020/lib/message-reactions/message-reactions.component.mjs +0 -168
- package/esm2020/lib/message-reactions-selector/message-reactions-selector.component.mjs +0 -61
- package/esm2020/lib/message.service.mjs +0 -43
- package/esm2020/lib/notification-list/notification-list.component.mjs +0 -36
- package/esm2020/lib/paginated-list/paginated-list.component.mjs +0 -94
- package/esm2020/lib/stream-autocomplete-textarea.module.mjs +0 -33
- package/esm2020/lib/stream-textarea.module.mjs +0 -31
- package/esm2020/lib/types.mjs +0 -2
- package/esm2020/lib/virtualized-list.service.mjs +0 -271
- package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +0 -61
- package/esm2020/lib/voice-recorder/transcoder.service.mjs +0 -121
- package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +0 -35
- package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +0 -80
- package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +0 -112
- package/esm2020/lib/voice-recording/voice-recording.component.mjs +0 -91
- package/esm2020/public-api.mjs +0 -83
- package/fesm2015/stream-chat-angular.mjs +0 -9152
- package/fesm2015/stream-chat-angular.mjs.map +0 -1
- package/fesm2020/stream-chat-angular.mjs.map +0 -1
- package/lib/channel-query.d.ts +0 -26
- /package/{esm2020 → esm2022}/lib/format-duration.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/injection-tokens.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-image-attachment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-on-separate-date.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-safari.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/message-input/textarea.interface.mjs +0 -0
- /package/{esm2020 → esm2022}/stream-chat-angular.mjs +0 -0
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { Component, } from '@angular/core';
|
|
2
|
+
import { getChannelDisplayText } from '../get-channel-display-text';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../channel.service";
|
|
5
|
+
import * as i2 from "../custom-templates.service";
|
|
6
|
+
import * as i3 from "../chat-client.service";
|
|
7
|
+
import * as i4 from "@angular/common";
|
|
8
|
+
import * as i5 from "../avatar-placeholder/avatar-placeholder.component";
|
|
9
|
+
import * as i6 from "@ngx-translate/core";
|
|
10
|
+
/**
|
|
11
|
+
* The `ChannelHeader` component displays the avatar and name of the currently active channel along with member and watcher information. You can read about [the difference between members and watchers](/chat/docs/javascript/watch_channel/#watchers-vs-members) in the platform documentation. Please note that number of watchers is only displayed if the user has [`connect-events` capability](/chat/docs/javascript/channel_capabilities/)
|
|
12
|
+
*/
|
|
13
|
+
export class ChannelHeaderComponent {
|
|
14
|
+
constructor(channelService, customTemplatesService, cdRef, chatClientService) {
|
|
15
|
+
this.channelService = channelService;
|
|
16
|
+
this.customTemplatesService = customTemplatesService;
|
|
17
|
+
this.cdRef = cdRef;
|
|
18
|
+
this.chatClientService = chatClientService;
|
|
19
|
+
this.subscriptions = [];
|
|
20
|
+
this.channelService.activeChannel$.subscribe((c) => {
|
|
21
|
+
this.activeChannel = c;
|
|
22
|
+
const capabilities = this.activeChannel?.data
|
|
23
|
+
?.own_capabilities;
|
|
24
|
+
if (!capabilities) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
this.canReceiveConnectEvents =
|
|
28
|
+
capabilities.indexOf('connect-events') !== -1;
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
ngOnInit() {
|
|
32
|
+
this.subscriptions.push(this.customTemplatesService.channelActionsTemplate$.subscribe((template) => {
|
|
33
|
+
this.channelActionsTemplate = template;
|
|
34
|
+
this.cdRef.detectChanges();
|
|
35
|
+
}));
|
|
36
|
+
this.subscriptions.push(this.customTemplatesService.channelHeaderInfoTemplate$.subscribe((template) => {
|
|
37
|
+
this.channelHeaderInfoTemplate = template;
|
|
38
|
+
this.cdRef.detectChanges();
|
|
39
|
+
}));
|
|
40
|
+
}
|
|
41
|
+
ngOnDestroy() {
|
|
42
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
43
|
+
}
|
|
44
|
+
getChannelActionsContext() {
|
|
45
|
+
return { channel: this.activeChannel };
|
|
46
|
+
}
|
|
47
|
+
getChannelInfoContext() {
|
|
48
|
+
return { channel: this.activeChannel };
|
|
49
|
+
}
|
|
50
|
+
get memberCountParam() {
|
|
51
|
+
return { memberCount: this.activeChannel?.data?.member_count || 0 };
|
|
52
|
+
}
|
|
53
|
+
get watcherCountParam() {
|
|
54
|
+
return { watcherCount: this.activeChannel?.state?.watcher_count || 0 };
|
|
55
|
+
}
|
|
56
|
+
get displayText() {
|
|
57
|
+
if (!this.activeChannel) {
|
|
58
|
+
return '';
|
|
59
|
+
}
|
|
60
|
+
return getChannelDisplayText(this.activeChannel, this.chatClientService.chatClient.user);
|
|
61
|
+
}
|
|
62
|
+
get avatarName() {
|
|
63
|
+
return this.activeChannel?.data?.name;
|
|
64
|
+
}
|
|
65
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelHeaderComponent, deps: [{ token: i1.ChannelService }, { token: i2.CustomTemplatesService }, { token: i0.ChangeDetectorRef }, { token: i3.ChatClientService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
66
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: ChannelHeaderComponent, selector: "stream-channel-header", ngImport: i0, template: "<div class=\"str-chat__header-livestream str-chat__channel-header\">\n <ng-content />\n <stream-avatar-placeholder\n type=\"channel\"\n location=\"channel-header\"\n imageUrl=\"{{ activeChannel?.data?.image }}\"\n name=\"{{ avatarName }}\"\n [channel]=\"activeChannel\"\n />\n <div class=\"str-chat__header-livestream-left str-chat__channel-header-end\">\n <p\n data-testid=\"name\"\n class=\"str-chat__header-livestream-left--title str-chat__channel-header-title\"\n >\n {{ displayText }}\n </p>\n <ng-container\n *ngTemplateOutlet=\"\n channelHeaderInfoTemplate || defaultChannelInfo;\n context: getChannelInfoContext()\n \"\n />\n <ng-template #defaultChannelInfo>\n <p\n data-testid=\"info\"\n class=\"str-chat__header-livestream-left--members str-chat__channel-header-info\"\n >\n {{'streamChat.{{ memberCount }} members' | translate:memberCountParam}}\n {{canReceiveConnectEvents ? ('streamChat.{{ watcherCount }} online' |\n translate:watcherCountParam) : ''}}\n </p>\n </ng-template>\n </div>\n @if (channelActionsTemplate) {\n <ng-container\n *ngTemplateOutlet=\"\n channelActionsTemplate;\n context: getChannelActionsContext()\n \"\n />\n }\n</div>\n", dependencies: [{ kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i5.AvatarPlaceholderComponent, selector: "stream-avatar-placeholder", inputs: ["name", "imageUrl", "location", "channel", "user", "type", "initialsType", "showOnlineIndicator"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] }); }
|
|
67
|
+
}
|
|
68
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelHeaderComponent, decorators: [{
|
|
69
|
+
type: Component,
|
|
70
|
+
args: [{ selector: 'stream-channel-header', template: "<div class=\"str-chat__header-livestream str-chat__channel-header\">\n <ng-content />\n <stream-avatar-placeholder\n type=\"channel\"\n location=\"channel-header\"\n imageUrl=\"{{ activeChannel?.data?.image }}\"\n name=\"{{ avatarName }}\"\n [channel]=\"activeChannel\"\n />\n <div class=\"str-chat__header-livestream-left str-chat__channel-header-end\">\n <p\n data-testid=\"name\"\n class=\"str-chat__header-livestream-left--title str-chat__channel-header-title\"\n >\n {{ displayText }}\n </p>\n <ng-container\n *ngTemplateOutlet=\"\n channelHeaderInfoTemplate || defaultChannelInfo;\n context: getChannelInfoContext()\n \"\n />\n <ng-template #defaultChannelInfo>\n <p\n data-testid=\"info\"\n class=\"str-chat__header-livestream-left--members str-chat__channel-header-info\"\n >\n {{'streamChat.{{ memberCount }} members' | translate:memberCountParam}}\n {{canReceiveConnectEvents ? ('streamChat.{{ watcherCount }} online' |\n translate:watcherCountParam) : ''}}\n </p>\n </ng-template>\n </div>\n @if (channelActionsTemplate) {\n <ng-container\n *ngTemplateOutlet=\"\n channelActionsTemplate;\n context: getChannelActionsContext()\n \"\n />\n }\n</div>\n" }]
|
|
71
|
+
}], ctorParameters: () => [{ type: i1.ChannelService }, { type: i2.CustomTemplatesService }, { type: i0.ChangeDetectorRef }, { type: i3.ChatClientService }] });
|
|
72
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel-header.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/channel-header/channel-header.component.ts","../../../../../projects/stream-chat-angular/src/lib/channel-header/channel-header.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,SAAS,GAIV,MAAM,eAAe,CAAC;AAMvB,OAAO,EAAE,qBAAqB,EAAE,MAAM,6BAA6B,CAAC;;;;;;;;AAOpE;;GAEG;AAMH,MAAM,OAAO,sBAAsB;IAOjC,YACU,cAA8B,EAC9B,sBAA8C,EAC9C,KAAwB,EACxB,iBAAoC;QAHpC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,2BAAsB,GAAtB,sBAAsB,CAAwB;QAC9C,UAAK,GAAL,KAAK,CAAmB;QACxB,sBAAiB,GAAjB,iBAAiB,CAAmB;QANtC,kBAAa,GAAmB,EAAE,CAAC;QAQzC,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YACjD,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;YACvB,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,EAAE,IAAI;gBAC3C,EAAE,gBAA4B,CAAC;YACjC,IAAI,CAAC,YAAY,EAAE;gBACjB,OAAO;aACR;YACD,IAAI,CAAC,uBAAuB;gBAC1B,YAAY,CAAC,OAAO,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;IACL,CAAC;IACD,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,sBAAsB,CAAC,uBAAuB,CAAC,SAAS,CAC3D,CAAC,QAAQ,EAAE,EAAE;YACX,IAAI,CAAC,sBAAsB,GAAG,QAAQ,CAAC;YACvC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;QAC7B,CAAC,CACF,CACF,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,sBAAsB,CAAC,0BAA0B,CAAC,SAAS,CAC9D,CAAC,QAAQ,EAAE,EAAE;YACX,IAAI,CAAC,yBAAyB,GAAG,QAAQ,CAAC;YAC1C,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;QAC7B,CAAC,CACF,CACF,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,wBAAwB;QACtB,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,aAAc,EAAE,CAAC;IAC1C,CAAC;IAED,qBAAqB;QACnB,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,aAAc,EAAE,CAAC;IAC1C,CAAC;IAED,IAAI,gBAAgB;QAClB,OAAO,EAAE,WAAW,EAAE,IAAI,CAAC,aAAa,EAAE,IAAI,EAAE,YAAY,IAAI,CAAC,EAAE,CAAC;IACtE,CAAC;IAED,IAAI,iBAAiB;QACnB,OAAO,EAAE,YAAY,EAAE,IAAI,CAAC,aAAa,EAAE,KAAK,EAAE,aAAa,IAAI,CAAC,EAAE,CAAC;IACzE,CAAC;IAED,IAAI,WAAW;QACb,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE;YACvB,OAAO,EAAE,CAAC;SACX;QACD,OAAO,qBAAqB,CAC1B,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAK,CACxC,CAAC;IACJ,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,aAAa,EAAE,IAAI,EAAE,IAAI,CAAC;IACxC,CAAC;8GA3EU,sBAAsB;kGAAtB,sBAAsB,6DC3BnC,4yCA0CA;;2FDfa,sBAAsB;kBALlC,SAAS;+BACE,uBAAuB","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { Channel } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { getChannelDisplayText } from '../get-channel-display-text';\nimport {\n  ChannelActionsContext,\n  ChannelHeaderInfoContext,\n  DefaultStreamChatGenerics,\n} from '../types';\n\n/**\n * The `ChannelHeader` component displays the avatar and name of the currently active channel along with member and watcher information. You can read about [the difference between members and watchers](/chat/docs/javascript/watch_channel/#watchers-vs-members) in the platform documentation. Please note that number of watchers is only displayed if the user has [`connect-events` capability](/chat/docs/javascript/channel_capabilities/)\n */\n@Component({\n  selector: 'stream-channel-header',\n  templateUrl: './channel-header.component.html',\n  styles: [],\n})\nexport class ChannelHeaderComponent implements OnInit, OnDestroy {\n  channelActionsTemplate?: TemplateRef<ChannelActionsContext>;\n  channelHeaderInfoTemplate?: TemplateRef<ChannelHeaderInfoContext>;\n  activeChannel: Channel<DefaultStreamChatGenerics> | undefined;\n  canReceiveConnectEvents: boolean | undefined;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private channelService: ChannelService,\n    private customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef,\n    private chatClientService: ChatClientService,\n  ) {\n    this.channelService.activeChannel$.subscribe((c) => {\n      this.activeChannel = c;\n      const capabilities = this.activeChannel?.data\n        ?.own_capabilities as string[];\n      if (!capabilities) {\n        return;\n      }\n      this.canReceiveConnectEvents =\n        capabilities.indexOf('connect-events') !== -1;\n    });\n  }\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.customTemplatesService.channelActionsTemplate$.subscribe(\n        (template) => {\n          this.channelActionsTemplate = template;\n          this.cdRef.detectChanges();\n        },\n      ),\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.channelHeaderInfoTemplate$.subscribe(\n        (template) => {\n          this.channelHeaderInfoTemplate = template;\n          this.cdRef.detectChanges();\n        },\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  getChannelActionsContext(): ChannelActionsContext {\n    return { channel: this.activeChannel! };\n  }\n\n  getChannelInfoContext(): ChannelHeaderInfoContext {\n    return { channel: this.activeChannel! };\n  }\n\n  get memberCountParam() {\n    return { memberCount: this.activeChannel?.data?.member_count || 0 };\n  }\n\n  get watcherCountParam() {\n    return { watcherCount: this.activeChannel?.state?.watcher_count || 0 };\n  }\n\n  get displayText() {\n    if (!this.activeChannel) {\n      return '';\n    }\n    return getChannelDisplayText(\n      this.activeChannel,\n      this.chatClientService.chatClient.user!,\n    );\n  }\n\n  get avatarName() {\n    return this.activeChannel?.data?.name;\n  }\n}\n","<div class=\"str-chat__header-livestream str-chat__channel-header\">\n  <ng-content />\n  <stream-avatar-placeholder\n    type=\"channel\"\n    location=\"channel-header\"\n    imageUrl=\"{{ activeChannel?.data?.image }}\"\n    name=\"{{ avatarName }}\"\n    [channel]=\"activeChannel\"\n  />\n  <div class=\"str-chat__header-livestream-left str-chat__channel-header-end\">\n    <p\n      data-testid=\"name\"\n      class=\"str-chat__header-livestream-left--title str-chat__channel-header-title\"\n    >\n      {{ displayText }}\n    </p>\n    <ng-container\n      *ngTemplateOutlet=\"\n        channelHeaderInfoTemplate || defaultChannelInfo;\n        context: getChannelInfoContext()\n      \"\n    />\n    <ng-template #defaultChannelInfo>\n      <p\n        data-testid=\"info\"\n        class=\"str-chat__header-livestream-left--members str-chat__channel-header-info\"\n      >\n        {{'streamChat.{{ memberCount }} members' | translate:memberCountParam}}\n        {{canReceiveConnectEvents ? ('streamChat.{{ watcherCount }} online' |\n        translate:watcherCountParam) : ''}}\n      </p>\n    </ng-template>\n  </div>\n  @if (channelActionsTemplate) {\n    <ng-container\n      *ngTemplateOutlet=\"\n        channelActionsTemplate;\n        context: getChannelActionsContext()\n      \"\n    />\n  }\n</div>\n"]}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { map } from 'rxjs/operators';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../channel.service";
|
|
5
|
+
import * as i2 from "../custom-templates.service";
|
|
6
|
+
import * as i3 from "../theme.service";
|
|
7
|
+
import * as i4 from "@angular/common";
|
|
8
|
+
import * as i5 from "../icon/icon.component";
|
|
9
|
+
import * as i6 from "../channel-preview/channel-preview.component";
|
|
10
|
+
import * as i7 from "../paginated-list/paginated-list.component";
|
|
11
|
+
import * as i8 from "@ngx-translate/core";
|
|
12
|
+
/**
|
|
13
|
+
* The `ChannelList` component renders the list of channels.
|
|
14
|
+
*/
|
|
15
|
+
export class ChannelListComponent {
|
|
16
|
+
constructor(channelService, customTemplatesService, themeService) {
|
|
17
|
+
this.channelService = channelService;
|
|
18
|
+
this.customTemplatesService = customTemplatesService;
|
|
19
|
+
this.themeService = themeService;
|
|
20
|
+
this.isLoadingMoreChannels = false;
|
|
21
|
+
this.subscriptions = [];
|
|
22
|
+
this.theme$ = this.themeService.theme$;
|
|
23
|
+
this.channels$ = this.channelService.channels$;
|
|
24
|
+
this.hasMoreChannels$ = this.channelService.hasMoreChannels$;
|
|
25
|
+
this.isError$ = this.channelService.shouldRecoverState$;
|
|
26
|
+
this.isInitializing$ = this.channelService.channelQueryState$.pipe(map((s) => !this.isLoadingMoreChannels && s?.state === 'in-progress'));
|
|
27
|
+
this.subscriptions.push(this.customTemplatesService.channelPreviewTemplate$.subscribe((template) => (this.customChannelPreviewTemplate = template)));
|
|
28
|
+
}
|
|
29
|
+
ngOnDestroy() {
|
|
30
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
31
|
+
}
|
|
32
|
+
async loadMoreChannels() {
|
|
33
|
+
this.isLoadingMoreChannels = true;
|
|
34
|
+
await this.channelService.loadMoreChannels();
|
|
35
|
+
this.isLoadingMoreChannels = false;
|
|
36
|
+
}
|
|
37
|
+
recoverState() {
|
|
38
|
+
void this.channelService.recoverState();
|
|
39
|
+
}
|
|
40
|
+
trackByChannelId(_, item) {
|
|
41
|
+
return item.cid;
|
|
42
|
+
}
|
|
43
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelListComponent, deps: [{ token: i1.ChannelService }, { token: i2.CustomTemplatesService }, { token: i3.ThemeService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
44
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: ChannelListComponent, selector: "stream-channel-list", ngImport: i0, template: "<div\n #container\n data-testid=\"channel-list-container\"\n class=\"str-chat str-chat-angular__channel-list str-chat__channel-list str-chat-channel-list messaging str-chat__theme-{{\n theme$ | async\n }}\"\n>\n @if ((isError$ | async) === false && (isInitializing$ | async) === false) {\n <div class=\"str-chat__channel-list-messenger\">\n <div class=\"str-chat__channel-list-messenger__main\">\n <ng-content select=\"[channel-list-top]\" />\n @if (!(channels$ | async)?.length) {\n <div class=\"str-chat__channel-list-empty\">\n <stream-icon icon=\"chat-bubble\" />\n <p data-testid=\"empty-channel-list-indicator\">\n {{ \"streamChat.You have no channels currently\" | translate }}\n </p>\n </div>\n }\n <stream-paginated-list\n [items]=\"(channels$ | async) ?? []\"\n [hasMore]=\"(hasMoreChannels$ | async) ?? false\"\n [isLoading]=\"isLoadingMoreChannels\"\n (loadMore)=\"loadMoreChannels()\"\n [trackBy]=\"trackByChannelId\"\n >\n <ng-template let-channel=\"item\">\n <ng-template #defaultTemplate let-channelInput=\"channel\">\n <stream-channel-preview\n data-testclass=\"channel-preview\"\n [channel]=\"channelInput\"\n />\n </ng-template>\n <div>\n <ng-container\n *ngTemplateOutlet=\"\n customChannelPreviewTemplate || defaultTemplate;\n context: { channel: channel }\n \"\n />\n </div>\n </ng-template>\n </stream-paginated-list>\n <ng-content select=\"[channel-list-bottom]\" />\n </div>\n </div>\n } @else {\n @if (isError$ | async) {\n <div\n data-testid=\"chatdown-container\"\n class=\"str-chat__dow str-chat__channel-list-empty\"\n >\n <button (click)=\"recoverState()\" class=\"str-chat__cta-button\">\n {{ \"streamChat.Reload channels\" | translate }}\n </button>\n </div>\n }\n @if (isInitializing$ | async) {\n <ng-container *ngTemplateOutlet=\"loadingChannels\" />\n }\n }\n</div>\n\n<ng-template #loadingChannels>\n <div\n data-testid=\"loading-indicator-full-size\"\n class=\"str-chat__loading-channels\"\n >\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n </div>\n</ng-template>\n\n<ng-template #loadingChannel>\n <div\n class=\"str-chat__loading-channels-item str-chat__channel-preview-loading\"\n >\n <div class=\"str-chat__loading-channels-avatar\"></div>\n <div\n class=\"str-chat__loading-channels-meta str-chat__channel-preview-end-loading\"\n >\n <div class=\"str-chat__loading-channels-username\"></div>\n <div class=\"str-chat__loading-channels-status\"></div>\n </div>\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i5.IconComponent, selector: "stream-icon", inputs: ["icon"] }, { kind: "component", type: i6.ChannelPreviewComponent, selector: "stream-channel-preview", inputs: ["channel"] }, { kind: "component", type: i7.PaginatedListComponent, selector: "stream-paginated-list", inputs: ["items", "isLoading", "hasMore", "trackBy"], outputs: ["loadMore"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }] }); }
|
|
45
|
+
}
|
|
46
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelListComponent, decorators: [{
|
|
47
|
+
type: Component,
|
|
48
|
+
args: [{ selector: 'stream-channel-list', template: "<div\n #container\n data-testid=\"channel-list-container\"\n class=\"str-chat str-chat-angular__channel-list str-chat__channel-list str-chat-channel-list messaging str-chat__theme-{{\n theme$ | async\n }}\"\n>\n @if ((isError$ | async) === false && (isInitializing$ | async) === false) {\n <div class=\"str-chat__channel-list-messenger\">\n <div class=\"str-chat__channel-list-messenger__main\">\n <ng-content select=\"[channel-list-top]\" />\n @if (!(channels$ | async)?.length) {\n <div class=\"str-chat__channel-list-empty\">\n <stream-icon icon=\"chat-bubble\" />\n <p data-testid=\"empty-channel-list-indicator\">\n {{ \"streamChat.You have no channels currently\" | translate }}\n </p>\n </div>\n }\n <stream-paginated-list\n [items]=\"(channels$ | async) ?? []\"\n [hasMore]=\"(hasMoreChannels$ | async) ?? false\"\n [isLoading]=\"isLoadingMoreChannels\"\n (loadMore)=\"loadMoreChannels()\"\n [trackBy]=\"trackByChannelId\"\n >\n <ng-template let-channel=\"item\">\n <ng-template #defaultTemplate let-channelInput=\"channel\">\n <stream-channel-preview\n data-testclass=\"channel-preview\"\n [channel]=\"channelInput\"\n />\n </ng-template>\n <div>\n <ng-container\n *ngTemplateOutlet=\"\n customChannelPreviewTemplate || defaultTemplate;\n context: { channel: channel }\n \"\n />\n </div>\n </ng-template>\n </stream-paginated-list>\n <ng-content select=\"[channel-list-bottom]\" />\n </div>\n </div>\n } @else {\n @if (isError$ | async) {\n <div\n data-testid=\"chatdown-container\"\n class=\"str-chat__dow str-chat__channel-list-empty\"\n >\n <button (click)=\"recoverState()\" class=\"str-chat__cta-button\">\n {{ \"streamChat.Reload channels\" | translate }}\n </button>\n </div>\n }\n @if (isInitializing$ | async) {\n <ng-container *ngTemplateOutlet=\"loadingChannels\" />\n }\n }\n</div>\n\n<ng-template #loadingChannels>\n <div\n data-testid=\"loading-indicator-full-size\"\n class=\"str-chat__loading-channels\"\n >\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n </div>\n</ng-template>\n\n<ng-template #loadingChannel>\n <div\n class=\"str-chat__loading-channels-item str-chat__channel-preview-loading\"\n >\n <div class=\"str-chat__loading-channels-avatar\"></div>\n <div\n class=\"str-chat__loading-channels-meta str-chat__channel-preview-end-loading\"\n >\n <div class=\"str-chat__loading-channels-username\"></div>\n <div class=\"str-chat__loading-channels-status\"></div>\n </div>\n </div>\n</ng-template>\n" }]
|
|
49
|
+
}], ctorParameters: () => [{ type: i1.ChannelService }, { type: i2.CustomTemplatesService }, { type: i3.ThemeService }] });
|
|
50
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/channel-list/channel-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/channel-list/channel-list.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAA0B,MAAM,eAAe,CAAC;AAElE,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;;;AAOrC;;GAEG;AAMH,MAAM,OAAO,oBAAoB;IAU/B,YACU,cAA8B,EAC9B,sBAA8C,EAC9C,YAA0B;QAF1B,mBAAc,GAAd,cAAc,CAAgB;QAC9B,2BAAsB,GAAtB,sBAAsB,CAAwB;QAC9C,iBAAY,GAAZ,YAAY,CAAc;QATpC,0BAAqB,GAAG,KAAK,CAAC;QAI9B,kBAAa,GAAmB,EAAE,CAAC;QAOjC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC;QACvC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC;QAC/C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC;QAC7D,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,mBAAmB,CAAC;QACxD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,IAAI,CAChE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,IAAI,CAAC,EAAE,KAAK,KAAK,aAAa,CAAC,CACtE,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,sBAAsB,CAAC,uBAAuB,CAAC,SAAS,CAC3D,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,4BAA4B,GAAG,QAAQ,CAAC,CAC7D,CACF,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,gBAAgB;QACpB,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;QAClC,MAAM,IAAI,CAAC,cAAc,CAAC,gBAAgB,EAAE,CAAC;QAC7C,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;IACrC,CAAC;IAED,YAAY;QACV,KAAK,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;IAC1C,CAAC;IAED,gBAAgB,CAAC,CAAS,EAAE,IAAwC;QAClE,OAAO,IAAI,CAAC,GAAG,CAAC;IAClB,CAAC;8GA7CU,oBAAoB;kGAApB,oBAAoB,2DCjBjC,0/FAuFA;;2FDtEa,oBAAoB;kBALhC,SAAS;+BACE,qBAAqB","sourcesContent":["import { Component, OnDestroy, TemplateRef } from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { Channel } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { ThemeService } from '../theme.service';\nimport { ChannelPreviewContext, DefaultStreamChatGenerics } from '../types';\n\n/**\n * The `ChannelList` component renders the list of channels.\n */\n@Component({\n  selector: 'stream-channel-list',\n  templateUrl: './channel-list.component.html',\n  styles: [],\n})\nexport class ChannelListComponent implements OnDestroy {\n  channels$: Observable<Channel<DefaultStreamChatGenerics>[] | undefined>;\n  isError$: Observable<boolean>;\n  isInitializing$: Observable<boolean>;\n  isLoadingMoreChannels = false;\n  hasMoreChannels$: Observable<boolean>;\n  customChannelPreviewTemplate: TemplateRef<ChannelPreviewContext> | undefined;\n  theme$: Observable<string>;\n  subscriptions: Subscription[] = [];\n\n  constructor(\n    private channelService: ChannelService,\n    private customTemplatesService: CustomTemplatesService,\n    private themeService: ThemeService,\n  ) {\n    this.theme$ = this.themeService.theme$;\n    this.channels$ = this.channelService.channels$;\n    this.hasMoreChannels$ = this.channelService.hasMoreChannels$;\n    this.isError$ = this.channelService.shouldRecoverState$;\n    this.isInitializing$ = this.channelService.channelQueryState$.pipe(\n      map((s) => !this.isLoadingMoreChannels && s?.state === 'in-progress'),\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.channelPreviewTemplate$.subscribe(\n        (template) => (this.customChannelPreviewTemplate = template),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async loadMoreChannels() {\n    this.isLoadingMoreChannels = true;\n    await this.channelService.loadMoreChannels();\n    this.isLoadingMoreChannels = false;\n  }\n\n  recoverState() {\n    void this.channelService.recoverState();\n  }\n\n  trackByChannelId(_: number, item: Channel<DefaultStreamChatGenerics>) {\n    return item.cid;\n  }\n}\n","<div\n  #container\n  data-testid=\"channel-list-container\"\n  class=\"str-chat str-chat-angular__channel-list str-chat__channel-list str-chat-channel-list messaging str-chat__theme-{{\n    theme$ | async\n  }}\"\n>\n  @if ((isError$ | async) === false && (isInitializing$ | async) === false) {\n    <div class=\"str-chat__channel-list-messenger\">\n      <div class=\"str-chat__channel-list-messenger__main\">\n        <ng-content select=\"[channel-list-top]\" />\n        @if (!(channels$ | async)?.length) {\n          <div class=\"str-chat__channel-list-empty\">\n            <stream-icon icon=\"chat-bubble\" />\n            <p data-testid=\"empty-channel-list-indicator\">\n              {{ \"streamChat.You have no channels currently\" | translate }}\n            </p>\n          </div>\n        }\n        <stream-paginated-list\n          [items]=\"(channels$ | async) ?? []\"\n          [hasMore]=\"(hasMoreChannels$ | async) ?? false\"\n          [isLoading]=\"isLoadingMoreChannels\"\n          (loadMore)=\"loadMoreChannels()\"\n          [trackBy]=\"trackByChannelId\"\n        >\n          <ng-template let-channel=\"item\">\n            <ng-template #defaultTemplate let-channelInput=\"channel\">\n              <stream-channel-preview\n                data-testclass=\"channel-preview\"\n                [channel]=\"channelInput\"\n              />\n            </ng-template>\n            <div>\n              <ng-container\n                *ngTemplateOutlet=\"\n                  customChannelPreviewTemplate || defaultTemplate;\n                  context: { channel: channel }\n                \"\n              />\n            </div>\n          </ng-template>\n        </stream-paginated-list>\n        <ng-content select=\"[channel-list-bottom]\" />\n      </div>\n    </div>\n  } @else {\n    @if (isError$ | async) {\n      <div\n        data-testid=\"chatdown-container\"\n        class=\"str-chat__dow str-chat__channel-list-empty\"\n      >\n        <button (click)=\"recoverState()\" class=\"str-chat__cta-button\">\n          {{ \"streamChat.Reload channels\" | translate }}\n        </button>\n      </div>\n    }\n    @if (isInitializing$ | async) {\n      <ng-container *ngTemplateOutlet=\"loadingChannels\" />\n    }\n  }\n</div>\n\n<ng-template #loadingChannels>\n  <div\n    data-testid=\"loading-indicator-full-size\"\n    class=\"str-chat__loading-channels\"\n  >\n    <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n    <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n    <ng-container *ngTemplateOutlet=\"loadingChannel\" />\n  </div>\n</ng-template>\n\n<ng-template #loadingChannel>\n  <div\n    class=\"str-chat__loading-channels-item str-chat__channel-preview-loading\"\n  >\n    <div class=\"str-chat__loading-channels-avatar\"></div>\n    <div\n      class=\"str-chat__loading-channels-meta str-chat__channel-preview-end-loading\"\n    >\n      <div class=\"str-chat__loading-channels-username\"></div>\n      <div class=\"str-chat__loading-channels-status\"></div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { filter } from 'rxjs/operators';
|
|
3
|
+
import { getChannelDisplayText } from '../get-channel-display-text';
|
|
4
|
+
import { getMessageTranslation } from '../get-message-translation';
|
|
5
|
+
import { getReadBy } from '../read-by';
|
|
6
|
+
import { isOnSeparateDate } from '../is-on-separate-date';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "../channel.service";
|
|
9
|
+
import * as i2 from "../chat-client.service";
|
|
10
|
+
import * as i3 from "../message.service";
|
|
11
|
+
import * as i4 from "../custom-templates.service";
|
|
12
|
+
import * as i5 from "../date-parser.service";
|
|
13
|
+
import * as i6 from "@angular/common";
|
|
14
|
+
import * as i7 from "../avatar-placeholder/avatar-placeholder.component";
|
|
15
|
+
import * as i8 from "../icon/icon-placeholder/icon-placeholder.component";
|
|
16
|
+
import * as i9 from "@ngx-translate/core";
|
|
17
|
+
/**
|
|
18
|
+
* The `ChannelPreview` component displays a channel preview in the channel list, it consists of the image, name and latest message of the channel.
|
|
19
|
+
*/
|
|
20
|
+
export class ChannelPreviewComponent {
|
|
21
|
+
constructor(channelService, chatClientService, messageService, customTemplatesService, dateParser) {
|
|
22
|
+
this.channelService = channelService;
|
|
23
|
+
this.chatClientService = chatClientService;
|
|
24
|
+
this.customTemplatesService = customTemplatesService;
|
|
25
|
+
this.dateParser = dateParser;
|
|
26
|
+
this.isActive = false;
|
|
27
|
+
this.isUnreadMessageWasCalled = false;
|
|
28
|
+
this.isUnread = false;
|
|
29
|
+
this.latestMessageText = 'streamChat.Nothing yet...';
|
|
30
|
+
this.subscriptions = [];
|
|
31
|
+
this.canSendReadEvents = true;
|
|
32
|
+
this.displayAs = messageService.displayAs;
|
|
33
|
+
}
|
|
34
|
+
ngOnInit() {
|
|
35
|
+
this.subscriptions.push(this.chatClientService.user$.subscribe((user) => {
|
|
36
|
+
if (user?.id !== this.userId) {
|
|
37
|
+
this.userId = user?.id;
|
|
38
|
+
}
|
|
39
|
+
}));
|
|
40
|
+
this.subscriptions.push(this.channelService.activeChannel$.subscribe((activeChannel) => (this.isActive = activeChannel?.id === this.channel?.id)));
|
|
41
|
+
const messages = this.channel?.state?.latestMessages;
|
|
42
|
+
if (messages && messages.length > 0) {
|
|
43
|
+
this.setLatestMessage(messages[messages.length - 1]);
|
|
44
|
+
}
|
|
45
|
+
this.updateUnreadState();
|
|
46
|
+
const capabilities = this.channel?.data?.own_capabilities || [];
|
|
47
|
+
this.canSendReadEvents = capabilities.indexOf('read-events') !== -1;
|
|
48
|
+
this.subscriptions.push(this.channel.on('message.new', this.handleMessageEvent.bind(this)));
|
|
49
|
+
this.subscriptions.push(this.channel.on('message.updated', this.handleMessageEvent.bind(this)));
|
|
50
|
+
this.subscriptions.push(this.channel.on('message.deleted', this.handleMessageEvent.bind(this)));
|
|
51
|
+
this.subscriptions.push(this.channel.on('channel.truncated', this.handleMessageEvent.bind(this)));
|
|
52
|
+
this.subscriptions.push(this.channel.on('message.read', () => {
|
|
53
|
+
this.isUnreadMessageWasCalled = false;
|
|
54
|
+
this.updateUnreadState();
|
|
55
|
+
}));
|
|
56
|
+
this.subscriptions.push(this.chatClientService.events$
|
|
57
|
+
.pipe(filter((e) => e.eventType === 'notification.mark_unread' &&
|
|
58
|
+
this.channel.id === e.event?.channel_id))
|
|
59
|
+
.subscribe(() => {
|
|
60
|
+
this.isUnreadMessageWasCalled = true;
|
|
61
|
+
this.updateUnreadState();
|
|
62
|
+
}));
|
|
63
|
+
}
|
|
64
|
+
ngOnDestroy() {
|
|
65
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
66
|
+
}
|
|
67
|
+
get avatarImage() {
|
|
68
|
+
return this.channel?.data?.image;
|
|
69
|
+
}
|
|
70
|
+
get avatarName() {
|
|
71
|
+
return this.channel?.data?.name;
|
|
72
|
+
}
|
|
73
|
+
get title() {
|
|
74
|
+
if (!this.channel) {
|
|
75
|
+
return '';
|
|
76
|
+
}
|
|
77
|
+
return getChannelDisplayText(this.channel, this.chatClientService.chatClient.user);
|
|
78
|
+
}
|
|
79
|
+
setAsActiveChannel() {
|
|
80
|
+
void this.channelService.setAsActiveChannel(this.channel);
|
|
81
|
+
}
|
|
82
|
+
handleMessageEvent(event) {
|
|
83
|
+
if (this.channel?.state.latestMessages.length === 0) {
|
|
84
|
+
this.latestMessage = undefined;
|
|
85
|
+
this.latestMessageStatus = undefined;
|
|
86
|
+
this.latestMessageText = 'streamChat.Nothing yet...';
|
|
87
|
+
this.latestMessageTime = undefined;
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
const latestMessage = this.channel?.state.latestMessages[this.channel?.state.latestMessages.length - 1];
|
|
91
|
+
if (!event.message || latestMessage?.id !== event.message.id) {
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
this.setLatestMessage(latestMessage);
|
|
95
|
+
this.updateUnreadState();
|
|
96
|
+
}
|
|
97
|
+
setLatestMessage(message) {
|
|
98
|
+
this.latestMessage = message;
|
|
99
|
+
if (message?.deleted_at) {
|
|
100
|
+
this.latestMessageText = 'streamChat.Message deleted';
|
|
101
|
+
}
|
|
102
|
+
else if (message?.text) {
|
|
103
|
+
this.latestMessageText =
|
|
104
|
+
getMessageTranslation(message, this.channel, this.chatClientService.chatClient.user) || message.text;
|
|
105
|
+
}
|
|
106
|
+
else if (message?.attachments && message.attachments.length) {
|
|
107
|
+
this.latestMessageText = 'streamChat.🏙 Attachment...';
|
|
108
|
+
}
|
|
109
|
+
if (this.latestMessage && this.latestMessage.type === 'regular') {
|
|
110
|
+
this.latestMessageTime = isOnSeparateDate(new Date(), this.latestMessage.created_at)
|
|
111
|
+
? this.dateParser.parseDate(this.latestMessage.created_at)
|
|
112
|
+
: this.dateParser.parseTime(this.latestMessage.created_at);
|
|
113
|
+
}
|
|
114
|
+
else {
|
|
115
|
+
this.latestMessageTime = undefined;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
updateUnreadState() {
|
|
119
|
+
if (this.channel &&
|
|
120
|
+
this.latestMessage &&
|
|
121
|
+
this.latestMessage.user?.id === this.userId &&
|
|
122
|
+
this.latestMessage.status === 'received' &&
|
|
123
|
+
this.latestMessage.type === 'regular') {
|
|
124
|
+
this.latestMessageStatus =
|
|
125
|
+
getReadBy(this.latestMessage, this.channel).length > 0
|
|
126
|
+
? 'read'
|
|
127
|
+
: 'delivered';
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
this.latestMessageStatus = undefined;
|
|
131
|
+
}
|
|
132
|
+
if ((this.isActive && !this.isUnreadMessageWasCalled) ||
|
|
133
|
+
!this.canSendReadEvents) {
|
|
134
|
+
this.unreadCount = 0;
|
|
135
|
+
this.isUnread = false;
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
this.unreadCount = this.channel.countUnread();
|
|
139
|
+
this.isUnread = !!this.unreadCount;
|
|
140
|
+
}
|
|
141
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelPreviewComponent, deps: [{ token: i1.ChannelService }, { token: i2.ChatClientService }, { token: i3.MessageService }, { token: i4.CustomTemplatesService }, { token: i5.DateParserService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
142
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: ChannelPreviewComponent, selector: "stream-channel-preview", inputs: { channel: "channel" }, ngImport: i0, template: "<button\n class=\"str-chat__channel-preview-messenger str-chat__channel-preview\"\n data-testid=\"channel-preview-container\"\n [class.str-chat__channel-preview-messenger--active]=\"isActive\"\n [class.str-chat__channel-preview--active]=\"isActive\"\n [class.str-chat__channel-preview-messenger--unread]=\"isUnread\"\n (click)=\"setAsActiveChannel()\"\n>\n <div class=\"str-chat__channel-preview-messenger--left\">\n <stream-avatar-placeholder\n type=\"channel\"\n location=\"channel-preview\"\n name=\"{{ avatarName }}\"\n imageUrl=\"{{ avatarImage }}\"\n [channel]=\"channel\"\n />\n </div>\n <div\n class=\"str-chat__channel-preview-messenger--right str-chat__channel-preview-end\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.channelPreviewInfoTemplate$ | async) ||\n defaultChannelInfo;\n context: {\n channelDisplayTitle: title,\n channel: channel,\n unreadCount: unreadCount,\n latestMessageText: latestMessageText,\n latestMessageStatus: latestMessageStatus,\n latestMessageTime: latestMessageTime,\n latestMessage: latestMessage,\n }\n \"\n />\n <ng-template\n #defaultChannelInfo\n let-channelDisplayTitle=\"channelDisplayTitle\"\n let-unreadCount=\"unreadCount\"\n let-latestMessageText=\"latestMessageText\"\n let-latestMessageStatus=\"latestMessageStatus\"\n let-latestMessageTime=\"latestMessageTime\"\n >\n <div class=\"str-chat__channel-preview-end-first-row\">\n <div class=\"str-chat__channel-preview-messenger--name\">\n <span data-testid=\"channel-preview-title\">{{\n channelDisplayTitle\n }}</span>\n </div>\n @if (unreadCount) {\n <div\n data-testid=\"unread-badge\"\n class=\"str-chat__channel-preview-unread-badge\"\n >\n {{ unreadCount }}\n </div>\n }\n </div>\n <div class=\"str-chat__channel-preview-end-second-row\">\n <div\n data-testid=\"latest-message\"\n class=\"str-chat__channel-preview-messenger--last-message\"\n >\n @if (displayAs === \"text\") {\n {{ latestMessageText | translate }}\n } @else {\n <span\n data-testid=\"html-content\"\n [innerHTML]=\"latestMessageText | translate\"\n ></span>\n }\n </div>\n @if (latestMessageStatus) {\n <div\n data-testid=\"latest-message-status\"\n class=\"str-chat__channel-preview-messenger--status str-chat__channel-preview-messenger--status-{{\n latestMessageStatus\n }}\"\n >\n <stream-icon-placeholder\n [icon]=\"\n latestMessageStatus === 'delivered' ? 'delivered' : 'read'\n \"\n />\n </div>\n }\n @if (latestMessageTime) {\n <div\n data-testid=\"latest-message-time\"\n class=\"str-chat__channel-preview-messenger--time\"\n >\n {{ latestMessageTime }}\n </div>\n }\n </div>\n </ng-template>\n </div>\n</button>\n", dependencies: [{ kind: "directive", type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i7.AvatarPlaceholderComponent, selector: "stream-avatar-placeholder", inputs: ["name", "imageUrl", "location", "channel", "user", "type", "initialsType", "showOnlineIndicator"] }, { kind: "component", type: i8.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }, { kind: "pipe", type: i6.AsyncPipe, name: "async" }, { kind: "pipe", type: i9.TranslatePipe, name: "translate" }] }); }
|
|
143
|
+
}
|
|
144
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ChannelPreviewComponent, decorators: [{
|
|
145
|
+
type: Component,
|
|
146
|
+
args: [{ selector: 'stream-channel-preview', template: "<button\n class=\"str-chat__channel-preview-messenger str-chat__channel-preview\"\n data-testid=\"channel-preview-container\"\n [class.str-chat__channel-preview-messenger--active]=\"isActive\"\n [class.str-chat__channel-preview--active]=\"isActive\"\n [class.str-chat__channel-preview-messenger--unread]=\"isUnread\"\n (click)=\"setAsActiveChannel()\"\n>\n <div class=\"str-chat__channel-preview-messenger--left\">\n <stream-avatar-placeholder\n type=\"channel\"\n location=\"channel-preview\"\n name=\"{{ avatarName }}\"\n imageUrl=\"{{ avatarImage }}\"\n [channel]=\"channel\"\n />\n </div>\n <div\n class=\"str-chat__channel-preview-messenger--right str-chat__channel-preview-end\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.channelPreviewInfoTemplate$ | async) ||\n defaultChannelInfo;\n context: {\n channelDisplayTitle: title,\n channel: channel,\n unreadCount: unreadCount,\n latestMessageText: latestMessageText,\n latestMessageStatus: latestMessageStatus,\n latestMessageTime: latestMessageTime,\n latestMessage: latestMessage,\n }\n \"\n />\n <ng-template\n #defaultChannelInfo\n let-channelDisplayTitle=\"channelDisplayTitle\"\n let-unreadCount=\"unreadCount\"\n let-latestMessageText=\"latestMessageText\"\n let-latestMessageStatus=\"latestMessageStatus\"\n let-latestMessageTime=\"latestMessageTime\"\n >\n <div class=\"str-chat__channel-preview-end-first-row\">\n <div class=\"str-chat__channel-preview-messenger--name\">\n <span data-testid=\"channel-preview-title\">{{\n channelDisplayTitle\n }}</span>\n </div>\n @if (unreadCount) {\n <div\n data-testid=\"unread-badge\"\n class=\"str-chat__channel-preview-unread-badge\"\n >\n {{ unreadCount }}\n </div>\n }\n </div>\n <div class=\"str-chat__channel-preview-end-second-row\">\n <div\n data-testid=\"latest-message\"\n class=\"str-chat__channel-preview-messenger--last-message\"\n >\n @if (displayAs === \"text\") {\n {{ latestMessageText | translate }}\n } @else {\n <span\n data-testid=\"html-content\"\n [innerHTML]=\"latestMessageText | translate\"\n ></span>\n }\n </div>\n @if (latestMessageStatus) {\n <div\n data-testid=\"latest-message-status\"\n class=\"str-chat__channel-preview-messenger--status str-chat__channel-preview-messenger--status-{{\n latestMessageStatus\n }}\"\n >\n <stream-icon-placeholder\n [icon]=\"\n latestMessageStatus === 'delivered' ? 'delivered' : 'read'\n \"\n />\n </div>\n }\n @if (latestMessageTime) {\n <div\n data-testid=\"latest-message-time\"\n class=\"str-chat__channel-preview-messenger--time\"\n >\n {{ latestMessageTime }}\n </div>\n }\n </div>\n </ng-template>\n </div>\n</button>\n" }]
|
|
147
|
+
}], ctorParameters: () => [{ type: i1.ChannelService }, { type: i2.ChatClientService }, { type: i3.MessageService }, { type: i4.CustomTemplatesService }, { type: i5.DateParserService }], propDecorators: { channel: [{
|
|
148
|
+
type: Input
|
|
149
|
+
}] } });
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel-preview.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.ts","../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnDestroy, OnInit } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { filter } from 'rxjs/operators';\nimport { Channel, Event, FormatMessageResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { getChannelDisplayText } from '../get-channel-display-text';\nimport { DefaultStreamChatGenerics } from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getMessageTranslation } from '../get-message-translation';\nimport { MessageService } from '../message.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { getReadBy } from '../read-by';\nimport { isOnSeparateDate } from '../is-on-separate-date';\nimport { DateParserService } from '../date-parser.service';\n\n/**\n * The `ChannelPreview` component displays a channel preview in the channel list, it consists of the image, name and latest message of the channel.\n */\n@Component({\n  selector: 'stream-channel-preview',\n  templateUrl: './channel-preview.component.html',\n  styles: [],\n})\nexport class ChannelPreviewComponent implements OnInit, OnDestroy {\n  /**\n   * The channel to be displayed\n   */\n  @Input() channel: Channel<DefaultStreamChatGenerics> | undefined;\n  isActive = false;\n  isUnreadMessageWasCalled = false;\n  isUnread = false;\n  unreadCount: number | undefined;\n  latestMessageText: string = 'streamChat.Nothing yet...';\n  latestMessageStatus?: 'delivered' | 'read';\n  latestMessageTime?: string;\n  latestMessage?: FormatMessageResponse<DefaultStreamChatGenerics>;\n  displayAs: 'text' | 'html';\n  userId?: string;\n  private subscriptions: (Subscription | { unsubscribe: () => void })[] = [];\n  private canSendReadEvents = true;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    messageService: MessageService,\n    public customTemplatesService: CustomTemplatesService,\n    private dateParser: DateParserService,\n  ) {\n    this.displayAs = messageService.displayAs;\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((user) => {\n        if (user?.id !== this.userId) {\n          this.userId = user?.id;\n        }\n      }),\n    );\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe(\n        (activeChannel) =>\n          (this.isActive = activeChannel?.id === this.channel?.id),\n      ),\n    );\n    const messages = this.channel?.state?.latestMessages;\n    if (messages && messages.length > 0) {\n      this.setLatestMessage(messages[messages.length - 1]);\n    }\n    this.updateUnreadState();\n    const capabilities =\n      (this.channel?.data?.own_capabilities as string[]) || [];\n    this.canSendReadEvents = capabilities.indexOf('read-events') !== -1;\n    this.subscriptions.push(\n      this.channel!.on('message.new', this.handleMessageEvent.bind(this)),\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.updated', this.handleMessageEvent.bind(this)),\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.deleted', this.handleMessageEvent.bind(this)),\n    );\n    this.subscriptions.push(\n      this.channel!.on('channel.truncated', this.handleMessageEvent.bind(this)),\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.read', () => {\n        this.isUnreadMessageWasCalled = false;\n        this.updateUnreadState();\n      }),\n    );\n    this.subscriptions.push(\n      this.chatClientService.events$\n        .pipe(\n          filter(\n            (e) =>\n              e.eventType === 'notification.mark_unread' &&\n              this.channel!.id === e.event?.channel_id,\n          ),\n        )\n        .subscribe(() => {\n          this.isUnreadMessageWasCalled = true;\n          this.updateUnreadState();\n        }),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get avatarImage() {\n    return this.channel?.data?.image;\n  }\n\n  get avatarName() {\n    return this.channel?.data?.name;\n  }\n\n  get title() {\n    if (!this.channel) {\n      return '';\n    }\n    return getChannelDisplayText(\n      this.channel,\n      this.chatClientService.chatClient.user!,\n    );\n  }\n\n  setAsActiveChannel(): void {\n    void this.channelService.setAsActiveChannel(this.channel!);\n  }\n\n  private handleMessageEvent(event: Event) {\n    if (this.channel?.state.latestMessages.length === 0) {\n      this.latestMessage = undefined;\n      this.latestMessageStatus = undefined;\n      this.latestMessageText = 'streamChat.Nothing yet...';\n      this.latestMessageTime = undefined;\n      return;\n    }\n    const latestMessage =\n      this.channel?.state.latestMessages[\n        this.channel?.state.latestMessages.length - 1\n      ];\n    if (!event.message || latestMessage?.id !== event.message.id) {\n      return;\n    }\n    this.setLatestMessage(latestMessage);\n    this.updateUnreadState();\n  }\n\n  private setLatestMessage(\n    message?: FormatMessageResponse<DefaultStreamChatGenerics>,\n  ) {\n    this.latestMessage = message;\n    if (message?.deleted_at) {\n      this.latestMessageText = 'streamChat.Message deleted';\n    } else if (message?.text) {\n      this.latestMessageText =\n        getMessageTranslation(\n          message,\n          this.channel,\n          this.chatClientService.chatClient.user,\n        ) || message.text;\n    } else if (message?.attachments && message.attachments.length) {\n      this.latestMessageText = 'streamChat.🏙 Attachment...';\n    }\n    if (this.latestMessage && this.latestMessage.type === 'regular') {\n      this.latestMessageTime = isOnSeparateDate(\n        new Date(),\n        this.latestMessage.created_at,\n      )\n        ? this.dateParser.parseDate(this.latestMessage.created_at)\n        : this.dateParser.parseTime(this.latestMessage.created_at);\n    } else {\n      this.latestMessageTime = undefined;\n    }\n  }\n\n  private updateUnreadState() {\n    if (\n      this.channel &&\n      this.latestMessage &&\n      this.latestMessage.user?.id === this.userId &&\n      this.latestMessage.status === 'received' &&\n      this.latestMessage.type === 'regular'\n    ) {\n      this.latestMessageStatus =\n        getReadBy(this.latestMessage, this.channel).length > 0\n          ? 'read'\n          : 'delivered';\n    } else {\n      this.latestMessageStatus = undefined;\n    }\n    if (\n      (this.isActive && !this.isUnreadMessageWasCalled) ||\n      !this.canSendReadEvents\n    ) {\n      this.unreadCount = 0;\n      this.isUnread = false;\n      return;\n    }\n    this.unreadCount = this.channel!.countUnread();\n    this.isUnread = !!this.unreadCount;\n  }\n}\n","<button\n  class=\"str-chat__channel-preview-messenger str-chat__channel-preview\"\n  data-testid=\"channel-preview-container\"\n  [class.str-chat__channel-preview-messenger--active]=\"isActive\"\n  [class.str-chat__channel-preview--active]=\"isActive\"\n  [class.str-chat__channel-preview-messenger--unread]=\"isUnread\"\n  (click)=\"setAsActiveChannel()\"\n>\n  <div class=\"str-chat__channel-preview-messenger--left\">\n    <stream-avatar-placeholder\n      type=\"channel\"\n      location=\"channel-preview\"\n      name=\"{{ avatarName }}\"\n      imageUrl=\"{{ avatarImage }}\"\n      [channel]=\"channel\"\n    />\n  </div>\n  <div\n    class=\"str-chat__channel-preview-messenger--right str-chat__channel-preview-end\"\n  >\n    <ng-container\n      *ngTemplateOutlet=\"\n        (customTemplatesService.channelPreviewInfoTemplate$ | async) ||\n          defaultChannelInfo;\n        context: {\n          channelDisplayTitle: title,\n          channel: channel,\n          unreadCount: unreadCount,\n          latestMessageText: latestMessageText,\n          latestMessageStatus: latestMessageStatus,\n          latestMessageTime: latestMessageTime,\n          latestMessage: latestMessage,\n        }\n      \"\n    />\n    <ng-template\n      #defaultChannelInfo\n      let-channelDisplayTitle=\"channelDisplayTitle\"\n      let-unreadCount=\"unreadCount\"\n      let-latestMessageText=\"latestMessageText\"\n      let-latestMessageStatus=\"latestMessageStatus\"\n      let-latestMessageTime=\"latestMessageTime\"\n    >\n      <div class=\"str-chat__channel-preview-end-first-row\">\n        <div class=\"str-chat__channel-preview-messenger--name\">\n          <span data-testid=\"channel-preview-title\">{{\n            channelDisplayTitle\n          }}</span>\n        </div>\n        @if (unreadCount) {\n          <div\n            data-testid=\"unread-badge\"\n            class=\"str-chat__channel-preview-unread-badge\"\n          >\n            {{ unreadCount }}\n          </div>\n        }\n      </div>\n      <div class=\"str-chat__channel-preview-end-second-row\">\n        <div\n          data-testid=\"latest-message\"\n          class=\"str-chat__channel-preview-messenger--last-message\"\n        >\n          @if (displayAs === \"text\") {\n            {{ latestMessageText | translate }}\n          } @else {\n            <span\n              data-testid=\"html-content\"\n              [innerHTML]=\"latestMessageText | translate\"\n            ></span>\n          }\n        </div>\n        @if (latestMessageStatus) {\n          <div\n            data-testid=\"latest-message-status\"\n            class=\"str-chat__channel-preview-messenger--status str-chat__channel-preview-messenger--status-{{\n              latestMessageStatus\n            }}\"\n          >\n            <stream-icon-placeholder\n              [icon]=\"\n                latestMessageStatus === 'delivered' ? 'delivered' : 'read'\n              \"\n            />\n          </div>\n        }\n        @if (latestMessageTime) {\n          <div\n            data-testid=\"latest-message-time\"\n            class=\"str-chat__channel-preview-messenger--time\"\n          >\n            {{ latestMessageTime }}\n          </div>\n        }\n      </div>\n    </ng-template>\n  </div>\n</button>\n"]}
|