stream-chat-angular 4.20.0 → 4.22.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 +23 -4
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/chat-client.service.js +4 -3
- package/esm2015/lib/message-list/message-list.component.js +19 -2
- package/fesm2015/stream-chat-angular.js +22 -4
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/chat-client.service.d.ts +3 -2
- package/lib/message-list/message-list.component.d.ts +7 -1
- package/package.json +1 -1
- package/src/assets/version.ts +1 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export const version = '4.
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
1
|
+
export const version = '4.22.0';
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxRQUFRLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjIyLjAnO1xuIl19
|
|
@@ -29,11 +29,12 @@ export class ChatClientService {
|
|
|
29
29
|
* @param apiKey
|
|
30
30
|
* @param userOrId
|
|
31
31
|
* @param userTokenOrProvider You can provide a token, or the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users)
|
|
32
|
+
* @param clientOptions Setting to provide to the Stream client instance
|
|
32
33
|
*/
|
|
33
|
-
init(apiKey, userOrId, userTokenOrProvider) {
|
|
34
|
+
init(apiKey, userOrId, userTokenOrProvider, clientOptions) {
|
|
34
35
|
var _a;
|
|
35
36
|
return __awaiter(this, void 0, void 0, function* () {
|
|
36
|
-
this.chatClient = StreamChat.getInstance(apiKey);
|
|
37
|
+
this.chatClient = StreamChat.getInstance(apiKey, clientOptions);
|
|
37
38
|
this.chatClient.devToken;
|
|
38
39
|
let result;
|
|
39
40
|
yield this.ngZone.runOutsideAngular(() => __awaiter(this, void 0, void 0, function* () {
|
|
@@ -162,4 +163,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
162
163
|
providedIn: 'root',
|
|
163
164
|
}]
|
|
164
165
|
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i1.NotificationService }]; } });
|
|
165
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-client.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/chat-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable, ReplaySubject } from 'rxjs';\nimport {\n  Channel,\n  ChannelFilters,\n  ChannelResponse,\n  ConnectAPIResponse,\n  OwnUserResponse,\n  UserFilters,\n  UserResponse,\n} from 'stream-chat';\nimport { AppSettings, Event, StreamChat, TokenOrProvider } from 'stream-chat';\nimport { version } from '../assets/version';\nimport { NotificationService } from './notification.service';\nimport { DefaultStreamChatGenerics } from './types';\n\nexport type ClientEvent<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  eventType: string;\n  event: Event<T>;\n};\n\n/**\n * The `ChatClient` service connects the user to the Stream chat.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class ChatClientService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * The [StreamChat client](https://github.com/GetStream/stream-chat-js/blob/master/src/client.ts) instance. In general you shouldn't need to access the client, but it's there if you want to use it.\n   */\n  chatClient!: StreamChat<T>;\n  /**\n   * Emits [`ClientEvent`](https://github.com/GetStream/stream-chat-angular/blob/master/projects/stream-chat-angular/src/lib/chat-client.service.ts) events. The platform documentation covers [the list of client, user presence and notification events](https://getstream.io/chat/docs/javascript/event_object/?language=javascript).\n   * :::important\n   * For performance reasons this Observable operates outside of the Angular change detection zone. If you subscribe to it, you need to manually reenter Angular's change detection zone, our [Change detection guide](../concepts/change-detection.mdx) explains this in detail.\n   * :::\n   */\n  events$: Observable<ClientEvent<T>>;\n  /**\n   * Emits the current [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript). Since getting the application settings is an expensive API call and we don't always need the result, this is not initialized by default, you need to call `getApplicationSettings` to load them.\n   */\n  appSettings$: Observable<AppSettings | undefined>;\n  /**\n   * Emits the current connection state of the user (`online` or `offline`)\n   */\n  connectionState$: Observable<'offline' | 'online'>;\n  /**\n   * Emits the list of pending invites of the user. It emits every pending invitation during initialization and then extends the list when a new invite is received. More information can be found in the [channel invitations](../code-examples/channel-invites.mdx) guide.\n   */\n  pendingInvites$: Observable<(ChannelResponse<T> | Channel<T>)[]>;\n  /**\n   * Emits the current chat user\n   */\n  user$: Observable<UserResponse<T> | undefined>;\n  private notificationSubject = new ReplaySubject<ClientEvent<T>>(1);\n  private connectionStateSubject = new ReplaySubject<'offline' | 'online'>(1);\n  private appSettingsSubject = new BehaviorSubject<AppSettings | undefined>(\n    undefined\n  );\n  private pendingInvitesSubject = new BehaviorSubject<\n    (ChannelResponse<T> | Channel<T>)[]\n  >([]);\n  private userSubject = new ReplaySubject<UserResponse<T> | undefined>(1);\n  private subscriptions: { unsubscribe: () => void }[] = [];\n\n  constructor(\n    private ngZone: NgZone,\n    private notificationService: NotificationService\n  ) {\n    this.events$ = this.notificationSubject.asObservable();\n    this.connectionState$ = this.connectionStateSubject.asObservable();\n    this.appSettings$ = this.appSettingsSubject.asObservable();\n    this.pendingInvites$ = this.pendingInvitesSubject.asObservable();\n    this.user$ = this.userSubject.asObservable();\n  }\n\n  /**\n   * Creates a [`StreamChat`](https://github.com/GetStream/stream-chat-js/blob/668b3e5521339f4e14fc657834531b4c8bf8176b/src/client.ts#L124) instance using the provided `apiKey`, and connects a user with the given meta data and token. More info about [connecting users](https://getstream.io/chat/docs/javascript/init_and_users/?language=javascript) can be found in the platform documentation.\n   * @param apiKey\n   * @param userOrId\n   * @param userTokenOrProvider You can provide a token, or the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users)\n   */\n  async init(\n    apiKey: string,\n    userOrId: string | OwnUserResponse<T> | UserResponse<T>,\n    userTokenOrProvider: TokenOrProvider | 'guest'\n  ): ConnectAPIResponse<T> {\n    this.chatClient = StreamChat.getInstance<T>(apiKey);\n    this.chatClient.devToken;\n    let result;\n    await this.ngZone.runOutsideAngular(async () => {\n      const user = typeof userOrId === 'string' ? { id: userOrId } : userOrId;\n      try {\n        result =\n          userTokenOrProvider === 'guest'\n            ? await this.chatClient.setGuestUser(user)\n            : await this.chatClient.connectUser(user, userTokenOrProvider);\n      } catch (error) {\n        this.notificationService.addPermanentNotification(\n          'streamChat.Error connecting to chat, refresh the page to try again.',\n          'error'\n        );\n        throw error;\n      }\n      this.userSubject.next(this.chatClient.user);\n      const sdkPrefix = 'stream-chat-angular';\n      if (!this.chatClient.getUserAgent().includes(sdkPrefix)) {\n        this.chatClient.setUserAgent(\n          `${sdkPrefix}-${version}-${this.chatClient.getUserAgent()}`\n        );\n      }\n    });\n    const channels = await this.chatClient.queryChannels(\n      { invite: 'pending' } as any as ChannelFilters<T>, // TODO: find out why we need this typecast\n      {},\n      { user_id: this.chatClient.user?.id }\n    );\n    this.pendingInvitesSubject.next(channels);\n    this.appSettingsSubject.next(undefined);\n    this.subscriptions.push(\n      this.chatClient.on((e) => {\n        this.updatePendingInvites(e);\n        this.notificationSubject.next({\n          eventType: e.type,\n          event: e,\n        });\n      })\n    );\n    let removeNotification: undefined | Function;\n    this.subscriptions.push(\n      this.chatClient.on('connection.changed', (e) => {\n        this.ngZone.run(() => {\n          const isOnline = e.online;\n          if (isOnline) {\n            if (removeNotification) {\n              removeNotification();\n            }\n          } else {\n            removeNotification =\n              this.notificationService.addPermanentNotification(\n                'streamChat.Connection failure, reconnecting now...'\n              );\n          }\n          this.connectionStateSubject.next(isOnline ? 'online' : 'offline');\n        });\n      })\n    );\n    return result;\n  }\n\n  /**\n   * Disconnects the current user, and closes the WebSocket connection. Useful when disconnecting a chat user, use in combination with [`reset`](./ChannelService.mdx/#reset).\n   */\n  async disconnectUser() {\n    this.pendingInvitesSubject.next([]);\n    await this.chatClient.disconnectUser();\n    this.userSubject.next(undefined);\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  /**\n   * Loads the current [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript), if the application settings have already been loaded, it does nothing.\n   */\n  async getAppSettings() {\n    if (this.appSettingsSubject.getValue()) {\n      return;\n    }\n    const settings = await this.chatClient.getAppSettings();\n    this.appSettingsSubject.next((settings.app as AppSettings) || {});\n  }\n\n  /**\n   * Flag the message with the given ID. If you want to know [more about flags](https://getstream.io/chat/docs/javascript/moderation/?language=javascript) check out the platform documentation.\n   * @param messageId\n   */\n  async flagMessage(messageId: string) {\n    await this.chatClient.flagMessage(messageId);\n  }\n\n  /**\n   * Searches for users in the application that have ID or name matching the provided search term\n   * @param searchTerm\n   * @returns The users matching the search\n   */\n  async autocompleteUsers(searchTerm: string) {\n    if (!searchTerm) {\n      return [];\n    }\n    const result = await this.chatClient.queryUsers({\n      $or: [\n        { id: { $autocomplete: searchTerm } },\n        { name: { $autocomplete: searchTerm } },\n      ],\n      id: { $ne: this.chatClient.userID! },\n    } as UserFilters<T>); // TODO: find out why we need this typecast\n    return result.users;\n  }\n\n  private updatePendingInvites(e: Event<T>) {\n    if (e.member?.user?.id === this.chatClient.user?.id && e.channel) {\n      const pendingInvites = this.pendingInvitesSubject.getValue();\n      if (e.type === 'notification.invited') {\n        this.pendingInvitesSubject.next([...pendingInvites, e.channel]);\n      } else if (\n        e.type === 'notification.invite_accepted' ||\n        e.type === 'notification.invite_rejected'\n      ) {\n        const index = pendingInvites.findIndex(\n          (i) => i?.cid === e.channel?.cid\n        );\n        if (index !== -1) {\n          pendingInvites.splice(index, 1);\n          this.pendingInvitesSubject.next([...pendingInvites]);\n        }\n      }\n    }\n  }\n}\n"]}
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-client.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/chat-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable, ReplaySubject } from 'rxjs';\nimport {\n  Channel,\n  ChannelFilters,\n  ChannelResponse,\n  ConnectAPIResponse,\n  OwnUserResponse,\n  StreamChatOptions,\n  UserFilters,\n  UserResponse,\n} from 'stream-chat';\nimport { AppSettings, Event, StreamChat, TokenOrProvider } from 'stream-chat';\nimport { version } from '../assets/version';\nimport { NotificationService } from './notification.service';\nimport { DefaultStreamChatGenerics } from './types';\n\nexport type ClientEvent<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  eventType: string;\n  event: Event<T>;\n};\n\n/**\n * The `ChatClient` service connects the user to the Stream chat.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class ChatClientService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * The [StreamChat client](https://github.com/GetStream/stream-chat-js/blob/master/src/client.ts) instance. In general you shouldn't need to access the client, but it's there if you want to use it.\n   */\n  chatClient!: StreamChat<T>;\n  /**\n   * Emits [`ClientEvent`](https://github.com/GetStream/stream-chat-angular/blob/master/projects/stream-chat-angular/src/lib/chat-client.service.ts) events. The platform documentation covers [the list of client, user presence and notification events](https://getstream.io/chat/docs/javascript/event_object/?language=javascript).\n   * :::important\n   * For performance reasons this Observable operates outside of the Angular change detection zone. If you subscribe to it, you need to manually reenter Angular's change detection zone, our [Change detection guide](../concepts/change-detection.mdx) explains this in detail.\n   * :::\n   */\n  events$: Observable<ClientEvent<T>>;\n  /**\n   * Emits the current [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript). Since getting the application settings is an expensive API call and we don't always need the result, this is not initialized by default, you need to call `getApplicationSettings` to load them.\n   */\n  appSettings$: Observable<AppSettings | undefined>;\n  /**\n   * Emits the current connection state of the user (`online` or `offline`)\n   */\n  connectionState$: Observable<'offline' | 'online'>;\n  /**\n   * Emits the list of pending invites of the user. It emits every pending invitation during initialization and then extends the list when a new invite is received. More information can be found in the [channel invitations](../code-examples/channel-invites.mdx) guide.\n   */\n  pendingInvites$: Observable<(ChannelResponse<T> | Channel<T>)[]>;\n  /**\n   * Emits the current chat user\n   */\n  user$: Observable<UserResponse<T> | undefined>;\n  private notificationSubject = new ReplaySubject<ClientEvent<T>>(1);\n  private connectionStateSubject = new ReplaySubject<'offline' | 'online'>(1);\n  private appSettingsSubject = new BehaviorSubject<AppSettings | undefined>(\n    undefined\n  );\n  private pendingInvitesSubject = new BehaviorSubject<\n    (ChannelResponse<T> | Channel<T>)[]\n  >([]);\n  private userSubject = new ReplaySubject<UserResponse<T> | undefined>(1);\n  private subscriptions: { unsubscribe: () => void }[] = [];\n\n  constructor(\n    private ngZone: NgZone,\n    private notificationService: NotificationService\n  ) {\n    this.events$ = this.notificationSubject.asObservable();\n    this.connectionState$ = this.connectionStateSubject.asObservable();\n    this.appSettings$ = this.appSettingsSubject.asObservable();\n    this.pendingInvites$ = this.pendingInvitesSubject.asObservable();\n    this.user$ = this.userSubject.asObservable();\n  }\n\n  /**\n   * Creates a [`StreamChat`](https://github.com/GetStream/stream-chat-js/blob/668b3e5521339f4e14fc657834531b4c8bf8176b/src/client.ts#L124) instance using the provided `apiKey`, and connects a user with the given meta data and token. More info about [connecting users](https://getstream.io/chat/docs/javascript/init_and_users/?language=javascript) can be found in the platform documentation.\n   * @param apiKey\n   * @param userOrId\n   * @param userTokenOrProvider You can provide a token, or the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users)\n   * @param clientOptions Setting to provide to the Stream client instance\n   */\n  async init(\n    apiKey: string,\n    userOrId: string | OwnUserResponse<T> | UserResponse<T>,\n    userTokenOrProvider: TokenOrProvider | 'guest',\n    clientOptions?: StreamChatOptions\n  ): ConnectAPIResponse<T> {\n    this.chatClient = StreamChat.getInstance<T>(apiKey, clientOptions);\n    this.chatClient.devToken;\n    let result;\n    await this.ngZone.runOutsideAngular(async () => {\n      const user = typeof userOrId === 'string' ? { id: userOrId } : userOrId;\n      try {\n        result =\n          userTokenOrProvider === 'guest'\n            ? await this.chatClient.setGuestUser(user)\n            : await this.chatClient.connectUser(user, userTokenOrProvider);\n      } catch (error) {\n        this.notificationService.addPermanentNotification(\n          'streamChat.Error connecting to chat, refresh the page to try again.',\n          'error'\n        );\n        throw error;\n      }\n      this.userSubject.next(this.chatClient.user);\n      const sdkPrefix = 'stream-chat-angular';\n      if (!this.chatClient.getUserAgent().includes(sdkPrefix)) {\n        this.chatClient.setUserAgent(\n          `${sdkPrefix}-${version}-${this.chatClient.getUserAgent()}`\n        );\n      }\n    });\n    const channels = await this.chatClient.queryChannels(\n      { invite: 'pending' } as any as ChannelFilters<T>, // TODO: find out why we need this typecast\n      {},\n      { user_id: this.chatClient.user?.id }\n    );\n    this.pendingInvitesSubject.next(channels);\n    this.appSettingsSubject.next(undefined);\n    this.subscriptions.push(\n      this.chatClient.on((e) => {\n        this.updatePendingInvites(e);\n        this.notificationSubject.next({\n          eventType: e.type,\n          event: e,\n        });\n      })\n    );\n    let removeNotification: undefined | Function;\n    this.subscriptions.push(\n      this.chatClient.on('connection.changed', (e) => {\n        this.ngZone.run(() => {\n          const isOnline = e.online;\n          if (isOnline) {\n            if (removeNotification) {\n              removeNotification();\n            }\n          } else {\n            removeNotification =\n              this.notificationService.addPermanentNotification(\n                'streamChat.Connection failure, reconnecting now...'\n              );\n          }\n          this.connectionStateSubject.next(isOnline ? 'online' : 'offline');\n        });\n      })\n    );\n    return result;\n  }\n\n  /**\n   * Disconnects the current user, and closes the WebSocket connection. Useful when disconnecting a chat user, use in combination with [`reset`](./ChannelService.mdx/#reset).\n   */\n  async disconnectUser() {\n    this.pendingInvitesSubject.next([]);\n    await this.chatClient.disconnectUser();\n    this.userSubject.next(undefined);\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  /**\n   * Loads the current [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript), if the application settings have already been loaded, it does nothing.\n   */\n  async getAppSettings() {\n    if (this.appSettingsSubject.getValue()) {\n      return;\n    }\n    const settings = await this.chatClient.getAppSettings();\n    this.appSettingsSubject.next((settings.app as AppSettings) || {});\n  }\n\n  /**\n   * Flag the message with the given ID. If you want to know [more about flags](https://getstream.io/chat/docs/javascript/moderation/?language=javascript) check out the platform documentation.\n   * @param messageId\n   */\n  async flagMessage(messageId: string) {\n    await this.chatClient.flagMessage(messageId);\n  }\n\n  /**\n   * Searches for users in the application that have ID or name matching the provided search term\n   * @param searchTerm\n   * @returns The users matching the search\n   */\n  async autocompleteUsers(searchTerm: string) {\n    if (!searchTerm) {\n      return [];\n    }\n    const result = await this.chatClient.queryUsers({\n      $or: [\n        { id: { $autocomplete: searchTerm } },\n        { name: { $autocomplete: searchTerm } },\n      ],\n      id: { $ne: this.chatClient.userID! },\n    } as UserFilters<T>); // TODO: find out why we need this typecast\n    return result.users;\n  }\n\n  private updatePendingInvites(e: Event<T>) {\n    if (e.member?.user?.id === this.chatClient.user?.id && e.channel) {\n      const pendingInvites = this.pendingInvitesSubject.getValue();\n      if (e.type === 'notification.invited') {\n        this.pendingInvitesSubject.next([...pendingInvites, e.channel]);\n      } else if (\n        e.type === 'notification.invite_accepted' ||\n        e.type === 'notification.invite_rejected'\n      ) {\n        const index = pendingInvites.findIndex(\n          (i) => i?.cid === e.channel?.cid\n        );\n        if (index !== -1) {\n          pendingInvites.splice(index, 1);\n          this.pendingInvitesSubject.next([...pendingInvites]);\n        }\n      }\n    }\n  }\n}\n"]}
|
|
@@ -31,11 +31,16 @@ export class MessageListComponent {
|
|
|
31
31
|
* 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.
|
|
32
32
|
*/
|
|
33
33
|
this.messageOptionsTrigger = 'message-row';
|
|
34
|
+
/**
|
|
35
|
+
* 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)
|
|
36
|
+
*/
|
|
37
|
+
this.hideJumpToLatestButtonDuringScroll = false;
|
|
34
38
|
this.enabledMessageActions = [];
|
|
35
39
|
this.class = 'str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner';
|
|
36
40
|
this.unreadMessageCount = 0;
|
|
37
41
|
this.groupStyles = [];
|
|
38
42
|
this.isLoading = false;
|
|
43
|
+
this.isScrollInProgress = false;
|
|
39
44
|
this.subscriptions = [];
|
|
40
45
|
this.isLatestMessageInList = true;
|
|
41
46
|
this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
|
|
@@ -157,6 +162,9 @@ export class MessageListComponent {
|
|
|
157
162
|
var _a;
|
|
158
163
|
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
159
164
|
(_a = this.newMessageSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
|
|
165
|
+
if (this.scrollEndTimeout) {
|
|
166
|
+
clearTimeout(this.scrollEndTimeout);
|
|
167
|
+
}
|
|
160
168
|
}
|
|
161
169
|
trackByMessageId(index, item) {
|
|
162
170
|
return item.id;
|
|
@@ -176,6 +184,13 @@ export class MessageListComponent {
|
|
|
176
184
|
this.scrollContainer.nativeElement.scrollTop = 0;
|
|
177
185
|
}
|
|
178
186
|
scrolled() {
|
|
187
|
+
this.isScrollInProgress = true;
|
|
188
|
+
if (this.scrollEndTimeout) {
|
|
189
|
+
clearTimeout(this.scrollEndTimeout);
|
|
190
|
+
}
|
|
191
|
+
this.scrollEndTimeout = setTimeout(() => {
|
|
192
|
+
this.isScrollInProgress = false;
|
|
193
|
+
}, 100);
|
|
179
194
|
if (this.scrollContainer.nativeElement.scrollHeight ===
|
|
180
195
|
this.scrollContainer.nativeElement.clientHeight) {
|
|
181
196
|
return;
|
|
@@ -348,7 +363,7 @@ export class MessageListComponent {
|
|
|
348
363
|
}
|
|
349
364
|
}
|
|
350
365
|
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 }], target: i0.ɵɵFactoryTarget.Component });
|
|
351
|
-
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" }, 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'\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\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 <li\n tabindex=\"0\"\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\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=\"isUserScrolled\"\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\">\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 >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: getMessageContext(message)\n \"\n ></ng-container>\n</ng-template>\n", components: [{ type: i4.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: i5.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i6.MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
|
|
366
|
+
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" }, 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'\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\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 <li\n tabindex=\"0\"\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\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\">\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 >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: getMessageContext(message)\n \"\n ></ng-container>\n</ng-template>\n", components: [{ type: i4.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: i5.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i6.MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
|
|
352
367
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
|
|
353
368
|
type: Component,
|
|
354
369
|
args: [{
|
|
@@ -362,6 +377,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
362
377
|
type: Input
|
|
363
378
|
}], messageOptionsTrigger: [{
|
|
364
379
|
type: Input
|
|
380
|
+
}], hideJumpToLatestButtonDuringScroll: [{
|
|
381
|
+
type: Input
|
|
365
382
|
}], class: [{
|
|
366
383
|
type: HostBinding,
|
|
367
384
|
args: ['class']
|
|
@@ -372,4 +389,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
372
389
|
type: ViewChild,
|
|
373
390
|
args: ['parentMessageElement']
|
|
374
391
|
}] } });
|
|
375
|
-
//# 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} 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';\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  messageTemplate: TemplateRef<MessageContext> | 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  @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\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private customTemplatesService: CustomTemplatesService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.resetScrollState();\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.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.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  }\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    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  getMessageContext(message: StreamMessage): MessageContext {\n    return {\n      message,\n      isLastSentMessage: !!(\n        this.lastSentMessageId && message?.id === this.lastSentMessageId\n      ),\n      enabledMessageActions: this.enabledMessageActions,\n      mode: this.mode,\n      isHighlighted: message?.id === this.highlightedMessageId,\n    };\n  }\n\n  getTypingIndicatorText(users: UserResponse[]) {\n    const text = listUsers(users);\n\n    return text;\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          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'\"\n        data-testid=\"parent-message\"\n        class=\"str-chat__parent-message-li\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage }\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      <li\n        tabindex=\"0\"\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n        id=\"{{ message.id }}\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: message }\n          \"\n        ></ng-container>\n      </li>\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=\"isUserScrolled\"\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\">\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  >\n    <stream-message\n      [message]=\"messageInput\"\n      [isLastSentMessage]=\"isLastSentMessage\"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [mode]=\"mode\"\n      [isHighlighted]=\"isHighlighted\"\n    ></stream-message>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      messageTemplate || defaultMessageTemplate;\n      context: getMessageContext(message)\n    \"\n  ></ng-container>\n</ng-template>\n"]}
|
|
392
|
+
//# 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} 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';\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  messageTemplate: TemplateRef<MessageContext> | 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\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private customTemplatesService: CustomTemplatesService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.resetScrollState();\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.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.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  getMessageContext(message: StreamMessage): MessageContext {\n    return {\n      message,\n      isLastSentMessage: !!(\n        this.lastSentMessageId && message?.id === this.lastSentMessageId\n      ),\n      enabledMessageActions: this.enabledMessageActions,\n      mode: this.mode,\n      isHighlighted: message?.id === this.highlightedMessageId,\n    };\n  }\n\n  getTypingIndicatorText(users: UserResponse[]) {\n    const text = listUsers(users);\n\n    return text;\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          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'\"\n        data-testid=\"parent-message\"\n        class=\"str-chat__parent-message-li\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage }\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      <li\n        tabindex=\"0\"\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n        id=\"{{ message.id }}\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: message }\n          \"\n        ></ng-container>\n      </li>\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\">\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  >\n    <stream-message\n      [message]=\"messageInput\"\n      [isLastSentMessage]=\"isLastSentMessage\"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [mode]=\"mode\"\n      [isHighlighted]=\"isHighlighted\"\n    ></stream-message>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      messageTemplate || defaultMessageTemplate;\n      context: getMessageContext(message)\n    \"\n  ></ng-container>\n</ng-template>\n"]}
|
|
@@ -19,7 +19,7 @@ import transliterate from '@stream-io/transliterate';
|
|
|
19
19
|
import * as i8$1 from 'angular-mentions';
|
|
20
20
|
import { MentionModule } from 'angular-mentions';
|
|
21
21
|
|
|
22
|
-
const version = '4.
|
|
22
|
+
const version = '4.22.0';
|
|
23
23
|
|
|
24
24
|
/**
|
|
25
25
|
* The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
|
|
@@ -121,11 +121,12 @@ class ChatClientService {
|
|
|
121
121
|
* @param apiKey
|
|
122
122
|
* @param userOrId
|
|
123
123
|
* @param userTokenOrProvider You can provide a token, or the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users)
|
|
124
|
+
* @param clientOptions Setting to provide to the Stream client instance
|
|
124
125
|
*/
|
|
125
|
-
init(apiKey, userOrId, userTokenOrProvider) {
|
|
126
|
+
init(apiKey, userOrId, userTokenOrProvider, clientOptions) {
|
|
126
127
|
var _a;
|
|
127
128
|
return __awaiter(this, void 0, void 0, function* () {
|
|
128
|
-
this.chatClient = StreamChat.getInstance(apiKey);
|
|
129
|
+
this.chatClient = StreamChat.getInstance(apiKey, clientOptions);
|
|
129
130
|
this.chatClient.devToken;
|
|
130
131
|
let result;
|
|
131
132
|
yield this.ngZone.runOutsideAngular(() => __awaiter(this, void 0, void 0, function* () {
|
|
@@ -4831,11 +4832,16 @@ class MessageListComponent {
|
|
|
4831
4832
|
* 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.
|
|
4832
4833
|
*/
|
|
4833
4834
|
this.messageOptionsTrigger = 'message-row';
|
|
4835
|
+
/**
|
|
4836
|
+
* 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)
|
|
4837
|
+
*/
|
|
4838
|
+
this.hideJumpToLatestButtonDuringScroll = false;
|
|
4834
4839
|
this.enabledMessageActions = [];
|
|
4835
4840
|
this.class = 'str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner';
|
|
4836
4841
|
this.unreadMessageCount = 0;
|
|
4837
4842
|
this.groupStyles = [];
|
|
4838
4843
|
this.isLoading = false;
|
|
4844
|
+
this.isScrollInProgress = false;
|
|
4839
4845
|
this.subscriptions = [];
|
|
4840
4846
|
this.isLatestMessageInList = true;
|
|
4841
4847
|
this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
|
|
@@ -4957,6 +4963,9 @@ class MessageListComponent {
|
|
|
4957
4963
|
var _a;
|
|
4958
4964
|
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
4959
4965
|
(_a = this.newMessageSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
|
|
4966
|
+
if (this.scrollEndTimeout) {
|
|
4967
|
+
clearTimeout(this.scrollEndTimeout);
|
|
4968
|
+
}
|
|
4960
4969
|
}
|
|
4961
4970
|
trackByMessageId(index, item) {
|
|
4962
4971
|
return item.id;
|
|
@@ -4976,6 +4985,13 @@ class MessageListComponent {
|
|
|
4976
4985
|
this.scrollContainer.nativeElement.scrollTop = 0;
|
|
4977
4986
|
}
|
|
4978
4987
|
scrolled() {
|
|
4988
|
+
this.isScrollInProgress = true;
|
|
4989
|
+
if (this.scrollEndTimeout) {
|
|
4990
|
+
clearTimeout(this.scrollEndTimeout);
|
|
4991
|
+
}
|
|
4992
|
+
this.scrollEndTimeout = setTimeout(() => {
|
|
4993
|
+
this.isScrollInProgress = false;
|
|
4994
|
+
}, 100);
|
|
4979
4995
|
if (this.scrollContainer.nativeElement.scrollHeight ===
|
|
4980
4996
|
this.scrollContainer.nativeElement.clientHeight) {
|
|
4981
4997
|
return;
|
|
@@ -5148,7 +5164,7 @@ class MessageListComponent {
|
|
|
5148
5164
|
}
|
|
5149
5165
|
}
|
|
5150
5166
|
MessageListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, deps: [{ token: ChannelService }, { token: ChatClientService }, { token: CustomTemplatesService }], target: i0.ɵɵFactoryTarget.Component });
|
|
5151
|
-
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" }, 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'\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\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 <li\n tabindex=\"0\"\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\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=\"isUserScrolled\"\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\">\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 >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: getMessageContext(message)\n \"\n ></ng-container>\n</ng-template>\n", components: [{ type: LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
|
|
5167
|
+
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" }, 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'\"\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\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 <li\n tabindex=\"0\"\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\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\">\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 >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n ></stream-message>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: getMessageContext(message)\n \"\n ></ng-container>\n</ng-template>\n", components: [{ type: LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }, { type: IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
|
|
5152
5168
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
|
|
5153
5169
|
type: Component,
|
|
5154
5170
|
args: [{
|
|
@@ -5162,6 +5178,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
5162
5178
|
type: Input
|
|
5163
5179
|
}], messageOptionsTrigger: [{
|
|
5164
5180
|
type: Input
|
|
5181
|
+
}], hideJumpToLatestButtonDuringScroll: [{
|
|
5182
|
+
type: Input
|
|
5165
5183
|
}], class: [{
|
|
5166
5184
|
type: HostBinding,
|
|
5167
5185
|
args: ['class']
|