stream-chat-angular 4.47.1 → 4.47.3
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 +25 -8
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/channel.service.js +21 -2
- package/esm2015/lib/chat-client.service.js +3 -2
- package/esm2015/lib/message/message.component.js +2 -4
- package/fesm2015/stream-chat-angular.js +28 -11
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/package.json +1 -1
- package/src/assets/version.ts +1 -1
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import { __awaiter } from "tslib";
|
|
2
2
|
import { Injectable } from '@angular/core';
|
|
3
|
-
import { BehaviorSubject, ReplaySubject
|
|
3
|
+
import { BehaviorSubject, ReplaySubject } from 'rxjs';
|
|
4
4
|
import { StreamChat } from 'stream-chat';
|
|
5
5
|
import { version } from '../assets/version';
|
|
6
|
+
import { take } from 'rxjs/operators';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "./notification.service";
|
|
8
9
|
/**
|
|
@@ -205,4 +206,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
205
206
|
providedIn: 'root',
|
|
206
207
|
}]
|
|
207
208
|
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i1.NotificationService }]; } });
|
|
208
|
-
//# 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, take } 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<OwnUserResponse<T> | 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<\n    OwnUserResponse<T> | UserResponse<T> | undefined\n  >(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 you can emit this for anonymous logins\n   * @param userTokenOrProvider You can provide:<ul>\n   *  <li> a token, </li>\n   *  <li> the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users) </li>\n   *  <li> the keyword 'anonymous' to connect as [anonymous user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#anonymous-users) </li>\n   *  </ul>\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> | undefined,\n    userTokenOrProvider: TokenOrProvider | 'anonymous' | '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 = await (\n          {\n            guest: () => this.chatClient.setGuestUser(user!),\n            anonymous: () => this.chatClient.connectAnonymousUser(),\n            // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n          }[`${userTokenOrProvider}`] ??\n          (() => this.chatClient.connectUser(user!, userTokenOrProvider))\n        )();\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(\n        this.chatClient.user ? { ...this.chatClient.user } : undefined\n      );\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.updateUser(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  private updateUser(e: Event<T>) {\n    if (typeof e.total_unread_count !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.total_unread_count !== e.total_unread_count) {\n        this.userSubject.next({\n          ...user,\n          total_unread_count: e.total_unread_count,\n        });\n      }\n    }\n    if (typeof e.unread_channels !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.unread_channels !== e.unread_channels) {\n        this.userSubject.next({\n          ...user,\n          unread_channels: e.unread_channels,\n        });\n      }\n    }\n    if (typeof e.unread_count !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.unread_count !== e.unread_count) {\n        this.userSubject.next({\n          ...user,\n          unread_count: e.unread_count,\n        });\n      }\n    }\n    if (\n      e.type === 'user.updated' &&\n      this.chatClient.user &&\n      e.user?.id === this.chatClient.user.id\n    ) {\n      this.userSubject.next({ ...this.chatClient.user });\n    }\n  }\n}\n"]}
|
|
209
|
+
//# 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":";AAAA,OAAO,EAAE,UAAU,EAAU,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,eAAe,EAAc,aAAa,EAAE,MAAM,MAAM,CAAC;AAWlE,OAAO,EAAsB,UAAU,EAAmB,MAAM,aAAa,CAAC;AAC9E,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAG5C,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;;;AAStC;;GAEG;AAIH,MAAM,OAAO,iBAAiB;IA2C5B,YACU,MAAc,EACd,mBAAwC;QADxC,WAAM,GAAN,MAAM,CAAQ;QACd,wBAAmB,GAAnB,mBAAmB,CAAqB;QAf1C,wBAAmB,GAAG,IAAI,aAAa,CAAiB,CAAC,CAAC,CAAC;QAC3D,2BAAsB,GAAG,IAAI,aAAa,CAAuB,CAAC,CAAC,CAAC;QACpE,uBAAkB,GAAG,IAAI,eAAe,CAC9C,SAAS,CACV,CAAC;QACM,0BAAqB,GAAG,IAAI,eAAe,CAEjD,EAAE,CAAC,CAAC;QACE,gBAAW,GAAG,IAAI,aAAa,CAErC,CAAC,CAAC,CAAC;QACG,kBAAa,GAAkC,EAAE,CAAC;QAMxD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,mBAAmB,CAAC,YAAY,EAAE,CAAC;QACvD,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,sBAAsB,CAAC,YAAY,EAAE,CAAC;QACnE,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC3D,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QACjE,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,EAAE,CAAC;IAC/C,CAAC;IAED;;;;;;;;;;OAUG;IACG,IAAI,CACR,MAAc,EACd,QAAmE,EACnE,mBAA4D,EAC5D,aAAiC;;;YAEjC,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC,WAAW,CAAI,MAAM,EAAE,aAAa,CAAC,CAAC;YACnE,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;YACzB,IAAI,MAAM,CAAC;YACX,MAAM,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAS,EAAE;;gBAC7C,MAAM,IAAI,GAAG,OAAO,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,EAAE,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC;gBACxE,IAAI;oBACF,MAAM,GAAG,MAAM,CACb,MAAA;wBACE,KAAK,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,IAAK,CAAC;wBAChD,SAAS,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,oBAAoB,EAAE;wBACvD,4EAA4E;qBAC7E,CAAC,GAAG,mBAAmB,EAAE,CAAC,mCAC3B,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,IAAK,EAAE,mBAAmB,CAAC,CAAC,CAChE,EAAE,CAAC;iBACL;gBAAC,OAAO,KAAK,EAAE;oBACd,IAAI,CAAC,mBAAmB,CAAC,wBAAwB,CAC/C,qEAAqE,EACrE,OAAO,CACR,CAAC;oBACF,MAAM,KAAK,CAAC;iBACb;gBACD,IAAI,CAAC,WAAW,CAAC,IAAI,CACnB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,mBAAM,IAAI,CAAC,UAAU,CAAC,IAAI,EAAG,CAAC,CAAC,SAAS,CAC/D,CAAC;gBACF,MAAM,SAAS,GAAG,qBAAqB,CAAC;gBACxC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE;oBACvD,IAAI,CAAC,UAAU,CAAC,YAAY,CAC1B,GAAG,SAAS,IAAI,OAAO,IAAI,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE,EAAE,CAC5D,CAAC;iBACH;YACH,CAAC,CAAA,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,aAAa,CAClD,EAAE,MAAM,EAAE,SAAS,EAA8B,EAAE,2CAA2C;YAC9F,EAAE,EACF,EAAE,OAAO,EAAE,MAAA,IAAI,CAAC,UAAU,CAAC,IAAI,0CAAE,EAAE,EAAE,CACtC,CAAC;YACF,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC1C,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACxC,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE;gBACvB,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;gBACnB,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC;gBAC7B,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC;oBAC5B,SAAS,EAAE,CAAC,CAAC,IAAI;oBACjB,KAAK,EAAE,CAAC;iBACT,CAAC,CAAC;YACL,CAAC,CAAC,CACH,CAAC;YACF,IAAI,kBAAwC,CAAC;YAC7C,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,oBAAoB,EAAE,CAAC,CAAC,EAAE,EAAE;gBAC7C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,MAAM,QAAQ,GAAG,CAAC,CAAC,MAAM,CAAC;oBAC1B,IAAI,QAAQ,EAAE;wBACZ,IAAI,kBAAkB,EAAE;4BACtB,kBAAkB,EAAE,CAAC;yBACtB;qBACF;yBAAM;wBACL,kBAAkB;4BAChB,IAAI,CAAC,mBAAmB,CAAC,wBAAwB,CAC/C,oDAAoD,CACrD,CAAC;qBACL;oBACD,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;gBACpE,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CACH,CAAC;YACF,OAAO,MAAM,CAAC;;KACf;IAED;;OAEG;IACG,cAAc;;YAClB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACpC,MAAM,IAAI,CAAC,UAAU,CAAC,cAAc,EAAE,CAAC;YACvC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACjC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;QACrD,CAAC;KAAA;IAED;;OAEG;IACG,cAAc;;YAClB,IAAI,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,EAAE;gBACtC,OAAO;aACR;YACD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,cAAc,EAAE,CAAC;YACxD,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAE,QAAQ,CAAC,GAAmB,IAAI,EAAE,CAAC,CAAC;QACpE,CAAC;KAAA;IAED;;;OAGG;IACG,WAAW,CAAC,SAAiB;;YACjC,MAAM,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAC/C,CAAC;KAAA;IAED;;;;OAIG;IACG,iBAAiB,CAAC,UAAkB;;YACxC,IAAI,CAAC,UAAU,EAAE;gBACf,OAAO,EAAE,CAAC;aACX;YACD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC;gBAC9C,GAAG,EAAE;oBACH,EAAE,EAAE,EAAE,EAAE,aAAa,EAAE,UAAU,EAAE,EAAE;oBACrC,EAAE,IAAI,EAAE,EAAE,aAAa,EAAE,UAAU,EAAE,EAAE;iBACxC;gBACD,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,UAAU,CAAC,MAAO,EAAE;aACnB,CAAC,CAAC,CAAC,2CAA2C;YACjE,OAAO,MAAM,CAAC,KAAK,CAAC;QACtB,CAAC;KAAA;IAEO,oBAAoB,CAAC,CAAW;;QACtC,IAAI,CAAA,MAAA,MAAA,CAAC,CAAC,MAAM,0CAAE,IAAI,0CAAE,EAAE,OAAK,MAAA,IAAI,CAAC,UAAU,CAAC,IAAI,0CAAE,EAAE,CAAA,IAAI,CAAC,CAAC,OAAO,EAAE;YAChE,MAAM,cAAc,GAAG,IAAI,CAAC,qBAAqB,CAAC,QAAQ,EAAE,CAAC;YAC7D,IAAI,CAAC,CAAC,IAAI,KAAK,sBAAsB,EAAE;gBACrC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,GAAG,cAAc,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;aACjE;iBAAM,IACL,CAAC,CAAC,IAAI,KAAK,8BAA8B;gBACzC,CAAC,CAAC,IAAI,KAAK,8BAA8B,EACzC;gBACA,MAAM,KAAK,GAAG,cAAc,CAAC,SAAS,CACpC,CAAC,CAAC,EAAE,EAAE,WAAC,OAAA,CAAA,CAAC,aAAD,CAAC,uBAAD,CAAC,CAAE,GAAG,OAAK,MAAA,CAAC,CAAC,OAAO,0CAAE,GAAG,CAAA,CAAA,EAAA,CACjC,CAAC;gBACF,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE;oBAChB,cAAc,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;oBAChC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,GAAG,cAAc,CAAC,CAAC,CAAC;iBACtD;aACF;SACF;IACH,CAAC;IAEO,UAAU,CAAC,CAAW;;QAC5B,IAAI,OAAO,CAAC,CAAC,kBAAkB,KAAK,WAAW,EAAE;YAC/C,IAAI,IAAsD,CAAC;YAC3D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC7C,IAAI,GAAG,CAAC,CAAC;YACX,CAAC,CAAC,CAAC;YACH,IAAI,IAAI,IAAI,IAAI,CAAC,kBAAkB,KAAK,CAAC,CAAC,kBAAkB,EAAE;gBAC5D,IAAI,CAAC,WAAW,CAAC,IAAI,iCAChB,IAAI,KACP,kBAAkB,EAAE,CAAC,CAAC,kBAAkB,IACxC,CAAC;aACJ;SACF;QACD,IAAI,OAAO,CAAC,CAAC,eAAe,KAAK,WAAW,EAAE;YAC5C,IAAI,IAAsD,CAAC;YAC3D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC7C,IAAI,GAAG,CAAC,CAAC;YACX,CAAC,CAAC,CAAC;YACH,IAAI,IAAI,IAAI,IAAI,CAAC,eAAe,KAAK,CAAC,CAAC,eAAe,EAAE;gBACtD,IAAI,CAAC,WAAW,CAAC,IAAI,iCAChB,IAAI,KACP,eAAe,EAAE,CAAC,CAAC,eAAe,IAClC,CAAC;aACJ;SACF;QACD,IAAI,OAAO,CAAC,CAAC,YAAY,KAAK,WAAW,EAAE;YACzC,IAAI,IAAsD,CAAC;YAC3D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC7C,IAAI,GAAG,CAAC,CAAC;YACX,CAAC,CAAC,CAAC;YACH,IAAI,IAAI,IAAI,IAAI,CAAC,YAAY,KAAK,CAAC,CAAC,YAAY,EAAE;gBAChD,IAAI,CAAC,WAAW,CAAC,IAAI,iCAChB,IAAI,KACP,YAAY,EAAE,CAAC,CAAC,YAAY,IAC5B,CAAC;aACJ;SACF;QACD,IACE,CAAC,CAAC,IAAI,KAAK,cAAc;YACzB,IAAI,CAAC,UAAU,CAAC,IAAI;YACpB,CAAA,MAAA,CAAC,CAAC,IAAI,0CAAE,EAAE,MAAK,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,EACtC;YACA,IAAI,CAAC,WAAW,CAAC,IAAI,mBAAM,IAAI,CAAC,UAAU,CAAC,IAAI,EAAG,CAAC;SACpD;IACH,CAAC;;8GA7PU,iBAAiB;kHAAjB,iBAAiB,cAFhB,MAAM;2FAEP,iBAAiB;kBAH7B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","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';\nimport { take } from 'rxjs/operators';\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<OwnUserResponse<T> | 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<\n    OwnUserResponse<T> | UserResponse<T> | undefined\n  >(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 you can emit this for anonymous logins\n   * @param userTokenOrProvider You can provide:<ul>\n   *  <li> a token, </li>\n   *  <li> the keyword 'guest' to connect as [guest user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#guest-users) </li>\n   *  <li> the keyword 'anonymous' to connect as [anonymous user](https://getstream.io/chat/docs/javascript/authless_users/?language=javascript#anonymous-users) </li>\n   *  </ul>\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> | undefined,\n    userTokenOrProvider: TokenOrProvider | 'anonymous' | '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 = await (\n          {\n            guest: () => this.chatClient.setGuestUser(user!),\n            anonymous: () => this.chatClient.connectAnonymousUser(),\n            // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n          }[`${userTokenOrProvider}`] ??\n          (() => this.chatClient.connectUser(user!, userTokenOrProvider))\n        )();\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(\n        this.chatClient.user ? { ...this.chatClient.user } : undefined\n      );\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.updateUser(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  private updateUser(e: Event<T>) {\n    if (typeof e.total_unread_count !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.total_unread_count !== e.total_unread_count) {\n        this.userSubject.next({\n          ...user,\n          total_unread_count: e.total_unread_count,\n        });\n      }\n    }\n    if (typeof e.unread_channels !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.unread_channels !== e.unread_channels) {\n        this.userSubject.next({\n          ...user,\n          unread_channels: e.unread_channels,\n        });\n      }\n    }\n    if (typeof e.unread_count !== 'undefined') {\n      let user: OwnUserResponse<T> | UserResponse<T> | undefined;\n      this.userSubject.pipe(take(1)).subscribe((u) => {\n        user = u;\n      });\n      if (user && user.unread_count !== e.unread_count) {\n        this.userSubject.next({\n          ...user,\n          unread_count: e.unread_count,\n        });\n      }\n    }\n    if (\n      e.type === 'user.updated' &&\n      this.chatClient.user &&\n      e.user?.id === this.chatClient.user.id\n    ) {\n      this.userSubject.next({ ...this.chatClient.user });\n    }\n  }\n}\n"]}
|
|
@@ -134,9 +134,7 @@ export class MessageComponent {
|
|
|
134
134
|
}
|
|
135
135
|
if (changes.message || changes.enabledMessageActions || changes.mode) {
|
|
136
136
|
this.shouldDisplayThreadLink =
|
|
137
|
-
!!((_g = this.message) === null || _g === void 0 ? void 0 : _g.reply_count) &&
|
|
138
|
-
this.mode !== 'thread' &&
|
|
139
|
-
this.enabledMessageActions.indexOf('send-reply') !== -1;
|
|
137
|
+
!!((_g = this.message) === null || _g === void 0 ? void 0 : _g.reply_count) && this.mode !== 'thread';
|
|
140
138
|
}
|
|
141
139
|
if (changes.message || changes.mode) {
|
|
142
140
|
this.areOptionsVisible = this.message
|
|
@@ -380,4 +378,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
380
378
|
type: ViewChild,
|
|
381
379
|
args: ['container']
|
|
382
380
|
}] } });
|
|
383
|
-
//# 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":"AAAA,OAAO,EACL,SAAS,EAET,KAAK,EAGL,SAAS,EAIT,uBAAuB,GAGxB,MAAM,eAAe,CAAC;AAiBvB,OAAO,UAAU,MAAM,aAAa,CAAC;AAGrC,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAE1C,OAAO,EAAE,mBAAmB,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;;;;;;;;;;;;;;;;AAS1E;;GAEG;AAOH,MAAM,OAAO,gBAAgB;IAoE3B,YACU,iBAAoC,EACpC,cAA8B,EAC/B,sBAA8C,EAC7C,KAAwB,EAChC,YAA0B,EAClB,UAA6B,EAC7B,MAAc;QANd,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,mBAAc,GAAd,cAAc,CAAgB;QAC/B,2BAAsB,GAAtB,sBAAsB,CAAwB;QAC7C,UAAK,GAAL,KAAK,CAAmB;QAExB,eAAU,GAAV,UAAU,CAAmB;QAC7B,WAAM,GAAN,MAAM,CAAQ;QApExB;;WAEG;QACM,0BAAqB,GAAa,EAAE,CAAC;QAK9C;;WAEG;QACM,SAAI,GAAsB,MAAM,CAAC;QAC1C;;WAEG;QACM,kBAAa,GAAG,KAAK,CAAC;QAC/B;;WAEG;QACM,kBAAa,GAA8B,EAAE,CAAC;QAKvD,oBAAe,GAAG,KAAK,CAAC;QACxB,2BAAsB,GAAG,KAAK,CAAC;QAC/B,+BAA0B,GAAG,CAAC,CAAC;QAC/B,qBAAgB,GAA8B,EAAE,CAAC;QAEjD,uBAAkB,GAAG,mBAAmB,CAAC,KAAK,CAAC;QAC/C,uBAAkB,GAAG,mBAAmB,CAAC,KAAK,CAAC;QAC/C,wBAAmB,GAAG,qBAAqB,CAAC,IAAI,CAAC;QACjD,mCAA8B,GAAG,KAAK,CAAC;QACvC,gCAA2B,GAA+B,UAAU,CAAC;QACrE,8BAAyB,GAAwB,QAAQ,CAAC;QAC1D,4BAAuB,GAAG,KAAK,CAAC;QAChC,wBAAmB,GAAG,KAAK,CAAC;QAC5B,eAAU,GAAG,EAAE,CAAC;QAChB,iBAAY,GAAwD,SAAS,CAAC;QAC9E,mBAAc,GAAG,KAAK,CAAC;QACvB,0BAAqB,GAAG,KAAK,CAAC;QAC9B,8BAAyB,GAAG,KAAK,CAAC;QAClC,eAAU,GAAG,EAAE,CAAC;QAChB,sBAAiB,GAAG,KAAK,CAAC;QAC1B,kBAAa,GAAG,KAAK,CAAC;QACtB,iBAAY,GAAG,KAAK,CAAC;QACrB,oBAAe,GAAuC;YACpD,UAAU,EAAE,SAAS;SACtB,CAAC;QACF,yBAAoB,GAAG,KAAK,CAAC;QAG7B,8BAAyB,GAAG,KAAK,CAAC;QAC1B,kBAAa,GAAmB,EAAE,CAAC;QAI1B,cAAS,GACxB,gKAAgK,CAAC;QAC3J,gBAAW,GAAG,IAAI,MAAM,CAAC,UAAU,EAAE,EAAE,GAAG,CAAC,CAAC;QAWlD,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC,YAAY,CAAC;IAChD,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YAC3C,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,EAAE;gBACnB,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC;gBACd,IAAI,CAAC,sBAAsB,EAAE,CAAC;gBAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;gBACvB,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;aAC5B;QACH,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,WAAW,CAAC,OAAsB;;QAChC,IAAI,OAAO,CAAC,OAAO,EAAE;YACnB,IAAI,CAAC,8BAA8B,GAAG,KAAK,CAAC;YAC5C,IAAI,CAAC,2BAA2B,GAAG,UAAU,CAAC;YAC9C,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,MAAM,mBAAmB,GAAG,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,cAAc,0CAAE,WAAW,CAAC;YACtE,IAAI,CAAC,wBAAwB;gBAC3B,mBAAmB,IAAI,mBAAmB,CAAC,MAAM;oBAC/C,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC;oBAC1B,CAAC,CAAC,EAAE,CAAC;YACT,IAAI,CAAC,sBAAsB,EAAE,CAAC;YAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,UAAU,GAAG,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,MAAM;gBACpC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC;gBAChC,CAAC,CAAC,EAAE,CAAC;YACP,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC,CACtB,IAAI,CAAC,OAAO;gBACZ,IAAI,CAAC,OAAO,CAAC,MAAM;gBACnB,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,CACjC,CAAC;YACF,IAAI,CAAC,qBAAqB,GAAG,CAAC,CAAC,CAC7B,IAAI,CAAC,OAAO;gBACZ,IAAI,CAAC,OAAO,CAAC,MAAM;gBACnB,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAC/B,CAAC;YACF,IAAI,CAAC,yBAAyB,GAAG,CAAC,CAAC,CACjC,IAAI,CAAC,OAAO;gBACZ,IAAI,CAAC,OAAO,CAAC,MAAM;gBACnB,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,UAAU;gBAClC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAC/B,CAAC;YACF,IAAI,CAAC,UAAU;gBACb,CAAC,IAAI,CAAC,OAAO;oBACX,IAAI,CAAC,OAAO,CAAC,UAAU;oBACvB,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;oBACzD,EAAE,CAAC;YACL,IAAI,CAAC,aAAa;gBAChB,CAAC,CAAC,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,WAAW,CAAA,IAAI,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC;YACnE,IAAI,CAAC,YAAY;gBACf,CAAC,CAAC,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,eAAe,CAAA;oBAC/B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC;YACvD,IAAI,CAAC,eAAe,GAAG,EAAE,UAAU,EAAE,MAAA,IAAI,CAAC,OAAO,0CAAE,WAAW,EAAE,CAAC;SAClE;QACD,IAAI,OAAO,CAAC,qBAAqB,EAAE;YACjC,IAAI,CAAC,iBAAiB;gBACpB,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC;YAC7D,IAAI,CAAC,oBAAoB;gBACvB,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC;YAC3D,IAAI,CAAC,oBAAoB;gBACvB,IAAI,CAAC,oBAAoB,KAAK,KAAK;oBACnC,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC;SAC5D;QACD,IAAI,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,qBAAqB,IAAI,OAAO,CAAC,IAAI,EAAE;YACpE,IAAI,CAAC,uBAAuB;gBAC1B,CAAC,CAAC,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,WAAW,CAAA;oBAC3B,IAAI,CAAC,IAAI,KAAK,QAAQ;oBACtB,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;SAC3D;QACD,IAAI,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,IAAI,EAAE;YACnC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,OAAO;gBACnC,CAAC,CAAC,CAAC,CACC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI;oBAClB,IAAI,CAAC,OAAO,CAAC,IAAI,KAAK,OAAO;oBAC7B,IAAI,CAAC,OAAO,CAAC,IAAI,KAAK,QAAQ;oBAC9B,IAAI,CAAC,OAAO,CAAC,IAAI,KAAK,WAAW;oBACjC,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,QAAQ;oBAChC,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,SAAS;oBACjC,CAAC,IAAI,CAAC,IAAI,KAAK,QAAQ,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CACpD;gBACH,CAAC,CAAC,KAAK,CAAC;SACX;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;;YACjC,MAAA,IAAI,CAAC,SAAS,0CAAE,aAAa,CAAC,gBAAgB,CAAC,YAAY,EAAE,GAAG,EAAE,CAChE,IAAI,CAAC,SAAS,EAAE,CACjB,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,yBAAyB,GAAG,IAAI,CAAC;YACtC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;QAC7B,CAAC,EAAE,CAAC,CAAC,CAAC;IACR,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,wBAAwB;;QACtB,OAAO;YACL,SAAS,EAAE,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,EAAE,KAAI,EAAE;YACjC,WAAW,EAAE,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,WAAW,KAAI,EAAE;YAC5C,eAAe,EAAE,MAAA,IAAI,CAAC,OAAO,0CAAE,SAAS;YACxC,4BAA4B,EAAE,CAAC,KAAK,EAAE,EAAE,CACtC,CAAC,IAAI,CAAC,yBAAyB,GAAG,KAAK,CAAC;SAC3C,CAAC;IACJ,CAAC;IAED,iBAAiB;QACf,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,qBAAqB,EAAE,IAAI,CAAC,qBAAqB;YACjD,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;YACzC,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,UAAU,EAAE,IAAI,CAAC,UAAU;SAC5B,CAAC;IACJ,CAAC;IAED,qCAAqC;;QACnC,OAAO;YACL,SAAS,EAAE,CAAA,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,cAAc,0CAAE,EAAE,KAAI,EAAE;YACjD,WAAW,EAAE,IAAI,CAAC,wBAAyB;YAC3C,eAAe,EAAE,MAAA,MAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,OAAO,0CAAE,cAAc,0CAAE,SAAS;SAC1D,CAAC;IACJ,CAAC;IAED,0BAA0B;;QACxB,OAAO;YACL,qBAAqB,EAAE,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,eAAe,KAAI,EAAE;YAC1D,eAAe,EAAE,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,gBAAgB,KAAI,EAAE;YACrD,cAAc,EAAE,IAAI,CAAC,sBAAsB;YAC3C,2BAA2B,EAAE,CAAC,MAAM,EAAE,EAAE,CACtC,CAAC,IAAI,CAAC,sBAAsB,GAAG,MAAM,CAAC;YACxC,SAAS,EAAE,MAAA,IAAI,CAAC,OAAO,0CAAE,EAAE;YAC3B,YAAY,EAAE,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,aAAa,KAAI,EAAE;SAChD,CAAC;IACJ,CAAC;IAED,aAAa;QACX,KAAK,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,OAAQ,CAAC,CAAC;IACxD,CAAC;IAED,wBAAwB;QACtB,KAAK,IAAI,CAAC,cAAc,CAAC,wBAAwB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAClE,CAAC;IAED,iBAAiB,CAAC,WAAwB;QACxC,OAAO;YACL,OAAO,EAAE,WAAW,CAAC,OAAO;YAC5B,IAAI,EAAE,WAAW,CAAC,IAAK;SACxB,CAAC;IACJ,CAAC;IAED,2BAA2B;QACzB,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,eAAe;YAC5B,MAAM,EAAE,IAAI,CAAC,mBAAmB;YAChC,cAAc,EAAE,IAAI,CAAC,qBAAqB;YAC1C,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,gCAAgC,EAAE,CAAC,KAAK,EAAE,EAAE;gBAC1C,IAAI,CAAC,0BAA0B,GAAG,KAAK,CAAC;gBACxC,4FAA4F;gBAC5F,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YAC7B,CAAC;YACD,kCAAkC,EAAE,CAAC,KAAK,EAAE,EAAE;gBAC5C,IAAI,CAAC,0BAA0B,GAAG,KAAK,CAAC;gBACxC,4FAA4F;gBAC5F,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YAC7B,CAAC;YACD,sBAAsB,EAAE,CAAC,SAAS,EAAE,EAAE;gBACpC,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;gBAC3B,IAAI,CAAC,eAAe,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC;YACzC,CAAC;YACD,aAAa,EAAE,IAAI,CAAC,aAAa,IAAI,EAAE;SACxC,CAAC;IACJ,CAAC;IAED,yBAAyB;QACvB,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,OAAQ;SACvB,CAAC;IACJ,CAAC;IAED,uBAAuB;QACrB,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,OAAQ;SACvB,CAAC;IACJ,CAAC;IAED,oBAAoB;QAClB,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,OAAQ;YACtB,UAAU,EAAE,IAAI,CAAC,UAAU;SAC5B,CAAC;IACJ,CAAC;IAED,aAAa,CAAC,SAAiB,EAAE,eAAwB;QACvD,KAAK,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;IACrE,CAAC;IAED,wBAAwB;QACtB,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,SAAS;QACP,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;YAC/B,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAEO,kBAAkB,CAAC,eAAe,GAAG,IAAI;QAC/C,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;QAClC,IAAI,CAAC,WAAW,GAAG,SAAS,CAAC;QAC7B,IAAI,OAAO,GAAG,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAAC,CAAC;QACtD,IACE,CAAC,CAAC,IAAI,CAAC,OAAQ,CAAC,eAAe;YAC7B,IAAI,CAAC,OAAQ,CAAC,eAAe,CAAC,MAAM,KAAK,CAAC,CAAC;YAC7C,CAAC,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAA;YACjC,CAAC,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA,EAC/B;YACA,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;YAClC,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC;YAC3B,OAAO;SACR;QACD,IAAI,CAAC,OAAO,EAAE;YACZ,OAAO;SACR;QACD,IACE,CAAC,IAAI,CAAC,OAAQ,CAAC,eAAe;YAC9B,IAAI,CAAC,OAAQ,CAAC,eAAe,CAAC,MAAM,KAAK,CAAC,EAC1C;YACA,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;YACxC,OAAO,GAAG,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;YAC/C,IAAI,CAAC,gBAAgB,GAAG,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;SACrD;aAAM;YACL,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAC;YAC3B,IAAI,IAAI,GAAG,OAAO,CAAC;YACnB,IAAI,CAAC,OAAQ,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;gBAC7C,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,EAAE,EAAE,CAAC;gBAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC/D,IAAI,sBAAsB,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;gBACjE,sBAAsB,GAAG,IAAI,CAAC,sBAAsB,CAClD,sBAAsB,CACvB,CAAC;gBACF,IAAI,CAAC,gBAAiB,CAAC,IAAI,CAAC;oBAC1B,OAAO,EAAE,sBAAsB;oBAC/B,IAAI,EAAE,MAAM;iBACb,CAAC,CAAC;gBACH,IAAI,CAAC,gBAAiB,CAAC,IAAI,CAAC;oBAC1B,OAAO,EAAE,OAAO;oBAChB,IAAI,EAAE,SAAS;oBACf,IAAI;iBACL,CAAC,CAAC;gBACH,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,aAAa,GAAG,OAAO,EAAE,EAAE,CAAC,CAAC;YACnD,CAAC,CAAC,CAAC;YACH,IAAI,IAAI,EAAE;gBACR,IAAI,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;gBAClC,IAAI,GAAG,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,CAAC;gBACzC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;aAC7D;SACF;IACH,CAAC;IAEO,iBAAiB,CAAC,eAAwB;;QAChD,MAAM,eAAe,GAAG,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,CAAC;QAC3C,IAAI,eAAe,EAAE;YACnB,MAAM,WAAW,GAAG,MAAA,IAAI,CAAC,OAAO,0CAAE,WAAW,CAAC;YAC9C,IAAI,WAAW,EAAE;gBACf,IAAI,CAAC,8BAA8B,GAAG,IAAI,CAAC;gBAC3C,IAAI,CAAC,2BAA2B,GAAG,aAAa,CAAC;aAClD;YACD,OAAO,WAAW,IAAI,eAAe,CAAC;SACvC;aAAM;YACL,IAAI,CAAC,2BAA2B,GAAG,UAAU,CAAC;YAC9C,OAAO,eAAe,CAAC;SACxB;IACH,CAAC;IAEO,eAAe,CAAC,OAAe;QACrC,yHAAyH;QACzH,sHAAsH;QACtH,+DAA+D;QAC/D,MAAM,QAAQ,GACZ,CAAC,CAAE,MAAc,CAAC,MAAM,IAAI,OAAQ,MAAc,CAAC,GAAG,KAAK,WAAW,CAAC;QACzE,8DAA8D;QAC9D,OAAO,GAAG,OAAO,CAAC,OAAO,CACvB,IAAI,CAAC,WAAW,EAChB,CAAC,KAAK,EAAE,EAAE,CACR,SACE,QAAQ,CAAC,CAAC,CAAC,qCAAqC,CAAC,CAAC,CAAC,EACrD,IAAI,KAAK,SAAS,CACrB,CAAC;QAEF,OAAO,OAAO,CAAC;IACjB,CAAC;IAEO,sBAAsB,CAAC,OAAe;QAC5C,OAAO,GAAG,OAAO,CAAC,OAAO,CACvB,IAAI,CAAC,SAAS,EACd,CAAC,KAAK,EAAE,EAAE,CAAC,YAAY,KAAK,oBAAoB,KAAK,MAAM,CAC5D,CAAC;QAEF,OAAO,OAAO,CAAC;IACjB,CAAC;IAEO,sBAAsB;;QAC5B,IAAI,CAAC,mBAAmB,GAAG,CAAA,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,EAAE,OAAK,MAAA,IAAI,CAAC,IAAI,0CAAE,EAAE,CAAA,CAAC;IACtE,CAAC;IAEO,eAAe;;QACrB,IAAI,CAAC,YAAY,GAAG,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,MAAM,0CAAE,MAAM,CAC9C,CAAC,CAAC,EAAE,EAAE,WAAC,OAAA,CAAC,CAAC,EAAE,MAAK,MAAA,IAAI,CAAC,IAAI,0CAAE,EAAE,CAAA,CAAA,EAAA,EAC7B,CAAC,CAAC,CAAC;IACP,CAAC;;6GArZU,gBAAgB;iGAAhB,gBAAgB,yXCrD7B,y6wBAspBA;2FDjmBa,gBAAgB;kBAN5B,SAAS;mBAAC;oBACT,QAAQ,EAAE,gBAAgB;oBAC1B,WAAW,EAAE,0BAA0B;oBACvC,MAAM,EAAE,EAAE;oBACV,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;0RAOU,OAAO;sBAAf,KAAK;gBAIG,qBAAqB;sBAA7B,KAAK;gBAIG,iBAAiB;sBAAzB,KAAK;gBAIG,IAAI;sBAAZ,KAAK;gBAIG,aAAa;sBAArB,KAAK;gBAIG,aAAa;sBAArB,KAAK;gBAmC0B,SAAS;sBAAxC,SAAS;uBAAC,WAAW","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  OnDestroy,\n  OnInit,\n  ChangeDetectorRef,\n  ChangeDetectionStrategy,\n  NgZone,\n  AfterViewInit,\n} from '@angular/core';\nimport { Attachment, UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport {\n  AttachmentListContext,\n  MentionTemplateContext,\n  MessageActionsBoxContext,\n  MessageReactionsContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n  DeliveredStatusContext,\n  SendingStatusContext,\n  ReadStatusContext,\n  CustomMessageActionItem,\n  SystemMessageContext,\n} from '../types';\nimport emojiRegex from 'emoji-regex';\nimport { Subscription } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\nimport { ThemeService } from '../theme.service';\nimport { NgxPopperjsTriggers, NgxPopperjsPlacements } from 'ngx-popperjs';\nimport { DateParserService } from '../date-parser.service';\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  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MessageComponent\n  implements OnInit, OnChanges, OnDestroy, AfterViewInit\n{\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  /**\n   * Highlighting is used to add visual emphasize to a message when jumping to the message\n   */\n  @Input() isHighlighted = false;\n  /**\n   * A list of custom message actions to be displayed in the action box\n   */\n  @Input() customActions: CustomMessageActionItem[] = [];\n  readonly themeVersion: '1' | '2';\n  canReceiveReadEvents: boolean | undefined;\n  canReactToMessage: boolean | undefined;\n  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] | undefined = [];\n  messageText?: string;\n  popperTriggerClick = NgxPopperjsTriggers.click;\n  popperTriggerHover = NgxPopperjsTriggers.hover;\n  popperPlacementAuto = NgxPopperjsPlacements.AUTO;\n  shouldDisplayTranslationNotice = false;\n  displayedMessageTextContent: 'original' | 'translation' = 'original';\n  imageAttachmentModalState: 'opened' | 'closed' = 'closed';\n  shouldDisplayThreadLink = false;\n  isSentByCurrentUser = false;\n  readByText = '';\n  lastReadUser: UserResponse<DefaultStreamChatGenerics> | undefined = undefined;\n  isOnlyReadByMe = false;\n  isReadByMultipleUsers = false;\n  isMessageDeliveredAndRead = false;\n  parsedDate = '';\n  areOptionsVisible = false;\n  hasAttachment = false;\n  hasReactions = false;\n  replyCountParam: { replyCount: number | undefined } = {\n    replyCount: undefined,\n  };\n  canDisplayReadStatus = false;\n  private quotedMessageAttachments: Attachment[] | undefined;\n  user: UserResponse<DefaultStreamChatGenerics> | undefined;\n  optionsRenderTimeoutEnded = false;\n  private subscriptions: Subscription[] = [];\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n  private readonly urlRegexp =\n    /(?:(?:https?|ftp|file):\\/\\/|www\\.|ftp\\.)(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[-A-Z0-9+&@#/%=~_|$?!:,.])*(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[A-Z0-9+&@#/%=~_|$])/gim;\n  private emojiRegexp = new RegExp(emojiRegex(), 'g');\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService,\n    public customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef,\n    themeService: ThemeService,\n    private dateParser: DateParserService,\n    private ngZone: NgZone\n  ) {\n    this.themeVersion = themeService.themeVersion;\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((u) => {\n        if (u !== this.user) {\n          this.user = u;\n          this.setIsSentByCurrentUser();\n          this.setLastReadUser();\n          this.cdRef.detectChanges();\n        }\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.shouldDisplayTranslationNotice = false;\n      this.displayedMessageTextContent = 'original';\n      this.createMessageParts();\n      const originalAttachments = this.message?.quoted_message?.attachments;\n      this.quotedMessageAttachments =\n        originalAttachments && originalAttachments.length\n          ? [originalAttachments[0]]\n          : [];\n      this.setIsSentByCurrentUser();\n      this.setLastReadUser();\n      this.readByText = this.message?.readBy\n        ? listUsers(this.message.readBy)\n        : '';\n      this.isOnlyReadByMe = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length === 0\n      );\n      this.isReadByMultipleUsers = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length > 1\n      );\n      this.isMessageDeliveredAndRead = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.status === 'received' &&\n        this.message.readBy.length > 0\n      );\n      this.parsedDate =\n        (this.message &&\n          this.message.created_at &&\n          this.dateParser.parseDateTime(this.message.created_at)) ||\n        '';\n      this.hasAttachment =\n        !!this.message?.attachments && !!this.message.attachments.length;\n      this.hasReactions =\n        !!this.message?.reaction_counts &&\n        Object.keys(this.message.reaction_counts).length > 0;\n      this.replyCountParam = { replyCount: this.message?.reply_count };\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      this.canDisplayReadStatus =\n        this.canReceiveReadEvents !== false &&\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n    }\n    if (changes.message || changes.enabledMessageActions || changes.mode) {\n      this.shouldDisplayThreadLink =\n        !!this.message?.reply_count &&\n        this.mode !== 'thread' &&\n        this.enabledMessageActions.indexOf('send-reply') !== -1;\n    }\n    if (changes.message || changes.mode) {\n      this.areOptionsVisible = this.message\n        ? !(\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        : false;\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.ngZone.runOutsideAngular(() => {\n      this.container?.nativeElement.addEventListener('mouseleave', () =>\n        this.mouseLeft()\n      );\n    });\n    setTimeout(() => {\n      this.optionsRenderTimeoutEnded = true;\n      this.cdRef.detectChanges();\n    }, 0);\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  getAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.id || '',\n      attachments: this.message?.attachments || [],\n      parentMessageId: this.message?.parent_id,\n      imageModalStateChangeHandler: (state) =>\n        (this.imageAttachmentModalState = state),\n    };\n  }\n\n  getMessageContext(): SystemMessageContext {\n    return {\n      message: this.message,\n      enabledMessageActions: this.enabledMessageActions,\n      isHighlighted: this.isHighlighted,\n      isLastSentMessage: this.isLastSentMessage,\n      mode: this.mode,\n      customActions: this.customActions,\n      parsedDate: this.parsedDate,\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  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      displayedActionsCountChangeHandler: (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      customActions: this.customActions || [],\n    };\n  }\n\n  getDeliveredStatusContext(): DeliveredStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getSendingStatusContext(): SendingStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getReadStatusContext(): ReadStatusContext {\n    return {\n      message: this.message!,\n      readByText: this.readByText,\n    };\n  }\n\n  jumpToMessage(messageId: string, parentMessageId?: string) {\n    void this.channelService.jumpToMessage(messageId, parentMessageId);\n  }\n\n  displayTranslatedMessage() {\n    this.createMessageParts(true);\n  }\n\n  displayOriginalMessage() {\n    this.createMessageParts(false);\n  }\n\n  mouseLeft() {\n    if (this.isActionBoxOpen) {\n      this.ngZone.run(() => {\n        this.isActionBoxOpen = false;\n      });\n    }\n  }\n\n  private createMessageParts(shouldTranslate = true) {\n    this.messageTextParts = undefined;\n    this.messageText = undefined;\n    let content = this.getMessageContent(shouldTranslate);\n    if (\n      (!this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0) &&\n      !content?.match(this.emojiRegexp) &&\n      !content?.match(this.urlRegexp)\n    ) {\n      this.messageTextParts = undefined;\n      this.messageText = content;\n      return;\n    }\n    if (!content) {\n      return;\n    }\n    if (\n      !this.message!.mentioned_users ||\n      this.message!.mentioned_users.length === 0\n    ) {\n      content = this.fixEmojiDisplay(content);\n      content = this.wrapLinskWithAnchorTag(content);\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        let formattedPrecedingText = this.fixEmojiDisplay(precedingText);\n        formattedPrecedingText = this.wrapLinskWithAnchorTag(\n          formattedPrecedingText\n        );\n        this.messageTextParts!.push({\n          content: formattedPrecedingText,\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        text = this.fixEmojiDisplay(text);\n        text = this.wrapLinskWithAnchorTag(text);\n        this.messageTextParts.push({ content: text, type: 'text' });\n      }\n    }\n  }\n\n  private getMessageContent(shouldTranslate: boolean) {\n    const originalContent = this.message?.text;\n    if (shouldTranslate) {\n      const translation = this.message?.translation;\n      if (translation) {\n        this.shouldDisplayTranslationNotice = true;\n        this.displayedMessageTextContent = 'translation';\n      }\n      return translation || originalContent;\n    } else {\n      this.displayedMessageTextContent = 'original';\n      return originalContent;\n    }\n  }\n\n  private fixEmojiDisplay(content: string) {\n    // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\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 && typeof (window as any).opr === 'undefined';\n    /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n    content = content.replace(\n      this.emojiRegexp,\n      (match) =>\n        `<span ${\n          isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n        }>${match}</span>`\n    );\n\n    return content;\n  }\n\n  private wrapLinskWithAnchorTag(content: string) {\n    content = content.replace(\n      this.urlRegexp,\n      (match) => `<a href=\"${match}\" rel=\"nofollow\">${match}</a>`\n    );\n\n    return content;\n  }\n\n  private setIsSentByCurrentUser() {\n    this.isSentByCurrentUser = this.message?.user?.id === this.user?.id;\n  }\n\n  private setLastReadUser() {\n    this.lastReadUser = this.message?.readBy?.filter(\n      (u) => u.id !== this.user?.id\n    )[0];\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--other]=\"!isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"hasReactions\"\n  [class.str-chat__message--highlighted]=\"isHighlighted\"\n  [class.str-chat__message-with-thread-link]=\"shouldDisplayThreadLink\"\n  [class.str-chat__message-send-can-be-retried]=\"\n    message?.status === 'failed' && message?.errorStatusCode !== 403\n  \"\n  data-testid=\"message-container\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container *ngIf=\"themeVersion === '1'\">\n        <ng-container *ngTemplateOutlet=\"messageStatus\"></ng-container>\n      </ng-container>\n      <stream-avatar-placeholder\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\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 str-chat__message-options\"\n          data-testid=\"message-options\"\n          [class.str-chat__message-edit-in-progress]=\"isEditing\"\n          *ngIf=\"areOptionsVisible && optionsRenderTimeoutEnded\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            #messageActionsToggle\n            class=\"\n              str-chat__message-actions-container\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            [popper]=\"popperContent\"\n            [popperTrigger]=\"popperTriggerClick\"\n            [popperPlacement]=\"popperPlacementAuto\"\n            [popperHideOnScroll]=\"false\"\n            [popperHideOnClickOutside]=\"true\"\n            [popperHideOnMouseLeave]=\"false\"\n            [popperDisableAnimation]=\"true\"\n          >\n            <popper-content #popperContent>\n              <ng-template\n                #defaultMessageActionsBox\n                let-isOpen=\"isOpen\"\n                let-isMine=\"isMine\"\n                let-enabledActions=\"enabledActions\"\n                let-messageInput=\"message\"\n                let-displayedActionsCountChangeHandler=\"displayedActionsCountChangeHandler\"\n                let-isEditingChangeHandler=\"isEditingChangeHandler\"\n                let-customActions=\"customActions\"\n              >\n                <stream-message-actions-box\n                  [isOpen]=\"isOpen\"\n                  [isMine]=\"isMine\"\n                  [enabledActions]=\"enabledActions\"\n                  [message]=\"messageInput\"\n                  [customActions]=\"customActions\"\n                  (displayedActionsCount)=\"\n                    displayedActionsCountChangeHandler($event)\n                  \"\n                  (isEditing)=\"isEditingChangeHandler($event)\"\n                ></stream-message-actions-box>\n              </ng-template>\n              <ng-container\n                *ngTemplateOutlet=\"\n                  (customTemplatesService.messageActionsBoxTemplate$ | async) ||\n                    defaultMessageActionsBox;\n                  context: getMessageActionsBoxContext()\n                \"\n              >\n              </ng-container>\n            </popper-content>\n            <div\n              class=\"str-chat__message-actions-box-button\"\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n            >\n              <stream-icon-placeholder\n                icon=\"action-icon\"\n                class=\"str-chat__message-action-icon\"\n              ></stream-icon-placeholder>\n            </div>\n          </div>\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              str-chat__message-reply-in-thread-button\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder\n              class=\"str-chat__message-action-icon\"\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              str-chat__message-reactions-button\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon-placeholder\n              class=\"str-chat__message-action-icon\"\n              icon=\"reaction-icon\"\n            ></stream-icon-placeholder>\n          </div>\n        </div>\n        <div class=\"str-chat__message-reactions-host\">\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              (customTemplatesService.messageReactionsTemplate$ | async) ||\n                defaultMessageReactions;\n              context: getMessageReactionsContext()\n            \"\n          ></ng-container>\n        </div>\n        <!-- transform: translate3d(0, 0, 0) fixes scrolling issues on iOS, see: https://stackoverflow.com/questions/50105780/elements-disappear-when-scrolling-in-safari-webkit-transform-fix-only-works-t/50144295#50144295 -->\n        <!-- transform: none is required when image carousel is open in order for the modal to be correctly positioned, see how the transform property changes the behavior of fixed positioned elements https://developer.mozilla.org/en-US/docs/Web/CSS/position  -->\n        <div\n          class=\"str-chat__message-bubble\"\n          style=\"transform: {{\n            imageAttachmentModalState === 'opened'\n              ? 'none'\n              : 'translate3d(0, 0, 0)'\n          }}\"\n        >\n          <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n            <ng-container\n              *ngTemplateOutlet=\"attachmentsTemplate\"\n            ></ng-container>\n          </ng-container>\n          <div\n            class=\"str-chat__message-text\"\n            tabindex=\"0\"\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              <ng-container *ngIf=\"hasAttachment && message?.quoted_message\">\n                <ng-container\n                  *ngTemplateOutlet=\"attachmentsTemplate\"\n                ></ng-container>\n              </ng-container>\n              <div\n                data-testid=\"client-error-message\"\n                *ngIf=\"message?.type === 'error'\"\n                class=\"\n                  str-chat__simple-message--error-message\n                  str-chat__message--error-message\n                \"\n              >\n                {{ \"streamChat.Error · Unsent\" | translate }}\n              </div>\n              <div\n                data-testid=\"error-message\"\n                *ngIf=\"message?.status === 'failed'\"\n                class=\"\n                  str-chat__simple-message--error-message\n                  str-chat__message--error-message\n                \"\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 data-testid=\"text\">\n                <p>\n                  <ng-container *ngIf=\"messageTextParts; else defaultContent\">\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                          <span class=\"str-chat__message-mention\">{{\n                            content\n                          }}</span>\n                        </ng-template>\n                        <ng-container\n                          *ngTemplateOutlet=\"\n                            (customTemplatesService.mentionTemplate$ | async) ||\n                              defaultMention;\n                            context: getMentionContext(part)\n                          \"\n                        ></ng-container>\n                      </ng-template>\n                    </ng-container>\n                  </ng-container>\n                  <ng-template #defaultContent>\n                    {{ messageText || \"\" }}\n                  </ng-template>\n                </p>\n              </div>\n            </div>\n          </div>\n          <stream-icon-placeholder\n            *ngIf=\"themeVersion === '2'\"\n            class=\"str-chat__message-error-icon\"\n            icon=\"error\"\n          ></stream-icon-placeholder>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"\n            replyCountButton;\n            context: { position: 'inside-message-bubble' }\n          \"\n        ></ng-container>\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageDateAndSender;\n            context: { position: 'inside-message-bubble' }\n          \"\n        ></ng-container>\n      </div>\n      <ng-container\n        *ngTemplateOutlet=\"\n          replyCountButton;\n          context: { position: 'outside-message-bubble', message: message }\n        \"\n      ></ng-container>\n\n      <ng-container\n        *ngTemplateOutlet=\"\n          messageDateAndSender;\n          context: { position: 'outside-message-bubble' }\n        \"\n      ></ng-container>\n    </ng-container>\n  </ng-container>\n</div>\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  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.systemMessageTemplate$ | async) ||\n        defaultSystemMessage;\n      context: getMessageContext()\n    \"\n  ></ng-container>\n  <ng-template #defaultSystemMessage let-messageInput=\"message\">\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>{{ messageInput?.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</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message str-chat__quoted-message-preview\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n    (click)=\"\n      jumpToMessage(\n        (message?.quoted_message)!.id,\n        message?.quoted_message?.parent_id\n      )\n    \"\n    (keyup.enter)=\"\n      jumpToMessage(\n        (message?.quoted_message)!.id,\n        message?.quoted_message?.parent_id\n      )\n    \"\n  >\n    <stream-avatar-placeholder\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\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 str-chat__quoted-message-bubble\">\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          let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n        >\n          <stream-attachment-list\n            [messageId]=\"messageId\"\n            [attachments]=\"attachments\"\n            [parentMessageId]=\"parentMessageId\"\n            (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n          ></stream-attachment-list>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.attachmentListTemplate$ | async) ||\n              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?.translation ||\n          message?.quoted_message?.html ||\n          message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n\n<!-- We need these markups in slightly different positions in theme-v1 and theme-v2, this soultion makes that possible without duplicating the code -->\n<ng-template #messageDateAndSender let-position=\"position\">\n  <ng-container\n    *ngIf=\"\n      (position === 'inside-message-bubble' && themeVersion === '1') ||\n      (position === 'outside-message-bubble' && themeVersion === '2')\n    \"\n  >\n    <div\n      class=\"str-chat__translation-notice\"\n      *ngIf=\"shouldDisplayTranslationNotice\"\n      data-testid=\"translation-notice\"\n    >\n      <button\n        data-testid=\"see-original\"\n        *ngIf=\"displayedMessageTextContent === 'translation'\"\n        (click)=\"displayOriginalMessage()\"\n        (keyup.enter)=\"displayOriginalMessage()\"\n        translate\n      >\n        streamChat.See original (automatically translated)\n      </button>\n      <button\n        data-testid=\"see-translation\"\n        *ngIf=\"displayedMessageTextContent === 'original'\"\n        (click)=\"displayTranslatedMessage()\"\n        (keyup.enter)=\"displayTranslatedMessage()\"\n        translate\n      >\n        streamChat.See translation\n      </button>\n    </div>\n    <div\n      class=\"\n        str-chat__message-data\n        str-chat__message-simple-data\n        str-chat__message-metadata\n      \"\n    >\n      <ng-container *ngIf=\"themeVersion === '2'\">\n        <ng-container *ngTemplateOutlet=\"messageStatus\"></ng-container>\n      </ng-container>\n      <span\n        data-testid=\"sender\"\n        *ngIf=\"!isSentByCurrentUser\"\n        class=\"str-chat__message-simple-name str-chat__message-sender-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  </ng-container>\n</ng-template>\n\n<ng-template #messageStatus>\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' && isMessageDeliveredAndRead && canDisplayReadStatus;\n          else deliveredStatus\n        \"\n      >\n        <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n      </ng-container>\n    </ng-template>\n    <ng-template #deliveredStatus>\n      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.deliveredStatusTemplate$ | async) ||\n            defaultDeliveredStatus;\n          context: getDeliveredStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultDeliveredStatus>\n      <span\n        *ngIf=\"mode === 'main'\"\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"delivered-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n            {{ \"streamChat.Delivered\" | translate }}\n          </div>\n        </popper-content>\n        <stream-icon-placeholder\n          data-testid=\"delivered-icon\"\n          icon=\"delivered-icon\"\n        ></stream-icon-placeholder>\n      </span>\n    </ng-template>\n    <ng-template #sendingStatus>\n      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.sendingStatusTemplate$ | async) ||\n            defaultSendingStatus;\n          context: getSendingStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultSendingStatus>\n      <span\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"sending-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n            {{ \"streamChat.Sending...\" | translate }}\n          </div>\n        </popper-content>\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      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.readStatusTemplate$ | async) ||\n            defaultReadStatus;\n          context: getReadStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultReadStatus>\n      <span\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"read-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div\n            class=\"str-chat__tooltip str-chat__tooltip-angular\"\n            data-testid=\"read-by-tooltip\"\n          >\n            {{ readByText }}\n          </div>\n        </popper-content>\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-container>\n</ng-template>\n\n<ng-template #replyCountButton let-position=\"position\">\n  <div\n    *ngIf=\"\n      (position === 'inside-message-bubble' && themeVersion === '1') ||\n      (position === 'outside-message-bubble' && themeVersion === '2')\n    \"\n    class=\"\n      str-chat__message-simple-reply-button\n      str-chat__message-replies-count-button-wrapper\n    \"\n  >\n    <button\n      *ngIf=\"shouldDisplayThreadLink\"\n      class=\"str-chat__message-replies-count-button\"\n      data-testid=\"reply-count-button\"\n      (click)=\"setAsActiveParentMessage()\"\n    >\n      <stream-icon-placeholder\n        *ngIf=\"themeVersion === '1'\"\n        stream-icon-placeholder\n        icon=\"reply\"\n      ></stream-icon-placeholder>\n      {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n      replies' | translate:replyCountParam)}}\n    </button>\n  </div>\n</ng-template>\n\n<ng-template #attachmentsTemplate>\n  <ng-template\n    #defaultAttachments\n    let-messageId=\"messageId\"\n    let-attachments=\"attachments\"\n    let-parentMessageId=\"parentMessageId\"\n    let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n  >\n    <stream-attachment-list\n      [messageId]=\"messageId\"\n      [attachments]=\"attachments\"\n      [parentMessageId]=\"parentMessageId\"\n      (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n    ></stream-attachment-list>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.attachmentListTemplate$ | async) ||\n        defaultAttachments;\n      context: getAttachmentListContext()\n    \"\n  ></ng-container>\n</ng-template>\n"]}
|
|
381
|
+
//# 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  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  OnDestroy,\n  OnInit,\n  ChangeDetectorRef,\n  ChangeDetectionStrategy,\n  NgZone,\n  AfterViewInit,\n} from '@angular/core';\nimport { Attachment, UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport {\n  AttachmentListContext,\n  MentionTemplateContext,\n  MessageActionsBoxContext,\n  MessageReactionsContext,\n  DefaultStreamChatGenerics,\n  StreamMessage,\n  DeliveredStatusContext,\n  SendingStatusContext,\n  ReadStatusContext,\n  CustomMessageActionItem,\n  SystemMessageContext,\n} from '../types';\nimport emojiRegex from 'emoji-regex';\nimport { Subscription } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\nimport { ThemeService } from '../theme.service';\nimport { NgxPopperjsTriggers, NgxPopperjsPlacements } from 'ngx-popperjs';\nimport { DateParserService } from '../date-parser.service';\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  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MessageComponent\n  implements OnInit, OnChanges, OnDestroy, AfterViewInit\n{\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  /**\n   * Highlighting is used to add visual emphasize to a message when jumping to the message\n   */\n  @Input() isHighlighted = false;\n  /**\n   * A list of custom message actions to be displayed in the action box\n   */\n  @Input() customActions: CustomMessageActionItem[] = [];\n  readonly themeVersion: '1' | '2';\n  canReceiveReadEvents: boolean | undefined;\n  canReactToMessage: boolean | undefined;\n  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] | undefined = [];\n  messageText?: string;\n  popperTriggerClick = NgxPopperjsTriggers.click;\n  popperTriggerHover = NgxPopperjsTriggers.hover;\n  popperPlacementAuto = NgxPopperjsPlacements.AUTO;\n  shouldDisplayTranslationNotice = false;\n  displayedMessageTextContent: 'original' | 'translation' = 'original';\n  imageAttachmentModalState: 'opened' | 'closed' = 'closed';\n  shouldDisplayThreadLink = false;\n  isSentByCurrentUser = false;\n  readByText = '';\n  lastReadUser: UserResponse<DefaultStreamChatGenerics> | undefined = undefined;\n  isOnlyReadByMe = false;\n  isReadByMultipleUsers = false;\n  isMessageDeliveredAndRead = false;\n  parsedDate = '';\n  areOptionsVisible = false;\n  hasAttachment = false;\n  hasReactions = false;\n  replyCountParam: { replyCount: number | undefined } = {\n    replyCount: undefined,\n  };\n  canDisplayReadStatus = false;\n  private quotedMessageAttachments: Attachment[] | undefined;\n  user: UserResponse<DefaultStreamChatGenerics> | undefined;\n  optionsRenderTimeoutEnded = false;\n  private subscriptions: Subscription[] = [];\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n  private readonly urlRegexp =\n    /(?:(?:https?|ftp|file):\\/\\/|www\\.|ftp\\.)(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[-A-Z0-9+&@#/%=~_|$?!:,.])*(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[A-Z0-9+&@#/%=~_|$])/gim;\n  private emojiRegexp = new RegExp(emojiRegex(), 'g');\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService,\n    public customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef,\n    themeService: ThemeService,\n    private dateParser: DateParserService,\n    private ngZone: NgZone\n  ) {\n    this.themeVersion = themeService.themeVersion;\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((u) => {\n        if (u !== this.user) {\n          this.user = u;\n          this.setIsSentByCurrentUser();\n          this.setLastReadUser();\n          this.cdRef.detectChanges();\n        }\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.shouldDisplayTranslationNotice = false;\n      this.displayedMessageTextContent = 'original';\n      this.createMessageParts();\n      const originalAttachments = this.message?.quoted_message?.attachments;\n      this.quotedMessageAttachments =\n        originalAttachments && originalAttachments.length\n          ? [originalAttachments[0]]\n          : [];\n      this.setIsSentByCurrentUser();\n      this.setLastReadUser();\n      this.readByText = this.message?.readBy\n        ? listUsers(this.message.readBy)\n        : '';\n      this.isOnlyReadByMe = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length === 0\n      );\n      this.isReadByMultipleUsers = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length > 1\n      );\n      this.isMessageDeliveredAndRead = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.status === 'received' &&\n        this.message.readBy.length > 0\n      );\n      this.parsedDate =\n        (this.message &&\n          this.message.created_at &&\n          this.dateParser.parseDateTime(this.message.created_at)) ||\n        '';\n      this.hasAttachment =\n        !!this.message?.attachments && !!this.message.attachments.length;\n      this.hasReactions =\n        !!this.message?.reaction_counts &&\n        Object.keys(this.message.reaction_counts).length > 0;\n      this.replyCountParam = { replyCount: this.message?.reply_count };\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      this.canDisplayReadStatus =\n        this.canReceiveReadEvents !== false &&\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n    }\n    if (changes.message || changes.enabledMessageActions || changes.mode) {\n      this.shouldDisplayThreadLink =\n        !!this.message?.reply_count && this.mode !== 'thread';\n    }\n    if (changes.message || changes.mode) {\n      this.areOptionsVisible = this.message\n        ? !(\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        : false;\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.ngZone.runOutsideAngular(() => {\n      this.container?.nativeElement.addEventListener('mouseleave', () =>\n        this.mouseLeft()\n      );\n    });\n    setTimeout(() => {\n      this.optionsRenderTimeoutEnded = true;\n      this.cdRef.detectChanges();\n    }, 0);\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  getAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.id || '',\n      attachments: this.message?.attachments || [],\n      parentMessageId: this.message?.parent_id,\n      imageModalStateChangeHandler: (state) =>\n        (this.imageAttachmentModalState = state),\n    };\n  }\n\n  getMessageContext(): SystemMessageContext {\n    return {\n      message: this.message,\n      enabledMessageActions: this.enabledMessageActions,\n      isHighlighted: this.isHighlighted,\n      isLastSentMessage: this.isLastSentMessage,\n      mode: this.mode,\n      customActions: this.customActions,\n      parsedDate: this.parsedDate,\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  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      displayedActionsCountChangeHandler: (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      customActions: this.customActions || [],\n    };\n  }\n\n  getDeliveredStatusContext(): DeliveredStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getSendingStatusContext(): SendingStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getReadStatusContext(): ReadStatusContext {\n    return {\n      message: this.message!,\n      readByText: this.readByText,\n    };\n  }\n\n  jumpToMessage(messageId: string, parentMessageId?: string) {\n    void this.channelService.jumpToMessage(messageId, parentMessageId);\n  }\n\n  displayTranslatedMessage() {\n    this.createMessageParts(true);\n  }\n\n  displayOriginalMessage() {\n    this.createMessageParts(false);\n  }\n\n  mouseLeft() {\n    if (this.isActionBoxOpen) {\n      this.ngZone.run(() => {\n        this.isActionBoxOpen = false;\n      });\n    }\n  }\n\n  private createMessageParts(shouldTranslate = true) {\n    this.messageTextParts = undefined;\n    this.messageText = undefined;\n    let content = this.getMessageContent(shouldTranslate);\n    if (\n      (!this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0) &&\n      !content?.match(this.emojiRegexp) &&\n      !content?.match(this.urlRegexp)\n    ) {\n      this.messageTextParts = undefined;\n      this.messageText = content;\n      return;\n    }\n    if (!content) {\n      return;\n    }\n    if (\n      !this.message!.mentioned_users ||\n      this.message!.mentioned_users.length === 0\n    ) {\n      content = this.fixEmojiDisplay(content);\n      content = this.wrapLinskWithAnchorTag(content);\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        let formattedPrecedingText = this.fixEmojiDisplay(precedingText);\n        formattedPrecedingText = this.wrapLinskWithAnchorTag(\n          formattedPrecedingText\n        );\n        this.messageTextParts!.push({\n          content: formattedPrecedingText,\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        text = this.fixEmojiDisplay(text);\n        text = this.wrapLinskWithAnchorTag(text);\n        this.messageTextParts.push({ content: text, type: 'text' });\n      }\n    }\n  }\n\n  private getMessageContent(shouldTranslate: boolean) {\n    const originalContent = this.message?.text;\n    if (shouldTranslate) {\n      const translation = this.message?.translation;\n      if (translation) {\n        this.shouldDisplayTranslationNotice = true;\n        this.displayedMessageTextContent = 'translation';\n      }\n      return translation || originalContent;\n    } else {\n      this.displayedMessageTextContent = 'original';\n      return originalContent;\n    }\n  }\n\n  private fixEmojiDisplay(content: string) {\n    // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\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 && typeof (window as any).opr === 'undefined';\n    /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n    content = content.replace(\n      this.emojiRegexp,\n      (match) =>\n        `<span ${\n          isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n        }>${match}</span>`\n    );\n\n    return content;\n  }\n\n  private wrapLinskWithAnchorTag(content: string) {\n    content = content.replace(\n      this.urlRegexp,\n      (match) => `<a href=\"${match}\" rel=\"nofollow\">${match}</a>`\n    );\n\n    return content;\n  }\n\n  private setIsSentByCurrentUser() {\n    this.isSentByCurrentUser = this.message?.user?.id === this.user?.id;\n  }\n\n  private setLastReadUser() {\n    this.lastReadUser = this.message?.readBy?.filter(\n      (u) => u.id !== this.user?.id\n    )[0];\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--other]=\"!isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"hasReactions\"\n  [class.str-chat__message--highlighted]=\"isHighlighted\"\n  [class.str-chat__message-with-thread-link]=\"shouldDisplayThreadLink\"\n  [class.str-chat__message-send-can-be-retried]=\"\n    message?.status === 'failed' && message?.errorStatusCode !== 403\n  \"\n  data-testid=\"message-container\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container *ngIf=\"themeVersion === '1'\">\n        <ng-container *ngTemplateOutlet=\"messageStatus\"></ng-container>\n      </ng-container>\n      <stream-avatar-placeholder\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\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 str-chat__message-options\"\n          data-testid=\"message-options\"\n          [class.str-chat__message-edit-in-progress]=\"isEditing\"\n          *ngIf=\"areOptionsVisible && optionsRenderTimeoutEnded\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            #messageActionsToggle\n            class=\"\n              str-chat__message-actions-container\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            [popper]=\"popperContent\"\n            [popperTrigger]=\"popperTriggerClick\"\n            [popperPlacement]=\"popperPlacementAuto\"\n            [popperHideOnScroll]=\"false\"\n            [popperHideOnClickOutside]=\"true\"\n            [popperHideOnMouseLeave]=\"false\"\n            [popperDisableAnimation]=\"true\"\n          >\n            <popper-content #popperContent>\n              <ng-template\n                #defaultMessageActionsBox\n                let-isOpen=\"isOpen\"\n                let-isMine=\"isMine\"\n                let-enabledActions=\"enabledActions\"\n                let-messageInput=\"message\"\n                let-displayedActionsCountChangeHandler=\"displayedActionsCountChangeHandler\"\n                let-isEditingChangeHandler=\"isEditingChangeHandler\"\n                let-customActions=\"customActions\"\n              >\n                <stream-message-actions-box\n                  [isOpen]=\"isOpen\"\n                  [isMine]=\"isMine\"\n                  [enabledActions]=\"enabledActions\"\n                  [message]=\"messageInput\"\n                  [customActions]=\"customActions\"\n                  (displayedActionsCount)=\"\n                    displayedActionsCountChangeHandler($event)\n                  \"\n                  (isEditing)=\"isEditingChangeHandler($event)\"\n                ></stream-message-actions-box>\n              </ng-template>\n              <ng-container\n                *ngTemplateOutlet=\"\n                  (customTemplatesService.messageActionsBoxTemplate$ | async) ||\n                    defaultMessageActionsBox;\n                  context: getMessageActionsBoxContext()\n                \"\n              >\n              </ng-container>\n            </popper-content>\n            <div\n              class=\"str-chat__message-actions-box-button\"\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n            >\n              <stream-icon-placeholder\n                icon=\"action-icon\"\n                class=\"str-chat__message-action-icon\"\n              ></stream-icon-placeholder>\n            </div>\n          </div>\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              str-chat__message-reply-in-thread-button\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon-placeholder\n              class=\"str-chat__message-action-icon\"\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              str-chat__message-reactions-button\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon-placeholder\n              class=\"str-chat__message-action-icon\"\n              icon=\"reaction-icon\"\n            ></stream-icon-placeholder>\n          </div>\n        </div>\n        <div class=\"str-chat__message-reactions-host\">\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              (customTemplatesService.messageReactionsTemplate$ | async) ||\n                defaultMessageReactions;\n              context: getMessageReactionsContext()\n            \"\n          ></ng-container>\n        </div>\n        <!-- transform: translate3d(0, 0, 0) fixes scrolling issues on iOS, see: https://stackoverflow.com/questions/50105780/elements-disappear-when-scrolling-in-safari-webkit-transform-fix-only-works-t/50144295#50144295 -->\n        <!-- transform: none is required when image carousel is open in order for the modal to be correctly positioned, see how the transform property changes the behavior of fixed positioned elements https://developer.mozilla.org/en-US/docs/Web/CSS/position  -->\n        <div\n          class=\"str-chat__message-bubble\"\n          style=\"transform: {{\n            imageAttachmentModalState === 'opened'\n              ? 'none'\n              : 'translate3d(0, 0, 0)'\n          }}\"\n        >\n          <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n            <ng-container\n              *ngTemplateOutlet=\"attachmentsTemplate\"\n            ></ng-container>\n          </ng-container>\n          <div\n            class=\"str-chat__message-text\"\n            tabindex=\"0\"\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              <ng-container *ngIf=\"hasAttachment && message?.quoted_message\">\n                <ng-container\n                  *ngTemplateOutlet=\"attachmentsTemplate\"\n                ></ng-container>\n              </ng-container>\n              <div\n                data-testid=\"client-error-message\"\n                *ngIf=\"message?.type === 'error'\"\n                class=\"\n                  str-chat__simple-message--error-message\n                  str-chat__message--error-message\n                \"\n              >\n                {{ \"streamChat.Error · Unsent\" | translate }}\n              </div>\n              <div\n                data-testid=\"error-message\"\n                *ngIf=\"message?.status === 'failed'\"\n                class=\"\n                  str-chat__simple-message--error-message\n                  str-chat__message--error-message\n                \"\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 data-testid=\"text\">\n                <p>\n                  <ng-container *ngIf=\"messageTextParts; else defaultContent\">\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                          <span class=\"str-chat__message-mention\">{{\n                            content\n                          }}</span>\n                        </ng-template>\n                        <ng-container\n                          *ngTemplateOutlet=\"\n                            (customTemplatesService.mentionTemplate$ | async) ||\n                              defaultMention;\n                            context: getMentionContext(part)\n                          \"\n                        ></ng-container>\n                      </ng-template>\n                    </ng-container>\n                  </ng-container>\n                  <ng-template #defaultContent>\n                    {{ messageText || \"\" }}\n                  </ng-template>\n                </p>\n              </div>\n            </div>\n          </div>\n          <stream-icon-placeholder\n            *ngIf=\"themeVersion === '2'\"\n            class=\"str-chat__message-error-icon\"\n            icon=\"error\"\n          ></stream-icon-placeholder>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"\n            replyCountButton;\n            context: { position: 'inside-message-bubble' }\n          \"\n        ></ng-container>\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageDateAndSender;\n            context: { position: 'inside-message-bubble' }\n          \"\n        ></ng-container>\n      </div>\n      <ng-container\n        *ngTemplateOutlet=\"\n          replyCountButton;\n          context: { position: 'outside-message-bubble', message: message }\n        \"\n      ></ng-container>\n\n      <ng-container\n        *ngTemplateOutlet=\"\n          messageDateAndSender;\n          context: { position: 'outside-message-bubble' }\n        \"\n      ></ng-container>\n    </ng-container>\n  </ng-container>\n</div>\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  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.systemMessageTemplate$ | async) ||\n        defaultSystemMessage;\n      context: getMessageContext()\n    \"\n  ></ng-container>\n  <ng-template #defaultSystemMessage let-messageInput=\"message\">\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>{{ messageInput?.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</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message str-chat__quoted-message-preview\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n    (click)=\"\n      jumpToMessage(\n        (message?.quoted_message)!.id,\n        message?.quoted_message?.parent_id\n      )\n    \"\n    (keyup.enter)=\"\n      jumpToMessage(\n        (message?.quoted_message)!.id,\n        message?.quoted_message?.parent_id\n      )\n    \"\n  >\n    <stream-avatar-placeholder\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\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 str-chat__quoted-message-bubble\">\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          let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n        >\n          <stream-attachment-list\n            [messageId]=\"messageId\"\n            [attachments]=\"attachments\"\n            [parentMessageId]=\"parentMessageId\"\n            (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n          ></stream-attachment-list>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.attachmentListTemplate$ | async) ||\n              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?.translation ||\n          message?.quoted_message?.html ||\n          message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n\n<!-- We need these markups in slightly different positions in theme-v1 and theme-v2, this soultion makes that possible without duplicating the code -->\n<ng-template #messageDateAndSender let-position=\"position\">\n  <ng-container\n    *ngIf=\"\n      (position === 'inside-message-bubble' && themeVersion === '1') ||\n      (position === 'outside-message-bubble' && themeVersion === '2')\n    \"\n  >\n    <div\n      class=\"str-chat__translation-notice\"\n      *ngIf=\"shouldDisplayTranslationNotice\"\n      data-testid=\"translation-notice\"\n    >\n      <button\n        data-testid=\"see-original\"\n        *ngIf=\"displayedMessageTextContent === 'translation'\"\n        (click)=\"displayOriginalMessage()\"\n        (keyup.enter)=\"displayOriginalMessage()\"\n        translate\n      >\n        streamChat.See original (automatically translated)\n      </button>\n      <button\n        data-testid=\"see-translation\"\n        *ngIf=\"displayedMessageTextContent === 'original'\"\n        (click)=\"displayTranslatedMessage()\"\n        (keyup.enter)=\"displayTranslatedMessage()\"\n        translate\n      >\n        streamChat.See translation\n      </button>\n    </div>\n    <div\n      class=\"\n        str-chat__message-data\n        str-chat__message-simple-data\n        str-chat__message-metadata\n      \"\n    >\n      <ng-container *ngIf=\"themeVersion === '2'\">\n        <ng-container *ngTemplateOutlet=\"messageStatus\"></ng-container>\n      </ng-container>\n      <span\n        data-testid=\"sender\"\n        *ngIf=\"!isSentByCurrentUser\"\n        class=\"str-chat__message-simple-name str-chat__message-sender-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  </ng-container>\n</ng-template>\n\n<ng-template #messageStatus>\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' && isMessageDeliveredAndRead && canDisplayReadStatus;\n          else deliveredStatus\n        \"\n      >\n        <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n      </ng-container>\n    </ng-template>\n    <ng-template #deliveredStatus>\n      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.deliveredStatusTemplate$ | async) ||\n            defaultDeliveredStatus;\n          context: getDeliveredStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultDeliveredStatus>\n      <span\n        *ngIf=\"mode === 'main'\"\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"delivered-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n            {{ \"streamChat.Delivered\" | translate }}\n          </div>\n        </popper-content>\n        <stream-icon-placeholder\n          data-testid=\"delivered-icon\"\n          icon=\"delivered-icon\"\n        ></stream-icon-placeholder>\n      </span>\n    </ng-template>\n    <ng-template #sendingStatus>\n      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.sendingStatusTemplate$ | async) ||\n            defaultSendingStatus;\n          context: getSendingStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultSendingStatus>\n      <span\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"sending-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n            {{ \"streamChat.Sending...\" | translate }}\n          </div>\n        </popper-content>\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      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.readStatusTemplate$ | async) ||\n            defaultReadStatus;\n          context: getReadStatusContext()\n        \"\n      ></ng-container>\n    </ng-template>\n    <ng-template #defaultReadStatus>\n      <span\n        class=\"\n          str-chat__message-simple-status\n          str-chat__message-simple-status-angular\n          str-chat__message-status\n        \"\n        data-testid=\"read-indicator\"\n        tabindex=\"0\"\n        [popper]=\"popperContent\"\n        [popperTrigger]=\"popperTriggerHover\"\n        [popperPlacement]=\"popperPlacementAuto\"\n        [popperHideOnScroll]=\"false\"\n        [popperHideOnClickOutside]=\"false\"\n        (hover)=\"$event.stopPropagation()\"\n      >\n        <popper-content #popperContent>\n          <div\n            class=\"str-chat__tooltip str-chat__tooltip-angular\"\n            data-testid=\"read-by-tooltip\"\n          >\n            {{ readByText }}\n          </div>\n        </popper-content>\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-container>\n</ng-template>\n\n<ng-template #replyCountButton let-position=\"position\">\n  <div\n    *ngIf=\"\n      (position === 'inside-message-bubble' && themeVersion === '1') ||\n      (position === 'outside-message-bubble' && themeVersion === '2')\n    \"\n    class=\"\n      str-chat__message-simple-reply-button\n      str-chat__message-replies-count-button-wrapper\n    \"\n  >\n    <button\n      *ngIf=\"shouldDisplayThreadLink\"\n      class=\"str-chat__message-replies-count-button\"\n      data-testid=\"reply-count-button\"\n      (click)=\"setAsActiveParentMessage()\"\n    >\n      <stream-icon-placeholder\n        *ngIf=\"themeVersion === '1'\"\n        stream-icon-placeholder\n        icon=\"reply\"\n      ></stream-icon-placeholder>\n      {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n      replies' | translate:replyCountParam)}}\n    </button>\n  </div>\n</ng-template>\n\n<ng-template #attachmentsTemplate>\n  <ng-template\n    #defaultAttachments\n    let-messageId=\"messageId\"\n    let-attachments=\"attachments\"\n    let-parentMessageId=\"parentMessageId\"\n    let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n  >\n    <stream-attachment-list\n      [messageId]=\"messageId\"\n      [attachments]=\"attachments\"\n      [parentMessageId]=\"parentMessageId\"\n      (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n    ></stream-attachment-list>\n  </ng-template>\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.attachmentListTemplate$ | async) ||\n        defaultAttachments;\n      context: getAttachmentListContext()\n    \"\n  ></ng-container>\n</ng-template>\n"]}
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { __awaiter } from 'tslib';
|
|
2
2
|
import * as i0 from '@angular/core';
|
|
3
3
|
import { Injectable, Component, Input, EventEmitter, Output, ViewChild, InjectionToken, Directive, HostBinding, Inject, ChangeDetectionStrategy, NgModule } from '@angular/core';
|
|
4
|
-
import { BehaviorSubject, ReplaySubject,
|
|
4
|
+
import { BehaviorSubject, ReplaySubject, combineLatest, Subject, timer } from 'rxjs';
|
|
5
5
|
import { StreamChat } from 'stream-chat';
|
|
6
|
-
import { shareReplay, map,
|
|
6
|
+
import { take, shareReplay, map, first, filter, tap, distinctUntilChanged, debounceTime } from 'rxjs/operators';
|
|
7
7
|
import { v4 } from 'uuid';
|
|
8
8
|
import * as i9 from '@ngx-translate/core';
|
|
9
9
|
import { TranslateModule } from '@ngx-translate/core';
|
|
@@ -20,7 +20,7 @@ import transliterate from '@stream-io/transliterate';
|
|
|
20
20
|
import * as i8 from 'angular-mentions';
|
|
21
21
|
import { MentionModule } from 'angular-mentions';
|
|
22
22
|
|
|
23
|
-
const version = '4.47.
|
|
23
|
+
const version = '4.47.3';
|
|
24
24
|
|
|
25
25
|
/**
|
|
26
26
|
* The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
|
|
@@ -890,7 +890,7 @@ class ChannelService {
|
|
|
890
890
|
: this.activeChannelMessagesSubject.next([...channel.state.messages]);
|
|
891
891
|
let messages;
|
|
892
892
|
(isThreadReply ? this.activeThreadMessages$ : this.activeChannelMessages$)
|
|
893
|
-
.pipe(take
|
|
893
|
+
.pipe(take(1))
|
|
894
894
|
.subscribe((m) => (messages = m));
|
|
895
895
|
const newMessage = messages[messages.length - 1];
|
|
896
896
|
return newMessage;
|
|
@@ -908,7 +908,7 @@ class ChannelService {
|
|
|
908
908
|
: this.activeChannelMessagesSubject.next([...channel.state.messages]);
|
|
909
909
|
let messages;
|
|
910
910
|
(isThreadReply ? this.activeThreadMessages$ : this.activeChannelMessages$)
|
|
911
|
-
.pipe(take
|
|
911
|
+
.pipe(take(1))
|
|
912
912
|
.subscribe((m) => (messages = m));
|
|
913
913
|
const newMessage = messages[messages.length - 1];
|
|
914
914
|
return newMessage;
|
|
@@ -1000,7 +1000,7 @@ class ChannelService {
|
|
|
1000
1000
|
this.setChannelState(this.activeChannelSubject.getValue());
|
|
1001
1001
|
let messages;
|
|
1002
1002
|
this.activeChannelMessages$
|
|
1003
|
-
.pipe(take
|
|
1003
|
+
.pipe(take(1))
|
|
1004
1004
|
.subscribe((m) => (messages = m));
|
|
1005
1005
|
const updatedMessageToQuote = messages.find((m) => m.id === (messageToQuote === null || messageToQuote === void 0 ? void 0 : messageToQuote.id));
|
|
1006
1006
|
if (updatedMessageToQuote) {
|
|
@@ -1324,7 +1324,9 @@ class ChannelService {
|
|
|
1324
1324
|
}
|
|
1325
1325
|
watchForChannelEvents(channel) {
|
|
1326
1326
|
const unsubscribe = channel.on((event) => {
|
|
1327
|
-
|
|
1327
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion
|
|
1328
|
+
const type = event.type;
|
|
1329
|
+
switch (type) {
|
|
1328
1330
|
case 'message.new': {
|
|
1329
1331
|
this.ngZone.run(() => {
|
|
1330
1332
|
if (this.customNewMessageHandler) {
|
|
@@ -1391,6 +1393,23 @@ class ChannelService {
|
|
|
1391
1393
|
});
|
|
1392
1394
|
break;
|
|
1393
1395
|
}
|
|
1396
|
+
case 'capabilities.changed': {
|
|
1397
|
+
this.ngZone.run(() => {
|
|
1398
|
+
var _a;
|
|
1399
|
+
const cid = event.cid;
|
|
1400
|
+
if (cid) {
|
|
1401
|
+
const currentChannels = this.channelsSubject.getValue();
|
|
1402
|
+
const index = currentChannels === null || currentChannels === void 0 ? void 0 : currentChannels.findIndex((c) => c.cid === cid);
|
|
1403
|
+
if (index !== -1 && index !== undefined) {
|
|
1404
|
+
this.channelsSubject.next([...currentChannels]);
|
|
1405
|
+
if (cid === ((_a = this.activeChannelSubject.getValue()) === null || _a === void 0 ? void 0 : _a.cid)) {
|
|
1406
|
+
this.activeChannelSubject.next(this.activeChannelSubject.getValue());
|
|
1407
|
+
}
|
|
1408
|
+
}
|
|
1409
|
+
}
|
|
1410
|
+
});
|
|
1411
|
+
break;
|
|
1412
|
+
}
|
|
1394
1413
|
}
|
|
1395
1414
|
});
|
|
1396
1415
|
this.channelSubscriptions[channel.cid] = unsubscribe.unsubscribe;
|
|
@@ -3607,7 +3626,7 @@ class MessageInputComponent {
|
|
|
3607
3626
|
startCooldown(cooldown) {
|
|
3608
3627
|
this.textareaPlaceholder = this.slowModeTextareaPlaceholder;
|
|
3609
3628
|
this.isCooldownInProgress = true;
|
|
3610
|
-
this.cooldown$ = timer(0, 1000).pipe(take
|
|
3629
|
+
this.cooldown$ = timer(0, 1000).pipe(take(cooldown + 1), map((v) => cooldown - v), tap((v) => {
|
|
3611
3630
|
if (v === 0) {
|
|
3612
3631
|
this.stopCooldown();
|
|
3613
3632
|
}
|
|
@@ -4618,9 +4637,7 @@ class MessageComponent {
|
|
|
4618
4637
|
}
|
|
4619
4638
|
if (changes.message || changes.enabledMessageActions || changes.mode) {
|
|
4620
4639
|
this.shouldDisplayThreadLink =
|
|
4621
|
-
!!((_g = this.message) === null || _g === void 0 ? void 0 : _g.reply_count) &&
|
|
4622
|
-
this.mode !== 'thread' &&
|
|
4623
|
-
this.enabledMessageActions.indexOf('send-reply') !== -1;
|
|
4640
|
+
!!((_g = this.message) === null || _g === void 0 ? void 0 : _g.reply_count) && this.mode !== 'thread';
|
|
4624
4641
|
}
|
|
4625
4642
|
if (changes.message || changes.mode) {
|
|
4626
4643
|
this.areOptionsVisible = this.message
|