stream-chat-angular 4.66.0 → 5.0.0-v5.1

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 (157) 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 +422 -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 +255 -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} +8 -10
  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} +345 -433
  59. package/fesm2015/stream-chat-angular.mjs.map +1 -0
  60. package/fesm2020/stream-chat-angular.mjs +7128 -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 +1 -2
  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 -1
  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/theme.service.d.ts +1 -2
  97. package/lib/thread/thread.component.d.ts +1 -1
  98. package/lib/types.d.ts +18 -18
  99. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +2 -2
  100. package/lib/voice-recording/voice-recording.component.d.ts +1 -1
  101. package/package.json +28 -15
  102. package/src/assets/version.ts +1 -1
  103. package/bundles/stream-chat-angular.umd.js +0 -8425
  104. package/bundles/stream-chat-angular.umd.js.map +0 -1
  105. package/esm2015/lib/attachment-configuration.service.js +0 -166
  106. package/esm2015/lib/attachment-list/attachment-list.component.js +0 -209
  107. package/esm2015/lib/attachment-preview-list/attachment-preview-list.component.js +0 -49
  108. package/esm2015/lib/attachment.service.js +0 -276
  109. package/esm2015/lib/avatar/avatar.component.js +0 -172
  110. package/esm2015/lib/avatar-placeholder/avatar-placeholder.component.js +0 -78
  111. package/esm2015/lib/channel/channel.component.js +0 -50
  112. package/esm2015/lib/channel-header/channel-header.component.js +0 -86
  113. package/esm2015/lib/channel-list/channel-list-toggle.service.js +0 -73
  114. package/esm2015/lib/channel-list/channel-list.component.js +0 -67
  115. package/esm2015/lib/channel-preview/channel-preview.component.js +0 -167
  116. package/esm2015/lib/channel.service.js +0 -1487
  117. package/esm2015/lib/chat-client.service.js +0 -211
  118. package/esm2015/lib/edit-message-form/edit-message-form.component.js +0 -87
  119. package/esm2015/lib/get-channel-display-text.js +0 -15
  120. package/esm2015/lib/get-message-translation.js +0 -13
  121. package/esm2015/lib/icon/icon.component.js +0 -25
  122. package/esm2015/lib/icon-placeholder/icon-placeholder.component.js +0 -35
  123. package/esm2015/lib/loading-indicator/loading-indicator.component.js +0 -35
  124. package/esm2015/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.js +0 -42
  125. package/esm2015/lib/message/message.component.js +0 -436
  126. package/esm2015/lib/message-actions-box/message-actions-box.component.js +0 -137
  127. package/esm2015/lib/message-actions.service.js +0 -114
  128. package/esm2015/lib/message-bounce-prompt/message-bounce-prompt.component.js +0 -80
  129. package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +0 -262
  130. package/esm2015/lib/message-input/message-input.component.js +0 -455
  131. package/esm2015/lib/message-input/textarea.directive.js +0 -90
  132. package/esm2015/lib/message-list/group-styles.js +0 -53
  133. package/esm2015/lib/message-list/message-list.component.js +0 -726
  134. package/esm2015/lib/message-preview.js +0 -7
  135. package/esm2015/lib/message-reactions/message-reactions.component.js +0 -266
  136. package/esm2015/lib/modal/modal.component.js +0 -74
  137. package/esm2015/lib/notification/notification.component.js +0 -24
  138. package/esm2015/lib/notification-list/notification-list.component.js +0 -38
  139. package/esm2015/lib/notification.service.js +0 -79
  140. package/esm2015/lib/read-by.js +0 -13
  141. package/esm2015/lib/theme.service.js +0 -122
  142. package/esm2015/lib/thread/thread.component.js +0 -55
  143. package/esm2015/lib/types.js +0 -2
  144. package/esm2015/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.js +0 -192
  145. package/esm2015/lib/voice-recording/voice-recording.component.js +0 -115
  146. package/fesm2015/stream-chat-angular.js.map +0 -1
  147. /package/{esm2015/assets/i18n/en.js → esm2020/assets/i18n/en.mjs} +0 -0
  148. /package/{esm2015/lib/injection-tokens.js → esm2020/lib/injection-tokens.mjs} +0 -0
  149. /package/{esm2015/lib/is-image-attachment.js → esm2020/lib/is-image-attachment.mjs} +0 -0
  150. /package/{esm2015/lib/is-image-file.js → esm2020/lib/is-image-file.mjs} +0 -0
  151. /package/{esm2015/lib/is-on-separate-date.js → esm2020/lib/is-on-separate-date.mjs} +0 -0
  152. /package/{esm2015/lib/list-users.js → esm2020/lib/list-users.mjs} +0 -0
  153. /package/{esm2015/lib/message-input/textarea.interface.js → esm2020/lib/message-input/textarea.interface.mjs} +0 -0
  154. /package/{esm2015/lib/parse-date.js → esm2020/lib/parse-date.mjs} +0 -0
  155. /package/{esm2015/public-api.js → esm2020/public-api.mjs} +0 -0
  156. /package/{esm2015/stream-chat-angular.js → esm2020/stream-chat-angular.mjs} +0 -0
  157. /package/{stream-chat-angular.d.ts → index.d.ts} +0 -0
