stream-chat-angular 2.20.0 → 2.21.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/version.d.ts +1 -1
- package/bundles/stream-chat-angular.umd.js +39 -18
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/channel.service.js +29 -12
- package/esm2015/lib/message/message.component.js +7 -2
- package/esm2015/lib/message-list/message-list.component.js +2 -2
- package/fesm2015/stream-chat-angular.js +36 -14
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/channel.service.d.ts +7 -1
- package/package.json +1 -1
- package/src/assets/version.ts +1 -1
|
@@ -149,7 +149,12 @@ export class MessageComponent {
|
|
|
149
149
|
this.message.mentioned_users.length === 0) {
|
|
150
150
|
// Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223
|
|
151
151
|
const regex = new RegExp(emojiRegex(), 'g');
|
|
152
|
-
|
|
152
|
+
// Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome
|
|
153
|
+
/* eslint-disable @typescript-eslint/no-unsafe-member-access */
|
|
154
|
+
const isChrome = !!window.chrome &&
|
|
155
|
+
typeof window.opr === 'undefined';
|
|
156
|
+
/* eslint-enable @typescript-eslint/no-unsafe-member-access */
|
|
157
|
+
content = content.replace(regex, (match) => `<span ${isChrome ? 'class="str-chat__emoji-display-fix"' : ''}>${match}</span>`);
|
|
153
158
|
this.messageTextParts = [{ content, type: 'text' }];
|
|
154
159
|
}
|
|
155
160
|
else {
|
|
@@ -207,4 +212,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
207
212
|
type: ViewChild,
|
|
208
213
|
args: ['container']
|
|
209
214
|
}] } });
|
|
210
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message/message.component.ts","../../../../../projects/stream-chat-angular/src/lib/message/message.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  TemplateRef,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { getDeviceWidth } from '../device-width';\nimport { DefaultUserType, StreamMessage } from '../types';\nimport { parseDate } from './parse-date';\nimport { getReadByText } from './read-by-text';\nimport emojiRegex from 'emoji-regex';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](../concepts/message-interactions.mdx).\n */\n@Component({\n  selector: 'stream-message',\n  templateUrl: './message.component.html',\n  styles: [],\n})\nexport class MessageComponent implements OnChanges {\n  /**\n   * The input used for message edit. By default, the [default message input component](./MessageInputComponent.mdx) is used. To change the input for message edit, provide [your own custom template](./MessageInputComponent.mdx/#customization).\n   */\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  /**\n   * The template used to display a mention in a message. It receives the mentioned user in a variable called `user` with the type [`UserResponse`](https://github.com/GetStream/stream-chat-js/blob/master/src/types.ts). You can provide your own template if you want to [add actions to mentions](../code-examples/mention-actions.mdx).\n   */\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\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 reactions are displayed. If you use the default chat UI you can also set this using the [`MessageList`](./MessageListComponent.mdx) component.\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() areReactionsEnabled: boolean | undefined;\n  /**\n   * If true, the user can add reactions to the message. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() canReactToMessage: boolean | undefined;\n  /**\n   * If `true`, the message status (sending, sent, who read the message) is displayed.\n   */\n  @Input() isLastSentMessage: boolean | undefined;\n  /**\n   * If true, the read indicator is displayed. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() canReceiveReadEvents: 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  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  isPressedOnMobile = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] = [];\n  private user: UserResponse<DefaultUserType> | undefined;\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService\n  ) {\n    this.user = this.chatClientService.chatClient.user;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.createMessageParts();\n    }\n  }\n\n  get isSentByCurrentUser() {\n    return this.message?.user?.id === this.user?.id;\n  }\n\n  get readByText() {\n    return getReadByText(this.message!.readBy);\n  }\n\n  get lastReadUser() {\n    return this.message?.readBy.filter((u) => u.id !== this.user?.id)[0];\n  }\n\n  get isOnlyReadByMe() {\n    return this.message && this.message.readBy.length === 0;\n  }\n\n  get isReadByMultipleUsers() {\n    return this.message && this.message.readBy.length > 1;\n  }\n\n  get isMessageDeliveredAndRead() {\n    return (\n      this.message &&\n      this.message.readBy &&\n      this.message.status === 'received' &&\n      this.message.readBy.length > 0\n    );\n  }\n\n  get parsedDate() {\n    if (!this.message || !this.message?.created_at) {\n      return;\n    }\n    return parseDate(this.message.created_at);\n  }\n\n  get areOptionsVisible() {\n    if (!this.message) {\n      return false;\n    }\n    return !(\n      !this.message.type ||\n      this.message.type === 'error' ||\n      this.message.type === 'system' ||\n      this.message.type === 'ephemeral' ||\n      this.message.status === 'failed' ||\n      this.message.status === 'sending' ||\n      (this.mode === 'thread' && !this.message.parent_id)\n    );\n  }\n\n  get hasAttachment() {\n    return !!this.message?.attachments && !!this.message.attachments.length;\n  }\n\n  get hasReactions() {\n    return (\n      !!this.message?.reaction_counts &&\n      Object.keys(this.message.reaction_counts).length > 0\n    );\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.message?.reply_count };\n  }\n\n  get canDisplayReadStatus() {\n    return (\n      this.canReceiveReadEvents !== false &&\n      this.enabledMessageActions.indexOf('read-events') !== -1\n    );\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.message?.quoted_message?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  resendMessage() {\n    void this.channelService.resendMessage(this.message!);\n  }\n\n  textClicked() {\n    if (getDeviceWidth().device !== 'mobile') {\n      this.isPressedOnMobile = false;\n      return;\n    }\n    if (this.isPressedOnMobile) {\n      return;\n    }\n    this.isPressedOnMobile = true;\n    const eventHandler = (event: Event) => {\n      if (!this.container?.nativeElement.contains(event.target as Node)) {\n        this.isPressedOnMobile = false;\n        window.removeEventListener('click', eventHandler);\n      }\n    };\n    window.addEventListener('click', eventHandler);\n  }\n\n  setAsActiveParentMessage() {\n    void this.channelService.setAsActiveParentMessage(this.message);\n  }\n\n  private createMessageParts() {\n    let content = this.message?.html || this.message?.text;\n    if (!content) {\n      this.messageTextParts = [];\n    } else {\n      // Backend will wrap HTML content with <p></p>\\n\n      if (content.startsWith('<p>')) {\n        content = content.replace('<p>', '');\n      }\n      if (content.endsWith('</p>\\n')) {\n        content = content.replace('</p>\\n', '');\n      }\n      if (\n        !this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0\n      ) {\n        // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n        const regex = new RegExp(emojiRegex(), 'g');\n        content = content.replace(\n          regex,\n          (match) => `<span class=\"str-chat__emoji-display-fix\">${match}</span>`\n        );\n        this.messageTextParts = [{ content, type: 'text' }];\n      } else {\n        this.messageTextParts = [];\n        let text = content;\n        this.message!.mentioned_users.forEach((user) => {\n          const mention = `@${user.name || user.id}`;\n          const precedingText = text.substring(0, text.indexOf(mention));\n          this.messageTextParts.push({\n            content: precedingText,\n            type: 'text',\n          });\n          this.messageTextParts.push({\n            content: mention,\n            type: 'mention',\n            user,\n          });\n          text = text.replace(precedingText + mention, '');\n        });\n        if (text) {\n          this.messageTextParts.push({ content: text, type: 'text' });\n        }\n      }\n    }\n  }\n}\n","<div\n  #container\n  class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n    message?.type\n  }} str-chat__message--{{ message?.status }} {{\n    message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n  }}\"\n  [class.str-chat__message--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.mobile-press]=\"isPressedOnMobile\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"\n    areReactionsEnabled !== false && hasReactions\n  \"\n  data-testid=\"message-container\"\n  (mouseleave)=\"isActionBoxOpen = false\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container\n        *ngIf=\"\n          isSentByCurrentUser &&\n          ((isLastSentMessage && message?.status === 'received') ||\n            message?.status === 'sending')\n        \"\n      >\n        <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n          <ng-container *ngTemplateOutlet=\"sendingStatus\"></ng-container>\n        </ng-container>\n        <ng-template #sentStatus>\n          <ng-container\n            *ngIf=\"\n              mode === 'main' &&\n                isMessageDeliveredAndRead &&\n                canDisplayReadStatus;\n              else deliveredStatus\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n          </ng-container>\n        </ng-template>\n      </ng-container>\n      <stream-avatar\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"message?.user?.image\"\n        [name]=\"message?.user?.name || message?.user?.id\"\n      ></stream-avatar>\n      <div class=\"str-chat__message-inner\">\n        <div\n          class=\"str-chat__message-simple__actions\"\n          data-testid=\"message-options\"\n          *ngIf=\"areOptionsVisible\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--options\n            \"\n            [class.str-chat-angular__message-simple__actions__action--options--editing]=\"\n              isEditing\n            \"\n          >\n            <stream-message-actions-box\n              [isOpen]=\"isActionBoxOpen\"\n              [isMine]=\"isSentByCurrentUser\"\n              [enabledActions]=\"enabledMessageActions\"\n              [message]=\"message\"\n              (displayedActionsCount)=\"visibleMessageActionsCount = $event\"\n              (isEditing)=\"isEditing = $event; isActionBoxOpen = !isEditing\"\n              [messageInputTemplate]=\"messageInputTemplate\"\n            ></stream-message-actions-box>\n            <stream-icon\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              icon=\"action-icon\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n            ></stream-icon>\n          </div>\n          <!-- eslint-disable @angular-eslint/template/conditional-complexity -->\n          <div\n            *ngIf=\"\n              enabledMessageActions.indexOf('send-reply') !== -1 &&\n              mode === 'main'\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--thread\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon icon=\"reply-in-thread\"></stream-icon>\n          </div>\n          <div\n            *ngIf=\"\n              areReactionsEnabled !== false &&\n              canReactToMessage !== false &&\n              enabledMessageActions.indexOf('send-reaction') !== -1\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--reactions\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon icon=\"reaction-icon\"></stream-icon>\n          </div>\n        </div>\n        <!-- eslint-enable @angular-eslint/template/conditional-complexity -->\n        <stream-message-reactions\n          *ngIf=\"areReactionsEnabled !== false\"\n          [messageReactionCounts]=\"message?.reaction_counts || {}\"\n          [latestReactions]=\"message?.latest_reactions || []\"\n          [(isSelectorOpen)]=\"isReactionSelectorOpen\"\n          [messageId]=\"message?.id\"\n          [ownReactions]=\"message?.own_reactions || []\"\n        ></stream-message-reactions>\n        <stream-attachment-list\n          *ngIf=\"hasAttachment && !message?.quoted_message\"\n          [attachments]=\"message!.attachments!\"\n          [messageId]=\"message!.id\"\n        ></stream-attachment-list>\n        <div class=\"str-chat__message-text\" *ngIf=\"message?.text\">\n          <div\n            data-testid=\"inner-message\"\n            class=\"\n              str-chat__message-text-inner str-chat__message-simple-text-inner\n            \"\n            [class.str-chat__message-light-text-inner--has-attachment]=\"\n              hasAttachment\n            \"\n            (click)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n            (keyup.enter)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"quotedMessage\"></ng-container>\n            <stream-attachment-list\n              *ngIf=\"hasAttachment && message?.quoted_message\"\n              [attachments]=\"message!.attachments!\"\n              [messageId]=\"message!.id\"\n            ></stream-attachment-list>\n            <div\n              data-testid=\"client-error-message\"\n              *ngIf=\"message?.type === 'error'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{ \"streamChat.Error · Unsent\" | translate }}\n            </div>\n            <div\n              data-testid=\"error-message\"\n              *ngIf=\"message?.status === 'failed'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{\n                (message?.errorStatusCode === 403\n                  ? \"streamChat.Message Failed · Unauthorized\"\n                  : \"streamChat.Message Failed · Click to try again\"\n                ) | translate\n              }}\n            </div>\n            <div\n              (click)=\"textClicked()\"\n              (keyup.enter)=\"textClicked()\"\n              data-testid=\"text\"\n            >\n              <p>\n                <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n                <ng-container *ngFor=\"let part of messageTextParts\">\n                  <span\n                    *ngIf=\"part.type === 'text'; else mention\"\n                    [innerHTML]=\"part.content\"\n                  ></span>\n                  <ng-template #mention>\n                    <ng-container *ngIf=\"mentionTemplate; else defaultMention\">\n                      <ng-container\n                        *ngTemplateOutlet=\"\n                          mentionTemplate;\n                          context: { user: part.user! }\n                        \"\n                      ></ng-container>\n                    </ng-container>\n                    <ng-template #defaultMention>\n                      <b>{{ part.content }}</b>\n                    </ng-template>\n                  </ng-template>\n                </ng-container>\n              </p>\n            </div>\n          </div>\n        </div>\n        <div class=\"str-chat__message-simple-reply-button\">\n          <button\n            *ngIf=\"\n              !!message?.reply_count &&\n              mode !== 'thread' &&\n              enabledMessageActions.indexOf('send-reply') !== -1\n            \"\n            class=\"str-chat__message-replies-count-button\"\n            data-testid=\"reply-count-button\"\n            (click)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon icon=\"reply\"></stream-icon>\n            {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n            replies' | translate:replyCountParam)}}\n          </button>\n        </div>\n        <div class=\"str-chat__message-data str-chat__message-simple-data\">\n          <span\n            data-testid=\"sender\"\n            *ngIf=\"!isSentByCurrentUser\"\n            class=\"str-chat__message-simple-name\"\n          >\n            {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n          </span>\n          <span data-testid=\"date\" class=\"str-chat__message-simple-timestamp\">\n            {{ parsedDate }}\n          </span>\n        </div>\n      </div>\n    </ng-container>\n  </ng-container>\n</div>\n\n<ng-template #sendingStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"sending-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Sending...\" | translate }}\n    </div>\n    <stream-loading-indicator\n      data-testid=\"loading-indicator\"\n    ></stream-loading-indicator>\n  </span>\n</ng-template>\n<ng-template #readStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"read-indicator\"\n  >\n    <div class=\"str-chat__tooltip\" data-testid=\"read-by-tooltip\">\n      {{ readByText }}\n    </div>\n    <stream-avatar\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    ></stream-avatar>\n    <span\n      data-test-id=\"read-by-length\"\n      *ngIf=\"isReadByMultipleUsers\"\n      class=\"str-chat__message-simple-status-number\"\n    >\n      {{ (message?.readBy)!.length }}\n    </span>\n  </span>\n</ng-template>\n<ng-template #deliveredStatus>\n  <span\n    *ngIf=\"mode === 'main'\"\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"delivered-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Delivered\" | translate }}\n    </div>\n    <stream-icon\n      data-testid=\"delivered-icon\"\n      icon=\"delivered-icon\"\n    ></stream-icon>\n  </span>\n</ng-template>\n\n<ng-template #deletedMessage>\n  <div data-testid=\"message-deleted-component\">\n    <div class=\"str-chat__message--deleted-inner\" translate>\n      streamChat.This message was deleted...\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #systemMessage>\n  <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n    <div class=\"str-chat__message--system__text\">\n      <div class=\"str-chat__message--system__line\"></div>\n      <p>{{ message?.text }}</p>\n      <div class=\"str-chat__message--system__line\"></div>\n    </div>\n    <div class=\"str-chat__message--system__date\">\n      {{ parsedDate }}\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n  >\n    <stream-avatar\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host\"\n      [imageUrl]=\"message?.quoted_message?.user?.image\"\n      [name]=\"\n        message?.quoted_message?.user?.name || message?.quoted_message?.user?.id\n      \"\n      [size]=\"20\"\n    ></stream-avatar>\n    <div class=\"quoted-message-inner\">\n      <stream-attachment-list\n        *ngIf=\"\n          message?.quoted_message?.attachments &&\n          message?.quoted_message?.attachments?.length\n        \"\n        [attachments]=\"quotedMessageAttachments\"\n        [messageId]=\"message?.quoted_message?.id\"\n      ></stream-attachment-list>\n      <div\n        data-testid=\"quoted-message-text\"\n        [innerHTML]=\"\n          message?.quoted_message?.html || message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message/message.component.ts","../../../../../projects/stream-chat-angular/src/lib/message/message.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  TemplateRef,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { getDeviceWidth } from '../device-width';\nimport { DefaultUserType, StreamMessage } from '../types';\nimport { parseDate } from './parse-date';\nimport { getReadByText } from './read-by-text';\nimport emojiRegex from 'emoji-regex';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](../concepts/message-interactions.mdx).\n */\n@Component({\n  selector: 'stream-message',\n  templateUrl: './message.component.html',\n  styles: [],\n})\nexport class MessageComponent implements OnChanges {\n  /**\n   * The input used for message edit. By default, the [default message input component](./MessageInputComponent.mdx) is used. To change the input for message edit, provide [your own custom template](./MessageInputComponent.mdx/#customization).\n   */\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  /**\n   * The template used to display a mention in a message. It receives the mentioned user in a variable called `user` with the type [`UserResponse`](https://github.com/GetStream/stream-chat-js/blob/master/src/types.ts). You can provide your own template if you want to [add actions to mentions](../code-examples/mention-actions.mdx).\n   */\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\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 reactions are displayed. If you use the default chat UI you can also set this using the [`MessageList`](./MessageListComponent.mdx) component.\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() areReactionsEnabled: boolean | undefined;\n  /**\n   * If true, the user can add reactions to the message. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() canReactToMessage: boolean | undefined;\n  /**\n   * If `true`, the message status (sending, sent, who read the message) is displayed.\n   */\n  @Input() isLastSentMessage: boolean | undefined;\n  /**\n   * If true, the read indicator is displayed. The [`MessageList`](./MessageListComponent.mdx) component automatically sets this based on [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead\n   */\n  @Input() canReceiveReadEvents: 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  isEditing: boolean | undefined;\n  isActionBoxOpen = false;\n  isReactionSelectorOpen = false;\n  isPressedOnMobile = false;\n  visibleMessageActionsCount = 0;\n  messageTextParts: MessagePart[] = [];\n  private user: UserResponse<DefaultUserType> | undefined;\n  @ViewChild('container') private container:\n    | ElementRef<HTMLElement>\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService\n  ) {\n    this.user = this.chatClientService.chatClient.user;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.createMessageParts();\n    }\n  }\n\n  get isSentByCurrentUser() {\n    return this.message?.user?.id === this.user?.id;\n  }\n\n  get readByText() {\n    return getReadByText(this.message!.readBy);\n  }\n\n  get lastReadUser() {\n    return this.message?.readBy.filter((u) => u.id !== this.user?.id)[0];\n  }\n\n  get isOnlyReadByMe() {\n    return this.message && this.message.readBy.length === 0;\n  }\n\n  get isReadByMultipleUsers() {\n    return this.message && this.message.readBy.length > 1;\n  }\n\n  get isMessageDeliveredAndRead() {\n    return (\n      this.message &&\n      this.message.readBy &&\n      this.message.status === 'received' &&\n      this.message.readBy.length > 0\n    );\n  }\n\n  get parsedDate() {\n    if (!this.message || !this.message?.created_at) {\n      return;\n    }\n    return parseDate(this.message.created_at);\n  }\n\n  get areOptionsVisible() {\n    if (!this.message) {\n      return false;\n    }\n    return !(\n      !this.message.type ||\n      this.message.type === 'error' ||\n      this.message.type === 'system' ||\n      this.message.type === 'ephemeral' ||\n      this.message.status === 'failed' ||\n      this.message.status === 'sending' ||\n      (this.mode === 'thread' && !this.message.parent_id)\n    );\n  }\n\n  get hasAttachment() {\n    return !!this.message?.attachments && !!this.message.attachments.length;\n  }\n\n  get hasReactions() {\n    return (\n      !!this.message?.reaction_counts &&\n      Object.keys(this.message.reaction_counts).length > 0\n    );\n  }\n\n  get replyCountParam() {\n    return { replyCount: this.message?.reply_count };\n  }\n\n  get canDisplayReadStatus() {\n    return (\n      this.canReceiveReadEvents !== false &&\n      this.enabledMessageActions.indexOf('read-events') !== -1\n    );\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.message?.quoted_message?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  resendMessage() {\n    void this.channelService.resendMessage(this.message!);\n  }\n\n  textClicked() {\n    if (getDeviceWidth().device !== 'mobile') {\n      this.isPressedOnMobile = false;\n      return;\n    }\n    if (this.isPressedOnMobile) {\n      return;\n    }\n    this.isPressedOnMobile = true;\n    const eventHandler = (event: Event) => {\n      if (!this.container?.nativeElement.contains(event.target as Node)) {\n        this.isPressedOnMobile = false;\n        window.removeEventListener('click', eventHandler);\n      }\n    };\n    window.addEventListener('click', eventHandler);\n  }\n\n  setAsActiveParentMessage() {\n    void this.channelService.setAsActiveParentMessage(this.message);\n  }\n\n  private createMessageParts() {\n    let content = this.message?.html || this.message?.text;\n    if (!content) {\n      this.messageTextParts = [];\n    } else {\n      // Backend will wrap HTML content with <p></p>\\n\n      if (content.startsWith('<p>')) {\n        content = content.replace('<p>', '');\n      }\n      if (content.endsWith('</p>\\n')) {\n        content = content.replace('</p>\\n', '');\n      }\n      if (\n        !this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0\n      ) {\n        // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n        const regex = new RegExp(emojiRegex(), 'g');\n        // Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome\n        /* eslint-disable @typescript-eslint/no-unsafe-member-access */\n        const isChrome =\n          !!(window as any).chrome &&\n          typeof (window as any).opr === 'undefined';\n        /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n        content = content.replace(\n          regex,\n          (match) =>\n            `<span ${\n              isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n            }>${match}</span>`\n        );\n        this.messageTextParts = [{ content, type: 'text' }];\n      } else {\n        this.messageTextParts = [];\n        let text = content;\n        this.message!.mentioned_users.forEach((user) => {\n          const mention = `@${user.name || user.id}`;\n          const precedingText = text.substring(0, text.indexOf(mention));\n          this.messageTextParts.push({\n            content: precedingText,\n            type: 'text',\n          });\n          this.messageTextParts.push({\n            content: mention,\n            type: 'mention',\n            user,\n          });\n          text = text.replace(precedingText + mention, '');\n        });\n        if (text) {\n          this.messageTextParts.push({ content: text, type: 'text' });\n        }\n      }\n    }\n  }\n}\n","<div\n  #container\n  class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n    message?.type\n  }} str-chat__message--{{ message?.status }} {{\n    message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n  }}\"\n  [class.str-chat__message--me]=\"isSentByCurrentUser\"\n  [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n  [class.mobile-press]=\"isPressedOnMobile\"\n  [class.str-chat__message--has-attachment]=\"hasAttachment\"\n  [class.str-chat__message--with-reactions]=\"\n    areReactionsEnabled !== false && hasReactions\n  \"\n  data-testid=\"message-container\"\n  (mouseleave)=\"isActionBoxOpen = false\"\n>\n  <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n    <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n      <ng-container\n        *ngIf=\"\n          isSentByCurrentUser &&\n          ((isLastSentMessage && message?.status === 'received') ||\n            message?.status === 'sending')\n        \"\n      >\n        <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n          <ng-container *ngTemplateOutlet=\"sendingStatus\"></ng-container>\n        </ng-container>\n        <ng-template #sentStatus>\n          <ng-container\n            *ngIf=\"\n              mode === 'main' &&\n                isMessageDeliveredAndRead &&\n                canDisplayReadStatus;\n              else deliveredStatus\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"readStatus\"></ng-container>\n          </ng-container>\n        </ng-template>\n      </ng-container>\n      <stream-avatar\n        data-testid=\"avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"message?.user?.image\"\n        [name]=\"message?.user?.name || message?.user?.id\"\n      ></stream-avatar>\n      <div class=\"str-chat__message-inner\">\n        <div\n          class=\"str-chat__message-simple__actions\"\n          data-testid=\"message-options\"\n          *ngIf=\"areOptionsVisible\"\n        >\n          <div\n            data-testid=\"message-actions-container\"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--options\n            \"\n            [class.str-chat-angular__message-simple__actions__action--options--editing]=\"\n              isEditing\n            \"\n          >\n            <stream-message-actions-box\n              [isOpen]=\"isActionBoxOpen\"\n              [isMine]=\"isSentByCurrentUser\"\n              [enabledActions]=\"enabledMessageActions\"\n              [message]=\"message\"\n              (displayedActionsCount)=\"visibleMessageActionsCount = $event\"\n              (isEditing)=\"isEditing = $event; isActionBoxOpen = !isEditing\"\n              [messageInputTemplate]=\"messageInputTemplate\"\n            ></stream-message-actions-box>\n            <stream-icon\n              *ngIf=\"visibleMessageActionsCount > 0\"\n              data-testid=\"action-icon\"\n              icon=\"action-icon\"\n              (keyup.enter)=\"isActionBoxOpen = !isActionBoxOpen\"\n              (click)=\"isActionBoxOpen = !isActionBoxOpen\"\n            ></stream-icon>\n          </div>\n          <!-- eslint-disable @angular-eslint/template/conditional-complexity -->\n          <div\n            *ngIf=\"\n              enabledMessageActions.indexOf('send-reply') !== -1 &&\n              mode === 'main'\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--thread\n            \"\n            data-testid=\"reply-in-thread\"\n            (click)=\"setAsActiveParentMessage()\"\n            (keyup.enter)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon icon=\"reply-in-thread\"></stream-icon>\n          </div>\n          <div\n            *ngIf=\"\n              areReactionsEnabled !== false &&\n              canReactToMessage !== false &&\n              enabledMessageActions.indexOf('send-reaction') !== -1\n            \"\n            class=\"\n              str-chat__message-simple__actions__action\n              str-chat__message-simple__actions__action--reactions\n            \"\n            data-testid=\"reaction-icon\"\n            (click)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n            (keyup.enter)=\"isReactionSelectorOpen = !isReactionSelectorOpen\"\n          >\n            <stream-icon icon=\"reaction-icon\"></stream-icon>\n          </div>\n        </div>\n        <!-- eslint-enable @angular-eslint/template/conditional-complexity -->\n        <stream-message-reactions\n          *ngIf=\"areReactionsEnabled !== false\"\n          [messageReactionCounts]=\"message?.reaction_counts || {}\"\n          [latestReactions]=\"message?.latest_reactions || []\"\n          [(isSelectorOpen)]=\"isReactionSelectorOpen\"\n          [messageId]=\"message?.id\"\n          [ownReactions]=\"message?.own_reactions || []\"\n        ></stream-message-reactions>\n        <stream-attachment-list\n          *ngIf=\"hasAttachment && !message?.quoted_message\"\n          [attachments]=\"message!.attachments!\"\n          [messageId]=\"message!.id\"\n        ></stream-attachment-list>\n        <div class=\"str-chat__message-text\" *ngIf=\"message?.text\">\n          <div\n            data-testid=\"inner-message\"\n            class=\"\n              str-chat__message-text-inner str-chat__message-simple-text-inner\n            \"\n            [class.str-chat__message-light-text-inner--has-attachment]=\"\n              hasAttachment\n            \"\n            (click)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n            (keyup.enter)=\"\n              message?.status === 'failed' && message?.errorStatusCode !== 403\n                ? resendMessage()\n                : undefined\n            \"\n          >\n            <ng-container *ngTemplateOutlet=\"quotedMessage\"></ng-container>\n            <stream-attachment-list\n              *ngIf=\"hasAttachment && message?.quoted_message\"\n              [attachments]=\"message!.attachments!\"\n              [messageId]=\"message!.id\"\n            ></stream-attachment-list>\n            <div\n              data-testid=\"client-error-message\"\n              *ngIf=\"message?.type === 'error'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{ \"streamChat.Error · Unsent\" | translate }}\n            </div>\n            <div\n              data-testid=\"error-message\"\n              *ngIf=\"message?.status === 'failed'\"\n              class=\"str-chat__simple-message--error-message\"\n            >\n              {{\n                (message?.errorStatusCode === 403\n                  ? \"streamChat.Message Failed · Unauthorized\"\n                  : \"streamChat.Message Failed · Click to try again\"\n                ) | translate\n              }}\n            </div>\n            <div\n              (click)=\"textClicked()\"\n              (keyup.enter)=\"textClicked()\"\n              data-testid=\"text\"\n            >\n              <p>\n                <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n                <ng-container *ngFor=\"let part of messageTextParts\">\n                  <span\n                    *ngIf=\"part.type === 'text'; else mention\"\n                    [innerHTML]=\"part.content\"\n                  ></span>\n                  <ng-template #mention>\n                    <ng-container *ngIf=\"mentionTemplate; else defaultMention\">\n                      <ng-container\n                        *ngTemplateOutlet=\"\n                          mentionTemplate;\n                          context: { user: part.user! }\n                        \"\n                      ></ng-container>\n                    </ng-container>\n                    <ng-template #defaultMention>\n                      <b>{{ part.content }}</b>\n                    </ng-template>\n                  </ng-template>\n                </ng-container>\n              </p>\n            </div>\n          </div>\n        </div>\n        <div class=\"str-chat__message-simple-reply-button\">\n          <button\n            *ngIf=\"\n              !!message?.reply_count &&\n              mode !== 'thread' &&\n              enabledMessageActions.indexOf('send-reply') !== -1\n            \"\n            class=\"str-chat__message-replies-count-button\"\n            data-testid=\"reply-count-button\"\n            (click)=\"setAsActiveParentMessage()\"\n          >\n            <stream-icon icon=\"reply\"></stream-icon>\n            {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n            replies' | translate:replyCountParam)}}\n          </button>\n        </div>\n        <div class=\"str-chat__message-data str-chat__message-simple-data\">\n          <span\n            data-testid=\"sender\"\n            *ngIf=\"!isSentByCurrentUser\"\n            class=\"str-chat__message-simple-name\"\n          >\n            {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n          </span>\n          <span data-testid=\"date\" class=\"str-chat__message-simple-timestamp\">\n            {{ parsedDate }}\n          </span>\n        </div>\n      </div>\n    </ng-container>\n  </ng-container>\n</div>\n\n<ng-template #sendingStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"sending-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Sending...\" | translate }}\n    </div>\n    <stream-loading-indicator\n      data-testid=\"loading-indicator\"\n    ></stream-loading-indicator>\n  </span>\n</ng-template>\n<ng-template #readStatus>\n  <span\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"read-indicator\"\n  >\n    <div class=\"str-chat__tooltip\" data-testid=\"read-by-tooltip\">\n      {{ readByText }}\n    </div>\n    <stream-avatar\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    ></stream-avatar>\n    <span\n      data-test-id=\"read-by-length\"\n      *ngIf=\"isReadByMultipleUsers\"\n      class=\"str-chat__message-simple-status-number\"\n    >\n      {{ (message?.readBy)!.length }}\n    </span>\n  </span>\n</ng-template>\n<ng-template #deliveredStatus>\n  <span\n    *ngIf=\"mode === 'main'\"\n    class=\"\n      str-chat__message-simple-status str-chat__message-simple-status-angular\n    \"\n    data-testid=\"delivered-indicator\"\n  >\n    <div class=\"str-chat__tooltip\">\n      {{ \"streamChat.Delivered\" | translate }}\n    </div>\n    <stream-icon\n      data-testid=\"delivered-icon\"\n      icon=\"delivered-icon\"\n    ></stream-icon>\n  </span>\n</ng-template>\n\n<ng-template #deletedMessage>\n  <div data-testid=\"message-deleted-component\">\n    <div class=\"str-chat__message--deleted-inner\" translate>\n      streamChat.This message was deleted...\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #systemMessage>\n  <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n    <div class=\"str-chat__message--system__text\">\n      <div class=\"str-chat__message--system__line\"></div>\n      <p>{{ message?.text }}</p>\n      <div class=\"str-chat__message--system__line\"></div>\n    </div>\n    <div class=\"str-chat__message--system__date\">\n      {{ parsedDate }}\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #quotedMessage>\n  <div\n    *ngIf=\"message?.quoted_message\"\n    class=\"quoted-message\"\n    data-testid=\"quoted-message-container\"\n    [class.mine]=\"isSentByCurrentUser\"\n  >\n    <stream-avatar\n      data-testid=\"qouted-message-avatar\"\n      class=\"str-chat-angular__avatar-host\"\n      [imageUrl]=\"message?.quoted_message?.user?.image\"\n      [name]=\"\n        message?.quoted_message?.user?.name || message?.quoted_message?.user?.id\n      \"\n      [size]=\"20\"\n    ></stream-avatar>\n    <div class=\"quoted-message-inner\">\n      <stream-attachment-list\n        *ngIf=\"\n          message?.quoted_message?.attachments &&\n          message?.quoted_message?.attachments?.length\n        \"\n        [attachments]=\"quotedMessageAttachments\"\n        [messageId]=\"message?.quoted_message?.id\"\n      ></stream-attachment-list>\n      <div\n        data-testid=\"quoted-message-text\"\n        [innerHTML]=\"\n          message?.quoted_message?.html || message?.quoted_message?.text\n        \"\n      ></div>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
@@ -252,7 +252,7 @@ export class MessageListComponent {
|
|
|
252
252
|
}
|
|
253
253
|
}
|
|
254
254
|
MessageListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, deps: [{ token: i1.ChannelService }, { token: i2.ChatClientService }, { token: i3.ImageLoadService }], target: i0.ɵɵFactoryTarget.Component });
|
|
255
|
-
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", typingIndicatorTemplate: "typingIndicatorTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"], mode: "mode" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread'\"\n data-testid=\"parent-message\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\n \"\n ></ng-container>\n <div class=\"str-chat__thread-start\" translate>\n streamChat.Start of a new thread\n </div>\n </li>\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\n </ul>\n <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate;\n context: { usersTyping$: usersTyping$ }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultTypingIndicator>\n <div\n *ngIf=\"(usersTyping$ | async)?.length\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <stream-avatar\n *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n [name]=\"user.name || user.id\"\n [imageUrl]=\"user.image\"\n ></stream-avatar>\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n </div>\n </ng-template>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate,\n mode: mode\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n [mode]=\"mode\"\n ></stream-message>\n </ng-template>\n</ng-template>\n", components: [{ type: i4.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: i5.MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents", "mode"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i6.AsyncPipe } });
|
|
255
|
+
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", typingIndicatorTemplate: "typingIndicatorTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"], mode: "mode" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread'\"\n data-testid=\"parent-message\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\n \"\n ></ng-container>\n <div class=\"str-chat__thread-start\" translate>\n streamChat.Start of a new thread\n </div>\n </li>\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\n </ul>\n <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate;\n context: { usersTyping$: usersTyping$ }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultTypingIndicator>\n <div\n *ngIf=\"(usersTyping$ | async)?.length\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <stream-avatar\n *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n [name]=\"user.name || user.id\"\n [imageUrl]=\"user.image\"\n ></stream-avatar>\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n </div>\n </ng-template>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate,\n mode: mode,\n enabledMessageActions: enabledMessageActions\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n [mode]=\"mode\"\n ></stream-message>\n </ng-template>\n</ng-template>\n", components: [{ type: i4.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: i5.MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents", "mode"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i6.AsyncPipe } });
|
|
256
256
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
|
|
257
257
|
type: Component,
|
|
258
258
|
args: [{
|
|
@@ -285,4 +285,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
285
285
|
type: ViewChild,
|
|
286
286
|
args: ['parentMessageElement']
|
|
287
287
|
}] } });
|
|
288
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { Observable, Subscription } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { DefaultUserType, StreamMessage } from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { ImageLoadService } from './image-load.service';\nimport { UserResponse } from 'stream-chat';\n\n/**\n * The `MessageList` component renders a scrollable list of messages.\n */\n@Component({\n  selector: 'stream-message-list',\n  templateUrl: './message-list.component.html',\n  styles: [],\n})\nexport class MessageListComponent\n  implements AfterViewChecked, OnChanges, OnInit, OnDestroy\n{\n  /**\n   * By default, the [default message component](./MessageComponent.mdx) is used. To change the contents of the message, provide [your own custom message template](./MessageComponent.mdx/#customization).\n   */\n  @Input() messageTemplate: TemplateRef<any> | undefined;\n  /**\n   * The input used for message edit. By default, the [default message input component](./MessageInputComponent.mdx) is used. To change the input for message edit, provide [your own custom template](./MessageInputComponent.mdx/#customization).\n   */\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  /**\n   * The template used to display a mention in a message. It receives the mentioned user in a variable called `user` with the type [`UserResponse`](https://github.com/GetStream/stream-chat-js/blob/master/src/types.ts). You can provide your own template if you want to [add actions to mentions](../code-examples/mention-actions.mdx).\n   */\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\n  /**\n   * You can provide your own typing indicator template instead of the default one.\n   */\n  @Input() typingIndicatorTemplate:\n    | TemplateRef<{ usersTyping$: Observable<UserResponse<DefaultUserType>[]> }>\n    | undefined;\n  /**\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead. If true, the message reactions are displayed. Users can also react to messages if they have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   */\n  @Input() areReactionsEnabled: boolean | undefined = undefined;\n  /**\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead. The list of [actions that are enabled](./MessageActionsBoxComponent.mdx), please note that the user also has to have the necessary [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) for actions to work. Unathorized actions won't be displayed on the UI. The `MessgaeList` component makes the necessary checks before passing the actions to the `Message` component.\n   */\n  /* eslint-disable-next-line @angular-eslint/no-input-rename */\n  @Input('enabledMessageActions') enabledMessageActionsInput:\n    | string[]\n    | undefined = undefined;\n  /**\n   * Determines if the message list should display channel messages or [thread messages](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'main' | 'thread' = 'main';\n  messages$!: Observable<StreamMessage[]>;\n  canReactToMessage: boolean | undefined;\n  canReceiveReadEvents: boolean | undefined;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host';\n  unreadMessageCount = 0;\n  isUserScrolledUp: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  parentMessage: StreamMessage | undefined;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  @ViewChild('parentMessageElement')\n  private parentMessageElement!: ElementRef<HTMLElement>;\n  private latestMessageDate: Date | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessageDate: Date | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private authorizedMessageActions: string[] = ['flag'];\n  private readonly isUserScrolledUpThreshold = 300;\n  private subscriptions: Subscription[] = [];\n  private prevScrollTop: number | undefined;\n  private usersTypingInChannel$!: Observable<UserResponse<DefaultUserType>[]>;\n  private usersTypingInThread$!: Observable<UserResponse<DefaultUserType>[]>;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private imageLoadService: ImageLoadService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.resetScrollState();\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.canReactToMessage = capabilites.indexOf('send-reaction') !== -1;\n          this.canReceiveReadEvents = capabilites.indexOf('read-events') !== -1;\n          this.authorizedMessageActions = [];\n          if (this.canReactToMessage) {\n            this.authorizedMessageActions.push('send-reaction');\n          }\n          if (this.canReceiveReadEvents) {\n            this.authorizedMessageActions.push('read-events');\n          }\n          if (capabilites.indexOf('flag-message') !== -1) {\n            this.authorizedMessageActions.push('flag');\n          }\n          if (capabilites.indexOf('update-own-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n          }\n          if (capabilites.indexOf('update-any-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n            this.authorizedMessageActions.push('edit-any');\n          }\n          if (capabilites.indexOf('delete-own-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n          }\n          if (capabilites.indexOf('delete-any-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n            this.authorizedMessageActions.push('delete-any');\n          }\n          if (capabilites.indexOf('send-reply') !== -1) {\n            this.authorizedMessageActions.push('send-reply');\n          }\n          if (capabilites.indexOf('quote-message') !== -1) {\n            this.authorizedMessageActions.push('quote-message');\n          }\n          this.setEnabledActions();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.imageLoadService.imageLoad$.subscribe(() => {\n        if (!this.isUserScrolledUp) {\n          this.scrollToBottom();\n          // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n          setTimeout(() => {\n            this.scrollToBottom();\n          }, 300);\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.channelService.activeParentMessage$.subscribe((message) => {\n        if (\n          message &&\n          this.parentMessage &&\n          message.id !== this.parentMessage.id &&\n          this.mode === 'thread'\n        ) {\n          this.resetScrollState();\n        }\n        this.parentMessage = message;\n      })\n    );\n    this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;\n    this.usersTypingInThread$ = this.channelService.usersTypingInThread$;\n  }\n\n  ngOnInit(): void {\n    this.setMessages$();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.enabledMessageActionsInput) {\n      this.setEnabledActions();\n    }\n    if (changes.mode) {\n      this.setMessages$();\n    }\n  }\n\n  ngAfterViewChecked() {\n    if (this.hasNewMessages) {\n      if (!this.isUserScrolledUp || this.isNewMessageSentByUser) {\n        this.scrollToBottom();\n        // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n        setTimeout(() => {\n          this.scrollToBottom();\n        }, 300);\n      }\n      this.hasNewMessages = false;\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    } else if (this.olderMassagesLoaded) {\n      this.preserveScrollbarPosition();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.olderMassagesLoaded = false;\n    } else if (\n      this.containerHeight !== undefined &&\n      this.containerHeight < this.scrollContainer.nativeElement.scrollHeight &&\n      !this.isUserScrolledUp\n    ) {\n      this.scrollToBottom();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get usersTyping$() {\n    return this.mode === 'thread'\n      ? this.usersTypingInThread$\n      : this.usersTypingInChannel$;\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  trackByUserId(index: number, user: UserResponse) {\n    return user.id;\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrolled() {\n    this.isUserScrolledUp =\n      this.scrollContainer.nativeElement.scrollHeight -\n        (this.scrollContainer.nativeElement.scrollTop +\n          this.scrollContainer.nativeElement.clientHeight) >\n      this.isUserScrolledUpThreshold;\n    if (!this.isUserScrolledUp) {\n      this.unreadMessageCount = 0;\n    }\n    if (\n      this.scrollContainer.nativeElement.scrollTop <=\n        (this.parentMessageElement?.nativeElement.clientHeight || 0) &&\n      (this.prevScrollTop === undefined ||\n        this.prevScrollTop >\n          (this.parentMessageElement?.nativeElement.clientHeight || 0))\n    ) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.mode === 'main'\n        ? void this.channelService.loadMoreMessages()\n        : void this.channelService.loadMoreThreadReplies();\n    }\n    this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;\n  }\n\n  private preserveScrollbarPosition() {\n    this.scrollContainer.nativeElement.scrollTop =\n      (this.prevScrollTop || 0) +\n      (this.scrollContainer.nativeElement.scrollHeight - this.containerHeight!);\n  }\n\n  private setEnabledActions() {\n    this.enabledMessageActions = [];\n    if (!this.enabledMessageActionsInput) {\n      this.enabledMessageActions = this.authorizedMessageActions;\n      return;\n    }\n    this.enabledMessageActionsInput = [\n      ...this.enabledMessageActionsInput,\n      'send-reaction',\n      'read-events',\n      'send-reply',\n      'quote-message',\n    ];\n    this.enabledMessageActionsInput.forEach((action) => {\n      const isAuthorized = this.authorizedMessageActions.indexOf(action) !== -1;\n      if (isAuthorized) {\n        this.enabledMessageActions.push(action);\n      }\n    });\n  }\n\n  private setMessages$() {\n    this.messages$ = (\n      this.mode === 'main'\n        ? this.channelService.activeChannelMessages$\n        : this.channelService.activeThreadMessages$\n    ).pipe(\n      tap((messages) => {\n        if (messages.length === 0) {\n          return;\n        }\n        const currentLatestMessageDate =\n          messages[messages.length - 1].created_at;\n        if (\n          !this.latestMessageDate ||\n          this.latestMessageDate?.getTime() < currentLatestMessageDate.getTime()\n        ) {\n          this.latestMessageDate = currentLatestMessageDate;\n          this.hasNewMessages = true;\n          this.isNewMessageSentByUser =\n            messages[messages.length - 1].user?.id ===\n            this.chatClientService.chatClient?.user?.id;\n          if (this.isUserScrolledUp) {\n            this.unreadMessageCount++;\n          }\n        }\n        const currentOldestMessageDate = messages[0].created_at;\n        if (!this.oldestMessageDate) {\n          this.oldestMessageDate = currentOldestMessageDate;\n        } else if (\n          this.oldestMessageDate?.getTime() > currentOldestMessageDate.getTime()\n        ) {\n          this.oldestMessageDate = currentOldestMessageDate;\n          this.olderMassagesLoaded = true;\n        }\n      }),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      }),\n      tap(\n        (messages) =>\n          (this.lastSentMessageId = [...messages]\n            .reverse()\n            .find(\n              (m) =>\n                m.user?.id === this.chatClientService.chatClient?.user?.id &&\n                m.status !== 'sending'\n            )?.id)\n      )\n    );\n  }\n\n  private resetScrollState() {\n    this.latestMessageDate = undefined;\n    this.hasNewMessages = true;\n    this.isUserScrolledUp = false;\n    this.containerHeight = undefined;\n    this.olderMassagesLoaded = false;\n    this.oldestMessageDate = undefined;\n    this.unreadMessageCount = 0;\n    this.prevScrollTop = undefined;\n    this.isNewMessageSentByUser = undefined;\n  }\n}\n","<div\n  #scrollContainer\n  data-testid=\"scroll-container\"\n  class=\"str-chat__list\"\n  (scroll)=\"scrolled()\"\n>\n  <div class=\"str-chat__reverse-infinite-scroll\">\n    <ul class=\"str-chat__ul\">\n      <li\n        #parentMessageElement\n        *ngIf=\"mode === 'thread'\"\n        data-testid=\"parent-message\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage }\n          \"\n        ></ng-container>\n        <div class=\"str-chat__thread-start\" translate>\n          streamChat.Start of a new thread\n        </div>\n      </li>\n      <li\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: message }\n          \"\n        ></ng-container>\n      </li>\n    </ul>\n    <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n      <ng-container\n        *ngTemplateOutlet=\"\n          typingIndicatorTemplate;\n          context: { usersTyping$: usersTyping$ }\n        \"\n      ></ng-container>\n    </ng-container>\n    <ng-template #defaultTypingIndicator>\n      <div\n        *ngIf=\"(usersTyping$ | async)?.length\"\n        data-testid=\"typing-indicator\"\n        class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n      >\n        <stream-avatar\n          *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n          [name]=\"user.name || user.id\"\n          [imageUrl]=\"user.image\"\n        ></stream-avatar>\n        <div class=\"str-chat__typing-indicator__dots\">\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n        </div>\n      </div>\n    </ng-template>\n  </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n  <button\n    data-testid=\"scroll-to-bottom\"\n    *ngIf=\"isUserScrolledUp\"\n    class=\"\n      str-chat__message-notification\n      str-chat__message-notification-right\n      str-chat__message-notification-scroll-down\n    \"\n    (keyup.enter)=\"scrollToBottom()\"\n    (click)=\"scrollToBottom()\"\n  >\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-down-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n  <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        messageTemplate;\n        context: {\n          message: message,\n          areReactionsEnabled: areReactionsEnabled,\n          canReactToMessage: canReactToMessage,\n          lastSentMessageId: !!(\n            lastSentMessageId && message?.id === lastSentMessageId\n          ),\n          canReceiveReadEvents: canReceiveReadEvents,\n          messageInputTemplate: messageInputTemplate,\n          mentionTemplate: mentionTemplate,\n          mode: mode\n        }\n      \"\n    ></ng-container>\n  </ng-container>\n  <ng-template #defaultMessageTemplate>\n    <stream-message\n      [message]=\"message\"\n      [areReactionsEnabled]=\"areReactionsEnabled\"\n      [canReactToMessage]=\"canReactToMessage\"\n      [isLastSentMessage]=\"\n        !!(lastSentMessageId && message?.id === lastSentMessageId)\n      \"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [canReceiveReadEvents]=\"canReceiveReadEvents\"\n      [messageInputTemplate]=\"messageInputTemplate\"\n      [mentionTemplate]=\"mentionTemplate\"\n      [mode]=\"mode\"\n    ></stream-message>\n  </ng-template>\n</ng-template>\n"]}
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-list/message-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { Observable, Subscription } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { DefaultUserType, StreamMessage } from '../types';\nimport { ChatClientService } from '../chat-client.service';\nimport { getGroupStyles, GroupStyle } from './group-styles';\nimport { ImageLoadService } from './image-load.service';\nimport { UserResponse } from 'stream-chat';\n\n/**\n * The `MessageList` component renders a scrollable list of messages.\n */\n@Component({\n  selector: 'stream-message-list',\n  templateUrl: './message-list.component.html',\n  styles: [],\n})\nexport class MessageListComponent\n  implements AfterViewChecked, OnChanges, OnInit, OnDestroy\n{\n  /**\n   * By default, the [default message component](./MessageComponent.mdx) is used. To change the contents of the message, provide [your own custom message template](./MessageComponent.mdx/#customization).\n   */\n  @Input() messageTemplate: TemplateRef<any> | undefined;\n  /**\n   * The input used for message edit. By default, the [default message input component](./MessageInputComponent.mdx) is used. To change the input for message edit, provide [your own custom template](./MessageInputComponent.mdx/#customization).\n   */\n  @Input() messageInputTemplate: TemplateRef<any> | undefined;\n  /**\n   * The template used to display a mention in a message. It receives the mentioned user in a variable called `user` with the type [`UserResponse`](https://github.com/GetStream/stream-chat-js/blob/master/src/types.ts). You can provide your own template if you want to [add actions to mentions](../code-examples/mention-actions.mdx).\n   */\n  @Input() mentionTemplate: TemplateRef<any> | undefined;\n  /**\n   * You can provide your own typing indicator template instead of the default one.\n   */\n  @Input() typingIndicatorTemplate:\n    | TemplateRef<{ usersTyping$: Observable<UserResponse<DefaultUserType>[]> }>\n    | undefined;\n  /**\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead. If true, the message reactions are displayed. Users can also react to messages if they have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n   */\n  @Input() areReactionsEnabled: boolean | undefined = undefined;\n  /**\n   * @deprecated use [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) instead. The list of [actions that are enabled](./MessageActionsBoxComponent.mdx), please note that the user also has to have the necessary [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) for actions to work. Unathorized actions won't be displayed on the UI. The `MessgaeList` component makes the necessary checks before passing the actions to the `Message` component.\n   */\n  /* eslint-disable-next-line @angular-eslint/no-input-rename */\n  @Input('enabledMessageActions') enabledMessageActionsInput:\n    | string[]\n    | undefined = undefined;\n  /**\n   * Determines if the message list should display channel messages or [thread messages](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'main' | 'thread' = 'main';\n  messages$!: Observable<StreamMessage[]>;\n  canReactToMessage: boolean | undefined;\n  canReceiveReadEvents: boolean | undefined;\n  enabledMessageActions: string[] = [];\n  @HostBinding('class') private class =\n    'str-chat-angular__main-panel-inner str-chat-angular__message-list-host';\n  unreadMessageCount = 0;\n  isUserScrolledUp: boolean | undefined;\n  groupStyles: GroupStyle[] = [];\n  lastSentMessageId: string | undefined;\n  parentMessage: StreamMessage | undefined;\n  @ViewChild('scrollContainer')\n  private scrollContainer!: ElementRef<HTMLElement>;\n  @ViewChild('parentMessageElement')\n  private parentMessageElement!: ElementRef<HTMLElement>;\n  private latestMessageDate: Date | undefined;\n  private hasNewMessages: boolean | undefined;\n  private containerHeight: number | undefined;\n  private oldestMessageDate: Date | undefined;\n  private olderMassagesLoaded: boolean | undefined;\n  private isNewMessageSentByUser: boolean | undefined;\n  private authorizedMessageActions: string[] = ['flag'];\n  private readonly isUserScrolledUpThreshold = 300;\n  private subscriptions: Subscription[] = [];\n  private prevScrollTop: number | undefined;\n  private usersTypingInChannel$!: Observable<UserResponse<DefaultUserType>[]>;\n  private usersTypingInThread$!: Observable<UserResponse<DefaultUserType>[]>;\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService,\n    private imageLoadService: ImageLoadService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.resetScrollState();\n        const capabilites = channel?.data?.own_capabilities as string[];\n        if (capabilites) {\n          this.canReactToMessage = capabilites.indexOf('send-reaction') !== -1;\n          this.canReceiveReadEvents = capabilites.indexOf('read-events') !== -1;\n          this.authorizedMessageActions = [];\n          if (this.canReactToMessage) {\n            this.authorizedMessageActions.push('send-reaction');\n          }\n          if (this.canReceiveReadEvents) {\n            this.authorizedMessageActions.push('read-events');\n          }\n          if (capabilites.indexOf('flag-message') !== -1) {\n            this.authorizedMessageActions.push('flag');\n          }\n          if (capabilites.indexOf('update-own-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n          }\n          if (capabilites.indexOf('update-any-message') !== -1) {\n            this.authorizedMessageActions.push('edit');\n            this.authorizedMessageActions.push('edit-any');\n          }\n          if (capabilites.indexOf('delete-own-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n          }\n          if (capabilites.indexOf('delete-any-message') !== -1) {\n            this.authorizedMessageActions.push('delete');\n            this.authorizedMessageActions.push('delete-any');\n          }\n          if (capabilites.indexOf('send-reply') !== -1) {\n            this.authorizedMessageActions.push('send-reply');\n          }\n          if (capabilites.indexOf('quote-message') !== -1) {\n            this.authorizedMessageActions.push('quote-message');\n          }\n          this.setEnabledActions();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.imageLoadService.imageLoad$.subscribe(() => {\n        if (!this.isUserScrolledUp) {\n          this.scrollToBottom();\n          // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n          setTimeout(() => {\n            this.scrollToBottom();\n          }, 300);\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.channelService.activeParentMessage$.subscribe((message) => {\n        if (\n          message &&\n          this.parentMessage &&\n          message.id !== this.parentMessage.id &&\n          this.mode === 'thread'\n        ) {\n          this.resetScrollState();\n        }\n        this.parentMessage = message;\n      })\n    );\n    this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;\n    this.usersTypingInThread$ = this.channelService.usersTypingInThread$;\n  }\n\n  ngOnInit(): void {\n    this.setMessages$();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.enabledMessageActionsInput) {\n      this.setEnabledActions();\n    }\n    if (changes.mode) {\n      this.setMessages$();\n    }\n  }\n\n  ngAfterViewChecked() {\n    if (this.hasNewMessages) {\n      if (!this.isUserScrolledUp || this.isNewMessageSentByUser) {\n        this.scrollToBottom();\n        // Hacky and unreliable workaround to scroll down after loaded images move the scrollbar\n        setTimeout(() => {\n          this.scrollToBottom();\n        }, 300);\n      }\n      this.hasNewMessages = false;\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    } else if (this.olderMassagesLoaded) {\n      this.preserveScrollbarPosition();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.olderMassagesLoaded = false;\n    } else if (\n      this.containerHeight !== undefined &&\n      this.containerHeight < this.scrollContainer.nativeElement.scrollHeight &&\n      !this.isUserScrolledUp\n    ) {\n      this.scrollToBottom();\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get usersTyping$() {\n    return this.mode === 'thread'\n      ? this.usersTypingInThread$\n      : this.usersTypingInChannel$;\n  }\n\n  trackByMessageId(index: number, item: StreamMessage) {\n    return item.id;\n  }\n\n  trackByUserId(index: number, user: UserResponse) {\n    return user.id;\n  }\n\n  scrollToBottom(): void {\n    this.scrollContainer.nativeElement.scrollTop =\n      this.scrollContainer.nativeElement.scrollHeight;\n  }\n\n  scrolled() {\n    this.isUserScrolledUp =\n      this.scrollContainer.nativeElement.scrollHeight -\n        (this.scrollContainer.nativeElement.scrollTop +\n          this.scrollContainer.nativeElement.clientHeight) >\n      this.isUserScrolledUpThreshold;\n    if (!this.isUserScrolledUp) {\n      this.unreadMessageCount = 0;\n    }\n    if (\n      this.scrollContainer.nativeElement.scrollTop <=\n        (this.parentMessageElement?.nativeElement.clientHeight || 0) &&\n      (this.prevScrollTop === undefined ||\n        this.prevScrollTop >\n          (this.parentMessageElement?.nativeElement.clientHeight || 0))\n    ) {\n      this.containerHeight = this.scrollContainer.nativeElement.scrollHeight;\n      this.mode === 'main'\n        ? void this.channelService.loadMoreMessages()\n        : void this.channelService.loadMoreThreadReplies();\n    }\n    this.prevScrollTop = this.scrollContainer.nativeElement.scrollTop;\n  }\n\n  private preserveScrollbarPosition() {\n    this.scrollContainer.nativeElement.scrollTop =\n      (this.prevScrollTop || 0) +\n      (this.scrollContainer.nativeElement.scrollHeight - this.containerHeight!);\n  }\n\n  private setEnabledActions() {\n    this.enabledMessageActions = [];\n    if (!this.enabledMessageActionsInput) {\n      this.enabledMessageActions = this.authorizedMessageActions;\n      return;\n    }\n    this.enabledMessageActionsInput = [\n      ...this.enabledMessageActionsInput,\n      'send-reaction',\n      'read-events',\n      'send-reply',\n      'quote-message',\n    ];\n    this.enabledMessageActionsInput.forEach((action) => {\n      const isAuthorized = this.authorizedMessageActions.indexOf(action) !== -1;\n      if (isAuthorized) {\n        this.enabledMessageActions.push(action);\n      }\n    });\n  }\n\n  private setMessages$() {\n    this.messages$ = (\n      this.mode === 'main'\n        ? this.channelService.activeChannelMessages$\n        : this.channelService.activeThreadMessages$\n    ).pipe(\n      tap((messages) => {\n        if (messages.length === 0) {\n          return;\n        }\n        const currentLatestMessageDate =\n          messages[messages.length - 1].created_at;\n        if (\n          !this.latestMessageDate ||\n          this.latestMessageDate?.getTime() < currentLatestMessageDate.getTime()\n        ) {\n          this.latestMessageDate = currentLatestMessageDate;\n          this.hasNewMessages = true;\n          this.isNewMessageSentByUser =\n            messages[messages.length - 1].user?.id ===\n            this.chatClientService.chatClient?.user?.id;\n          if (this.isUserScrolledUp) {\n            this.unreadMessageCount++;\n          }\n        }\n        const currentOldestMessageDate = messages[0].created_at;\n        if (!this.oldestMessageDate) {\n          this.oldestMessageDate = currentOldestMessageDate;\n        } else if (\n          this.oldestMessageDate?.getTime() > currentOldestMessageDate.getTime()\n        ) {\n          this.oldestMessageDate = currentOldestMessageDate;\n          this.olderMassagesLoaded = true;\n        }\n      }),\n      tap((messages) => {\n        this.groupStyles = messages.map((m, i) =>\n          getGroupStyles(m, messages[i - 1], messages[i + 1])\n        );\n      }),\n      tap(\n        (messages) =>\n          (this.lastSentMessageId = [...messages]\n            .reverse()\n            .find(\n              (m) =>\n                m.user?.id === this.chatClientService.chatClient?.user?.id &&\n                m.status !== 'sending'\n            )?.id)\n      )\n    );\n  }\n\n  private resetScrollState() {\n    this.latestMessageDate = undefined;\n    this.hasNewMessages = true;\n    this.isUserScrolledUp = false;\n    this.containerHeight = undefined;\n    this.olderMassagesLoaded = false;\n    this.oldestMessageDate = undefined;\n    this.unreadMessageCount = 0;\n    this.prevScrollTop = undefined;\n    this.isNewMessageSentByUser = undefined;\n  }\n}\n","<div\n  #scrollContainer\n  data-testid=\"scroll-container\"\n  class=\"str-chat__list\"\n  (scroll)=\"scrolled()\"\n>\n  <div class=\"str-chat__reverse-infinite-scroll\">\n    <ul class=\"str-chat__ul\">\n      <li\n        #parentMessageElement\n        *ngIf=\"mode === 'thread'\"\n        data-testid=\"parent-message\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: parentMessage }\n          \"\n        ></ng-container>\n        <div class=\"str-chat__thread-start\" translate>\n          streamChat.Start of a new thread\n        </div>\n      </li>\n      <li\n        data-testclass=\"message\"\n        *ngFor=\"\n          let message of messages$ | async;\n          let i = index;\n          trackBy: trackByMessageId\n        \"\n        class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            messageTemplateContainer;\n            context: { message: message }\n          \"\n        ></ng-container>\n      </li>\n    </ul>\n    <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n      <ng-container\n        *ngTemplateOutlet=\"\n          typingIndicatorTemplate;\n          context: { usersTyping$: usersTyping$ }\n        \"\n      ></ng-container>\n    </ng-container>\n    <ng-template #defaultTypingIndicator>\n      <div\n        *ngIf=\"(usersTyping$ | async)?.length\"\n        data-testid=\"typing-indicator\"\n        class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n      >\n        <stream-avatar\n          *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n          [name]=\"user.name || user.id\"\n          [imageUrl]=\"user.image\"\n        ></stream-avatar>\n        <div class=\"str-chat__typing-indicator__dots\">\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n          <span class=\"str-chat__typing-indicator__dot\"></span>\n        </div>\n      </div>\n    </ng-template>\n  </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n  <button\n    data-testid=\"scroll-to-bottom\"\n    *ngIf=\"isUserScrolledUp\"\n    class=\"\n      str-chat__message-notification\n      str-chat__message-notification-right\n      str-chat__message-notification-scroll-down\n    \"\n    (keyup.enter)=\"scrollToBottom()\"\n    (click)=\"scrollToBottom()\"\n  >\n    <div\n      *ngIf=\"unreadMessageCount > 0\"\n      class=\"\n        str-chat__message-notification\n        str-chat__message-notification-scroll-down-unread-count\n      \"\n    >\n      {{ unreadMessageCount }}\n    </div>\n  </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n  <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        messageTemplate;\n        context: {\n          message: message,\n          areReactionsEnabled: areReactionsEnabled,\n          canReactToMessage: canReactToMessage,\n          lastSentMessageId: !!(\n            lastSentMessageId && message?.id === lastSentMessageId\n          ),\n          canReceiveReadEvents: canReceiveReadEvents,\n          messageInputTemplate: messageInputTemplate,\n          mentionTemplate: mentionTemplate,\n          mode: mode,\n          enabledMessageActions: enabledMessageActions\n        }\n      \"\n    ></ng-container>\n  </ng-container>\n  <ng-template #defaultMessageTemplate>\n    <stream-message\n      [message]=\"message\"\n      [areReactionsEnabled]=\"areReactionsEnabled\"\n      [canReactToMessage]=\"canReactToMessage\"\n      [isLastSentMessage]=\"\n        !!(lastSentMessageId && message?.id === lastSentMessageId)\n      \"\n      [enabledMessageActions]=\"enabledMessageActions\"\n      [canReceiveReadEvents]=\"canReceiveReadEvents\"\n      [messageInputTemplate]=\"messageInputTemplate\"\n      [mentionTemplate]=\"mentionTemplate\"\n      [mode]=\"mode\"\n    ></stream-message>\n  </ng-template>\n</ng-template>\n"]}
|
|
@@ -17,7 +17,7 @@ import transliterate from '@stream-io/transliterate';
|
|
|
17
17
|
import * as i6$1 from 'angular-mentions';
|
|
18
18
|
import { MentionModule } from 'angular-mentions';
|
|
19
19
|
|
|
20
|
-
const version = '2.
|
|
20
|
+
const version = '2.21.0';
|
|
21
21
|
|
|
22
22
|
/**
|
|
23
23
|
* The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
|
|
@@ -358,6 +358,21 @@ class ChannelService {
|
|
|
358
358
|
this.activeThreadMessagesSubject.next([]);
|
|
359
359
|
this.messageToQuoteSubject.next(undefined);
|
|
360
360
|
}
|
|
361
|
+
/**
|
|
362
|
+
* Deselects the currently active (if any) channel
|
|
363
|
+
*/
|
|
364
|
+
deselectActiveChannel() {
|
|
365
|
+
const activeChannel = this.activeChannelSubject.getValue();
|
|
366
|
+
if (!activeChannel) {
|
|
367
|
+
return;
|
|
368
|
+
}
|
|
369
|
+
this.activeChannelMessagesSubject.next([]);
|
|
370
|
+
this.activeChannelSubject.next(undefined);
|
|
371
|
+
this.activeParentMessageIdSubject.next(undefined);
|
|
372
|
+
this.activeThreadMessagesSubject.next([]);
|
|
373
|
+
this.latestMessageDateByUserByChannelsSubject.next({});
|
|
374
|
+
this.selectMessageToQuote(undefined);
|
|
375
|
+
}
|
|
361
376
|
/**
|
|
362
377
|
* Sets the given `message` as an active parent message. If `undefined` is provided, it will deleselect the current parent message.
|
|
363
378
|
* @param message
|
|
@@ -428,8 +443,10 @@ class ChannelService {
|
|
|
428
443
|
* @param filters
|
|
429
444
|
* @param sort
|
|
430
445
|
* @param options
|
|
446
|
+
* @param shouldSetActiveChannel Decides if the first channel in the result should be made as an active channel, or no channel should be marked as active
|
|
447
|
+
* @returns the list of channels found by the query
|
|
431
448
|
*/
|
|
432
|
-
init(filters, sort, options) {
|
|
449
|
+
init(filters, sort, options, shouldSetActiveChannel = true) {
|
|
433
450
|
return __awaiter(this, void 0, void 0, function* () {
|
|
434
451
|
this.filters = filters;
|
|
435
452
|
this.options = options || {
|
|
@@ -441,21 +458,17 @@ class ChannelService {
|
|
|
441
458
|
message_limit: this.messagePageSize,
|
|
442
459
|
};
|
|
443
460
|
this.sort = sort || { last_message_at: -1, updated_at: -1 };
|
|
444
|
-
yield this.queryChannels();
|
|
461
|
+
const result = yield this.queryChannels(shouldSetActiveChannel);
|
|
445
462
|
this.chatClientService.notification$.subscribe((notification) => void this.handleNotification(notification));
|
|
463
|
+
return result;
|
|
446
464
|
});
|
|
447
465
|
}
|
|
448
466
|
/**
|
|
449
467
|
* Resets the `activeChannel$`, `channels$` and `activeChannelMessages$` Observables. Useful when disconnecting a chat user, use in combination with [`disconnectUser`](./ChatClientService.mdx/#disconnectuser).
|
|
450
468
|
*/
|
|
451
469
|
reset() {
|
|
452
|
-
this.
|
|
453
|
-
this.activeChannelSubject.next(undefined);
|
|
454
|
-
this.activeParentMessageIdSubject.next(undefined);
|
|
455
|
-
this.activeThreadMessagesSubject.next([]);
|
|
470
|
+
this.deselectActiveChannel();
|
|
456
471
|
this.channelsSubject.next(undefined);
|
|
457
|
-
this.latestMessageDateByUserByChannelsSubject.next({});
|
|
458
|
-
this.selectMessageToQuote(undefined);
|
|
459
472
|
}
|
|
460
473
|
/**
|
|
461
474
|
* Loads the next page of channels. The page size can be set in the [query option](https://getstream.io/chat/docs/javascript/query_channels/?language=javascript#query-options) object.
|
|
@@ -463,7 +476,7 @@ class ChannelService {
|
|
|
463
476
|
loadMoreChannels() {
|
|
464
477
|
return __awaiter(this, void 0, void 0, function* () {
|
|
465
478
|
this.options.offset = this.channels.length;
|
|
466
|
-
yield this.queryChannels();
|
|
479
|
+
yield this.queryChannels(false);
|
|
467
480
|
});
|
|
468
481
|
}
|
|
469
482
|
/**
|
|
@@ -884,20 +897,24 @@ class ChannelService {
|
|
|
884
897
|
this.activeChannelSubscriptions.forEach((s) => s.unsubscribe());
|
|
885
898
|
this.activeChannelSubscriptions = [];
|
|
886
899
|
}
|
|
887
|
-
queryChannels() {
|
|
900
|
+
queryChannels(shouldSetActiveChannel) {
|
|
888
901
|
return __awaiter(this, void 0, void 0, function* () {
|
|
889
902
|
try {
|
|
890
903
|
const channels = yield this.chatClientService.chatClient.queryChannels(this.filters, this.sort, this.options);
|
|
891
904
|
channels.forEach((c) => this.watchForChannelEvents(c));
|
|
892
905
|
const prevChannels = this.channelsSubject.getValue() || [];
|
|
893
906
|
this.channelsSubject.next([...prevChannels, ...channels]);
|
|
894
|
-
if (channels.length > 0 &&
|
|
907
|
+
if (channels.length > 0 &&
|
|
908
|
+
!this.activeChannelSubject.getValue() &&
|
|
909
|
+
shouldSetActiveChannel) {
|
|
895
910
|
this.setAsActiveChannel(channels[0]);
|
|
896
911
|
}
|
|
897
912
|
this.hasMoreChannelsSubject.next(channels.length >= this.options.limit);
|
|
913
|
+
return channels;
|
|
898
914
|
}
|
|
899
915
|
catch (error) {
|
|
900
916
|
this.channelsSubject.error(error);
|
|
917
|
+
throw error;
|
|
901
918
|
}
|
|
902
919
|
});
|
|
903
920
|
}
|
|
@@ -3181,7 +3198,12 @@ class MessageComponent {
|
|
|
3181
3198
|
this.message.mentioned_users.length === 0) {
|
|
3182
3199
|
// Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223
|
|
3183
3200
|
const regex = new RegExp(emojiRegex(), 'g');
|
|
3184
|
-
|
|
3201
|
+
// Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome
|
|
3202
|
+
/* eslint-disable @typescript-eslint/no-unsafe-member-access */
|
|
3203
|
+
const isChrome = !!window.chrome &&
|
|
3204
|
+
typeof window.opr === 'undefined';
|
|
3205
|
+
/* eslint-enable @typescript-eslint/no-unsafe-member-access */
|
|
3206
|
+
content = content.replace(regex, (match) => `<span ${isChrome ? 'class="str-chat__emoji-display-fix"' : ''}>${match}</span>`);
|
|
3185
3207
|
this.messageTextParts = [{ content, type: 'text' }];
|
|
3186
3208
|
}
|
|
3187
3209
|
else {
|
|
@@ -3824,7 +3846,7 @@ class MessageListComponent {
|
|
|
3824
3846
|
}
|
|
3825
3847
|
}
|
|
3826
3848
|
MessageListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, deps: [{ token: ChannelService }, { token: ChatClientService }, { token: ImageLoadService }], target: i0.ɵɵFactoryTarget.Component });
|
|
3827
|
-
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", typingIndicatorTemplate: "typingIndicatorTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"], mode: "mode" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread'\"\n data-testid=\"parent-message\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\n \"\n ></ng-container>\n <div class=\"str-chat__thread-start\" translate>\n streamChat.Start of a new thread\n </div>\n </li>\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\n </ul>\n <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate;\n context: { usersTyping$: usersTyping$ }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultTypingIndicator>\n <div\n *ngIf=\"(usersTyping$ | async)?.length\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <stream-avatar\n *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n [name]=\"user.name || user.id\"\n [imageUrl]=\"user.image\"\n ></stream-avatar>\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n </div>\n </ng-template>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate,\n mode: mode\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n [mode]=\"mode\"\n ></stream-message>\n </ng-template>\n</ng-template>\n", components: [{ type: AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents", "mode"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i2.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i6.AsyncPipe } });
|
|
3849
|
+
MessageListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageListComponent, selector: "stream-message-list", inputs: { messageTemplate: "messageTemplate", messageInputTemplate: "messageInputTemplate", mentionTemplate: "mentionTemplate", typingIndicatorTemplate: "typingIndicatorTemplate", areReactionsEnabled: "areReactionsEnabled", enabledMessageActionsInput: ["enabledMessageActions", "enabledMessageActionsInput"], mode: "mode" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #scrollContainer\n data-testid=\"scroll-container\"\n class=\"str-chat__list\"\n (scroll)=\"scrolled()\"\n>\n <div class=\"str-chat__reverse-infinite-scroll\">\n <ul class=\"str-chat__ul\">\n <li\n #parentMessageElement\n *ngIf=\"mode === 'thread'\"\n data-testid=\"parent-message\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage }\n \"\n ></ng-container>\n <div class=\"str-chat__thread-start\" translate>\n streamChat.Start of a new thread\n </div>\n </li>\n <li\n data-testclass=\"message\"\n *ngFor=\"\n let message of messages$ | async;\n let i = index;\n trackBy: trackByMessageId\n \"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message }\n \"\n ></ng-container>\n </li>\n </ul>\n <ng-container *ngIf=\"typingIndicatorTemplate; else defaultTypingIndicator\">\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate;\n context: { usersTyping$: usersTyping$ }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultTypingIndicator>\n <div\n *ngIf=\"(usersTyping$ | async)?.length\"\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <stream-avatar\n *ngFor=\"let user of usersTyping$ | async; trackBy: trackByUserId\"\n [name]=\"user.name || user.id\"\n [imageUrl]=\"user.image\"\n ></stream-avatar>\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n </div>\n </ng-template>\n </div>\n</div>\n<div class=\"str-chat__list-notifications\">\n <button\n data-testid=\"scroll-to-bottom\"\n *ngIf=\"isUserScrolledUp\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-right\n str-chat__message-notification-scroll-down\n \"\n (keyup.enter)=\"scrollToBottom()\"\n (click)=\"scrollToBottom()\"\n >\n <div\n *ngIf=\"unreadMessageCount > 0\"\n class=\"\n str-chat__message-notification\n str-chat__message-notification-scroll-down-unread-count\n \"\n >\n {{ unreadMessageCount }}\n </div>\n </button>\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\">\n <ng-container *ngIf=\"messageTemplate; else defaultMessageTemplate\">\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate;\n context: {\n message: message,\n areReactionsEnabled: areReactionsEnabled,\n canReactToMessage: canReactToMessage,\n lastSentMessageId: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n canReceiveReadEvents: canReceiveReadEvents,\n messageInputTemplate: messageInputTemplate,\n mentionTemplate: mentionTemplate,\n mode: mode,\n enabledMessageActions: enabledMessageActions\n }\n \"\n ></ng-container>\n </ng-container>\n <ng-template #defaultMessageTemplate>\n <stream-message\n [message]=\"message\"\n [areReactionsEnabled]=\"areReactionsEnabled\"\n [canReactToMessage]=\"canReactToMessage\"\n [isLastSentMessage]=\"\n !!(lastSentMessageId && message?.id === lastSentMessageId)\n \"\n [enabledMessageActions]=\"enabledMessageActions\"\n [canReceiveReadEvents]=\"canReceiveReadEvents\"\n [messageInputTemplate]=\"messageInputTemplate\"\n [mentionTemplate]=\"mentionTemplate\"\n [mode]=\"mode\"\n ></stream-message>\n </ng-template>\n</ng-template>\n", components: [{ type: AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: MessageComponent, selector: "stream-message", inputs: ["messageInputTemplate", "mentionTemplate", "message", "enabledMessageActions", "areReactionsEnabled", "canReactToMessage", "isLastSentMessage", "canReceiveReadEvents", "mode"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i2.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i6.AsyncPipe } });
|
|
3828
3850
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageListComponent, decorators: [{
|
|
3829
3851
|
type: Component,
|
|
3830
3852
|
args: [{
|