stream-chat-angular 4.66.1 → 5.0.0-v5.2

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 (161) hide show
  1. package/assets/version.d.ts +1 -1
  2. package/{esm2015/assets/version.js → esm2020/assets/version.mjs} +2 -2
  3. package/esm2020/lib/attachment-configuration.service.mjs +185 -0
  4. package/esm2020/lib/attachment-list/attachment-list.component.mjs +205 -0
  5. package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +45 -0
  6. package/esm2020/lib/attachment.service.mjs +262 -0
  7. package/esm2020/lib/avatar/avatar.component.mjs +163 -0
  8. package/esm2020/lib/avatar-placeholder/avatar-placeholder.component.mjs +74 -0
  9. package/esm2020/lib/channel/channel.component.mjs +46 -0
  10. package/esm2020/lib/channel-header/channel-header.component.mjs +79 -0
  11. package/esm2020/lib/channel-list/channel-list-toggle.service.mjs +72 -0
  12. package/esm2020/lib/channel-list/channel-list.component.mjs +60 -0
  13. package/esm2020/lib/channel-preview/channel-preview.component.mjs +155 -0
  14. package/esm2020/lib/channel.service.mjs +1460 -0
  15. package/esm2020/lib/chat-client.service.mjs +206 -0
  16. package/{esm2015/lib/custom-templates.service.js → esm2020/lib/custom-templates.service.mjs} +3 -3
  17. package/{esm2015/lib/date-parser.service.js → esm2020/lib/date-parser.service.mjs} +3 -3
  18. package/esm2020/lib/edit-message-form/edit-message-form.component.mjs +83 -0
  19. package/esm2020/lib/get-channel-display-text.mjs +14 -0
  20. package/esm2020/lib/get-message-translation.mjs +12 -0
  21. package/esm2020/lib/icon/icon.component.mjs +21 -0
  22. package/esm2020/lib/icon-placeholder/icon-placeholder.component.mjs +31 -0
  23. package/esm2020/lib/loading-indicator/loading-indicator.component.mjs +31 -0
  24. package/esm2020/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +38 -0
  25. package/esm2020/lib/message/message.component.mjs +418 -0
  26. package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +130 -0
  27. package/esm2020/lib/message-actions.service.mjs +119 -0
  28. package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
  29. package/esm2020/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +262 -0
  30. package/{esm2015/lib/message-input/emoji-input.service.js → esm2020/lib/message-input/emoji-input.service.mjs} +3 -3
  31. package/{esm2015/lib/message-input/message-input-config.service.js → esm2020/lib/message-input/message-input-config.service.mjs} +3 -3
  32. package/esm2020/lib/message-input/message-input.component.mjs +443 -0
  33. package/{esm2015/lib/message-input/textarea/textarea.component.js → esm2020/lib/message-input/textarea/textarea.component.mjs} +5 -9
  34. package/esm2020/lib/message-input/textarea.directive.mjs +89 -0
  35. package/esm2020/lib/message-list/group-styles.mjs +52 -0
  36. package/{esm2015/lib/message-list/image-load.service.js → esm2020/lib/message-list/image-load.service.mjs} +3 -3
  37. package/esm2020/lib/message-list/message-list.component.mjs +699 -0
  38. package/esm2020/lib/message-preview.mjs +21 -0
  39. package/esm2020/lib/message-reactions/message-reactions.component.mjs +252 -0
  40. package/{esm2015/lib/message-reactions.service.js → esm2020/lib/message-reactions.service.mjs} +3 -3
  41. package/{esm2015/lib/message.service.js → esm2020/lib/message.service.mjs} +4 -4
  42. package/esm2020/lib/modal/modal.component.mjs +69 -0
  43. package/esm2020/lib/notification/notification.component.mjs +20 -0
  44. package/esm2020/lib/notification-list/notification-list.component.mjs +37 -0
  45. package/esm2020/lib/notification.service.mjs +79 -0
  46. package/esm2020/lib/read-by.mjs +12 -0
  47. package/{esm2015/lib/stream-autocomplete-textarea.module.js → esm2020/lib/stream-autocomplete-textarea.module.mjs} +6 -6
  48. package/{esm2015/lib/stream-avatar.module.js → esm2020/lib/stream-avatar.module.mjs} +5 -5
  49. package/{esm2015/lib/stream-chat.module.js → esm2020/lib/stream-chat.module.mjs} +14 -16
  50. package/{esm2015/lib/stream-i18n.service.js → esm2020/lib/stream-i18n.service.mjs} +5 -5
  51. package/{esm2015/lib/stream-textarea.module.js → esm2020/lib/stream-textarea.module.mjs} +6 -6
  52. package/esm2020/lib/theme.service.mjs +123 -0
  53. package/esm2020/lib/thread/thread.component.mjs +51 -0
  54. package/{esm2015/lib/transliteration.service.js → esm2020/lib/transliteration.service.mjs} +3 -3
  55. package/esm2020/lib/types.mjs +2 -0
  56. package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +183 -0
  57. package/esm2020/lib/voice-recording/voice-recording.component.mjs +102 -0
  58. package/fesm2015/{stream-chat-angular.js → stream-chat-angular.mjs} +313 -426
  59. package/fesm2015/stream-chat-angular.mjs.map +1 -0
  60. package/fesm2020/stream-chat-angular.mjs +7123 -0
  61. package/fesm2020/stream-chat-angular.mjs.map +1 -0
  62. package/lib/attachment-list/attachment-list.component.d.ts +3 -3
  63. package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
  64. package/lib/attachment.service.d.ts +0 -1
  65. package/lib/avatar/avatar.component.d.ts +1 -1
  66. package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
  67. package/lib/channel/channel.component.d.ts +1 -1
  68. package/lib/channel-header/channel-header.component.d.ts +1 -1
  69. package/lib/channel-list/channel-list-toggle.service.d.ts +0 -1
  70. package/lib/channel-list/channel-list.component.d.ts +1 -1
  71. package/lib/channel-preview/channel-preview.component.d.ts +1 -1
  72. package/lib/channel.service.d.ts +7 -7
  73. package/lib/chat-client.service.d.ts +1 -1
  74. package/lib/edit-message-form/edit-message-form.component.d.ts +1 -1
  75. package/lib/get-message-translation.d.ts +1 -1
  76. package/lib/icon/icon.component.d.ts +1 -1
  77. package/lib/icon-placeholder/icon-placeholder.component.d.ts +1 -1
  78. package/lib/loading-indicator/loading-indicator.component.d.ts +1 -1
  79. package/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.d.ts +1 -1
  80. package/lib/message/message.component.d.ts +3 -7
  81. package/lib/message-actions-box/message-actions-box.component.d.ts +2 -4
  82. package/lib/message-actions.service.d.ts +0 -1
  83. package/lib/message-bounce-prompt/message-bounce-prompt.component.d.ts +1 -1
  84. package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +1 -1
  85. package/lib/message-input/message-input.component.d.ts +2 -2
  86. package/lib/message-input/textarea/textarea.component.d.ts +1 -1
  87. package/lib/message-input/textarea.directive.d.ts +1 -1
  88. package/lib/message-list/group-styles.d.ts +1 -1
  89. package/lib/message-list/message-list.component.d.ts +4 -5
  90. package/lib/message-reactions/message-reactions.component.d.ts +1 -4
  91. package/lib/message.service.d.ts +0 -1
  92. package/lib/modal/modal.component.d.ts +1 -1
  93. package/lib/notification/notification.component.d.ts +1 -1
  94. package/lib/notification-list/notification-list.component.d.ts +2 -2
  95. package/lib/notification.service.d.ts +2 -5
  96. package/lib/stream-chat.module.d.ts +3 -3
  97. package/lib/theme.service.d.ts +1 -2
  98. package/lib/thread/thread.component.d.ts +1 -1
  99. package/lib/types.d.ts +18 -18
  100. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +2 -2
  101. package/lib/voice-recording/voice-recording.component.d.ts +1 -1
  102. package/package.json +31 -17
  103. package/src/assets/styles/v2/css/index.css +1 -1
  104. package/src/assets/styles/v2/css/index.layout.css +1 -1
  105. package/src/assets/styles/v2/scss/_base.scss +2 -2
  106. package/src/assets/version.ts +1 -1
  107. package/bundles/stream-chat-angular.umd.js +0 -8445
  108. package/bundles/stream-chat-angular.umd.js.map +0 -1
  109. package/esm2015/lib/attachment-configuration.service.js +0 -186
  110. package/esm2015/lib/attachment-list/attachment-list.component.js +0 -209
  111. package/esm2015/lib/attachment-preview-list/attachment-preview-list.component.js +0 -49
  112. package/esm2015/lib/attachment.service.js +0 -276
  113. package/esm2015/lib/avatar/avatar.component.js +0 -172
  114. package/esm2015/lib/avatar-placeholder/avatar-placeholder.component.js +0 -78
  115. package/esm2015/lib/channel/channel.component.js +0 -50
  116. package/esm2015/lib/channel-header/channel-header.component.js +0 -86
  117. package/esm2015/lib/channel-list/channel-list-toggle.service.js +0 -73
  118. package/esm2015/lib/channel-list/channel-list.component.js +0 -67
  119. package/esm2015/lib/channel-preview/channel-preview.component.js +0 -167
  120. package/esm2015/lib/channel.service.js +0 -1487
  121. package/esm2015/lib/chat-client.service.js +0 -211
  122. package/esm2015/lib/edit-message-form/edit-message-form.component.js +0 -87
  123. package/esm2015/lib/get-channel-display-text.js +0 -15
  124. package/esm2015/lib/get-message-translation.js +0 -13
  125. package/esm2015/lib/icon/icon.component.js +0 -25
  126. package/esm2015/lib/icon-placeholder/icon-placeholder.component.js +0 -35
  127. package/esm2015/lib/loading-indicator/loading-indicator.component.js +0 -35
  128. package/esm2015/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.js +0 -42
  129. package/esm2015/lib/message/message.component.js +0 -436
  130. package/esm2015/lib/message-actions-box/message-actions-box.component.js +0 -137
  131. package/esm2015/lib/message-actions.service.js +0 -114
  132. package/esm2015/lib/message-bounce-prompt/message-bounce-prompt.component.js +0 -80
  133. package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +0 -262
  134. package/esm2015/lib/message-input/message-input.component.js +0 -455
  135. package/esm2015/lib/message-input/textarea.directive.js +0 -90
  136. package/esm2015/lib/message-list/group-styles.js +0 -53
  137. package/esm2015/lib/message-list/message-list.component.js +0 -726
  138. package/esm2015/lib/message-preview.js +0 -7
  139. package/esm2015/lib/message-reactions/message-reactions.component.js +0 -266
  140. package/esm2015/lib/modal/modal.component.js +0 -74
  141. package/esm2015/lib/notification/notification.component.js +0 -24
  142. package/esm2015/lib/notification-list/notification-list.component.js +0 -38
  143. package/esm2015/lib/notification.service.js +0 -79
  144. package/esm2015/lib/read-by.js +0 -13
  145. package/esm2015/lib/theme.service.js +0 -122
  146. package/esm2015/lib/thread/thread.component.js +0 -55
  147. package/esm2015/lib/types.js +0 -2
  148. package/esm2015/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.js +0 -192
  149. package/esm2015/lib/voice-recording/voice-recording.component.js +0 -115
  150. package/fesm2015/stream-chat-angular.js.map +0 -1
  151. /package/{esm2015/assets/i18n/en.js → esm2020/assets/i18n/en.mjs} +0 -0
  152. /package/{esm2015/lib/injection-tokens.js → esm2020/lib/injection-tokens.mjs} +0 -0
  153. /package/{esm2015/lib/is-image-attachment.js → esm2020/lib/is-image-attachment.mjs} +0 -0
  154. /package/{esm2015/lib/is-image-file.js → esm2020/lib/is-image-file.mjs} +0 -0
  155. /package/{esm2015/lib/is-on-separate-date.js → esm2020/lib/is-on-separate-date.mjs} +0 -0
  156. /package/{esm2015/lib/list-users.js → esm2020/lib/list-users.mjs} +0 -0
  157. /package/{esm2015/lib/message-input/textarea.interface.js → esm2020/lib/message-input/textarea.interface.mjs} +0 -0
  158. /package/{esm2015/lib/parse-date.js → esm2020/lib/parse-date.mjs} +0 -0
  159. /package/{esm2015/public-api.js → esm2020/public-api.mjs} +0 -0
  160. /package/{esm2015/stream-chat-angular.js → esm2020/stream-chat-angular.mjs} +0 -0
  161. /package/{stream-chat-angular.d.ts → index.d.ts} +0 -0
