stream-chat-angular 4.0.0 → 4.1.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 (33) 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 +512 -176
  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 +83 -0
  8. package/esm2015/lib/attachment-list/attachment-list.component.js +22 -11
  9. package/esm2015/lib/channel-preview/channel-preview.component.js +4 -4
  10. package/esm2015/lib/channel.service.js +123 -16
  11. package/esm2015/lib/message/message.component.js +11 -2
  12. package/esm2015/lib/message-actions-box/message-actions-box.component.js +5 -3
  13. package/esm2015/lib/message-list/image-load.service.js +2 -1
  14. package/esm2015/lib/message-list/message-list.component.js +158 -75
  15. package/esm2015/lib/types.js +1 -1
  16. package/esm2015/public-api.js +2 -1
  17. package/fesm2015/stream-chat-angular.js +440 -143
  18. package/fesm2015/stream-chat-angular.js.map +1 -1
  19. package/lib/attachment-configuration.service.d.ts +47 -0
  20. package/lib/attachment-list/attachment-list.component.d.ts +6 -4
  21. package/lib/channel.service.d.ts +35 -3
  22. package/lib/message/message.component.d.ts +6 -1
  23. package/lib/message-list/image-load.service.d.ts +1 -0
  24. package/lib/message-list/message-list.component.d.ts +14 -9
  25. package/lib/types.d.ts +6 -0
  26. package/package.json +1 -1
  27. package/public-api.d.ts +1 -0
  28. package/src/assets/i18n/en.ts +2 -0
  29. package/src/assets/styles/css/index.css +1 -1
  30. package/src/assets/styles/css/index.css.map +1 -1
  31. package/src/assets/styles/scss/ImageCarousel.scss +6 -0
  32. package/src/assets/styles/scss/Message.scss +6 -1
  33. 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":"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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    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.1.0';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjEuMCc7XG4iXX0=
