stream-chat-angular 4.0.0 → 4.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/assets/i18n/en.d.ts +2 -0
  2. package/assets/version.d.ts +1 -1
  3. package/bundles/stream-chat-angular.umd.js +669 -204
  4. package/bundles/stream-chat-angular.umd.js.map +1 -1
  5. package/esm2015/assets/i18n/en.js +3 -1
  6. package/esm2015/assets/version.js +2 -2
  7. package/esm2015/lib/attachment-configuration.service.js +158 -0
  8. package/esm2015/lib/attachment-list/attachment-list.component.js +64 -25
  9. package/esm2015/lib/attachment.service.js +4 -1
  10. package/esm2015/lib/avatar/avatar.component.js +30 -6
  11. package/esm2015/lib/channel-preview/channel-preview.component.js +4 -4
  12. package/esm2015/lib/channel.service.js +127 -18
  13. package/esm2015/lib/message/message.component.js +17 -9
  14. package/esm2015/lib/message-actions-box/message-actions-box.component.js +5 -3
  15. package/esm2015/lib/message-list/image-load.service.js +2 -1
  16. package/esm2015/lib/message-list/message-list.component.js +158 -75
  17. package/esm2015/lib/types.js +1 -1
  18. package/esm2015/public-api.js +2 -1
  19. package/fesm2015/stream-chat-angular.js +598 -170
  20. package/fesm2015/stream-chat-angular.js.map +1 -1
  21. package/lib/attachment-configuration.service.d.ts +58 -0
  22. package/lib/attachment-list/attachment-list.component.d.ts +12 -8
  23. package/lib/avatar/avatar.component.d.ts +2 -0
  24. package/lib/channel.service.d.ts +35 -3
  25. package/lib/message/message.component.d.ts +7 -21
  26. package/lib/message-list/image-load.service.d.ts +1 -0
  27. package/lib/message-list/message-list.component.d.ts +14 -9
  28. package/lib/types.d.ts +10 -0
  29. package/package.json +1 -1
  30. package/public-api.d.ts +1 -0
  31. package/src/assets/i18n/en.ts +2 -0
  32. package/src/assets/styles/css/index.css +1 -1
  33. package/src/assets/styles/css/index.css.map +1 -1
  34. package/src/assets/styles/scss/Attachment.scss +45 -2
  35. package/src/assets/styles/scss/Gallery.scss +12 -6
  36. package/src/assets/styles/scss/ImageCarousel.scss +6 -0
  37. package/src/assets/styles/scss/Message.scss +8 -2
  38. package/src/assets/styles/v2/css/index.css +1 -1
  39. package/src/assets/styles/v2/css/index.css.map +1 -1
  40. package/src/assets/styles/v2/css/index.layout.css +1 -1
  41. package/src/assets/styles/v2/css/index.layout.css.map +1 -1
  42. package/src/assets/styles/v2/scss/AttachmentList/AttachmentList-layout.scss +72 -46
  43. package/src/assets/styles/v2/scss/Message/Message-layout.scss +0 -16
  44. package/src/assets/styles/v2/scss/MessageReactions/MessageReactions-layout.scss +1 -10
  45. package/src/assets/styles/v2/scss/Tooltip/Tooltip-layout.scss +2 -23
  46. package/src/assets/version.ts +1 -1