@@ -1,49 +0,0 @@
1
- import { Component, EventEmitter, Input, Output } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- import * as i1 from "../theme.service";
4
- import * as i2 from "../icon-placeholder/icon-placeholder.component";
5
- import * as i3 from "../loading-indicator-placeholder/loading-indicator-placeholder.component";
6
- import * as i4 from "@angular/common";
7
- import * as i5 from "@ngx-translate/core";
8
- /**
9
- * The `AttachmentPreviewList` component displays a preview of the attachments uploaded to a message. Users can delete attachments using the preview component, or retry upload if it failed previously.
10
- */
11
- export class AttachmentPreviewListComponent {
12
- constructor(themeService) {
13
- /**
14
- * An output to notify the parent component if the user tries to retry a failed upload
15
- */
16
- this.retryAttachmentUpload = new EventEmitter();
17
- /**
18
- * An output to notify the parent component if the user wants to delete a file
19
- */
20
- this.deleteAttachment = new EventEmitter();
21
- this.themeVersion = themeService.themeVersion;
22
- }
23
- attachmentUploadRetried(file) {
24
- this.retryAttachmentUpload.emit(file);
25
- }
26
- attachmentDeleted(upload) {
27
- this.deleteAttachment.emit(upload);
28
- }
29
- trackByFile(_, item) {
30
- return item.file;
31
- }
32
- }
33
- AttachmentPreviewListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentPreviewListComponent, deps: [{ token: i1.ThemeService }], target: i0.ɵɵFactoryTarget.Component });
34
- AttachmentPreviewListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AttachmentPreviewListComponent, selector: "stream-attachment-preview-list", inputs: { attachmentUploads$: "attachmentUploads$" }, outputs: { retryAttachmentUpload: "retryAttachmentUpload", deleteAttachment: "deleteAttachment" }, ngImport: i0, template: "<div\n class=\"rfu-image-previewer\"\n *ngIf=\"(attachmentUploads$ | async)?.length && themeVersion === '1'\"\n>\n <ng-container\n *ngFor=\"\n let attachmentUpload of attachmentUploads$ | async;\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)=\"attachmentUploadRetried(attachmentUpload.file)\"\n (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n data-testclass=\"upload-retry\"\n >\n <stream-icon-placeholder icon=\"retry\"></stream-icon-placeholder>\n </div>\n <div class=\"rfu-thumbnail__wrapper\">\n <div class=\"rfu-thumbnail__overlay\">\n <div\n class=\"rfu-icon-button\"\n data-testclass=\"file-delete\"\n role=\"button\"\n (click)=\"attachmentDeleted(attachmentUpload)\"\n (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n >\n <stream-icon-placeholder\n icon=\"close-no-outline\"\n ></stream-icon-placeholder>\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-placeholder\n data-testclass=\"loading-indicator\"\n color=\"rgba(255,255,255,0.7)\"\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n ></stream-loading-indicator-placeholder>\n </div>\n <div\n class=\"rfu-file-previewer\"\n *ngIf=\"\n attachmentUpload.type === 'file' || attachmentUpload.type === 'video'\n \"\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-placeholder icon=\"file\"></stream-icon-placeholder>\n\n <a\n data-testclass=\"file-download-link\"\n href=\"{{ attachmentUpload.url }}\"\n (click)=\"attachmentUpload.url ? null : $event.preventDefault()\"\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=\"upload-retry\"\n class=\"rfu-file-previewer__failed\"\n (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n translate\n >\n streamChat.failed\n </div>\n <div\n class=\"rfu-file-previewer__retry\"\n (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\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)=\"attachmentDeleted(attachmentUpload)\"\n (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n >\n <stream-icon-placeholder\n icon=\"close-no-outline\"\n ></stream-icon-placeholder>\n </span>\n <div\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n class=\"rfu-file-previewer__loading-indicator\"\n >\n <stream-loading-indicator-placeholder></stream-loading-indicator-placeholder>\n </div>\n </li>\n </ol>\n </div>\n </ng-container>\n</div>\n\n<div\n class=\"str-chat__attachment-preview-list\"\n *ngIf=\"(attachmentUploads$ | async)?.length && themeVersion === '2'\"\n>\n <div class=\"str-chat__attachment-list-scroll-container\">\n <ng-container\n *ngFor=\"\n let attachmentUpload of attachmentUploads$ | async;\n trackBy: trackByFile\n \"\n >\n <div\n *ngIf=\"attachmentUpload.type === 'image'\"\n class=\"str-chat__attachment-preview-image\"\n data-testclass=\"attachment-image-preview\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n deleteButton;\n context: { attachmentUpload: attachmentUpload }\n \"\n ></ng-container>\n <div\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n class=\"str-chat__attachment-preview-image-loading\"\n >\n <stream-loading-indicator-placeholder\n data-testclass=\"loading-indicator\"\n [size]=\"18\"\n ></stream-loading-indicator-placeholder>\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n retryButton;\n context: { attachmentUpload: attachmentUpload }\n \"\n ></ng-container>\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=\"str-chat__attachment-preview-thumbnail\"\n data-testclass=\"attachment-image\"\n />\n </div>\n <div\n class=\"str-chat__attachment-preview-file\"\n *ngIf=\"\n attachmentUpload.type === 'file' || attachmentUpload.type === 'video'\n \"\n data-testclass=\"attachment-file-preview\"\n >\n <stream-icon-placeholder\n icon=\"unspecified-filetype\"\n ></stream-icon-placeholder>\n\n <div class=\"str-chat__attachment-preview-file-end\">\n <div class=\"str-chat__attachment-preview-file-name\">\n {{ attachmentUpload.file.name }}\n </div>\n <a\n *ngIf=\"attachmentUpload.state === 'success'\"\n class=\"str-chat__attachment-preview-file-download\"\n data-testclass=\"file-download-link\"\n href=\"{{ attachmentUpload.url }}\"\n (click)=\"attachmentUpload.url ? null : $event.preventDefault()\"\n (keyup.enter)=\"\n attachmentUpload.url ? null : $event.preventDefault()\n \"\n download\n >\n <stream-icon-placeholder icon=\"download\"></stream-icon-placeholder>\n </a>\n <stream-loading-indicator-placeholder\n *ngIf=\"attachmentUpload.state === 'uploading'\"\n data-testclass=\"loading-indicator\"\n [size]=\"18\"\n ></stream-loading-indicator-placeholder>\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n deleteButton;\n context: { attachmentUpload: attachmentUpload }\n \"\n ></ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n retryButton;\n context: { attachmentUpload: attachmentUpload }\n \"\n ></ng-container>\n </div>\n </ng-container>\n </div>\n</div>\n\n<ng-template #deleteButton let-attachmentUpload=\"attachmentUpload\">\n <div\n class=\"str-chat__attachment-preview-delete\"\n data-testclass=\"file-delete\"\n role=\"button\"\n (click)=\"attachmentDeleted(attachmentUpload)\"\n (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n >\n <stream-icon-placeholder icon=\"close-no-outline\"></stream-icon-placeholder>\n </div>\n</ng-template>\n\n<ng-template #retryButton let-attachmentUpload=\"attachmentUpload\">\n <div\n *ngIf=\"attachmentUpload.state === 'error'\"\n class=\"str-chat__attachment-preview-error str-chat__attachment-preview-error-{{\n attachmentUpload.type === 'image' ? 'image' : 'file'\n }}\"\n (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n data-testclass=\"upload-retry\"\n >\n <stream-icon-placeholder icon=\"retry\"></stream-icon-placeholder>\n </div>\n</ng-template>\n", components: [{ type: i2.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon", "size"] }, { type: i3.LoadingIndicatorPlaceholderComponent, selector: "stream-loading-indicator-placeholder", inputs: ["size", "color"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "async": i4.AsyncPipe } });
35
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentPreviewListComponent, decorators: [{
36
- type: Component,
37
- args: [{
38
- selector: 'stream-attachment-preview-list',
39
- templateUrl: './attachment-preview-list.component.html',
40
- styles: [],
41
- }]
42
- }], ctorParameters: function () { return [{ type: i1.ThemeService }]; }, propDecorators: { attachmentUploads$: [{
43
- type: Input
44
- }], retryAttachmentUpload: [{
45
- type: Output
46
- }], deleteAttachment: [{
47
- type: Output
48
- }] } });
49
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment-preview-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/attachment-preview-list/attachment-preview-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/attachment-preview-list/attachment-preview-list.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;;;;;;;AAKvE;;GAEG;AAMH,MAAM,OAAO,8BAA8B;IAezC,YAAY,YAA0B;QAVtC;;WAEG;QACgB,0BAAqB,GAAG,IAAI,YAAY,EAAQ,CAAC;QACpE;;WAEG;QACgB,qBAAgB,GAAG,IAAI,YAAY,EAAoB,CAAC;QAIzE,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC,YAAY,CAAC;IAChD,CAAC;IAED,uBAAuB,CAAC,IAAU;QAChC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACxC,CAAC;IAED,iBAAiB,CAAC,MAAwB;QACxC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACrC,CAAC;IAED,WAAW,CAAC,CAAS,EAAE,IAAsB;QAC3C,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;;2HA7BU,8BAA8B;+GAA9B,8BAA8B,+NCb3C,o2RAgQA;2FDnPa,8BAA8B;kBAL1C,SAAS;mBAAC;oBACT,QAAQ,EAAE,gCAAgC;oBAC1C,WAAW,EAAE,0CAA0C;oBACvD,MAAM,EAAE,EAAE;iBACX;mGAKU,kBAAkB;sBAA1B,KAAK;gBAIa,qBAAqB;sBAAvC,MAAM;gBAIY,gBAAgB;sBAAlC,MAAM","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { ThemeService } from '../theme.service';\nimport { AttachmentUpload } from '../types';\n\n/**\n * The `AttachmentPreviewList` component displays a preview of the attachments uploaded to a message. Users can delete attachments using the preview component, or retry upload if it failed previously.\n */\n@Component({\n  selector: 'stream-attachment-preview-list',\n  templateUrl: './attachment-preview-list.component.html',\n  styles: [],\n})\nexport class AttachmentPreviewListComponent {\n  /**\n   * A stream that emits the current file uploads and their states\n   */\n  @Input() attachmentUploads$: Observable<AttachmentUpload[]> | undefined;\n  /**\n   * An output to notify the parent component if the user tries to retry a failed upload\n   */\n  @Output() readonly retryAttachmentUpload = new EventEmitter<File>();\n  /**\n   * An output to notify the parent component if the user wants to delete a file\n   */\n  @Output() readonly deleteAttachment = new EventEmitter<AttachmentUpload>();\n  themeVersion: '1' | '2';\n\n  constructor(themeService: ThemeService) {\n    this.themeVersion = themeService.themeVersion;\n  }\n\n  attachmentUploadRetried(file: File) {\n    this.retryAttachmentUpload.emit(file);\n  }\n\n  attachmentDeleted(upload: AttachmentUpload) {\n    this.deleteAttachment.emit(upload);\n  }\n\n  trackByFile(_: number, item: AttachmentUpload) {\n    return item.file;\n  }\n}\n","<div\n  class=\"rfu-image-previewer\"\n  *ngIf=\"(attachmentUploads$ | async)?.length && themeVersion === '1'\"\n>\n  <ng-container\n    *ngFor=\"\n      let attachmentUpload of attachmentUploads$ | async;\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)=\"attachmentUploadRetried(attachmentUpload.file)\"\n        (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n        data-testclass=\"upload-retry\"\n      >\n        <stream-icon-placeholder icon=\"retry\"></stream-icon-placeholder>\n      </div>\n      <div class=\"rfu-thumbnail__wrapper\">\n        <div class=\"rfu-thumbnail__overlay\">\n          <div\n            class=\"rfu-icon-button\"\n            data-testclass=\"file-delete\"\n            role=\"button\"\n            (click)=\"attachmentDeleted(attachmentUpload)\"\n            (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n          >\n            <stream-icon-placeholder\n              icon=\"close-no-outline\"\n            ></stream-icon-placeholder>\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-placeholder\n        data-testclass=\"loading-indicator\"\n        color=\"rgba(255,255,255,0.7)\"\n        *ngIf=\"attachmentUpload.state === 'uploading'\"\n      ></stream-loading-indicator-placeholder>\n    </div>\n    <div\n      class=\"rfu-file-previewer\"\n      *ngIf=\"\n        attachmentUpload.type === 'file' || attachmentUpload.type === 'video'\n      \"\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-placeholder icon=\"file\"></stream-icon-placeholder>\n\n          <a\n            data-testclass=\"file-download-link\"\n            href=\"{{ attachmentUpload.url }}\"\n            (click)=\"attachmentUpload.url ? null : $event.preventDefault()\"\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=\"upload-retry\"\n                class=\"rfu-file-previewer__failed\"\n                (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n                (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n                translate\n              >\n                streamChat.failed\n              </div>\n              <div\n                class=\"rfu-file-previewer__retry\"\n                (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n                (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\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)=\"attachmentDeleted(attachmentUpload)\"\n            (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n          >\n            <stream-icon-placeholder\n              icon=\"close-no-outline\"\n            ></stream-icon-placeholder>\n          </span>\n          <div\n            *ngIf=\"attachmentUpload.state === 'uploading'\"\n            class=\"rfu-file-previewer__loading-indicator\"\n          >\n            <stream-loading-indicator-placeholder></stream-loading-indicator-placeholder>\n          </div>\n        </li>\n      </ol>\n    </div>\n  </ng-container>\n</div>\n\n<div\n  class=\"str-chat__attachment-preview-list\"\n  *ngIf=\"(attachmentUploads$ | async)?.length && themeVersion === '2'\"\n>\n  <div class=\"str-chat__attachment-list-scroll-container\">\n    <ng-container\n      *ngFor=\"\n        let attachmentUpload of attachmentUploads$ | async;\n        trackBy: trackByFile\n      \"\n    >\n      <div\n        *ngIf=\"attachmentUpload.type === 'image'\"\n        class=\"str-chat__attachment-preview-image\"\n        data-testclass=\"attachment-image-preview\"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            deleteButton;\n            context: { attachmentUpload: attachmentUpload }\n          \"\n        ></ng-container>\n        <div\n          *ngIf=\"attachmentUpload.state === 'uploading'\"\n          class=\"str-chat__attachment-preview-image-loading\"\n        >\n          <stream-loading-indicator-placeholder\n            data-testclass=\"loading-indicator\"\n            [size]=\"18\"\n          ></stream-loading-indicator-placeholder>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"\n            retryButton;\n            context: { attachmentUpload: attachmentUpload }\n          \"\n        ></ng-container>\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=\"str-chat__attachment-preview-thumbnail\"\n          data-testclass=\"attachment-image\"\n        />\n      </div>\n      <div\n        class=\"str-chat__attachment-preview-file\"\n        *ngIf=\"\n          attachmentUpload.type === 'file' || attachmentUpload.type === 'video'\n        \"\n        data-testclass=\"attachment-file-preview\"\n      >\n        <stream-icon-placeholder\n          icon=\"unspecified-filetype\"\n        ></stream-icon-placeholder>\n\n        <div class=\"str-chat__attachment-preview-file-end\">\n          <div class=\"str-chat__attachment-preview-file-name\">\n            {{ attachmentUpload.file.name }}\n          </div>\n          <a\n            *ngIf=\"attachmentUpload.state === 'success'\"\n            class=\"str-chat__attachment-preview-file-download\"\n            data-testclass=\"file-download-link\"\n            href=\"{{ attachmentUpload.url }}\"\n            (click)=\"attachmentUpload.url ? null : $event.preventDefault()\"\n            (keyup.enter)=\"\n              attachmentUpload.url ? null : $event.preventDefault()\n            \"\n            download\n          >\n            <stream-icon-placeholder icon=\"download\"></stream-icon-placeholder>\n          </a>\n          <stream-loading-indicator-placeholder\n            *ngIf=\"attachmentUpload.state === 'uploading'\"\n            data-testclass=\"loading-indicator\"\n            [size]=\"18\"\n          ></stream-loading-indicator-placeholder>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"\n            deleteButton;\n            context: { attachmentUpload: attachmentUpload }\n          \"\n        ></ng-container>\n        <ng-container\n          *ngTemplateOutlet=\"\n            retryButton;\n            context: { attachmentUpload: attachmentUpload }\n          \"\n        ></ng-container>\n      </div>\n    </ng-container>\n  </div>\n</div>\n\n<ng-template #deleteButton let-attachmentUpload=\"attachmentUpload\">\n  <div\n    class=\"str-chat__attachment-preview-delete\"\n    data-testclass=\"file-delete\"\n    role=\"button\"\n    (click)=\"attachmentDeleted(attachmentUpload)\"\n    (keyup.enter)=\"attachmentDeleted(attachmentUpload)\"\n  >\n    <stream-icon-placeholder icon=\"close-no-outline\"></stream-icon-placeholder>\n  </div>\n</ng-template>\n\n<ng-template #retryButton let-attachmentUpload=\"attachmentUpload\">\n  <div\n    *ngIf=\"attachmentUpload.state === 'error'\"\n    class=\"str-chat__attachment-preview-error str-chat__attachment-preview-error-{{\n      attachmentUpload.type === 'image' ? 'image' : 'file'\n    }}\"\n    (click)=\"attachmentUploadRetried(attachmentUpload.file)\"\n    (keyup.enter)=\"attachmentUploadRetried(attachmentUpload.file)\"\n    data-testclass=\"upload-retry\"\n  >\n    <stream-icon-placeholder icon=\"retry\"></stream-icon-placeholder>\n  </div>\n</ng-template>\n"]}
@@ -1,276 +0,0 @@
1
- import { __awaiter } from "tslib";
2
- import { Injectable } from '@angular/core';
3
- import { isImageFile } from './is-image-file';
4
- import { BehaviorSubject } from 'rxjs';
5
- import { isImageAttachment } from './is-image-attachment';
6
- import * as i0 from "@angular/core";
7
- import * as i1 from "./channel.service";
8
- import * as i2 from "./notification.service";
9
- /**
10
- * The `AttachmentService` manages the uploads of a message input.
11
- */
12
- export class AttachmentService {
13
- constructor(channelService, notificationService) {
14
- this.channelService = channelService;
15
- this.notificationService = notificationService;
16
- this.attachmentUploadInProgressCounterSubject = new BehaviorSubject(0);
17
- this.attachmentUploadsSubject = new BehaviorSubject([]);
18
- this.attachmentUploadInProgressCounter$ =
19
- this.attachmentUploadInProgressCounterSubject.asObservable();
20
- this.attachmentUploads$ = this.attachmentUploadsSubject.asObservable();
21
- }
22
- /**
23
- * Resets the attachments uploads (for example after the message with the attachments sent successfully)
24
- */
25
- resetAttachmentUploads() {
26
- this.attachmentUploadsSubject.next([]);
27
- }
28
- /**
29
- * Uploads the selected files, and creates preview for image files. The result is propagated throught the `attachmentUploads$` stream.
30
- * @param fileList The files selected by the user
31
- * @returns A promise with the result
32
- */
33
- filesSelected(fileList) {
34
- return __awaiter(this, void 0, void 0, function* () {
35
- if (!fileList) {
36
- return;
37
- }
38
- const imageFiles = [];
39
- const dataFiles = [];
40
- const videoFiles = [];
41
- Array.from(fileList).forEach((file) => {
42
- if (isImageFile(file)) {
43
- imageFiles.push(file);
44
- }
45
- else if (file.type.startsWith('video/')) {
46
- videoFiles.push(file);
47
- }
48
- else {
49
- dataFiles.push(file);
50
- }
51
- });
52
- imageFiles.forEach((f) => this.createPreview(f));
53
- const newUploads = [
54
- ...imageFiles.map((file) => ({
55
- file,
56
- state: 'uploading',
57
- type: 'image',
58
- })),
59
- ...videoFiles.map((file) => ({
60
- file,
61
- state: 'uploading',
62
- type: 'video',
63
- })),
64
- ...dataFiles.map((file) => ({
65
- file,
66
- state: 'uploading',
67
- type: 'file',
68
- })),
69
- ];
70
- this.attachmentUploadsSubject.next([
71
- ...this.attachmentUploadsSubject.getValue(),
72
- ...newUploads,
73
- ]);
74
- yield this.uploadAttachments(newUploads);
75
- });
76
- }
77
- /**
78
- * You can add custom `image`, `video` and `file` attachments using this method.
79
- *
80
- * Note: If you just want to use your own CDN for file uploads, you don't necessary need this method, you can just specify you own upload function in the [`ChannelService`](./ChannelService.mdx)
81
- *
82
- * @param attachment
83
- */
84
- addAttachment(attachment) {
85
- attachment.isCustomAttachment = true;
86
- this.createFromAttachments([attachment]);
87
- }
88
- /**
89
- * Retries to upload an attachment.
90
- * @param file
91
- * @returns A promise with the result
92
- */
93
- retryAttachmentUpload(file) {
94
- return __awaiter(this, void 0, void 0, function* () {
95
- const attachmentUploads = this.attachmentUploadsSubject.getValue();
96
- const upload = attachmentUploads.find((u) => u.file === file);
97
- if (!upload) {
98
- return;
99
- }
100
- upload.state = 'uploading';
101
- this.attachmentUploadsSubject.next([...attachmentUploads]);
102
- yield this.uploadAttachments([upload]);
103
- });
104
- }
105
- /**
106
- * Deletes an attachment, the attachment can have any state (`error`, `uploading` or `success`).
107
- * @param upload
108
- */
109
- deleteAttachment(upload) {
110
- var _a;
111
- return __awaiter(this, void 0, void 0, function* () {
112
- const attachmentUploads = this.attachmentUploadsSubject.getValue();
113
- let result;
114
- if (upload.state === 'success' &&
115
- !((_a = upload.fromAttachment) === null || _a === void 0 ? void 0 : _a.isCustomAttachment)) {
116
- try {
117
- yield this.channelService.deleteAttachment(upload);
118
- result = [...attachmentUploads];
119
- const index = attachmentUploads.indexOf(upload);
120
- result.splice(index, 1);
121
- }
122
- catch (error) {
123
- result = attachmentUploads;
124
- this.notificationService.addTemporaryNotification('streamChat.Error deleting attachment');
125
- }
126
- }
127
- else {
128
- result = [...attachmentUploads];
129
- const index = attachmentUploads.indexOf(upload);
130
- result.splice(index, 1);
131
- }
132
- this.attachmentUploadsSubject.next([...result]);
133
- });
134
- }
135
- /**
136
- * Maps the current uploads to a format that can be sent along with the message to the Stream API.
137
- * @returns the attachments
138
- */
139
- mapToAttachments() {
140
- const attachmentUploads = this.attachmentUploadsSubject.getValue();
141
- return attachmentUploads
142
- .filter((r) => r.state === 'success')
143
- .map((r) => {
144
- var _a, _b, _c, _d;
145
- const attachment = {
146
- type: r.type,
147
- };
148
- if (r.fromAttachment) {
149
- return r.fromAttachment;
150
- }
151
- else {
152
- attachment.mime_type = (_a = r.file) === null || _a === void 0 ? void 0 : _a.type;
153
- if (r.type === 'image') {
154
- attachment.fallback = (_b = r.file) === null || _b === void 0 ? void 0 : _b.name;
155
- attachment.image_url = r.url;
156
- }
157
- else {
158
- attachment.asset_url = r.url;
159
- attachment.title = (_c = r.file) === null || _c === void 0 ? void 0 : _c.name;
160
- attachment.file_size = (_d = r.file) === null || _d === void 0 ? void 0 : _d.size;
161
- attachment.thumb_url = r.thumb_url;
162
- }
163
- }
164
- return attachment;
165
- });
166
- }
167
- /**
168
- * Maps attachments received from the Stream API to uploads. This is useful when editing a message.
169
- * @param attachments Attachemnts received with the message
170
- */
171
- createFromAttachments(attachments) {
172
- const attachmentUploads = [];
173
- attachments.forEach((attachment) => {
174
- if (isImageAttachment(attachment)) {
175
- attachmentUploads.push({
176
- url: (attachment.img_url ||
177
- attachment.thumb_url ||
178
- attachment.image_url),
179
- state: 'success',
180
- type: 'image',
181
- file: {
182
- name: attachment.fallback,
183
- type: attachment.mime_type,
184
- },
185
- fromAttachment: attachment,
186
- });
187
- }
188
- else if (attachment.type === 'file' || attachment.type === 'video') {
189
- attachmentUploads.push({
190
- url: attachment.asset_url,
191
- state: 'success',
192
- file: {
193
- name: attachment.title,
194
- size: attachment.file_size,
195
- type: attachment.mime_type,
196
- },
197
- type: attachment.type,
198
- thumb_url: attachment.thumb_url,
199
- fromAttachment: attachment,
200
- });
201
- }
202
- });
203
- if (attachmentUploads.length > 0) {
204
- this.attachmentUploadsSubject.next([
205
- ...this.attachmentUploadsSubject.getValue(),
206
- ...attachmentUploads,
207
- ]);
208
- }
209
- }
210
- createPreview(file) {
211
- const reader = new FileReader();
212
- reader.onload = (event) => {
213
- var _a;
214
- const attachmentUploads = this.attachmentUploadsSubject.getValue();
215
- const upload = attachmentUploads.find((upload) => upload.file === file);
216
- if (!upload) {
217
- return;
218
- }
219
- upload.previewUri = ((_a = event.target) === null || _a === void 0 ? void 0 : _a.result) || undefined;
220
- this.attachmentUploadsSubject.next([...attachmentUploads]);
221
- };
222
- reader.readAsDataURL(file);
223
- }
224
- uploadAttachments(uploads) {
225
- return __awaiter(this, void 0, void 0, function* () {
226
- this.attachmentUploadInProgressCounterSubject.next(this.attachmentUploadInProgressCounterSubject.getValue() + 1);
227
- const result = yield this.channelService.uploadAttachments(uploads);
228
- const attachmentUploads = this.attachmentUploadsSubject.getValue();
229
- result.forEach((r) => {
230
- var _a, _b, _c, _d;
231
- const upload = attachmentUploads.find((upload) => upload.file === r.file);
232
- if (!upload) {
233
- if (r.url) {
234
- void this.channelService.deleteAttachment(r);
235
- }
236
- return;
237
- }
238
- upload.state = r.state;
239
- upload.url = r.url;
240
- upload.thumb_url = r.thumb_url;
241
- if (upload.state === 'error') {
242
- upload.errorReason = r.errorReason;
243
- upload.errorExtraInfo = r.errorExtraInfo;
244
- let errorKey;
245
- const translateParams = { name: upload.file.name };
246
- switch (upload.errorReason) {
247
- case 'file-extension':
248
- errorKey =
249
- 'streamChat.Error uploading file, extension not supported';
250
- translateParams.ext = (_b = (_a = upload.errorExtraInfo) === null || _a === void 0 ? void 0 : _a[0]) === null || _b === void 0 ? void 0 : _b.param;
251
- break;
252
- case 'file-size':
253
- errorKey =
254
- 'streamChat.Error uploading file, maximum file size exceeded';
255
- translateParams.limit = (_d = (_c = upload.errorExtraInfo) === null || _c === void 0 ? void 0 : _c[0]) === null || _d === void 0 ? void 0 : _d.param;
256
- break;
257
- default:
258
- errorKey = 'streamChat.Error uploading file';
259
- }
260
- this.notificationService.addTemporaryNotification(errorKey, 'error', undefined, translateParams);
261
- }
262
- });
263
- this.attachmentUploadInProgressCounterSubject.next(this.attachmentUploadInProgressCounterSubject.getValue() - 1);
264
- this.attachmentUploadsSubject.next([...attachmentUploads]);
265
- });
266
- }
267
- }
268
- AttachmentService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentService, deps: [{ token: i1.ChannelService }, { token: i2.NotificationService }], target: i0.ɵɵFactoryTarget.Injectable });
269
- AttachmentService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentService, providedIn: 'root' });
270
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentService, decorators: [{
271
- type: Injectable,
272
- args: [{
273
- providedIn: 'root',
274
- }]
275
- }], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.NotificationService }]; } });
276
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/attachment.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { isImageFile } from './is-image-file';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { Attachment } from 'stream-chat';\nimport { ChannelService } from './channel.service';\nimport { isImageAttachment } from './is-image-attachment';\nimport { NotificationService } from './notification.service';\nimport { AttachmentUpload, DefaultStreamChatGenerics } from './types';\n\n/**\n * The `AttachmentService` manages the uploads of a message input.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class AttachmentService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * Emits the number of uploads in progress.\n   */\n  attachmentUploadInProgressCounter$: Observable<number>;\n  /**\n   * Emits the state of the uploads ([`AttachmentUpload[]`](https://github.com/GetStream/stream-chat-angular/blob/master/projects/stream-chat-angular/src/lib/types.ts)), it adds a state (`success`, `error` or `uploading`) to each file the user selects for upload. It is used by the [`AttachmentPreviewList`](../components/AttachmentPreviewListComponent.mdx) to display the attachment previews.\n   */\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  private attachmentUploadInProgressCounterSubject =\n    new BehaviorSubject<number>(0);\n  private attachmentUploadsSubject = new BehaviorSubject<AttachmentUpload[]>(\n    []\n  );\n\n  constructor(\n    private channelService: ChannelService,\n    private notificationService: NotificationService\n  ) {\n    this.attachmentUploadInProgressCounter$ =\n      this.attachmentUploadInProgressCounterSubject.asObservable();\n    this.attachmentUploads$ = this.attachmentUploadsSubject.asObservable();\n  }\n\n  /**\n   * Resets the attachments uploads (for example after the message with the attachments sent successfully)\n   */\n  resetAttachmentUploads() {\n    this.attachmentUploadsSubject.next([]);\n  }\n\n  /**\n   * Uploads the selected files, and creates preview for image files. The result is propagated throught the `attachmentUploads$` stream.\n   * @param fileList The files selected by the user\n   * @returns A promise with the result\n   */\n  async filesSelected(fileList: FileList | null) {\n    if (!fileList) {\n      return;\n    }\n    const imageFiles: File[] = [];\n    const dataFiles: File[] = [];\n    const videoFiles: File[] = [];\n\n    Array.from(fileList).forEach((file) => {\n      if (isImageFile(file)) {\n        imageFiles.push(file);\n      } else if (file.type.startsWith('video/')) {\n        videoFiles.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      ...videoFiles.map((file) => ({\n        file,\n        state: 'uploading' as 'uploading',\n        type: 'video' as 'video',\n      })),\n      ...dataFiles.map((file) => ({\n        file,\n        state: 'uploading' as 'uploading',\n        type: 'file' as 'file',\n      })),\n    ];\n    this.attachmentUploadsSubject.next([\n      ...this.attachmentUploadsSubject.getValue(),\n      ...newUploads,\n    ]);\n    await this.uploadAttachments(newUploads);\n  }\n\n  /**\n   * You can add custom `image`, `video` and `file` attachments using this method.\n   *\n   * Note: If you just want to use your own CDN for file uploads, you don't necessary need this method, you can just specify you own upload function in the [`ChannelService`](./ChannelService.mdx)\n   *\n   * @param attachment\n   */\n  addAttachment(attachment: Attachment<T>) {\n    attachment.isCustomAttachment = true;\n    this.createFromAttachments([attachment]);\n  }\n\n  /**\n   * Retries to upload an attachment.\n   * @param file\n   * @returns A promise with the result\n   */\n  async retryAttachmentUpload(file: File) {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    const upload = attachmentUploads.find((u) => u.file === file);\n    if (!upload) {\n      return;\n    }\n    upload.state = 'uploading';\n    this.attachmentUploadsSubject.next([...attachmentUploads]);\n    await this.uploadAttachments([upload]);\n  }\n\n  /**\n   * Deletes an attachment, the attachment can have any state (`error`, `uploading` or `success`).\n   * @param upload\n   */\n  async deleteAttachment(upload: AttachmentUpload) {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    let result!: AttachmentUpload[];\n    if (\n      upload.state === 'success' &&\n      !upload.fromAttachment?.isCustomAttachment\n    ) {\n      try {\n        await this.channelService.deleteAttachment(upload);\n        result = [...attachmentUploads];\n        const index = attachmentUploads.indexOf(upload);\n        result.splice(index, 1);\n      } catch (error) {\n        result = attachmentUploads;\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Error deleting attachment'\n        );\n      }\n    } else {\n      result = [...attachmentUploads];\n      const index = attachmentUploads.indexOf(upload);\n      result.splice(index, 1);\n    }\n    this.attachmentUploadsSubject.next([...result]);\n  }\n\n  /**\n   * Maps the current uploads to a format that can be sent along with the message to the Stream API.\n   * @returns the attachments\n   */\n  mapToAttachments() {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    return attachmentUploads\n      .filter((r) => r.state === 'success')\n      .map((r) => {\n        const attachment: Attachment = {\n          type: r.type,\n        };\n        if (r.fromAttachment) {\n          return r.fromAttachment;\n        } else {\n          attachment.mime_type = r.file?.type;\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            attachment.thumb_url = r.thumb_url;\n          }\n        }\n\n        return attachment;\n      });\n  }\n\n  /**\n   * Maps attachments received from the Stream API to uploads. This is useful when editing a message.\n   * @param attachments Attachemnts received with the message\n   */\n  createFromAttachments(attachments: Attachment<T>[]) {\n    const attachmentUploads: AttachmentUpload[] = [];\n    attachments.forEach((attachment) => {\n      if (isImageAttachment(attachment)) {\n        attachmentUploads.push({\n          url: (attachment.img_url ||\n            attachment.thumb_url ||\n            attachment.image_url) as string,\n          state: 'success',\n          type: 'image',\n          file: {\n            name: attachment.fallback,\n            type: attachment.mime_type,\n          } as File,\n          fromAttachment: attachment,\n        });\n      } else if (attachment.type === 'file' || attachment.type === 'video') {\n        attachmentUploads.push({\n          url: attachment.asset_url,\n          state: 'success',\n          file: {\n            name: attachment.title,\n            size: attachment.file_size,\n            type: attachment.mime_type,\n          } as File,\n          type: attachment.type,\n          thumb_url: attachment.thumb_url,\n          fromAttachment: attachment,\n        });\n      }\n    });\n\n    if (attachmentUploads.length > 0) {\n      this.attachmentUploadsSubject.next([\n        ...this.attachmentUploadsSubject.getValue(),\n        ...attachmentUploads,\n      ]);\n    }\n  }\n\n  private createPreview(file: File) {\n    const reader = new FileReader();\n    reader.onload = (event) => {\n      const attachmentUploads = this.attachmentUploadsSubject.getValue();\n      const upload = attachmentUploads.find((upload) => upload.file === file);\n      if (!upload) {\n        return;\n      }\n      upload.previewUri = event.target?.result || undefined;\n      this.attachmentUploadsSubject.next([...attachmentUploads]);\n    };\n    reader.readAsDataURL(file as Blob);\n  }\n\n  private async uploadAttachments(uploads: AttachmentUpload[]) {\n    this.attachmentUploadInProgressCounterSubject.next(\n      this.attachmentUploadInProgressCounterSubject.getValue() + 1\n    );\n    const result = await this.channelService.uploadAttachments(uploads);\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    result.forEach((r) => {\n      const upload = attachmentUploads.find((upload) => upload.file === r.file);\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      upload.thumb_url = r.thumb_url;\n      if (upload.state === 'error') {\n        upload.errorReason = r.errorReason;\n        upload.errorExtraInfo = r.errorExtraInfo;\n        let errorKey;\n        const translateParams: { name: string; ext?: string; limit?: string } =\n          { name: upload.file.name };\n        switch (upload.errorReason) {\n          case 'file-extension':\n            errorKey =\n              'streamChat.Error uploading file, extension not supported';\n            translateParams.ext = upload.errorExtraInfo?.[0]?.param;\n            break;\n          case 'file-size':\n            errorKey =\n              'streamChat.Error uploading file, maximum file size exceeded';\n            translateParams.limit = upload.errorExtraInfo?.[0]?.param;\n            break;\n          default:\n            errorKey = 'streamChat.Error uploading file';\n        }\n        this.notificationService.addTemporaryNotification(\n          errorKey,\n          'error',\n          undefined,\n          translateParams\n        );\n      }\n    });\n    this.attachmentUploadInProgressCounterSubject.next(\n      this.attachmentUploadInProgressCounterSubject.getValue() - 1\n    );\n    this.attachmentUploadsSubject.next([...attachmentUploads]);\n  }\n}\n"]}
@@ -1,172 +0,0 @@
1
- import { Component, Input, } from '@angular/core';
2
- import { filter } from 'rxjs/operators';
3
- import * as i0 from "@angular/core";
4
- import * as i1 from "../chat-client.service";
5
- import * as i2 from "@angular/common";
6
- /**
7
- * The `Avatar` component displays the provided image, with fallback to the first letter of the optional name input.
8
- */
9
- export class AvatarComponent {
10
- constructor(chatClientService, ngZone, cdRef) {
11
- this.chatClientService = chatClientService;
12
- this.ngZone = ngZone;
13
- this.cdRef = cdRef;
14
- /**
15
- * The size in pixels of the avatar image.
16
- */
17
- this.size = 32;
18
- /**
19
- * If a channel avatar is displayed, and if the channel has exactly two members a green dot is displayed if the other member is online. Set this flag to `false` to turn off this behavior.
20
- */
21
- this.showOnlineIndicator = true;
22
- /**
23
- * If channel/user image isn't provided the initials of the name of the channel/user is shown instead, you can choose how the initals should be computed
24
- */
25
- this.initialsType = 'first-letter-of-first-word';
26
- this.isError = false;
27
- this.isOnline = false;
28
- this.initials = '';
29
- this.isViewInited = false;
30
- this.subscriptions = [];
31
- }
32
- ngOnInit() {
33
- this.subscriptions.push(this.chatClientService.user$.subscribe((u) => {
34
- if ((u === null || u === void 0 ? void 0 : u.id) !== this.userId) {
35
- this.userId = u === null || u === void 0 ? void 0 : u.id;
36
- if (this.type || this.channel || this.name) {
37
- this.setInitials();
38
- this.setFallbackChannelImage();
39
- this.updateIsOnlineSubscription();
40
- }
41
- if (this.isViewInited) {
42
- this.cdRef.detectChanges();
43
- }
44
- }
45
- }));
46
- }
47
- ngOnChanges(changes) {
48
- if (changes['channel']) {
49
- this.updateIsOnlineSubscription();
50
- }
51
- if (changes.type || changes.name || changes.channel) {
52
- this.setInitials();
53
- }
54
- if (changes.type || changes.channel) {
55
- this.setFallbackChannelImage();
56
- }
57
- }
58
- setFallbackChannelImage() {
59
- if (this.type !== 'channel') {
60
- this.fallbackChannelImage = undefined;
61
- }
62
- else {
63
- const otherMember = this.getOtherMemberIfOneToOneChannel();
64
- if (otherMember) {
65
- this.fallbackChannelImage = otherMember.image;
66
- }
67
- else {
68
- this.fallbackChannelImage = undefined;
69
- }
70
- }
71
- }
72
- setInitials() {
73
- var _a, _b, _c, _d, _e;
74
- let result = '';
75
- if (this.type === 'user') {
76
- result = ((_a = this.name) === null || _a === void 0 ? void 0 : _a.toString()) || '';
77
- }
78
- else if (this.type === 'channel') {
79
- if ((_c = (_b = this.channel) === null || _b === void 0 ? void 0 : _b.data) === null || _c === void 0 ? void 0 : _c.name) {
80
- result = (_e = (_d = this.channel) === null || _d === void 0 ? void 0 : _d.data) === null || _e === void 0 ? void 0 : _e.name;
81
- }
82
- else {
83
- const otherMember = this.getOtherMemberIfOneToOneChannel();
84
- if (otherMember) {
85
- result = otherMember.name || otherMember.id || '';
86
- }
87
- else {
88
- result = '#';
89
- }
90
- }
91
- }
92
- const words = result.split(' ');
93
- let initials;
94
- if (this.initialsType === 'first-letter-of-each-word') {
95
- initials = words.map((w) => w.charAt(0) || '').join('');
96
- }
97
- else {
98
- initials = words[0].charAt(0) || '';
99
- }
100
- this.initials = initials;
101
- }
102
- updateIsOnlineSubscription() {
103
- var _a, _b;
104
- if (this.channel) {
105
- const otherMember = this.getOtherMemberIfOneToOneChannel();
106
- if (otherMember) {
107
- this.isOnline = otherMember.online || false;
108
- this.isOnlineSubscription = this.chatClientService.events$
109
- .pipe(filter((e) => e.eventType === 'user.presence.changed'))
110
- .subscribe((event) => {
111
- var _a;
112
- if (((_a = event.event.user) === null || _a === void 0 ? void 0 : _a.id) === otherMember.id) {
113
- this.ngZone.run(() => {
114
- var _a;
115
- this.isOnline = ((_a = event.event.user) === null || _a === void 0 ? void 0 : _a.online) || false;
116
- });
117
- }
118
- });
119
- }
120
- else {
121
- this.isOnline = false;
122
- (_a = this.isOnlineSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
123
- }
124
- }
125
- else {
126
- this.isOnline = false;
127
- (_b = this.isOnlineSubscription) === null || _b === void 0 ? void 0 : _b.unsubscribe();
128
- }
129
- }
130
- ngAfterViewInit() {
131
- this.isViewInited = true;
132
- }
133
- getOtherMemberIfOneToOneChannel() {
134
- var _a, _b;
135
- const otherMembers = Object.values(((_b = (_a = this.channel) === null || _a === void 0 ? void 0 : _a.state) === null || _b === void 0 ? void 0 : _b.members) || {}).filter((m) => m.user_id !== this.userId);
136
- if (otherMembers.length === 1) {
137
- return otherMembers[0].user;
138
- }
139
- else {
140
- return undefined;
141
- }
142
- }
143
- }
144
- AvatarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, deps: [{ token: i1.ChatClientService }, { token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
145
- AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AvatarComponent, selector: "stream-avatar", inputs: { name: "name", imageUrl: "imageUrl", size: "size", location: "location", channel: "channel", user: "user", type: "type", showOnlineIndicator: "showOnlineIndicator", initialsType: "initialsType" }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n location\n }}\"\n title=\"{{ name }}\"\n [ngStyle]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n fontSize:\n initialsType === 'first-letter-of-first-word'\n ? 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')'\n : 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 3 + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n>\n <img\n *ngIf=\"(imageUrl || fallbackChannelImage) && !isError; else fallback\"\n class=\"str-chat__avatar-image str-chat__avatar-image\"\n src=\"{{ imageUrl || fallbackChannelImage }}\"\n alt=\"{{ initials }}\"\n data-testid=\"avatar-img\"\n (error)=\"isError = true\"\n [ngStyle]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n objectFit: 'cover',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n fetchpriority=\"high\"\n />\n <ng-template #fallback>\n <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n {{ initials }}\n </div>\n </ng-template>\n <div\n data-testid=\"online-indicator\"\n *ngIf=\"isOnline && showOnlineIndicator\"\n class=\"str-chat__avatar--online-indicator\"\n ></div>\n</div>\n", styles: [""], directives: [{ type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
146
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, decorators: [{
147
- type: Component,
148
- args: [{
149
- selector: 'stream-avatar',
150
- templateUrl: './avatar.component.html',
151
- styleUrls: ['./avatar.component.scss'],
152
- }]
153
- }], ctorParameters: function () { return [{ type: i1.ChatClientService }, { type: i0.NgZone }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { name: [{
154
- type: Input
155
- }], imageUrl: [{
156
- type: Input
157
- }], size: [{
158
- type: Input
159
- }], location: [{
160
- type: Input
161
- }], channel: [{
162
- type: Input
163
- }], user: [{
164
- type: Input
165
- }], type: [{
166
- type: Input
167
- }], showOnlineIndicator: [{
168
- type: Input
169
- }], initialsType: [{
170
- type: Input
171
- }] } });
172
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/avatar/avatar.component.ts","../../../../../projects/stream-chat-angular/src/lib/avatar/avatar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  Input,\n  NgZone,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { filter } from 'rxjs/operators';\nimport { Channel, User } from 'stream-chat';\nimport { ChatClientService } from '../chat-client.service';\nimport {\n  AvatarLocation,\n  AvatarType,\n  DefaultStreamChatGenerics,\n} from '../types';\n\n/**\n * The `Avatar` component displays the provided image, with fallback to the first letter of the optional name input.\n */\n@Component({\n  selector: 'stream-avatar',\n  templateUrl: './avatar.component.html',\n  styleUrls: ['./avatar.component.scss'],\n})\nexport class AvatarComponent\n  implements OnChanges, OnInit, OnChanges, AfterViewInit\n{\n  /**\n   * An optional name of the image, used for fallback image or image title (if `imageUrl` is provided)\n   */\n  @Input() name: string | undefined;\n  /**\n   * The URL of the image to be displayed. If the image can't be displayed the first letter of the name input is displayed.\n   */\n  @Input() imageUrl: string | undefined;\n  /**\n   * The size in pixels of the avatar image.\n   */\n  @Input() size = 32;\n  /**\n   * The location the avatar will be displayed in\n   */\n  @Input() location: AvatarLocation | undefined;\n  /**\n   * The channel the avatar belongs to (if avatar of a channel is displayed)\n   */\n  @Input() channel?: Channel<DefaultStreamChatGenerics>;\n  /**\n   * The user the avatar belongs to (if avatar of a user is displayed)\n   */\n  @Input() user?: User<DefaultStreamChatGenerics>;\n  /**\n   * The type of the avatar: channel if channel avatar is displayed, user if user avatar is displayed\n   */\n  @Input() type: AvatarType | undefined;\n  /**\n   * If a channel avatar is displayed, and if the channel has exactly two members a green dot is displayed if the other member is online. Set this flag to `false` to turn off this behavior.\n   */\n  @Input() showOnlineIndicator = true;\n  /**\n   * If channel/user image isn't provided the initials of the name of the channel/user is shown instead, you can choose how the initals should be computed\n   */\n  @Input() initialsType:\n    | 'first-letter-of-first-word'\n    | 'first-letter-of-each-word' = 'first-letter-of-first-word';\n  isError = false;\n  isOnline = false;\n  private isOnlineSubscription?: Subscription;\n  initials: string = '';\n  fallbackChannelImage: string | undefined;\n  private userId?: string;\n  private isViewInited = false;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private ngZone: NgZone,\n    private cdRef: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((u) => {\n        if (u?.id !== this.userId) {\n          this.userId = u?.id;\n          if (this.type || this.channel || this.name) {\n            this.setInitials();\n            this.setFallbackChannelImage();\n            this.updateIsOnlineSubscription();\n          }\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        }\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['channel']) {\n      this.updateIsOnlineSubscription();\n    }\n    if (changes.type || changes.name || changes.channel) {\n      this.setInitials();\n    }\n\n    if (changes.type || changes.channel) {\n      this.setFallbackChannelImage();\n    }\n  }\n\n  private setFallbackChannelImage() {\n    if (this.type !== 'channel') {\n      this.fallbackChannelImage = undefined;\n    } else {\n      const otherMember = this.getOtherMemberIfOneToOneChannel();\n      if (otherMember) {\n        this.fallbackChannelImage = otherMember.image;\n      } else {\n        this.fallbackChannelImage = undefined;\n      }\n    }\n  }\n\n  private setInitials() {\n    let result: string = '';\n    if (this.type === 'user') {\n      result = this.name?.toString() || '';\n    } else if (this.type === 'channel') {\n      if (this.channel?.data?.name) {\n        result = this.channel?.data?.name;\n      } else {\n        const otherMember = this.getOtherMemberIfOneToOneChannel();\n        if (otherMember) {\n          result = otherMember.name || otherMember.id || '';\n        } else {\n          result = '#';\n        }\n      }\n    }\n\n    const words = result.split(' ');\n    let initials: string;\n    if (this.initialsType === 'first-letter-of-each-word') {\n      initials = words.map((w) => w.charAt(0) || '').join('');\n    } else {\n      initials = words[0].charAt(0) || '';\n    }\n    this.initials = initials;\n  }\n\n  private updateIsOnlineSubscription() {\n    if (this.channel) {\n      const otherMember = this.getOtherMemberIfOneToOneChannel();\n      if (otherMember) {\n        this.isOnline = otherMember.online || false;\n        this.isOnlineSubscription = this.chatClientService.events$\n          .pipe(filter((e) => e.eventType === 'user.presence.changed'))\n          .subscribe((event) => {\n            if (event.event.user?.id === otherMember.id) {\n              this.ngZone.run(() => {\n                this.isOnline = event.event.user?.online || false;\n              });\n            }\n          });\n      } else {\n        this.isOnline = false;\n        this.isOnlineSubscription?.unsubscribe();\n      }\n    } else {\n      this.isOnline = false;\n      this.isOnlineSubscription?.unsubscribe();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n  }\n\n  private getOtherMemberIfOneToOneChannel() {\n    const otherMembers = Object.values(\n      this.channel?.state?.members || {}\n    ).filter((m) => m.user_id !== this.userId);\n    if (otherMembers.length === 1) {\n      return otherMembers[0].user;\n    } else {\n      return undefined;\n    }\n  }\n}\n","<div\n  class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n    location\n  }}\"\n  title=\"{{ name }}\"\n  [ngStyle]=\"{\n    flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    fontSize:\n      initialsType === 'first-letter-of-first-word'\n        ? 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')'\n        : 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 3 + ')',\n    height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n  }\"\n>\n  <img\n    *ngIf=\"(imageUrl || fallbackChannelImage) && !isError; else fallback\"\n    class=\"str-chat__avatar-image str-chat__avatar-image\"\n    src=\"{{ imageUrl || fallbackChannelImage }}\"\n    alt=\"{{ initials }}\"\n    data-testid=\"avatar-img\"\n    (error)=\"isError = true\"\n    [ngStyle]=\"{\n      flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n      height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n      objectFit: 'cover',\n      width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n    }\"\n    fetchpriority=\"high\"\n  />\n  <ng-template #fallback>\n    <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n      {{ initials }}\n    </div>\n  </ng-template>\n  <div\n    data-testid=\"online-indicator\"\n    *ngIf=\"isOnline && showOnlineIndicator\"\n    class=\"str-chat__avatar--online-indicator\"\n  ></div>\n</div>\n"]}