stream-chat-angular 1.3.0 → 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.
Files changed (64) hide show
  1. package/assets/i18n/en.d.ts +1 -0
  2. package/assets/version.d.ts +1 -1
  3. package/bundles/stream-chat-angular.umd.js +1196 -365
  4. package/bundles/stream-chat-angular.umd.js.map +1 -1
  5. package/esm2015/assets/i18n/en.js +2 -1
  6. package/esm2015/assets/version.js +2 -2
  7. package/esm2015/lib/attachment-list/attachment-list.component.js +3 -4
  8. package/esm2015/lib/attachment-preview-list/attachment-preview-list.component.js +38 -0
  9. package/esm2015/lib/attachment.service.js +190 -0
  10. package/esm2015/lib/channel.service.js +53 -4
  11. package/esm2015/lib/chat-client.service.js +16 -1
  12. package/esm2015/lib/injection-tokens.js +3 -0
  13. package/esm2015/lib/is-image-attachment.js +6 -0
  14. package/esm2015/lib/message/highlight-mentions.pipe.js +23 -0
  15. package/esm2015/lib/message/message.component.js +3 -2
  16. package/esm2015/lib/message-actions-box/message-actions-box.component.js +43 -12
  17. package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +145 -0
  18. package/esm2015/lib/message-input/message-input-config.service.js +19 -0
  19. package/esm2015/lib/message-input/message-input.component.js +122 -140
  20. package/esm2015/lib/message-input/textarea/textarea.component.js +42 -0
  21. package/esm2015/lib/message-input/textarea.directive.js +69 -0
  22. package/esm2015/lib/message-input/textarea.interface.js +2 -0
  23. package/esm2015/lib/message-list/message-list.component.js +19 -3
  24. package/esm2015/lib/message-preview.js +3 -2
  25. package/esm2015/lib/modal/modal.component.js +63 -0
  26. package/esm2015/lib/notification-list/notification-list.component.js +1 -1
  27. package/esm2015/lib/notification.service.js +1 -1
  28. package/esm2015/lib/stream-autocomplete-textarea.module.js +33 -0
  29. package/esm2015/lib/stream-avatar.module.js +19 -0
  30. package/esm2015/lib/stream-chat.module.js +25 -11
  31. package/esm2015/lib/stream-textarea.module.js +31 -0
  32. package/esm2015/lib/types.js +1 -1
  33. package/esm2015/public-api.js +19 -1
  34. package/fesm2015/stream-chat-angular.js +955 -241
  35. package/fesm2015/stream-chat-angular.js.map +1 -1
  36. package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +14 -0
  37. package/lib/attachment.service.d.ts +25 -0
  38. package/lib/channel.service.d.ts +6 -2
  39. package/lib/chat-client.service.d.ts +1 -0
  40. package/lib/injection-tokens.d.ts +3 -0
  41. package/lib/is-image-attachment.d.ts +2 -0
  42. package/lib/message/highlight-mentions.pipe.d.ts +8 -0
  43. package/lib/message/message.component.d.ts +1 -0
  44. package/lib/message-actions-box/message-actions-box.component.d.ts +11 -4
  45. package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +42 -0
  46. package/lib/message-input/message-input-config.service.d.ts +14 -0
  47. package/lib/message-input/message-input.component.d.ts +30 -20
  48. package/lib/message-input/textarea/textarea.component.d.ts +16 -0
  49. package/lib/message-input/textarea.directive.d.ts +21 -0
  50. package/lib/message-input/textarea.interface.d.ts +12 -0
  51. package/lib/message-list/message-list.component.d.ts +1 -1
  52. package/lib/message-preview.d.ts +1 -1
  53. package/lib/modal/modal.component.d.ts +16 -0
  54. package/lib/notification-list/notification-list.component.d.ts +3 -3
  55. package/lib/notification.service.d.ts +2 -2
  56. package/lib/stream-autocomplete-textarea.module.d.ts +11 -0
  57. package/lib/stream-avatar.module.d.ts +9 -0
  58. package/lib/stream-chat.module.d.ts +23 -19
  59. package/lib/stream-textarea.module.d.ts +9 -0
  60. package/lib/types.d.ts +7 -1
  61. package/package.json +3 -2
  62. package/public-api.d.ts +18 -0
  63. package/src/assets/i18n/en.ts +1 -0
  64. package/src/assets/version.ts +1 -1
@@ -1,68 +1,129 @@
1
1
  import { __awaiter } from "tslib";
2
- import { Component, Input, ViewChild, } from '@angular/core';
2
+ import { Component, EventEmitter, Inject, Input, Output, ViewChild, } from '@angular/core';
3
+ import { first } from 'rxjs/operators';
4
+ import { AttachmentService } from '../attachment.service';
5
+ import { textareaInjectionToken } from '../injection-tokens';
6
+ import { TextareaDirective } from './textarea.directive';
3
7
  import * as i0 from "@angular/core";
4
8
  import * as i1 from "../channel.service";
5
9
  import * as i2 from "../notification.service";
