stream-chat-angular 2.15.0 → 2.17.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.
@@ -1,7 +1,7 @@
1
1
  import { __awaiter } from "tslib";
2
2
  import { Component, EventEmitter, Inject, Input, Output, ViewChild, } from '@angular/core';
3
- import { Subject } from 'rxjs';
4
- import { first } from 'rxjs/operators';
3
+ import { combineLatest, Subject, timer } from 'rxjs';
4
+ import { first, map, take, tap } from 'rxjs/operators';
5
5
  import { AttachmentService } from '../attachment.service';
6
6
  import { textareaInjectionToken } from '../injection-tokens';
7
7
  import { TextareaDirective } from './textarea.directive';
@@ -46,6 +46,7 @@ export class MessageInputComponent {
46
46
  this.textareaValue = '';
47
47
  this.mentionedUsers = [];
48
48
  this.typingStart$ = new Subject();
49
+ this.isCooldownInProgress = false;
49
50
  this.subscriptions = [];
50
51
  this.isViewInited = false;
51
52
  this.subscriptions.push(this.attachmentService.attachmentUploadInProgressCounter$.subscribe((counter) => {
@@ -89,6 +90,26 @@ export class MessageInputComponent {
89
90
  this.configService.commandAutocompleteItemTemplate;
90
91
  this.emojiPickerTemplate = this.configService.emojiPickerTemplate;
91
92
  this.subscriptions.push(this.typingStart$.subscribe(() => void this.channelService.typingStarted(this.parentMessageId)));
93
+ this.subscriptions.push(combineLatest([
94
+ this.channelService.latestMessageDateByUserByChannels$,
95
+ this.channelService.activeChannel$,
96
+ ])
97
+ .pipe(map(([latestMessages, channel]) => [latestMessages[(channel === null || channel === void 0 ? void 0 : channel.cid) || ''], channel]))
98
+ .subscribe(([latestMessageDate, channel]) => {
99
+ var _a, _b, _c;
100
+ const cooldown = ((_a = channel === null || channel === void 0 ? void 0 : channel.data) === null || _a === void 0 ? void 0 : _a.cooldown) &&
101
+ latestMessageDate &&
102
+ Math.round(((_b = channel === null || channel === void 0 ? void 0 : channel.data) === null || _b === void 0 ? void 0 : _b.cooldown) -
103
+ (new Date().getTime() - latestMessageDate.getTime()) / 1000);
104
+ if (cooldown &&
105
+ cooldown > 0 &&
106
+ ((_c = channel === null || channel === void 0 ? void 0 : channel.data) === null || _c === void 0 ? void 0 : _c.own_capabilities).includes('slow-mode')) {
107
+ this.startCooldown(cooldown);
108
+ }
109
+ else if (this.isCooldownInProgress) {
110
+ this.stopCooldown();
111
+ }
112
+ }));
92
113
  }
93
114
  ngAfterViewInit() {
94
115
  this.isViewInited = true;
@@ -196,6 +217,17 @@ export class MessageInputComponent {
196
217
  ? [originalAttachments[0]]
197
218
  : [];
198
219
  }
220
+ get disabledTextareaText() {
221
+ if (!this.canSendMessages) {
222
+ return this.mode === 'thread'
223
+ ? "streamChat.You can't send thread replies in this channel"
224
+ : "streamChat.You can't send messages in this channel";
225
+ }
226
+ else if (this.cooldown$) {
227
+ return 'streamChat.Slow Mode ON';
228
+ }
229
+ return '';
230
+ }
199
231
  filesSelected(fileList) {
200
232
  return __awaiter(this, void 0, void 0, function* () {
201
233
  if (!(yield this.areAttachemntsValid(fileList))) {
@@ -297,9 +329,25 @@ export class MessageInputComponent {
297
329
  }
298
330
  return parentMessageId;
299
331
  }
332
+ startCooldown(cooldown) {
333
+ this.isCooldownInProgress = true;
334
+ this.cooldown$ = timer(0, 1000).pipe(take(cooldown + 1), map((v) => cooldown - v), tap((v) => {
335
+ if (v === 0) {
336
+ this.stopCooldown();
337
+ }
338
+ }));
339
+ }
340
+ stopCooldown() {
341
+ this.cooldown$ = undefined;
342
+ this.isCooldownInProgress = false;
343
+ // the anchor directive will be recreated because of *ngIf, so we will have to reinit the textarea as well
344
+ this.textareaRef = undefined;
345
+ // we can only create the textarea after the anchor was recreated, so we will have to wait a change detection cycle with setTimeout
346
+ setTimeout(() => this.initTextarea());
347
+ }
300
348
  }
301
349
  MessageInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, deps: [{ token: i1.ChannelService }, { token: i2.NotificationService }, { token: i3.AttachmentService }, { token: i4.MessageInputConfigService }, { token: textareaInjectionToken }, { token: i0.ComponentFactoryResolver }, { token: i0.ChangeDetectorRef }, { token: i5.ChatClientService }, { token: i6.EmojiInputService }], target: i0.ɵɵFactoryTarget.Component });
302
- MessageInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageInputComponent, selector: "stream-message-input", inputs: { isFileUploadEnabled: "isFileUploadEnabled", areMentionsEnabled: "areMentionsEnabled", mentionScope: "mentionScope", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", commandAutocompleteItemTemplate: "commandAutocompleteItemTemplate", emojiPickerTemplate: "emojiPickerTemplate", mode: "mode", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService, EmojiInputService], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "textareaAnchor", first: true, predicate: TextareaDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"{{\n mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n }}\"\n [class.str-chat__input-flat-has-attachments]=\"\n (attachmentUploads$ | async)!.length > 0\n \"\n [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n <div\n data-testid=\"quoted-message-container\"\n class=\"quoted-message-preview\"\n *ngIf=\"quotedMessage\"\n >\n <div class=\"quoted-message-preview-header\">\n <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n <button\n class=\"str-chat__square-button\"\n data-testid=\"remove-quote\"\n (click)=\"deselectMessageToQuote()\"\n (keyup.enter)=\"deselectMessageToQuote()\"\n >\n <stream-icon\n icon=\"close-no-outline\"\n style=\"font-size: 10px; line-height: 10px\"\n ></stream-icon>\n </button>\n </div>\n <div class=\"quoted-message-preview-content\">\n <stream-avatar\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host\"\n [imageUrl]=\"quotedMessage?.user?.image\"\n [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n [size]=\"20\"\n ></stream-avatar>\n <div class=\"quoted-message-preview-content-inner\">\n <stream-attachment-list\n *ngIf=\"\n quotedMessage?.attachments && quotedMessage?.attachments?.length\n \"\n [attachments]=\"quotedMessageAttachments\"\n [messageId]=\"quotedMessage?.id\"\n ></stream-attachment-list>\n <div\n data-testid=\"quoted-message-text\"\n [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n ></div>\n </div>\n </div>\n </div>\n <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n <div\n class=\"{{\n mode === 'main'\n ? 'str-chat__input-flat--textarea-wrapper'\n : 'str-chat__small-message-input--textarea-wrapper'\n }}\"\n >\n <stream-attachment-preview-list\n class=\"rfu-image-previewer-angular-host\"\n ></stream-attachment-preview-list>\n <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n <ng-container *ngIf=\"emojiPickerTemplate\">\n <div\n class=\"\n str-chat__input-flat-emojiselect\n str-chat-angular__emojiselect\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n emojiPickerTemplate;\n context: { emojiInput$: emojiInputService.emojiInput$ }\n \"\n ></ng-container>\n </div>\n </ng-container>\n <ng-template\n *ngIf=\"canSendMessages; else notAllowed\"\n streamTextarea\n [(value)]=\"textareaValue\"\n (valueChange)=\"typingStart$.next()\"\n (send)=\"messageSent()\"\n [componentRef]=\"textareaRef\"\n (userMentions)=\"mentionedUsers = $event\"\n [areMentionsEnabled]=\"areMentionsEnabled\"\n [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n [mentionScope]=\"mentionScope\"\n ></ng-template>\n <ng-template #notAllowed>\n <textarea\n disabled\n rows=\"1\"\n [value]=\"\n (mode === 'thread'\n ? 'You can\\'t send thread replies in this channel'\n : 'streamChat.You can\\'t send messages in this channel'\n ) | translate\n \"\n class=\"rta__textarea str-chat__textarea__textarea\"\n ></textarea>\n </ng-template>\n </div>\n <div\n *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized && canSendMessages\"\n class=\"str-chat__fileupload-wrapper\"\n data-testid=\"file-upload-button\"\n >\n <div class=\"str-chat__tooltip\">\n {{ \"streamChat.Attach files\" | translate }}\n </div>\n <div class=\"rfu-file-upload-button\">\n <label>\n <input\n #fileInput\n type=\"file\"\n class=\"rfu-file-input\"\n data-testid=\"file-input\"\n [accept]=\"accept\"\n [multiple]=\"isMultipleFileUploadEnabled\"\n (change)=\"filesSelected(fileInput.files)\"\n />\n <span class=\"str-chat__input-flat-fileupload\">\n <stream-icon icon=\"file-upload\"></stream-icon>\n </span>\n </label>\n </div>\n </div>\n </div>\n <button\n *ngIf=\"canSendMessages\"\n data-testid=\"send-button\"\n class=\"str-chat__send-button\"\n (click)=\"messageSent()\"\n (keyup.enter)=\"messageSent()\"\n >\n <stream-icon icon=\"send\"></stream-icon>\n </button>\n </div>\n</div>\n", components: [{ type: i7.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i8.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: i9.AttachmentListComponent, selector: "stream-attachment-list", inputs: ["messageId", "attachments"] }, { type: i10.AttachmentPreviewListComponent, selector: "stream-attachment-preview-list" }], directives: [{ type: i11.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i11.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i12.TextareaDirective, selector: "[streamTextarea]", inputs: ["componentRef", "areMentionsEnabled", "mentionAutocompleteItemTemplate", "mentionScope", "commandAutocompleteItemTemplate", "value"], outputs: ["valueChange", "send", "userMentions"] }], pipes: { "async": i11.AsyncPipe, "translate": i13.TranslatePipe } });
350
+ MessageInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageInputComponent, selector: "stream-message-input", inputs: { isFileUploadEnabled: "isFileUploadEnabled", areMentionsEnabled: "areMentionsEnabled", mentionScope: "mentionScope", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", commandAutocompleteItemTemplate: "commandAutocompleteItemTemplate", emojiPickerTemplate: "emojiPickerTemplate", mode: "mode", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService, EmojiInputService], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "textareaAnchor", first: true, predicate: TextareaDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"{{\n mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n }}\"\n [class.str-chat__input-flat-has-attachments]=\"\n (attachmentUploads$ | async)!.length > 0\n \"\n [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n <div\n data-testid=\"quoted-message-container\"\n class=\"quoted-message-preview\"\n *ngIf=\"quotedMessage\"\n >\n <div class=\"quoted-message-preview-header\">\n <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n <button\n class=\"str-chat__square-button\"\n data-testid=\"remove-quote\"\n (click)=\"deselectMessageToQuote()\"\n (keyup.enter)=\"deselectMessageToQuote()\"\n >\n <stream-icon\n icon=\"close-no-outline\"\n style=\"font-size: 10px; line-height: 10px\"\n ></stream-icon>\n </button>\n </div>\n <div class=\"quoted-message-preview-content\">\n <stream-avatar\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host\"\n [imageUrl]=\"quotedMessage?.user?.image\"\n [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n [size]=\"20\"\n ></stream-avatar>\n <div class=\"quoted-message-preview-content-inner\">\n <stream-attachment-list\n *ngIf=\"\n quotedMessage?.attachments && quotedMessage?.attachments?.length\n \"\n [attachments]=\"quotedMessageAttachments\"\n [messageId]=\"quotedMessage?.id\"\n ></stream-attachment-list>\n <div\n data-testid=\"quoted-message-text\"\n [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n ></div>\n </div>\n </div>\n </div>\n <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n <div\n class=\"{{\n mode === 'main'\n ? 'str-chat__input-flat--textarea-wrapper'\n : 'str-chat__small-message-input--textarea-wrapper'\n }}\"\n >\n <stream-attachment-preview-list\n class=\"rfu-image-previewer-angular-host\"\n ></stream-attachment-preview-list>\n <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n <ng-container\n *ngIf=\"emojiPickerTemplate && !isCooldownInProgress\"\n data-testid=\"emoji-picker\"\n >\n <div\n class=\"\n str-chat__input-flat-emojiselect\n str-chat-angular__emojiselect\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n emojiPickerTemplate;\n context: { emojiInput$: emojiInputService.emojiInput$ }\n \"\n ></ng-container>\n </div>\n </ng-container>\n <div\n class=\"str-chat__input-flat-cooldown str-chat-angular__cooldown\"\n *ngIf=\"isCooldownInProgress\"\n data-testid=\"cooldown-timer\"\n >\n {{ cooldown$ | async }}\n </div>\n <ng-template\n *ngIf=\"canSendMessages && !isCooldownInProgress; else notAllowed\"\n streamTextarea\n [(value)]=\"textareaValue\"\n (valueChange)=\"typingStart$.next()\"\n (send)=\"messageSent()\"\n [componentRef]=\"textareaRef\"\n (userMentions)=\"mentionedUsers = $event\"\n [areMentionsEnabled]=\"areMentionsEnabled\"\n [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n [mentionScope]=\"mentionScope\"\n ></ng-template>\n <ng-template #notAllowed>\n <textarea\n disabled\n rows=\"1\"\n [value]=\"disabledTextareaText | translate\"\n class=\"rta__textarea str-chat__textarea__textarea\"\n data-testid=\"disabled-textarea\"\n ></textarea>\n </ng-template>\n </div>\n <div\n *ngIf=\"\n isFileUploadEnabled &&\n isFileUploadAuthorized &&\n canSendMessages &&\n !isCooldownInProgress\n \"\n class=\"str-chat__fileupload-wrapper\"\n data-testid=\"file-upload-button\"\n >\n <div class=\"str-chat__tooltip\">\n {{ \"streamChat.Attach files\" | translate }}\n </div>\n <div class=\"rfu-file-upload-button\">\n <label>\n <input\n #fileInput\n type=\"file\"\n class=\"rfu-file-input\"\n data-testid=\"file-input\"\n [accept]=\"accept\"\n [multiple]=\"isMultipleFileUploadEnabled\"\n (change)=\"filesSelected(fileInput.files)\"\n />\n <span class=\"str-chat__input-flat-fileupload\">\n <stream-icon icon=\"file-upload\"></stream-icon>\n </span>\n </label>\n </div>\n </div>\n </div>\n <button\n *ngIf=\"canSendMessages\"\n data-testid=\"send-button\"\n class=\"str-chat__send-button\"\n (click)=\"messageSent()\"\n (keyup.enter)=\"messageSent()\"\n >\n <stream-icon icon=\"send\"></stream-icon>\n </button>\n </div>\n</div>\n", components: [{ type: i7.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i8.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: i9.AttachmentListComponent, selector: "stream-attachment-list", inputs: ["messageId", "attachments"] }, { type: i10.AttachmentPreviewListComponent, selector: "stream-attachment-preview-list" }], directives: [{ type: i11.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i11.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i12.TextareaDirective, selector: "[streamTextarea]", inputs: ["componentRef", "areMentionsEnabled", "mentionAutocompleteItemTemplate", "mentionScope", "commandAutocompleteItemTemplate", "value"], outputs: ["valueChange", "send", "userMentions"] }], pipes: { "async": i11.AsyncPipe, "translate": i13.TranslatePipe } });
303
351
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, decorators: [{
304
352
  type: Component,
305
353
  args: [{
@@ -340,4 +388,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
340
388
  type: ViewChild,
341
389
  args: [TextareaDirective, { static: false }]
342
390
  }] } });
343
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-input.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-input/message-input.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-input/message-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ComponentFactoryResolver,\n  ComponentRef,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  ViewChild,\n} from '@angular/core';\nimport { ChatClientService } from '../chat-client.service';\nimport { Observable, Subject, Subscription } from 'rxjs';\nimport { first } from 'rxjs/operators';\nimport { AppSettings, Channel, UserResponse } from 'stream-chat';\nimport { AttachmentService } from '../attachment.service';\nimport { ChannelService } from '../channel.service';\nimport { textareaInjectionToken } from '../injection-tokens';\nimport { NotificationService } from '../notification.service';\nimport {\n  AttachmentUpload,\n  CommandAutocompleteListItemContext,\n  MentionAutcompleteListItemContext,\n  StreamMessage,\n} from '../types';\nimport { MessageInputConfigService } from './message-input-config.service';\nimport { TextareaDirective } from './textarea.directive';\nimport { TextareaInterface } from './textarea.interface';\nimport { isImageFile } from '../is-image-file';\nimport { EmojiInputService } from './emoji-input.service';\n\n/**\n * The `MessageInput` component displays an input where users can type their messages and upload files, and sends the message to the active channel. The component can be used to compose new messages or update existing ones. To send messages, the chat user needs to have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n */\n@Component({\n  selector: 'stream-message-input',\n  templateUrl: './message-input.component.html',\n  styles: [],\n  providers: [AttachmentService, EmojiInputService],\n})\nexport class MessageInputComponent\n  implements OnChanges, OnDestroy, AfterViewInit\n{\n  /**\n   * If file upload is enabled, the user can open a file selector from the input. Please note that the user also needs to have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript). If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() isFileUploadEnabled: boolean | undefined;\n  /**\n   * If true, users can mention other users in messages. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() areMentionsEnabled: boolean | undefined;\n  /**\n   * The scope for user mentions, either members of the current channel of members of the application. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() mentionScope: 'channel' | 'application' | undefined;\n  /**\n   * You can provide your own template for the autocomplete list for user mentions. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() mentionAutocompleteItemTemplate:\n    | TemplateRef<MentionAutcompleteListItemContext>\n    | undefined;\n  /**\n   * You can provide your own template for the autocomplete list for commands. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() commandAutocompleteItemTemplate:\n    | TemplateRef<CommandAutocompleteListItemContext>\n    | undefined;\n  /**\n   * You can add an emoji picker by [providing your own emoji picker template](../code-examples/emoji-picker.mdx)\n   */\n  @Input() emojiPickerTemplate: TemplateRef<void> | undefined;\n  /**\n   * Determines if the message is being dispalyed in a channel or in a [thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'thread' | 'main' = 'main';\n  /**\n   * You can narrow the accepted file types by providing the [accepted types](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept). By default every file type is accepted.\n   * If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   * @deprecated use [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript#file-uploads) instead\n   */\n  @Input() acceptedFileTypes: string[] | undefined;\n  /**\n   * If true, users can select multiple files to upload. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() isMultipleFileUploadEnabled: boolean | undefined;\n  /**\n   * The message to edit\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * Emits when a message was successfuly sent or updated\n   */\n  @Output() readonly messageUpdate = new EventEmitter<void>();\n  isFileUploadAuthorized: boolean | undefined;\n  canSendLinks: boolean | undefined;\n  canSendMessages: boolean | undefined;\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  textareaValue = '';\n  textareaRef: ComponentRef<TextareaInterface> | undefined;\n  mentionedUsers: UserResponse[] = [];\n  quotedMessage: undefined | StreamMessage;\n  typingStart$ = new Subject<void>();\n  @ViewChild('fileInput') private fileInput!: ElementRef<HTMLInputElement>;\n  @ViewChild(TextareaDirective, { static: false })\n  private textareaAnchor!: TextareaDirective;\n  private subscriptions: Subscription[] = [];\n  private hideNotification: Function | undefined;\n  private isViewInited = false;\n  private appSettings: AppSettings | undefined;\n  private channel: Channel | undefined;\n  constructor(\n    private channelService: ChannelService,\n    private notificationService: NotificationService,\n    private attachmentService: AttachmentService,\n    private configService: MessageInputConfigService,\n    @Inject(textareaInjectionToken)\n    private textareaType: Type<TextareaInterface>,\n    private componentFactoryResolver: ComponentFactoryResolver,\n    private cdRef: ChangeDetectorRef,\n    private chatClient: ChatClientService,\n    public emojiInputService: EmojiInputService\n  ) {\n    this.subscriptions.push(\n      this.attachmentService.attachmentUploadInProgressCounter$.subscribe(\n        (counter) => {\n          if (counter === 0 && this.hideNotification) {\n            this.hideNotification();\n            this.hideNotification = undefined;\n          }\n        }\n      )\n    );\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.textareaValue = '';\n        this.attachmentService.resetAttachmentUploads();\n        const capabilities = channel?.data?.own_capabilities as string[];\n        if (capabilities) {\n          this.isFileUploadAuthorized =\n            capabilities.indexOf('upload-file') !== -1;\n          this.canSendLinks = capabilities.indexOf('send-links') !== -1;\n          this.channel = channel;\n          this.setCanSendMessages();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.chatClient.appSettings$.subscribe(\n        (appSettings) => (this.appSettings = appSettings)\n      )\n    );\n    this.subscriptions.push(\n      this.channelService.messageToQuote$.subscribe((m) => {\n        const isThreadReply = m && m.parent_id;\n        if (\n          (this.mode === 'thread' && isThreadReply) ||\n          (this.mode === 'thread' && this.quotedMessage && !m) ||\n          (this.mode === 'main' && !isThreadReply)\n        ) {\n          this.quotedMessage = m;\n        }\n      })\n    );\n    this.attachmentUploads$ = this.attachmentService.attachmentUploads$;\n    this.isFileUploadEnabled = this.configService.isFileUploadEnabled;\n    this.acceptedFileTypes = this.configService.acceptedFileTypes;\n    this.isMultipleFileUploadEnabled =\n      this.configService.isMultipleFileUploadEnabled;\n    this.areMentionsEnabled = this.configService.areMentionsEnabled;\n    this.mentionAutocompleteItemTemplate =\n      this.configService.mentionAutocompleteItemTemplate;\n    this.mentionScope = this.configService.mentionScope;\n    this.commandAutocompleteItemTemplate =\n      this.configService.commandAutocompleteItemTemplate;\n    this.emojiPickerTemplate = this.configService.emojiPickerTemplate;\n\n    this.subscriptions.push(\n      this.typingStart$.subscribe(\n        () => void this.channelService.typingStarted(this.parentMessageId)\n      )\n    );\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n    this.initTextarea();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.attachmentService.resetAttachmentUploads();\n      if (this.isUpdate) {\n        this.attachmentService.createFromAttachments(\n          this.message!.attachments || []\n        );\n        this.textareaValue = this.message!.text || '';\n      }\n    }\n    if (changes.isFileUploadEnabled) {\n      this.configService.isFileUploadEnabled = this.isFileUploadEnabled;\n    }\n    if (changes.acceptedFileTypes) {\n      this.configService.acceptedFileTypes = this.acceptedFileTypes;\n    }\n    if (changes.isMultipleFileUploadEnabled) {\n      this.configService.isMultipleFileUploadEnabled =\n        this.isMultipleFileUploadEnabled;\n    }\n    if (changes.areMentionsEnabled) {\n      this.configService.areMentionsEnabled = this.areMentionsEnabled;\n    }\n    if (changes.mentionAutocompleteItemTemplate) {\n      this.configService.mentionAutocompleteItemTemplate =\n        this.mentionAutocompleteItemTemplate;\n    }\n    if (changes.commandAutocompleteItemTemplate) {\n      this.configService.commandAutocompleteItemTemplate =\n        this.commandAutocompleteItemTemplate;\n    }\n    if (changes.mentionScope) {\n      this.configService.mentionScope = this.mentionScope;\n    }\n    if (changes.emojiPickerTemplate) {\n      this.configService.emojiPickerTemplate = this.emojiPickerTemplate;\n    }\n    if (changes.mode) {\n      this.setCanSendMessages();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async messageSent() {\n    let attachmentUploadInProgressCounter!: number;\n    this.attachmentService.attachmentUploadInProgressCounter$\n      .pipe(first())\n      .subscribe((counter) => (attachmentUploadInProgressCounter = counter));\n    if (attachmentUploadInProgressCounter > 0) {\n      if (!this.hideNotification) {\n        this.hideNotification =\n          this.notificationService.addPermanentNotification(\n            'streamChat.Wait until all attachments have uploaded'\n          );\n      }\n      return;\n    }\n    const attachments = this.attachmentService.mapToAttachments();\n    const text = this.textareaValue;\n    if (!text && (!attachments || attachments.length === 0)) {\n      return;\n    }\n    if (this.containsLinks && !this.canSendLinks) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Sending links is not allowed in this conversation'\n      );\n      return;\n    }\n    if (!this.isUpdate) {\n      this.textareaValue = '';\n    }\n    try {\n      await (this.isUpdate\n        ? this.channelService.updateMessage({\n            ...this.message!,\n            text: text,\n            attachments: attachments,\n          })\n        : this.channelService.sendMessage(\n            text,\n            attachments,\n            this.mentionedUsers,\n            this.parentMessageId,\n            this.quotedMessage?.id\n          ));\n      this.messageUpdate.emit();\n      if (!this.isUpdate) {\n        this.attachmentService.resetAttachmentUploads();\n      }\n    } catch (error) {\n      if (this.isUpdate) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Edit message request failed'\n        );\n      }\n    }\n    void this.channelService.typingStopped(this.parentMessageId);\n    if (this.quotedMessage) {\n      this.deselectMessageToQuote();\n    }\n  }\n\n  get containsLinks() {\n    return /(?:(?:https?|ftp):\\/\\/)?[\\w/\\-?=%.]+\\.[\\w/\\-&?=%.]+/.test(\n      this.textareaValue\n    );\n  }\n\n  get accept() {\n    return this.acceptedFileTypes ? this.acceptedFileTypes?.join(',') : '';\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.quotedMessage?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  async filesSelected(fileList: FileList | null) {\n    if (!(await this.areAttachemntsValid(fileList))) {\n      return;\n    }\n    await this.attachmentService.filesSelected(fileList);\n    this.clearFileInput();\n  }\n\n  deselectMessageToQuote() {\n    this.channelService.selectMessageToQuote(undefined);\n  }\n\n  private clearFileInput() {\n    this.fileInput.nativeElement.value = '';\n  }\n\n  private get isUpdate() {\n    return !!this.message;\n  }\n\n  private initTextarea() {\n    if (!this.canSendMessages || this.textareaRef || !this.textareaAnchor) {\n      return;\n    }\n    const componentFactory =\n      this.componentFactoryResolver.resolveComponentFactory(this.textareaType);\n    this.textareaRef =\n      this.textareaAnchor.viewContainerRef.createComponent<any>(\n        componentFactory\n      );\n    this.cdRef.detectChanges();\n  }\n\n  private async areAttachemntsValid(fileList: FileList | null) {\n    if (!fileList || this.acceptedFileTypes) {\n      return true;\n    }\n    if (!this.appSettings) {\n      await this.chatClient.getAppSettings();\n    }\n    let isValid = true;\n    Array.from(fileList).forEach((f) => {\n      let hasBlockedExtension: boolean;\n      let hasBlockedMimeType: boolean;\n      let hasNotAllowedExtension: boolean;\n      let hasNotAllowedMimeType: boolean;\n      if (isImageFile(f)) {\n        hasBlockedExtension =\n          !!this.appSettings?.image_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.image_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.image_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.image_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.image_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.image_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      } else {\n        hasBlockedExtension =\n          !!this.appSettings?.file_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.file_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.file_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.file_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.file_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.file_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      }\n      if (\n        hasBlockedExtension ||\n        hasBlockedMimeType ||\n        hasNotAllowedExtension ||\n        hasNotAllowedMimeType\n      ) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Unsupported file type: {{type}}',\n          undefined,\n          undefined,\n          { type: f.type }\n        );\n        isValid = false;\n      }\n    });\n    return isValid;\n  }\n\n  private setCanSendMessages() {\n    const capabilities = this.channel?.data?.own_capabilities as string[];\n    if (!capabilities) {\n      this.canSendMessages = false;\n    } else {\n      this.canSendMessages =\n        capabilities.indexOf(\n          this.mode === 'main' ? 'send-message' : 'send-reply'\n        ) !== -1;\n    }\n    if (this.isViewInited) {\n      this.cdRef.detectChanges();\n      this.initTextarea();\n    }\n  }\n\n  private get parentMessageId() {\n    let parentMessageId: string | undefined = undefined;\n    if (this.mode === 'thread') {\n      this.channelService.activeParentMessageId$\n        .pipe(first())\n        .subscribe((id) => (parentMessageId = id));\n    }\n\n    return parentMessageId;\n  }\n}\n","<div\n  class=\"{{\n    mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n  }}\"\n  [class.str-chat__input-flat-has-attachments]=\"\n    (attachmentUploads$ | async)!.length > 0\n  \"\n  [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n  <div\n    data-testid=\"quoted-message-container\"\n    class=\"quoted-message-preview\"\n    *ngIf=\"quotedMessage\"\n  >\n    <div class=\"quoted-message-preview-header\">\n      <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n      <button\n        class=\"str-chat__square-button\"\n        data-testid=\"remove-quote\"\n        (click)=\"deselectMessageToQuote()\"\n        (keyup.enter)=\"deselectMessageToQuote()\"\n      >\n        <stream-icon\n          icon=\"close-no-outline\"\n          style=\"font-size: 10px; line-height: 10px\"\n        ></stream-icon>\n      </button>\n    </div>\n    <div class=\"quoted-message-preview-content\">\n      <stream-avatar\n        data-testid=\"qouted-message-avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"quotedMessage?.user?.image\"\n        [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n        [size]=\"20\"\n      ></stream-avatar>\n      <div class=\"quoted-message-preview-content-inner\">\n        <stream-attachment-list\n          *ngIf=\"\n            quotedMessage?.attachments && quotedMessage?.attachments?.length\n          \"\n          [attachments]=\"quotedMessageAttachments\"\n          [messageId]=\"quotedMessage?.id\"\n        ></stream-attachment-list>\n        <div\n          data-testid=\"quoted-message-text\"\n          [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n        ></div>\n      </div>\n    </div>\n  </div>\n  <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n    <div\n      class=\"{{\n        mode === 'main'\n          ? 'str-chat__input-flat--textarea-wrapper'\n          : 'str-chat__small-message-input--textarea-wrapper'\n      }}\"\n    >\n      <stream-attachment-preview-list\n        class=\"rfu-image-previewer-angular-host\"\n      ></stream-attachment-preview-list>\n      <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n        <ng-container *ngIf=\"emojiPickerTemplate\">\n          <div\n            class=\"\n              str-chat__input-flat-emojiselect\n              str-chat-angular__emojiselect\n            \"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"\n                emojiPickerTemplate;\n                context: { emojiInput$: emojiInputService.emojiInput$ }\n              \"\n            ></ng-container>\n          </div>\n        </ng-container>\n        <ng-template\n          *ngIf=\"canSendMessages; else notAllowed\"\n          streamTextarea\n          [(value)]=\"textareaValue\"\n          (valueChange)=\"typingStart$.next()\"\n          (send)=\"messageSent()\"\n          [componentRef]=\"textareaRef\"\n          (userMentions)=\"mentionedUsers = $event\"\n          [areMentionsEnabled]=\"areMentionsEnabled\"\n          [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n          [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n          [mentionScope]=\"mentionScope\"\n        ></ng-template>\n        <ng-template #notAllowed>\n          <textarea\n            disabled\n            rows=\"1\"\n            [value]=\"\n              (mode === 'thread'\n                ? 'You can\\'t send thread replies in this channel'\n                : 'streamChat.You can\\'t send messages in this channel'\n              ) | translate\n            \"\n            class=\"rta__textarea str-chat__textarea__textarea\"\n          ></textarea>\n        </ng-template>\n      </div>\n      <div\n        *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized && canSendMessages\"\n        class=\"str-chat__fileupload-wrapper\"\n        data-testid=\"file-upload-button\"\n      >\n        <div class=\"str-chat__tooltip\">\n          {{ \"streamChat.Attach files\" | translate }}\n        </div>\n        <div class=\"rfu-file-upload-button\">\n          <label>\n            <input\n              #fileInput\n              type=\"file\"\n              class=\"rfu-file-input\"\n              data-testid=\"file-input\"\n              [accept]=\"accept\"\n              [multiple]=\"isMultipleFileUploadEnabled\"\n              (change)=\"filesSelected(fileInput.files)\"\n            />\n            <span class=\"str-chat__input-flat-fileupload\">\n              <stream-icon icon=\"file-upload\"></stream-icon>\n            </span>\n          </label>\n        </div>\n      </div>\n    </div>\n    <button\n      *ngIf=\"canSendMessages\"\n      data-testid=\"send-button\"\n      class=\"str-chat__send-button\"\n      (click)=\"messageSent()\"\n      (keyup.enter)=\"messageSent()\"\n    >\n      <stream-icon icon=\"send\"></stream-icon>\n    </button>\n  </div>\n</div>\n"]}
391
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-input.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-input/message-input.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-input/message-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ComponentFactoryResolver,\n  ComponentRef,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  ViewChild,\n} from '@angular/core';\nimport { ChatClientService } from '../chat-client.service';\nimport { combineLatest, Observable, Subject, Subscription, timer } from 'rxjs';\nimport { first, map, take, tap } from 'rxjs/operators';\nimport { AppSettings, Channel, UserResponse } from 'stream-chat';\nimport { AttachmentService } from '../attachment.service';\nimport { ChannelService } from '../channel.service';\nimport { textareaInjectionToken } from '../injection-tokens';\nimport { NotificationService } from '../notification.service';\nimport {\n  AttachmentUpload,\n  CommandAutocompleteListItemContext,\n  MentionAutcompleteListItemContext,\n  StreamMessage,\n} from '../types';\nimport { MessageInputConfigService } from './message-input-config.service';\nimport { TextareaDirective } from './textarea.directive';\nimport { TextareaInterface } from './textarea.interface';\nimport { isImageFile } from '../is-image-file';\nimport { EmojiInputService } from './emoji-input.service';\n\n/**\n * The `MessageInput` component displays an input where users can type their messages and upload files, and sends the message to the active channel. The component can be used to compose new messages or update existing ones. To send messages, the chat user needs to have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript).\n */\n@Component({\n  selector: 'stream-message-input',\n  templateUrl: './message-input.component.html',\n  styles: [],\n  providers: [AttachmentService, EmojiInputService],\n})\nexport class MessageInputComponent\n  implements OnChanges, OnDestroy, AfterViewInit\n{\n  /**\n   * If file upload is enabled, the user can open a file selector from the input. Please note that the user also needs to have the necessary [channel capability](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript). If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() isFileUploadEnabled: boolean | undefined;\n  /**\n   * If true, users can mention other users in messages. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() areMentionsEnabled: boolean | undefined;\n  /**\n   * The scope for user mentions, either members of the current channel of members of the application. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() mentionScope: 'channel' | 'application' | undefined;\n  /**\n   * You can provide your own template for the autocomplete list for user mentions. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() mentionAutocompleteItemTemplate:\n    | TemplateRef<MentionAutcompleteListItemContext>\n    | undefined;\n  /**\n   * You can provide your own template for the autocomplete list for commands. You also [need to use the `AutocompleteTextarea`](../concepts/opt-in-architecture.mdx) for this feature to work. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() commandAutocompleteItemTemplate:\n    | TemplateRef<CommandAutocompleteListItemContext>\n    | undefined;\n  /**\n   * You can add an emoji picker by [providing your own emoji picker template](../code-examples/emoji-picker.mdx)\n   */\n  @Input() emojiPickerTemplate: TemplateRef<void> | undefined;\n  /**\n   * Determines if the message is being dispalyed in a channel or in a [thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript).\n   */\n  @Input() mode: 'thread' | 'main' = 'main';\n  /**\n   * You can narrow the accepted file types by providing the [accepted types](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept). By default every file type is accepted.\n   * If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   * @deprecated use [application settings](https://getstream.io/chat/docs/javascript/app_setting_overview/?language=javascript#file-uploads) instead\n   */\n  @Input() acceptedFileTypes: string[] | undefined;\n  /**\n   * If true, users can select multiple files to upload. If no value is provided, it is set from the [`MessageInputConfigService`](../services/MessageInputConfigService.mdx).\n   */\n  @Input() isMultipleFileUploadEnabled: boolean | undefined;\n  /**\n   * The message to edit\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * Emits when a message was successfuly sent or updated\n   */\n  @Output() readonly messageUpdate = new EventEmitter<void>();\n  isFileUploadAuthorized: boolean | undefined;\n  canSendLinks: boolean | undefined;\n  canSendMessages: boolean | undefined;\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  textareaValue = '';\n  textareaRef: ComponentRef<TextareaInterface> | undefined;\n  mentionedUsers: UserResponse[] = [];\n  quotedMessage: undefined | StreamMessage;\n  typingStart$ = new Subject<void>();\n  cooldown$: Observable<number> | undefined;\n  isCooldownInProgress = false;\n  @ViewChild('fileInput') private fileInput!: ElementRef<HTMLInputElement>;\n  @ViewChild(TextareaDirective, { static: false })\n  private textareaAnchor!: TextareaDirective;\n  private subscriptions: Subscription[] = [];\n  private hideNotification: Function | undefined;\n  private isViewInited = false;\n  private appSettings: AppSettings | undefined;\n  private channel: Channel | undefined;\n  constructor(\n    private channelService: ChannelService,\n    private notificationService: NotificationService,\n    private attachmentService: AttachmentService,\n    private configService: MessageInputConfigService,\n    @Inject(textareaInjectionToken)\n    private textareaType: Type<TextareaInterface>,\n    private componentFactoryResolver: ComponentFactoryResolver,\n    private cdRef: ChangeDetectorRef,\n    private chatClient: ChatClientService,\n    public emojiInputService: EmojiInputService\n  ) {\n    this.subscriptions.push(\n      this.attachmentService.attachmentUploadInProgressCounter$.subscribe(\n        (counter) => {\n          if (counter === 0 && this.hideNotification) {\n            this.hideNotification();\n            this.hideNotification = undefined;\n          }\n        }\n      )\n    );\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        this.textareaValue = '';\n        this.attachmentService.resetAttachmentUploads();\n        const capabilities = channel?.data?.own_capabilities as string[];\n        if (capabilities) {\n          this.isFileUploadAuthorized =\n            capabilities.indexOf('upload-file') !== -1;\n          this.canSendLinks = capabilities.indexOf('send-links') !== -1;\n          this.channel = channel;\n          this.setCanSendMessages();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.chatClient.appSettings$.subscribe(\n        (appSettings) => (this.appSettings = appSettings)\n      )\n    );\n    this.subscriptions.push(\n      this.channelService.messageToQuote$.subscribe((m) => {\n        const isThreadReply = m && m.parent_id;\n        if (\n          (this.mode === 'thread' && isThreadReply) ||\n          (this.mode === 'thread' && this.quotedMessage && !m) ||\n          (this.mode === 'main' && !isThreadReply)\n        ) {\n          this.quotedMessage = m;\n        }\n      })\n    );\n    this.attachmentUploads$ = this.attachmentService.attachmentUploads$;\n    this.isFileUploadEnabled = this.configService.isFileUploadEnabled;\n    this.acceptedFileTypes = this.configService.acceptedFileTypes;\n    this.isMultipleFileUploadEnabled =\n      this.configService.isMultipleFileUploadEnabled;\n    this.areMentionsEnabled = this.configService.areMentionsEnabled;\n    this.mentionAutocompleteItemTemplate =\n      this.configService.mentionAutocompleteItemTemplate;\n    this.mentionScope = this.configService.mentionScope;\n    this.commandAutocompleteItemTemplate =\n      this.configService.commandAutocompleteItemTemplate;\n    this.emojiPickerTemplate = this.configService.emojiPickerTemplate;\n\n    this.subscriptions.push(\n      this.typingStart$.subscribe(\n        () => void this.channelService.typingStarted(this.parentMessageId)\n      )\n    );\n\n    this.subscriptions.push(\n      combineLatest([\n        this.channelService.latestMessageDateByUserByChannels$,\n        this.channelService.activeChannel$,\n      ])\n        .pipe(\n          map(\n            ([latestMessages, channel]): [\n              Date | undefined,\n              Channel | undefined\n            ] => [latestMessages[channel?.cid || ''], channel!]\n          )\n        )\n        .subscribe(([latestMessageDate, channel]) => {\n          const cooldown =\n            (channel?.data?.cooldown as number) &&\n            latestMessageDate &&\n            Math.round(\n              (channel?.data?.cooldown as number) -\n                (new Date().getTime() - latestMessageDate.getTime()) / 1000\n            );\n          if (\n            cooldown &&\n            cooldown > 0 &&\n            (channel?.data?.own_capabilities as string[]).includes('slow-mode')\n          ) {\n            this.startCooldown(cooldown);\n          } else if (this.isCooldownInProgress) {\n            this.stopCooldown();\n          }\n        })\n    );\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n    this.initTextarea();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.attachmentService.resetAttachmentUploads();\n      if (this.isUpdate) {\n        this.attachmentService.createFromAttachments(\n          this.message!.attachments || []\n        );\n        this.textareaValue = this.message!.text || '';\n      }\n    }\n    if (changes.isFileUploadEnabled) {\n      this.configService.isFileUploadEnabled = this.isFileUploadEnabled;\n    }\n    if (changes.acceptedFileTypes) {\n      this.configService.acceptedFileTypes = this.acceptedFileTypes;\n    }\n    if (changes.isMultipleFileUploadEnabled) {\n      this.configService.isMultipleFileUploadEnabled =\n        this.isMultipleFileUploadEnabled;\n    }\n    if (changes.areMentionsEnabled) {\n      this.configService.areMentionsEnabled = this.areMentionsEnabled;\n    }\n    if (changes.mentionAutocompleteItemTemplate) {\n      this.configService.mentionAutocompleteItemTemplate =\n        this.mentionAutocompleteItemTemplate;\n    }\n    if (changes.commandAutocompleteItemTemplate) {\n      this.configService.commandAutocompleteItemTemplate =\n        this.commandAutocompleteItemTemplate;\n    }\n    if (changes.mentionScope) {\n      this.configService.mentionScope = this.mentionScope;\n    }\n    if (changes.emojiPickerTemplate) {\n      this.configService.emojiPickerTemplate = this.emojiPickerTemplate;\n    }\n    if (changes.mode) {\n      this.setCanSendMessages();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async messageSent() {\n    let attachmentUploadInProgressCounter!: number;\n    this.attachmentService.attachmentUploadInProgressCounter$\n      .pipe(first())\n      .subscribe((counter) => (attachmentUploadInProgressCounter = counter));\n    if (attachmentUploadInProgressCounter > 0) {\n      if (!this.hideNotification) {\n        this.hideNotification =\n          this.notificationService.addPermanentNotification(\n            'streamChat.Wait until all attachments have uploaded'\n          );\n      }\n      return;\n    }\n    const attachments = this.attachmentService.mapToAttachments();\n    const text = this.textareaValue;\n    if (!text && (!attachments || attachments.length === 0)) {\n      return;\n    }\n    if (this.containsLinks && !this.canSendLinks) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Sending links is not allowed in this conversation'\n      );\n      return;\n    }\n    if (!this.isUpdate) {\n      this.textareaValue = '';\n    }\n    try {\n      await (this.isUpdate\n        ? this.channelService.updateMessage({\n            ...this.message!,\n            text: text,\n            attachments: attachments,\n          })\n        : this.channelService.sendMessage(\n            text,\n            attachments,\n            this.mentionedUsers,\n            this.parentMessageId,\n            this.quotedMessage?.id\n          ));\n      this.messageUpdate.emit();\n      if (!this.isUpdate) {\n        this.attachmentService.resetAttachmentUploads();\n      }\n    } catch (error) {\n      if (this.isUpdate) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Edit message request failed'\n        );\n      }\n    }\n    void this.channelService.typingStopped(this.parentMessageId);\n    if (this.quotedMessage) {\n      this.deselectMessageToQuote();\n    }\n  }\n\n  get containsLinks() {\n    return /(?:(?:https?|ftp):\\/\\/)?[\\w/\\-?=%.]+\\.[\\w/\\-&?=%.]+/.test(\n      this.textareaValue\n    );\n  }\n\n  get accept() {\n    return this.acceptedFileTypes ? this.acceptedFileTypes?.join(',') : '';\n  }\n\n  get quotedMessageAttachments() {\n    const originalAttachments = this.quotedMessage?.attachments;\n    return originalAttachments && originalAttachments.length\n      ? [originalAttachments[0]]\n      : [];\n  }\n\n  get disabledTextareaText() {\n    if (!this.canSendMessages) {\n      return this.mode === 'thread'\n        ? \"streamChat.You can't send thread replies in this channel\"\n        : \"streamChat.You can't send messages in this channel\";\n    } else if (this.cooldown$) {\n      return 'streamChat.Slow Mode ON';\n    }\n    return '';\n  }\n\n  async filesSelected(fileList: FileList | null) {\n    if (!(await this.areAttachemntsValid(fileList))) {\n      return;\n    }\n    await this.attachmentService.filesSelected(fileList);\n    this.clearFileInput();\n  }\n\n  deselectMessageToQuote() {\n    this.channelService.selectMessageToQuote(undefined);\n  }\n\n  private clearFileInput() {\n    this.fileInput.nativeElement.value = '';\n  }\n\n  private get isUpdate() {\n    return !!this.message;\n  }\n\n  private initTextarea() {\n    if (!this.canSendMessages || this.textareaRef || !this.textareaAnchor) {\n      return;\n    }\n    const componentFactory =\n      this.componentFactoryResolver.resolveComponentFactory(this.textareaType);\n    this.textareaRef =\n      this.textareaAnchor.viewContainerRef.createComponent<any>(\n        componentFactory\n      );\n    this.cdRef.detectChanges();\n  }\n\n  private async areAttachemntsValid(fileList: FileList | null) {\n    if (!fileList || this.acceptedFileTypes) {\n      return true;\n    }\n    if (!this.appSettings) {\n      await this.chatClient.getAppSettings();\n    }\n    let isValid = true;\n    Array.from(fileList).forEach((f) => {\n      let hasBlockedExtension: boolean;\n      let hasBlockedMimeType: boolean;\n      let hasNotAllowedExtension: boolean;\n      let hasNotAllowedMimeType: boolean;\n      if (isImageFile(f)) {\n        hasBlockedExtension =\n          !!this.appSettings?.image_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.image_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.image_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.image_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.image_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.image_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      } else {\n        hasBlockedExtension =\n          !!this.appSettings?.file_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.file_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.file_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.file_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.file_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.file_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      }\n      if (\n        hasBlockedExtension ||\n        hasBlockedMimeType ||\n        hasNotAllowedExtension ||\n        hasNotAllowedMimeType\n      ) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Unsupported file type: {{type}}',\n          undefined,\n          undefined,\n          { type: f.type }\n        );\n        isValid = false;\n      }\n    });\n    return isValid;\n  }\n\n  private setCanSendMessages() {\n    const capabilities = this.channel?.data?.own_capabilities as string[];\n    if (!capabilities) {\n      this.canSendMessages = false;\n    } else {\n      this.canSendMessages =\n        capabilities.indexOf(\n          this.mode === 'main' ? 'send-message' : 'send-reply'\n        ) !== -1;\n    }\n    if (this.isViewInited) {\n      this.cdRef.detectChanges();\n      this.initTextarea();\n    }\n  }\n\n  private get parentMessageId() {\n    let parentMessageId: string | undefined = undefined;\n    if (this.mode === 'thread') {\n      this.channelService.activeParentMessageId$\n        .pipe(first())\n        .subscribe((id) => (parentMessageId = id));\n    }\n\n    return parentMessageId;\n  }\n\n  private startCooldown(cooldown: number) {\n    this.isCooldownInProgress = true;\n    this.cooldown$ = timer(0, 1000).pipe(\n      take(cooldown + 1),\n      map((v) => cooldown - v),\n      tap((v) => {\n        if (v === 0) {\n          this.stopCooldown();\n        }\n      })\n    );\n  }\n\n  private stopCooldown() {\n    this.cooldown$ = undefined;\n    this.isCooldownInProgress = false;\n    // the anchor directive will be recreated because of *ngIf, so we will have to reinit the textarea as well\n    this.textareaRef = undefined;\n    // we can only create the textarea after the anchor was recreated, so we will have to wait a change detection cycle with setTimeout\n    setTimeout(() => this.initTextarea());\n  }\n}\n","<div\n  class=\"{{\n    mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n  }}\"\n  [class.str-chat__input-flat-has-attachments]=\"\n    (attachmentUploads$ | async)!.length > 0\n  \"\n  [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n  <div\n    data-testid=\"quoted-message-container\"\n    class=\"quoted-message-preview\"\n    *ngIf=\"quotedMessage\"\n  >\n    <div class=\"quoted-message-preview-header\">\n      <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n      <button\n        class=\"str-chat__square-button\"\n        data-testid=\"remove-quote\"\n        (click)=\"deselectMessageToQuote()\"\n        (keyup.enter)=\"deselectMessageToQuote()\"\n      >\n        <stream-icon\n          icon=\"close-no-outline\"\n          style=\"font-size: 10px; line-height: 10px\"\n        ></stream-icon>\n      </button>\n    </div>\n    <div class=\"quoted-message-preview-content\">\n      <stream-avatar\n        data-testid=\"qouted-message-avatar\"\n        class=\"str-chat-angular__avatar-host\"\n        [imageUrl]=\"quotedMessage?.user?.image\"\n        [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n        [size]=\"20\"\n      ></stream-avatar>\n      <div class=\"quoted-message-preview-content-inner\">\n        <stream-attachment-list\n          *ngIf=\"\n            quotedMessage?.attachments && quotedMessage?.attachments?.length\n          \"\n          [attachments]=\"quotedMessageAttachments\"\n          [messageId]=\"quotedMessage?.id\"\n        ></stream-attachment-list>\n        <div\n          data-testid=\"quoted-message-text\"\n          [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n        ></div>\n      </div>\n    </div>\n  </div>\n  <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n    <div\n      class=\"{{\n        mode === 'main'\n          ? 'str-chat__input-flat--textarea-wrapper'\n          : 'str-chat__small-message-input--textarea-wrapper'\n      }}\"\n    >\n      <stream-attachment-preview-list\n        class=\"rfu-image-previewer-angular-host\"\n      ></stream-attachment-preview-list>\n      <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n        <ng-container\n          *ngIf=\"emojiPickerTemplate && !isCooldownInProgress\"\n          data-testid=\"emoji-picker\"\n        >\n          <div\n            class=\"\n              str-chat__input-flat-emojiselect\n              str-chat-angular__emojiselect\n            \"\n          >\n            <ng-container\n              *ngTemplateOutlet=\"\n                emojiPickerTemplate;\n                context: { emojiInput$: emojiInputService.emojiInput$ }\n              \"\n            ></ng-container>\n          </div>\n        </ng-container>\n        <div\n          class=\"str-chat__input-flat-cooldown str-chat-angular__cooldown\"\n          *ngIf=\"isCooldownInProgress\"\n          data-testid=\"cooldown-timer\"\n        >\n          {{ cooldown$ | async }}\n        </div>\n        <ng-template\n          *ngIf=\"canSendMessages && !isCooldownInProgress; else notAllowed\"\n          streamTextarea\n          [(value)]=\"textareaValue\"\n          (valueChange)=\"typingStart$.next()\"\n          (send)=\"messageSent()\"\n          [componentRef]=\"textareaRef\"\n          (userMentions)=\"mentionedUsers = $event\"\n          [areMentionsEnabled]=\"areMentionsEnabled\"\n          [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n          [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n          [mentionScope]=\"mentionScope\"\n        ></ng-template>\n        <ng-template #notAllowed>\n          <textarea\n            disabled\n            rows=\"1\"\n            [value]=\"disabledTextareaText | translate\"\n            class=\"rta__textarea str-chat__textarea__textarea\"\n            data-testid=\"disabled-textarea\"\n          ></textarea>\n        </ng-template>\n      </div>\n      <div\n        *ngIf=\"\n          isFileUploadEnabled &&\n          isFileUploadAuthorized &&\n          canSendMessages &&\n          !isCooldownInProgress\n        \"\n        class=\"str-chat__fileupload-wrapper\"\n        data-testid=\"file-upload-button\"\n      >\n        <div class=\"str-chat__tooltip\">\n          {{ \"streamChat.Attach files\" | translate }}\n        </div>\n        <div class=\"rfu-file-upload-button\">\n          <label>\n            <input\n              #fileInput\n              type=\"file\"\n              class=\"rfu-file-input\"\n              data-testid=\"file-input\"\n              [accept]=\"accept\"\n              [multiple]=\"isMultipleFileUploadEnabled\"\n              (change)=\"filesSelected(fileInput.files)\"\n            />\n            <span class=\"str-chat__input-flat-fileupload\">\n              <stream-icon icon=\"file-upload\"></stream-icon>\n            </span>\n          </label>\n        </div>\n      </div>\n    </div>\n    <button\n      *ngIf=\"canSendMessages\"\n      data-testid=\"send-button\"\n      class=\"str-chat__send-button\"\n      (click)=\"messageSent()\"\n      (keyup.enter)=\"messageSent()\"\n    >\n      <stream-icon icon=\"send\"></stream-icon>\n    </button>\n  </div>\n</div>\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9zdHJlYW0tY2hhdC1hbmd1bGFyL3NyYy9saWIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgQ2hhbm5lbE1lbWJlclJlc3BvbnNlLFxuICBDb21tYW5kUmVzcG9uc2UsXG4gIEV2ZW50LFxuICBGb3JtYXRNZXNzYWdlUmVzcG9uc2UsXG4gIExpdGVyYWxTdHJpbmdGb3JVbmlvbixcbiAgTXV0ZSxcbiAgVXNlclJlc3BvbnNlLFxufSBmcm9tICdzdHJlYW0tY2hhdCc7XG5cbmV4cG9ydCB0eXBlIFVua25vd25UeXBlID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG5cbmV4cG9ydCB0eXBlIEN1c3RvbVRyaWdnZXIgPSB7XG4gIFtrZXk6IHN0cmluZ106IHtcbiAgICBjb21wb25lbnRQcm9wczogVW5rbm93blR5cGU7XG4gICAgZGF0YTogVW5rbm93blR5cGU7XG4gIH07XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0QXR0YWNobWVudFR5cGUgPSBVbmtub3duVHlwZSAmIHtcbiAgYXNzZXRfdXJsPzogc3RyaW5nO1xuICBmaWxlX3NpemU/OiBudW1iZXI7XG4gIGlkPzogc3RyaW5nO1xuICBpbWFnZXM/OiBBcnJheTx7XG4gICAgaW1hZ2VfdXJsPzogc3RyaW5nO1xuICAgIHRodW1iX3VybD86IHN0cmluZztcbiAgfT47XG4gIG1pbWVfdHlwZT86IHN0cmluZztcbn07XG5cbmV4cG9ydCB0eXBlIERlZmF1bHRDaGFubmVsVHlwZSA9IFVua25vd25UeXBlICYge1xuICBpbWFnZT86IHN0cmluZztcbiAgbWVtYmVyX2NvdW50PzogbnVtYmVyO1xuICBzdWJ0aXRsZT86IHN0cmluZztcbn07XG5cbmV4cG9ydCB0eXBlIERlZmF1bHRDb21tYW5kVHlwZSA9IExpdGVyYWxTdHJpbmdGb3JVbmlvbjtcblxuZXhwb3J0IHR5cGUgRGVmYXVsdEV2ZW50VHlwZSA9IFVua25vd25UeXBlO1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0TWVzc2FnZVR5cGUgPSBVbmtub3duVHlwZSAmIHtcbiAgY3VzdG9tVHlwZT86ICdjaGFubmVsLmludHJvJyB8ICdtZXNzYWdlLmRhdGUnO1xuICBkYXRlPzogc3RyaW5nIHwgRGF0ZTtcbiAgZXJyb3JTdGF0dXNDb2RlPzogbnVtYmVyO1xuICBldmVudD86IEV2ZW50PFxuICAgIERlZmF1bHRBdHRhY2htZW50VHlwZSxcbiAgICBEZWZhdWx0Q2hhbm5lbFR5cGUsXG4gICAgRGVmYXVsdENvbW1hbmRUeXBlLFxuICAgIERlZmF1bHRFdmVudFR5cGUsXG4gICAgRGVmYXVsdE1lc3NhZ2VUeXBlLFxuICAgIERlZmF1bHRSZWFjdGlvblR5cGUsXG4gICAgRGVmYXVsdFVzZXJUeXBlXG4gID47XG4gIHVucmVhZD86IGJvb2xlYW47XG4gIHJlYWRCeTogVXNlclJlc3BvbnNlPERlZmF1bHRVc2VyVHlwZT5bXTtcbn07XG5cbmV4cG9ydCB0eXBlIERlZmF1bHRSZWFjdGlvblR5cGUgPSBVbmtub3duVHlwZTtcblxuZXhwb3J0IHR5cGUgRGVmYXVsdFVzZXJUeXBlSW50ZXJuYWwgPSB7XG4gIGltYWdlPzogc3RyaW5nO1xuICBzdGF0dXM/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0VXNlclR5cGU8XG4gIFVzZXJUeXBlIGV4dGVuZHMgRGVmYXVsdFVzZXJUeXBlSW50ZXJuYWwgPSBEZWZhdWx0VXNlclR5cGVJbnRlcm5hbFxuPiA9IFVua25vd25UeXBlICZcbiAgRGVmYXVsdFVzZXJUeXBlSW50ZXJuYWwgJiB7XG4gICAgbXV0ZXM/OiBBcnJheTxNdXRlPFVzZXJUeXBlPj47XG4gIH07XG5cbmV4cG9ydCB0eXBlIFN0cmVhbU1lc3NhZ2U8XG4gIEF0IGV4dGVuZHMgRGVmYXVsdEF0dGFjaG1lbnRUeXBlID0gRGVmYXVsdEF0dGFjaG1lbnRUeXBlLFxuICBDaCBleHRlbmRzIERlZmF1bHRDaGFubmVsVHlwZSA9IERlZmF1bHRDaGFubmVsVHlwZSxcbiAgQ28gZXh0ZW5kcyBEZWZhdWx0Q29tbWFuZFR5cGUgPSBEZWZhdWx0Q29tbWFuZFR5cGUsXG4gIE1lIGV4dGVuZHMgRGVmYXVsdE1lc3NhZ2VUeXBlID0gRGVmYXVsdE1lc3NhZ2VUeXBlLFxuICBSZSBleHRlbmRzIERlZmF1bHRSZWFjdGlvblR5cGUgPSBEZWZhdWx0UmVhY3Rpb25UeXBlLFxuICBVcyBleHRlbmRzIERlZmF1bHRVc2VyVHlwZTxVcz4gPSBEZWZhdWx0VXNlclR5cGVcbj4gPSBGb3JtYXRNZXNzYWdlUmVzcG9uc2U8QXQsIENoLCBDbywgTWUsIFJlLCBVcz47XG5cbmV4cG9ydCB0eXBlIEF0dGFjaG1lbnRVcGxvYWQgPSB7XG4gIGZpbGU6IEZpbGU7XG4gIHN0YXRlOiAnZXJyb3InIHwgJ3N1Y2Nlc3MnIHwgJ3VwbG9hZGluZyc7XG4gIHVybD86IHN0cmluZztcbiAgdHlwZTogJ2ltYWdlJyB8ICdmaWxlJztcbiAgcHJldmlld1VyaT86IHN0cmluZyB8IEFycmF5QnVmZmVyO1xufTtcblxuZXhwb3J0IHR5cGUgTWVudGlvbkF1dGNvbXBsZXRlTGlzdEl0ZW1Db250ZXh0ID0ge1xuICBpdGVtOiBNZW50aW9uQXV0Y29tcGxldGVMaXN0SXRlbTtcbn07XG5cbmV4cG9ydCB0eXBlIENvbW1hbmRBdXRvY29tcGxldGVMaXN0SXRlbUNvbnRleHQgPSB7XG4gIGl0ZW06IENvbWFuZEF1dG9jb21wbGV0ZUxpc3RJdGVtO1xufTtcblxuZXhwb3J0IHR5cGUgTWVudGlvbkF1dGNvbXBsZXRlTGlzdEl0ZW0gPSAoXG4gIHwgQ2hhbm5lbE1lbWJlclJlc3BvbnNlXG4gIHwgVXNlclJlc3BvbnNlXG4pICYge1xuICBhdXRvY29tcGxldGVMYWJlbDogc3RyaW5nO1xufTtcblxuZXhwb3J0IHR5cGUgQ29tYW5kQXV0b2NvbXBsZXRlTGlzdEl0ZW0gPSBDb21tYW5kUmVzcG9uc2UgJiB7XG4gIGF1dG9jb21wbGV0ZUxhYmVsOiBzdHJpbmc7XG59O1xuIl19
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9zdHJlYW0tY2hhdC1hbmd1bGFyL3NyYy9saWIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHtcbiAgQ2hhbm5lbE1lbWJlclJlc3BvbnNlLFxuICBDb21tYW5kUmVzcG9uc2UsXG4gIEV2ZW50LFxuICBGb3JtYXRNZXNzYWdlUmVzcG9uc2UsXG4gIExpdGVyYWxTdHJpbmdGb3JVbmlvbixcbiAgTXV0ZSxcbiAgVXNlclJlc3BvbnNlLFxufSBmcm9tICdzdHJlYW0tY2hhdCc7XG5cbmV4cG9ydCB0eXBlIFVua25vd25UeXBlID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG5cbmV4cG9ydCB0eXBlIEN1c3RvbVRyaWdnZXIgPSB7XG4gIFtrZXk6IHN0cmluZ106IHtcbiAgICBjb21wb25lbnRQcm9wczogVW5rbm93blR5cGU7XG4gICAgZGF0YTogVW5rbm93blR5cGU7XG4gIH07XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0QXR0YWNobWVudFR5cGUgPSBVbmtub3duVHlwZSAmIHtcbiAgYXNzZXRfdXJsPzogc3RyaW5nO1xuICBpZD86IHN0cmluZztcbiAgaW1hZ2VzPzogQXJyYXk8e1xuICAgIGltYWdlX3VybD86IHN0cmluZztcbiAgICB0aHVtYl91cmw/OiBzdHJpbmc7XG4gIH0+O1xuICBtaW1lX3R5cGU/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0Q2hhbm5lbFR5cGUgPSBVbmtub3duVHlwZSAmIHtcbiAgaW1hZ2U/OiBzdHJpbmc7XG4gIG1lbWJlcl9jb3VudD86IG51bWJlcjtcbiAgc3VidGl0bGU/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0Q29tbWFuZFR5cGUgPSBMaXRlcmFsU3RyaW5nRm9yVW5pb247XG5cbmV4cG9ydCB0eXBlIERlZmF1bHRFdmVudFR5cGUgPSBVbmtub3duVHlwZTtcblxuZXhwb3J0IHR5cGUgRGVmYXVsdE1lc3NhZ2VUeXBlID0gVW5rbm93blR5cGUgJiB7XG4gIGN1c3RvbVR5cGU/OiAnY2hhbm5lbC5pbnRybycgfCAnbWVzc2FnZS5kYXRlJztcbiAgZGF0ZT86IHN0cmluZyB8IERhdGU7XG4gIGVycm9yU3RhdHVzQ29kZT86IG51bWJlcjtcbiAgZXZlbnQ/OiBFdmVudDxcbiAgICBEZWZhdWx0QXR0YWNobWVudFR5cGUsXG4gICAgRGVmYXVsdENoYW5uZWxUeXBlLFxuICAgIERlZmF1bHRDb21tYW5kVHlwZSxcbiAgICBEZWZhdWx0RXZlbnRUeXBlLFxuICAgIERlZmF1bHRNZXNzYWdlVHlwZSxcbiAgICBEZWZhdWx0UmVhY3Rpb25UeXBlLFxuICAgIERlZmF1bHRVc2VyVHlwZVxuICA+O1xuICB1bnJlYWQ/OiBib29sZWFuO1xuICByZWFkQnk6IFVzZXJSZXNwb25zZTxEZWZhdWx0VXNlclR5cGU+W107XG59O1xuXG5leHBvcnQgdHlwZSBEZWZhdWx0UmVhY3Rpb25UeXBlID0gVW5rbm93blR5cGU7XG5cbmV4cG9ydCB0eXBlIERlZmF1bHRVc2VyVHlwZUludGVybmFsID0ge1xuICBpbWFnZT86IHN0cmluZztcbiAgc3RhdHVzPzogc3RyaW5nO1xufTtcblxuZXhwb3J0IHR5cGUgRGVmYXVsdFVzZXJUeXBlPFxuICBVc2VyVHlwZSBleHRlbmRzIERlZmF1bHRVc2VyVHlwZUludGVybmFsID0gRGVmYXVsdFVzZXJUeXBlSW50ZXJuYWxcbj4gPSBVbmtub3duVHlwZSAmXG4gIERlZmF1bHRVc2VyVHlwZUludGVybmFsICYge1xuICAgIG11dGVzPzogQXJyYXk8TXV0ZTxVc2VyVHlwZT4+O1xuICB9O1xuXG5leHBvcnQgdHlwZSBTdHJlYW1NZXNzYWdlPFxuICBBdCBleHRlbmRzIERlZmF1bHRBdHRhY2htZW50VHlwZSA9IERlZmF1bHRBdHRhY2htZW50VHlwZSxcbiAgQ2ggZXh0ZW5kcyBEZWZhdWx0Q2hhbm5lbFR5cGUgPSBEZWZhdWx0Q2hhbm5lbFR5cGUsXG4gIENvIGV4dGVuZHMgRGVmYXVsdENvbW1hbmRUeXBlID0gRGVmYXVsdENvbW1hbmRUeXBlLFxuICBNZSBleHRlbmRzIERlZmF1bHRNZXNzYWdlVHlwZSA9IERlZmF1bHRNZXNzYWdlVHlwZSxcbiAgUmUgZXh0ZW5kcyBEZWZhdWx0UmVhY3Rpb25UeXBlID0gRGVmYXVsdFJlYWN0aW9uVHlwZSxcbiAgVXMgZXh0ZW5kcyBEZWZhdWx0VXNlclR5cGU8VXM+ID0gRGVmYXVsdFVzZXJUeXBlXG4+ID0gRm9ybWF0TWVzc2FnZVJlc3BvbnNlPEF0LCBDaCwgQ28sIE1lLCBSZSwgVXM+O1xuXG5leHBvcnQgdHlwZSBBdHRhY2htZW50VXBsb2FkID0ge1xuICBmaWxlOiBGaWxlO1xuICBzdGF0ZTogJ2Vycm9yJyB8ICdzdWNjZXNzJyB8ICd1cGxvYWRpbmcnO1xuICB1cmw/OiBzdHJpbmc7XG4gIHR5cGU6ICdpbWFnZScgfCAnZmlsZSc7XG4gIHByZXZpZXdVcmk/OiBzdHJpbmcgfCBBcnJheUJ1ZmZlcjtcbn07XG5cbmV4cG9ydCB0eXBlIE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtQ29udGV4dCA9IHtcbiAgaXRlbTogTWVudGlvbkF1dGNvbXBsZXRlTGlzdEl0ZW07XG59O1xuXG5leHBvcnQgdHlwZSBDb21tYW5kQXV0b2NvbXBsZXRlTGlzdEl0ZW1Db250ZXh0ID0ge1xuICBpdGVtOiBDb21hbmRBdXRvY29tcGxldGVMaXN0SXRlbTtcbn07XG5cbmV4cG9ydCB0eXBlIE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtID0gKFxuICB8IENoYW5uZWxNZW1iZXJSZXNwb25zZVxuICB8IFVzZXJSZXNwb25zZVxuKSAmIHtcbiAgYXV0b2NvbXBsZXRlTGFiZWw6IHN0cmluZztcbn07XG5cbmV4cG9ydCB0eXBlIENvbWFuZEF1dG9jb21wbGV0ZUxpc3RJdGVtID0gQ29tbWFuZFJlc3BvbnNlICYge1xuICBhdXRvY29tcGxldGVMYWJlbDogc3RyaW5nO1xufTtcbiJdfQ==
@@ -1,9 +1,9 @@
1
1
  import { __awaiter } from 'tslib';
2
2
  import * as i0 from '@angular/core';
3
3
  import { Injectable, Component, Input, InjectionToken, EventEmitter, Directive, Output, Inject, ViewChild, HostBinding, NgModule } from '@angular/core';
4
- import { BehaviorSubject, ReplaySubject, combineLatest, Subject, of } from 'rxjs';
4
+ import { BehaviorSubject, ReplaySubject, combineLatest, Subject, timer, of } from 'rxjs';
5
5
  import { StreamChat } from 'stream-chat';
6
- import { map, shareReplay, filter, first, catchError, startWith, distinctUntilChanged, debounceTime, tap } from 'rxjs/operators';
6
+ import { map, shareReplay, filter, first, take, tap, catchError, startWith, distinctUntilChanged, debounceTime } from 'rxjs/operators';
7
7
  import { v4 } from 'uuid';
8
8
  import * as i2 from '@ngx-translate/core';
9
9
  import { TranslateModule } from '@ngx-translate/core';
@@ -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.15.0';
20
+ const version = '2.17.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.
@@ -238,6 +238,7 @@ class ChannelService {
238
238
  this.activeChannelSubscriptions = [];
239
239
  this.activeParentMessageIdSubject = new BehaviorSubject(undefined);
240
240
  this.activeThreadMessagesSubject = new BehaviorSubject([]);
241
+ this.latestMessageDateByUserByChannelsSubject = new BehaviorSubject({});
241
242
  this.messagePageSize = 25;
242
243
  this.messageToQuoteSubject = new BehaviorSubject(undefined);
243
244
  this.usersTypingInChannelSubject = new BehaviorSubject([]);
@@ -287,6 +288,8 @@ class ChannelService {
287
288
  this.usersTypingInChannel$ =
288
289
  this.usersTypingInChannelSubject.asObservable();
289
290
  this.usersTypingInThread$ = this.usersTypingInThreadSubject.asObservable();
291
+ this.latestMessageDateByUserByChannels$ =
292
+ this.latestMessageDateByUserByChannelsSubject.asObservable();
290
293
  }
291
294
  /**
292
295
  * Sets the given `channel` as active.
@@ -404,6 +407,7 @@ class ChannelService {
404
407
  this.activeParentMessageIdSubject.next(undefined);
405
408
  this.activeThreadMessagesSubject.next([]);
406
409
  this.channelsSubject.next(undefined);
410
+ this.latestMessageDateByUserByChannelsSubject.next({});
407
411
  this.selectMessageToQuote(undefined);
408
412
  }
409
413
  /**
@@ -724,6 +728,7 @@ class ChannelService {
724
728
  void (c === null || c === void 0 ? void 0 : c.markRead());
725
729
  }
726
730
  });
731
+ this.updateLatestMessages(event);
727
732
  });
728
733
  }));
729
734
  this.activeChannelSubscriptions.push(channel.on('message.updated', (event) => this.messageUpdated(event)));
@@ -945,7 +950,6 @@ class ChannelService {
945
950
  }
946
951
  }
947
952
  }
948
- // truncate active thread as well
949
953
  handleChannelTruncate(event) {
950
954
  var _a, _b;
951
955
  const channelIndex = this.channels.findIndex((c) => c.cid === event.channel.cid);
@@ -1023,6 +1027,26 @@ class ChannelService {
1023
1027
  return;
1024
1028
  }
1025
1029
  }
1030
+ updateLatestMessages(event) {
1031
+ var _a, _b, _c, _d, _e, _f, _g;
1032
+ if (((_b = (_a = event.message) === null || _a === void 0 ? void 0 : _a.user) === null || _b === void 0 ? void 0 : _b.id) !== ((_d = (_c = this.chatClientService) === null || _c === void 0 ? void 0 : _c.chatClient.user) === null || _d === void 0 ? void 0 : _d.id)) {
1033
+ return;
1034
+ }
1035
+ const latestMessages = this.latestMessageDateByUserByChannelsSubject.getValue();
1036
+ if (!((_e = event.message) === null || _e === void 0 ? void 0 : _e.created_at)) {
1037
+ return;
1038
+ }
1039
+ const channelId = (_f = event === null || event === void 0 ? void 0 : event.message) === null || _f === void 0 ? void 0 : _f.cid;
1040
+ if (!channelId) {
1041
+ return;
1042
+ }
1043
+ const messageDate = new Date(event.message.created_at);
1044
+ if (!latestMessages[channelId] ||
1045
+ ((_g = latestMessages[channelId]) === null || _g === void 0 ? void 0 : _g.getTime()) < messageDate.getTime()) {
1046
+ latestMessages[channelId] = messageDate;
1047
+ this.latestMessageDateByUserByChannelsSubject.next(Object.assign({}, latestMessages));
1048
+ }
1049
+ }
1026
1050
  }
1027
1051
  ChannelService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ChannelService, deps: [{ token: ChatClientService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
1028
1052
  ChannelService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: ChannelService, providedIn: 'root' });
@@ -1765,7 +1789,7 @@ class AttachmentListComponent {
1765
1789
  return (attachment.file_size && Number.isFinite(Number(attachment.file_size)));
1766
1790
  }
1767
1791
  getFileSize(attachment) {
1768
- return prettybytes(attachment.file_size);
1792
+ return prettybytes(Number(attachment.file_size));
1769
1793
  }
1770
1794
  trimUrl(url) {
1771
1795
  if (url !== undefined && url !== null) {
@@ -1858,6 +1882,7 @@ class MessageInputComponent {
1858
1882
  this.textareaValue = '';
1859
1883
  this.mentionedUsers = [];
1860
1884
  this.typingStart$ = new Subject();
1885
+ this.isCooldownInProgress = false;
1861
1886
  this.subscriptions = [];
1862
1887
  this.isViewInited = false;
1863
1888
  this.subscriptions.push(this.attachmentService.attachmentUploadInProgressCounter$.subscribe((counter) => {
@@ -1901,6 +1926,26 @@ class MessageInputComponent {
1901
1926
  this.configService.commandAutocompleteItemTemplate;
1902
1927
  this.emojiPickerTemplate = this.configService.emojiPickerTemplate;
1903
1928
  this.subscriptions.push(this.typingStart$.subscribe(() => void this.channelService.typingStarted(this.parentMessageId)));
1929
+ this.subscriptions.push(combineLatest([
1930
+ this.channelService.latestMessageDateByUserByChannels$,
1931
+ this.channelService.activeChannel$,
1932
+ ])
1933
+ .pipe(map(([latestMessages, channel]) => [latestMessages[(channel === null || channel === void 0 ? void 0 : channel.cid) || ''], channel]))
1934
+ .subscribe(([latestMessageDate, channel]) => {
1935
+ var _a, _b, _c;
1936
+ const cooldown = ((_a = channel === null || channel === void 0 ? void 0 : channel.data) === null || _a === void 0 ? void 0 : _a.cooldown) &&
1937
+ latestMessageDate &&
1938
+ Math.round(((_b = channel === null || channel === void 0 ? void 0 : channel.data) === null || _b === void 0 ? void 0 : _b.cooldown) -
1939
+ (new Date().getTime() - latestMessageDate.getTime()) / 1000);
1940
+ if (cooldown &&
1941
+ cooldown > 0 &&
1942
+ ((_c = channel === null || channel === void 0 ? void 0 : channel.data) === null || _c === void 0 ? void 0 : _c.own_capabilities).includes('slow-mode')) {
1943
+ this.startCooldown(cooldown);
1944
+ }
1945
+ else if (this.isCooldownInProgress) {
1946
+ this.stopCooldown();
1947
+ }
1948
+ }));
1904
1949
  }
1905
1950
  ngAfterViewInit() {
1906
1951
  this.isViewInited = true;
@@ -2008,6 +2053,17 @@ class MessageInputComponent {
2008
2053
  ? [originalAttachments[0]]
2009
2054
  : [];
2010
2055
  }
2056
+ get disabledTextareaText() {
2057
+ if (!this.canSendMessages) {
2058
+ return this.mode === 'thread'
2059
+ ? "streamChat.You can't send thread replies in this channel"
2060
+ : "streamChat.You can't send messages in this channel";
2061
+ }
2062
+ else if (this.cooldown$) {
2063
+ return 'streamChat.Slow Mode ON';
2064
+ }
2065
+ return '';
2066
+ }
2011
2067
  filesSelected(fileList) {
2012
2068
  return __awaiter(this, void 0, void 0, function* () {
2013
2069
  if (!(yield this.areAttachemntsValid(fileList))) {
@@ -2109,9 +2165,25 @@ class MessageInputComponent {
2109
2165
  }
2110
2166
  return parentMessageId;
2111
2167
  }
2168
+ startCooldown(cooldown) {
2169
+ this.isCooldownInProgress = true;
2170
+ this.cooldown$ = timer(0, 1000).pipe(take(cooldown + 1), map((v) => cooldown - v), tap((v) => {
2171
+ if (v === 0) {
2172
+ this.stopCooldown();
2173
+ }
2174
+ }));
2175
+ }
2176
+ stopCooldown() {
2177
+ this.cooldown$ = undefined;
2178
+ this.isCooldownInProgress = false;
2179
+ // the anchor directive will be recreated because of *ngIf, so we will have to reinit the textarea as well
2180
+ this.textareaRef = undefined;
2181
+ // we can only create the textarea after the anchor was recreated, so we will have to wait a change detection cycle with setTimeout
2182
+ setTimeout(() => this.initTextarea());
2183
+ }
2112
2184
  }
2113
2185
  MessageInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, deps: [{ token: ChannelService }, { token: NotificationService }, { token: AttachmentService }, { token: MessageInputConfigService }, { token: textareaInjectionToken }, { token: i0.ComponentFactoryResolver }, { token: i0.ChangeDetectorRef }, { token: ChatClientService }, { token: EmojiInputService }], target: i0.ɵɵFactoryTarget.Component });
2114
- MessageInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageInputComponent, selector: "stream-message-input", inputs: { isFileUploadEnabled: "isFileUploadEnabled", areMentionsEnabled: "areMentionsEnabled", mentionScope: "mentionScope", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", commandAutocompleteItemTemplate: "commandAutocompleteItemTemplate", emojiPickerTemplate: "emojiPickerTemplate", mode: "mode", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService, EmojiInputService], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "textareaAnchor", first: true, predicate: TextareaDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"{{\n mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n }}\"\n [class.str-chat__input-flat-has-attachments]=\"\n (attachmentUploads$ | async)!.length > 0\n \"\n [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n <div\n data-testid=\"quoted-message-container\"\n class=\"quoted-message-preview\"\n *ngIf=\"quotedMessage\"\n >\n <div class=\"quoted-message-preview-header\">\n <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n <button\n class=\"str-chat__square-button\"\n data-testid=\"remove-quote\"\n (click)=\"deselectMessageToQuote()\"\n (keyup.enter)=\"deselectMessageToQuote()\"\n >\n <stream-icon\n icon=\"close-no-outline\"\n style=\"font-size: 10px; line-height: 10px\"\n ></stream-icon>\n </button>\n </div>\n <div class=\"quoted-message-preview-content\">\n <stream-avatar\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host\"\n [imageUrl]=\"quotedMessage?.user?.image\"\n [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n [size]=\"20\"\n ></stream-avatar>\n <div class=\"quoted-message-preview-content-inner\">\n <stream-attachment-list\n *ngIf=\"\n quotedMessage?.attachments && quotedMessage?.attachments?.length\n \"\n [attachments]=\"quotedMessageAttachments\"\n [messageId]=\"quotedMessage?.id\"\n ></stream-attachment-list>\n <div\n data-testid=\"quoted-message-text\"\n [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n ></div>\n </div>\n </div>\n </div>\n <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n <div\n class=\"{{\n mode === 'main'\n ? 'str-chat__input-flat--textarea-wrapper'\n : 'str-chat__small-message-input--textarea-wrapper'\n }}\"\n >\n <stream-attachment-preview-list\n class=\"rfu-image-previewer-angular-host\"\n ></stream-attachment-preview-list>\n <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n <ng-container *ngIf=\"emojiPickerTemplate\">\n <div\n class=\"\n str-chat__input-flat-emojiselect\n str-chat-angular__emojiselect\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n emojiPickerTemplate;\n context: { emojiInput$: emojiInputService.emojiInput$ }\n \"\n ></ng-container>\n </div>\n </ng-container>\n <ng-template\n *ngIf=\"canSendMessages; else notAllowed\"\n streamTextarea\n [(value)]=\"textareaValue\"\n (valueChange)=\"typingStart$.next()\"\n (send)=\"messageSent()\"\n [componentRef]=\"textareaRef\"\n (userMentions)=\"mentionedUsers = $event\"\n [areMentionsEnabled]=\"areMentionsEnabled\"\n [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n [mentionScope]=\"mentionScope\"\n ></ng-template>\n <ng-template #notAllowed>\n <textarea\n disabled\n rows=\"1\"\n [value]=\"\n (mode === 'thread'\n ? 'You can\\'t send thread replies in this channel'\n : 'streamChat.You can\\'t send messages in this channel'\n ) | translate\n \"\n class=\"rta__textarea str-chat__textarea__textarea\"\n ></textarea>\n </ng-template>\n </div>\n <div\n *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized && canSendMessages\"\n class=\"str-chat__fileupload-wrapper\"\n data-testid=\"file-upload-button\"\n >\n <div class=\"str-chat__tooltip\">\n {{ \"streamChat.Attach files\" | translate }}\n </div>\n <div class=\"rfu-file-upload-button\">\n <label>\n <input\n #fileInput\n type=\"file\"\n class=\"rfu-file-input\"\n data-testid=\"file-input\"\n [accept]=\"accept\"\n [multiple]=\"isMultipleFileUploadEnabled\"\n (change)=\"filesSelected(fileInput.files)\"\n />\n <span class=\"str-chat__input-flat-fileupload\">\n <stream-icon icon=\"file-upload\"></stream-icon>\n </span>\n </label>\n </div>\n </div>\n </div>\n <button\n *ngIf=\"canSendMessages\"\n data-testid=\"send-button\"\n class=\"str-chat__send-button\"\n (click)=\"messageSent()\"\n (keyup.enter)=\"messageSent()\"\n >\n <stream-icon icon=\"send\"></stream-icon>\n </button>\n </div>\n</div>\n", components: [{ type: IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: AttachmentListComponent, selector: "stream-attachment-list", inputs: ["messageId", "attachments"] }, { type: AttachmentPreviewListComponent, selector: "stream-attachment-preview-list" }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: TextareaDirective, selector: "[streamTextarea]", inputs: ["componentRef", "areMentionsEnabled", "mentionAutocompleteItemTemplate", "mentionScope", "commandAutocompleteItemTemplate", "value"], outputs: ["valueChange", "send", "userMentions"] }], pipes: { "async": i6.AsyncPipe, "translate": i2.TranslatePipe } });
2186
+ MessageInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageInputComponent, selector: "stream-message-input", inputs: { isFileUploadEnabled: "isFileUploadEnabled", areMentionsEnabled: "areMentionsEnabled", mentionScope: "mentionScope", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", commandAutocompleteItemTemplate: "commandAutocompleteItemTemplate", emojiPickerTemplate: "emojiPickerTemplate", mode: "mode", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService, EmojiInputService], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "textareaAnchor", first: true, predicate: TextareaDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"{{\n mode === 'main' ? 'str-chat__input-flat' : 'str-chat__small-message-input'\n }}\"\n [class.str-chat__input-flat-has-attachments]=\"\n (attachmentUploads$ | async)!.length > 0\n \"\n [class.str-chat__input-flat-quoted]=\"!!quotedMessage\"\n>\n <div\n data-testid=\"quoted-message-container\"\n class=\"quoted-message-preview\"\n *ngIf=\"quotedMessage\"\n >\n <div class=\"quoted-message-preview-header\">\n <div>{{ \"streamChat.Reply to Message\" | translate }}</div>\n <button\n class=\"str-chat__square-button\"\n data-testid=\"remove-quote\"\n (click)=\"deselectMessageToQuote()\"\n (keyup.enter)=\"deselectMessageToQuote()\"\n >\n <stream-icon\n icon=\"close-no-outline\"\n style=\"font-size: 10px; line-height: 10px\"\n ></stream-icon>\n </button>\n </div>\n <div class=\"quoted-message-preview-content\">\n <stream-avatar\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host\"\n [imageUrl]=\"quotedMessage?.user?.image\"\n [name]=\"quotedMessage?.user?.name || quotedMessage?.user?.id\"\n [size]=\"20\"\n ></stream-avatar>\n <div class=\"quoted-message-preview-content-inner\">\n <stream-attachment-list\n *ngIf=\"\n quotedMessage?.attachments && quotedMessage?.attachments?.length\n \"\n [attachments]=\"quotedMessageAttachments\"\n [messageId]=\"quotedMessage?.id\"\n ></stream-attachment-list>\n <div\n data-testid=\"quoted-message-text\"\n [innerHTML]=\"quotedMessage?.html || quotedMessage?.text\"\n ></div>\n </div>\n </div>\n </div>\n <div class=\"str-chat__input-flat-wrapper\" style=\"width: 100%\">\n <div\n class=\"{{\n mode === 'main'\n ? 'str-chat__input-flat--textarea-wrapper'\n : 'str-chat__small-message-input--textarea-wrapper'\n }}\"\n >\n <stream-attachment-preview-list\n class=\"rfu-image-previewer-angular-host\"\n ></stream-attachment-preview-list>\n <div class=\"rta str-chat__textarea str-chat-angular__textarea\">\n <ng-container\n *ngIf=\"emojiPickerTemplate && !isCooldownInProgress\"\n data-testid=\"emoji-picker\"\n >\n <div\n class=\"\n str-chat__input-flat-emojiselect\n str-chat-angular__emojiselect\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n emojiPickerTemplate;\n context: { emojiInput$: emojiInputService.emojiInput$ }\n \"\n ></ng-container>\n </div>\n </ng-container>\n <div\n class=\"str-chat__input-flat-cooldown str-chat-angular__cooldown\"\n *ngIf=\"isCooldownInProgress\"\n data-testid=\"cooldown-timer\"\n >\n {{ cooldown$ | async }}\n </div>\n <ng-template\n *ngIf=\"canSendMessages && !isCooldownInProgress; else notAllowed\"\n streamTextarea\n [(value)]=\"textareaValue\"\n (valueChange)=\"typingStart$.next()\"\n (send)=\"messageSent()\"\n [componentRef]=\"textareaRef\"\n (userMentions)=\"mentionedUsers = $event\"\n [areMentionsEnabled]=\"areMentionsEnabled\"\n [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n [commandAutocompleteItemTemplate]=\"commandAutocompleteItemTemplate\"\n [mentionScope]=\"mentionScope\"\n ></ng-template>\n <ng-template #notAllowed>\n <textarea\n disabled\n rows=\"1\"\n [value]=\"disabledTextareaText | translate\"\n class=\"rta__textarea str-chat__textarea__textarea\"\n data-testid=\"disabled-textarea\"\n ></textarea>\n </ng-template>\n </div>\n <div\n *ngIf=\"\n isFileUploadEnabled &&\n isFileUploadAuthorized &&\n canSendMessages &&\n !isCooldownInProgress\n \"\n class=\"str-chat__fileupload-wrapper\"\n data-testid=\"file-upload-button\"\n >\n <div class=\"str-chat__tooltip\">\n {{ \"streamChat.Attach files\" | translate }}\n </div>\n <div class=\"rfu-file-upload-button\">\n <label>\n <input\n #fileInput\n type=\"file\"\n class=\"rfu-file-input\"\n data-testid=\"file-input\"\n [accept]=\"accept\"\n [multiple]=\"isMultipleFileUploadEnabled\"\n (change)=\"filesSelected(fileInput.files)\"\n />\n <span class=\"str-chat__input-flat-fileupload\">\n <stream-icon icon=\"file-upload\"></stream-icon>\n </span>\n </label>\n </div>\n </div>\n </div>\n <button\n *ngIf=\"canSendMessages\"\n data-testid=\"send-button\"\n class=\"str-chat__send-button\"\n (click)=\"messageSent()\"\n (keyup.enter)=\"messageSent()\"\n >\n <stream-icon icon=\"send\"></stream-icon>\n </button>\n </div>\n</div>\n", components: [{ type: IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }, { type: AttachmentListComponent, selector: "stream-attachment-list", inputs: ["messageId", "attachments"] }, { type: AttachmentPreviewListComponent, selector: "stream-attachment-preview-list" }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: TextareaDirective, selector: "[streamTextarea]", inputs: ["componentRef", "areMentionsEnabled", "mentionAutocompleteItemTemplate", "mentionScope", "commandAutocompleteItemTemplate", "value"], outputs: ["valueChange", "send", "userMentions"] }], pipes: { "async": i6.AsyncPipe, "translate": i2.TranslatePipe } });
2115
2187
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, decorators: [{
2116
2188
  type: Component,
2117
2189
  args: [{