@@ -1,166 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- /**
4
- * The `AttachmentConfigurationService` provides customization for certain attributes of attachments displayed inside the message component. If you're using your own CDN, you can integrate resizing features of it by providing your own handlers.
5
- */
6
- export class AttachmentConfigurationService {
7
- constructor() {
8
- /**
9
- * You can turn on/off thumbnail generation for video attachments
10
- */
11
- this.shouldGenerateVideoThumbnail = true;
12
- }
13
- /**
14
- * Handles the configuration for image attachments, it's possible to provide your own function to override the default logic
15
- * @param attachment The attachment to configure
16
- * @param location Specifies where the image is being displayed
17
- * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.
18
- */
19
- getImageAttachmentConfiguration(attachment, location, element) {
20
- if (this.customImageAttachmentConfigurationHandler) {
21
- return this.customImageAttachmentConfigurationHandler(attachment, location, element);
22
- }
23
- const url = new URL((attachment.img_url ||
24
- attachment.thumb_url ||
25
- attachment.image_url ||
26
- ''));
27
- const originalHeight = Number(url.searchParams.get('oh')) > 1
28
- ? Number(url.searchParams.get('oh'))
29
- : 1000000;
30
- const originalWidth = Number(url.searchParams.get('ow')) > 1
31
- ? Number(url.searchParams.get('ow'))
32
- : 1000000;
33
- const displayWarning = location === 'gallery' || location === 'single';
34
- const sizeRestriction = this.getSizingRestrictions(url, element, displayWarning);
35
- if (sizeRestriction) {
36
- // Apply 2x for retina displays
37
- sizeRestriction.height *= 2;
38
- sizeRestriction.width *= 2;
39
- this.addResizingParamsToUrl(sizeRestriction, url);
40
- }
41
- return {
42
- url: url.href,
43
- width: '',
44
- height: '',
45
- originalHeight,
46
- originalWidth,
47
- };
48
- }
49
- /**
50
- * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic
51
- * @param attachment The attachment to configure
52
- * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.
53
- */
54
- getVideoAttachmentConfiguration(attachment, element) {
55
- if (this.customVideoAttachmentConfigurationHandler) {
56
- return this.customVideoAttachmentConfigurationHandler(attachment, element);
57
- }
58
- let thumbUrl = undefined;
59
- let originalHeight = 1000000;
60
- let originalWidth = 1000000;
61
- if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {
62
- const url = new URL(attachment.thumb_url);
63
- originalHeight =
64
- Number(url.searchParams.get('oh')) > 1
65
- ? Number(url.searchParams.get('oh'))
66
- : originalHeight;
67
- originalWidth =
68
- Number(url.searchParams.get('ow')) > 1
69
- ? Number(url.searchParams.get('ow'))
70
- : originalWidth;
71
- const displayWarning = true;
72
- const sizeRestriction = this.getSizingRestrictions(url, element, displayWarning);
73
- if (sizeRestriction) {
74
- sizeRestriction.height *= 2;
75
- sizeRestriction.width *= 2;
76
- this.addResizingParamsToUrl(sizeRestriction, url);
77
- }
78
- thumbUrl = url.href;
79
- }
80
- return {
81
- url: attachment.asset_url || '',
82
- width: '',
83
- height: '',
84
- thumbUrl: thumbUrl,
85
- originalHeight,
86
- originalWidth,
87
- };
88
- }
89
- /**
90
- * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic
91
- * @param attachment The attachment to configure
92
- */
93
- getGiphyAttachmentConfiguration(attachment) {
94
- var _a;
95
- if (this.customGiphyAttachmentConfigurationHandler) {
96
- return this.customGiphyAttachmentConfigurationHandler(attachment);
97
- }
98
- const giphy = (_a = attachment.giphy) === null || _a === void 0 ? void 0 : _a.fixed_height_downsampled;
99
- return {
100
- url: (giphy === null || giphy === void 0 ? void 0 : giphy.url) || attachment.image_url || attachment.thumb_url || '',
101
- height: (giphy === null || giphy === void 0 ? void 0 : giphy.height) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.height}px` : '300px',
102
- width: (giphy === null || giphy === void 0 ? void 0 : giphy.width) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.width}px` : '',
103
- };
104
- }
105
- /**
106
- * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic
107
- * @param attachment The attachment to configure
108
- */
109
- getScrapedImageAttachmentConfiguration(attachment) {
110
- if (this.customScrapedImageAttachmentConfigurationHandler) {
111
- return this.customScrapedImageAttachmentConfigurationHandler(attachment);
112
- }
113
- return {
114
- url: attachment.image_url || attachment.thumb_url || '',
115
- width: '',
116
- height: '', // Set from CSS
117
- };
118
- }
119
- addResizingParamsToUrl(sizeRestriction, url) {
120
- url.searchParams.set('h', sizeRestriction.height.toString());
121
- url.searchParams.set('w', sizeRestriction.width.toString());
122
- }
123
- getSizingRestrictions(url, htmlElement, displayWarning = false) {
124
- const urlParams = url.searchParams;
125
- const originalHeight = Number(urlParams.get('oh')) || 1;
126
- const originalWidth = Number(urlParams.get('ow')) || 1;
127
- const cssSizeRestriction = this.getCSSSizeRestriction(htmlElement);
128
- let sizeRestriction;
129
- if ((cssSizeRestriction.maxHeight || cssSizeRestriction.height) &&
130
- cssSizeRestriction.maxWidth) {
131
- sizeRestriction = this.getSizeRestrictions(originalHeight, originalWidth, (cssSizeRestriction.maxHeight || cssSizeRestriction.height), cssSizeRestriction.maxWidth);
132
- }
133
- else {
134
- sizeRestriction = undefined;
135
- if (displayWarning) {
136
- console.warn(`Invalid value set for height/max-height and/or max-width for HTML element, this can cause scrolling issues inside the message list, more info https://getstream.io/chat/docs/sdk/angular/components/AttachmentListComponent/#image-and-video-sizing, attachment URL: ${url.toString()}`);
137
- }
138
- }
139
- return sizeRestriction;
140
- }
141
- getSizeRestrictions(originalHeight, originalWidth, maxHeight, maxWidth) {
142
- return {
143
- height: Math.round(Math.max(maxHeight, (maxWidth / originalWidth) * originalHeight)),
144
- width: Math.round(Math.max(maxHeight, (maxWidth / originalHeight) * originalWidth)),
145
- };
146
- }
147
- getCSSSizeRestriction(htmlElement) {
148
- const computedStylesheet = getComputedStyle(htmlElement);
149
- const height = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('height'));
150
- const maxHeight = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('max-height'));
151
- const maxWidth = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('max-width'));
152
- return { height, maxHeight, maxWidth };
153
- }
154
- getValueRepresentationOfCSSProperty(property) {
155
- return Number(property.replace('px', '')) || undefined;
156
- }
157
- }
158
- AttachmentConfigurationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
159
- AttachmentConfigurationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, providedIn: 'root' });
160
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, decorators: [{
161
- type: Injectable,
162
- args: [{
163
- providedIn: 'root',
164
- }]
165
- }] });
166
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment-configuration.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/attachment-configuration.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport {\n  AttachmentConfigration,\n  DefaultStreamChatGenerics,\n  ImageAttachmentConfiguration,\n  VideoAttachmentConfiguration,\n} from './types';\n\n/**\n * The `AttachmentConfigurationService` provides customization for certain attributes of attachments displayed inside the message component. If you're using your own CDN, you can integrate resizing features of it by providing your own handlers.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class AttachmentConfigurationService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * A custom handler can be provided to override the default image attachment (images uploaded from files) configuration. By default the SDK uses fixed image height (a size that's known before image is loaded), if you override that with dynamic image height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    type: 'gallery' | 'single' | 'carousel',\n    containerElement: HTMLElement\n  ) => ImageAttachmentConfiguration;\n  /**\n   * A custom handler can be provided to override the default video attachment (videos uploaded from files) configuration. By default the SDK uses fixed height (a size that's known before video is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customVideoAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    containerElement: HTMLElement\n  ) => VideoAttachmentConfiguration;\n  /**\n   * A custom handler can be provided to override the default giphy attachment (GIFs sent with the /giphy command) configuration. By default the SDK uses fixed height (a size that's known before the GIF is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customGiphyAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * A custom handler can be provided to override the default scraped image attachment (images found in links inside messages) configuration. By default the SDK uses fixed height (a size that's known before image is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customScrapedImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * You can turn on/off thumbnail generation for video attachments\n   */\n  shouldGenerateVideoThumbnail = true;\n\n  /**\n   * Handles the configuration for image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param location Specifies where the image is being displayed\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getImageAttachmentConfiguration(\n    attachment: Attachment<T>,\n    location: 'gallery' | 'single' | 'carousel',\n    element: HTMLElement\n  ): ImageAttachmentConfiguration {\n    if (this.customImageAttachmentConfigurationHandler) {\n      return this.customImageAttachmentConfigurationHandler(\n        attachment,\n        location,\n        element\n      );\n    }\n\n    const url = new URL(\n      (attachment.img_url ||\n        attachment.thumb_url ||\n        attachment.image_url ||\n        '') as string\n    );\n    const originalHeight =\n      Number(url.searchParams.get('oh')) > 1\n        ? Number(url.searchParams.get('oh'))\n        : 1000000;\n    const originalWidth =\n      Number(url.searchParams.get('ow')) > 1\n        ? Number(url.searchParams.get('ow'))\n        : 1000000;\n    const displayWarning = location === 'gallery' || location === 'single';\n    const sizeRestriction = this.getSizingRestrictions(\n      url,\n      element,\n      displayWarning\n    );\n\n    if (sizeRestriction) {\n      // Apply 2x for retina displays\n      sizeRestriction.height *= 2;\n      sizeRestriction.width *= 2;\n      this.addResizingParamsToUrl(sizeRestriction, url);\n    }\n\n    return {\n      url: url.href,\n      width: '', // Not set to respect responsive width\n      height: '', // Set from CSS\n      originalHeight,\n      originalWidth,\n    };\n  }\n\n  /**\n   * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getVideoAttachmentConfiguration(\n    attachment: Attachment<T>,\n    element: HTMLElement\n  ): VideoAttachmentConfiguration {\n    if (this.customVideoAttachmentConfigurationHandler) {\n      return this.customVideoAttachmentConfigurationHandler(\n        attachment,\n        element\n      );\n    }\n\n    let thumbUrl = undefined;\n    let originalHeight = 1000000;\n    let originalWidth = 1000000;\n    if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {\n      const url = new URL(attachment.thumb_url);\n      originalHeight =\n        Number(url.searchParams.get('oh')) > 1\n          ? Number(url.searchParams.get('oh'))\n          : originalHeight;\n      originalWidth =\n        Number(url.searchParams.get('ow')) > 1\n          ? Number(url.searchParams.get('ow'))\n          : originalWidth;\n      const displayWarning = true;\n      const sizeRestriction = this.getSizingRestrictions(\n        url,\n        element,\n        displayWarning\n      );\n\n      if (sizeRestriction) {\n        sizeRestriction.height *= 2;\n        sizeRestriction.width *= 2;\n        this.addResizingParamsToUrl(sizeRestriction, url);\n      }\n      thumbUrl = url.href;\n    }\n    return {\n      url: attachment.asset_url || '',\n      width: '', // Not set to respect responsive width\n      height: '', // Set from CSS\n      thumbUrl: thumbUrl,\n      originalHeight,\n      originalWidth,\n    };\n  }\n\n  /**\n   * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getGiphyAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customGiphyAttachmentConfigurationHandler) {\n      return this.customGiphyAttachmentConfigurationHandler(attachment);\n    }\n\n    const giphy = attachment.giphy?.fixed_height_downsampled;\n\n    return {\n      url: giphy?.url || attachment.image_url || attachment.thumb_url || '',\n      height: giphy?.height ? `${giphy?.height}px` : '300px',\n      width: giphy?.width ? `${giphy?.width}px` : '',\n    };\n  }\n\n  /**\n   * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getScrapedImageAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customScrapedImageAttachmentConfigurationHandler) {\n      return this.customScrapedImageAttachmentConfigurationHandler(attachment);\n    }\n\n    return {\n      url: attachment.image_url || attachment.thumb_url || '',\n      width: '',\n      height: '', // Set from CSS\n    };\n  }\n\n  private addResizingParamsToUrl(\n    sizeRestriction: { width: number; height: number },\n    url: URL\n  ) {\n    url.searchParams.set('h', sizeRestriction.height.toString());\n    url.searchParams.set('w', sizeRestriction.width.toString());\n  }\n\n  private getSizingRestrictions(\n    url: URL,\n    htmlElement: HTMLElement,\n    displayWarning = false\n  ) {\n    const urlParams = url.searchParams;\n    const originalHeight = Number(urlParams.get('oh')) || 1;\n    const originalWidth = Number(urlParams.get('ow')) || 1;\n    const cssSizeRestriction = this.getCSSSizeRestriction(htmlElement);\n    let sizeRestriction: { width: number; height: number } | undefined;\n\n    if (\n      (cssSizeRestriction.maxHeight || cssSizeRestriction.height) &&\n      cssSizeRestriction.maxWidth\n    ) {\n      sizeRestriction = this.getSizeRestrictions(\n        originalHeight,\n        originalWidth,\n        (cssSizeRestriction.maxHeight || cssSizeRestriction.height)!,\n        cssSizeRestriction.maxWidth\n      );\n    } else {\n      sizeRestriction = undefined;\n      if (displayWarning) {\n        console.warn(\n          `Invalid value set for height/max-height and/or max-width for HTML element, this can cause scrolling issues inside the message list, more info https://getstream.io/chat/docs/sdk/angular/components/AttachmentListComponent/#image-and-video-sizing, attachment URL: ${url.toString()}`\n        );\n      }\n    }\n\n    return sizeRestriction;\n  }\n\n  private getSizeRestrictions(\n    originalHeight: number,\n    originalWidth: number,\n    maxHeight: number,\n    maxWidth: number\n  ) {\n    return {\n      height: Math.round(\n        Math.max(maxHeight, (maxWidth / originalWidth) * originalHeight)\n      ),\n      width: Math.round(\n        Math.max(maxHeight, (maxWidth / originalHeight) * originalWidth)\n      ),\n    };\n  }\n\n  private getCSSSizeRestriction(htmlElement: HTMLElement) {\n    const computedStylesheet = getComputedStyle(htmlElement);\n    const height = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('height')\n    );\n    const maxHeight = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-height')\n    );\n    const maxWidth = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-width')\n    );\n\n    return { height, maxHeight, maxWidth };\n  }\n\n  private getValueRepresentationOfCSSProperty(property: string) {\n    return Number(property.replace('px', '')) || undefined;\n  }\n}\n"]}
@@ -1,209 +0,0 @@
1
- import { Component, EventEmitter, HostBinding, Input, Output, ViewChild, } from '@angular/core';
2
- import prettybytes from 'pretty-bytes';
3
- import { isImageAttachment } from '../is-image-attachment';
4
- import * as i0 from "@angular/core";
5
- import * as i1 from "../custom-templates.service";
6
- import * as i2 from "../channel.service";
7
- import * as i3 from "../attachment-configuration.service";
8
- import * as i4 from "../theme.service";
9
- import * as i5 from "../icon-placeholder/icon-placeholder.component";
10
- import * as i6 from "../voice-recording/voice-recording.component";
11
- import * as i7 from "../modal/modal.component";
12
- import * as i8 from "@angular/common";
13
- import * as i9 from "@ngx-translate/core";
14
- /**
15
- * The `AttachmentList` component displays the attachments of a message
16
- */
17
- export class AttachmentListComponent {
18
- constructor(customTemplatesService, channelService, attachmentConfigurationService, themeService) {
19
- this.customTemplatesService = customTemplatesService;
20
- this.channelService = channelService;
21
- this.attachmentConfigurationService = attachmentConfigurationService;
22
- /**
23
- * The attachments to display
24
- */
25
- this.attachments = [];
26
- /**
27
- * Emits the state of the image carousel window
28
- */
29
- this.imageModalStateChange = new EventEmitter();
30
- this.class = 'str-chat__attachment-list-angular-host';
31
- this.orderedAttachments = [];
32
- this.imagesToView = [];
33
- this.imagesToViewCurrentIndex = 0;
34
- this.attachmentConfigurations = new Map();
35
- this.themeVersion = themeService.themeVersion;
36
- }
37
- ngOnChanges(changes) {
38
- if (changes.attachments) {
39
- const images = this.attachments.filter(this.isImage);
40
- const containsGallery = images.length >= 2;
41
- this.orderedAttachments = [
42
- ...(containsGallery ? this.createGallery(images) : images),
43
- ...this.attachments.filter((a) => this.isVideo(a)),
44
- ...this.attachments.filter((a) => this.isVoiceMessage(a)),
45
- ...this.attachments.filter((a) => this.isFile(a)),
46
- ];
47
- this.attachmentConfigurations = new Map();
48
- // Display link attachments only if there are no other attachments
49
- // Giphy-s always sent without other attachments
50
- if (this.orderedAttachments.length === 0) {
51
- this.orderedAttachments.push(...this.attachments.filter((a) => this.isCard(a)));
52
- }
53
- }
54
- }
55
- trackByUrl(_, attachment) {
56
- return (attachment.image_url ||
57
- attachment.img_url ||
58
- attachment.asset_url ||
59
- attachment.thumb_url);
60
- }
61
- isImage(attachment) {
62
- return isImageAttachment(attachment);
63
- }
64
- isSvg(attachment) {
65
- const filename = attachment.fallback || '';
66
- return !!filename.toLowerCase().endsWith('.svg');
67
- }
68
- isFile(attachment) {
69
- return attachment.type === 'file';
70
- }
71
- isGallery(attachment) {
72
- return attachment.type === 'gallery';
73
- }
74
- isVideo(attachment) {
75
- return (attachment.type === 'video' &&
76
- attachment.asset_url &&
77
- !attachment.og_scrape_url // links from video share services (such as YouTube or Facebook) are can't be played
78
- );
79
- }
80
- isCard(attachment) {
81
- return (!attachment.type ||
82
- (attachment.type === 'image' && !this.isImage(attachment)) ||
83
- attachment.type === 'giphy');
84
- }
85
- isVoiceMessage(attachment) {
86
- return attachment.type === 'voiceRecording';
87
- }
88
- hasFileSize(attachment) {
89
- return (attachment.file_size && Number.isFinite(Number(attachment.file_size)));
90
- }
91
- getFileSize(attachment) {
92
- return prettybytes(Number(attachment.file_size));
93
- }
94
- getModalContext() {
95
- return {
96
- isOpen: this.imagesToView && this.imagesToView.length > 0,
97
- isOpenChangeHandler: (isOpen) => (isOpen ? null : this.closeImageModal()),
98
- content: this.modalContent,
99
- };
100
- }
101
- trimUrl(url) {
102
- if (url !== undefined && url !== null) {
103
- const [trimmedUrl] = url
104
- .replace(/^(?:https?:\/\/)?(?:www\.)?/i, '')
105
- .split('/');
106
- return trimmedUrl;
107
- }
108
- return null;
109
- }
110
- sendAction(action) {
111
- void this.channelService.sendAction(this.messageId, {
112
- [action.name]: action.value,
113
- }, this.parentMessageId);
114
- }
115
- trackByActionValue(_, item) {
116
- return item.value;
117
- }
118
- openImageModal(attachments, selectedIndex = 0) {
119
- this.imageModalStateChange.next('opened');
120
- this.imagesToView = attachments;
121
- this.imagesToViewCurrentIndex = selectedIndex;
122
- }
123
- stepImages(dir) {
124
- this.imagesToViewCurrentIndex += dir * 1;
125
- }
126
- trackByImageUrl(_, item) {
127
- return item.image_url || item.img_url || item.thumb_url;
128
- }
129
- getAttachmentContext(attachment) {
130
- return { attachment };
131
- }
132
- getImageAttachmentConfiguration(attachment, type, element) {
133
- const existingConfiguration = this.attachmentConfigurations.get(attachment);
134
- if (existingConfiguration) {
135
- return existingConfiguration;
136
- }
137
- const configuration = this.attachmentConfigurationService.getImageAttachmentConfiguration(attachment, type, element);
138
- this.attachmentConfigurations.set(attachment, configuration);
139
- return configuration;
140
- }
141
- getCarouselImageAttachmentConfiguration(attachment, element) {
142
- return this.attachmentConfigurationService.getImageAttachmentConfiguration(attachment, 'carousel', element);
143
- }
144
- getVideoAttachmentConfiguration(attachment, element) {
145
- const existingConfiguration = this.attachmentConfigurations.get(attachment);
146
- if (existingConfiguration) {
147
- return existingConfiguration;
148
- }
149
- const configuration = this.attachmentConfigurationService.getVideoAttachmentConfiguration(attachment, element);
150
- this.attachmentConfigurations.set(attachment, configuration);
151
- return configuration;
152
- }
153
- getCardAttachmentConfiguration(attachment) {
154
- const existingConfiguration = this.attachmentConfigurations.get(attachment);
155
- if (existingConfiguration) {
156
- return existingConfiguration;
157
- }
158
- if (attachment.type === 'giphy') {
159
- return this.attachmentConfigurationService.getGiphyAttachmentConfiguration(attachment);
160
- }
161
- else {
162
- const configuration = this.attachmentConfigurationService.getScrapedImageAttachmentConfiguration(attachment);
163
- this.attachmentConfigurations.set(attachment, configuration);
164
- return configuration;
165
- }
166
- }
167
- get isImageModalPrevButtonVisible() {
168
- return this.imagesToViewCurrentIndex !== 0;
169
- }
170
- get isImageModalNextButtonVisible() {
171
- return this.imagesToViewCurrentIndex !== this.imagesToView.length - 1;
172
- }
173
- createGallery(images) {
174
- return [
175
- {
176
- type: 'gallery',
177
- images,
178
- },
179
- ];
180
- }
181
- closeImageModal() {
182
- this.imageModalStateChange.next('closed');
183
- this.imagesToView = [];
184
- }
185
- }
186
- AttachmentListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentListComponent, deps: [{ token: i1.CustomTemplatesService }, { token: i2.ChannelService }, { token: i3.AttachmentConfigurationService }, { token: i4.ThemeService }], target: i0.ɵɵFactoryTarget.Component });
187
- AttachmentListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AttachmentListComponent, selector: "stream-attachment-list", inputs: { messageId: "messageId", parentMessageId: "parentMessageId", attachments: "attachments" }, outputs: { imageModalStateChange: "imageModalStateChange" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "modalContent", first: true, predicate: ["modalContent"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"orderedAttachments.length > 0\" class=\"str-chat__attachment-list\">\n <ng-container\n *ngFor=\"let attachment of orderedAttachments; trackBy: trackByUrl\"\n >\n <div\n data-testclass=\"attachment-container\"\n class=\"str-chat__message-attachment str-chat__message-attachment--{{\n attachment.type\n }} str-chat__message-attachment-dynamic-size\"\n [class.str-chat__message-attachment--card]=\"isCard(attachment)\"\n [class.str-chat-angular__message-attachment-file-single]=\"\n isFile(attachment)\n \"\n [class.str-chat__message-attachment--voice-recording]=\"\n isVoiceMessage(attachment)\n \"\n [class.str-chat__message-attachment-with-actions]=\"\n attachment.actions && attachment.actions.length > 0\n \"\n [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n >\n <ng-container *ngIf=\"isImage(attachment)\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.imageAttachmentTemplate$ | async) ||\n defaultImage;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultImage let-attachmentContext=\"attachment\">\n <img\n #imgElement\n class=\"str-chat__message-attachment--img\"\n data-testclass=\"image\"\n [src]=\"\n getImageAttachmentConfiguration(\n attachmentContext,\n 'single',\n imgElement\n ).url\n \"\n [alt]=\"attachmentContext?.fallback\"\n (click)=\"openImageModal([attachmentContext])\"\n (keyup.enter)=\"openImageModal([attachmentContext])\"\n [style.--original-height]=\"\n getImageAttachmentConfiguration(\n attachmentContext,\n 'single',\n imgElement\n ).originalHeight\n \"\n [style.--original-width]=\"\n getImageAttachmentConfiguration(\n attachmentContext,\n 'single',\n imgElement\n ).originalWidth\n \"\n [ngStyle]=\"{\n height: getImageAttachmentConfiguration(\n attachmentContext,\n 'single',\n imgElement\n ).height,\n width: getImageAttachmentConfiguration(\n attachmentContext,\n 'single',\n imgElement\n ).width\n }\"\n />\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"isGallery(attachment)\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.galleryAttachmentTemplate$ | async) ||\n defaultGallery;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultGallery let-attachmentContext=\"attachment\">\n <div\n class=\"str-chat__gallery\"\n data-testid=\"image-gallery\"\n [class.str-chat__gallery--square]=\"\n (attachmentContext?.images)!.length > 3\n \"\n [class.str-chat__gallery-two-rows]=\"\n (attachmentContext?.images)!.length > 2\n \"\n >\n <ng-container\n *ngFor=\"\n let galleryImage of attachmentContext.images;\n let index = index;\n let isLast = last;\n trackBy: trackByImageUrl\n \"\n >\n <button\n *ngIf=\"index < 3 || (index === 3 && isLast)\"\n class=\"str-chat__gallery-image\"\n data-testclass=\"gallery-image\"\n (click)=\"openImageModal(attachmentContext.images!, index)\"\n (keyup.enter)=\"openImageModal(attachmentContext.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\n \"\n >\n <img\n fetchpriority=\"low\"\n loading=\"lazy\"\n #imgElement\n [src]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).url\n \"\n [alt]=\"galleryImage.fallback\"\n [style.--original-height]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).originalHeight\n \"\n [style.--original-width]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).originalWidth\n \"\n [ngStyle]=\"{\n height: getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).height,\n width: getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).width\n }\"\n />\n </button>\n <button\n #element\n *ngIf=\"index === 3 && !isLast\"\n class=\"str-chat__gallery-placeholder\"\n data-testclass=\"gallery-image\"\n data-testid=\"more-image-button\"\n (click)=\"openImageModal(attachmentContext.images!, index)\"\n (keyup.enter)=\"openImageModal(attachmentContext.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\n \"\n [style.--original-height]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n element\n ).originalHeight\n \"\n [style.--original-width]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n element\n ).originalWidth\n \"\n [ngStyle]=\"{\n 'background-image':\n 'url(' +\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n element\n ).url +\n ')',\n height: getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n element\n ).height,\n width: getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n element\n ).width\n }\"\n >\n <p\n [innerHTML]=\"\n 'streamChat.{{ imageCount }} more'\n | translate\n : { imageCount: attachmentContext!.images!.length - 4 }\n \"\n ></p>\n </button>\n </ng-container>\n </div>\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"isVideo(attachment)\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.videoAttachmentTemplate$ | async) ||\n defaultVideo;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultVideo let-attachmentContext=\"attachment\">\n <div\n class=\"str-chat__player-wrapper\"\n data-testclass=\"video-attachment-parent\"\n [style.--original-height]=\"\n getVideoAttachmentConfiguration(attachmentContext, videoElement)\n .originalHeight\n \"\n [style.--original-width]=\"\n getVideoAttachmentConfiguration(attachmentContext, videoElement)\n .originalWidth\n \"\n [ngStyle]=\"{\n height: getVideoAttachmentConfiguration(\n attachmentContext,\n videoElement\n ).height,\n width: getVideoAttachmentConfiguration(\n attachmentContext,\n videoElement\n ).width\n }\"\n >\n <video\n #videoElement\n class=\"str-chat__video-angular\"\n controls\n data-testclass=\"video-attachment\"\n [src]=\"\n getVideoAttachmentConfiguration(attachmentContext, videoElement)\n .url\n \"\n [poster]=\"\n getVideoAttachmentConfiguration(attachmentContext, videoElement)\n .thumbUrl\n \"\n ></video>\n </div>\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"isFile(attachment)\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.fileAttachmentTemplate$ | async) ||\n defaultFile;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultFile let-attachmentContext=\"attachment\">\n <div\n class=\"\n str-chat__message-attachment-file--item\n str-chat-angular__message-attachment-file-single\n \"\n >\n <stream-icon-placeholder\n *ngIf=\"themeVersion === '1'\"\n icon=\"file\"\n [size]=\"30\"\n ></stream-icon-placeholder>\n <stream-icon-placeholder\n *ngIf=\"themeVersion === '2'\"\n icon=\"unspecified-filetype\"\n [size]=\"30\"\n ></stream-icon-placeholder>\n <div class=\"str-chat__message-attachment-file--item-text\">\n <a\n class=\"str-chat__message-attachment-file--item-first-row\"\n data-testclass=\"file-link\"\n href=\"{{ attachmentContext.asset_url }}\"\n target=\"_blank\"\n >\n <div\n data-testclass=\"file-title\"\n class=\"str-chat__message-attachment-file--item-name\"\n >\n {{ attachmentContext.title }}\n </div>\n <stream-icon-placeholder\n class=\"str-chat__message-attachment-download-icon\"\n icon=\"download\"\n ></stream-icon-placeholder>\n </a>\n <span\n class=\"str-chat__message-attachment-file--item-size\"\n data-testclass=\"size\"\n *ngIf=\"hasFileSize(attachmentContext)\"\n >{{ getFileSize(attachmentContext) }}</span\n >\n </div>\n </div>\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"isVoiceMessage(attachment)\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.voiceRecordingAttachmentTemplate$\n | async) || defaultRecording;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultRecording>\n <stream-voice-recording\n data-testclass=\"voice-recording\"\n [attachment]=\"attachment\"\n ></stream-voice-recording>\n </ng-template>\n </ng-container>\n <ng-container\n *ngIf=\"\n isCard(attachment) &&\n getCardAttachmentConfiguration(attachment) as attachmentConfiguration\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.cardAttachmentTemplate$ | async) ||\n defaultCard;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultCard let-attachmentContext=\"attachment\">\n <div\n class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n attachmentContext.type\n }}\"\n >\n <div\n *ngIf=\"attachmentConfiguration.url\"\n class=\"str-chat__message-attachment-card--header\"\n >\n <img\n fetchpriority=\"low\"\n loading=\"lazy\"\n data-testclass=\"card-img\"\n alt=\"{{ attachmentConfiguration.url }}\"\n src=\"{{ attachmentConfiguration.url }}\"\n [ngStyle]=\"{\n height: attachmentConfiguration.height,\n width: attachmentConfiguration.width\n }\"\n />\n </div>\n <div class=\"str-chat__message-attachment-card--content\">\n <div class=\"str-chat__message-attachment-card--flex\">\n <div\n *ngIf=\"attachmentContext.title\"\n data-testclass=\"card-title\"\n class=\"str-chat__message-attachment-card--title\"\n >\n {{ attachmentContext.title }}\n </div>\n <div\n *ngIf=\"attachmentContext.text\"\n class=\"str-chat__message-attachment-card--text\"\n data-testclass=\"card-text\"\n >\n {{ attachmentContext.text }}\n </div>\n <a\n class=\"str-chat__message-attachment-card--url\"\n *ngIf=\"\n attachmentContext.title_link ||\n attachmentContext.og_scrape_url\n \"\n data-testclass=\"url-link\"\n noopener\n noreferrer\n href=\"{{\n attachmentContext.title_link ||\n attachmentContext.og_scrape_url\n }}\"\n target=\"_blank\"\n >\n {{\n trimUrl(\n attachmentContext.title_link ||\n attachmentContext.og_scrape_url\n )\n }}\n </a>\n </div>\n </div>\n </div>\n </ng-template>\n </ng-container>\n <ng-container *ngIf=\"attachment.actions && attachment.actions.length > 0\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.attachmentActionsTemplate$ | async) ||\n defaultActions;\n context: getAttachmentContext(attachment)\n \"\n ></ng-container>\n <ng-template #defaultActions let-attachmentContext=\"attachment\">\n <div class=\"str-chat__message-attachment-actions\">\n <div class=\"str-chat__message-attachment-actions-form\">\n <button\n *ngFor=\"\n let action of attachmentContext.actions;\n trackBy: trackByActionValue\n \"\n class=\"str-chat__message-attachment-actions-button str-chat__message-attachment-actions-button--{{\n action.style\n }}\"\n data-testclass=\"attachment-action\"\n (click)=\"sendAction(action)\"\n (keyup.enter)=\"sendAction(action)\"\n >\n {{ action.text }}\n </button>\n </div>\n </div>\n </ng-template>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"imagesToView && imagesToView.length > 0\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: getModalContext()\n \"\n ></ng-container>\n </ng-container>\n</div>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n class=\"stream-chat-angular__image-modal-host\"\n [isOpen]=\"isOpen\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n [content]=\"content\"\n >\n </stream-modal>\n</ng-template>\n\n<ng-template #modalContent>\n <div class=\"stream-chat-angular__image-modal str-chat__image-carousel\">\n <img\n #imgElement\n class=\"\n stream-chat-angular__image-modal-image\n str-chat__image-carousel-image\n \"\n data-testid=\"modal-image\"\n [src]=\"\n getCarouselImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n imgElement\n ).url\n \"\n [style.--original-height]=\"\n getCarouselImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n imgElement\n ).originalHeight\n \"\n [style.--original-width]=\"\n getCarouselImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n imgElement\n ).originalWidth\n \"\n [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n [ngStyle]=\"{\n width: getCarouselImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n imgElement\n ).width,\n height: getCarouselImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n imgElement\n ).height\n }\"\n />\n <div>\n <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper str-chat__image-carousel-stepper-prev\n \"\n [ngStyle]=\"{\n visibility: isImageModalPrevButtonVisible ? 'visible' : 'hidden'\n }\"\n data-testid=\"image-modal-prev\"\n type=\"button\"\n (click)=\"stepImages(-1)\"\n (keyup.enter)=\"stepImages(-1)\"\n >\n <stream-icon-placeholder icon=\"arrow-left\"></stream-icon-placeholder>\n </button>\n <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper str-chat__image-carousel-stepper-next\n \"\n type=\"button\"\n [ngStyle]=\"{\n visibility: isImageModalNextButtonVisible ? 'visible' : 'hidden'\n }\"\n data-testid=\"image-modal-next\"\n (click)=\"stepImages(1)\"\n (keyup.enter)=\"stepImages(1)\"\n >\n <stream-icon-placeholder icon=\"arrow-right\"></stream-icon-placeholder>\n </button>\n </div>\n </div>\n</ng-template>\n", components: [{ type: i5.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon", "size"] }, { type: i6.VoiceRecordingComponent, selector: "stream-voice-recording", inputs: ["attachment"] }, { type: i7.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i8.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i8.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }], pipes: { "async": i8.AsyncPipe, "translate": i9.TranslatePipe } });
188
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentListComponent, decorators: [{
189
- type: Component,
190
- args: [{
191
- selector: 'stream-attachment-list',
192
- templateUrl: './attachment-list.component.html',
193
- styles: [],
194
- }]
195
- }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ChannelService }, { type: i3.AttachmentConfigurationService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
196
- type: Input
197
- }], parentMessageId: [{
198
- type: Input
199
- }], attachments: [{
200
- type: Input
201
- }], imageModalStateChange: [{
202
- type: Output
203
- }], class: [{
204
- type: HostBinding
205
- }], modalContent: [{
206
- type: ViewChild,
207
- args: ['modalContent', { static: true }]
208
- }] } });
209
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment-list.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/attachment-list/attachment-list.component.ts","../../../../../projects/stream-chat-angular/src/lib/attachment-list/attachment-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { Action, Attachment } from 'stream-chat';\nimport {\n  ModalContext,\n  DefaultStreamChatGenerics,\n  AttachmentConfigration,\n  VideoAttachmentConfiguration,\n  ImageAttachmentConfiguration,\n  AttachmentContext,\n} from '../types';\nimport prettybytes from 'pretty-bytes';\nimport { isImageAttachment } from '../is-image-attachment';\nimport { ChannelService } from '../channel.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { AttachmentConfigurationService } from '../attachment-configuration.service';\nimport { ThemeService } from '../theme.service';\n\n/**\n * The `AttachmentList` component displays the attachments of a message\n */\n@Component({\n  selector: 'stream-attachment-list',\n  templateUrl: './attachment-list.component.html',\n  styles: [],\n})\nexport class AttachmentListComponent implements OnChanges {\n  /**\n   * The id of the message the attachments belong to\n   */\n  @Input() messageId: string | undefined;\n  /**\n   * The parent id of the message the attachments belong to\n   */\n  @Input() parentMessageId: string | undefined;\n  /**\n   * The attachments to display\n   */\n  @Input() attachments: Attachment<DefaultStreamChatGenerics>[] = [];\n  /**\n   * Emits the state of the image carousel window\n   */\n  @Output() readonly imageModalStateChange = new EventEmitter<\n    'opened' | 'closed'\n  >();\n  @HostBinding() class = 'str-chat__attachment-list-angular-host';\n  orderedAttachments: Attachment<DefaultStreamChatGenerics>[] = [];\n  imagesToView: Attachment<DefaultStreamChatGenerics>[] = [];\n  imagesToViewCurrentIndex = 0;\n  themeVersion: '1' | '2';\n  @ViewChild('modalContent', { static: true })\n  private modalContent!: TemplateRef<void>;\n  private attachmentConfigurations: Map<\n    Attachment,\n    | AttachmentConfigration\n    | VideoAttachmentConfiguration\n    | ImageAttachmentConfiguration\n  > = new Map();\n\n  constructor(\n    public readonly customTemplatesService: CustomTemplatesService,\n    private channelService: ChannelService,\n    private attachmentConfigurationService: AttachmentConfigurationService,\n    themeService: ThemeService\n  ) {\n    this.themeVersion = themeService.themeVersion;\n  }\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.attachments) {\n      const images = this.attachments.filter(this.isImage);\n      const containsGallery = images.length >= 2;\n      this.orderedAttachments = [\n        ...(containsGallery ? this.createGallery(images) : images),\n        ...this.attachments.filter((a) => this.isVideo(a)),\n        ...this.attachments.filter((a) => this.isVoiceMessage(a)),\n        ...this.attachments.filter((a) => this.isFile(a)),\n      ];\n      this.attachmentConfigurations = new Map();\n      // Display link attachments only if there are no other attachments\n      // Giphy-s always sent without other attachments\n      if (this.orderedAttachments.length === 0) {\n        this.orderedAttachments.push(\n          ...this.attachments.filter((a) => this.isCard(a))\n        );\n      }\n    }\n  }\n\n  trackByUrl(_: number, attachment: Attachment) {\n    return (\n      attachment.image_url ||\n      attachment.img_url ||\n      attachment.asset_url ||\n      attachment.thumb_url\n    );\n  }\n\n  isImage(attachment: Attachment) {\n    return isImageAttachment(attachment);\n  }\n\n  isSvg(attachment: Attachment) {\n    const filename = attachment.fallback || '';\n    return !!filename.toLowerCase().endsWith('.svg');\n  }\n\n  isFile(attachment: Attachment) {\n    return attachment.type === 'file';\n  }\n\n  isGallery(attachment: Attachment) {\n    return attachment.type === 'gallery';\n  }\n\n  isVideo(attachment: Attachment) {\n    return (\n      attachment.type === 'video' &&\n      attachment.asset_url &&\n      !attachment.og_scrape_url // links from video share services (such as YouTube or Facebook) are can't be played\n    );\n  }\n\n  isCard(attachment: Attachment) {\n    return (\n      !attachment.type ||\n      (attachment.type === 'image' && !this.isImage(attachment)) ||\n      attachment.type === 'giphy'\n    );\n  }\n\n  isVoiceMessage(attachment: Attachment) {\n    return attachment.type === 'voiceRecording';\n  }\n\n  hasFileSize(attachment: Attachment<DefaultStreamChatGenerics>) {\n    return (\n      attachment.file_size && Number.isFinite(Number(attachment.file_size))\n    );\n  }\n\n  getFileSize(attachment: Attachment<DefaultStreamChatGenerics>) {\n    return prettybytes(Number(attachment.file_size!));\n  }\n\n  getModalContext(): ModalContext {\n    return {\n      isOpen: this.imagesToView && this.imagesToView.length > 0,\n      isOpenChangeHandler: (isOpen) => (isOpen ? null : this.closeImageModal()),\n      content: this.modalContent,\n    };\n  }\n\n  trimUrl(url?: string | null) {\n    if (url !== undefined && url !== null) {\n      const [trimmedUrl] = url\n        .replace(/^(?:https?:\\/\\/)?(?:www\\.)?/i, '')\n        .split('/');\n\n      return trimmedUrl;\n    }\n    return null;\n  }\n\n  sendAction(action: Action) {\n    void this.channelService.sendAction(\n      this.messageId!,\n      {\n        [action.name!]: action.value!,\n      },\n      this.parentMessageId\n    );\n  }\n\n  trackByActionValue(_: number, item: Action) {\n    return item.value;\n  }\n\n  openImageModal(attachments: Attachment[], selectedIndex = 0) {\n    this.imageModalStateChange.next('opened');\n    this.imagesToView = attachments;\n    this.imagesToViewCurrentIndex = selectedIndex;\n  }\n\n  stepImages(dir: -1 | 1) {\n    this.imagesToViewCurrentIndex += dir * 1;\n  }\n\n  trackByImageUrl(_: number, item: Attachment) {\n    return item.image_url || item.img_url || item.thumb_url;\n  }\n\n  getAttachmentContext(\n    attachment: Attachment<DefaultStreamChatGenerics>\n  ): AttachmentContext {\n    return { attachment };\n  }\n\n  getImageAttachmentConfiguration(\n    attachment: Attachment,\n    type: 'gallery' | 'single',\n    element: HTMLElement\n  ): ImageAttachmentConfiguration {\n    const existingConfiguration = this.attachmentConfigurations.get(attachment);\n    if (existingConfiguration) {\n      return existingConfiguration as ImageAttachmentConfiguration;\n    }\n    const configuration =\n      this.attachmentConfigurationService.getImageAttachmentConfiguration(\n        attachment,\n        type,\n        element\n      );\n    this.attachmentConfigurations.set(attachment, configuration);\n    return configuration;\n  }\n\n  getCarouselImageAttachmentConfiguration(\n    attachment: Attachment,\n    element: HTMLElement\n  ) {\n    return this.attachmentConfigurationService.getImageAttachmentConfiguration(\n      attachment,\n      'carousel',\n      element\n    );\n  }\n\n  getVideoAttachmentConfiguration(\n    attachment: Attachment,\n    element: HTMLElement\n  ): VideoAttachmentConfiguration {\n    const existingConfiguration = this.attachmentConfigurations.get(attachment);\n    if (existingConfiguration) {\n      return existingConfiguration as VideoAttachmentConfiguration;\n    }\n    const configuration =\n      this.attachmentConfigurationService.getVideoAttachmentConfiguration(\n        attachment,\n        element\n      );\n    this.attachmentConfigurations.set(attachment, configuration);\n    return configuration;\n  }\n\n  getCardAttachmentConfiguration(attachment: Attachment) {\n    const existingConfiguration = this.attachmentConfigurations.get(attachment);\n    if (existingConfiguration) {\n      return existingConfiguration;\n    }\n    if (attachment.type === 'giphy') {\n      return this.attachmentConfigurationService.getGiphyAttachmentConfiguration(\n        attachment\n      );\n    } else {\n      const configuration =\n        this.attachmentConfigurationService.getScrapedImageAttachmentConfiguration(\n          attachment\n        );\n      this.attachmentConfigurations.set(attachment, configuration);\n      return configuration;\n    }\n  }\n\n  get isImageModalPrevButtonVisible() {\n    return this.imagesToViewCurrentIndex !== 0;\n  }\n\n  get isImageModalNextButtonVisible() {\n    return this.imagesToViewCurrentIndex !== this.imagesToView.length - 1;\n  }\n\n  private createGallery(images: Attachment[]) {\n    return [\n      {\n        type: 'gallery',\n        images,\n      },\n    ];\n  }\n\n  private closeImageModal() {\n    this.imageModalStateChange.next('closed');\n    this.imagesToView = [];\n  }\n}\n","<div *ngIf=\"orderedAttachments.length > 0\" class=\"str-chat__attachment-list\">\n  <ng-container\n    *ngFor=\"let attachment of orderedAttachments; trackBy: trackByUrl\"\n  >\n    <div\n      data-testclass=\"attachment-container\"\n      class=\"str-chat__message-attachment str-chat__message-attachment--{{\n        attachment.type\n      }} str-chat__message-attachment-dynamic-size\"\n      [class.str-chat__message-attachment--card]=\"isCard(attachment)\"\n      [class.str-chat-angular__message-attachment-file-single]=\"\n        isFile(attachment)\n      \"\n      [class.str-chat__message-attachment--voice-recording]=\"\n        isVoiceMessage(attachment)\n      \"\n      [class.str-chat__message-attachment-with-actions]=\"\n        attachment.actions && attachment.actions.length > 0\n      \"\n      [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n    >\n      <ng-container *ngIf=\"isImage(attachment)\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.imageAttachmentTemplate$ | async) ||\n              defaultImage;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultImage let-attachmentContext=\"attachment\">\n          <img\n            #imgElement\n            class=\"str-chat__message-attachment--img\"\n            data-testclass=\"image\"\n            [src]=\"\n              getImageAttachmentConfiguration(\n                attachmentContext,\n                'single',\n                imgElement\n              ).url\n            \"\n            [alt]=\"attachmentContext?.fallback\"\n            (click)=\"openImageModal([attachmentContext])\"\n            (keyup.enter)=\"openImageModal([attachmentContext])\"\n            [style.--original-height]=\"\n              getImageAttachmentConfiguration(\n                attachmentContext,\n                'single',\n                imgElement\n              ).originalHeight\n            \"\n            [style.--original-width]=\"\n              getImageAttachmentConfiguration(\n                attachmentContext,\n                'single',\n                imgElement\n              ).originalWidth\n            \"\n            [ngStyle]=\"{\n              height: getImageAttachmentConfiguration(\n                attachmentContext,\n                'single',\n                imgElement\n              ).height,\n              width: getImageAttachmentConfiguration(\n                attachmentContext,\n                'single',\n                imgElement\n              ).width\n            }\"\n          />\n        </ng-template>\n      </ng-container>\n      <ng-container *ngIf=\"isGallery(attachment)\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.galleryAttachmentTemplate$ | async) ||\n              defaultGallery;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultGallery let-attachmentContext=\"attachment\">\n          <div\n            class=\"str-chat__gallery\"\n            data-testid=\"image-gallery\"\n            [class.str-chat__gallery--square]=\"\n              (attachmentContext?.images)!.length > 3\n            \"\n            [class.str-chat__gallery-two-rows]=\"\n              (attachmentContext?.images)!.length > 2\n            \"\n          >\n            <ng-container\n              *ngFor=\"\n                let galleryImage of attachmentContext.images;\n                let index = index;\n                let isLast = last;\n                trackBy: trackByImageUrl\n              \"\n            >\n              <button\n                *ngIf=\"index < 3 || (index === 3 && isLast)\"\n                class=\"str-chat__gallery-image\"\n                data-testclass=\"gallery-image\"\n                (click)=\"openImageModal(attachmentContext.images!, index)\"\n                (keyup.enter)=\"openImageModal(attachmentContext.images!, index)\"\n                [class.str-chat__message-attachment--svg-image]=\"\n                  isSvg(galleryImage)\n                \"\n              >\n                <img\n                  fetchpriority=\"low\"\n                  loading=\"lazy\"\n                  #imgElement\n                  [src]=\"\n                    getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      imgElement\n                    ).url\n                  \"\n                  [alt]=\"galleryImage.fallback\"\n                  [style.--original-height]=\"\n                    getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      imgElement\n                    ).originalHeight\n                  \"\n                  [style.--original-width]=\"\n                    getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      imgElement\n                    ).originalWidth\n                  \"\n                  [ngStyle]=\"{\n                    height: getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      imgElement\n                    ).height,\n                    width: getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      imgElement\n                    ).width\n                  }\"\n                />\n              </button>\n              <button\n                #element\n                *ngIf=\"index === 3 && !isLast\"\n                class=\"str-chat__gallery-placeholder\"\n                data-testclass=\"gallery-image\"\n                data-testid=\"more-image-button\"\n                (click)=\"openImageModal(attachmentContext.images!, index)\"\n                (keyup.enter)=\"openImageModal(attachmentContext.images!, index)\"\n                [class.str-chat__message-attachment--svg-image]=\"\n                  isSvg(galleryImage)\n                \"\n                [style.--original-height]=\"\n                  getImageAttachmentConfiguration(\n                    galleryImage,\n                    'gallery',\n                    element\n                  ).originalHeight\n                \"\n                [style.--original-width]=\"\n                  getImageAttachmentConfiguration(\n                    galleryImage,\n                    'gallery',\n                    element\n                  ).originalWidth\n                \"\n                [ngStyle]=\"{\n                  'background-image':\n                    'url(' +\n                    getImageAttachmentConfiguration(\n                      galleryImage,\n                      'gallery',\n                      element\n                    ).url +\n                    ')',\n                  height: getImageAttachmentConfiguration(\n                    galleryImage,\n                    'gallery',\n                    element\n                  ).height,\n                  width: getImageAttachmentConfiguration(\n                    galleryImage,\n                    'gallery',\n                    element\n                  ).width\n                }\"\n              >\n                <p\n                  [innerHTML]=\"\n                    'streamChat.{{ imageCount }} more'\n                      | translate\n                        : { imageCount: attachmentContext!.images!.length - 4 }\n                  \"\n                ></p>\n              </button>\n            </ng-container>\n          </div>\n        </ng-template>\n      </ng-container>\n      <ng-container *ngIf=\"isVideo(attachment)\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.videoAttachmentTemplate$ | async) ||\n              defaultVideo;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultVideo let-attachmentContext=\"attachment\">\n          <div\n            class=\"str-chat__player-wrapper\"\n            data-testclass=\"video-attachment-parent\"\n            [style.--original-height]=\"\n              getVideoAttachmentConfiguration(attachmentContext, videoElement)\n                .originalHeight\n            \"\n            [style.--original-width]=\"\n              getVideoAttachmentConfiguration(attachmentContext, videoElement)\n                .originalWidth\n            \"\n            [ngStyle]=\"{\n              height: getVideoAttachmentConfiguration(\n                attachmentContext,\n                videoElement\n              ).height,\n              width: getVideoAttachmentConfiguration(\n                attachmentContext,\n                videoElement\n              ).width\n            }\"\n          >\n            <video\n              #videoElement\n              class=\"str-chat__video-angular\"\n              controls\n              data-testclass=\"video-attachment\"\n              [src]=\"\n                getVideoAttachmentConfiguration(attachmentContext, videoElement)\n                  .url\n              \"\n              [poster]=\"\n                getVideoAttachmentConfiguration(attachmentContext, videoElement)\n                  .thumbUrl\n              \"\n            ></video>\n          </div>\n        </ng-template>\n      </ng-container>\n      <ng-container *ngIf=\"isFile(attachment)\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.fileAttachmentTemplate$ | async) ||\n              defaultFile;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultFile let-attachmentContext=\"attachment\">\n          <div\n            class=\"\n              str-chat__message-attachment-file--item\n              str-chat-angular__message-attachment-file-single\n            \"\n          >\n            <stream-icon-placeholder\n              *ngIf=\"themeVersion === '1'\"\n              icon=\"file\"\n              [size]=\"30\"\n            ></stream-icon-placeholder>\n            <stream-icon-placeholder\n              *ngIf=\"themeVersion === '2'\"\n              icon=\"unspecified-filetype\"\n              [size]=\"30\"\n            ></stream-icon-placeholder>\n            <div class=\"str-chat__message-attachment-file--item-text\">\n              <a\n                class=\"str-chat__message-attachment-file--item-first-row\"\n                data-testclass=\"file-link\"\n                href=\"{{ attachmentContext.asset_url }}\"\n                target=\"_blank\"\n              >\n                <div\n                  data-testclass=\"file-title\"\n                  class=\"str-chat__message-attachment-file--item-name\"\n                >\n                  {{ attachmentContext.title }}\n                </div>\n                <stream-icon-placeholder\n                  class=\"str-chat__message-attachment-download-icon\"\n                  icon=\"download\"\n                ></stream-icon-placeholder>\n              </a>\n              <span\n                class=\"str-chat__message-attachment-file--item-size\"\n                data-testclass=\"size\"\n                *ngIf=\"hasFileSize(attachmentContext)\"\n                >{{ getFileSize(attachmentContext) }}</span\n              >\n            </div>\n          </div>\n        </ng-template>\n      </ng-container>\n      <ng-container *ngIf=\"isVoiceMessage(attachment)\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.voiceRecordingAttachmentTemplate$\n              | async) || defaultRecording;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultRecording>\n          <stream-voice-recording\n            data-testclass=\"voice-recording\"\n            [attachment]=\"attachment\"\n          ></stream-voice-recording>\n        </ng-template>\n      </ng-container>\n      <ng-container\n        *ngIf=\"\n          isCard(attachment) &&\n          getCardAttachmentConfiguration(attachment) as attachmentConfiguration\n        \"\n      >\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.cardAttachmentTemplate$ | async) ||\n              defaultCard;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultCard let-attachmentContext=\"attachment\">\n          <div\n            class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n              attachmentContext.type\n            }}\"\n          >\n            <div\n              *ngIf=\"attachmentConfiguration.url\"\n              class=\"str-chat__message-attachment-card--header\"\n            >\n              <img\n                fetchpriority=\"low\"\n                loading=\"lazy\"\n                data-testclass=\"card-img\"\n                alt=\"{{ attachmentConfiguration.url }}\"\n                src=\"{{ attachmentConfiguration.url }}\"\n                [ngStyle]=\"{\n                  height: attachmentConfiguration.height,\n                  width: attachmentConfiguration.width\n                }\"\n              />\n            </div>\n            <div class=\"str-chat__message-attachment-card--content\">\n              <div class=\"str-chat__message-attachment-card--flex\">\n                <div\n                  *ngIf=\"attachmentContext.title\"\n                  data-testclass=\"card-title\"\n                  class=\"str-chat__message-attachment-card--title\"\n                >\n                  {{ attachmentContext.title }}\n                </div>\n                <div\n                  *ngIf=\"attachmentContext.text\"\n                  class=\"str-chat__message-attachment-card--text\"\n                  data-testclass=\"card-text\"\n                >\n                  {{ attachmentContext.text }}\n                </div>\n                <a\n                  class=\"str-chat__message-attachment-card--url\"\n                  *ngIf=\"\n                    attachmentContext.title_link ||\n                    attachmentContext.og_scrape_url\n                  \"\n                  data-testclass=\"url-link\"\n                  noopener\n                  noreferrer\n                  href=\"{{\n                    attachmentContext.title_link ||\n                      attachmentContext.og_scrape_url\n                  }}\"\n                  target=\"_blank\"\n                >\n                  {{\n                    trimUrl(\n                      attachmentContext.title_link ||\n                        attachmentContext.og_scrape_url\n                    )\n                  }}\n                </a>\n              </div>\n            </div>\n          </div>\n        </ng-template>\n      </ng-container>\n      <ng-container *ngIf=\"attachment.actions && attachment.actions.length > 0\">\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.attachmentActionsTemplate$ | async) ||\n              defaultActions;\n            context: getAttachmentContext(attachment)\n          \"\n        ></ng-container>\n        <ng-template #defaultActions let-attachmentContext=\"attachment\">\n          <div class=\"str-chat__message-attachment-actions\">\n            <div class=\"str-chat__message-attachment-actions-form\">\n              <button\n                *ngFor=\"\n                  let action of attachmentContext.actions;\n                  trackBy: trackByActionValue\n                \"\n                class=\"str-chat__message-attachment-actions-button str-chat__message-attachment-actions-button--{{\n                  action.style\n                }}\"\n                data-testclass=\"attachment-action\"\n                (click)=\"sendAction(action)\"\n                (keyup.enter)=\"sendAction(action)\"\n              >\n                {{ action.text }}\n              </button>\n            </div>\n          </div>\n        </ng-template>\n      </ng-container>\n    </div>\n  </ng-container>\n\n  <ng-container *ngIf=\"imagesToView && imagesToView.length > 0\">\n    <ng-container\n      *ngTemplateOutlet=\"\n        (customTemplatesService.modalTemplate$ | async) || defaultModal;\n        context: getModalContext()\n      \"\n    ></ng-container>\n  </ng-container>\n</div>\n\n<ng-template\n  #defaultModal\n  let-isOpen=\"isOpen\"\n  let-isOpenChangeHandler=\"isOpenChangeHandler\"\n  let-content=\"content\"\n>\n  <stream-modal\n    class=\"stream-chat-angular__image-modal-host\"\n    [isOpen]=\"isOpen\"\n    (isOpenChange)=\"isOpenChangeHandler($event)\"\n    [content]=\"content\"\n  >\n  </stream-modal>\n</ng-template>\n\n<ng-template #modalContent>\n  <div class=\"stream-chat-angular__image-modal str-chat__image-carousel\">\n    <img\n      #imgElement\n      class=\"\n        stream-chat-angular__image-modal-image\n        str-chat__image-carousel-image\n      \"\n      data-testid=\"modal-image\"\n      [src]=\"\n        getCarouselImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          imgElement\n        ).url\n      \"\n      [style.--original-height]=\"\n        getCarouselImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          imgElement\n        ).originalHeight\n      \"\n      [style.--original-width]=\"\n        getCarouselImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          imgElement\n        ).originalWidth\n      \"\n      [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n      [ngStyle]=\"{\n        width: getCarouselImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          imgElement\n        ).width,\n        height: getCarouselImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          imgElement\n        ).height\n      }\"\n    />\n    <div>\n      <button\n        class=\"\n          stream-chat-angular__image-modal-stepper\n          str-chat__image-carousel-stepper str-chat__image-carousel-stepper-prev\n        \"\n        [ngStyle]=\"{\n          visibility: isImageModalPrevButtonVisible ? 'visible' : 'hidden'\n        }\"\n        data-testid=\"image-modal-prev\"\n        type=\"button\"\n        (click)=\"stepImages(-1)\"\n        (keyup.enter)=\"stepImages(-1)\"\n      >\n        <stream-icon-placeholder icon=\"arrow-left\"></stream-icon-placeholder>\n      </button>\n      <button\n        class=\"\n          stream-chat-angular__image-modal-stepper\n          str-chat__image-carousel-stepper str-chat__image-carousel-stepper-next\n        \"\n        type=\"button\"\n        [ngStyle]=\"{\n          visibility: isImageModalNextButtonVisible ? 'visible' : 'hidden'\n        }\"\n        data-testid=\"image-modal-next\"\n        (click)=\"stepImages(1)\"\n        (keyup.enter)=\"stepImages(1)\"\n      >\n        <stream-icon-placeholder icon=\"arrow-right\"></stream-icon-placeholder>\n      </button>\n    </div>\n  </div>\n</ng-template>\n"]}