6
- import * as i3 from "../icon/icon.component";
7
- import * as i4 from "../loading-indicator/loading-indicator.component";
8
- import * as i5 from "@angular/common";
9
- import * as i6 from "@ngx-translate/core";
10
+ import * as i3 from "../attachment.service";
11
+ import * as i4 from "./message-input-config.service";
12
+ import * as i5 from "../attachment-preview-list/attachment-preview-list.component";
13
+ import * as i6 from "../icon/icon.component";
14
+ import * as i7 from "./textarea.directive";
15
+ import * as i8 from "@angular/common";
16
+ import * as i9 from "@ngx-translate/core";
10
17
  export class MessageInputComponent {
11
- constructor(channelService, notificationService) {
18
+ constructor(channelService, notificationService, attachmentService, configService, textareaType, componentFactoryResolver) {
12
19
  this.channelService = channelService;
13
20
  this.notificationService = notificationService;
14
- this.isFileUploadEnabled = true;
15
- this.isMultipleFileUploadEnabled = true;
16
- this.attachmentUploads = [];
17
- this.attachmentUploadInProgressCounter = 0;
21
+ this.attachmentService = attachmentService;
22
+ this.configService = configService;
23
+ this.textareaType = textareaType;
24
+ this.componentFactoryResolver = componentFactoryResolver;
25
+ this.messageUpdate = new EventEmitter();
26
+ this.textareaValue = '';
27
+ this.mentionedUsers = [];
18
28
  this.subscriptions = [];
19
29
  this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
20
30
  var _a;
21
- if (this.messageInput) {
22
- this.messageInput.nativeElement.value = '';
23
- }
24
- this.attachmentUploads = [];
31
+ this.textareaValue = '';
32
+ this.attachmentService.resetAttachmentUploads();
25
33
  const capabilities = (_a = channel === null || channel === void 0 ? void 0 : channel.data) === null || _a === void 0 ? void 0 : _a.own_capabilities;
26
34
  if (capabilities) {
27
35
  this.isFileUploadAuthorized =
28
36
  capabilities.indexOf('upload-file') !== -1;
29
37
  }
30
38
  }));
39
+ this.subscriptions.push(this.attachmentService.attachmentUploadInProgressCounter$.subscribe((counter) => {
40
+ if (counter === 0 && this.hideNotification) {
41
+ this.hideNotification();
42
+ this.hideNotification = undefined;
43
+ }
44
+ }));
45
+ this.attachmentUploads$ = this.attachmentService.attachmentUploads$;
46
+ this.isFileUploadEnabled = this.configService.isFileUploadEnabled;
47
+ this.acceptedFileTypes = this.configService.acceptedFileTypes;
48
+ this.isMultipleFileUploadEnabled =
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);
59
+ }
60
+ ngOnChanges(changes) {
61
+ if (changes.message) {
62
+ this.attachmentService.resetAttachmentUploads();
63
+ if (this.isUpdate) {
64
+ this.attachmentService.createFromAttachments(this.message.attachments || []);
65
+ this.textareaValue = this.message.text || '';
66
+ }
67
+ }
68
+ if (changes.isFileUploadEnabled) {
69
+ this.configService.isFileUploadEnabled = this.isFileUploadEnabled;
70
+ }
71
+ if (changes.acceptedFileTypes) {
72
+ this.configService.acceptedFileTypes = this.acceptedFileTypes;
73
+ }
74
+ if (changes.isMultipleFileUploadEnabled) {
75
+ this.configService.isMultipleFileUploadEnabled =
76
+ this.isMultipleFileUploadEnabled;
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
+ }
31
88
  }
32
89
  ngOnDestroy() {
33
90
  this.subscriptions.forEach((s) => s.unsubscribe());
34
91
  }