@@ -33,6 +33,7 @@ export const en = {
33
33
  'Message deleted': 'Message deleted',
34
34
  'Message has been successfully flagged': 'Message has been successfully flagged',
35
35
  'Message pinned': 'Message pinned',
36
+ 'Message unpinned': 'Message unpinned',
36
37
  Mute: 'Mute',
37
38
  New: 'New',
38
39
  'New Messages!': 'New Messages!',
@@ -82,10 +83,11 @@ export const en = {
82
83
  "You can't send messages in this channel": "You can't send messages in this channel",
83
84
  "You can't send thread replies in this channel": "You can't send thread replies in this channel",
84
85
  'Unsupported file type: {{type}}': 'Unsupported file type: {{type}}',
86
+ 'Message not found': 'Message not found',
85
87
  'No chats here yet…': 'No chats here yet…',
86
88
  'user is typing': '{{ user }} is typing',
87
89
  'users are typing': '{{ users }} are typing',
88
90
  'Error loading channels': 'Error loading channels',
89
91
  },
90
92
  };
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"en.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/assets/i18n/en.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,EAAE,GAAG;IAChB,UAAU,EAAE;QACV,SAAS,EAAE,SAAS;QACpB,cAAc,EAAE,cAAc;QAC9B,MAAM,EAAE,QAAQ;QAChB,iBAAiB,EAAE,iBAAiB;QACpC,KAAK,EAAE,OAAO;QACd,oBAAoB,EAAE,oBAAoB;QAC1C,mBAAmB,EAAE,mBAAmB;QACxC,yCAAyC,EACvC,yCAAyC;QAC3C,MAAM,EAAE,QAAQ;QAChB,SAAS,EAAE,WAAW;QACtB,cAAc,EAAE,cAAc;QAC9B,6BAA6B,EAAE,6BAA6B;QAC5D,gBAAgB,EAAE,gBAAgB;QAClC,kBAAkB,EAAE,kBAAkB;QACtC,mBAAmB,EAAE,mBAAmB;QACxC,0DAA0D,EACxD,yDAAyD;QAC3D,wBAAwB,EAAE,wBAAwB;QAClD,yBAAyB,EAAE,yBAAyB;QACpD,uBAAuB,EAAE,uBAAuB;QAChD,4BAA4B,EAAE,4BAA4B;QAC1D,2BAA2B,EAAE,2BAA2B;QACxD,sBAAsB,EAAE,sBAAsB;QAC9C,uBAAuB,EAAE,uBAAuB;QAChD,2BAA2B,EAAE,2BAA2B;QACxD,gBAAgB,EAAE,0BAA0B;QAC5C,2BAA2B,EAAE,2BAA2B;QACxD,IAAI,EAAE,MAAM;QACZ,gBAAgB,EAAE,gBAAgB;QAClC,+BAA+B,EAAE,8BAA8B;QAC/D,qCAAqC,EACnC,8CAA8C;QAChD,iBAAiB,EAAE,iBAAiB;QACpC,uCAAuC,EACrC,uCAAuC;QACzC,gBAAgB,EAAE,gBAAgB;QAClC,IAAI,EAAE,MAAM;QACZ,GAAG,EAAE,KAAK;QACV,eAAe,EAAE,eAAe;QAChC,kBAAkB,EAAE,kBAAkB;QACtC,gBAAgB,EAAE,gBAAgB;QAClC,qBAAqB,EAAE,qBAAqB;QAC5C,mBAAmB,EAAE,mBAAmB;QACxC,iBAAiB,EAAE,iBAAiB;QACpC,iBAAiB,EAAE,iBAAiB;QACpC,GAAG,EAAE,KAAK;QACV,WAAW,EAAE,WAAW;QACxB,KAAK,EAAE,OAAO;QACd,kBAAkB,EAAE,kBAAkB;QACtC,MAAM,EAAE,QAAQ;QAChB,cAAc,EAAE,cAAc;QAC9B,IAAI,EAAE,MAAM;QACZ,6BAA6B,EAAE,6BAA6B;QAC5D,YAAY,EAAE,YAAY;QAC1B,cAAc,EAAE,cAAc;QAC9B,uBAAuB,EAAE,uBAAuB;QAChD,6BAA6B,EAAE,6BAA6B;QAC5D,MAAM,EAAE,cAAc;QACtB,mBAAmB,EAAE,mBAAmB;QACxC,MAAM,EAAE,QAAQ;QAChB,KAAK,EAAE,OAAO;QACd,0CAA0C,EACxC,0CAA0C;QAC5C,gCAAgC,EAAE,gCAAgC;QAClE,4CAA4C,EAC1C,4CAA4C;QAC9C,IAAI,EAAE,MAAM;QACZ,qCAAqC,EACnC,qCAAqC;QACvC,oDAAoD,EAClD,oDAAoD;QACtD,+CAA+C,EAC7C,+CAA+C;QACjD,sCAAsC,EACpC,sCAAsC;QACxC,uBAAuB,EAAE,uBAAuB;QAChD,2BAA2B,EAAE,2BAA2B;QACxD,0BAA0B,EAAE,0BAA0B;QACtD,2BAA2B,EAAE,2BAA2B;QACxD,6BAA6B,EAAE,6BAA6B;QAC5D,2BAA2B,EAAE,2BAA2B;QACxD,kBAAkB,EAAE,kBAAkB;QACtC,kBAAkB,EAAE,kBAAkB;QACtC,WAAW,EAAE,WAAW;QACxB,MAAM,EAAE,QAAQ;QAChB,KAAK,EAAE,OAAO;QACd,IAAI,EAAE,SAAS;QACf,mDAAmD,EACjD,mDAAmD;QACrD,yCAAyC,EACvC,yCAAyC;QAC3C,+CAA+C,EAC7C,+CAA+C;QACjD,iCAAiC,EAAE,iCAAiC;QACpE,oBAAoB,EAAE,oBAAoB;QAC1C,gBAAgB,EAAE,sBAAsB;QACxC,kBAAkB,EAAE,wBAAwB;QAC5C,wBAAwB,EAAE,wBAAwB;KACnD;CACF,CAAC","sourcesContent":["export const en = {\n  streamChat: {\n    '1 reply': '1 reply',\n    'Attach files': 'Attach files',\n    Cancel: 'Cancel',\n    'Channel Missing': 'Channel Missing',\n    Close: 'Close',\n    'Close emoji picker': 'Close emoji picker',\n    'Commands matching': 'Commands matching',\n    'Connection failure, reconnecting now...':\n      'Connection failure, reconnecting now...',\n    Delete: 'Delete',\n    Delivered: 'Delivered',\n    'Edit Message': 'Edit Message',\n    'Edit message request failed': 'Edit message request failed',\n    'Emoji matching': 'Emoji matching',\n    'Empty message...': 'Empty message...',\n    'Error adding flag': 'Error adding flag',\n    'Error connecting to chat, refresh the page to try again.':\n      'Error connecting to chat, refresh the page to try again',\n    'Error deleting message': 'Error deleting message',\n    'Error muting a user ...': 'Error muting a user ...',\n    'Error pinning message': 'Error pinning message',\n    'Error removing message pin': 'Error removing message pin',\n    'Error unmuting a user ...': 'Error unmuting a user ...',\n    'Error uploading file': 'Error uploading file',\n    'Error uploading image': 'Error uploading image',\n    'Error deleting attachment': 'Error deleting attachment',\n    'Error · Unsent': \"Message couldn't be sent\",\n    'Error: {{ errorMessage }}': 'Error: {{ errorMessage }}',\n    Flag: 'Flag',\n    'Message Failed': 'Message Failed',\n    'Message Failed · Unauthorized': 'Unauthorized to send message',\n    'Message Failed · Click to try again':\n      \"Message couldn't be sent, click to try again\",\n    'Message deleted': 'Message deleted',\n    'Message has been successfully flagged':\n      'Message has been successfully flagged',\n    'Message pinned': 'Message pinned',\n    Mute: 'Mute',\n    New: 'New',\n    'New Messages!': 'New Messages!',\n    'No results found': 'No results found',\n    'Nothing yet...': 'Nothing yet...',\n    'Only visible to you': 'Only visible to you',\n    'Open emoji picker': 'Open emoji picker',\n    'People matching': 'People matching',\n    'Pick your emoji': 'Pick your emoji',\n    Pin: 'Pin',\n    'Pinned by': 'Pinned by',\n    Reply: 'Reply',\n    'Reply to Message': 'Reply to Message',\n    Search: 'Search',\n    'Searching...': 'Searching...',\n    Send: 'Send',\n    'Send message request failed': 'Send message request failed',\n    'Sending...': 'Sending...',\n    'Slow Mode ON': 'Slow Mode ON',\n    'Start of a new thread': 'Start of a new thread',\n    'This message was deleted...': 'This message was deleted...',\n    Thread: 'Thread reply',\n    'Type your message': 'Type your message',\n    Unmute: 'Unmute',\n    Unpin: 'Unpin',\n    'Wait until all attachments have uploaded':\n      'Wait until all attachments have uploaded',\n    'You have no channels currently': 'You have no channels currently',\n    \"You've reached the maximum number of files\":\n      \"You've reached the maximum number of files\",\n    live: 'live',\n    'this content could not be displayed':\n      'this content could not be displayed',\n    '{{ commaSeparatedUsers }} and {{ moreCount }} more':\n      '{{ commaSeparatedUsers }} and {{ moreCount }} more',\n    '{{ commaSeparatedUsers }}, and {{ lastUser }}':\n      '{{ commaSeparatedUsers }}, and {{ lastUser }}',\n    '{{ firstUser }} and {{ secondUser }}':\n      '{{ firstUser }} and {{ secondUser }}',\n    '{{ imageCount }} more': '{{ imageCount }} more',\n    '{{ memberCount }} members': '{{ memberCount }} members',\n    '{{ replyCount }} replies': '{{ replyCount }} replies',\n    '{{ user }} has been muted': '{{ user }} has been muted',\n    '{{ user }} has been unmuted': '{{ user }} has been unmuted',\n    '{{ watcherCount }} online': '{{ watcherCount }} online',\n    '🏙 Attachment...': '🏙 Attachment...',\n    'Connection error': 'Connection error',\n    'Load more': 'Load more',\n    failed: 'failed',\n    retry: 'retry',\n    test: 'success',\n    'Sending links is not allowed in this conversation':\n      'Sending links is not allowed in this conversation',\n    \"You can't send messages in this channel\":\n      \"You can't send messages in this channel\",\n    \"You can't send thread replies in this channel\":\n      \"You can't send thread replies in this channel\",\n    'Unsupported file type: {{type}}': 'Unsupported file type: {{type}}',\n    'No chats here yet…': 'No chats here yet…',\n    'user is typing': '{{ user }} is typing',\n    'users are typing': '{{ users }} are typing',\n    'Error loading channels': 'Error loading channels',\n  },\n};\n"]}
93
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"en.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/assets/i18n/en.ts"],"names":[],"mappings":"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","sourcesContent":["export const en = {\n  streamChat: {\n    '1 reply': '1 reply',\n    'Attach files': 'Attach files',\n    Cancel: 'Cancel',\n    'Channel Missing': 'Channel Missing',\n    Close: 'Close',\n    'Close emoji picker': 'Close emoji picker',\n    'Commands matching': 'Commands matching',\n    'Connection failure, reconnecting now...':\n      'Connection failure, reconnecting now...',\n    Delete: 'Delete',\n    Delivered: 'Delivered',\n    'Edit Message': 'Edit Message',\n    'Edit message request failed': 'Edit message request failed',\n    'Emoji matching': 'Emoji matching',\n    'Empty message...': 'Empty message...',\n    'Error adding flag': 'Error adding flag',\n    'Error connecting to chat, refresh the page to try again.':\n      'Error connecting to chat, refresh the page to try again',\n    'Error deleting message': 'Error deleting message',\n    'Error muting a user ...': 'Error muting a user ...',\n    'Error pinning message': 'Error pinning message',\n    'Error removing message pin': 'Error removing message pin',\n    'Error unmuting a user ...': 'Error unmuting a user ...',\n    'Error uploading file': 'Error uploading file',\n    'Error uploading image': 'Error uploading image',\n    'Error deleting attachment': 'Error deleting attachment',\n    'Error · Unsent': \"Message couldn't be sent\",\n    'Error: {{ errorMessage }}': 'Error: {{ errorMessage }}',\n    Flag: 'Flag',\n    'Message Failed': 'Message Failed',\n    'Message Failed · Unauthorized': 'Unauthorized to send message',\n    'Message Failed · Click to try again':\n      \"Message couldn't be sent, click to try again\",\n    'Message deleted': 'Message deleted',\n    'Message has been successfully flagged':\n      'Message has been successfully flagged',\n    'Message pinned': 'Message pinned',\n    'Message unpinned': 'Message unpinned',\n    Mute: 'Mute',\n    New: 'New',\n    'New Messages!': 'New Messages!',\n    'No results found': 'No results found',\n    'Nothing yet...': 'Nothing yet...',\n    'Only visible to you': 'Only visible to you',\n    'Open emoji picker': 'Open emoji picker',\n    'People matching': 'People matching',\n    'Pick your emoji': 'Pick your emoji',\n    Pin: 'Pin',\n    'Pinned by': 'Pinned by',\n    Reply: 'Reply',\n    'Reply to Message': 'Reply to Message',\n    Search: 'Search',\n    'Searching...': 'Searching...',\n    Send: 'Send',\n    'Send message request failed': 'Send message request failed',\n    'Sending...': 'Sending...',\n    'Slow Mode ON': 'Slow Mode ON',\n    'Start of a new thread': 'Start of a new thread',\n    'This message was deleted...': 'This message was deleted...',\n    Thread: 'Thread reply',\n    'Type your message': 'Type your message',\n    Unmute: 'Unmute',\n    Unpin: 'Unpin',\n    'Wait until all attachments have uploaded':\n      'Wait until all attachments have uploaded',\n    'You have no channels currently': 'You have no channels currently',\n    \"You've reached the maximum number of files\":\n      \"You've reached the maximum number of files\",\n    live: 'live',\n    'this content could not be displayed':\n      'this content could not be displayed',\n    '{{ commaSeparatedUsers }} and {{ moreCount }} more':\n      '{{ commaSeparatedUsers }} and {{ moreCount }} more',\n    '{{ commaSeparatedUsers }}, and {{ lastUser }}':\n      '{{ commaSeparatedUsers }}, and {{ lastUser }}',\n    '{{ firstUser }} and {{ secondUser }}':\n      '{{ firstUser }} and {{ secondUser }}',\n    '{{ imageCount }} more': '{{ imageCount }} more',\n    '{{ memberCount }} members': '{{ memberCount }} members',\n    '{{ replyCount }} replies': '{{ replyCount }} replies',\n    '{{ user }} has been muted': '{{ user }} has been muted',\n    '{{ user }} has been unmuted': '{{ user }} has been unmuted',\n    '{{ watcherCount }} online': '{{ watcherCount }} online',\n    '🏙 Attachment...': '🏙 Attachment...',\n    'Connection error': 'Connection error',\n    'Load more': 'Load more',\n    failed: 'failed',\n    retry: 'retry',\n    test: 'success',\n    'Sending links is not allowed in this conversation':\n      'Sending links is not allowed in this conversation',\n    \"You can't send messages in this channel\":\n      \"You can't send messages in this channel\",\n    \"You can't send thread replies in this channel\":\n      \"You can't send thread replies in this channel\",\n    'Unsupported file type: {{type}}': 'Unsupported file type: {{type}}',\n    'Message not found': 'Message not found',\n    'No chats here yet…': 'No chats here yet…',\n    'user is typing': '{{ user }} is typing',\n    'users are typing': '{{ users }} are typing',\n    'Error loading channels': 'Error loading channels',\n  },\n};\n"]}
@@ -1,2 +1,2 @@
1
- export const version = '4.0.0';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjAuMCc7XG4iXX0=
1
+ export const version = '4.3.0';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjMuMCc7XG4iXX0=
@@ -0,0 +1,158 @@
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 { sizeRestriction, height } = this.getSizingRestrictions(url, element);
28
+ if (sizeRestriction) {
29
+ // Apply 2x for retina displays
30
+ sizeRestriction.height *= 2;
31
+ sizeRestriction.width *= 2;
32
+ this.addResizingParamsToUrl(sizeRestriction, url);
33
+ }
34
+ return {
35
+ url: url.href,
36
+ width: '',
37
+ height,
38
+ };
39
+ }
40
+ /**
41
+ * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic
42
+ * @param attachment The attachment to configure
43
+ * @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.
44
+ */
45
+ getVideoAttachmentConfiguration(attachment, element) {
46
+ if (this.customVideoAttachmentConfigurationHandler) {
47
+ return this.customVideoAttachmentConfigurationHandler(attachment, element);
48
+ }
49
+ let attachmentHeight = ``;
50
+ let thumbUrl = undefined;
51
+ if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {
52
+ const url = new URL(attachment.thumb_url);
53
+ const { sizeRestriction, height } = this.getSizingRestrictions(url, element);
54
+ if (sizeRestriction) {
55
+ sizeRestriction.height *= 2;
56
+ sizeRestriction.width *= 2;
57
+ this.addResizingParamsToUrl(sizeRestriction, url);
58
+ }
59
+ thumbUrl = url.href;
60
+ attachmentHeight = height;
61
+ }
62
+ else {
63
+ const cssSizeRestriction = this.getCSSSizeRestriction(element);
64
+ attachmentHeight = `${cssSizeRestriction.maxHeight || cssSizeRestriction.height || ''}px`;
65
+ }
66
+ return {
67
+ url: attachment.asset_url || '',
68
+ width: '',
69
+ height: attachmentHeight,
70
+ thumbUrl: thumbUrl,
71
+ };
72
+ }
73
+ /**
74
+ * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic
75
+ * @param attachment The attachment to configure
76
+ */
77
+ getGiphyAttachmentConfiguration(attachment) {
78
+ var _a;
79
+ if (this.customGiphyAttachmentConfigurationHandler) {
80
+ return this.customGiphyAttachmentConfigurationHandler(attachment);
81
+ }
82
+ const giphy = (_a = attachment.giphy) === null || _a === void 0 ? void 0 : _a.fixed_height_downsampled;
83
+ return {
84
+ url: (giphy === null || giphy === void 0 ? void 0 : giphy.url) || attachment.image_url || attachment.thumb_url || '',
85
+ height: (giphy === null || giphy === void 0 ? void 0 : giphy.height) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.height}px` : '300px',
86
+ width: (giphy === null || giphy === void 0 ? void 0 : giphy.width) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.width}px` : '',
87
+ };
88
+ }
89
+ /**
90
+ * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic
91
+ * @param attachment The attachment to configure
92
+ */
93
+ getScrapedImageAttachmentConfiguration(attachment) {
94
+ if (this.customScrapedImageAttachmentConfigurationHandler) {
95
+ return this.customScrapedImageAttachmentConfigurationHandler(attachment);
96
+ }
97
+ return {
98
+ url: attachment.image_url || attachment.thumb_url || '',
99
+ width: '',
100
+ height: '', // Set from CSS
101
+ };
102
+ }
103
+ addResizingParamsToUrl(sizeRestriction, url) {
104
+ url.searchParams.set('h', sizeRestriction.height.toString());
105
+ url.searchParams.set('w', sizeRestriction.width.toString());
106
+ }
107
+ getSizingRestrictions(url, htmlElement) {
108
+ const urlParams = url.searchParams;
109
+ const originalHeight = Number(urlParams.get('oh')) || 1;
110
+ const originalWidth = Number(urlParams.get('ow')) || 1;
111
+ const cssSizeRestriction = this.getCSSSizeRestriction(htmlElement);
112
+ let sizeRestriction;
113
+ let height = '';
114
+ if ((cssSizeRestriction.maxHeight || cssSizeRestriction.height) &&
115
+ cssSizeRestriction.maxWidth) {
116
+ sizeRestriction = this.getSizeRestrictions(originalHeight, originalWidth, (cssSizeRestriction.maxHeight || cssSizeRestriction.height), cssSizeRestriction.maxWidth);
117
+ if (cssSizeRestriction.maxHeight) {
118
+ const heightNum = originalHeight > 1 && originalWidth > 1
119
+ ? originalHeight <= cssSizeRestriction.maxHeight &&
120
+ originalWidth <= cssSizeRestriction.maxWidth
121
+ ? originalHeight
122
+ : Math.round(Math.min(cssSizeRestriction.maxHeight, (cssSizeRestriction.maxWidth / originalWidth) *
123
+ originalHeight))
124
+ : cssSizeRestriction.maxHeight;
125
+ height = `${heightNum}px`;
126
+ }
127
+ }
128
+ else {
129
+ sizeRestriction = undefined;
130
+ }
131
+ return { sizeRestriction, height };
132
+ }
133
+ getSizeRestrictions(originalHeight, originalWidth, maxHeight, maxWidth) {
134
+ return {
135
+ height: Math.round(Math.max(maxHeight, (maxWidth / originalWidth) * originalHeight)),
136
+ width: Math.round(Math.max(maxHeight, (maxWidth / originalHeight) * originalWidth)),
137
+ };
138
+ }
139
+ getCSSSizeRestriction(htmlElement) {
140
+ const computedStylesheet = getComputedStyle(htmlElement);
141
+ const height = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('height'));
142
+ const maxHeight = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('max-height'));
143
+ const maxWidth = this.getValueRepresentationOfCSSProperty(computedStylesheet.getPropertyValue('max-width'));
144
+ return { height, maxHeight, maxWidth };
145
+ }
146
+ getValueRepresentationOfCSSProperty(property) {
147
+ return Number(property.replace('px', '')) || undefined;
148
+ }
149
+ }
150
+ AttachmentConfigurationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
151
+ AttachmentConfigurationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, providedIn: 'root' });
152
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, decorators: [{
153
+ type: Injectable,
154
+ args: [{
155
+ providedIn: 'root',
156
+ }]
157
+ }] });
158
+ //# 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":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;AAQ3C;;GAEG;AAIH,MAAM,OAAO,8BAA8B;IAH3C;QAiCE;;WAEG;QACH,iCAA4B,GAAG,IAAI,CAAC;KAqNrC;IAnNC;;;;;OAKG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,QAA2C,EAC3C,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,QAAQ,EACR,OAAO,CACR,CAAC;SACH;QAED,MAAM,GAAG,GAAG,IAAI,GAAG,CACjB,CAAC,UAAU,CAAC,OAAO;YACjB,UAAU,CAAC,SAAS;YACpB,UAAU,CAAC,SAAS;YACpB,EAAE,CAAW,CAChB,CAAC;QACF,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,CACR,CAAC;QAEF,IAAI,eAAe,EAAE;YACnB,+BAA+B;YAC/B,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;YAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;YAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;SACnD;QAED,OAAO;YACL,GAAG,EAAE,GAAG,CAAC,IAAI;YACb,KAAK,EAAE,EAAE;YACT,MAAM;SACP,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,OAAO,CACR,CAAC;SACH;QAED,IAAI,gBAAgB,GAAG,EAAE,CAAC;QAC1B,IAAI,QAAQ,GAAG,SAAS,CAAC;QACzB,IAAI,UAAU,CAAC,SAAS,IAAI,IAAI,CAAC,4BAA4B,EAAE;YAC7D,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC1C,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,CACR,CAAC;YAEF,IAAI,eAAe,EAAE;gBACnB,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;gBAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;gBAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;aACnD;YACD,QAAQ,GAAG,GAAG,CAAC,IAAI,CAAC;YACpB,gBAAgB,GAAG,MAAM,CAAC;SAC3B;aAAM;YACL,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YAC/D,gBAAgB,GAAG,GACjB,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,IAAI,EAC/D,IAAI,CAAC;SACN;QACD,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,EAAE;YAC/B,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,gBAAgB;YACxB,QAAQ,EAAE,QAAQ;SACnB,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,+BAA+B,CAC7B,UAAyB;;QAEzB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CAAC,UAAU,CAAC,CAAC;SACnE;QAED,MAAM,KAAK,GAAG,MAAA,UAAU,CAAC,KAAK,0CAAE,wBAAwB,CAAC;QAEzD,OAAO;YACL,GAAG,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,GAAG,KAAI,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACrE,MAAM,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,IAAI,CAAC,CAAC,CAAC,OAAO;YACtD,KAAK,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,IAAI,CAAC,CAAC,CAAC,EAAE;SAC/C,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,sCAAsC,CACpC,UAAyB;QAEzB,IAAI,IAAI,CAAC,gDAAgD,EAAE;YACzD,OAAO,IAAI,CAAC,gDAAgD,CAAC,UAAU,CAAC,CAAC;SAC1E;QAED,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACvD,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,EAAE,EAAE,eAAe;SAC5B,CAAC;IACJ,CAAC;IAEO,sBAAsB,CAC5B,eAAkD,EAClD,GAAQ;QAER,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7D,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC9D,CAAC;IAEO,qBAAqB,CAAC,GAAQ,EAAE,WAAwB;QAC9D,MAAM,SAAS,GAAG,GAAG,CAAC,YAAY,CAAC;QACnC,MAAM,cAAc,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACxD,MAAM,aAAa,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACvD,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;QACnE,IAAI,eAA8D,CAAC;QACnE,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,IACE,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAC;YAC3D,kBAAkB,CAAC,QAAQ,EAC3B;YACA,eAAe,GAAG,IAAI,CAAC,mBAAmB,CACxC,cAAc,EACd,aAAa,EACb,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAE,EAC5D,kBAAkB,CAAC,QAAQ,CAC5B,CAAC;YACF,IAAI,kBAAkB,CAAC,SAAS,EAAE;gBAChC,MAAM,SAAS,GACb,cAAc,GAAG,CAAC,IAAI,aAAa,GAAG,CAAC;oBACrC,CAAC,CAAC,cAAc,IAAI,kBAAkB,CAAC,SAAS;wBAC9C,aAAa,IAAI,kBAAkB,CAAC,QAAQ;wBAC5C,CAAC,CAAC,cAAc;wBAChB,CAAC,CAAC,IAAI,CAAC,KAAK,CACR,IAAI,CAAC,GAAG,CACN,kBAAkB,CAAC,SAAS,EAC5B,CAAC,kBAAkB,CAAC,QAAQ,GAAG,aAAa,CAAC;4BAC3C,cAAc,CACjB,CACF;oBACL,CAAC,CAAC,kBAAkB,CAAC,SAAS,CAAC;gBACnC,MAAM,GAAG,GAAG,SAAS,IAAI,CAAC;aAC3B;SACF;aAAM;YACL,eAAe,GAAG,SAAS,CAAC;SAC7B;QAED,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,CAAC;IACrC,CAAC;IAEO,mBAAmB,CACzB,cAAsB,EACtB,aAAqB,EACrB,SAAiB,EACjB,QAAgB;QAEhB,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,KAAK,CAChB,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,aAAa,CAAC,GAAG,cAAc,CAAC,CACjE;YACD,KAAK,EAAE,IAAI,CAAC,KAAK,CACf,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,cAAc,CAAC,GAAG,aAAa,CAAC,CACjE;SACF,CAAC;IACJ,CAAC;IAEO,qBAAqB,CAAC,WAAwB;QACpD,MAAM,kBAAkB,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;QACzD,MAAM,MAAM,GAAG,IAAI,CAAC,mCAAmC,CACrD,kBAAkB,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAC9C,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,CAAC,mCAAmC,CACxD,kBAAkB,CAAC,gBAAgB,CAAC,YAAY,CAAC,CAClD,CAAC;QACF,MAAM,QAAQ,GAAG,IAAI,CAAC,mCAAmC,CACvD,kBAAkB,CAAC,gBAAgB,CAAC,WAAW,CAAC,CACjD,CAAC;QAEF,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,CAAC;IACzC,CAAC;IAEO,mCAAmC,CAAC,QAAgB;QAC1D,OAAO,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,SAAS,CAAC;IACzD,CAAC;;2HArPU,8BAA8B;+HAA9B,8BAA8B,cAF7B,MAAM;2FAEP,8BAA8B;kBAH1C,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport {\n  AttachmentConfigration,\n  DefaultStreamChatGenerics,\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  ) => AttachmentConfigration;\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  ): AttachmentConfigration {\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 { sizeRestriction, height } = this.getSizingRestrictions(\n      url,\n      element\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,\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 attachmentHeight = ``;\n    let thumbUrl = undefined;\n    if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {\n      const url = new URL(attachment.thumb_url);\n      const { sizeRestriction, height } = this.getSizingRestrictions(\n        url,\n        element\n      );\n\n      if (sizeRestriction) {\n        sizeRestriction.height *= 2;\n        sizeRestriction.width *= 2;\n        this.addResizingParamsToUrl(sizeRestriction, url);\n      }\n      thumbUrl = url.href;\n      attachmentHeight = height;\n    } else {\n      const cssSizeRestriction = this.getCSSSizeRestriction(element);\n      attachmentHeight = `${\n        cssSizeRestriction.maxHeight || cssSizeRestriction.height || ''\n      }px`;\n    }\n    return {\n      url: attachment.asset_url || '',\n      width: '', // Not set to respect responsive width\n      height: attachmentHeight,\n      thumbUrl: thumbUrl,\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(url: URL, htmlElement: HTMLElement) {\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    let height = '';\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      if (cssSizeRestriction.maxHeight) {\n        const heightNum =\n          originalHeight > 1 && originalWidth > 1\n            ? originalHeight <= cssSizeRestriction.maxHeight &&\n              originalWidth <= cssSizeRestriction.maxWidth\n              ? originalHeight\n              : Math.round(\n                  Math.min(\n                    cssSizeRestriction.maxHeight,\n                    (cssSizeRestriction.maxWidth / originalWidth) *\n                      originalHeight\n                  )\n                )\n            : cssSizeRestriction.maxHeight;\n        height = `${heightNum}px`;\n      }\n    } else {\n      sizeRestriction = undefined;\n    }\n\n    return { sizeRestriction, height };\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"]}
@@ -3,8 +3,8 @@ import prettybytes from 'pretty-bytes';
3
3
  import { isImageAttachment } from '../is-image-attachment';
