stream-chat-angular 1.4.2 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/version.d.ts +1 -1
- package/bundles/stream-chat-angular.umd.js +508 -51
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/channel.service.js +29 -3
- package/esm2015/lib/chat-client.service.js +16 -1
- package/esm2015/lib/injection-tokens.js +3 -0
- package/esm2015/lib/message/highlight-mentions.pipe.js +23 -0
- package/esm2015/lib/message/message.component.js +3 -2
- package/esm2015/lib/message-actions-box/message-actions-box.component.js +2 -2
- package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +145 -0
- package/esm2015/lib/message-input/message-input-config.service.js +3 -1
- package/esm2015/lib/message-input/message-input.component.js +52 -19
- package/esm2015/lib/message-input/textarea/textarea.component.js +42 -0
- package/esm2015/lib/message-input/textarea.directive.js +69 -0
- package/esm2015/lib/message-input/textarea.interface.js +2 -0
- package/esm2015/lib/message-preview.js +3 -2
- package/esm2015/lib/stream-autocomplete-textarea.module.js +33 -0
- package/esm2015/lib/stream-avatar.module.js +19 -0
- package/esm2015/lib/stream-chat.module.js +15 -11
- package/esm2015/lib/stream-textarea.module.js +31 -0
- package/esm2015/lib/types.js +1 -1
- package/esm2015/public-api.js +9 -1
- package/fesm2015/stream-chat-angular.js +454 -50
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/channel.service.d.ts +3 -2
- package/lib/chat-client.service.d.ts +1 -0
- package/lib/injection-tokens.d.ts +3 -0
- package/lib/message/highlight-mentions.pipe.d.ts +8 -0
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +42 -0
- package/lib/message-input/message-input-config.service.d.ts +5 -0
- package/lib/message-input/message-input.component.d.ts +18 -7
- package/lib/message-input/textarea/textarea.component.d.ts +16 -0
- package/lib/message-input/textarea.directive.d.ts +21 -0
- package/lib/message-input/textarea.interface.d.ts +12 -0
- package/lib/message-preview.d.ts +1 -1
- package/lib/stream-autocomplete-textarea.module.d.ts +11 -0
- package/lib/stream-avatar.module.d.ts +9 -0
- package/lib/stream-chat.module.d.ts +23 -21
- package/lib/stream-textarea.module.d.ts +9 -0
- package/lib/types.d.ts +7 -1
- package/package.json +2 -1
- package/public-api.d.ts +8 -0
- package/src/assets/version.ts +1 -1
|
@@ -110,7 +110,7 @@ export class MessageActionsBoxComponent {
|
|
|
110
110
|
}
|
|
111
111
|
}
|
|
112
112
|
MessageActionsBoxComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageActionsBoxComponent, deps: [{ token: i1.ChatClientService }, { token: i2.NotificationService }, { token: i3.ChannelService }], target: i0.ɵɵFactoryTarget.Component });
|
|
113
|
-
MessageActionsBoxComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageActionsBoxComponent, selector: "stream-message-actions-box", inputs: { isOpen: "isOpen", isMine: "isMine", message: "message", enabledActions: "enabledActions" }, outputs: { displayedActionsCount: "displayedActionsCount", isEditing: "isEditing" }, viewQueries: [{ propertyName: "messageInput", first: true, predicate: MessageInputComponent, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n data-testid=\"action-box\"\n class=\"str-chat__message-actions-box\"\n [class.str-chat__message-actions-box--open]=\"isOpen\"\n [class.str-chat__message-actions-box--mine]=\"isMine\"\n>\n <ul class=\"str-chat__message-actions-list\">\n <button\n data-testid=\"quote-action\"\n *ngIf=\"isQuoteVisible\"\n (click)=\"quoteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Reply\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"pin-action\"\n *ngIf=\"isPinVisible\"\n (click)=\"pinClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{\n (message?.pinned ? \"streamChat.Unpin\" : \"streamChat.Pin\") | translate\n }}\n </li>\n </button>\n <button\n data-testid=\"flag-action\"\n *ngIf=\"isFlagVisible\"\n (click)=\"flagClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Flag\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"mute-action\"\n *ngIf=\"isMuteVisible\"\n (click)=\"muteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Mute\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"edit-action\"\n *ngIf=\"isEditVisible\"\n (click)=\"editClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Edit Message\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"delete-action\"\n *ngIf=\"isDeleteVisible\"\n (click)=\"deleteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Delete\" | translate }}\n </li>\n </button>\n </ul>\n</div>\n\n<stream-modal\n [isOpen]=\"isEditModalOpen\"\n (isOpenChange)=\"\n isEditModalOpen = $event; isEditModalOpen ? '' : modalClosed()\n \"\n>\n <div class=\"str-chat__edit-message-form\">\n <stream-message-input\n [message]=\"message\"\n (messageUpdate)=\"modalClosed()\"\n ></stream-message-input>\n <stream-notification-list></stream-notification-list>\n <div class=\"str-chat__message-team-form-footer\">\n <div></div>\n <div>\n <button translate data-testid=\"cancel-button\" (click)=\"modalClosed()\">\n streamChat.Cancel\n </button>\n <button\n type=\"submit\"\n translate\n data-testid=\"send-button\"\n (click)=\"sendClicked()\"\n (keyup.enter)=\"sendClicked()\"\n >\n streamChat.Send\n </button>\n </div>\n </div>\n </div>\n</stream-modal>\n", components: [{ type: i4.ModalComponent, selector: "stream-modal", inputs: ["isOpen"], outputs: ["isOpenChange"] }, { type: i5.MessageInputComponent, selector: "stream-message-input", inputs: ["isFileUploadEnabled", "acceptedFileTypes", "isMultipleFileUploadEnabled", "message"], outputs: ["messageUpdate"] }, { type: i6.NotificationListComponent, selector: "stream-notification-list" }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], pipes: { "translate": i8.TranslatePipe } });
|
|
113
|
+
MessageActionsBoxComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageActionsBoxComponent, selector: "stream-message-actions-box", inputs: { isOpen: "isOpen", isMine: "isMine", message: "message", enabledActions: "enabledActions" }, outputs: { displayedActionsCount: "displayedActionsCount", isEditing: "isEditing" }, viewQueries: [{ propertyName: "messageInput", first: true, predicate: MessageInputComponent, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n data-testid=\"action-box\"\n class=\"str-chat__message-actions-box\"\n [class.str-chat__message-actions-box--open]=\"isOpen\"\n [class.str-chat__message-actions-box--mine]=\"isMine\"\n>\n <ul class=\"str-chat__message-actions-list\">\n <button\n data-testid=\"quote-action\"\n *ngIf=\"isQuoteVisible\"\n (click)=\"quoteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Reply\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"pin-action\"\n *ngIf=\"isPinVisible\"\n (click)=\"pinClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{\n (message?.pinned ? \"streamChat.Unpin\" : \"streamChat.Pin\") | translate\n }}\n </li>\n </button>\n <button\n data-testid=\"flag-action\"\n *ngIf=\"isFlagVisible\"\n (click)=\"flagClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Flag\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"mute-action\"\n *ngIf=\"isMuteVisible\"\n (click)=\"muteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Mute\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"edit-action\"\n *ngIf=\"isEditVisible\"\n (click)=\"editClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Edit Message\" | translate }}\n </li>\n </button>\n <button\n data-testid=\"delete-action\"\n *ngIf=\"isDeleteVisible\"\n (click)=\"deleteClicked()\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ \"streamChat.Delete\" | translate }}\n </li>\n </button>\n </ul>\n</div>\n\n<stream-modal\n [isOpen]=\"isEditModalOpen\"\n (isOpenChange)=\"\n isEditModalOpen = $event; isEditModalOpen ? '' : modalClosed()\n \"\n>\n <div class=\"str-chat__edit-message-form\" *ngIf=\"isEditModalOpen\">\n <stream-message-input\n [message]=\"message\"\n (messageUpdate)=\"modalClosed()\"\n ></stream-message-input>\n <stream-notification-list></stream-notification-list>\n <div class=\"str-chat__message-team-form-footer\">\n <div></div>\n <div>\n <button translate data-testid=\"cancel-button\" (click)=\"modalClosed()\">\n streamChat.Cancel\n </button>\n <button\n type=\"submit\"\n translate\n data-testid=\"send-button\"\n (click)=\"sendClicked()\"\n (keyup.enter)=\"sendClicked()\"\n >\n streamChat.Send\n </button>\n </div>\n </div>\n </div>\n</stream-modal>\n", components: [{ type: i4.ModalComponent, selector: "stream-modal", inputs: ["isOpen"], outputs: ["isOpenChange"] }, { type: i5.MessageInputComponent, selector: "stream-message-input", inputs: ["isFileUploadEnabled", "areMentionsEnabled", "mentionScope", "mentionAutocompleteItemTemplate", "acceptedFileTypes", "isMultipleFileUploadEnabled", "message"], outputs: ["messageUpdate"] }, { type: i6.NotificationListComponent, selector: "stream-notification-list" }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], pipes: { "translate": i8.TranslatePipe } });
|
|
114
114
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageActionsBoxComponent, decorators: [{
|
|
115
115
|
type: Component,
|
|
116
116
|
args: [{
|
|
@@ -134,4 +134,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
134
134
|
type: ViewChild,
|
|
135
135
|
args: [MessageInputComponent]
|
|
136
136
|
}] } });
|
|
137
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-actions-box.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { MessageInputComponent } from '../message-input/message-input.component';\nimport { NotificationService } from '../notification.service';\nimport { StreamMessage } from '../types';\n\nexport type MessageActions =\n  | 'edit'\n  | 'delete'\n  | 'edit-any'\n  | 'delete-any'\n  | 'pin'\n  | 'quote'\n  | 'flag'\n  | 'mute';\n\n@Component({\n  selector: 'stream-message-actions-box',\n  templateUrl: './message-actions-box.component.html',\n  styles: [],\n})\nexport class MessageActionsBoxComponent implements OnChanges {\n  @Input() isOpen = false;\n  @Input() isMine = false;\n  @Input() message: StreamMessage | undefined;\n  @Input() enabledActions: MessageActions[] = [];\n  @Output() readonly displayedActionsCount = new EventEmitter<number>();\n  @Output() readonly isEditing = new EventEmitter<boolean>();\n  isEditModalOpen = false;\n  @ViewChild(MessageInputComponent) private messageInput:\n    | MessageInputComponent\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private notificationService: NotificationService,\n    private channelService: ChannelService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.isMine || changes.enabledActions) {\n      let displayedActionsCount = 0;\n      if (this.isQuoteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isEditVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isDeleteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isMuteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isFlagVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isPinVisible) {\n        displayedActionsCount++;\n      }\n      this.displayedActionsCount.next(displayedActionsCount);\n    }\n  }\n\n  get isQuoteVisible() {\n    return this.enabledActions.indexOf('quote') !== -1;\n  }\n\n  get isEditVisible() {\n    return (\n      (this.enabledActions.indexOf('edit') !== -1 && this.isMine) ||\n      this.enabledActions.indexOf('edit-any') !== -1\n    );\n  }\n\n  get isDeleteVisible() {\n    return (\n      (this.enabledActions.indexOf('delete') !== -1 && this.isMine) ||\n      this.enabledActions.indexOf('delete-any') !== -1\n    );\n  }\n\n  get isMuteVisible() {\n    return this.enabledActions.indexOf('mute') !== -1;\n  }\n\n  get isFlagVisible() {\n    return this.enabledActions.indexOf('flag') !== -1 && !this.isMine;\n  }\n\n  get isPinVisible() {\n    return this.enabledActions.indexOf('pin') !== -1;\n  }\n\n  pinClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  async flagClicked() {\n    try {\n      await this.chatClientService.flagMessage(this.message!.id);\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Message has been successfully flagged',\n        'success'\n      );\n    } catch (err) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Error adding flag'\n      );\n    }\n  }\n\n  muteClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  quoteClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  editClicked() {\n    this.isEditing.emit(true);\n    this.isEditModalOpen = true;\n  }\n\n  sendClicked() {\n    this.messageInput?.messageSent();\n  }\n\n  modalClosed() {\n    this.isEditModalOpen = false;\n    this.isEditing.emit(false);\n  }\n\n  async deleteClicked() {\n    try {\n      await this.channelService.deleteMessage(this.message!);\n    } catch (error) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Error deleting message'\n      );\n    }\n  }\n}\n","<div\n  data-testid=\"action-box\"\n  class=\"str-chat__message-actions-box\"\n  [class.str-chat__message-actions-box--open]=\"isOpen\"\n  [class.str-chat__message-actions-box--mine]=\"isMine\"\n>\n  <ul class=\"str-chat__message-actions-list\">\n    <button\n      data-testid=\"quote-action\"\n      *ngIf=\"isQuoteVisible\"\n      (click)=\"quoteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Reply\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"pin-action\"\n      *ngIf=\"isPinVisible\"\n      (click)=\"pinClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{\n          (message?.pinned ? \"streamChat.Unpin\" : \"streamChat.Pin\") | translate\n        }}\n      </li>\n    </button>\n    <button\n      data-testid=\"flag-action\"\n      *ngIf=\"isFlagVisible\"\n      (click)=\"flagClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Flag\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"mute-action\"\n      *ngIf=\"isMuteVisible\"\n      (click)=\"muteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Mute\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"edit-action\"\n      *ngIf=\"isEditVisible\"\n      (click)=\"editClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Edit Message\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"delete-action\"\n      *ngIf=\"isDeleteVisible\"\n      (click)=\"deleteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Delete\" | translate }}\n      </li>\n    </button>\n  </ul>\n</div>\n\n<stream-modal\n  [isOpen]=\"isEditModalOpen\"\n  (isOpenChange)=\"\n    isEditModalOpen = $event; isEditModalOpen ? '' : modalClosed()\n  \"\n>\n  <div class=\"str-chat__edit-message-form\">\n    <stream-message-input\n      [message]=\"message\"\n      (messageUpdate)=\"modalClosed()\"\n    ></stream-message-input>\n    <stream-notification-list></stream-notification-list>\n    <div class=\"str-chat__message-team-form-footer\">\n      <div></div>\n      <div>\n        <button translate data-testid=\"cancel-button\" (click)=\"modalClosed()\">\n          streamChat.Cancel\n        </button>\n        <button\n          type=\"submit\"\n          translate\n          data-testid=\"send-button\"\n          (click)=\"sendClicked()\"\n          (keyup.enter)=\"sendClicked()\"\n        >\n          streamChat.Send\n        </button>\n      </div>\n    </div>\n  </div>\n</stream-modal>\n"]}
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-actions-box.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport { MessageInputComponent } from '../message-input/message-input.component';\nimport { NotificationService } from '../notification.service';\nimport { StreamMessage } from '../types';\n\nexport type MessageActions =\n  | 'edit'\n  | 'delete'\n  | 'edit-any'\n  | 'delete-any'\n  | 'pin'\n  | 'quote'\n  | 'flag'\n  | 'mute';\n\n@Component({\n  selector: 'stream-message-actions-box',\n  templateUrl: './message-actions-box.component.html',\n  styles: [],\n})\nexport class MessageActionsBoxComponent implements OnChanges {\n  @Input() isOpen = false;\n  @Input() isMine = false;\n  @Input() message: StreamMessage | undefined;\n  @Input() enabledActions: MessageActions[] = [];\n  @Output() readonly displayedActionsCount = new EventEmitter<number>();\n  @Output() readonly isEditing = new EventEmitter<boolean>();\n  isEditModalOpen = false;\n  @ViewChild(MessageInputComponent) private messageInput:\n    | MessageInputComponent\n    | undefined;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private notificationService: NotificationService,\n    private channelService: ChannelService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.isMine || changes.enabledActions) {\n      let displayedActionsCount = 0;\n      if (this.isQuoteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isEditVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isDeleteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isMuteVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isFlagVisible) {\n        displayedActionsCount++;\n      }\n      if (this.isPinVisible) {\n        displayedActionsCount++;\n      }\n      this.displayedActionsCount.next(displayedActionsCount);\n    }\n  }\n\n  get isQuoteVisible() {\n    return this.enabledActions.indexOf('quote') !== -1;\n  }\n\n  get isEditVisible() {\n    return (\n      (this.enabledActions.indexOf('edit') !== -1 && this.isMine) ||\n      this.enabledActions.indexOf('edit-any') !== -1\n    );\n  }\n\n  get isDeleteVisible() {\n    return (\n      (this.enabledActions.indexOf('delete') !== -1 && this.isMine) ||\n      this.enabledActions.indexOf('delete-any') !== -1\n    );\n  }\n\n  get isMuteVisible() {\n    return this.enabledActions.indexOf('mute') !== -1;\n  }\n\n  get isFlagVisible() {\n    return this.enabledActions.indexOf('flag') !== -1 && !this.isMine;\n  }\n\n  get isPinVisible() {\n    return this.enabledActions.indexOf('pin') !== -1;\n  }\n\n  pinClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  async flagClicked() {\n    try {\n      await this.chatClientService.flagMessage(this.message!.id);\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Message has been successfully flagged',\n        'success'\n      );\n    } catch (err) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Error adding flag'\n      );\n    }\n  }\n\n  muteClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  quoteClicked() {\n    alert('Feature not yet implemented');\n  }\n\n  editClicked() {\n    this.isEditing.emit(true);\n    this.isEditModalOpen = true;\n  }\n\n  sendClicked() {\n    this.messageInput?.messageSent();\n  }\n\n  modalClosed() {\n    this.isEditModalOpen = false;\n    this.isEditing.emit(false);\n  }\n\n  async deleteClicked() {\n    try {\n      await this.channelService.deleteMessage(this.message!);\n    } catch (error) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Error deleting message'\n      );\n    }\n  }\n}\n","<div\n  data-testid=\"action-box\"\n  class=\"str-chat__message-actions-box\"\n  [class.str-chat__message-actions-box--open]=\"isOpen\"\n  [class.str-chat__message-actions-box--mine]=\"isMine\"\n>\n  <ul class=\"str-chat__message-actions-list\">\n    <button\n      data-testid=\"quote-action\"\n      *ngIf=\"isQuoteVisible\"\n      (click)=\"quoteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Reply\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"pin-action\"\n      *ngIf=\"isPinVisible\"\n      (click)=\"pinClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{\n          (message?.pinned ? \"streamChat.Unpin\" : \"streamChat.Pin\") | translate\n        }}\n      </li>\n    </button>\n    <button\n      data-testid=\"flag-action\"\n      *ngIf=\"isFlagVisible\"\n      (click)=\"flagClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Flag\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"mute-action\"\n      *ngIf=\"isMuteVisible\"\n      (click)=\"muteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Mute\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"edit-action\"\n      *ngIf=\"isEditVisible\"\n      (click)=\"editClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Edit Message\" | translate }}\n      </li>\n    </button>\n    <button\n      data-testid=\"delete-action\"\n      *ngIf=\"isDeleteVisible\"\n      (click)=\"deleteClicked()\"\n    >\n      <li class=\"str-chat__message-actions-list-item\">\n        {{ \"streamChat.Delete\" | translate }}\n      </li>\n    </button>\n  </ul>\n</div>\n\n<stream-modal\n  [isOpen]=\"isEditModalOpen\"\n  (isOpenChange)=\"\n    isEditModalOpen = $event; isEditModalOpen ? '' : modalClosed()\n  \"\n>\n  <div class=\"str-chat__edit-message-form\" *ngIf=\"isEditModalOpen\">\n    <stream-message-input\n      [message]=\"message\"\n      (messageUpdate)=\"modalClosed()\"\n    ></stream-message-input>\n    <stream-notification-list></stream-notification-list>\n    <div class=\"str-chat__message-team-form-footer\">\n      <div></div>\n      <div>\n        <button translate data-testid=\"cancel-button\" (click)=\"modalClosed()\">\n          streamChat.Cancel\n        </button>\n        <button\n          type=\"submit\"\n          translate\n          data-testid=\"send-button\"\n          (click)=\"sendClicked()\"\n          (keyup.enter)=\"sendClicked()\"\n        >\n          streamChat.Send\n        </button>\n      </div>\n    </div>\n  </div>\n</stream-modal>\n"]}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import { __awaiter } from "tslib";
|
|
2
|
+
import { Component, EventEmitter, HostBinding, Input, Output, ViewChild, } from '@angular/core';
|
|
3
|
+
import { BehaviorSubject } from 'rxjs';
|
|
4
|
+
import { debounceTime, distinctUntilChanged } from 'rxjs/operators';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../../channel.service";
|
|
7
|
+
import * as i2 from "../../chat-client.service";
|
|
8
|
+
import * as i3 from "../../avatar/avatar.component";
|
|
9
|
+
import * as i4 from "angular-mentions";
|
|
10
|
+
import * as i5 from "@angular/common";
|
|
11
|
+
import * as i6 from "@ngx-translate/core";
|
|
12
|
+
export class AutocompleteTextareaComponent {
|
|
13
|
+
constructor(channelService, chatClientService) {
|
|
14
|
+
this.channelService = channelService;
|
|
15
|
+
this.chatClientService = chatClientService;
|
|
16
|
+
this.class = 'str-chat__textarea';
|
|
17
|
+
this.value = '';
|
|
18
|
+
this.areMentionsEnabled = true;
|
|
19
|
+
this.mentionScope = 'channel';
|
|
20
|
+
this.valueChange = new EventEmitter();
|
|
21
|
+
this.send = new EventEmitter();
|
|
22
|
+
this.userMentions = new EventEmitter();
|
|
23
|
+
this.labelKey = 'autocompleteLabel';
|
|
24
|
+
this.triggerChar = '@';
|
|
25
|
+
this.autocompleteConfig = {
|
|
26
|
+
mentions: [],
|
|
27
|
+
};
|
|
28
|
+
this.subscriptions = [];
|
|
29
|
+
this.mentionedUsers = [];
|
|
30
|
+
this.userMentionConfig = {
|
|
31
|
+
triggerChar: this.triggerChar,
|
|
32
|
+
dropUp: true,
|
|
33
|
+
labelKey: this.labelKey,
|
|
34
|
+
mentionFilter: this.filter,
|
|
35
|
+
mentionSelect: (item, triggerChar) => this.mentioned(item, triggerChar),
|
|
36
|
+
};
|
|
37
|
+
this.searchTerm$ = new BehaviorSubject('');
|
|
38
|
+
this.searchTerm$
|
|
39
|
+
.pipe(debounceTime(300), distinctUntilChanged())
|
|
40
|
+
.subscribe((searchTerm) => void this.updateMentionOptions(searchTerm));
|
|
41
|
+
this.subscriptions.push(this.channelService.activeChannel$.subscribe(() => {
|
|
42
|
+
this.mentionedUsers = [];
|
|
43
|
+
this.userMentions.next([...this.mentionedUsers]);
|
|
44
|
+
void this.updateMentionOptions(this.searchTerm$.getValue());
|
|
45
|
+
}));
|
|
46
|
+
}
|
|
47
|
+
ngOnChanges(changes) {
|
|
48
|
+
if (changes.areMentionsEnabled) {
|
|
49
|
+
if (this.areMentionsEnabled) {
|
|
50
|
+
this.autocompleteConfig.mentions = [this.userMentionConfig];
|
|
51
|
+
this.autocompleteConfig = Object.assign({}, this.autocompleteConfig);
|
|
52
|
+
}
|
|
53
|
+
else {
|
|
54
|
+
this.autocompleteConfig.mentions = [];
|
|
55
|
+
this.autocompleteConfig = Object.assign({}, this.autocompleteConfig);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
if (changes.mentionScope) {
|
|
59
|
+
void this.updateMentionOptions(this.searchTerm$.getValue());
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
filter(searchString, items) {
|
|
63
|
+
return items.filter((item) => item.autocompleteLabel.toLowerCase().includes(searchString.toLowerCase()));
|
|
64
|
+
}
|
|
65
|
+
mentioned(item, triggerChar = '') {
|
|
66
|
+
this.mentionedUsers.push((item.user ? item.user : item));
|
|
67
|
+
this.userMentions.next([...this.mentionedUsers]);
|
|
68
|
+
return triggerChar + item.autocompleteLabel;
|
|
69
|
+
}
|
|
70
|
+
autcompleteSearchTermChanged(searchTerm) {
|
|
71
|
+
this.searchTerm$.next(searchTerm);
|
|
72
|
+
}
|
|
73
|
+
inputChanged() {
|
|
74
|
+
this.valueChange.emit(this.messageInput.nativeElement.value);
|
|
75
|
+
}
|
|
76
|
+
inputLeft() {
|
|
77
|
+
this.updateMentionedUsersFromText();
|
|
78
|
+
}
|
|
79
|
+
sent(event) {
|
|
80
|
+
event.preventDefault();
|
|
81
|
+
this.updateMentionedUsersFromText();
|
|
82
|
+
this.send.next();
|
|
83
|
+
}
|
|
84
|
+
updateMentionOptions(searchTerm) {
|
|
85
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
86
|
+
if (!this.areMentionsEnabled) {
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
const request = this.mentionScope === 'application'
|
|
90
|
+
? (s) => this.chatClientService.autocompleteUsers(s)
|
|
91
|
+
: (s) => this.channelService.autocompleteMembers(s);
|
|
92
|
+
const result = yield request(searchTerm || '');
|
|
93
|
+
const items = this.filter(searchTerm || '', result.map((i) => {
|
|
94
|
+
const user = (i.user ? i.user : i);
|
|
95
|
+
return Object.assign(Object.assign({}, i), { autocompleteLabel: user.name || user.id, type: 'mention' });
|
|
96
|
+
}));
|
|
97
|
+
this.userMentionConfig.items = items;
|
|
98
|
+
this.autocompleteConfig.mentions = [this.userMentionConfig];
|
|
99
|
+
this.autocompleteConfig = Object.assign({}, this.autocompleteConfig);
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
updateMentionedUsersFromText() {
|
|
103
|
+
const updatedMentionedUsers = [];
|
|
104
|
+
this.mentionedUsers.forEach((u) => {
|
|
105
|
+
const key = u.name || u.id;
|
|
106
|
+
if (this.value.includes(`${this.triggerChar}${key}`)) {
|
|
107
|
+
updatedMentionedUsers.push(u);
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
if (updatedMentionedUsers.length !== this.mentionedUsers.length) {
|
|
111
|
+
this.userMentions.next([...updatedMentionedUsers]);
|
|
112
|
+
this.mentionedUsers = updatedMentionedUsers;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
AutocompleteTextareaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AutocompleteTextareaComponent, deps: [{ token: i1.ChannelService }, { token: i2.ChatClientService }], target: i0.ɵɵFactoryTarget.Component });
|
|
117
|
+
AutocompleteTextareaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AutocompleteTextareaComponent, selector: "stream-autocomplete-textarea", inputs: { value: "value", areMentionsEnabled: "areMentionsEnabled", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", mentionScope: "mentionScope" }, outputs: { valueChange: "valueChange", send: "send", userMentions: "userMentions" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "messageInput", first: true, predicate: ["input"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<textarea\n [value]=\"value || ''\"\n autofocus\n data-testid=\"textarea\"\n #input\n placeholder=\"{{ 'streamChat.Type your message' | translate }}\"\n class=\"rta__textarea str-chat__textarea__textarea\"\n rows=\"1\"\n (input)=\"inputChanged()\"\n (keydown.enter)=\"sent($event)\"\n [mentionConfig]=\"autocompleteConfig\"\n (searchTerm)=\"autcompleteSearchTermChanged($event)\"\n [mentionListTemplate]=\"autocompleteItem\"\n (blur)=\"inputLeft()\"\n></textarea>\n<ng-template #autocompleteItem let-item=\"item\">\n <div class=\"rta rta__item str-chat__emojisearch__item\" [ngSwitch]=\"item.type\">\n <div class=\"rta__entity\" *ngSwitchCase=\"'mention'\">\n <ng-container\n *ngTemplateOutlet=\"\n mentionAutocompleteItemTemplate || defaultMentionTemplate;\n context: { item: item }\n \"\n ></ng-container>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultMentionTemplate let-item=\"item\">\n <div class=\"str-chat__user-item\">\n <stream-avatar\n data-testclass=\"avatar\"\n class=\"str-chat__avatar str-chat__avatar--circle\"\n style=\"height: 20px\"\n [size]=\"20\"\n [imageUrl]=\"item.image || item.user?.image\"\n [name]=\"item.autocompleteLabel\"\n ></stream-avatar>\n <span data-testclass=\"username\" class=\"str-chat__user-item--name\">{{\n item.autocompleteLabel\n }}</span>\n </div>\n</ng-template>\n", components: [{ type: i3.AvatarComponent, selector: "stream-avatar", inputs: ["name", "imageUrl", "size"] }], directives: [{ type: i4.MentionDirective, selector: "[mention], [mentionConfig]", inputs: ["mentionConfig", "mention", "mentionListTemplate"], outputs: ["searchTerm", "itemSelected", "opened", "closed"] }, { type: i5.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i5.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "translate": i6.TranslatePipe } });
|
|
118
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AutocompleteTextareaComponent, decorators: [{
|
|
119
|
+
type: Component,
|
|
120
|
+
args: [{
|
|
121
|
+
selector: 'stream-autocomplete-textarea',
|
|
122
|
+
templateUrl: './autocomplete-textarea.component.html',
|
|
123
|
+
styles: [],
|
|
124
|
+
}]
|
|
125
|
+
}], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.ChatClientService }]; }, propDecorators: { class: [{
|
|
126
|
+
type: HostBinding
|
|
127
|
+
}], value: [{
|
|
128
|
+
type: Input
|
|
129
|
+
}], areMentionsEnabled: [{
|
|
130
|
+
type: Input
|
|
131
|
+
}], mentionAutocompleteItemTemplate: [{
|
|
132
|
+
type: Input
|
|
133
|
+
}], mentionScope: [{
|
|
134
|
+
type: Input
|
|
135
|
+
}], valueChange: [{
|
|
136
|
+
type: Output
|
|
137
|
+
}], send: [{
|
|
138
|
+
type: Output
|
|
139
|
+
}], userMentions: [{
|
|
140
|
+
type: Output
|
|
141
|
+
}], messageInput: [{
|
|
142
|
+
type: ViewChild,
|
|
143
|
+
args: ['input']
|
|
144
|
+
}] } });
|
|
145
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete-textarea.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.ts","../../../../../../projects/stream-chat-angular/src/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { MentionConfig, Mentions } from 'angular-mentions';\nimport {\n  MentionAutcompleteListItemContext,\n  MentionAutcompleteListItem,\n} from '../../types';\n\nimport { BehaviorSubject, Subscription } from 'rxjs';\nimport { UserResponse } from 'stream-chat';\nimport { ChannelService } from '../../channel.service';\nimport { TextareaInterface } from '../textarea.interface';\nimport { ChatClientService } from '../../chat-client.service';\nimport { debounceTime, distinctUntilChanged } from 'rxjs/operators';\n\n@Component({\n  selector: 'stream-autocomplete-textarea',\n  templateUrl: './autocomplete-textarea.component.html',\n  styles: [],\n})\nexport class AutocompleteTextareaComponent\n  implements TextareaInterface, OnChanges\n{\n  @HostBinding() class = 'str-chat__textarea';\n  @Input() value = '';\n  @Input() areMentionsEnabled: boolean | undefined = true;\n  @Input() mentionAutocompleteItemTemplate:\n    | TemplateRef<MentionAutcompleteListItemContext>\n    | undefined;\n  @Input() mentionScope: 'channel' | 'application' = 'channel';\n  @Output() readonly valueChange = new EventEmitter<string>();\n  @Output() readonly send = new EventEmitter<void>();\n  @Output() readonly userMentions = new EventEmitter<UserResponse[]>();\n  private readonly labelKey = 'autocompleteLabel';\n  private readonly triggerChar = '@';\n  autocompleteConfig: MentionConfig = {\n    mentions: [],\n  };\n  @ViewChild('input') private messageInput!: ElementRef<HTMLInputElement>;\n  private subscriptions: Subscription[] = [];\n  private mentionedUsers: UserResponse[] = [];\n  private userMentionConfig: Mentions = {\n    triggerChar: this.triggerChar,\n    dropUp: true,\n    labelKey: this.labelKey,\n    mentionFilter: this.filter,\n    mentionSelect: (item, triggerChar) => this.mentioned(item, triggerChar),\n  };\n  private searchTerm$ = new BehaviorSubject<string>('');\n\n  constructor(\n    private channelService: ChannelService,\n    private chatClientService: ChatClientService\n  ) {\n    this.searchTerm$\n      .pipe(debounceTime(300), distinctUntilChanged())\n      .subscribe((searchTerm) => void this.updateMentionOptions(searchTerm));\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe(() => {\n        this.mentionedUsers = [];\n        this.userMentions.next([...this.mentionedUsers]);\n        void this.updateMentionOptions(this.searchTerm$.getValue());\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.areMentionsEnabled) {\n      if (this.areMentionsEnabled) {\n        this.autocompleteConfig.mentions = [this.userMentionConfig];\n        this.autocompleteConfig = { ...this.autocompleteConfig };\n      } else {\n        this.autocompleteConfig.mentions = [];\n        this.autocompleteConfig = { ...this.autocompleteConfig };\n      }\n    }\n    if (changes.mentionScope) {\n      void this.updateMentionOptions(this.searchTerm$.getValue());\n    }\n  }\n\n  filter(searchString: string, items: { autocompleteLabel: string }[]): any[] {\n    return items.filter((item) =>\n      item.autocompleteLabel.toLowerCase().includes(searchString.toLowerCase())\n    );\n  }\n\n  mentioned(item: MentionAutcompleteListItem, triggerChar = '') {\n    this.mentionedUsers.push((item.user ? item.user : item) as UserResponse);\n    this.userMentions.next([...this.mentionedUsers]);\n    return triggerChar + item.autocompleteLabel;\n  }\n\n  autcompleteSearchTermChanged(searchTerm: string) {\n    this.searchTerm$.next(searchTerm);\n  }\n\n  inputChanged() {\n    this.valueChange.emit(this.messageInput.nativeElement.value);\n  }\n\n  inputLeft() {\n    this.updateMentionedUsersFromText();\n  }\n\n  sent(event: Event) {\n    event.preventDefault();\n    this.updateMentionedUsersFromText();\n    this.send.next();\n  }\n\n  private async updateMentionOptions(searchTerm?: string) {\n    if (!this.areMentionsEnabled) {\n      return;\n    }\n    const request =\n      this.mentionScope === 'application'\n        ? (s: string) => this.chatClientService.autocompleteUsers(s)\n        : (s: string) => this.channelService.autocompleteMembers(s);\n    const result = await request(searchTerm || '');\n    const items = this.filter(\n      searchTerm || '',\n      result.map((i) => {\n        const user = (i.user ? i.user : i) as UserResponse;\n        return {\n          ...i,\n          autocompleteLabel: user.name || user.id,\n          type: 'mention',\n        };\n      })\n    );\n    this.userMentionConfig.items = items;\n    this.autocompleteConfig.mentions = [this.userMentionConfig];\n    this.autocompleteConfig = { ...this.autocompleteConfig };\n  }\n\n  private updateMentionedUsersFromText() {\n    const updatedMentionedUsers: UserResponse[] = [];\n    this.mentionedUsers.forEach((u) => {\n      const key = u.name || u.id;\n      if (this.value.includes(`${this.triggerChar}${key}`)) {\n        updatedMentionedUsers.push(u);\n      }\n    });\n    if (updatedMentionedUsers.length !== this.mentionedUsers.length) {\n      this.userMentions.next([...updatedMentionedUsers]);\n      this.mentionedUsers = updatedMentionedUsers;\n    }\n  }\n}\n","<textarea\n  [value]=\"value || ''\"\n  autofocus\n  data-testid=\"textarea\"\n  #input\n  placeholder=\"{{ 'streamChat.Type your message' | translate }}\"\n  class=\"rta__textarea str-chat__textarea__textarea\"\n  rows=\"1\"\n  (input)=\"inputChanged()\"\n  (keydown.enter)=\"sent($event)\"\n  [mentionConfig]=\"autocompleteConfig\"\n  (searchTerm)=\"autcompleteSearchTermChanged($event)\"\n  [mentionListTemplate]=\"autocompleteItem\"\n  (blur)=\"inputLeft()\"\n></textarea>\n<ng-template #autocompleteItem let-item=\"item\">\n  <div class=\"rta rta__item str-chat__emojisearch__item\" [ngSwitch]=\"item.type\">\n    <div class=\"rta__entity\" *ngSwitchCase=\"'mention'\">\n      <ng-container\n        *ngTemplateOutlet=\"\n          mentionAutocompleteItemTemplate || defaultMentionTemplate;\n          context: { item: item }\n        \"\n      ></ng-container>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #defaultMentionTemplate let-item=\"item\">\n  <div class=\"str-chat__user-item\">\n    <stream-avatar\n      data-testclass=\"avatar\"\n      class=\"str-chat__avatar str-chat__avatar--circle\"\n      style=\"height: 20px\"\n      [size]=\"20\"\n      [imageUrl]=\"item.image || item.user?.image\"\n      [name]=\"item.autocompleteLabel\"\n    ></stream-avatar>\n    <span data-testclass=\"username\" class=\"str-chat__user-item--name\">{{\n      item.autocompleteLabel\n    }}</span>\n  </div>\n</ng-template>\n"]}
|
|
@@ -3,7 +3,9 @@ import * as i0 from "@angular/core";
|
|
|
3
3
|
export class MessageInputConfigService {
|
|
4
4
|
constructor() {
|
|
5
5
|
this.isFileUploadEnabled = true;
|
|
6
|
+
this.areMentionsEnabled = true;
|
|
6
7
|
this.isMultipleFileUploadEnabled = true;
|
|
8
|
+
this.mentionScope = 'channel';
|
|
7
9
|
}
|
|
8
10
|
}
|
|
9
11
|
MessageInputConfigService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputConfigService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
@@ -14,4 +16,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
14
16
|
providedIn: 'root',
|
|
15
17
|
}]
|
|
16
18
|
}], ctorParameters: function () { return []; } });
|
|
17
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1pbnB1dC1jb25maWcuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tZXNzYWdlLWlucHV0L21lc3NhZ2UtaW5wdXQtY29uZmlnLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBZSxNQUFNLGVBQWUsQ0FBQzs7QUFNeEQsTUFBTSxPQUFPLHlCQUF5QjtJQVVwQztRQVRBLHdCQUFtQixHQUF3QixJQUFJLENBQUM7UUFDaEQsdUJBQWtCLEdBQXdCLElBQUksQ0FBQztRQUsvQyxnQ0FBMkIsR0FBd0IsSUFBSSxDQUFDO1FBQ3hELGlCQUFZLEdBQTBDLFNBQVMsQ0FBQztJQUVqRCxDQUFDOztzSEFWTCx5QkFBeUI7MEhBQXpCLHlCQUF5QixjQUZ4QixNQUFNOzJGQUVQLHlCQUF5QjtrQkFIckMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBUZW1wbGF0ZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTWVudGlvbkF1dGNvbXBsZXRlTGlzdEl0ZW1Db250ZXh0IH0gZnJvbSAnLi4vdHlwZXMnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgTWVzc2FnZUlucHV0Q29uZmlnU2VydmljZSB7XG4gIGlzRmlsZVVwbG9hZEVuYWJsZWQ6IGJvb2xlYW4gfCB1bmRlZmluZWQgPSB0cnVlO1xuICBhcmVNZW50aW9uc0VuYWJsZWQ6IGJvb2xlYW4gfCB1bmRlZmluZWQgPSB0cnVlO1xuICBtZW50aW9uQXV0b2NvbXBsZXRlSXRlbVRlbXBsYXRlOlxuICAgIHwgVGVtcGxhdGVSZWY8TWVudGlvbkF1dGNvbXBsZXRlTGlzdEl0ZW1Db250ZXh0PlxuICAgIHwgdW5kZWZpbmVkO1xuICBhY2NlcHRlZEZpbGVUeXBlczogc3RyaW5nW10gfCB1bmRlZmluZWQ7XG4gIGlzTXVsdGlwbGVGaWxlVXBsb2FkRW5hYmxlZDogYm9vbGVhbiB8IHVuZGVmaW5lZCA9IHRydWU7XG4gIG1lbnRpb25TY29wZTogJ2NoYW5uZWwnIHwgJ2FwcGxpY2F0aW9uJyB8IHVuZGVmaW5lZCA9ICdjaGFubmVsJztcblxuICBjb25zdHJ1Y3RvcigpIHt9XG59XG4iXX0=
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import { __awaiter } from "tslib";
|
|
2
|
-
import { Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { Component, EventEmitter, Inject, Input, Output, ViewChild, } from '@angular/core';
|
|
3
3
|
import { first } from 'rxjs/operators';
|
|
4
4
|
import { AttachmentService } from '../attachment.service';
|
|
5
|
+
import { textareaInjectionToken } from '../injection-tokens';
|
|
6
|
+
import { TextareaDirective } from './textarea.directive';
|
|
5
7
|
import * as i0 from "@angular/core";
|
|
6
8
|
import * as i1 from "../channel.service";
|
|
7
9
|
import * as i2 from "../notification.service";
|
|
@@ -9,21 +11,24 @@ import * as i3 from "../attachment.service";
|
|
|
9
11
|
import * as i4 from "./message-input-config.service";
|
|
10
12
|
import * as i5 from "../attachment-preview-list/attachment-preview-list.component";
|
|
11
13
|
import * as i6 from "../icon/icon.component";
|
|
12
|
-
import * as i7 from "
|
|
13
|
-
import * as i8 from "@
|
|
14
|
+
import * as i7 from "./textarea.directive";
|
|
15
|
+
import * as i8 from "@angular/common";
|
|
16
|
+
import * as i9 from "@ngx-translate/core";
|
|
14
17
|
export class MessageInputComponent {
|
|
15
|
-
constructor(channelService, notificationService, attachmentService, configService) {
|
|
18
|
+
constructor(channelService, notificationService, attachmentService, configService, textareaType, componentFactoryResolver) {
|
|
16
19
|
this.channelService = channelService;
|
|
17
20
|
this.notificationService = notificationService;
|
|
18
21
|
this.attachmentService = attachmentService;
|
|
19
22
|
this.configService = configService;
|
|
23
|
+
this.textareaType = textareaType;
|
|
24
|
+
this.componentFactoryResolver = componentFactoryResolver;
|
|
20
25
|
this.messageUpdate = new EventEmitter();
|
|
26
|
+
this.textareaValue = '';
|
|
27
|
+
this.mentionedUsers = [];
|
|
21
28
|
this.subscriptions = [];
|
|
22
29
|
this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
|
|
23
30
|
var _a;
|
|
24
|
-
|
|
25
|
-
this.messageInput.nativeElement.value = '';
|
|
26
|
-
}
|
|
31
|
+
this.textareaValue = '';
|
|
27
32
|
this.attachmentService.resetAttachmentUploads();
|
|
28
33
|
const capabilities = (_a = channel === null || channel === void 0 ? void 0 : channel.data) === null || _a === void 0 ? void 0 : _a.own_capabilities;
|
|
29
34
|
if (capabilities) {
|
|
@@ -42,12 +47,22 @@ export class MessageInputComponent {
|
|
|
42
47
|
this.acceptedFileTypes = this.configService.acceptedFileTypes;
|
|
43
48
|
this.isMultipleFileUploadEnabled =
|
|
44
49
|
this.configService.isMultipleFileUploadEnabled;
|
|
50
|
+
this.areMentionsEnabled = this.configService.areMentionsEnabled;
|
|
51
|
+
this.mentionAutocompleteItemTemplate =
|
|
52
|
+
this.configService.mentionAutocompleteItemTemplate;
|
|
53
|
+
this.mentionScope = this.configService.mentionScope;
|
|
54
|
+
}
|
|
55
|
+
ngOnInit() {
|
|
56
|
+
const componentFactory = this.componentFactoryResolver.resolveComponentFactory(this.textareaType);
|
|
57
|
+
this.textareaRef =
|
|
58
|
+
this.textareaAnchor.viewContainerRef.createComponent(componentFactory);
|
|
45
59
|
}
|
|
46
60
|
ngOnChanges(changes) {
|
|
47
61
|
if (changes.message) {
|
|
48
62
|
this.attachmentService.resetAttachmentUploads();
|
|
49
63
|
if (this.isUpdate) {
|
|
50
64
|
this.attachmentService.createFromAttachments(this.message.attachments || []);
|
|
65
|
+
this.textareaValue = this.message.text || '';
|
|
51
66
|
}
|
|
52
67
|
}
|
|
53
68
|
if (changes.isFileUploadEnabled) {
|
|
@@ -60,13 +75,22 @@ export class MessageInputComponent {
|
|
|
60
75
|
this.configService.isMultipleFileUploadEnabled =
|
|
61
76
|
this.isMultipleFileUploadEnabled;
|
|
62
77
|
}
|
|
78
|
+
if (changes.areMentionsEnabled) {
|
|
79
|
+
this.configService.areMentionsEnabled = this.areMentionsEnabled;
|
|
80
|
+
}
|
|
81
|
+
if (changes.mentionAutocompleteItemTemplate) {
|
|
82
|
+
this.configService.mentionAutocompleteItemTemplate =
|
|
83
|
+
this.mentionAutocompleteItemTemplate;
|
|
84
|
+
}
|
|
85
|
+
if (changes.mentionScope) {
|
|
86
|
+
this.configService.mentionScope = this.mentionScope;
|
|
87
|
+
}
|
|
63
88
|
}
|
|
64
89
|
ngOnDestroy() {
|
|
65
90
|
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
66
91
|
}
|
|
67
|
-
messageSent(
|
|
92
|
+
messageSent() {
|
|
68
93
|
return __awaiter(this, void 0, void 0, function* () {
|
|
69
|
-
event === null || event === void 0 ? void 0 : event.preventDefault();
|
|
70
94
|
let attachmentUploadInProgressCounter;
|
|
71
95
|
this.attachmentService.attachmentUploadInProgressCounter$
|
|
72
96
|
.pipe(first())
|
|
@@ -79,17 +103,17 @@ export class MessageInputComponent {
|
|
|
79
103
|
return;
|
|
80
104
|
}
|
|
81
105
|
const attachments = this.attachmentService.mapToAttachments();
|
|
82
|
-
const text = this.
|
|
106
|
+
const text = this.textareaValue;
|
|
83
107
|
if (!text && (!attachments || attachments.length === 0)) {
|
|
84
108
|
return;
|
|
85
109
|
}
|
|
86
110
|
if (!this.isUpdate) {
|
|
87
|
-
this.
|
|
111
|
+
this.textareaValue = '';
|
|
88
112
|
}
|
|
89
113
|
try {
|
|
90
114
|
yield (this.isUpdate
|
|
91
115
|
? this.channelService.updateMessage(Object.assign(Object.assign({}, this.message), { text: text, attachments: attachments }))
|
|
92
|
-
: this.channelService.sendMessage(text, attachments));
|
|
116
|
+
: this.channelService.sendMessage(text, attachments, this.mentionedUsers));
|
|
93
117
|
this.messageUpdate.emit();
|
|
94
118
|
if (!this.isUpdate) {
|
|
95
119
|
this.attachmentService.resetAttachmentUploads();
|
|
@@ -119,8 +143,8 @@ export class MessageInputComponent {
|
|
|
119
143
|
return !!this.message;
|
|
120
144
|
}
|
|
121
145
|
}
|
|
122
|
-
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 }], target: i0.ɵɵFactoryTarget.Component });
|
|
123
|
-
MessageInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageInputComponent, selector: "stream-message-input", inputs: { isFileUploadEnabled: "isFileUploadEnabled", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService], viewQueries: [{ propertyName: "
|
|
146
|
+
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 }], target: i0.ɵɵFactoryTarget.Component });
|
|
147
|
+
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", acceptedFileTypes: "acceptedFileTypes", isMultipleFileUploadEnabled: "isMultipleFileUploadEnabled", message: "message" }, outputs: { messageUpdate: "messageUpdate" }, providers: [AttachmentService], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }, { propertyName: "textareaAnchor", first: true, predicate: TextareaDirective, descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"str-chat__input-flat\"\n [class.str-chat__input-flat-has-attachments]=\"\n (attachmentUploads$ | async)!.length > 0\n \"\n>\n <div class=\"str-chat__input-flat-wrapper\">\n <div class=\"str-chat__input-flat--textarea-wrapper\">\n <stream-attachment-preview-list\n class=\"rfu-image-previewer-angular-host\"\n ></stream-attachment-preview-list>\n <div class=\"rta str-chat__textarea\">\n <ng-template\n streamTextarea\n [(value)]=\"textareaValue\"\n (send)=\"messageSent()\"\n [componentRef]=\"textareaRef\"\n (userMentions)=\"mentionedUsers = $event\"\n [areMentionsEnabled]=\"areMentionsEnabled\"\n [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n [mentionScope]=\"mentionScope\"\n ></ng-template>\n </div>\n <div\n *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized\"\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 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: i5.AttachmentPreviewListComponent, selector: "stream-attachment-preview-list" }, { type: i6.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }], directives: [{ type: i7.TextareaDirective, selector: "[streamTextarea]", inputs: ["componentRef", "areMentionsEnabled", "mentionAutocompleteItemTemplate", "mentionScope", "value"], outputs: ["valueChange", "send", "userMentions"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], pipes: { "async": i8.AsyncPipe, "translate": i9.TranslatePipe } });
|
|
124
148
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, decorators: [{
|
|
125
149
|
type: Component,
|
|
126
150
|
args: [{
|
|
@@ -129,7 +153,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
129
153
|
styles: [],
|
|
130
154
|
providers: [AttachmentService],
|
|
131
155
|
}]
|
|
132
|
-
}], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.NotificationService }, { type: i3.AttachmentService }, { type: i4.MessageInputConfigService }
|
|
156
|
+
}], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.NotificationService }, { type: i3.AttachmentService }, { type: i4.MessageInputConfigService }, { type: i0.Type, decorators: [{
|
|
157
|
+
type: Inject,
|
|
158
|
+
args: [textareaInjectionToken]
|
|
159
|
+
}] }, { type: i0.ComponentFactoryResolver }]; }, propDecorators: { isFileUploadEnabled: [{
|
|
160
|
+
type: Input
|
|
161
|
+
}], areMentionsEnabled: [{
|
|
162
|
+
type: Input
|
|
163
|
+
}], mentionScope: [{
|
|
164
|
+
type: Input
|
|
165
|
+
}], mentionAutocompleteItemTemplate: [{
|
|
133
166
|
type: Input
|
|
134
167
|
}], acceptedFileTypes: [{
|
|
135
168
|
type: Input
|
|
@@ -139,11 +172,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
139
172
|
type: Input
|
|
140
173
|
}], messageUpdate: [{
|
|
141
174
|
type: Output
|
|
142
|
-
}], messageInput: [{
|
|
143
|
-
type: ViewChild,
|
|
144
|
-
args: ['input']
|
|
145
175
|
}], fileInput: [{
|
|
146
176
|
type: ViewChild,
|
|
147
177
|
args: ['fileInput']
|
|
178
|
+
}], textareaAnchor: [{
|
|
179
|
+
type: ViewChild,
|
|
180
|
+
args: [TextareaDirective, { static: true }]
|
|
148
181
|
}] } });
|
|
149
|
-
//# 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  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { first } from 'rxjs/operators';\nimport { AttachmentService } from '../attachment.service';\nimport { ChannelService } from '../channel.service';\nimport { NotificationService } from '../notification.service';\nimport { AttachmentUpload, StreamMessage } from '../types';\nimport { MessageInputConfigService } from './message-input-config.service';\n\n@Component({\n  selector: 'stream-message-input',\n  templateUrl: './message-input.component.html',\n  styles: [],\n  providers: [AttachmentService],\n})\nexport class MessageInputComponent implements OnChanges, OnDestroy {\n  @Input() isFileUploadEnabled: boolean | undefined;\n  @Input() acceptedFileTypes: string[] | undefined;\n  @Input() isMultipleFileUploadEnabled: boolean | undefined;\n  @Input() message: StreamMessage | undefined;\n  @Output() readonly messageUpdate = new EventEmitter<void>();\n  isFileUploadAuthorized: boolean | undefined;\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  @ViewChild('input') private messageInput!: ElementRef<HTMLInputElement>;\n  @ViewChild('fileInput') private fileInput!: ElementRef<HTMLInputElement>;\n  private subscriptions: Subscription[] = [];\n  private hideNotification: Function | undefined;\n\n  constructor(\n    private channelService: ChannelService,\n    private notificationService: NotificationService,\n    private attachmentService: AttachmentService,\n    private configService: MessageInputConfigService\n  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        if (this.messageInput) {\n          this.messageInput.nativeElement.value = '';\n        }\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        }\n      })\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.attachmentUploads$ = this.attachmentService.attachmentUploads$;\n    this.isFileUploadEnabled = this.configService.isFileUploadEnabled;\n    this.acceptedFileTypes = this.configService.acceptedFileTypes;\n    this.isMultipleFileUploadEnabled =\n      this.configService.isMultipleFileUploadEnabled;\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      }\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  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async messageSent(event?: Event) {\n    event?.preventDefault();\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.messageInput.nativeElement.value;\n    if (!text && (!attachments || attachments.length === 0)) {\n      return;\n    }\n    if (!this.isUpdate) {\n      this.messageInput.nativeElement.value = '';\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(text, attachments));\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  }\n\n  get accept() {\n    return this.acceptedFileTypes ? this.acceptedFileTypes?.join(',') : '';\n  }\n\n  async filesSelected(fileList: FileList | null) {\n    await this.attachmentService.filesSelected(fileList);\n    this.clearFileInput();\n  }\n\n  private clearFileInput() {\n    this.fileInput.nativeElement.value = '';\n  }\n\n  private get isUpdate() {\n    return !!this.message;\n  }\n}\n","<div\n  class=\"str-chat__input-flat\"\n  [class.str-chat__input-flat-has-attachments]=\"\n    (attachmentUploads$ | async)!.length > 0\n  \"\n>\n  <div class=\"str-chat__input-flat-wrapper\">\n    <div class=\"str-chat__input-flat--textarea-wrapper\">\n      <stream-attachment-preview-list\n        class=\"rfu-image-previewer-angular-host\"\n      ></stream-attachment-preview-list>\n      <div class=\"rta str-chat__textarea\">\n        <textarea\n          [value]=\"message?.text || ''\"\n          autofocus\n          data-testid=\"textarea\"\n          #input\n          placeholder=\"{{ 'streamChat.Type your message' | translate }}\"\n          class=\"rta__textarea str-chat__textarea__textarea\"\n          rows=\"1\"\n          (keydown.enter)=\"messageSent($event)\"\n        ></textarea>\n      </div>\n      <div\n        *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized\"\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      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"]}
|
|
182
|
+
//# 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  Component,\n  ComponentFactoryResolver,\n  ComponentRef,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  ViewChild,\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { first } from 'rxjs/operators';\nimport { 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  MentionAutcompleteListItemContext,\n  StreamMessage,\n} from '../types';\nimport { MessageInputConfigService } from './message-input-config.service';\nimport { TextareaDirective } from './textarea.directive';\nimport { TextareaInterface } from './textarea.interface';\n\n@Component({\n  selector: 'stream-message-input',\n  templateUrl: './message-input.component.html',\n  styles: [],\n  providers: [AttachmentService],\n})\nexport class MessageInputComponent implements OnInit, OnChanges, OnDestroy {\n  @Input() isFileUploadEnabled: boolean | undefined;\n  @Input() areMentionsEnabled: boolean | undefined;\n  @Input() mentionScope: 'channel' | 'application' | undefined;\n  @Input() mentionAutocompleteItemTemplate:\n    | TemplateRef<MentionAutcompleteListItemContext>\n    | undefined;\n  @Input() acceptedFileTypes: string[] | undefined;\n  @Input() isMultipleFileUploadEnabled: boolean | undefined;\n  @Input() message: StreamMessage | undefined;\n  @Output() readonly messageUpdate = new EventEmitter<void>();\n  isFileUploadAuthorized: boolean | undefined;\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  textareaValue = '';\n  textareaRef: ComponentRef<TextareaInterface> | undefined;\n  mentionedUsers: UserResponse[] = [];\n  @ViewChild('fileInput') private fileInput!: ElementRef<HTMLInputElement>;\n  @ViewChild(TextareaDirective, { static: true })\n  private textareaAnchor!: TextareaDirective;\n  private subscriptions: Subscription[] = [];\n  private hideNotification: Function | undefined;\n\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  ) {\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        }\n      })\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.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  }\n\n  ngOnInit(): void {\n    const componentFactory =\n      this.componentFactoryResolver.resolveComponentFactory(this.textareaType);\n    this.textareaRef =\n      this.textareaAnchor.viewContainerRef.createComponent<any>(\n        componentFactory\n      );\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.mentionScope) {\n      this.configService.mentionScope = this.mentionScope;\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.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          ));\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  }\n\n  get accept() {\n    return this.acceptedFileTypes ? this.acceptedFileTypes?.join(',') : '';\n  }\n\n  async filesSelected(fileList: FileList | null) {\n    await this.attachmentService.filesSelected(fileList);\n    this.clearFileInput();\n  }\n\n  private clearFileInput() {\n    this.fileInput.nativeElement.value = '';\n  }\n\n  private get isUpdate() {\n    return !!this.message;\n  }\n}\n","<div\n  class=\"str-chat__input-flat\"\n  [class.str-chat__input-flat-has-attachments]=\"\n    (attachmentUploads$ | async)!.length > 0\n  \"\n>\n  <div class=\"str-chat__input-flat-wrapper\">\n    <div class=\"str-chat__input-flat--textarea-wrapper\">\n      <stream-attachment-preview-list\n        class=\"rfu-image-previewer-angular-host\"\n      ></stream-attachment-preview-list>\n      <div class=\"rta str-chat__textarea\">\n        <ng-template\n          streamTextarea\n          [(value)]=\"textareaValue\"\n          (send)=\"messageSent()\"\n          [componentRef]=\"textareaRef\"\n          (userMentions)=\"mentionedUsers = $event\"\n          [areMentionsEnabled]=\"areMentionsEnabled\"\n          [mentionAutocompleteItemTemplate]=\"mentionAutocompleteItemTemplate\"\n          [mentionScope]=\"mentionScope\"\n        ></ng-template>\n      </div>\n      <div\n        *ngIf=\"isFileUploadEnabled && isFileUploadAuthorized\"\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      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"]}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Component, EventEmitter, HostBinding, Input, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@ngx-translate/core";
|
|
4
|
+
export class TextareaComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.class = 'str-chat__textarea';
|
|
7
|
+
this.value = '';
|
|
8
|
+
this.valueChange = new EventEmitter();
|
|
9
|
+
this.send = new EventEmitter();
|
|
10
|
+
}
|
|
11
|
+
// eslint-disable-next-line @angular-eslint/no-empty-lifecycle-method
|
|
12
|
+
ngOnChanges() { }
|
|
13
|
+
inputChanged() {
|
|
14
|
+
this.valueChange.emit(this.messageInput.nativeElement.value);
|
|
15
|
+
}
|
|
16
|
+
sent(event) {
|
|
17
|
+
event.preventDefault();
|
|
18
|
+
this.send.next();
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
TextareaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: TextareaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
22
|
+
TextareaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: TextareaComponent, selector: "stream-textarea", inputs: { value: "value" }, outputs: { valueChange: "valueChange", send: "send" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "messageInput", first: true, predicate: ["input"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<textarea\n [value]=\"value || ''\"\n autofocus\n data-testid=\"textarea\"\n #input\n placeholder=\"{{ 'streamChat.Type your message' | translate }}\"\n class=\"rta__textarea str-chat__textarea__textarea\"\n rows=\"1\"\n (input)=\"inputChanged()\"\n (keydown.enter)=\"sent($event)\"\n></textarea>\n", pipes: { "translate": i1.TranslatePipe } });
|
|
23
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: TextareaComponent, decorators: [{
|
|
24
|
+
type: Component,
|
|
25
|
+
args: [{
|
|
26
|
+
selector: 'stream-textarea',
|
|
27
|
+
templateUrl: './textarea.component.html',
|
|
28
|
+
styles: [],
|
|
29
|
+
}]
|
|
30
|
+
}], ctorParameters: function () { return []; }, propDecorators: { class: [{
|
|
31
|
+
type: HostBinding
|
|
32
|
+
}], value: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], valueChange: [{
|
|
35
|
+
type: Output
|
|
36
|
+
}], send: [{
|
|
37
|
+
type: Output
|
|
38
|
+
}], messageInput: [{
|
|
39
|
+
type: ViewChild,
|
|
40
|
+
args: ['input']
|
|
41
|
+
}] } });
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dGFyZWEuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEvdGV4dGFyZWEuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEvdGV4dGFyZWEuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFFVCxZQUFZLEVBQ1osV0FBVyxFQUNYLEtBQUssRUFFTCxNQUFNLEVBQ04sU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDOzs7QUFRdkIsTUFBTSxPQUFPLGlCQUFpQjtJQU81QjtRQU5lLFVBQUssR0FBRyxvQkFBb0IsQ0FBQztRQUNuQyxVQUFLLEdBQUcsRUFBRSxDQUFDO1FBQ0QsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBVSxDQUFDO1FBQ3pDLFNBQUksR0FBRyxJQUFJLFlBQVksRUFBUSxDQUFDO0lBR3BDLENBQUM7SUFFaEIscUVBQXFFO0lBQ3JFLFdBQVcsS0FBVSxDQUFDO0lBRXRCLFlBQVk7UUFDVixJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvRCxDQUFDO0lBRUQsSUFBSSxDQUFDLEtBQVk7UUFDZixLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7UUFDdkIsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNuQixDQUFDOzs4R0FuQlUsaUJBQWlCO2tHQUFqQixpQkFBaUIsdVRDakI5QixxVEFXQTsyRkRNYSxpQkFBaUI7a0JBTDdCLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGlCQUFpQjtvQkFDM0IsV0FBVyxFQUFFLDJCQUEyQjtvQkFDeEMsTUFBTSxFQUFFLEVBQUU7aUJBQ1g7MEVBRWdCLEtBQUs7c0JBQW5CLFdBQVc7Z0JBQ0gsS0FBSztzQkFBYixLQUFLO2dCQUNhLFdBQVc7c0JBQTdCLE1BQU07Z0JBQ1ksSUFBSTtzQkFBdEIsTUFBTTtnQkFDcUIsWUFBWTtzQkFBdkMsU0FBUzt1QkFBQyxPQUFPIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBFbGVtZW50UmVmLFxuICBFdmVudEVtaXR0ZXIsXG4gIEhvc3RCaW5kaW5nLFxuICBJbnB1dCxcbiAgT25DaGFuZ2VzLFxuICBPdXRwdXQsXG4gIFZpZXdDaGlsZCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBUZXh0YXJlYUludGVyZmFjZSB9IGZyb20gJy4uL3RleHRhcmVhLmludGVyZmFjZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3N0cmVhbS10ZXh0YXJlYScsXG4gIHRlbXBsYXRlVXJsOiAnLi90ZXh0YXJlYS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlczogW10sXG59KVxuZXhwb3J0IGNsYXNzIFRleHRhcmVhQ29tcG9uZW50IGltcGxlbWVudHMgVGV4dGFyZWFJbnRlcmZhY2UsIE9uQ2hhbmdlcyB7XG4gIEBIb3N0QmluZGluZygpIGNsYXNzID0gJ3N0ci1jaGF0X190ZXh0YXJlYSc7XG4gIEBJbnB1dCgpIHZhbHVlID0gJyc7XG4gIEBPdXRwdXQoKSByZWFkb25seSB2YWx1ZUNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuICBAT3V0cHV0KCkgcmVhZG9ubHkgc2VuZCA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgQFZpZXdDaGlsZCgnaW5wdXQnKSBwcml2YXRlIG1lc3NhZ2VJbnB1dCE6IEVsZW1lbnRSZWY8SFRNTElucHV0RWxlbWVudD47XG5cbiAgY29uc3RydWN0b3IoKSB7fVxuXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAYW5ndWxhci1lc2xpbnQvbm8tZW1wdHktbGlmZWN5Y2xlLW1ldGhvZFxuICBuZ09uQ2hhbmdlcygpOiB2b2lkIHt9XG5cbiAgaW5wdXRDaGFuZ2VkKCkge1xuICAgIHRoaXMudmFsdWVDaGFuZ2UuZW1pdCh0aGlzLm1lc3NhZ2VJbnB1dC5uYXRpdmVFbGVtZW50LnZhbHVlKTtcbiAgfVxuXG4gIHNlbnQoZXZlbnQ6IEV2ZW50KSB7XG4gICAgZXZlbnQucHJldmVudERlZmF1bHQoKTtcbiAgICB0aGlzLnNlbmQubmV4dCgpO1xuICB9XG59XG4iLCI8dGV4dGFyZWFcbiAgW3ZhbHVlXT1cInZhbHVlIHx8ICcnXCJcbiAgYXV0b2ZvY3VzXG4gIGRhdGEtdGVzdGlkPVwidGV4dGFyZWFcIlxuICAjaW5wdXRcbiAgcGxhY2Vob2xkZXI9XCJ7eyAnc3RyZWFtQ2hhdC5UeXBlIHlvdXIgbWVzc2FnZScgfCB0cmFuc2xhdGUgfX1cIlxuICBjbGFzcz1cInJ0YV9fdGV4dGFyZWEgc3RyLWNoYXRfX3RleHRhcmVhX190ZXh0YXJlYVwiXG4gIHJvd3M9XCIxXCJcbiAgKGlucHV0KT1cImlucHV0Q2hhbmdlZCgpXCJcbiAgKGtleWRvd24uZW50ZXIpPVwic2VudCgkZXZlbnQpXCJcbj48L3RleHRhcmVhPlxuIl19
|