35
- messageSent(event) {
92
+ messageSent() {
36
93
  return __awaiter(this, void 0, void 0, function* () {
37
- event === null || event === void 0 ? void 0 : event.preventDefault();
38
- if (this.attachmentUploadInProgressCounter > 0) {
94
+ let attachmentUploadInProgressCounter;
95
+ this.attachmentService.attachmentUploadInProgressCounter$
96
+ .pipe(first())
97
+ .subscribe((counter) => (attachmentUploadInProgressCounter = counter));
98
+ if (attachmentUploadInProgressCounter > 0) {
39
99
  if (!this.hideNotification) {
40
100
  this.hideNotification =
41
101
  this.notificationService.addPermanentNotification('streamChat.Wait until all attachments have uploaded');
42
102
  }
43
103
  return;
44
104
  }
45
- const text = this.messageInput.nativeElement.value;
46
- const attachments = this.attachmentUploads
47
- .filter((r) => r.state === 'success')
48
- .map((r) => {
49
- const attachment = {
50
- type: r.type,
51
- };
52
- if (r.type === 'image') {
53
- attachment.fallback = r.file.name;
54
- attachment.image_url = r.url;
105
+ const attachments = this.attachmentService.mapToAttachments();
106
+ const text = this.textareaValue;
107
+ if (!text && (!attachments || attachments.length === 0)) {
108
+ return;
109
+ }
110
+ if (!this.isUpdate) {
111
+ this.textareaValue = '';
112
+ }
113
+ try {
114
+ yield (this.isUpdate
115
+ ? this.channelService.updateMessage(Object.assign(Object.assign({}, this.message), { text: text, attachments: attachments }))
116
+ : this.channelService.sendMessage(text, attachments, this.mentionedUsers));
117
+ this.messageUpdate.emit();
118
+ if (!this.isUpdate) {
119
+ this.attachmentService.resetAttachmentUploads();
55
120
  }
56
- else {
57
- attachment.asset_url = r.url;
58
- attachment.title = r.file.name;
59
- attachment.file_size = r.file.size;
121
+ }
122
+ catch (error) {
123
+ if (this.isUpdate) {
124
+ this.notificationService.addTemporaryNotification('streamChat.Edit message request failed');
60
125
  }
61
- return attachment;
62
- });
63
- this.messageInput.nativeElement.value = '';
64
- yield this.channelService.sendMessage(text, attachments);
65
- this.attachmentUploads = [];
126
+ }
66
127
  });
67
128
  }
68
129
  get accept() {
@@ -71,130 +132,51 @@ export class MessageInputComponent {
71
132
  }
72
133
  filesSelected(fileList) {
73
134
  return __awaiter(this, void 0, void 0, function* () {
74
- if (!fileList) {
75
- return;
76
- }
77
- const imageFiles = [];
78
- const dataFiles = [];
79
- Array.from(fileList).forEach((file) => {
80
- if (file.type.startsWith('image/') && !file.type.endsWith('.photoshop')) {
81
- // photoshop files begin with 'image/'
82
- imageFiles.push(file);
83
- }
84
- else {
85
- dataFiles.push(file);
86
- }
87
- });
88
- imageFiles.forEach((f) => this.createPreview(f));
89
- const newUploads = [
90
- ...imageFiles.map((file) => ({
91
- file,
92
- state: 'uploading',
93
- type: 'image',
94
- })),
95
- ...dataFiles.map((file) => ({
96
- file,
97
- state: 'uploading',
98
- type: 'file',
99
- })),
100
- ];
101
- this.attachmentUploads = [...this.attachmentUploads, ...newUploads];
135
+ yield this.attachmentService.filesSelected(fileList);
102
136
  this.clearFileInput();
103
- yield this.uploadAttachments(newUploads);
104
- });
105
- }
106
- retryAttachmentUpload(file) {
107
- return __awaiter(this, void 0, void 0, function* () {
108
- const upload = this.attachmentUploads.find((u) => u.file === file);
109
- if (!upload) {
110
- return;
111
- }
112
- upload.state = 'uploading';
113
- yield this.uploadAttachments([upload]);
114
- });
115
- }
116
- deleteAttachment(upload) {
117
- return __awaiter(this, void 0, void 0, function* () {
118
- if (upload.state === 'success') {
119
- try {
120
- yield this.channelService.deleteAttachment(upload);
121
- this.attachmentUploads.splice(this.attachmentUploads.indexOf(upload), 1);
122
- }
123
- catch (error) {
124
- // TODO error handling
125
- }
126
- }
127
- else {
128
- this.attachmentUploads.splice(this.attachmentUploads.indexOf(upload), 1);
129
- }
130
- });
131
- }
132
- trackByFile(_, item) {
133
- return item.file;
134
- }
135
- createPreview(file) {
136
- const reader = new FileReader();
137
- reader.onload = (event) => {
138
- var _a;
139
- const upload = this.attachmentUploads.find((upload) => upload.file === file);
140
- if (!upload) {
141
- return;
142
- }
143
- upload.previewUri = ((_a = event.target) === null || _a === void 0 ? void 0 : _a.result) || undefined;
144
- };
145
- reader.readAsDataURL(file);
146
- }
147
- uploadAttachments(uploads) {
148
- return __awaiter(this, void 0, void 0, function* () {
149
- this.attachmentUploadInProgressCounter++;
150
- const result = yield this.channelService.uploadAttachments(uploads);
151
- result.forEach((r) => {
152
- const upload = this.attachmentUploads.find((upload) => upload.file === r.file);
153
- if (!upload) {
154
- if (r.url) {
155
- void this.channelService.deleteAttachment(r);
156
- }
157
- return;
158
- }
159
- upload.state = r.state;
160
- upload.url = r.url;
161
- if (upload.state === 'error') {
162
- this.notificationService.addTemporaryNotification(upload.type === 'image'
163
- ? 'streamChat.Error uploading image'
164
- : 'streamChat.Error uploading file');
165
- }
166
- });
167
- this.attachmentUploadInProgressCounter--;
168
- if (this.attachmentUploadInProgressCounter === 0 && this.hideNotification) {
169
- this.hideNotification();
170
- this.hideNotification = undefined;
171
- }
172
137
  });
173
138
  }
174
139
  clearFileInput() {
175
140
  this.fileInput.nativeElement.value = '';
176
141
  }
142
+ get isUpdate() {
143
+ return !!this.message;
144
+ }
177
145
  }
178
- MessageInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, deps: [{ token: i1.ChannelService }, { token: i2.NotificationService }], target: i0.ɵɵFactoryTarget.Component });
179
- 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" }, viewQueries: [{ propertyName: "messageInput", first: true, predicate: ["input"], descendants: true }, { propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], ngImport: i0, template: "<div class=\"str-chat__input-flat\">\n <div class=\"str-chat__input-flat-wrapper\">\n <div class=\"str-chat__input-flat--textarea-wrapper\">\n <div class=\"rfu-image-previewer\">\n <ng-container\n *ngFor=\"\n let attachmentUpload of attachmentUploads;\n trackBy: trackByFile\n \"\n >\n <div\n *ngIf=\"attachmentUpload.type === 'image'\"\n class=\"rfu-image-previewer__image\"\n [class.rfu-image-previewer__image--loaded]=\"\n attachmentUpload.state === 'success'\n \"\n data-testclass=\"attachment-image-preview\"\n >\n <div\n *ngIf=\"attachmentUpload.state === 'error'\"\n class=\"rfu-image-previewer__retry\"\n (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n (keyup.enter)=\"retryAttachmentUpload(attachmentUpload.file)\"\n data-testclass=\"upload-error\"\n >\n <stream-icon icon=\"retry\"></stream-icon>\n </div>\n <div class=\"rfu-thumbnail__wrapper\" style=\"width: 100; height: 100\">\n <div class=\"rfu-thumbnail__overlay\">\n <div\n class=\"rfu-icon-button\"\n data-testclass=\"delete-attachment\"\n role=\"button\"\n (click)=\"deleteAttachment(attachmentUpload)\"\n (keyup.enter)=\"deleteAttachment(attachmentUpload)\"\n >\n <stream-icon icon=\"close\"></stream-icon>\n </div>\n </div>\n <img\n *ngIf=\"attachmentUpload.url || attachmentUpload.previewUri\"\n src=\"{{\n attachmentUpload.url\n ? attachmentUpload.url\n : attachmentUpload.previewUri\n }}\"\n alt=\"attachmentUpload.file.name\"\n class=\"rfu-thumbnail__image\"\n data-testclass=\"attachment-image\"\n />\n </div>\n <stream-loading-indicator\n data-testclass=\"loading-indicator\"\n color=\"rgba(255,255,255,0.7)\"\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n ></stream-loading-indicator>\n </div>\n <div\n class=\"rfu-file-previewer\"\n *ngIf=\"attachmentUpload.type === 'file'\"\n data-testclass=\"attachment-file-preview\"\n >\n <ol>\n <li\n class=\"rfu-file-previewer__file\"\n [class.rfu-file-previewer__file--uploading]=\"\n attachmentUpload.state === 'uploading'\n \"\n [class.rfu-file-previewer__file--failed]=\"\n attachmentUpload.state === 'error'\n \"\n >\n <stream-icon icon=\"file\"></stream-icon>\n\n <a\n data-testclass=\"file-download-link\"\n href=\"{{ attachmentUpload.url }}\"\n (click)=\"\n attachmentUpload.url ? null : $event.preventDefault()\n \"\n (keyup.enter)=\"\n attachmentUpload.url ? null : $event.preventDefault()\n \"\n download\n >\n {{ attachmentUpload.file.name }}\n <ng-container *ngIf=\"attachmentUpload.state === 'error'\">\n <div\n data-testclass=\"file-upload-retry\"\n class=\"rfu-file-previewer__failed\"\n (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n (keyup.enter)=\"\n retryAttachmentUpload(attachmentUpload.file)\n \"\n translate\n >\n streamChat.failed\n </div>\n <div\n class=\"rfu-file-previewer__retry\"\n (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n (keyup.enter)=\"\n retryAttachmentUpload(attachmentUpload.file)\n \"\n translate\n >\n streamChat.retry\n </div>\n </ng-container>\n </a>\n\n <span\n data-testclass=\"file-delete\"\n class=\"rfu-file-previewer__close-button\"\n (click)=\"deleteAttachment(attachmentUpload)\"\n (keyup.enter)=\"deleteAttachment(attachmentUpload)\"\n >\n \u2718\n </span>\n <div\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n class=\"rfu-file-previewer__loading-indicator\"\n >\n <stream-loading-indicator></stream-loading-indicator>\n </div>\n </li>\n </ol>\n </div>\n </ng-container>\n </div>\n <div class=\"rta str-chat__textarea\">\n <textarea\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", components: [{ type: i3.IconComponent, selector: "stream-icon", inputs: ["icon", "size"] }, { type: i4.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }], directives: [{ type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }], pipes: { "translate": i6.TranslatePipe } });
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 } });
180
148
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageInputComponent, decorators: [{
181
149
  type: Component,
182
150
  args: [{
183
151
  selector: 'stream-message-input',
184
152
  templateUrl: './message-input.component.html',
185
153
  styles: [],
154
+ providers: [AttachmentService],
186
155
  }]
187
- }], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.NotificationService }]; }, propDecorators: { isFileUploadEnabled: [{
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: [{
188
166
  type: Input
189
167
  }], acceptedFileTypes: [{
190
168
  type: Input
191
169
  }], isMultipleFileUploadEnabled: [{
192
170
  type: Input
193
- }], messageInput: [{
194
- type: ViewChild,
195
- args: ['input']
171
+ }], message: [{
172
+ type: Input
173
+ }], messageUpdate: [{
174
+ type: Output
196
175
  }], fileInput: [{
197
176
  type: ViewChild,
198
177
  args: ['fileInput']
178
+ }], textareaAnchor: [{
179
+ type: ViewChild,
180
+ args: [TextareaDirective, { static: true }]
199
181
  }] } });