@@ -0,0 +1,83 @@
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.
5
+ */
6
+ export class AttachmentConfigurationService {
7
+ /**
8
+ * Handles the configuration for image attachments, it's possible to provide your own function to override the default logic
9
+ * @param attachment The attachment to configure
10
+ * @param location Specifies where the image is being displayed
11
+ */
12
+ getImageAttachmentConfiguration(attachment, location) {
13
+ if (this.customImageAttachmentConfigurationHandler) {
14
+ return this.customImageAttachmentConfigurationHandler(attachment, location);
15
+ }
16
+ const height = {
17
+ gallery: '',
18
+ single: '300px',
19
+ carousel: '', // Set from CSS
20
+ }[location];
21
+ return {
22
+ url: (attachment.img_url ||
23
+ attachment.thumb_url ||
24
+ attachment.image_url ||
25
+ ''),
26
+ width: '',
27
+ height,
28
+ };
29
+ }
30
+ /**
31
+ * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic
32
+ * @param attachment The attachment to configure
33
+ */
34
+ getVideoAttachmentConfiguration(attachment) {
35
+ if (this.customVideoAttachmentConfigurationHandler) {
36
+ return this.customVideoAttachmentConfigurationHandler(attachment);
37
+ }
38
+ return {
39
+ url: attachment.asset_url || '',
40
+ width: '100%',
41
+ height: '100%',
42
+ };
43
+ }
44
+ /**
45
+ * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic
46
+ * @param attachment The attachment to configure
47
+ */
48
+ getGiphyAttachmentConfiguration(attachment) {
49
+ var _a;
50
+ if (this.customGiphyAttachmentConfigurationHandler) {
51
+ return this.customGiphyAttachmentConfigurationHandler(attachment);
52
+ }
53
+ const giphy = (_a = attachment.giphy) === null || _a === void 0 ? void 0 : _a.fixed_height_downsampled;
54
+ return {
55
+ url: (giphy === null || giphy === void 0 ? void 0 : giphy.url) || attachment.image_url || attachment.thumb_url || '',
56
+ height: (giphy === null || giphy === void 0 ? void 0 : giphy.height) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.height}px` : '300px',
57
+ width: (giphy === null || giphy === void 0 ? void 0 : giphy.width) ? `${giphy === null || giphy === void 0 ? void 0 : giphy.width}px` : '',
58
+ };
59
+ }
60
+ /**
61
+ * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic
62
+ * @param attachment The attachment to configure
63
+ */
64
+ getScrapedImageAttachmentConfiguration(attachment) {
65
+ if (this.customScrapedImageAttachmentConfigurationHandler) {
66
+ return this.customScrapedImageAttachmentConfigurationHandler(attachment);
67
+ }
68
+ return {
69
+ url: attachment.image_url || attachment.thumb_url || '',
70
+ width: '',
71
+ height: '', // Set from CSS
72
+ };
73
+ }
74
+ }
75
+ AttachmentConfigurationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
76
+ AttachmentConfigurationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, providedIn: 'root' });
77
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentConfigurationService, decorators: [{
78
+ type: Injectable,
79
+ args: [{
80
+ providedIn: 'root',
81
+ }]
82
+ }] });
83
+ //# 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;;AAI3C;;GAEG;AAIH,MAAM,OAAO,8BAA8B;IA6BzC;;;;OAIG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,QAA2C;QAE3C,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,QAAQ,CACT,CAAC;SACH;QAED,MAAM,MAAM,GAAG;YACb,OAAO,EAAE,EAAE;YACX,MAAM,EAAE,OAAO;YACf,QAAQ,EAAE,EAAE,EAAE,eAAe;SAC9B,CAAC,QAAQ,CAAC,CAAC;QAEZ,OAAO;YACL,GAAG,EAAE,CAAC,UAAU,CAAC,OAAO;gBACtB,UAAU,CAAC,SAAS;gBACpB,UAAU,CAAC,SAAS;gBACpB,EAAE,CAAW;YACf,KAAK,EAAE,EAAE;YACT,MAAM;SACP,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,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,EAAE;YAC/B,KAAK,EAAE,MAAM;YACb,MAAM,EAAE,MAAM;SACf,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;;2HAnHU,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 { AttachmentConfigration, DefaultStreamChatGenerics } from './types';\n\n/**\n * The `AttachmentConfigurationService` provides customization for certain attributes of attachments displayed inside the message component.\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  ) => 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  ) => AttachmentConfigration;\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  /**\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   */\n  getImageAttachmentConfiguration(\n    attachment: Attachment<T>,\n    location: 'gallery' | 'single' | 'carousel'\n  ): AttachmentConfigration {\n    if (this.customImageAttachmentConfigurationHandler) {\n      return this.customImageAttachmentConfigurationHandler(\n        attachment,\n        location\n      );\n    }\n\n    const height = {\n      gallery: '', // Set from CSS,\n      single: '300px',\n      carousel: '', // Set from CSS\n    }[location];\n\n    return {\n      url: (attachment.img_url ||\n        attachment.thumb_url ||\n        attachment.image_url ||\n        '') as string,\n      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   */\n  getVideoAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customVideoAttachmentConfigurationHandler) {\n      return this.customVideoAttachmentConfigurationHandler(attachment);\n    }\n\n    return {\n      url: attachment.asset_url || '',\n      width: '100%', // Set from CSS\n      height: '100%',\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"]}
@@ -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
  */
@@ -69,9 +69,6 @@ export class AttachmentListComponent {
69
69
  (attachment.type === 'image' && !this.isImage(attachment)) ||
70
70
  attachment.type === 'giphy');
71
71
  }
72
- imageLoaded() {
73
- this.imageLoadService.imageLoad$.next();
74
- }
75
72
  hasFileSize(attachment) {
76
73
  return (attachment.file_size && Number.isFinite(Number(attachment.file_size)));
77
74
  }
@@ -112,6 +109,20 @@ export class AttachmentListComponent {
112
109
  trackByImageUrl(_, item) {
113
110
  return item.image_url || item.img_url || item.thumb_url;
114
111
  }
112
+ getImageAttachmentConfiguration(attachment, type) {
113
+ return this.attachmentConfigurationService.getImageAttachmentConfiguration(attachment, type);
114
+ }
115
+ getVideoAttachmentConfiguration(attachment) {
116
+ return this.attachmentConfigurationService.getVideoAttachmentConfiguration(attachment);
117
+ }
118
+ getCardAttachmentConfiguration(attachment) {
119
+ if (attachment.type === 'giphy') {
120
+ return this.attachmentConfigurationService.getGiphyAttachmentConfiguration(attachment);
121
+ }
122
+ else {
123
+ return this.attachmentConfigurationService.getScrapedImageAttachmentConfiguration(attachment);
124
+ }
125
+ }
115
126
  get isImageModalPrevButtonVisible() {
116
127
  return this.imagesToViewCurrentIndex !== 0;
117
128
  }
@@ -130,8 +141,8 @@ export class AttachmentListComponent {
130
141
  this.imagesToView = [];
131
142
  }
132
143
  }
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 } });
144
+ 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 });
145
+ 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=\"\n isImage(attachment) &&\n getImageAttachmentConfiguration(\n attachment,\n 'single'\n ) as attachmentConfiguration\n \"\n class=\"str-chat__message-attachment--img\"\n data-testclass=\"image\"\n [src]=\"attachmentConfiguration.url\"\n [alt]=\"attachment?.fallback\"\n (click)=\"openImageModal([attachment])\"\n (keyup.enter)=\"openImageModal([attachment])\"\n [ngStyle]=\"{\n height: attachmentConfiguration.height,\n width: attachmentConfiguration.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 *ngIf=\"\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery'\n ) as attachmentConfiguration\n \"\n [src]=\"attachmentConfiguration.url\"\n [alt]=\"galleryImage.fallback\"\n [ngStyle]=\"{\n height: attachmentConfiguration.height,\n width: attachmentConfiguration.width\n }\"\n />\n </button>\n <button\n *ngIf=\"\n index === 3 &&\n !isLast &&\n getImageAttachmentConfiguration(\n galleryImage,\n 'gallery'\n ) as attachmentConfiguration\n \"\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': 'url(' + attachmentConfiguration.url + ')',\n height: attachmentConfiguration.height,\n width: attachmentConfiguration.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\n class=\"str-chat__player-wrapper\"\n *ngIf=\"\n isVideo(attachment) &&\n getVideoAttachmentConfiguration(attachment) as attachmentConfiguration\n \"\n >\n <video\n class=\"str-chat__video-angular\"\n controls\n data-testclass=\"video-attachment\"\n [src]=\"attachmentConfiguration.url\"\n [ngStyle]=\"{\n height: attachmentConfiguration.height,\n width: attachmentConfiguration.width\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 *ngIf=\"\n getImageAttachmentConfiguration(\n imagesToView[imagesToViewCurrentIndex],\n 'carousel'\n ) as attachmentConfiguration\n \"\n class=\"\n stream-chat-angular__image-modal-image\n str-chat__image-carousel-image\n \"\n data-testid=\"modal-image\"\n [src]=\"attachmentConfiguration.url\"\n [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n [ngStyle]=\"{\n width: attachmentConfiguration.width,\n height: attachmentConfiguration.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
146
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AttachmentListComponent, decorators: [{
136
147
  type: Component,
137
148
  args: [{
@@ -139,7 +150,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
139
150
  templateUrl: './attachment-list.component.html',
140
151
  styles: [],
141
152
  }]
142
- }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ImageLoadService }, { type: i3.ChannelService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
153
+ }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ChannelService }, { type: i3.AttachmentConfigurationService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
143
154
  type: Input
144
155
  }], parentMessageId: [{
145
156
  type: Input
@@ -151,4 +162,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
151
162
  type: ViewChild,
152
163
  args: ['modalContent', { static: true }]
153
164
  }] } });
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"]}
165
+ //# 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 { 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 { 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\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(): 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  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' | 'carousel'\n  ) {\n    return this.attachmentConfigurationService.getImageAttachmentConfiguration(\n      attachment,\n      type\n    );\n  }\n\n  getVideoAttachmentConfiguration(attachment: Attachment) {\n    return this.attachmentConfigurationService.getVideoAttachmentConfiguration(\n      attachment\n    );\n  }\n\n  getCardAttachmentConfiguration(attachment: Attachment) {\n    if (attachment.type === 'giphy') {\n      return this.attachmentConfigurationService.getGiphyAttachmentConfiguration(\n        attachment\n      );\n    } else {\n      return this.attachmentConfigurationService.getScrapedImageAttachmentConfiguration(\n        attachment\n      );\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: 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=\"\n          isImage(attachment) &&\n          getImageAttachmentConfiguration(\n            attachment,\n            'single'\n          ) as attachmentConfiguration\n        \"\n        class=\"str-chat__message-attachment--img\"\n        data-testclass=\"image\"\n        [src]=\"attachmentConfiguration.url\"\n        [alt]=\"attachment?.fallback\"\n        (click)=\"openImageModal([attachment])\"\n        (keyup.enter)=\"openImageModal([attachment])\"\n        [ngStyle]=\"{\n          height: attachmentConfiguration.height,\n          width: attachmentConfiguration.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              *ngIf=\"\n                getImageAttachmentConfiguration(\n                  galleryImage,\n                  'gallery'\n                ) as attachmentConfiguration\n              \"\n              [src]=\"attachmentConfiguration.url\"\n              [alt]=\"galleryImage.fallback\"\n              [ngStyle]=\"{\n                height: attachmentConfiguration.height,\n                width: attachmentConfiguration.width\n              }\"\n            />\n          </button>\n          <button\n            *ngIf=\"\n              index === 3 &&\n              !isLast &&\n              getImageAttachmentConfiguration(\n                galleryImage,\n                'gallery'\n              ) as attachmentConfiguration\n            \"\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': 'url(' + attachmentConfiguration.url + ')',\n              height: attachmentConfiguration.height,\n              width: attachmentConfiguration.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\n        class=\"str-chat__player-wrapper\"\n        *ngIf=\"\n          isVideo(attachment) &&\n          getVideoAttachmentConfiguration(attachment) as attachmentConfiguration\n        \"\n      >\n        <video\n          class=\"str-chat__video-angular\"\n          controls\n          data-testclass=\"video-attachment\"\n          [src]=\"attachmentConfiguration.url\"\n          [ngStyle]=\"{\n            height: attachmentConfiguration.height,\n            width: attachmentConfiguration.width\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      *ngIf=\"\n        getImageAttachmentConfiguration(\n          imagesToView[imagesToViewCurrentIndex],\n          'carousel'\n        ) as attachmentConfiguration\n      \"\n      class=\"\n        stream-chat-angular__image-modal-image\n        str-chat__image-carousel-image\n      \"\n      data-testid=\"modal-image\"\n      [src]=\"attachmentConfiguration.url\"\n      [alt]=\"imagesToView[imagesToViewCurrentIndex].fallback\"\n      [ngStyle]=\"{\n        width: attachmentConfiguration.width,\n        height: attachmentConfiguration.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"]}
@@ -23,7 +23,7 @@ export class ChannelPreviewComponent {
23
23
  ngOnInit() {
24
24
  var _a, _b, _c, _d;
25
25
  this.subscriptions.push(this.channelService.activeChannel$.subscribe((activeChannel) => { var _a; return (this.isActive = (activeChannel === null || activeChannel === void 0 ? void 0 : activeChannel.id) === ((_a = this.channel) === null || _a === void 0 ? void 0 : _a.id)); }));
26
- const messages = (_b = (_a = this.channel) === null || _a === void 0 ? void 0 : _a.state) === null || _b === void 0 ? void 0 : _b.messages;
26
+ const messages = (_b = (_a = this.channel) === null || _a === void 0 ? void 0 : _a.state) === null || _b === void 0 ? void 0 : _b.latestMessages;
27
27
  if (messages && messages.length > 0) {
28
28
  this.setLatestMessage(messages[messages.length - 1]);
29
29
  }
@@ -61,12 +61,12 @@ export class ChannelPreviewComponent {
61
61
  handleMessageEvent(event) {
62
62
  this.ngZone.run(() => {
63
63
  var _a, _b, _c;
64
- if (((_a = this.channel) === null || _a === void 0 ? void 0 : _a.state.messages.length) === 0) {
64
+ if (((_a = this.channel) === null || _a === void 0 ? void 0 : _a.state.latestMessages.length) === 0) {
65
65
  this.latestMessage = 'streamChat.Nothing yet...';
66
66
  return;
67
67
  }
68
68
  if (!event.message ||
69
- ((_b = this.channel) === null || _b === void 0 ? void 0 : _b.state.messages[((_c = this.channel) === null || _c === void 0 ? void 0 : _c.state.messages.length) - 1].id) !== event.message.id) {
69
+ ((_b = this.channel) === null || _b === void 0 ? void 0 : _b.state.latestMessages[((_c = this.channel) === null || _c === void 0 ? void 0 : _c.state.latestMessages.length) - 1].id) !== event.message.id) {
70
70
  return;
71
71
  }
72
72
  this.setLatestMessage(event.message);
@@ -106,4 +106,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
106
106
  }], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i0.NgZone }, { type: i2.ChatClientService }]; }, propDecorators: { channel: [{
107
107
  type: Input
108
108
  }] } });
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel-preview.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.ts","../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, NgZone, OnDestroy, OnInit } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport {\n  Channel,\n  Event,\n  FormatMessageResponse,\n  MessageResponse,\n} from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { getChannelDisplayText } from '../get-channel-display-text';\nimport { DefaultStreamChatGenerics } from '../types';\nimport { ChatClientService } from '../chat-client.service';\n\n/**\n * The `ChannelPreview` component displays a channel preview in the channel list, it consists of the image, name and latest message of the channel.\n */\n@Component({\n  selector: 'stream-channel-preview',\n  templateUrl: './channel-preview.component.html',\n  styles: [],\n})\nexport class ChannelPreviewComponent implements OnInit, OnDestroy {\n  /**\n   * The channel to be displayed\n   */\n  @Input() channel: Channel<DefaultStreamChatGenerics> | undefined;\n  isActive = false;\n  isUnread = false;\n  unreadCount: number | undefined;\n  latestMessage: string = 'streamChat.Nothing yet...';\n  private subscriptions: (Subscription | { unsubscribe: () => void })[] = [];\n  private canSendReadEvents = true;\n\n  constructor(\n    private channelService: ChannelService,\n    private ngZone: NgZone,\n    private chatClientService: ChatClientService\n  ) {}\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe(\n        (activeChannel) =>\n          (this.isActive = activeChannel?.id === this.channel?.id)\n      )\n    );\n    const messages = this.channel?.state?.messages;\n    if (messages && messages.length > 0) {\n      this.setLatestMessage(messages[messages.length - 1]);\n    }\n    this.updateUnreadState();\n    const capabilities =\n      (this.channel?.data?.own_capabilities as string[]) || [];\n    this.canSendReadEvents = capabilities.indexOf('read-events') !== -1;\n    this.subscriptions.push(\n      this.channel!.on('message.new', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.updated', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.deleted', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('channel.truncated', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.read', () =>\n        this.ngZone.run(() => {\n          this.updateUnreadState();\n        })\n      )\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get avatarImage() {\n    return this.channel?.data?.image;\n  }\n\n  get avatarName() {\n    return this.channel?.data?.name;\n  }\n\n  get title() {\n    if (!this.channel) {\n      return '';\n    }\n    return getChannelDisplayText(\n      this.channel,\n      this.chatClientService.chatClient.user!\n    );\n  }\n\n  setAsActiveChannel(): void {\n    void this.channelService.setAsActiveChannel(this.channel!);\n  }\n\n  private handleMessageEvent(event: Event) {\n    this.ngZone.run(() => {\n      if (this.channel?.state.messages.length === 0) {\n        this.latestMessage = 'streamChat.Nothing yet...';\n        return;\n      }\n      if (\n        !event.message ||\n        this.channel?.state.messages[this.channel?.state.messages.length - 1]\n          .id !== event.message.id\n      ) {\n        return;\n      }\n      this.setLatestMessage(event.message);\n      this.updateUnreadState();\n    });\n  }\n\n  private setLatestMessage(message?: FormatMessageResponse | MessageResponse) {\n    if (message?.deleted_at) {\n      this.latestMessage = 'streamChat.Message deleted';\n    } else if (message?.text) {\n      this.latestMessage = message.text;\n    } else if (message?.attachments && message.attachments.length) {\n      this.latestMessage = 'streamChat.🏙 Attachment...';\n    }\n  }\n\n  private updateUnreadState() {\n    if (this.isActive || !this.canSendReadEvents) {\n      this.unreadCount = 0;\n      this.isUnread = false;\n      return;\n    }\n    this.unreadCount = this.channel!.countUnread();\n    this.isUnread = !!this.unreadCount;\n  }\n}\n","<button\n  class=\"str-chat__channel-preview-messenger str-chat__channel-preview\"\n  [class.str-chat__channel-preview-messenger--active]=\"isActive\"\n  [class.str-chat__channel-preview--active]=\"isActive\"\n  [class.str-chat__channel-preview-messenger--unread]=\"isUnread\"\n  (click)=\"setAsActiveChannel()\"\n  data-testid=\"channel-preview-container\"\n>\n  <div class=\"str-chat__channel-preview-messenger--left\">\n    <stream-avatar-placeholder\n      name=\"{{ avatarName }}\"\n      imageUrl=\"{{ avatarImage }}\"\n      type=\"channel\"\n      [channel]=\"channel\"\n      location=\"channel-preview\"\n      [size]=\"49\"\n    ></stream-avatar-placeholder>\n  </div>\n  <div\n    class=\"\n      str-chat__channel-preview-messenger--right str-chat__channel-preview-end\n    \"\n  >\n    <div class=\"str-chat__channel-preview-end-first-row\">\n      <div class=\"str-chat__channel-preview-messenger--name\">\n        <span data-testid=\"channel-preview-title\">{{ title }}</span>\n      </div>\n      <div\n        data-testid=\"unread-badge\"\n        *ngIf=\"unreadCount\"\n        class=\"str-chat__channel-preview-unread-badge\"\n      >\n        {{ unreadCount }}\n      </div>\n    </div>\n    <div\n      data-testid=\"latest-message\"\n      class=\"str-chat__channel-preview-messenger--last-message\"\n    >\n      {{ latestMessage | translate }}\n    </div>\n  </div>\n</button>\n"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel-preview.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.ts","../../../../../projects/stream-chat-angular/src/lib/channel-preview/channel-preview.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAA6B,MAAM,eAAe,CAAC;AAS5E,OAAO,EAAE,qBAAqB,EAAE,MAAM,6BAA6B,CAAC;;;;;;;AAIpE;;GAEG;AAMH,MAAM,OAAO,uBAAuB;IAYlC,YACU,cAA8B,EAC9B,MAAc,EACd,iBAAoC;QAFpC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,WAAM,GAAN,MAAM,CAAQ;QACd,sBAAiB,GAAjB,iBAAiB,CAAmB;QAV9C,aAAQ,GAAG,KAAK,CAAC;QACjB,aAAQ,GAAG,KAAK,CAAC;QAEjB,kBAAa,GAAW,2BAA2B,CAAC;QAC5C,kBAAa,GAAmD,EAAE,CAAC;QACnE,sBAAiB,GAAG,IAAI,CAAC;IAM9B,CAAC;IAEJ,QAAQ;;QACN,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,SAAS,CAC1C,CAAC,aAAa,EAAE,EAAE,WAChB,OAAA,CAAC,IAAI,CAAC,QAAQ,GAAG,CAAA,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,EAAE,OAAK,MAAA,IAAI,CAAC,OAAO,0CAAE,EAAE,CAAA,CAAC,CAAA,EAAA,CAC3D,CACF,CAAC;QACF,MAAM,QAAQ,GAAG,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,0CAAE,cAAc,CAAC;QACrD,IAAI,QAAQ,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;YACnC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC;SACtD;QACD,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,MAAM,YAAY,GAChB,CAAC,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,gBAA6B,KAAI,EAAE,CAAC;QAC3D,IAAI,CAAC,iBAAiB,GAAG,YAAY,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC;QACpE,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,OAAQ,CAAC,EAAE,CAAC,aAAa,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CACpE,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,OAAQ,CAAC,EAAE,CAAC,iBAAiB,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CACxE,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,OAAQ,CAAC,EAAE,CAAC,iBAAiB,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CACxE,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,OAAQ,CAAC,EAAE,CAAC,mBAAmB,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAC1E,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,OAAQ,CAAC,EAAE,CAAC,cAAc,EAAE,GAAG,EAAE,CACpC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;YACnB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC3B,CAAC,CAAC,CACH,CACF,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,IAAI,WAAW;;QACb,OAAO,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,KAAK,CAAC;IACnC,CAAC;IAED,IAAI,UAAU;;QACZ,OAAO,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,IAAI,CAAC;IAClC,CAAC;IAED,IAAI,KAAK;QACP,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,OAAO,EAAE,CAAC;SACX;QACD,OAAO,qBAAqB,CAC1B,IAAI,CAAC,OAAO,EACZ,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAK,CACxC,CAAC;IACJ,CAAC;IAED,kBAAkB;QAChB,KAAK,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAQ,CAAC,CAAC;IAC7D,CAAC;IAEO,kBAAkB,CAAC,KAAY;QACrC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;;YACnB,IAAI,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,CAAC,cAAc,CAAC,MAAM,MAAK,CAAC,EAAE;gBACnD,IAAI,CAAC,aAAa,GAAG,2BAA2B,CAAC;gBACjD,OAAO;aACR;YACD,IACE,CAAC,KAAK,CAAC,OAAO;gBACd,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,CAAC,cAAc,CAChC,CAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,CAAC,cAAc,CAAC,MAAM,IAAG,CAAC,EAC7C,EAAE,MAAK,KAAK,CAAC,OAAO,CAAC,EAAE,EACzB;gBACA,OAAO;aACR;YACD,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YACrC,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,gBAAgB,CAAC,OAAiD;QACxE,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,UAAU,EAAE;YACvB,IAAI,CAAC,aAAa,GAAG,4BAA4B,CAAC;SACnD;aAAM,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,IAAI,EAAE;YACxB,IAAI,CAAC,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;SACnC;aAAM,IAAI,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,WAAW,KAAI,OAAO,CAAC,WAAW,CAAC,MAAM,EAAE;YAC7D,IAAI,CAAC,aAAa,GAAG,6BAA6B,CAAC;SACpD;IACH,CAAC;IAEO,iBAAiB;QACvB,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YAC5C,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;YACrB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,OAAO;SACR;QACD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,OAAQ,CAAC,WAAW,EAAE,CAAC;QAC/C,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC;IACrC,CAAC;;oHArHU,uBAAuB;wGAAvB,uBAAuB,8FCrBpC,63CA2CA;2FDtBa,uBAAuB;kBALnC,SAAS;mBAAC;oBACT,QAAQ,EAAE,wBAAwB;oBAClC,WAAW,EAAE,kCAAkC;oBAC/C,MAAM,EAAE,EAAE;iBACX;0JAKU,OAAO;sBAAf,KAAK","sourcesContent":["import { Component, Input, NgZone, OnDestroy, OnInit } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport {\n  Channel,\n  Event,\n  FormatMessageResponse,\n  MessageResponse,\n} from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { getChannelDisplayText } from '../get-channel-display-text';\nimport { DefaultStreamChatGenerics } from '../types';\nimport { ChatClientService } from '../chat-client.service';\n\n/**\n * The `ChannelPreview` component displays a channel preview in the channel list, it consists of the image, name and latest message of the channel.\n */\n@Component({\n  selector: 'stream-channel-preview',\n  templateUrl: './channel-preview.component.html',\n  styles: [],\n})\nexport class ChannelPreviewComponent implements OnInit, OnDestroy {\n  /**\n   * The channel to be displayed\n   */\n  @Input() channel: Channel<DefaultStreamChatGenerics> | undefined;\n  isActive = false;\n  isUnread = false;\n  unreadCount: number | undefined;\n  latestMessage: string = 'streamChat.Nothing yet...';\n  private subscriptions: (Subscription | { unsubscribe: () => void })[] = [];\n  private canSendReadEvents = true;\n\n  constructor(\n    private channelService: ChannelService,\n    private ngZone: NgZone,\n    private chatClientService: ChatClientService\n  ) {}\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe(\n        (activeChannel) =>\n          (this.isActive = activeChannel?.id === this.channel?.id)\n      )\n    );\n    const messages = this.channel?.state?.latestMessages;\n    if (messages && messages.length > 0) {\n      this.setLatestMessage(messages[messages.length - 1]);\n    }\n    this.updateUnreadState();\n    const capabilities =\n      (this.channel?.data?.own_capabilities as string[]) || [];\n    this.canSendReadEvents = capabilities.indexOf('read-events') !== -1;\n    this.subscriptions.push(\n      this.channel!.on('message.new', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.updated', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.deleted', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('channel.truncated', this.handleMessageEvent.bind(this))\n    );\n    this.subscriptions.push(\n      this.channel!.on('message.read', () =>\n        this.ngZone.run(() => {\n          this.updateUnreadState();\n        })\n      )\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  get avatarImage() {\n    return this.channel?.data?.image;\n  }\n\n  get avatarName() {\n    return this.channel?.data?.name;\n  }\n\n  get title() {\n    if (!this.channel) {\n      return '';\n    }\n    return getChannelDisplayText(\n      this.channel,\n      this.chatClientService.chatClient.user!\n    );\n  }\n\n  setAsActiveChannel(): void {\n    void this.channelService.setAsActiveChannel(this.channel!);\n  }\n\n  private handleMessageEvent(event: Event) {\n    this.ngZone.run(() => {\n      if (this.channel?.state.latestMessages.length === 0) {\n        this.latestMessage = 'streamChat.Nothing yet...';\n        return;\n      }\n      if (\n        !event.message ||\n        this.channel?.state.latestMessages[\n          this.channel?.state.latestMessages.length - 1\n        ].id !== event.message.id\n      ) {\n        return;\n      }\n      this.setLatestMessage(event.message);\n      this.updateUnreadState();\n    });\n  }\n\n  private setLatestMessage(message?: FormatMessageResponse | MessageResponse) {\n    if (message?.deleted_at) {\n      this.latestMessage = 'streamChat.Message deleted';\n    } else if (message?.text) {\n      this.latestMessage = message.text;\n    } else if (message?.attachments && message.attachments.length) {\n      this.latestMessage = 'streamChat.🏙 Attachment...';\n    }\n  }\n\n  private updateUnreadState() {\n    if (this.isActive || !this.canSendReadEvents) {\n      this.unreadCount = 0;\n      this.isUnread = false;\n      return;\n    }\n    this.unreadCount = this.channel!.countUnread();\n    this.isUnread = !!this.unreadCount;\n  }\n}\n","<button\n  class=\"str-chat__channel-preview-messenger str-chat__channel-preview\"\n  [class.str-chat__channel-preview-messenger--active]=\"isActive\"\n  [class.str-chat__channel-preview--active]=\"isActive\"\n  [class.str-chat__channel-preview-messenger--unread]=\"isUnread\"\n  (click)=\"setAsActiveChannel()\"\n  data-testid=\"channel-preview-container\"\n>\n  <div class=\"str-chat__channel-preview-messenger--left\">\n    <stream-avatar-placeholder\n      name=\"{{ avatarName }}\"\n      imageUrl=\"{{ avatarImage }}\"\n      type=\"channel\"\n      [channel]=\"channel\"\n      location=\"channel-preview\"\n      [size]=\"49\"\n    ></stream-avatar-placeholder>\n  </div>\n  <div\n    class=\"\n      str-chat__channel-preview-messenger--right str-chat__channel-preview-end\n    \"\n  >\n    <div class=\"str-chat__channel-preview-end-first-row\">\n      <div class=\"str-chat__channel-preview-messenger--name\">\n        <span data-testid=\"channel-preview-title\">{{ title }}</span>\n      </div>\n      <div\n        data-testid=\"unread-badge\"\n        *ngIf=\"unreadCount\"\n        class=\"str-chat__channel-preview-unread-badge\"\n      >\n        {{ unreadCount }}\n      </div>\n    </div>\n    <div\n      data-testid=\"latest-message\"\n      class=\"str-chat__channel-preview-messenger--last-message\"\n    >\n      {{ latestMessage | translate }}\n    </div>\n  </div>\n</button>\n"]}