stream-chat-angular 3.7.2 → 3.8.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 +29 -13
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/channel.service.js +14 -6
- package/esm2015/lib/chat-client.service.js +6 -3
- package/esm2015/lib/message/message.component.js +2 -2
- package/fesm2015/stream-chat-angular.js +20 -9
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/chat-client.service.d.ts +2 -2
- package/package.json +1 -1
- package/src/assets/version.ts +1 -1
|
@@ -28,7 +28,7 @@ export class ChatClientService {
|
|
|
28
28
|
* 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.
|
|
29
29
|
* @param apiKey
|
|
30
30
|
* @param userOrId
|
|
31
|
-
* @param userTokenOrProvider
|
|
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
32
|
*/
|
|
33
33
|
init(apiKey, userOrId, userTokenOrProvider) {
|
|
34
34
|
var _a;
|
|
@@ -38,7 +38,10 @@ export class ChatClientService {
|
|
|
38
38
|
let result;
|
|
39
39
|
yield this.ngZone.runOutsideAngular(() => __awaiter(this, void 0, void 0, function* () {
|
|
40
40
|
const user = typeof userOrId === 'string' ? { id: userOrId } : userOrId;
|
|
41
|
-
result =
|
|
41
|
+
result =
|
|
42
|
+
userTokenOrProvider === 'guest'
|
|
43
|
+
? yield this.chatClient.setGuestUser(user)
|
|
44
|
+
: yield this.chatClient.connectUser(user, userTokenOrProvider);
|
|
42
45
|
this.userSubject.next(this.chatClient.user);
|
|
43
46
|
this.chatClient.setUserAgent(`stream-chat-angular-${version}-${this.chatClient.getUserAgent()}`);
|
|
44
47
|
}));
|
|
@@ -150,4 +153,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
150
153
|
providedIn: 'root',
|
|
151
154
|
}]
|
|
152
155
|
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i1.NotificationService }]; } });
|
|
153
|
-
//# 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\n   */\n  async init(\n    apiKey: string,\n    userOrId: string | OwnUserResponse<T> | UserResponse<T>,\n    userTokenOrProvider: TokenOrProvider\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      result = await this.chatClient.connectUser(user, userTokenOrProvider);\n      this.userSubject.next(this.chatClient.user);\n      this.chatClient.setUserAgent(\n        `stream-chat-angular-${version}-${this.chatClient.getUserAgent()}`\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 || {});\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"]}
|
|
156
|
+
//# 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      result =\n        userTokenOrProvider === 'guest'\n          ? await this.chatClient.setGuestUser(user)\n          : await this.chatClient.connectUser(user, userTokenOrProvider);\n      this.userSubject.next(this.chatClient.user);\n      this.chatClient.setUserAgent(\n        `stream-chat-angular-${version}-${this.chatClient.getUserAgent()}`\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 || {});\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"]}
|
|
@@ -38,9 +38,9 @@ export class MessageComponent {
|
|
|
38
38
|
this.visibleMessageActionsCount = 0;
|
|
39
39
|
this.messageTextParts = [];
|
|
40
40
|
this.subscriptions = [];
|
|
41
|
-
this.user = this.chatClientService.chatClient.user;
|
|
42
41
|
}
|
|
43
42
|
ngOnInit() {
|
|
43
|
+
this.subscriptions.push(this.chatClientService.user$.subscribe((u) => (this.user = u)));
|
|
44
44
|
this.subscriptions.push(this.customTemplatesService.mentionTemplate$.subscribe((template) => (this.mentionTemplate = template)));
|
|
45
45
|
this.subscriptions.push(this.customTemplatesService.attachmentListTemplate$.subscribe((template) => (this.attachmentListTemplate = template)));
|
|
46
46
|
this.subscriptions.push(this.customTemplatesService.messageActionsBoxTemplate$.subscribe((template) => (this.messageActionsBoxTemplate = template)));
|
|
@@ -271,4 +271,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
271
271
|
type: ViewChild,
|
|
272
272
|
args: ['container']
|
|
273
273
|
}] } });
|
|
274
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message/message.component.ts","../../../../../projects/stream-chat-angular/src/lib/message/message.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  TemplateRef,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  OnDestroy,\n  OnInit,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { getDeviceWidth } from '../device-width';\nimport {\n  AttachmentListContext,\n  MentionTemplateContext,\n  MessageActionsBoxContext,\n  MessageReactionsContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n} from '../types';\nimport { parseDate } from './parse-date';\nimport emojiRegex from 'emoji-regex';\nimport { Subscription } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](../concepts/message-interactions.mdx).\n */\n@Component({\n  selector: 'stream-message',\n  templateUrl: './message.component.html',\n  styles: [],\n})\nexport class MessageComponent implements OnInit, OnChanges, OnDestroy {\n  /**\n   * The message to be displayed\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * The list of [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) that are enabled for the current user, the list of [supported interactions](../concepts/message-interactions.mdx) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   */\n  @Input() enabledMessageActions: string[] = [];\n  /**\n   * If `true`, the message status (sending, sent, who read the message) is displayed.\n   */\n  @Input() isLastSentMessage: boolean | undefined;\n  /**\n   * Determines if the message is being dispalyed in a channel or in a [thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'thread' | 'main' = 'main';\n  canReceiveReadEvents: boolean | undefined;\n  canReactToMessage: boolean | undefined;\n  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  isPressedOnMobile = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] = [];\n  mentionTemplate: TemplateRef<MentionTemplateContext> | undefined;\n  attachmentListTemplate: TemplateRef<AttachmentListContext> | undefined;\n  messageActionsBoxTemplate: TemplateRef<MessageActionsBoxContext> | undefined;\n  messageReactionsTemplate: TemplateRef<MessageReactionsContext> | undefined;\n  private user: UserResponse<DefaultStreamChatGenerics> | undefined;\n  private subscriptions: Subscription[] = [];\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService,\n    private customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef\n  ) {\n    this.user = this.chatClientService.chatClient.user;\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.customTemplatesService.mentionTemplate$.subscribe(\n        (template) => (this.mentionTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.attachmentListTemplate$.subscribe(\n        (template) => (this.attachmentListTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.messageActionsBoxTemplate$.subscribe(\n        (template) => (this.messageActionsBoxTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.messageReactionsTemplate$.subscribe(\n        (template) => (this.messageReactionsTemplate = template)\n      )\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.createMessageParts();\n    }\n    if (changes.enabledMessageActions) {\n      this.canReactToMessage =\n        this.enabledMessageActions.indexOf('send-reaction') !== -1;\n      this.canReceiveReadEvents =\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get isSentByCurrentUser() {\n    return this.message?.user?.id === this.user?.id;\n  }\n\n  get readByText() {\n    return listUsers(this.message!.readBy);\n  }\n\n  get lastReadUser() {\n    return this.message?.readBy.filter((u) => u.id !== this.user?.id)[0];\n  }\n\n  get isOnlyReadByMe() {\n    return this.message && this.message.readBy.length === 0;\n  }\n\n  get isReadByMultipleUsers() {\n    return this.message && this.message.readBy.length > 1;\n  }\n\n  get isMessageDeliveredAndRead() {\n    return (\n      this.message &&\n      this.message.readBy &&\n      this.message.status === 'received' &&\n      this.message.readBy.length > 0\n    );\n  }\n\n  get parsedDate() {\n    if (!this.message || !this.message?.created_at) {\n      return;\n    }\n    return parseDate(this.message.created_at);\n  }\n\n  get areOptionsVisible() {\n    if (!this.message) {\n      return false;\n    }\n    return !(\n      !this.message.type ||\n      this.message.type === 'error' ||\n      this.message.type === 'system' ||\n      this.message.type === 'ephemeral' ||\n      this.message.status === 'failed' ||\n      this.message.status === 'sending' ||\n      (this.mode === 'thread' && !this.message.parent_id)\n    );\n  }\n\n  get hasAttachment() {\n    return !!this.message?.attachments && !!this.message.attachments.length;\n  }\n\n  get hasReactions() {\n    return (\n      !!this.message?.reaction_counts &&\n      Object.keys(this.message.reaction_counts).length > 0\n    );\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.message?.reply_count };\n  }\n\n  get canDisplayReadStatus() {\n    return (\n      this.canReceiveReadEvents !== false &&\n      this.enabledMessageActions.indexOf('read-events') !== -1\n    );\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.message?.quoted_message?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  getAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.id || '',\n      attachments: this.message?.attachments || [],\n      parentMessageId: this.message?.parent_id,\n    };\n  }\n\n  getQuotedMessageAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.quoted_message?.id || '',\n      attachments: this.quotedMessageAttachments,\n      parentMessageId: this?.message?.quoted_message?.parent_id,\n    };\n  }\n\n  getMessageReactionsContext(): MessageReactionsContext {\n    return {\n      messageReactionCounts: this.message?.reaction_counts || {},\n      latestReactions: this.message?.latest_reactions || [],\n      isSelectorOpen: this.isReactionSelectorOpen,\n      isSelectorOpenChangeHandler: (isOpen) =>\n        (this.isReactionSelectorOpen = isOpen),\n      messageId: this.message?.id,\n      ownReactions: this.message?.own_reactions || [],\n    };\n  }\n\n  resendMessage() {\n    void this.channelService.resendMessage(this.message!);\n  }\n\n  textClicked() {\n    if (getDeviceWidth().device !== 'mobile') {\n      this.isPressedOnMobile = false;\n      return;\n    }\n    if (this.isPressedOnMobile) {\n      return;\n    }\n    this.isPressedOnMobile = true;\n    const eventHandler = (event: Event) => {\n      if (!this.container?.nativeElement.contains(event.target as Node)) {\n        this.isPressedOnMobile = false;\n        window.removeEventListener('click', eventHandler);\n      }\n    };\n    window.addEventListener('click', eventHandler);\n  }\n\n  setAsActiveParentMessage() {\n    void this.channelService.setAsActiveParentMessage(this.message);\n  }\n\n  getMentionContext(messagePart: MessagePart): MentionTemplateContext {\n    return {\n      content: messagePart.content,\n      user: messagePart.user!,\n    };\n  }\n\n  getMessageActionsBoxContext(): MessageActionsBoxContext {\n    return {\n      isOpen: this.isActionBoxOpen,\n      isMine: this.isSentByCurrentUser,\n      enabledActions: this.enabledMessageActions,\n      message: this.message,\n      displayedActionsCountChaneHanler: (count) => {\n        this.visibleMessageActionsCount = count;\n        // message action box changes UI bindings in parent, so we'll have to rerun change detection\n        this.cdRef.detectChanges();\n      },\n      isEditingChangeHandler: (isEditing) => {\n        this.isEditing = isEditing;\n        this.isActionBoxOpen = !this.isEditing;\n      },\n    };\n  }\n\n  private createMessageParts() {\n    let content = this.message?.html || this.message?.text;\n    if (!content) {\n      this.messageTextParts = [];\n    } else {\n      // Backend will wrap HTML content with <p></p>\\n\n      if (content.startsWith('<p>')) {\n        content = content.replace('<p>', '');\n      }\n      if (content.endsWith('</p>\\n')) {\n        content = content.replace('</p>\\n', '');\n      }\n      if (\n        !this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0\n      ) {\n        // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n        const regex = new RegExp(emojiRegex(), 'g');\n        // Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome\n        /* eslint-disable @typescript-eslint/no-unsafe-member-access */\n        const isChrome =\n          !!(window as any).chrome &&\n          typeof (window as any).opr === 'undefined';\n        /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n        content = content.replace(\n          regex,\n          (match) =>\n            `<span ${\n              isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n            }>${match}</span>`\n        );\n        this.messageTextParts = [{ content, type: 'text' }];\n      } else {\n        this.messageTextParts = [];\n        let text = content;\n        this.message!.mentioned_users.forEach((user) => {\n          const mention = `@${user.name || user.id}`;\n          const precedingText = text.substring(0, text.indexOf(mention));\n          this.messageTextParts.push({\n            content: precedingText,\n            type: 'text',\n          });\n          this.messageTextParts.push({\n            content: mention,\n            type: 'mention',\n            user,\n          });\n          text = text.replace(precedingText + mention, '');\n        });\n        if (text) {\n          this.messageTextParts.push({ content: text, type: 'text' });\n        }\n      }\n    }\n  }\n}\n","<div\n  #container\n  class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n    message?.type\n  }} str-chat__message--{{ message?.status }} {{\n    message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n  }}\"\n  [class.str-chat__message--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.mobile-press]=\"isPressedOnMobile\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"hasReactions\"\n  data-testid=\"message-container\"\n  (mouseleave)=\"isActionBoxOpen = false\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container\n        *ngIf=\"\n          isSentByCurrentUser &&\n          ((isLastSentMessage && message?.status === 'received') ||\n            message?.status === 'sending')\n        \"\n      >\n        <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n          <ng-container *ngTemplateOutlet=\"sendingStatus\"></ng-container>\n        </ng-container>\n        <ng-template #sentStatus>\n          <ng-container\n            *ngIf=\"\n              mode === 'main' &&\n                isMessageDeliveredAndRead &&\n                canDisplayReadStatus;\n              else deliveredStatus\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n          </ng-container>\n        </ng-template>\n      </ng-container>\n      <stream-avatar-placeholder\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"message?.user?.image\"\n        [name]=\"message?.user?.name || message?.user?.id\"\n        type=\"user\"\n        location=\"message-sender\"\n        [user]=\"message?.user || undefined\"\n      ></stream-avatar-placeholder>\n      <div class=\"str-chat__message-inner\">\n        <div\n          class=\"str-chat__message-simple__actions\"\n          data-testid=\"message-options\"\n          *ngIf=\"areOptionsVisible\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--options\n            \"\n            [class.str-chat-angular__message-simple__actions__action--options--editing]=\"\n              isEditing\n            \"\n          >\n            <ng-template\n              #defaultMessageActionsBox\n              let-isOpen=\"isOpen\"\n              let-isMine=\"isMine\"\n              let-enabledActions=\"enabledActions\"\n              let-messageInput=\"message\"\n              let-displayedActionsCountChaneHanler=\"displayedActionsCountChaneHanler\"\n              let-isEditingChangeHandler=\"isEditingChangeHandler\"\n            >\n              <stream-message-actions-box\n                [isOpen]=\"isOpen\"\n                [isMine]=\"isMine\"\n                [enabledActions]=\"enabledActions\"\n                [message]=\"messageInput\"\n                (displayedActionsCount)=\"\n                  displayedActionsCountChaneHanler($event)\n                \"\n                (isEditing)=\"isEditingChangeHandler($event)\"\n              ></stream-message-actions-box>\n            </ng-template>\n            <ng-container\n              *ngTemplateOutlet=\"\n                messageActionsBoxTemplate || defaultMessageActionsBox;\n                context: getMessageActionsBoxContext()\n              \"\n            ></ng-container>\n            <stream-icon-placeholder\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              icon=\"action-icon\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n            ></stream-icon-placeholder>\n          </div>\n          <!-- eslint-disable @angular-eslint/template/conditional-complexity -->\n          <div\n            *ngIf=\"\n              enabledMessageActions.indexOf('send-reply') !== -1 &&\n              mode === 'main'\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--thread\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder\n              icon=\"reply-in-thread\"\n            ></stream-icon-placeholder>\n          </div>\n          <div\n            *ngIf=\"canReactToMessage\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--reactions\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon-placeholder\n              icon=\"reaction-icon\"\n            ></stream-icon-placeholder>\n          </div>\n        </div>\n        <!-- eslint-enable @angular-eslint/template/conditional-complexity -->\n        <ng-template\n          #defaultMessageReactions\n          let-messageReactionCounts=\"messageReactionCounts\"\n          let-latestReactions=\"latestReactions\"\n          let-isSelectorOpen=\"isSelectorOpen\"\n          let-isSelectorOpenChangeHandler=\"isSelectorOpenChangeHandler\"\n          let-messageId=\"messageId\"\n          let-ownReactions=\"ownReactions\"\n        >\n          <stream-message-reactions\n            [messageReactionCounts]=\"messageReactionCounts\"\n            [latestReactions]=\"latestReactions\"\n            [isSelectorOpen]=\"isSelectorOpen\"\n            (isSelectorOpenChange)=\"isSelectorOpenChangeHandler($event)\"\n            [messageId]=\"messageId\"\n            [ownReactions]=\"ownReactions\"\n          ></stream-message-reactions>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageReactionsTemplate || defaultMessageReactions;\n            context: getMessageReactionsContext()\n          \"\n        ></ng-container>\n        <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n          <ng-template\n            #defaultAttachments\n            let-messageId=\"messageId\"\n            let-attachments=\"attachments\"\n            let-parentMessageId=\"parentMessageId\"\n          >\n            <stream-attachment-list\n              [messageId]=\"messageId\"\n              [attachments]=\"attachments\"\n              [parentMessageId]=\"parentMessageId\"\n            ></stream-attachment-list>\n          </ng-template>\n          <ng-container\n            *ngTemplateOutlet=\"\n              attachmentListTemplate || defaultAttachments;\n              context: getAttachmentListContext()\n            \"\n          ></ng-container>\n        </ng-container>\n        <div\n          class=\"str-chat__message-text\"\n          *ngIf=\"message?.text || (message?.quoted_message && hasAttachment)\"\n        >\n          <div\n            data-testid=\"inner-message\"\n            class=\"\n              str-chat__message-text-inner str-chat__message-simple-text-inner\n            \"\n            [class.str-chat__message-light-text-inner--has-attachment]=\"\n              hasAttachment\n            \"\n            (click)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n            (keyup.enter)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"quotedMessage\"></ng-container>\n            <stream-attachment-list\n              *ngIf=\"hasAttachment && message?.quoted_message\"\n              [attachments]=\"message!.attachments!\"\n              [messageId]=\"message!.id\"\n            ></stream-attachment-list>\n            <div\n              data-testid=\"client-error-message\"\n              *ngIf=\"message?.type === 'error'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{ \"streamChat.Error · Unsent\" | translate }}\n            </div>\n            <div\n              data-testid=\"error-message\"\n              *ngIf=\"message?.status === 'failed'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{\n                (message?.errorStatusCode === 403\n                  ? \"streamChat.Message Failed · Unauthorized\"\n                  : \"streamChat.Message Failed · Click to try again\"\n                ) | translate\n              }}\n            </div>\n            <div\n              (click)=\"textClicked()\"\n              (keyup.enter)=\"textClicked()\"\n              data-testid=\"text\"\n            >\n              <p>\n                <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n                <ng-container *ngFor=\"let part of messageTextParts\">\n                  <span\n                    *ngIf=\"part.type === 'text'; else mention\"\n                    [innerHTML]=\"part.content\"\n                  ></span>\n                  <ng-template #mention>\n                    <ng-template #defaultMention let-content=\"content\">\n                      <b>{{ content }}</b>\n                    </ng-template>\n                    <ng-container\n                      *ngTemplateOutlet=\"\n                        mentionTemplate || defaultMention;\n                        context: getMentionContext(part)\n                      \"\n                    ></ng-container>\n                  </ng-template>\n                </ng-container>\n              </p>\n            </div>\n          </div>\n        </div>\n        <div class=\"str-chat__message-simple-reply-button\">\n          <button\n            *ngIf=\"\n              !!message?.reply_count &&\n              mode !== 'thread' &&\n              enabledMessageActions.indexOf('send-reply') !== -1\n            \"\n            class=\"str-chat__message-replies-count-button\"\n            data-testid=\"reply-count-button\"\n            (click)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder icon=\"reply\"></stream-icon-placeholder>\n            {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n            replies' | translate:replyCountParam)}}\n          </button>\n        </div>\n        <div class=\"str-chat__message-data str-chat__message-simple-data\">\n          <span\n            data-testid=\"sender\"\n            *ngIf=\"!isSentByCurrentUser\"\n            class=\"str-chat__message-simple-name\"\n          >\n            {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n          </span>\n          <span data-testid=\"date\" class=\"str-chat__message-simple-timestamp\">\n            {{ parsedDate }}\n          </span>\n        </div>\n      </div>\n    </ng-container>\n  </ng-container>\n</div>\n\n<ng-template #sendingStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"sending-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Sending...\" | translate }}\n    </div>\n    <stream-loading-indicator-placeholder\n      data-testid=\"loading-indicator\"\n    ></stream-loading-indicator-placeholder>\n  </span>\n</ng-template>\n<ng-template #readStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"read-indicator\"\n  >\n    <div class=\"str-chat__tooltip\" data-testid=\"read-by-tooltip\">\n      {{ readByText }}\n    </div>\n    <stream-avatar-placeholder\n      class=\"str-chat-angular__avatar-host\"\n      data-test-id=\"last-read-user-avatar\"\n      [size]=\"15\"\n      [imageUrl]=\"lastReadUser?.image\"\n      [name]=\"lastReadUser?.name || lastReadUser?.id\"\n      type=\"user\"\n      location=\"message-reader\"\n      [user]=\"lastReadUser\"\n    ></stream-avatar-placeholder>\n    <span\n      data-test-id=\"read-by-length\"\n      *ngIf=\"isReadByMultipleUsers\"\n      class=\"str-chat__message-simple-status-number\"\n    >\n      {{ (message?.readBy)!.length }}\n    </span>\n  </span>\n</ng-template>\n<ng-template #deliveredStatus>\n  <span\n    *ngIf=\"mode === 'main'\"\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"delivered-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Delivered\" | translate }}\n    </div>\n    <stream-icon-placeholder\n      data-testid=\"delivered-icon\"\n      icon=\"delivered-icon\"\n    ></stream-icon-placeholder>\n  </span>\n</ng-template>\n\n<ng-template #deletedMessage>\n  <div data-testid=\"message-deleted-component\">\n    <div class=\"str-chat__message--deleted-inner\" translate>\n      streamChat.This message was deleted...\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #systemMessage>\n  <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n    <div class=\"str-chat__message--system__text\">\n      <div class=\"str-chat__message--system__line\"></div>\n      <p>{{ message?.text }}</p>\n      <div class=\"str-chat__message--system__line\"></div>\n    </div>\n    <div class=\"str-chat__message--system__date\">\n      {{ parsedDate }}\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n  >\n    <stream-avatar-placeholder\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host\"\n      [imageUrl]=\"message?.quoted_message?.user?.image\"\n      [name]=\"\n        message?.quoted_message?.user?.name || message?.quoted_message?.user?.id\n      \"\n      [size]=\"20\"\n      type=\"user\"\n      location=\"quoted-message-sender\"\n      [user]=\"message?.quoted_message?.user || undefined\"\n    ></stream-avatar-placeholder>\n    <div class=\"quoted-message-inner\">\n      <ng-container\n        *ngIf=\"\n          message?.quoted_message?.attachments &&\n          message?.quoted_message?.attachments?.length\n        \"\n      >\n        <ng-template\n          #defaultAttachments\n          let-messageId=\"messageId\"\n          let-attachments=\"attachments\"\n          let-parentMessageId=\"parentMessageId\"\n        >\n          <stream-attachment-list\n            [messageId]=\"messageId\"\n            [attachments]=\"attachments\"\n            [parentMessageId]=\"parentMessageId\"\n          ></stream-attachment-list>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            attachmentListTemplate || defaultAttachments;\n            context: getQuotedMessageAttachmentListContext()\n          \"\n        ></ng-container>\n      </ng-container>\n      <div\n        data-testid=\"quoted-message-text\"\n        [innerHTML]=\"\n          message?.quoted_message?.html || message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
274
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message/message.component.ts","../../../../../projects/stream-chat-angular/src/lib/message/message.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  TemplateRef,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  OnDestroy,\n  OnInit,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { getDeviceWidth } from '../device-width';\nimport {\n  AttachmentListContext,\n  MentionTemplateContext,\n  MessageActionsBoxContext,\n  MessageReactionsContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n} from '../types';\nimport { parseDate } from './parse-date';\nimport emojiRegex from 'emoji-regex';\nimport { Subscription } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](../concepts/message-interactions.mdx).\n */\n@Component({\n  selector: 'stream-message',\n  templateUrl: './message.component.html',\n  styles: [],\n})\nexport class MessageComponent implements OnInit, OnChanges, OnDestroy {\n  /**\n   * The message to be displayed\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * The list of [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) that are enabled for the current user, the list of [supported interactions](../concepts/message-interactions.mdx) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   */\n  @Input() enabledMessageActions: string[] = [];\n  /**\n   * If `true`, the message status (sending, sent, who read the message) is displayed.\n   */\n  @Input() isLastSentMessage: boolean | undefined;\n  /**\n   * Determines if the message is being dispalyed in a channel or in a [thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'thread' | 'main' = 'main';\n  canReceiveReadEvents: boolean | undefined;\n  canReactToMessage: boolean | undefined;\n  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  isPressedOnMobile = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] = [];\n  mentionTemplate: TemplateRef<MentionTemplateContext> | undefined;\n  attachmentListTemplate: TemplateRef<AttachmentListContext> | undefined;\n  messageActionsBoxTemplate: TemplateRef<MessageActionsBoxContext> | undefined;\n  messageReactionsTemplate: TemplateRef<MessageReactionsContext> | undefined;\n  private user: UserResponse<DefaultStreamChatGenerics> | undefined;\n  private subscriptions: Subscription[] = [];\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService,\n    private customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((u) => (this.user = u))\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.mentionTemplate$.subscribe(\n        (template) => (this.mentionTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.attachmentListTemplate$.subscribe(\n        (template) => (this.attachmentListTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.messageActionsBoxTemplate$.subscribe(\n        (template) => (this.messageActionsBoxTemplate = template)\n      )\n    );\n    this.subscriptions.push(\n      this.customTemplatesService.messageReactionsTemplate$.subscribe(\n        (template) => (this.messageReactionsTemplate = template)\n      )\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.createMessageParts();\n    }\n    if (changes.enabledMessageActions) {\n      this.canReactToMessage =\n        this.enabledMessageActions.indexOf('send-reaction') !== -1;\n      this.canReceiveReadEvents =\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get isSentByCurrentUser() {\n    return this.message?.user?.id === this.user?.id;\n  }\n\n  get readByText() {\n    return listUsers(this.message!.readBy);\n  }\n\n  get lastReadUser() {\n    return this.message?.readBy.filter((u) => u.id !== this.user?.id)[0];\n  }\n\n  get isOnlyReadByMe() {\n    return this.message && this.message.readBy.length === 0;\n  }\n\n  get isReadByMultipleUsers() {\n    return this.message && this.message.readBy.length > 1;\n  }\n\n  get isMessageDeliveredAndRead() {\n    return (\n      this.message &&\n      this.message.readBy &&\n      this.message.status === 'received' &&\n      this.message.readBy.length > 0\n    );\n  }\n\n  get parsedDate() {\n    if (!this.message || !this.message?.created_at) {\n      return;\n    }\n    return parseDate(this.message.created_at);\n  }\n\n  get areOptionsVisible() {\n    if (!this.message) {\n      return false;\n    }\n    return !(\n      !this.message.type ||\n      this.message.type === 'error' ||\n      this.message.type === 'system' ||\n      this.message.type === 'ephemeral' ||\n      this.message.status === 'failed' ||\n      this.message.status === 'sending' ||\n      (this.mode === 'thread' && !this.message.parent_id)\n    );\n  }\n\n  get hasAttachment() {\n    return !!this.message?.attachments && !!this.message.attachments.length;\n  }\n\n  get hasReactions() {\n    return (\n      !!this.message?.reaction_counts &&\n      Object.keys(this.message.reaction_counts).length > 0\n    );\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.message?.reply_count };\n  }\n\n  get canDisplayReadStatus() {\n    return (\n      this.canReceiveReadEvents !== false &&\n      this.enabledMessageActions.indexOf('read-events') !== -1\n    );\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.message?.quoted_message?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  getAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.id || '',\n      attachments: this.message?.attachments || [],\n      parentMessageId: this.message?.parent_id,\n    };\n  }\n\n  getQuotedMessageAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.quoted_message?.id || '',\n      attachments: this.quotedMessageAttachments,\n      parentMessageId: this?.message?.quoted_message?.parent_id,\n    };\n  }\n\n  getMessageReactionsContext(): MessageReactionsContext {\n    return {\n      messageReactionCounts: this.message?.reaction_counts || {},\n      latestReactions: this.message?.latest_reactions || [],\n      isSelectorOpen: this.isReactionSelectorOpen,\n      isSelectorOpenChangeHandler: (isOpen) =>\n        (this.isReactionSelectorOpen = isOpen),\n      messageId: this.message?.id,\n      ownReactions: this.message?.own_reactions || [],\n    };\n  }\n\n  resendMessage() {\n    void this.channelService.resendMessage(this.message!);\n  }\n\n  textClicked() {\n    if (getDeviceWidth().device !== 'mobile') {\n      this.isPressedOnMobile = false;\n      return;\n    }\n    if (this.isPressedOnMobile) {\n      return;\n    }\n    this.isPressedOnMobile = true;\n    const eventHandler = (event: Event) => {\n      if (!this.container?.nativeElement.contains(event.target as Node)) {\n        this.isPressedOnMobile = false;\n        window.removeEventListener('click', eventHandler);\n      }\n    };\n    window.addEventListener('click', eventHandler);\n  }\n\n  setAsActiveParentMessage() {\n    void this.channelService.setAsActiveParentMessage(this.message);\n  }\n\n  getMentionContext(messagePart: MessagePart): MentionTemplateContext {\n    return {\n      content: messagePart.content,\n      user: messagePart.user!,\n    };\n  }\n\n  getMessageActionsBoxContext(): MessageActionsBoxContext {\n    return {\n      isOpen: this.isActionBoxOpen,\n      isMine: this.isSentByCurrentUser,\n      enabledActions: this.enabledMessageActions,\n      message: this.message,\n      displayedActionsCountChaneHanler: (count) => {\n        this.visibleMessageActionsCount = count;\n        // message action box changes UI bindings in parent, so we'll have to rerun change detection\n        this.cdRef.detectChanges();\n      },\n      isEditingChangeHandler: (isEditing) => {\n        this.isEditing = isEditing;\n        this.isActionBoxOpen = !this.isEditing;\n      },\n    };\n  }\n\n  private createMessageParts() {\n    let content = this.message?.html || this.message?.text;\n    if (!content) {\n      this.messageTextParts = [];\n    } else {\n      // Backend will wrap HTML content with <p></p>\\n\n      if (content.startsWith('<p>')) {\n        content = content.replace('<p>', '');\n      }\n      if (content.endsWith('</p>\\n')) {\n        content = content.replace('</p>\\n', '');\n      }\n      if (\n        !this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0\n      ) {\n        // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n        const regex = new RegExp(emojiRegex(), 'g');\n        // Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome\n        /* eslint-disable @typescript-eslint/no-unsafe-member-access */\n        const isChrome =\n          !!(window as any).chrome &&\n          typeof (window as any).opr === 'undefined';\n        /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n        content = content.replace(\n          regex,\n          (match) =>\n            `<span ${\n              isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n            }>${match}</span>`\n        );\n        this.messageTextParts = [{ content, type: 'text' }];\n      } else {\n        this.messageTextParts = [];\n        let text = content;\n        this.message!.mentioned_users.forEach((user) => {\n          const mention = `@${user.name || user.id}`;\n          const precedingText = text.substring(0, text.indexOf(mention));\n          this.messageTextParts.push({\n            content: precedingText,\n            type: 'text',\n          });\n          this.messageTextParts.push({\n            content: mention,\n            type: 'mention',\n            user,\n          });\n          text = text.replace(precedingText + mention, '');\n        });\n        if (text) {\n          this.messageTextParts.push({ content: text, type: 'text' });\n        }\n      }\n    }\n  }\n}\n","<div\n  #container\n  class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n    message?.type\n  }} str-chat__message--{{ message?.status }} {{\n    message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n  }}\"\n  [class.str-chat__message--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.mobile-press]=\"isPressedOnMobile\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"hasReactions\"\n  data-testid=\"message-container\"\n  (mouseleave)=\"isActionBoxOpen = false\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container\n        *ngIf=\"\n          isSentByCurrentUser &&\n          ((isLastSentMessage && message?.status === 'received') ||\n            message?.status === 'sending')\n        \"\n      >\n        <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n          <ng-container *ngTemplateOutlet=\"sendingStatus\"></ng-container>\n        </ng-container>\n        <ng-template #sentStatus>\n          <ng-container\n            *ngIf=\"\n              mode === 'main' &&\n                isMessageDeliveredAndRead &&\n                canDisplayReadStatus;\n              else deliveredStatus\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n          </ng-container>\n        </ng-template>\n      </ng-container>\n      <stream-avatar-placeholder\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"message?.user?.image\"\n        [name]=\"message?.user?.name || message?.user?.id\"\n        type=\"user\"\n        location=\"message-sender\"\n        [user]=\"message?.user || undefined\"\n      ></stream-avatar-placeholder>\n      <div class=\"str-chat__message-inner\">\n        <div\n          class=\"str-chat__message-simple__actions\"\n          data-testid=\"message-options\"\n          *ngIf=\"areOptionsVisible\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--options\n            \"\n            [class.str-chat-angular__message-simple__actions__action--options--editing]=\"\n              isEditing\n            \"\n          >\n            <ng-template\n              #defaultMessageActionsBox\n              let-isOpen=\"isOpen\"\n              let-isMine=\"isMine\"\n              let-enabledActions=\"enabledActions\"\n              let-messageInput=\"message\"\n              let-displayedActionsCountChaneHanler=\"displayedActionsCountChaneHanler\"\n              let-isEditingChangeHandler=\"isEditingChangeHandler\"\n            >\n              <stream-message-actions-box\n                [isOpen]=\"isOpen\"\n                [isMine]=\"isMine\"\n                [enabledActions]=\"enabledActions\"\n                [message]=\"messageInput\"\n                (displayedActionsCount)=\"\n                  displayedActionsCountChaneHanler($event)\n                \"\n                (isEditing)=\"isEditingChangeHandler($event)\"\n              ></stream-message-actions-box>\n            </ng-template>\n            <ng-container\n              *ngTemplateOutlet=\"\n                messageActionsBoxTemplate || defaultMessageActionsBox;\n                context: getMessageActionsBoxContext()\n              \"\n            ></ng-container>\n            <stream-icon-placeholder\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              icon=\"action-icon\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n            ></stream-icon-placeholder>\n          </div>\n          <!-- eslint-disable @angular-eslint/template/conditional-complexity -->\n          <div\n            *ngIf=\"\n              enabledMessageActions.indexOf('send-reply') !== -1 &&\n              mode === 'main'\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--thread\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder\n              icon=\"reply-in-thread\"\n            ></stream-icon-placeholder>\n          </div>\n          <div\n            *ngIf=\"canReactToMessage\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--reactions\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon-placeholder\n              icon=\"reaction-icon\"\n            ></stream-icon-placeholder>\n          </div>\n        </div>\n        <!-- eslint-enable @angular-eslint/template/conditional-complexity -->\n        <ng-template\n          #defaultMessageReactions\n          let-messageReactionCounts=\"messageReactionCounts\"\n          let-latestReactions=\"latestReactions\"\n          let-isSelectorOpen=\"isSelectorOpen\"\n          let-isSelectorOpenChangeHandler=\"isSelectorOpenChangeHandler\"\n          let-messageId=\"messageId\"\n          let-ownReactions=\"ownReactions\"\n        >\n          <stream-message-reactions\n            [messageReactionCounts]=\"messageReactionCounts\"\n            [latestReactions]=\"latestReactions\"\n            [isSelectorOpen]=\"isSelectorOpen\"\n            (isSelectorOpenChange)=\"isSelectorOpenChangeHandler($event)\"\n            [messageId]=\"messageId\"\n            [ownReactions]=\"ownReactions\"\n          ></stream-message-reactions>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageReactionsTemplate || defaultMessageReactions;\n            context: getMessageReactionsContext()\n          \"\n        ></ng-container>\n        <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n          <ng-template\n            #defaultAttachments\n            let-messageId=\"messageId\"\n            let-attachments=\"attachments\"\n            let-parentMessageId=\"parentMessageId\"\n          >\n            <stream-attachment-list\n              [messageId]=\"messageId\"\n              [attachments]=\"attachments\"\n              [parentMessageId]=\"parentMessageId\"\n            ></stream-attachment-list>\n          </ng-template>\n          <ng-container\n            *ngTemplateOutlet=\"\n              attachmentListTemplate || defaultAttachments;\n              context: getAttachmentListContext()\n            \"\n          ></ng-container>\n        </ng-container>\n        <div\n          class=\"str-chat__message-text\"\n          *ngIf=\"message?.text || (message?.quoted_message && hasAttachment)\"\n        >\n          <div\n            data-testid=\"inner-message\"\n            class=\"\n              str-chat__message-text-inner str-chat__message-simple-text-inner\n            \"\n            [class.str-chat__message-light-text-inner--has-attachment]=\"\n              hasAttachment\n            \"\n            (click)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n            (keyup.enter)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"quotedMessage\"></ng-container>\n            <stream-attachment-list\n              *ngIf=\"hasAttachment && message?.quoted_message\"\n              [attachments]=\"message!.attachments!\"\n              [messageId]=\"message!.id\"\n            ></stream-attachment-list>\n            <div\n              data-testid=\"client-error-message\"\n              *ngIf=\"message?.type === 'error'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{ \"streamChat.Error · Unsent\" | translate }}\n            </div>\n            <div\n              data-testid=\"error-message\"\n              *ngIf=\"message?.status === 'failed'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{\n                (message?.errorStatusCode === 403\n                  ? \"streamChat.Message Failed · Unauthorized\"\n                  : \"streamChat.Message Failed · Click to try again\"\n                ) | translate\n              }}\n            </div>\n            <div\n              (click)=\"textClicked()\"\n              (keyup.enter)=\"textClicked()\"\n              data-testid=\"text\"\n            >\n              <p>\n                <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n                <ng-container *ngFor=\"let part of messageTextParts\">\n                  <span\n                    *ngIf=\"part.type === 'text'; else mention\"\n                    [innerHTML]=\"part.content\"\n                  ></span>\n                  <ng-template #mention>\n                    <ng-template #defaultMention let-content=\"content\">\n                      <b>{{ content }}</b>\n                    </ng-template>\n                    <ng-container\n                      *ngTemplateOutlet=\"\n                        mentionTemplate || defaultMention;\n                        context: getMentionContext(part)\n                      \"\n                    ></ng-container>\n                  </ng-template>\n                </ng-container>\n              </p>\n            </div>\n          </div>\n        </div>\n        <div class=\"str-chat__message-simple-reply-button\">\n          <button\n            *ngIf=\"\n              !!message?.reply_count &&\n              mode !== 'thread' &&\n              enabledMessageActions.indexOf('send-reply') !== -1\n            \"\n            class=\"str-chat__message-replies-count-button\"\n            data-testid=\"reply-count-button\"\n            (click)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder icon=\"reply\"></stream-icon-placeholder>\n            {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n            replies' | translate:replyCountParam)}}\n          </button>\n        </div>\n        <div class=\"str-chat__message-data str-chat__message-simple-data\">\n          <span\n            data-testid=\"sender\"\n            *ngIf=\"!isSentByCurrentUser\"\n            class=\"str-chat__message-simple-name\"\n          >\n            {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n          </span>\n          <span data-testid=\"date\" class=\"str-chat__message-simple-timestamp\">\n            {{ parsedDate }}\n          </span>\n        </div>\n      </div>\n    </ng-container>\n  </ng-container>\n</div>\n\n<ng-template #sendingStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"sending-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Sending...\" | translate }}\n    </div>\n    <stream-loading-indicator-placeholder\n      data-testid=\"loading-indicator\"\n    ></stream-loading-indicator-placeholder>\n  </span>\n</ng-template>\n<ng-template #readStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"read-indicator\"\n  >\n    <div class=\"str-chat__tooltip\" data-testid=\"read-by-tooltip\">\n      {{ readByText }}\n    </div>\n    <stream-avatar-placeholder\n      class=\"str-chat-angular__avatar-host\"\n      data-test-id=\"last-read-user-avatar\"\n      [size]=\"15\"\n      [imageUrl]=\"lastReadUser?.image\"\n      [name]=\"lastReadUser?.name || lastReadUser?.id\"\n      type=\"user\"\n      location=\"message-reader\"\n      [user]=\"lastReadUser\"\n    ></stream-avatar-placeholder>\n    <span\n      data-test-id=\"read-by-length\"\n      *ngIf=\"isReadByMultipleUsers\"\n      class=\"str-chat__message-simple-status-number\"\n    >\n      {{ (message?.readBy)!.length }}\n    </span>\n  </span>\n</ng-template>\n<ng-template #deliveredStatus>\n  <span\n    *ngIf=\"mode === 'main'\"\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"delivered-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Delivered\" | translate }}\n    </div>\n    <stream-icon-placeholder\n      data-testid=\"delivered-icon\"\n      icon=\"delivered-icon\"\n    ></stream-icon-placeholder>\n  </span>\n</ng-template>\n\n<ng-template #deletedMessage>\n  <div data-testid=\"message-deleted-component\">\n    <div class=\"str-chat__message--deleted-inner\" translate>\n      streamChat.This message was deleted...\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #systemMessage>\n  <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n    <div class=\"str-chat__message--system__text\">\n      <div class=\"str-chat__message--system__line\"></div>\n      <p>{{ message?.text }}</p>\n      <div class=\"str-chat__message--system__line\"></div>\n    </div>\n    <div class=\"str-chat__message--system__date\">\n      {{ parsedDate }}\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n  >\n    <stream-avatar-placeholder\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host\"\n      [imageUrl]=\"message?.quoted_message?.user?.image\"\n      [name]=\"\n        message?.quoted_message?.user?.name || message?.quoted_message?.user?.id\n      \"\n      [size]=\"20\"\n      type=\"user\"\n      location=\"quoted-message-sender\"\n      [user]=\"message?.quoted_message?.user || undefined\"\n    ></stream-avatar-placeholder>\n    <div class=\"quoted-message-inner\">\n      <ng-container\n        *ngIf=\"\n          message?.quoted_message?.attachments &&\n          message?.quoted_message?.attachments?.length\n        \"\n      >\n        <ng-template\n          #defaultAttachments\n          let-messageId=\"messageId\"\n          let-attachments=\"attachments\"\n          let-parentMessageId=\"parentMessageId\"\n        >\n          <stream-attachment-list\n            [messageId]=\"messageId\"\n            [attachments]=\"attachments\"\n            [parentMessageId]=\"parentMessageId\"\n          ></stream-attachment-list>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            attachmentListTemplate || defaultAttachments;\n            context: getQuotedMessageAttachmentListContext()\n          \"\n        ></ng-container>\n      </ng-container>\n      <div\n        data-testid=\"quoted-message-text\"\n        [innerHTML]=\"\n          message?.quoted_message?.html || message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
@@ -17,7 +17,7 @@ import transliterate from '@stream-io/transliterate';
|
|
|
17
17
|
import * as i7 from 'angular-mentions';
|
|
18
18
|
import { MentionModule } from 'angular-mentions';
|
|
19
19
|
|
|
20
|
-
const version = '3.
|
|
20
|
+
const version = '3.8.0';
|
|
21
21
|
|
|
22
22
|
/**
|
|
23
23
|
* The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
|
|
@@ -118,7 +118,7 @@ class ChatClientService {
|
|
|
118
118
|
* 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.
|
|
119
119
|
* @param apiKey
|
|
120
120
|
* @param userOrId
|
|
121
|
-
* @param userTokenOrProvider
|
|
121
|
+
* @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)
|
|
122
122
|
*/
|
|
123
123
|
init(apiKey, userOrId, userTokenOrProvider) {
|
|
124
124
|
var _a;
|
|
@@ -128,7 +128,10 @@ class ChatClientService {
|
|
|
128
128
|
let result;
|
|
129
129
|
yield this.ngZone.runOutsideAngular(() => __awaiter(this, void 0, void 0, function* () {
|
|
130
130
|
const user = typeof userOrId === 'string' ? { id: userOrId } : userOrId;
|
|
131
|
-
result =
|
|
131
|
+
result =
|
|
132
|
+
userTokenOrProvider === 'guest'
|
|
133
|
+
? yield this.chatClient.setGuestUser(user)
|
|
134
|
+
: yield this.chatClient.connectUser(user, userTokenOrProvider);
|
|
132
135
|
this.userSubject.next(this.chatClient.user);
|
|
133
136
|
this.chatClient.setUserAgent(`stream-chat-angular-${version}-${this.chatClient.getUserAgent()}`);
|
|
134
137
|
}));
|
|
@@ -875,13 +878,21 @@ class ChannelService {
|
|
|
875
878
|
}
|
|
876
879
|
messageUpdated(event) {
|
|
877
880
|
this.ngZone.run(() => {
|
|
881
|
+
var _a;
|
|
878
882
|
const isThreadReply = event.message && event.message.parent_id;
|
|
883
|
+
const channel = this.activeChannelSubject.getValue();
|
|
884
|
+
if (!channel) {
|
|
885
|
+
return;
|
|
886
|
+
}
|
|
887
|
+
// Get messages from state as message order could change, and message could've been deleted
|
|
879
888
|
const messages = isThreadReply
|
|
880
|
-
?
|
|
881
|
-
:
|
|
882
|
-
|
|
883
|
-
|
|
884
|
-
|
|
889
|
+
? channel.state.threads[((_a = event === null || event === void 0 ? void 0 : event.message) === null || _a === void 0 ? void 0 : _a.parent_id) || '']
|
|
890
|
+
: channel.state.messages;
|
|
891
|
+
if (!messages) {
|
|
892
|
+
return;
|
|
893
|
+
}
|
|
894
|
+
const messageIndex = messages.findIndex((m) => { var _a; return m.id === ((_a = event === null || event === void 0 ? void 0 : event.message) === null || _a === void 0 ? void 0 : _a.id); });
|
|
895
|
+
if (messageIndex !== -1) {
|
|
885
896
|
isThreadReply
|
|
886
897
|
? this.activeThreadMessagesSubject.next([...messages])
|
|
887
898
|
: this.activeChannelMessagesSubject.next([...messages]);
|
|
@@ -3496,9 +3507,9 @@ class MessageComponent {
|
|
|
3496
3507
|
this.visibleMessageActionsCount = 0;
|
|
3497
3508
|
this.messageTextParts = [];
|
|
3498
3509
|
this.subscriptions = [];
|
|
3499
|
-
this.user = this.chatClientService.chatClient.user;
|
|
3500
3510
|
}
|
|
3501
3511
|
ngOnInit() {
|
|
3512
|
+
this.subscriptions.push(this.chatClientService.user$.subscribe((u) => (this.user = u)));
|
|
3502
3513
|
this.subscriptions.push(this.customTemplatesService.mentionTemplate$.subscribe((template) => (this.mentionTemplate = template)));
|
|
3503
3514
|
this.subscriptions.push(this.customTemplatesService.attachmentListTemplate$.subscribe((template) => (this.attachmentListTemplate = template)));
|
|
3504
3515
|
this.subscriptions.push(this.customTemplatesService.messageActionsBoxTemplate$.subscribe((template) => (this.messageActionsBoxTemplate = template)));
|