200
- //# 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  Input,\n  OnDestroy,\n  ViewChild,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { Attachment } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { NotificationService } from '../notification.service';\nimport { AttachmentUpload } from '../types';\n\n@Component({\n  selector: 'stream-message-input',\n  templateUrl: './message-input.component.html',\n  styles: [],\n})\nexport class MessageInputComponent implements OnDestroy {\n  @Input() isFileUploadEnabled = true;\n  @Input() acceptedFileTypes: string[] | undefined;\n  @Input() isMultipleFileUploadEnabled = true;\n  attachmentUploads: AttachmentUpload[] = [];\n  isFileUploadAuthorized: boolean | undefined;\n  private attachmentUploadInProgressCounter = 0;\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  ) {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((channel) => {\n        if (this.messageInput) {\n          this.messageInput.nativeElement.value = '';\n        }\n        this.attachmentUploads = [];\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  }\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async messageSent(event?: Event) {\n    event?.preventDefault();\n    if (this.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 text = this.messageInput.nativeElement.value;\n    const attachments = this.attachmentUploads\n      .filter((r) => r.state === 'success')\n      .map((r) => {\n        const attachment: Attachment = {\n          type: r.type,\n        };\n        if (r.type === 'image') {\n          attachment.fallback = r.file.name;\n          attachment.image_url = r.url;\n        } else {\n          attachment.asset_url = r.url;\n          attachment.title = r.file.name;\n          attachment.file_size = r.file.size;\n        }\n\n        return attachment;\n      });\n    this.messageInput.nativeElement.value = '';\n    await this.channelService.sendMessage(text, attachments);\n    this.attachmentUploads = [];\n  }\n\n  get accept() {\n    return this.acceptedFileTypes ? this.acceptedFileTypes?.join(',') : '';\n  }\n\n  async filesSelected(fileList: FileList | null) {\n    if (!fileList) {\n      return;\n    }\n    const imageFiles: File[] = [];\n    const dataFiles: File[] = [];\n\n    Array.from(fileList).forEach((file) => {\n      if (file.type.startsWith('image/') && !file.type.endsWith('.photoshop')) {\n        // photoshop files begin with 'image/'\n        imageFiles.push(file);\n      } else {\n        dataFiles.push(file);\n      }\n    });\n    imageFiles.forEach((f) => this.createPreview(f));\n    const newUploads = [\n      ...imageFiles.map((file) => ({\n        file,\n        state: 'uploading' as 'uploading',\n        type: 'image' as 'image',\n      })),\n      ...dataFiles.map((file) => ({\n        file,\n        state: 'uploading' as 'uploading',\n        type: 'file' as 'file',\n      })),\n    ];\n    this.attachmentUploads = [...this.attachmentUploads, ...newUploads];\n    this.clearFileInput();\n    await this.uploadAttachments(newUploads);\n  }\n\n  async retryAttachmentUpload(file: File) {\n    const upload = this.attachmentUploads.find((u) => u.file === file);\n    if (!upload) {\n      return;\n    }\n    upload.state = 'uploading';\n    await this.uploadAttachments([upload]);\n  }\n\n  async deleteAttachment(upload: AttachmentUpload) {\n    if (upload.state === 'success') {\n      try {\n        await this.channelService.deleteAttachment(upload);\n        this.attachmentUploads.splice(\n          this.attachmentUploads.indexOf(upload),\n          1\n        );\n      } catch (error) {\n        // TODO error handling\n      }\n    } else {\n      this.attachmentUploads.splice(this.attachmentUploads.indexOf(upload), 1);\n    }\n  }\n\n  trackByFile(\n    _: number,\n    item: {\n      file: File;\n      state: 'error' | 'success' | 'uploading';\n      url?: string;\n    }\n  ) {\n    return item.file;\n  }\n\n  private createPreview(file: File) {\n    const reader = new FileReader();\n    reader.onload = (event) => {\n      const upload = this.attachmentUploads.find(\n        (upload) => upload.file === file\n      );\n      if (!upload) {\n        return;\n      }\n      upload.previewUri = event.target?.result || undefined;\n    };\n    reader.readAsDataURL(file as Blob);\n  }\n\n  private async uploadAttachments(uploads: AttachmentUpload[]) {\n    this.attachmentUploadInProgressCounter++;\n    const result = await this.channelService.uploadAttachments(uploads);\n    result.forEach((r) => {\n      const upload = this.attachmentUploads.find(\n        (upload) => upload.file === r.file\n      );\n      if (!upload) {\n        if (r.url) {\n          void this.channelService.deleteAttachment(r);\n        }\n        return;\n      }\n      upload.state = r.state;\n      upload.url = r.url;\n      if (upload.state === 'error') {\n        this.notificationService.addTemporaryNotification(\n          upload.type === 'image'\n            ? 'streamChat.Error uploading image'\n            : 'streamChat.Error uploading file'\n        );\n      }\n    });\n    this.attachmentUploadInProgressCounter--;\n    if (this.attachmentUploadInProgressCounter === 0 && this.hideNotification) {\n      this.hideNotification();\n      this.hideNotification = undefined;\n    }\n  }\n\n  private clearFileInput() {\n    this.fileInput.nativeElement.value = '';\n  }\n}\n","<div class=\"str-chat__input-flat\">\n  <div class=\"str-chat__input-flat-wrapper\">\n    <div class=\"str-chat__input-flat--textarea-wrapper\">\n      <div class=\"rfu-image-previewer\">\n        <ng-container\n          *ngFor=\"\n            let attachmentUpload of attachmentUploads;\n            trackBy: trackByFile\n          \"\n        >\n          <div\n            *ngIf=\"attachmentUpload.type === 'image'\"\n            class=\"rfu-image-previewer__image\"\n            [class.rfu-image-previewer__image--loaded]=\"\n              attachmentUpload.state === 'success'\n            \"\n            data-testclass=\"attachment-image-preview\"\n          >\n            <div\n              *ngIf=\"attachmentUpload.state === 'error'\"\n              class=\"rfu-image-previewer__retry\"\n              (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n              (keyup.enter)=\"retryAttachmentUpload(attachmentUpload.file)\"\n              data-testclass=\"upload-error\"\n            >\n              <stream-icon icon=\"retry\"></stream-icon>\n            </div>\n            <div class=\"rfu-thumbnail__wrapper\" style=\"width: 100; height: 100\">\n              <div class=\"rfu-thumbnail__overlay\">\n                <div\n                  class=\"rfu-icon-button\"\n                  data-testclass=\"delete-attachment\"\n                  role=\"button\"\n                  (click)=\"deleteAttachment(attachmentUpload)\"\n                  (keyup.enter)=\"deleteAttachment(attachmentUpload)\"\n                >\n                  <stream-icon icon=\"close\"></stream-icon>\n                </div>\n              </div>\n              <img\n                *ngIf=\"attachmentUpload.url || attachmentUpload.previewUri\"\n                src=\"{{\n                  attachmentUpload.url\n                    ? attachmentUpload.url\n                    : attachmentUpload.previewUri\n                }}\"\n                alt=\"attachmentUpload.file.name\"\n                class=\"rfu-thumbnail__image\"\n                data-testclass=\"attachment-image\"\n              />\n            </div>\n            <stream-loading-indicator\n              data-testclass=\"loading-indicator\"\n              color=\"rgba(255,255,255,0.7)\"\n              *ngIf=\"attachmentUpload.state === 'uploading'\"\n            ></stream-loading-indicator>\n          </div>\n          <div\n            class=\"rfu-file-previewer\"\n            *ngIf=\"attachmentUpload.type === 'file'\"\n            data-testclass=\"attachment-file-preview\"\n          >\n            <ol>\n              <li\n                class=\"rfu-file-previewer__file\"\n                [class.rfu-file-previewer__file--uploading]=\"\n                  attachmentUpload.state === 'uploading'\n                \"\n                [class.rfu-file-previewer__file--failed]=\"\n                  attachmentUpload.state === 'error'\n                \"\n              >\n                <stream-icon icon=\"file\"></stream-icon>\n\n                <a\n                  data-testclass=\"file-download-link\"\n                  href=\"{{ attachmentUpload.url }}\"\n                  (click)=\"\n                    attachmentUpload.url ? null : $event.preventDefault()\n                  \"\n                  (keyup.enter)=\"\n                    attachmentUpload.url ? null : $event.preventDefault()\n                  \"\n                  download\n                >\n                  {{ attachmentUpload.file.name }}\n                  <ng-container *ngIf=\"attachmentUpload.state === 'error'\">\n                    <div\n                      data-testclass=\"file-upload-retry\"\n                      class=\"rfu-file-previewer__failed\"\n                      (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n                      (keyup.enter)=\"\n                        retryAttachmentUpload(attachmentUpload.file)\n                      \"\n                      translate\n                    >\n                      streamChat.failed\n                    </div>\n                    <div\n                      class=\"rfu-file-previewer__retry\"\n                      (click)=\"retryAttachmentUpload(attachmentUpload.file)\"\n                      (keyup.enter)=\"\n                        retryAttachmentUpload(attachmentUpload.file)\n                      \"\n                      translate\n                    >\n                      streamChat.retry\n                    </div>\n                  </ng-container>\n                </a>\n\n                <span\n                  data-testclass=\"file-delete\"\n                  class=\"rfu-file-previewer__close-button\"\n                  (click)=\"deleteAttachment(attachmentUpload)\"\n                  (keyup.enter)=\"deleteAttachment(attachmentUpload)\"\n                >\n                  ✘\n                </span>\n                <div\n                  *ngIf=\"attachmentUpload.state === 'uploading'\"\n                  class=\"rfu-file-previewer__loading-indicator\"\n                >\n                  <stream-loading-indicator></stream-loading-indicator>\n                </div>\n              </li>\n            </ol>\n          </div>\n        </ng-container>\n      </div>\n      <div class=\"rta str-chat__textarea\">\n        <textarea\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
@@ -0,0 +1,69 @@
1
+ import { EventEmitter, Input, Output, } from '@angular/core';
2
+ import { Directive } from '@angular/core';
3
+ import * as i0 from "@angular/core";
4
+ export class TextareaDirective {
5
+ constructor(viewContainerRef) {
6
+ this.viewContainerRef = viewContainerRef;
7
+ this.value = '';
8
+ this.valueChange = new EventEmitter();
9
+ this.send = new EventEmitter();
10
+ this.userMentions = new EventEmitter();
11
+ this.subscriptions = [];
12
+ }
13
+ ngOnChanges(changes) {
14
+ if (!this.componentRef) {
15
+ return;
16
+ }
17
+ if (changes.componentRef) {
18
+ this.subscriptions.forEach((s) => s.unsubscribe());
19
+ if (this.componentRef) {
20
+ this.subscriptions.push(this.componentRef.instance.valueChange.subscribe((value) => this.valueChange.next(value)));
21
+ this.subscriptions.push(this.componentRef.instance.send.subscribe((value) => this.send.next(value)));
22
+ if (this.componentRef.instance.userMentions) {
23
+ this.subscriptions.push(this.componentRef.instance.userMentions.subscribe((value) => this.userMentions.next(value)));
24
+ }
25
+ }
26
+ }
27
+ if (changes.areMentionsEnabled) {
28
+ this.componentRef.instance.areMentionsEnabled = this.areMentionsEnabled;
29
+ }
30
+ if (changes.mentionAutocompleteItemTemplate) {
31
+ this.componentRef.instance.mentionAutocompleteItemTemplate =
32
+ this.mentionAutocompleteItemTemplate;
33
+ }
34
+ if (changes.mentionScope) {
35
+ this.componentRef.instance.mentionScope = this.mentionScope;
36
+ }
37
+ if (changes.value) {
38
+ this.componentRef.instance.value = this.value;
39
+ }
40
+ // ngOnChanges not called for dynamic components since we don't use template binding
41
+ // eslint-disable-next-line @angular-eslint/no-lifecycle-call
42
+ this.componentRef.instance.ngOnChanges(changes);
43
+ }
44
+ }
45
+ TextareaDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: TextareaDirective, deps: [{ token: i0.ViewContainerRef }], target: i0.ɵɵFactoryTarget.Directive });
46
+ TextareaDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "12.2.5", type: TextareaDirective, selector: "[streamTextarea]", inputs: { componentRef: "componentRef", areMentionsEnabled: "areMentionsEnabled", mentionAutocompleteItemTemplate: "mentionAutocompleteItemTemplate", mentionScope: "mentionScope", value: "value" }, outputs: { valueChange: "valueChange", send: "send", userMentions: "userMentions" }, usesOnChanges: true, ngImport: i0 });
47
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: TextareaDirective, decorators: [{
48
+ type: Directive,
49
+ args: [{
50
+ selector: '[streamTextarea]',
51
+ }]
52
+ }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }]; }, propDecorators: { componentRef: [{
53
+ type: Input
54
+ }], areMentionsEnabled: [{
55
+ type: Input
56
+ }], mentionAutocompleteItemTemplate: [{
57
+ type: Input
58
+ }], mentionScope: [{
59
+ type: Input
60
+ }], value: [{
61
+ type: Input
62
+ }], valueChange: [{
63
+ type: Output
64
+ }], send: [{
65
+ type: Output
66
+ }], userMentions: [{
67
+ type: Output
68
+ }] } });
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dGFyZWEuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFFTCxZQUFZLEVBQ1osS0FBSyxFQUVMLE1BQU0sR0FJUCxNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQVMxQyxNQUFNLE9BQU8saUJBQWlCO0lBYTVCLFlBQW1CLGdCQUFrQztRQUFsQyxxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQWtCO1FBTjVDLFVBQUssR0FBRyxFQUFFLENBQUM7UUFDRCxnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFDekMsU0FBSSxHQUFHLElBQUksWUFBWSxFQUFRLENBQUM7UUFDaEMsaUJBQVksR0FBRyxJQUFJLFlBQVksRUFBa0IsQ0FBQztRQUM3RCxrQkFBYSxHQUFtQixFQUFFLENBQUM7SUFFYSxDQUFDO0lBRXpELFdBQVcsQ0FBQyxPQUFzQjtRQUNoQyxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRTtZQUN0QixPQUFPO1NBQ1I7UUFDRCxJQUFJLE9BQU8sQ0FBQyxZQUFZLEVBQUU7WUFDeEIsSUFBSSxDQUFDLGFBQWEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDO1lBQ25ELElBQUksSUFBSSxDQUFDLFlBQVksRUFBRTtnQkFDckIsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQ3JCLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUN6RCxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FDN0IsQ0FDRixDQUFDO2dCQUNGLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUNyQixJQUFJLENBQUMsWUFBWSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FDbEQsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQ3RCLENBQ0YsQ0FBQztnQkFDRixJQUFJLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLFlBQVksRUFBRTtvQkFDM0MsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQ3JCLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUMxRCxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FDOUIsQ0FDRixDQUFDO2lCQUNIO2FBQ0Y7U0FDRjtRQUNELElBQUksT0FBTyxDQUFDLGtCQUFrQixFQUFFO1lBQzlCLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLGtCQUFrQixHQUFHLElBQUksQ0FBQyxrQkFBa0IsQ0FBQztTQUN6RTtRQUNELElBQUksT0FBTyxDQUFDLCtCQUErQixFQUFFO1lBQzNDLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLCtCQUErQjtnQkFDeEQsSUFBSSxDQUFDLCtCQUErQixDQUFDO1NBQ3hDO1FBQ0QsSUFBSSxPQUFPLENBQUMsWUFBWSxFQUFFO1lBQ3hCLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLFlBQVksR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDO1NBQzdEO1FBQ0QsSUFBSSxPQUFPLENBQUMsS0FBSyxFQUFFO1lBQ2pCLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDO1NBQy9DO1FBQ0Qsb0ZBQW9GO1FBQ3BGLDZEQUE2RDtRQUM3RCxJQUFJLENBQUMsWUFBWSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDbEQsQ0FBQzs7OEdBekRVLGlCQUFpQjtrR0FBakIsaUJBQWlCOzJGQUFqQixpQkFBaUI7a0JBSDdCLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGtCQUFrQjtpQkFDN0I7dUdBRVUsWUFBWTtzQkFBcEIsS0FBSztnQkFDRyxrQkFBa0I7c0JBQTFCLEtBQUs7Z0JBQ0csK0JBQStCO3NCQUF2QyxLQUFLO2dCQUdHLFlBQVk7c0JBQXBCLEtBQUs7Z0JBQ0csS0FBSztzQkFBYixLQUFLO2dCQUNhLFdBQVc7c0JBQTdCLE1BQU07Z0JBQ1ksSUFBSTtzQkFBdEIsTUFBTTtnQkFDWSxZQUFZO3NCQUE5QixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50UmVmLFxuICBFdmVudEVtaXR0ZXIsXG4gIElucHV0LFxuICBPbkNoYW5nZXMsXG4gIE91dHB1dCxcbiAgU2ltcGxlQ2hhbmdlcyxcbiAgVGVtcGxhdGVSZWYsXG4gIFZpZXdDb250YWluZXJSZWYsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTdWJzY3JpcHRpb24gfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IFVzZXJSZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtQ29udGV4dCB9IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCB7IFRleHRhcmVhSW50ZXJmYWNlIH0gZnJvbSAnLi90ZXh0YXJlYS5pbnRlcmZhY2UnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbc3RyZWFtVGV4dGFyZWFdJyxcbn0pXG5leHBvcnQgY2xhc3MgVGV4dGFyZWFEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkNoYW5nZXMge1xuICBASW5wdXQoKSBjb21wb25lbnRSZWY6IENvbXBvbmVudFJlZjxUZXh0YXJlYUludGVyZmFjZT4gfCB1bmRlZmluZWQ7XG4gIEBJbnB1dCgpIGFyZU1lbnRpb25zRW5hYmxlZDogYm9vbGVhbiB8IHVuZGVmaW5lZDtcbiAgQElucHV0KCkgbWVudGlvbkF1dG9jb21wbGV0ZUl0ZW1UZW1wbGF0ZTpcbiAgICB8IFRlbXBsYXRlUmVmPE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtQ29udGV4dD5cbiAgICB8IHVuZGVmaW5lZDtcbiAgQElucHV0KCkgbWVudGlvblNjb3BlPzogJ2NoYW5uZWwnIHwgJ2FwcGxpY2F0aW9uJztcbiAgQElucHV0KCkgdmFsdWUgPSAnJztcbiAgQE91dHB1dCgpIHJlYWRvbmx5IHZhbHVlQ2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxzdHJpbmc+KCk7XG4gIEBPdXRwdXQoKSByZWFkb25seSBzZW5kID0gbmV3IEV2ZW50RW1pdHRlcjx2b2lkPigpO1xuICBAT3V0cHV0KCkgcmVhZG9ubHkgdXNlck1lbnRpb25zID0gbmV3IEV2ZW50RW1pdHRlcjxVc2VyUmVzcG9uc2VbXT4oKTtcbiAgcHJpdmF0ZSBzdWJzY3JpcHRpb25zOiBTdWJzY3JpcHRpb25bXSA9IFtdO1xuXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyB2aWV3Q29udGFpbmVyUmVmOiBWaWV3Q29udGFpbmVyUmVmKSB7fVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMuY29tcG9uZW50UmVmKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmIChjaGFuZ2VzLmNvbXBvbmVudFJlZikge1xuICAgICAgdGhpcy5zdWJzY3JpcHRpb25zLmZvckVhY2goKHMpID0+IHMudW5zdWJzY3JpYmUoKSk7XG4gICAgICBpZiAodGhpcy5jb21wb25lbnRSZWYpIHtcbiAgICAgICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICAgICAgdGhpcy5jb21wb25lbnRSZWYuaW5zdGFuY2UudmFsdWVDaGFuZ2Uuc3Vic2NyaWJlKCh2YWx1ZSkgPT5cbiAgICAgICAgICAgIHRoaXMudmFsdWVDaGFuZ2UubmV4dCh2YWx1ZSlcbiAgICAgICAgICApXG4gICAgICAgICk7XG4gICAgICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgICAgIHRoaXMuY29tcG9uZW50UmVmLmluc3RhbmNlLnNlbmQuc3Vic2NyaWJlKCh2YWx1ZSkgPT5cbiAgICAgICAgICAgIHRoaXMuc2VuZC5uZXh0KHZhbHVlKVxuICAgICAgICAgIClcbiAgICAgICAgKTtcbiAgICAgICAgaWYgKHRoaXMuY29tcG9uZW50UmVmLmluc3RhbmNlLnVzZXJNZW50aW9ucykge1xuICAgICAgICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgICAgICAgdGhpcy5jb21wb25lbnRSZWYuaW5zdGFuY2UudXNlck1lbnRpb25zLnN1YnNjcmliZSgodmFsdWUpID0+XG4gICAgICAgICAgICAgIHRoaXMudXNlck1lbnRpb25zLm5leHQodmFsdWUpXG4gICAgICAgICAgICApXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICBpZiAoY2hhbmdlcy5hcmVNZW50aW9uc0VuYWJsZWQpIHtcbiAgICAgIHRoaXMuY29tcG9uZW50UmVmLmluc3RhbmNlLmFyZU1lbnRpb25zRW5hYmxlZCA9IHRoaXMuYXJlTWVudGlvbnNFbmFibGVkO1xuICAgIH1cbiAgICBpZiAoY2hhbmdlcy5tZW50aW9uQXV0b2NvbXBsZXRlSXRlbVRlbXBsYXRlKSB7XG4gICAgICB0aGlzLmNvbXBvbmVudFJlZi5pbnN0YW5jZS5tZW50aW9uQXV0b2NvbXBsZXRlSXRlbVRlbXBsYXRlID1cbiAgICAgICAgdGhpcy5tZW50aW9uQXV0b2NvbXBsZXRlSXRlbVRlbXBsYXRlO1xuICAgIH1cbiAgICBpZiAoY2hhbmdlcy5tZW50aW9uU2NvcGUpIHtcbiAgICAgIHRoaXMuY29tcG9uZW50UmVmLmluc3RhbmNlLm1lbnRpb25TY29wZSA9IHRoaXMubWVudGlvblNjb3BlO1xuICAgIH1cbiAgICBpZiAoY2hhbmdlcy52YWx1ZSkge1xuICAgICAgdGhpcy5jb21wb25lbnRSZWYuaW5zdGFuY2UudmFsdWUgPSB0aGlzLnZhbHVlO1xuICAgIH1cbiAgICAvLyBuZ09uQ2hhbmdlcyBub3QgY2FsbGVkIGZvciBkeW5hbWljIGNvbXBvbmVudHMgc2luY2Ugd2UgZG9uJ3QgdXNlIHRlbXBsYXRlIGJpbmRpbmdcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQGFuZ3VsYXItZXNsaW50L25vLWxpZmVjeWNsZS1jYWxsXG4gICAgdGhpcy5jb21wb25lbnRSZWYuaW5zdGFuY2UubmdPbkNoYW5nZXMoY2hhbmdlcyk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dGFyZWEuaW50ZXJmYWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtaW5wdXQvdGV4dGFyZWEuaW50ZXJmYWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFdmVudEVtaXR0ZXIsIE9uQ2hhbmdlcywgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFVzZXJSZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtQ29udGV4dCB9IGZyb20gJy4uL3R5cGVzJztcblxuZXhwb3J0IGludGVyZmFjZSBUZXh0YXJlYUludGVyZmFjZSBleHRlbmRzIE9uQ2hhbmdlcyB7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIHZhbHVlQ2hhbmdlOiBFdmVudEVtaXR0ZXI8c3RyaW5nPjtcbiAgc2VuZDogRXZlbnRFbWl0dGVyPHZvaWQ+O1xuICB1c2VyTWVudGlvbnM/OiBFdmVudEVtaXR0ZXI8VXNlclJlc3BvbnNlW10+O1xuICBhcmVNZW50aW9uc0VuYWJsZWQ/OiBib29sZWFuO1xuICBtZW50aW9uQXV0b2NvbXBsZXRlSXRlbVRlbXBsYXRlPzpcbiAgICB8IFRlbXBsYXRlUmVmPE1lbnRpb25BdXRjb21wbGV0ZUxpc3RJdGVtQ29udGV4dD5cbiAgICB8IHVuZGVmaW5lZDtcbiAgbWVudGlvblNjb3BlPzogJ2NoYW5uZWwnIHwgJ2FwcGxpY2F0aW9uJztcbn1cbiJdfQ==