4
4
  import * as i0 from "@angular/core";
5
5
  import * as i1 from "../custom-templates.service";
6
- import * as i2 from "../message-list/image-load.service";
7
- import * as i3 from "../channel.service";
6
+ import * as i2 from "../channel.service";
7
+ import * as i3 from "../attachment-configuration.service";
8
8
  import * as i4 from "../theme.service";
9
9
  import * as i5 from "../icon-placeholder/icon-placeholder.component";
10
10
  import * as i6 from "../modal/modal.component";
@@ -14,10 +14,10 @@ import * as i8 from "@ngx-translate/core";
14
14
  * The `AttachmentList` compontent displays the attachments of a message
15
15
  */
16
16
  export class AttachmentListComponent {
17
- constructor(customTemplatesService, imageLoadService, channelService, themeService) {
17
+ constructor(customTemplatesService, channelService, attachmentConfigurationService, themeService) {
18
18
  this.customTemplatesService = customTemplatesService;
19
- this.imageLoadService = imageLoadService;
20
19
  this.channelService = channelService;
20
+ this.attachmentConfigurationService = attachmentConfigurationService;
21
21
  /**
22
22
  * The attachments to display
23
23
  */
@@ -26,24 +26,31 @@ export class AttachmentListComponent {
26
26
  this.orderedAttachments = [];
27
27
  this.imagesToView = [];
28
28
  this.imagesToViewCurrentIndex = 0;
29
+ this.attachmentConfigurations = new Map();
29
30
  this.themeVersion = themeService.themeVersion;
30
31
  }
31
- ngOnChanges() {
32
- const images = this.attachments.filter(this.isImage);
33
- const containsGallery = images.length >= 2;
34
- this.orderedAttachments = [
35
- ...(containsGallery ? this.createGallery(images) : images),
36
- ...this.attachments.filter((a) => this.isVideo(a)),
37
- ...this.attachments.filter((a) => this.isFile(a)),
38
- ];
39
- // Display link attachments only if there are no other attachments
40
- // Giphy-s always sent without other attachments
41
- if (this.orderedAttachments.length === 0) {
42
- this.orderedAttachments.push(...this.attachments.filter((a) => this.isCard(a)));
32
+ ngOnChanges(changes) {
33
+ if (changes.attachments) {
34
+ const images = this.attachments.filter(this.isImage);
35
+ const containsGallery = images.length >= 2;
36
+ this.orderedAttachments = [
37
+ ...(containsGallery ? this.createGallery(images) : images),
38
+ ...this.attachments.filter((a) => this.isVideo(a)),
39
+ ...this.attachments.filter((a) => this.isFile(a)),
40
+ ];
41
+ this.attachmentConfigurations = new Map();
42
+ // Display link attachments only if there are no other attachments
43
+ // Giphy-s always sent without other attachments
44
+ if (this.orderedAttachments.length === 0) {
45
+ this.orderedAttachments.push(...this.attachments.filter((a) => this.isCard(a)));
46
+ }
43
47
  }
44
48
  }
45
- trackById(index) {
46
- return index;
49
+ trackByUrl(_, attachment) {
50
+ return (attachment.image_url ||
51
+ attachment.img_url ||
52
+ attachment.asset_url ||
53
+ attachment.thumb_url);
47
54
  }
48
55
  isImage(attachment) {
49
56
  return isImageAttachment(attachment);
@@ -69,9 +76,6 @@ export class AttachmentListComponent {
69
76
  (attachment.type === 'image' && !this.isImage(attachment)) ||
70
77
  attachment.type === 'giphy');
71
78
  }
72
- imageLoaded() {
73
- this.imageLoadService.imageLoad$.next();
74
- }
75
79
  hasFileSize(attachment) {
76
80
  return (attachment.file_size && Number.isFinite(Number(attachment.file_size)));
77
81
  }
@@ -112,6 +116,41 @@ export class AttachmentListComponent {
112
116
  trackByImageUrl(_, item) {
113
117
  return item.image_url || item.img_url || item.thumb_url;
114
118
  }
119
+ getImageAttachmentConfiguration(attachment, type, element) {
120
+ const existingConfiguration = this.attachmentConfigurations.get(attachment);
121
+ if (existingConfiguration) {
122
+ return existingConfiguration;
123
+ }
124
+ const configuration = this.attachmentConfigurationService.getImageAttachmentConfiguration(attachment, type, element);
125
+ this.attachmentConfigurations.set(attachment, configuration);
126
+ return configuration;
127
+ }
128
+ getCarouselImageAttachmentConfiguration(attachment, element) {
129
+ return this.attachmentConfigurationService.getImageAttachmentConfiguration(attachment, 'carousel', element);
130
+ }
131
+ getVideoAttachmentConfiguration(attachment, element) {
132
+ const existingConfiguration = this.attachmentConfigurations.get(attachment);
133
+ if (existingConfiguration) {
134
+ return existingConfiguration;
135
+ }
136
+ const configuration = this.attachmentConfigurationService.getVideoAttachmentConfiguration(attachment, element);
137
+ this.attachmentConfigurations.set(attachment, configuration);
138
+ return configuration;
139
+ }
140
+ getCardAttachmentConfiguration(attachment) {
141
+ const existingConfiguration = this.attachmentConfigurations.get(attachment);
142
+ if (existingConfiguration) {
143
+ return existingConfiguration;
144
+ }
145
+ if (attachment.type === 'giphy') {
146
+ return this.attachmentConfigurationService.getGiphyAttachmentConfiguration(attachment);
147
+ }
148
+ else {
149
+ const configuration = this.attachmentConfigurationService.getScrapedImageAttachmentConfiguration(attachment);
150
+ this.attachmentConfigurations.set(attachment, configuration);
151
+ return configuration;
152
+ }
153
+ }
115
154
  get isImageModalPrevButtonVisible() {
116
155
  return this.imagesToViewCurrentIndex !== 0;
117
156
  }
@@ -130,8 +169,8 @@ export class AttachmentListComponent {
130
169
  this.imagesToView = [];
131
170
  }
132
171
  }
133
- AttachmentListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentListComponent, deps: [{ token: i1.CustomTemplatesService }, { token: i2.ImageLoadService }, { token: i3.ChannelService }, { token: i4.ThemeService }], target: i0.ɵɵFactoryTarget.Component });
134
- 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" }, 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: trackById\"\n >\n <div\n data-testclass=\"attachment-container\"\n class=\"str-chat__message-attachment str-chat__message-attachment--{{\n attachment.type\n }}\"\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-with-actions]=\"\n attachment.actions && attachment.actions.length > 0\n \"\n [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n >\n <img\n *ngIf=\"isImage(attachment)\"\n class=\"str-chat__message-attachment--img\"\n data-testclass=\"image\"\n [src]=\"\n attachment.img_url || attachment.thumb_url || attachment.image_url\n \"\n [alt]=\"attachment?.fallback\"\n (load)=\"imageLoaded()\"\n (click)=\"openImageModal([attachment])\"\n (keyup.enter)=\"openImageModal([attachment])\"\n />\n <div\n class=\"str-chat__gallery\"\n data-testid=\"image-gallery\"\n *ngIf=\"isGallery(attachment)\"\n [class.str-chat__gallery--square]=\"(attachment?.images)!.length > 3\"\n [class.str-chat__gallery-two-rows]=\"(attachment?.images)!.length > 2\"\n >\n <ng-container\n *ngFor=\"\n let galleryImage of attachment.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(attachment.images!, index)\"\n (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\n \"\n >\n <img\n [src]=\"\n galleryImage.img_url ||\n galleryImage.thumb_url ||\n galleryImage.image_url\n \"\n [alt]=\"galleryImage.fallback\"\n (load)=\"imageLoaded()\"\n />\n </button>\n <button\n *ngIf=\"index === 3 && !isLast\"\n class=\"str-chat__gallery-placeholder\"\n data-testclass=\"gallery-image\"\n (click)=\"openImageModal(attachment.images!, index)\"\n (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\n \"\n [ngStyle]=\"{\n 'background-image':\n 'url(' +\n (galleryImage.img_url ||\n galleryImage.thumb_url ||\n galleryImage.image_url) +\n ')'\n }\"\n >\n <p\n [innerHTML]=\"\n 'streamChat.{{ imageCount }} more'\n | translate: { imageCount: attachment!.images!.length - 4 }\n \"\n ></p>\n </button>\n </ng-container>\n </div>\n <div class=\"str-chat__player-wrapper\" *ngIf=\"isVideo(attachment)\">\n <video\n class=\"str-chat__video-angular\"\n controls\n data-testclass=\"video-attachment\"\n [src]=\"attachment.asset_url\"\n ></video>\n </div>\n <div\n *ngIf=\"isFile(attachment)\"\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 <div class=\"str-chat__message-attachment-file--item-first-row\">\n <div\n data-testclass=\"file-title\"\n class=\"str-chat__message-attachment-file--item-name\"\n >\n {{ attachment.title }}\n </div>\n <a\n class=\"str-chat__message-attachment-file--item-download\"\n data-testclass=\"file-link\"\n download\n href=\"{{ attachment.asset_url }}\"\n target=\"_blank\"\n >\n <stream-icon-placeholder\n class=\"str-chat__message-attachment-download-icon\"\n icon=\"download\"\n ></stream-icon-placeholder>\n </a>\n </div>\n <span\n class=\"str-chat__message-attachment-file--item-size\"\n data-testclass=\"size\"\n *ngIf=\"hasFileSize(attachment)\"\n >{{ getFileSize(attachment) }}</span\n >\n </div>\n </div>\n <div\n *ngIf=\"isCard(attachment)\"\n class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n attachment.type\n }}\"\n >\n <div\n *ngIf=\"attachment.image_url || attachment.thumb_url\"\n class=\"str-chat__message-attachment-card--header\"\n >\n <img\n data-testclass=\"card-img\"\n alt=\"{{ attachment.image_url || attachment.thumb_url }}\"\n src=\"{{ attachment.image_url || attachment.thumb_url }}\"\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=\"attachment.title\"\n data-testclass=\"card-title\"\n class=\"str-chat__message-attachment-card--title\"\n >\n {{ attachment.title }}\n </div>\n <div\n *ngIf=\"attachment.text\"\n class=\"str-chat__message-attachment-card--text\"\n data-testclass=\"card-text\"\n >\n {{ attachment.text }}\n </div>\n <a\n class=\"str-chat__message-attachment-card--url\"\n *ngIf=\"attachment.title_link || attachment.og_scrape_url\"\n data-testclass=\"url-link\"\n noopener\n noreferrer\n href=\"{{ attachment.title_link || attachment.og_scrape_url }}\"\n target=\"_blank\"\n >\n {{ trimUrl(attachment.title_link || attachment.og_scrape_url) }}\n </a>\n </div>\n </div>\n </div>\n <div\n class=\"str-chat__message-attachment-actions\"\n *ngIf=\"attachment.actions && attachment.actions.length > 0\"\n >\n <div class=\"str-chat__message-attachment-actions-form\">\n <button\n *ngFor=\"\n let action of attachment.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 </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 [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 <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper\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 <img\n class=\"\n stream-chat-angular__image-modal-image\n str-chat__image-carousel-image\n \"\n data-testid=\"modal-image\"\n [src]=\"\n imagesToView[imagesToViewCurrentIndex].img_url ||\n imagesToView[imagesToViewCurrentIndex].thumb_url ||\n imagesToView[imagesToViewCurrentIndex].image_url\n \"\n [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n />\n <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper\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</ng-template>\n", components: [{ type: i5.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon", "size"] }, { type: i6.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
172
+ 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 });
173
+ 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" }, 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-with-actions]=\"\n attachment.actions && attachment.actions.length > 0\n \"\n [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n >\n <img\n #imgElement\n *ngIf=\"isImage(attachment)\"\n class=\"str-chat__message-attachment--img\"\n data-testclass=\"image\"\n [src]=\"\n getImageAttachmentConfiguration(attachment, 'single', imgElement).url\n \"\n [alt]=\"attachment?.fallback\"\n (click)=\"openImageModal([attachment])\"\n (keyup.enter)=\"openImageModal([attachment])\"\n [ngStyle]=\"{\n height: getImageAttachmentConfiguration(\n attachment,\n 'single',\n imgElement\n ).height,\n width: getImageAttachmentConfiguration(\n attachment,\n 'single',\n imgElement\n ).width\n }\"\n />\n <div\n class=\"str-chat__gallery\"\n data-testid=\"image-gallery\"\n *ngIf=\"isGallery(attachment)\"\n [class.str-chat__gallery--square]=\"(attachment?.images)!.length > 3\"\n [class.str-chat__gallery-two-rows]=\"(attachment?.images)!.length > 2\"\n >\n <ng-container\n *ngFor=\"\n let galleryImage of attachment.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(attachment.images!, index)\"\n (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\n \"\n >\n <img\n #imgElement\n [src]=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery',\n imgElement\n ).url\n \"\n [alt]=\"galleryImage.fallback\"\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(attachment.images!, index)\"\n (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n [class.str-chat__message-attachment--svg-image]=\"\n isSvg(galleryImage)\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: { imageCount: attachment!.images!.length - 4 }\n \"\n ></p>\n </button>\n </ng-container>\n </div>\n <div class=\"str-chat__player-wrapper\" *ngIf=\"isVideo(attachment)\">\n <video\n #videoElement\n class=\"str-chat__video-angular\"\n controls\n data-testclass=\"video-attachment\"\n [src]=\"getVideoAttachmentConfiguration(attachment, videoElement).url\"\n [ngStyle]=\"{\n height: getVideoAttachmentConfiguration(attachment, videoElement)\n .height,\n width: getVideoAttachmentConfiguration(attachment, videoElement)\n .width\n }\"\n [poster]=\"\n getVideoAttachmentConfiguration(attachment, videoElement).thumbUrl\n \"\n ></video>\n </div>\n <div\n *ngIf=\"isFile(attachment)\"\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 <div class=\"str-chat__message-attachment-file--item-first-row\">\n <div\n data-testclass=\"file-title\"\n class=\"str-chat__message-attachment-file--item-name\"\n >\n {{ attachment.title }}\n </div>\n <a\n class=\"str-chat__message-attachment-file--item-download\"\n data-testclass=\"file-link\"\n download\n href=\"{{ attachment.asset_url }}\"\n target=\"_blank\"\n >\n <stream-icon-placeholder\n class=\"str-chat__message-attachment-download-icon\"\n icon=\"download\"\n ></stream-icon-placeholder>\n </a>\n </div>\n <span\n class=\"str-chat__message-attachment-file--item-size\"\n data-testclass=\"size\"\n *ngIf=\"hasFileSize(attachment)\"\n >{{ getFileSize(attachment) }}</span\n >\n </div>\n </div>\n <div\n *ngIf=\"\n isCard(attachment) &&\n getCardAttachmentConfiguration(attachment) as attachmentConfiguration\n \"\n class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n attachment.type\n }}\"\n >\n <div\n *ngIf=\"attachmentConfiguration.url\"\n class=\"str-chat__message-attachment-card--header\"\n >\n <img\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=\"attachment.title\"\n data-testclass=\"card-title\"\n class=\"str-chat__message-attachment-card--title\"\n >\n {{ attachment.title }}\n </div>\n <div\n *ngIf=\"attachment.text\"\n class=\"str-chat__message-attachment-card--text\"\n data-testclass=\"card-text\"\n >\n {{ attachment.text }}\n </div>\n <a\n class=\"str-chat__message-attachment-card--url\"\n *ngIf=\"attachment.title_link || attachment.og_scrape_url\"\n data-testclass=\"url-link\"\n noopener\n noreferrer\n href=\"{{ attachment.title_link || attachment.og_scrape_url }}\"\n target=\"_blank\"\n >\n {{ trimUrl(attachment.title_link || attachment.og_scrape_url) }}\n </a>\n </div>\n </div>\n </div>\n <div\n class=\"str-chat__message-attachment-actions\"\n *ngIf=\"attachment.actions && attachment.actions.length > 0\"\n >\n <div class=\"str-chat__message-attachment-actions-form\">\n <button\n *ngFor=\"\n let action of attachment.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 </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 <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper\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 <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 [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 <button\n class=\"\n stream-chat-angular__image-modal-stepper\n str-chat__image-carousel-stepper\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</ng-template>\n", components: [{ type: i5.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon", "size"] }, { type: i6.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "translate": i8.TranslatePipe, "async": i7.AsyncPipe } });
135
174
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentListComponent, decorators: [{
136
175
  type: Component,
137
176
  args: [{
@@ -139,7 +178,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
139
178
  templateUrl: './attachment-list.component.html',
140
179
  styles: [],
141
180
  }]
142
- }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ImageLoadService }, { type: i3.ChannelService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
181
+ }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ChannelService }, { type: i3.AttachmentConfigurationService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
143
182
  type: Input
144
183
  }], parentMessageId: [{
145
184
  type: Input
@@ -151,4 +190,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
151
190
  type: ViewChild,
152
191
  args: ['modalContent', { static: true }]
153
192
  }] } });
154
- //# 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  HostBinding,\n  Input,\n  OnChanges,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { Action, Attachment } from 'stream-chat';\nimport { ImageLoadService } from '../message-list/image-load.service';\nimport { ModalContext, DefaultStreamChatGenerics } 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 { ThemeService } from '../theme.service';\n\n/**\n * The `AttachmentList` compontent 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  @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\n  constructor(\n    public readonly customTemplatesService: CustomTemplatesService,\n    private imageLoadService: ImageLoadService,\n    private channelService: ChannelService,\n    themeService: ThemeService\n  ) {\n    this.themeVersion = themeService.themeVersion;\n  }\n\n  ngOnChanges(): void {\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.isFile(a)),\n    ];\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  trackById(index: number) {\n    return index;\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  imageLoaded() {\n    this.imageLoadService.imageLoad$.next();\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.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  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.imagesToView = [];\n  }\n}\n","<div *ngIf=\"orderedAttachments.length > 0\" class=\"str-chat__attachment-list\">\n  <ng-container\n    *ngFor=\"let attachment of orderedAttachments; trackBy: trackById\"\n  >\n    <div\n      data-testclass=\"attachment-container\"\n      class=\"str-chat__message-attachment str-chat__message-attachment--{{\n        attachment.type\n      }}\"\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-with-actions]=\"\n        attachment.actions && attachment.actions.length > 0\n      \"\n      [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n    >\n      <img\n        *ngIf=\"isImage(attachment)\"\n        class=\"str-chat__message-attachment--img\"\n        data-testclass=\"image\"\n        [src]=\"\n          attachment.img_url || attachment.thumb_url || attachment.image_url\n        \"\n        [alt]=\"attachment?.fallback\"\n        (load)=\"imageLoaded()\"\n        (click)=\"openImageModal([attachment])\"\n        (keyup.enter)=\"openImageModal([attachment])\"\n      />\n      <div\n        class=\"str-chat__gallery\"\n        data-testid=\"image-gallery\"\n        *ngIf=\"isGallery(attachment)\"\n        [class.str-chat__gallery--square]=\"(attachment?.images)!.length > 3\"\n        [class.str-chat__gallery-two-rows]=\"(attachment?.images)!.length > 2\"\n      >\n        <ng-container\n          *ngFor=\"\n            let galleryImage of attachment.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(attachment.images!, index)\"\n            (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n            [class.str-chat__message-attachment--svg-image]=\"\n              isSvg(galleryImage)\n            \"\n          >\n            <img\n              [src]=\"\n                galleryImage.img_url ||\n                galleryImage.thumb_url ||\n                galleryImage.image_url\n              \"\n              [alt]=\"galleryImage.fallback\"\n              (load)=\"imageLoaded()\"\n            />\n          </button>\n          <button\n            *ngIf=\"index === 3 && !isLast\"\n            class=\"str-chat__gallery-placeholder\"\n            data-testclass=\"gallery-image\"\n            (click)=\"openImageModal(attachment.images!, index)\"\n            (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n            [class.str-chat__message-attachment--svg-image]=\"\n              isSvg(galleryImage)\n            \"\n            [ngStyle]=\"{\n              'background-image':\n                'url(' +\n                (galleryImage.img_url ||\n                  galleryImage.thumb_url ||\n                  galleryImage.image_url) +\n                ')'\n            }\"\n          >\n            <p\n              [innerHTML]=\"\n                'streamChat.{{ imageCount }} more'\n                  | translate: { imageCount: attachment!.images!.length - 4 }\n              \"\n            ></p>\n          </button>\n        </ng-container>\n      </div>\n      <div class=\"str-chat__player-wrapper\" *ngIf=\"isVideo(attachment)\">\n        <video\n          class=\"str-chat__video-angular\"\n          controls\n          data-testclass=\"video-attachment\"\n          [src]=\"attachment.asset_url\"\n        ></video>\n      </div>\n      <div\n        *ngIf=\"isFile(attachment)\"\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          <div class=\"str-chat__message-attachment-file--item-first-row\">\n            <div\n              data-testclass=\"file-title\"\n              class=\"str-chat__message-attachment-file--item-name\"\n            >\n              {{ attachment.title }}\n            </div>\n            <a\n              class=\"str-chat__message-attachment-file--item-download\"\n              data-testclass=\"file-link\"\n              download\n              href=\"{{ attachment.asset_url }}\"\n              target=\"_blank\"\n            >\n              <stream-icon-placeholder\n                class=\"str-chat__message-attachment-download-icon\"\n                icon=\"download\"\n              ></stream-icon-placeholder>\n            </a>\n          </div>\n          <span\n            class=\"str-chat__message-attachment-file--item-size\"\n            data-testclass=\"size\"\n            *ngIf=\"hasFileSize(attachment)\"\n            >{{ getFileSize(attachment) }}</span\n          >\n        </div>\n      </div>\n      <div\n        *ngIf=\"isCard(attachment)\"\n        class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n          attachment.type\n        }}\"\n      >\n        <div\n          *ngIf=\"attachment.image_url || attachment.thumb_url\"\n          class=\"str-chat__message-attachment-card--header\"\n        >\n          <img\n            data-testclass=\"card-img\"\n            alt=\"{{ attachment.image_url || attachment.thumb_url }}\"\n            src=\"{{ attachment.image_url || attachment.thumb_url }}\"\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=\"attachment.title\"\n              data-testclass=\"card-title\"\n              class=\"str-chat__message-attachment-card--title\"\n            >\n              {{ attachment.title }}\n            </div>\n            <div\n              *ngIf=\"attachment.text\"\n              class=\"str-chat__message-attachment-card--text\"\n              data-testclass=\"card-text\"\n            >\n              {{ attachment.text }}\n            </div>\n            <a\n              class=\"str-chat__message-attachment-card--url\"\n              *ngIf=\"attachment.title_link || attachment.og_scrape_url\"\n              data-testclass=\"url-link\"\n              noopener\n              noreferrer\n              href=\"{{ attachment.title_link || attachment.og_scrape_url }}\"\n              target=\"_blank\"\n            >\n              {{ trimUrl(attachment.title_link || attachment.og_scrape_url) }}\n            </a>\n          </div>\n        </div>\n      </div>\n      <div\n        class=\"str-chat__message-attachment-actions\"\n        *ngIf=\"attachment.actions && attachment.actions.length > 0\"\n      >\n        <div class=\"str-chat__message-attachment-actions-form\">\n          <button\n            *ngFor=\"\n              let action of attachment.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    </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    [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    <button\n      class=\"\n        stream-chat-angular__image-modal-stepper\n        str-chat__image-carousel-stepper\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    <img\n      class=\"\n        stream-chat-angular__image-modal-image\n        str-chat__image-carousel-image\n      \"\n      data-testid=\"modal-image\"\n      [src]=\"\n        imagesToView[imagesToViewCurrentIndex].img_url ||\n        imagesToView[imagesToViewCurrentIndex].thumb_url ||\n        imagesToView[imagesToViewCurrentIndex].image_url\n      \"\n      [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n    />\n    <button\n      class=\"\n        stream-chat-angular__image-modal-stepper\n        str-chat__image-carousel-stepper\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</ng-template>\n"]}
193
+ //# 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  HostBinding,\n  Input,\n  OnChanges,\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} 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` compontent 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  @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 | VideoAttachmentConfiguration\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\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.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  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.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  getImageAttachmentConfiguration(\n    attachment: Attachment,\n    type: 'gallery' | 'single',\n    element: HTMLElement\n  ) {\n    const existingConfiguration = this.attachmentConfigurations.get(attachment);\n    if (existingConfiguration) {\n      return existingConfiguration;\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  ) {\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.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-with-actions]=\"\n        attachment.actions && attachment.actions.length > 0\n      \"\n      [class.str-chat__message-attachment--svg-image]=\"isSvg(attachment)\"\n    >\n      <img\n        #imgElement\n        *ngIf=\"isImage(attachment)\"\n        class=\"str-chat__message-attachment--img\"\n        data-testclass=\"image\"\n        [src]=\"\n          getImageAttachmentConfiguration(attachment, 'single', imgElement).url\n        \"\n        [alt]=\"attachment?.fallback\"\n        (click)=\"openImageModal([attachment])\"\n        (keyup.enter)=\"openImageModal([attachment])\"\n        [ngStyle]=\"{\n          height: getImageAttachmentConfiguration(\n            attachment,\n            'single',\n            imgElement\n          ).height,\n          width: getImageAttachmentConfiguration(\n            attachment,\n            'single',\n            imgElement\n          ).width\n        }\"\n      />\n      <div\n        class=\"str-chat__gallery\"\n        data-testid=\"image-gallery\"\n        *ngIf=\"isGallery(attachment)\"\n        [class.str-chat__gallery--square]=\"(attachment?.images)!.length > 3\"\n        [class.str-chat__gallery-two-rows]=\"(attachment?.images)!.length > 2\"\n      >\n        <ng-container\n          *ngFor=\"\n            let galleryImage of attachment.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(attachment.images!, index)\"\n            (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n            [class.str-chat__message-attachment--svg-image]=\"\n              isSvg(galleryImage)\n            \"\n          >\n            <img\n              #imgElement\n              [src]=\"\n                getImageAttachmentConfiguration(\n                  galleryImage,\n                  'gallery',\n                  imgElement\n                ).url\n              \"\n              [alt]=\"galleryImage.fallback\"\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(attachment.images!, index)\"\n            (keyup.enter)=\"openImageModal(attachment.images!, index)\"\n            [class.str-chat__message-attachment--svg-image]=\"\n              isSvg(galleryImage)\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: { imageCount: attachment!.images!.length - 4 }\n              \"\n            ></p>\n          </button>\n        </ng-container>\n      </div>\n      <div class=\"str-chat__player-wrapper\" *ngIf=\"isVideo(attachment)\">\n        <video\n          #videoElement\n          class=\"str-chat__video-angular\"\n          controls\n          data-testclass=\"video-attachment\"\n          [src]=\"getVideoAttachmentConfiguration(attachment, videoElement).url\"\n          [ngStyle]=\"{\n            height: getVideoAttachmentConfiguration(attachment, videoElement)\n              .height,\n            width: getVideoAttachmentConfiguration(attachment, videoElement)\n              .width\n          }\"\n          [poster]=\"\n            getVideoAttachmentConfiguration(attachment, videoElement).thumbUrl\n          \"\n        ></video>\n      </div>\n      <div\n        *ngIf=\"isFile(attachment)\"\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          <div class=\"str-chat__message-attachment-file--item-first-row\">\n            <div\n              data-testclass=\"file-title\"\n              class=\"str-chat__message-attachment-file--item-name\"\n            >\n              {{ attachment.title }}\n            </div>\n            <a\n              class=\"str-chat__message-attachment-file--item-download\"\n              data-testclass=\"file-link\"\n              download\n              href=\"{{ attachment.asset_url }}\"\n              target=\"_blank\"\n            >\n              <stream-icon-placeholder\n                class=\"str-chat__message-attachment-download-icon\"\n                icon=\"download\"\n              ></stream-icon-placeholder>\n            </a>\n          </div>\n          <span\n            class=\"str-chat__message-attachment-file--item-size\"\n            data-testclass=\"size\"\n            *ngIf=\"hasFileSize(attachment)\"\n            >{{ getFileSize(attachment) }}</span\n          >\n        </div>\n      </div>\n      <div\n        *ngIf=\"\n          isCard(attachment) &&\n          getCardAttachmentConfiguration(attachment) as attachmentConfiguration\n        \"\n        class=\"str-chat__message-attachment-card str-chat__message-attachment-card--{{\n          attachment.type\n        }}\"\n      >\n        <div\n          *ngIf=\"attachmentConfiguration.url\"\n          class=\"str-chat__message-attachment-card--header\"\n        >\n          <img\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=\"attachment.title\"\n              data-testclass=\"card-title\"\n              class=\"str-chat__message-attachment-card--title\"\n            >\n              {{ attachment.title }}\n            </div>\n            <div\n              *ngIf=\"attachment.text\"\n              class=\"str-chat__message-attachment-card--text\"\n              data-testclass=\"card-text\"\n            >\n              {{ attachment.text }}\n            </div>\n            <a\n              class=\"str-chat__message-attachment-card--url\"\n              *ngIf=\"attachment.title_link || attachment.og_scrape_url\"\n              data-testclass=\"url-link\"\n              noopener\n              noreferrer\n              href=\"{{ attachment.title_link || attachment.og_scrape_url }}\"\n              target=\"_blank\"\n            >\n              {{ trimUrl(attachment.title_link || attachment.og_scrape_url) }}\n            </a>\n          </div>\n        </div>\n      </div>\n      <div\n        class=\"str-chat__message-attachment-actions\"\n        *ngIf=\"attachment.actions && attachment.actions.length > 0\"\n      >\n        <div class=\"str-chat__message-attachment-actions-form\">\n          <button\n            *ngFor=\"\n              let action of attachment.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    </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    <button\n      class=\"\n        stream-chat-angular__image-modal-stepper\n        str-chat__image-carousel-stepper\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    <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      [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    <button\n      class=\"\n        stream-chat-angular__image-modal-stepper\n        str-chat__image-carousel-stepper\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</ng-template>\n"]}