stream-chat-angular 4.2.0 → 4.4.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.
@@ -1,2 +1,2 @@
1
- export const version = '4.2.0';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjIuMCc7XG4iXX0=
1
+ export const version = '4.4.0';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2Fzc2V0cy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgdmVyc2lvbiA9ICc0LjQuMCc7XG4iXX0=
@@ -24,7 +24,8 @@ export class AttachmentConfigurationService {
24
24
  attachment.thumb_url ||
25
25
  attachment.image_url ||
26
26
  ''));
27
- const { sizeRestriction, height } = this.getSizingRestrictions(url, element);
27
+ const displayWarning = location === 'gallery' || location === 'single';
28
+ const { sizeRestriction, height } = this.getSizingRestrictions(url, element, displayWarning);
28
29
  if (sizeRestriction) {
29
30
  // Apply 2x for retina displays
30
31
  sizeRestriction.height *= 2;
@@ -50,7 +51,8 @@ export class AttachmentConfigurationService {
50
51
  let thumbUrl = undefined;
51
52
  if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {
52
53
  const url = new URL(attachment.thumb_url);
53
- const { sizeRestriction, height } = this.getSizingRestrictions(url, element);
54
+ const displayWarning = true;
55
+ const { sizeRestriction, height } = this.getSizingRestrictions(url, element, displayWarning);
54
56
  if (sizeRestriction) {
55
57
  sizeRestriction.height *= 2;
56
58
  sizeRestriction.width *= 2;
@@ -104,7 +106,7 @@ export class AttachmentConfigurationService {
104
106
  url.searchParams.set('h', sizeRestriction.height.toString());
105
107
  url.searchParams.set('w', sizeRestriction.width.toString());
106
108
  }
107
- getSizingRestrictions(url, htmlElement) {
109
+ getSizingRestrictions(url, htmlElement, displayWarning = false) {
108
110
  const urlParams = url.searchParams;
109
111
  const originalHeight = Number(urlParams.get('oh')) || 1;
110
112
  const originalWidth = Number(urlParams.get('ow')) || 1;
@@ -127,6 +129,9 @@ export class AttachmentConfigurationService {
127
129
  }
128
130
  else {
129
131
  sizeRestriction = undefined;
132
+ if (displayWarning) {
133
+ console.warn(`Invalid value set for height/max-height and/or max-width for HTML element, this can cause scrolling issues inside the message list, more info https://getstream.io/chat/docs/sdk/angular/components/AttachmentListComponent/#image-and-video-sizing`);
134
+ }
130
135
  }
131
136
  return { sizeRestriction, height };
132
137
  }
@@ -155,4 +160,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
155
160
  providedIn: 'root',
156
161
  }]
157
162
  }] });
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment-configuration.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/attachment-configuration.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;AAQ3C;;GAEG;AAIH,MAAM,OAAO,8BAA8B;IAH3C;QAiCE;;WAEG;QACH,iCAA4B,GAAG,IAAI,CAAC;KAqNrC;IAnNC;;;;;OAKG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,QAA2C,EAC3C,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,QAAQ,EACR,OAAO,CACR,CAAC;SACH;QAED,MAAM,GAAG,GAAG,IAAI,GAAG,CACjB,CAAC,UAAU,CAAC,OAAO;YACjB,UAAU,CAAC,SAAS;YACpB,UAAU,CAAC,SAAS;YACpB,EAAE,CAAW,CAChB,CAAC;QACF,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,CACR,CAAC;QAEF,IAAI,eAAe,EAAE;YACnB,+BAA+B;YAC/B,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;YAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;YAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;SACnD;QAED,OAAO;YACL,GAAG,EAAE,GAAG,CAAC,IAAI;YACb,KAAK,EAAE,EAAE;YACT,MAAM;SACP,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,OAAO,CACR,CAAC;SACH;QAED,IAAI,gBAAgB,GAAG,EAAE,CAAC;QAC1B,IAAI,QAAQ,GAAG,SAAS,CAAC;QACzB,IAAI,UAAU,CAAC,SAAS,IAAI,IAAI,CAAC,4BAA4B,EAAE;YAC7D,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC1C,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,CACR,CAAC;YAEF,IAAI,eAAe,EAAE;gBACnB,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;gBAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;gBAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;aACnD;YACD,QAAQ,GAAG,GAAG,CAAC,IAAI,CAAC;YACpB,gBAAgB,GAAG,MAAM,CAAC;SAC3B;aAAM;YACL,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YAC/D,gBAAgB,GAAG,GACjB,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,IAAI,EAC/D,IAAI,CAAC;SACN;QACD,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,EAAE;YAC/B,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,gBAAgB;YACxB,QAAQ,EAAE,QAAQ;SACnB,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,+BAA+B,CAC7B,UAAyB;;QAEzB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CAAC,UAAU,CAAC,CAAC;SACnE;QAED,MAAM,KAAK,GAAG,MAAA,UAAU,CAAC,KAAK,0CAAE,wBAAwB,CAAC;QAEzD,OAAO;YACL,GAAG,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,GAAG,KAAI,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACrE,MAAM,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,IAAI,CAAC,CAAC,CAAC,OAAO;YACtD,KAAK,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,IAAI,CAAC,CAAC,CAAC,EAAE;SAC/C,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,sCAAsC,CACpC,UAAyB;QAEzB,IAAI,IAAI,CAAC,gDAAgD,EAAE;YACzD,OAAO,IAAI,CAAC,gDAAgD,CAAC,UAAU,CAAC,CAAC;SAC1E;QAED,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACvD,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,EAAE,EAAE,eAAe;SAC5B,CAAC;IACJ,CAAC;IAEO,sBAAsB,CAC5B,eAAkD,EAClD,GAAQ;QAER,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7D,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC9D,CAAC;IAEO,qBAAqB,CAAC,GAAQ,EAAE,WAAwB;QAC9D,MAAM,SAAS,GAAG,GAAG,CAAC,YAAY,CAAC;QACnC,MAAM,cAAc,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACxD,MAAM,aAAa,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACvD,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;QACnE,IAAI,eAA8D,CAAC;QACnE,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,IACE,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAC;YAC3D,kBAAkB,CAAC,QAAQ,EAC3B;YACA,eAAe,GAAG,IAAI,CAAC,mBAAmB,CACxC,cAAc,EACd,aAAa,EACb,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAE,EAC5D,kBAAkB,CAAC,QAAQ,CAC5B,CAAC;YACF,IAAI,kBAAkB,CAAC,SAAS,EAAE;gBAChC,MAAM,SAAS,GACb,cAAc,GAAG,CAAC,IAAI,aAAa,GAAG,CAAC;oBACrC,CAAC,CAAC,cAAc,IAAI,kBAAkB,CAAC,SAAS;wBAC9C,aAAa,IAAI,kBAAkB,CAAC,QAAQ;wBAC5C,CAAC,CAAC,cAAc;wBAChB,CAAC,CAAC,IAAI,CAAC,KAAK,CACR,IAAI,CAAC,GAAG,CACN,kBAAkB,CAAC,SAAS,EAC5B,CAAC,kBAAkB,CAAC,QAAQ,GAAG,aAAa,CAAC;4BAC3C,cAAc,CACjB,CACF;oBACL,CAAC,CAAC,kBAAkB,CAAC,SAAS,CAAC;gBACnC,MAAM,GAAG,GAAG,SAAS,IAAI,CAAC;aAC3B;SACF;aAAM;YACL,eAAe,GAAG,SAAS,CAAC;SAC7B;QAED,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,CAAC;IACrC,CAAC;IAEO,mBAAmB,CACzB,cAAsB,EACtB,aAAqB,EACrB,SAAiB,EACjB,QAAgB;QAEhB,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,KAAK,CAChB,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,aAAa,CAAC,GAAG,cAAc,CAAC,CACjE;YACD,KAAK,EAAE,IAAI,CAAC,KAAK,CACf,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,cAAc,CAAC,GAAG,aAAa,CAAC,CACjE;SACF,CAAC;IACJ,CAAC;IAEO,qBAAqB,CAAC,WAAwB;QACpD,MAAM,kBAAkB,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;QACzD,MAAM,MAAM,GAAG,IAAI,CAAC,mCAAmC,CACrD,kBAAkB,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAC9C,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,CAAC,mCAAmC,CACxD,kBAAkB,CAAC,gBAAgB,CAAC,YAAY,CAAC,CAClD,CAAC;QACF,MAAM,QAAQ,GAAG,IAAI,CAAC,mCAAmC,CACvD,kBAAkB,CAAC,gBAAgB,CAAC,WAAW,CAAC,CACjD,CAAC;QAEF,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,CAAC;IACzC,CAAC;IAEO,mCAAmC,CAAC,QAAgB;QAC1D,OAAO,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,SAAS,CAAC;IACzD,CAAC;;2HArPU,8BAA8B;+HAA9B,8BAA8B,cAF7B,MAAM;2FAEP,8BAA8B;kBAH1C,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport {\n  AttachmentConfigration,\n  DefaultStreamChatGenerics,\n  VideoAttachmentConfiguration,\n} from './types';\n\n/**\n * The `AttachmentConfigurationService` provides customization for certain attributes of attachments displayed inside the message component. If you're using your own CDN, you can integrate resizing features of it by providing your own handlers.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class AttachmentConfigurationService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * A custom handler can be provided to override the default image attachment (images uploaded from files) configuration. By default the SDK uses fixed image height (a size that's known before image is loaded), if you override that with dynamic image height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    type: 'gallery' | 'single' | 'carousel',\n    containerElement: HTMLElement\n  ) => AttachmentConfigration;\n  /**\n   * A custom handler can be provided to override the default video attachment (videos uploaded from files) configuration. By default the SDK uses fixed height (a size that's known before video is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customVideoAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    containerElement: HTMLElement\n  ) => VideoAttachmentConfiguration;\n  /**\n   * A custom handler can be provided to override the default giphy attachment (GIFs sent with the /giphy command) configuration. By default the SDK uses fixed height (a size that's known before the GIF is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customGiphyAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * A custom handler can be provided to override the default scraped image attachment (images found in links inside messages) configuration. By default the SDK uses fixed height (a size that's known before image is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customScrapedImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * You can turn on/off thumbnail generation for video attachments\n   */\n  shouldGenerateVideoThumbnail = true;\n\n  /**\n   * Handles the configuration for image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param location Specifies where the image is being displayed\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getImageAttachmentConfiguration(\n    attachment: Attachment<T>,\n    location: 'gallery' | 'single' | 'carousel',\n    element: HTMLElement\n  ): AttachmentConfigration {\n    if (this.customImageAttachmentConfigurationHandler) {\n      return this.customImageAttachmentConfigurationHandler(\n        attachment,\n        location,\n        element\n      );\n    }\n\n    const url = new URL(\n      (attachment.img_url ||\n        attachment.thumb_url ||\n        attachment.image_url ||\n        '') as string\n    );\n    const { sizeRestriction, height } = this.getSizingRestrictions(\n      url,\n      element\n    );\n\n    if (sizeRestriction) {\n      // Apply 2x for retina displays\n      sizeRestriction.height *= 2;\n      sizeRestriction.width *= 2;\n      this.addResizingParamsToUrl(sizeRestriction, url);\n    }\n\n    return {\n      url: url.href,\n      width: '', // Not set to respect responsive width\n      height,\n    };\n  }\n\n  /**\n   * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getVideoAttachmentConfiguration(\n    attachment: Attachment<T>,\n    element: HTMLElement\n  ): VideoAttachmentConfiguration {\n    if (this.customVideoAttachmentConfigurationHandler) {\n      return this.customVideoAttachmentConfigurationHandler(\n        attachment,\n        element\n      );\n    }\n\n    let attachmentHeight = ``;\n    let thumbUrl = undefined;\n    if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {\n      const url = new URL(attachment.thumb_url);\n      const { sizeRestriction, height } = this.getSizingRestrictions(\n        url,\n        element\n      );\n\n      if (sizeRestriction) {\n        sizeRestriction.height *= 2;\n        sizeRestriction.width *= 2;\n        this.addResizingParamsToUrl(sizeRestriction, url);\n      }\n      thumbUrl = url.href;\n      attachmentHeight = height;\n    } else {\n      const cssSizeRestriction = this.getCSSSizeRestriction(element);\n      attachmentHeight = `${\n        cssSizeRestriction.maxHeight || cssSizeRestriction.height || ''\n      }px`;\n    }\n    return {\n      url: attachment.asset_url || '',\n      width: '', // Not set to respect responsive width\n      height: attachmentHeight,\n      thumbUrl: thumbUrl,\n    };\n  }\n\n  /**\n   * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getGiphyAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customGiphyAttachmentConfigurationHandler) {\n      return this.customGiphyAttachmentConfigurationHandler(attachment);\n    }\n\n    const giphy = attachment.giphy?.fixed_height_downsampled;\n\n    return {\n      url: giphy?.url || attachment.image_url || attachment.thumb_url || '',\n      height: giphy?.height ? `${giphy?.height}px` : '300px',\n      width: giphy?.width ? `${giphy?.width}px` : '',\n    };\n  }\n\n  /**\n   * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getScrapedImageAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customScrapedImageAttachmentConfigurationHandler) {\n      return this.customScrapedImageAttachmentConfigurationHandler(attachment);\n    }\n\n    return {\n      url: attachment.image_url || attachment.thumb_url || '',\n      width: '',\n      height: '', // Set from CSS\n    };\n  }\n\n  private addResizingParamsToUrl(\n    sizeRestriction: { width: number; height: number },\n    url: URL\n  ) {\n    url.searchParams.set('h', sizeRestriction.height.toString());\n    url.searchParams.set('w', sizeRestriction.width.toString());\n  }\n\n  private getSizingRestrictions(url: URL, htmlElement: HTMLElement) {\n    const urlParams = url.searchParams;\n    const originalHeight = Number(urlParams.get('oh')) || 1;\n    const originalWidth = Number(urlParams.get('ow')) || 1;\n    const cssSizeRestriction = this.getCSSSizeRestriction(htmlElement);\n    let sizeRestriction: { width: number; height: number } | undefined;\n    let height = '';\n\n    if (\n      (cssSizeRestriction.maxHeight || cssSizeRestriction.height) &&\n      cssSizeRestriction.maxWidth\n    ) {\n      sizeRestriction = this.getSizeRestrictions(\n        originalHeight,\n        originalWidth,\n        (cssSizeRestriction.maxHeight || cssSizeRestriction.height)!,\n        cssSizeRestriction.maxWidth\n      );\n      if (cssSizeRestriction.maxHeight) {\n        const heightNum =\n          originalHeight > 1 && originalWidth > 1\n            ? originalHeight <= cssSizeRestriction.maxHeight &&\n              originalWidth <= cssSizeRestriction.maxWidth\n              ? originalHeight\n              : Math.round(\n                  Math.min(\n                    cssSizeRestriction.maxHeight,\n                    (cssSizeRestriction.maxWidth / originalWidth) *\n                      originalHeight\n                  )\n                )\n            : cssSizeRestriction.maxHeight;\n        height = `${heightNum}px`;\n      }\n    } else {\n      sizeRestriction = undefined;\n    }\n\n    return { sizeRestriction, height };\n  }\n\n  private getSizeRestrictions(\n    originalHeight: number,\n    originalWidth: number,\n    maxHeight: number,\n    maxWidth: number\n  ) {\n    return {\n      height: Math.round(\n        Math.max(maxHeight, (maxWidth / originalWidth) * originalHeight)\n      ),\n      width: Math.round(\n        Math.max(maxHeight, (maxWidth / originalHeight) * originalWidth)\n      ),\n    };\n  }\n\n  private getCSSSizeRestriction(htmlElement: HTMLElement) {\n    const computedStylesheet = getComputedStyle(htmlElement);\n    const height = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('height')\n    );\n    const maxHeight = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-height')\n    );\n    const maxWidth = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-width')\n    );\n\n    return { height, maxHeight, maxWidth };\n  }\n\n  private getValueRepresentationOfCSSProperty(property: string) {\n    return Number(property.replace('px', '')) || undefined;\n  }\n}\n"]}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment-configuration.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/attachment-configuration.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;AAQ3C;;GAEG;AAIH,MAAM,OAAO,8BAA8B;IAH3C;QAiCE;;WAEG;QACH,iCAA4B,GAAG,IAAI,CAAC;KAkOrC;IAhOC;;;;;OAKG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,QAA2C,EAC3C,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,QAAQ,EACR,OAAO,CACR,CAAC;SACH;QAED,MAAM,GAAG,GAAG,IAAI,GAAG,CACjB,CAAC,UAAU,CAAC,OAAO;YACjB,UAAU,CAAC,SAAS;YACpB,UAAU,CAAC,SAAS;YACpB,EAAE,CAAW,CAChB,CAAC;QACF,MAAM,cAAc,GAAG,QAAQ,KAAK,SAAS,IAAI,QAAQ,KAAK,QAAQ,CAAC;QACvE,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,EACP,cAAc,CACf,CAAC;QAEF,IAAI,eAAe,EAAE;YACnB,+BAA+B;YAC/B,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;YAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;YAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;SACnD;QAED,OAAO;YACL,GAAG,EAAE,GAAG,CAAC,IAAI;YACb,KAAK,EAAE,EAAE;YACT,MAAM;SACP,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACH,+BAA+B,CAC7B,UAAyB,EACzB,OAAoB;QAEpB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CACnD,UAAU,EACV,OAAO,CACR,CAAC;SACH;QAED,IAAI,gBAAgB,GAAG,EAAE,CAAC;QAC1B,IAAI,QAAQ,GAAG,SAAS,CAAC;QACzB,IAAI,UAAU,CAAC,SAAS,IAAI,IAAI,CAAC,4BAA4B,EAAE;YAC7D,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YAC1C,MAAM,cAAc,GAAG,IAAI,CAAC;YAC5B,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,qBAAqB,CAC5D,GAAG,EACH,OAAO,EACP,cAAc,CACf,CAAC;YAEF,IAAI,eAAe,EAAE;gBACnB,eAAe,CAAC,MAAM,IAAI,CAAC,CAAC;gBAC5B,eAAe,CAAC,KAAK,IAAI,CAAC,CAAC;gBAC3B,IAAI,CAAC,sBAAsB,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC;aACnD;YACD,QAAQ,GAAG,GAAG,CAAC,IAAI,CAAC;YACpB,gBAAgB,GAAG,MAAM,CAAC;SAC3B;aAAM;YACL,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YAC/D,gBAAgB,GAAG,GACjB,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,IAAI,EAC/D,IAAI,CAAC;SACN;QACD,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,EAAE;YAC/B,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,gBAAgB;YACxB,QAAQ,EAAE,QAAQ;SACnB,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,+BAA+B,CAC7B,UAAyB;;QAEzB,IAAI,IAAI,CAAC,yCAAyC,EAAE;YAClD,OAAO,IAAI,CAAC,yCAAyC,CAAC,UAAU,CAAC,CAAC;SACnE;QAED,MAAM,KAAK,GAAG,MAAA,UAAU,CAAC,KAAK,0CAAE,wBAAwB,CAAC;QAEzD,OAAO;YACL,GAAG,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,GAAG,KAAI,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACrE,MAAM,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,MAAM,IAAI,CAAC,CAAC,CAAC,OAAO;YACtD,KAAK,EAAE,CAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,EAAC,CAAC,CAAC,GAAG,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,IAAI,CAAC,CAAC,CAAC,EAAE;SAC/C,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,sCAAsC,CACpC,UAAyB;QAEzB,IAAI,IAAI,CAAC,gDAAgD,EAAE;YACzD,OAAO,IAAI,CAAC,gDAAgD,CAAC,UAAU,CAAC,CAAC;SAC1E;QAED,OAAO;YACL,GAAG,EAAE,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,SAAS,IAAI,EAAE;YACvD,KAAK,EAAE,EAAE;YACT,MAAM,EAAE,EAAE,EAAE,eAAe;SAC5B,CAAC;IACJ,CAAC;IAEO,sBAAsB,CAC5B,eAAkD,EAClD,GAAQ;QAER,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7D,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,GAAG,EAAE,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC;IAC9D,CAAC;IAEO,qBAAqB,CAC3B,GAAQ,EACR,WAAwB,EACxB,cAAc,GAAG,KAAK;QAEtB,MAAM,SAAS,GAAG,GAAG,CAAC,YAAY,CAAC;QACnC,MAAM,cAAc,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACxD,MAAM,aAAa,GAAG,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QACvD,MAAM,kBAAkB,GAAG,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;QACnE,IAAI,eAA8D,CAAC;QACnE,IAAI,MAAM,GAAG,EAAE,CAAC;QAEhB,IACE,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAC;YAC3D,kBAAkB,CAAC,QAAQ,EAC3B;YACA,eAAe,GAAG,IAAI,CAAC,mBAAmB,CACxC,cAAc,EACd,aAAa,EACb,CAAC,kBAAkB,CAAC,SAAS,IAAI,kBAAkB,CAAC,MAAM,CAAE,EAC5D,kBAAkB,CAAC,QAAQ,CAC5B,CAAC;YACF,IAAI,kBAAkB,CAAC,SAAS,EAAE;gBAChC,MAAM,SAAS,GACb,cAAc,GAAG,CAAC,IAAI,aAAa,GAAG,CAAC;oBACrC,CAAC,CAAC,cAAc,IAAI,kBAAkB,CAAC,SAAS;wBAC9C,aAAa,IAAI,kBAAkB,CAAC,QAAQ;wBAC5C,CAAC,CAAC,cAAc;wBAChB,CAAC,CAAC,IAAI,CAAC,KAAK,CACR,IAAI,CAAC,GAAG,CACN,kBAAkB,CAAC,SAAS,EAC5B,CAAC,kBAAkB,CAAC,QAAQ,GAAG,aAAa,CAAC;4BAC3C,cAAc,CACjB,CACF;oBACL,CAAC,CAAC,kBAAkB,CAAC,SAAS,CAAC;gBACnC,MAAM,GAAG,GAAG,SAAS,IAAI,CAAC;aAC3B;SACF;aAAM;YACL,eAAe,GAAG,SAAS,CAAC;YAC5B,IAAI,cAAc,EAAE;gBAClB,OAAO,CAAC,IAAI,CACV,qPAAqP,CACtP,CAAC;aACH;SACF;QAED,OAAO,EAAE,eAAe,EAAE,MAAM,EAAE,CAAC;IACrC,CAAC;IAEO,mBAAmB,CACzB,cAAsB,EACtB,aAAqB,EACrB,SAAiB,EACjB,QAAgB;QAEhB,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,KAAK,CAChB,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,aAAa,CAAC,GAAG,cAAc,CAAC,CACjE;YACD,KAAK,EAAE,IAAI,CAAC,KAAK,CACf,IAAI,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,QAAQ,GAAG,cAAc,CAAC,GAAG,aAAa,CAAC,CACjE;SACF,CAAC;IACJ,CAAC;IAEO,qBAAqB,CAAC,WAAwB;QACpD,MAAM,kBAAkB,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;QACzD,MAAM,MAAM,GAAG,IAAI,CAAC,mCAAmC,CACrD,kBAAkB,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAC9C,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,CAAC,mCAAmC,CACxD,kBAAkB,CAAC,gBAAgB,CAAC,YAAY,CAAC,CAClD,CAAC;QACF,MAAM,QAAQ,GAAG,IAAI,CAAC,mCAAmC,CACvD,kBAAkB,CAAC,gBAAgB,CAAC,WAAW,CAAC,CACjD,CAAC;QAEF,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,CAAC;IACzC,CAAC;IAEO,mCAAmC,CAAC,QAAgB;QAC1D,OAAO,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,SAAS,CAAC;IACzD,CAAC;;2HAlQU,8BAA8B;+HAA9B,8BAA8B,cAF7B,MAAM;2FAEP,8BAA8B;kBAH1C,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Attachment } from 'stream-chat';\nimport {\n  AttachmentConfigration,\n  DefaultStreamChatGenerics,\n  VideoAttachmentConfiguration,\n} from './types';\n\n/**\n * The `AttachmentConfigurationService` provides customization for certain attributes of attachments displayed inside the message component. If you're using your own CDN, you can integrate resizing features of it by providing your own handlers.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class AttachmentConfigurationService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * A custom handler can be provided to override the default image attachment (images uploaded from files) configuration. By default the SDK uses fixed image height (a size that's known before image is loaded), if you override that with dynamic image height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    type: 'gallery' | 'single' | 'carousel',\n    containerElement: HTMLElement\n  ) => AttachmentConfigration;\n  /**\n   * A custom handler can be provided to override the default video attachment (videos uploaded from files) configuration. By default the SDK uses fixed height (a size that's known before video is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customVideoAttachmentConfigurationHandler?: (\n    a: Attachment<T>,\n    containerElement: HTMLElement\n  ) => VideoAttachmentConfiguration;\n  /**\n   * A custom handler can be provided to override the default giphy attachment (GIFs sent with the /giphy command) configuration. By default the SDK uses fixed height (a size that's known before the GIF is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customGiphyAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * A custom handler can be provided to override the default scraped image attachment (images found in links inside messages) configuration. By default the SDK uses fixed height (a size that's known before image is loaded), if you override that with dynamic height (for example: height: 100%) the scrolling logic inside the message list can break.\n   */\n  customScrapedImageAttachmentConfigurationHandler?: (\n    a: Attachment<T>\n  ) => AttachmentConfigration;\n  /**\n   * You can turn on/off thumbnail generation for video attachments\n   */\n  shouldGenerateVideoThumbnail = true;\n\n  /**\n   * Handles the configuration for image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param location Specifies where the image is being displayed\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getImageAttachmentConfiguration(\n    attachment: Attachment<T>,\n    location: 'gallery' | 'single' | 'carousel',\n    element: HTMLElement\n  ): AttachmentConfigration {\n    if (this.customImageAttachmentConfigurationHandler) {\n      return this.customImageAttachmentConfigurationHandler(\n        attachment,\n        location,\n        element\n      );\n    }\n\n    const url = new URL(\n      (attachment.img_url ||\n        attachment.thumb_url ||\n        attachment.image_url ||\n        '') as string\n    );\n    const displayWarning = location === 'gallery' || location === 'single';\n    const { sizeRestriction, height } = this.getSizingRestrictions(\n      url,\n      element,\n      displayWarning\n    );\n\n    if (sizeRestriction) {\n      // Apply 2x for retina displays\n      sizeRestriction.height *= 2;\n      sizeRestriction.width *= 2;\n      this.addResizingParamsToUrl(sizeRestriction, url);\n    }\n\n    return {\n      url: url.href,\n      width: '', // Not set to respect responsive width\n      height,\n    };\n  }\n\n  /**\n   * Handles the configuration for video attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   * @param element The default resizing logics reads the height/max-height and max-width propperties of this element and reduces file size based on the given values. File size reduction is done by Stream's CDN.\n   */\n  getVideoAttachmentConfiguration(\n    attachment: Attachment<T>,\n    element: HTMLElement\n  ): VideoAttachmentConfiguration {\n    if (this.customVideoAttachmentConfigurationHandler) {\n      return this.customVideoAttachmentConfigurationHandler(\n        attachment,\n        element\n      );\n    }\n\n    let attachmentHeight = ``;\n    let thumbUrl = undefined;\n    if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {\n      const url = new URL(attachment.thumb_url);\n      const displayWarning = true;\n      const { sizeRestriction, height } = this.getSizingRestrictions(\n        url,\n        element,\n        displayWarning\n      );\n\n      if (sizeRestriction) {\n        sizeRestriction.height *= 2;\n        sizeRestriction.width *= 2;\n        this.addResizingParamsToUrl(sizeRestriction, url);\n      }\n      thumbUrl = url.href;\n      attachmentHeight = height;\n    } else {\n      const cssSizeRestriction = this.getCSSSizeRestriction(element);\n      attachmentHeight = `${\n        cssSizeRestriction.maxHeight || cssSizeRestriction.height || ''\n      }px`;\n    }\n    return {\n      url: attachment.asset_url || '',\n      width: '', // Not set to respect responsive width\n      height: attachmentHeight,\n      thumbUrl: thumbUrl,\n    };\n  }\n\n  /**\n   * Handles the configuration for giphy attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getGiphyAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customGiphyAttachmentConfigurationHandler) {\n      return this.customGiphyAttachmentConfigurationHandler(attachment);\n    }\n\n    const giphy = attachment.giphy?.fixed_height_downsampled;\n\n    return {\n      url: giphy?.url || attachment.image_url || attachment.thumb_url || '',\n      height: giphy?.height ? `${giphy?.height}px` : '300px',\n      width: giphy?.width ? `${giphy?.width}px` : '',\n    };\n  }\n\n  /**\n   * Handles the configuration for scraped image attachments, it's possible to provide your own function to override the default logic\n   * @param attachment The attachment to configure\n   */\n  getScrapedImageAttachmentConfiguration(\n    attachment: Attachment<T>\n  ): AttachmentConfigration {\n    if (this.customScrapedImageAttachmentConfigurationHandler) {\n      return this.customScrapedImageAttachmentConfigurationHandler(attachment);\n    }\n\n    return {\n      url: attachment.image_url || attachment.thumb_url || '',\n      width: '',\n      height: '', // Set from CSS\n    };\n  }\n\n  private addResizingParamsToUrl(\n    sizeRestriction: { width: number; height: number },\n    url: URL\n  ) {\n    url.searchParams.set('h', sizeRestriction.height.toString());\n    url.searchParams.set('w', sizeRestriction.width.toString());\n  }\n\n  private getSizingRestrictions(\n    url: URL,\n    htmlElement: HTMLElement,\n    displayWarning = false\n  ) {\n    const urlParams = url.searchParams;\n    const originalHeight = Number(urlParams.get('oh')) || 1;\n    const originalWidth = Number(urlParams.get('ow')) || 1;\n    const cssSizeRestriction = this.getCSSSizeRestriction(htmlElement);\n    let sizeRestriction: { width: number; height: number } | undefined;\n    let height = '';\n\n    if (\n      (cssSizeRestriction.maxHeight || cssSizeRestriction.height) &&\n      cssSizeRestriction.maxWidth\n    ) {\n      sizeRestriction = this.getSizeRestrictions(\n        originalHeight,\n        originalWidth,\n        (cssSizeRestriction.maxHeight || cssSizeRestriction.height)!,\n        cssSizeRestriction.maxWidth\n      );\n      if (cssSizeRestriction.maxHeight) {\n        const heightNum =\n          originalHeight > 1 && originalWidth > 1\n            ? originalHeight <= cssSizeRestriction.maxHeight &&\n              originalWidth <= cssSizeRestriction.maxWidth\n              ? originalHeight\n              : Math.round(\n                  Math.min(\n                    cssSizeRestriction.maxHeight,\n                    (cssSizeRestriction.maxWidth / originalWidth) *\n                      originalHeight\n                  )\n                )\n            : cssSizeRestriction.maxHeight;\n        height = `${heightNum}px`;\n      }\n    } else {\n      sizeRestriction = undefined;\n      if (displayWarning) {\n        console.warn(\n          `Invalid value set for height/max-height and/or max-width for HTML element, this can cause scrolling issues inside the message list, more info https://getstream.io/chat/docs/sdk/angular/components/AttachmentListComponent/#image-and-video-sizing`\n        );\n      }\n    }\n\n    return { sizeRestriction, height };\n  }\n\n  private getSizeRestrictions(\n    originalHeight: number,\n    originalWidth: number,\n    maxHeight: number,\n    maxWidth: number\n  ) {\n    return {\n      height: Math.round(\n        Math.max(maxHeight, (maxWidth / originalWidth) * originalHeight)\n      ),\n      width: Math.round(\n        Math.max(maxHeight, (maxWidth / originalHeight) * originalWidth)\n      ),\n    };\n  }\n\n  private getCSSSizeRestriction(htmlElement: HTMLElement) {\n    const computedStylesheet = getComputedStyle(htmlElement);\n    const height = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('height')\n    );\n    const maxHeight = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-height')\n    );\n    const maxWidth = this.getValueRepresentationOfCSSProperty(\n      computedStylesheet.getPropertyValue('max-width')\n    );\n\n    return { height, maxHeight, maxWidth };\n  }\n\n  private getValueRepresentationOfCSSProperty(property: string) {\n    return Number(property.replace('px', '')) || undefined;\n  }\n}\n"]}
@@ -16,7 +16,7 @@ export class AvatarComponent {
16
16
  this.isError = false;
17
17
  }
18
18
  get initials() {
19
- var _a, _b, _c, _d, _e, _f, _g, _h, _j;
19
+ var _a, _b, _c, _d, _e;
20
20
  let result = '';
21
21
  if (this.type === 'user') {
22
22
  result = ((_a = this.name) === null || _a === void 0 ? void 0 : _a.toString()) || '';
@@ -26,9 +26,9 @@ export class AvatarComponent {
26
26
  result = (_e = (_d = this.channel) === null || _d === void 0 ? void 0 : _d.data) === null || _e === void 0 ? void 0 : _e.name;
27
27
  }
28
28
  else {
29
- const otherMembers = Object.values(((_g = (_f = this.channel) === null || _f === void 0 ? void 0 : _f.state) === null || _g === void 0 ? void 0 : _g.members) || {}).filter((m) => { var _a; return m.user_id !== ((_a = this.chatClientService.chatClient.user) === null || _a === void 0 ? void 0 : _a.id); });
30
- if (otherMembers.length === 1) {
31
- result = ((_h = otherMembers[0].user) === null || _h === void 0 ? void 0 : _h.name) || ((_j = otherMembers[0].user) === null || _j === void 0 ? void 0 : _j.id) || '';
29
+ const otherMember = this.getOtherMemberIfOneToOneChannel();
30
+ if (otherMember) {
31
+ result = otherMember.name || otherMember.id || '';
32
32
  }
33
33
  else {
34
34
  result = '#';
@@ -37,9 +37,33 @@ export class AvatarComponent {
37
37
  }
38
38
  return result.charAt(0) || '';
39
39
  }
40
+ get fallbackChannelImage() {
41
+ if (this.type !== 'channel') {
42
+ return undefined;
43
+ }
44
+ else {
45
+ const otherMember = this.getOtherMemberIfOneToOneChannel();
46
+ if (otherMember) {
47
+ return otherMember.image;
48
+ }
49
+ else {
50
+ return undefined;
51
+ }
52
+ }
53
+ }
54
+ getOtherMemberIfOneToOneChannel() {
55
+ var _a, _b;
56
+ const otherMembers = Object.values(((_b = (_a = this.channel) === null || _a === void 0 ? void 0 : _a.state) === null || _b === void 0 ? void 0 : _b.members) || {}).filter((m) => { var _a; return m.user_id !== ((_a = this.chatClientService.chatClient.user) === null || _a === void 0 ? void 0 : _a.id); });
57
+ if (otherMembers.length === 1) {
58
+ return otherMembers[0].user;
59
+ }
60
+ else {
61
+ return undefined;
62
+ }
63
+ }
40
64
  }
41
65
  AvatarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, deps: [{ token: i1.ChatClientService }], target: i0.ɵɵFactoryTarget.Component });
42
- AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AvatarComponent, selector: "stream-avatar", inputs: { name: "name", imageUrl: "imageUrl", size: "size", location: "location", channel: "channel", user: "user", type: "type" }, ngImport: i0, template: "<div\n class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n location\n }}\"\n title=\"{{ name }}\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n fontSize: 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n>\n <img\n *ngIf=\"imageUrl && !isError; else fallback\"\n class=\"str-chat__avatar-image str-chat__avatar-image{{\n isLoaded ? ' str-chat__avatar-image--loaded' : ''\n }}\"\n src=\"{{ imageUrl }}\"\n alt=\"{{ initials }}\"\n data-testid=\"avatar-img\"\n (load)=\"isLoaded = true\"\n (error)=\"isError = true\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n objectFit: 'cover',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n />\n <ng-template #fallback>\n <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n {{ initials }}\n </div>\n </ng-template>\n</div>\n", styles: [""], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
66
+ AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AvatarComponent, selector: "stream-avatar", inputs: { name: "name", imageUrl: "imageUrl", size: "size", location: "location", channel: "channel", user: "user", type: "type" }, ngImport: i0, template: "<div\n class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n location\n }}\"\n title=\"{{ name }}\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n fontSize: 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n>\n <img\n *ngIf=\"(imageUrl || fallbackChannelImage) && !isError; else fallback\"\n class=\"str-chat__avatar-image str-chat__avatar-image{{\n isLoaded ? ' str-chat__avatar-image--loaded' : ''\n }}\"\n src=\"{{ imageUrl || fallbackChannelImage }}\"\n alt=\"{{ initials }}\"\n data-testid=\"avatar-img\"\n (load)=\"isLoaded = true\"\n (error)=\"isError = true\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n objectFit: 'cover',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n />\n <ng-template #fallback>\n <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n {{ initials }}\n </div>\n </ng-template>\n</div>\n", styles: [""], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
43
67
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, decorators: [{
44
68
  type: Component,
45
69
  args: [{
@@ -62,4 +86,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
62
86
  }], type: [{
63
87
  type: Input
64
88
  }] } });
65
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXZhdGFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9hdmF0YXIvYXZhdGFyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9hdmF0YXIvYXZhdGFyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7O0FBU2pEOztHQUVHO0FBTUgsTUFBTSxPQUFPLGVBQWU7SUFnQzFCLFlBQW9CLGlCQUFvQztRQUFwQyxzQkFBaUIsR0FBakIsaUJBQWlCLENBQW1CO1FBdkJ4RDs7V0FFRztRQUNNLFNBQUksR0FBRyxFQUFFLENBQUM7UUFpQm5CLGFBQVEsR0FBRyxLQUFLLENBQUM7UUFDakIsWUFBTyxHQUFHLEtBQUssQ0FBQztJQUUyQyxDQUFDO0lBRTVELElBQUksUUFBUTs7UUFDVixJQUFJLE1BQU0sR0FBVyxFQUFFLENBQUM7UUFDeEIsSUFBSSxJQUFJLENBQUMsSUFBSSxLQUFLLE1BQU0sRUFBRTtZQUN4QixNQUFNLEdBQUcsQ0FBQSxNQUFBLElBQUksQ0FBQyxJQUFJLDBDQUFFLFFBQVEsRUFBRSxLQUFJLEVBQUUsQ0FBQztTQUN0QzthQUFNLElBQUksSUFBSSxDQUFDLElBQUksS0FBSyxTQUFTLEVBQUU7WUFDbEMsSUFBSSxNQUFBLE1BQUEsSUFBSSxDQUFDLE9BQU8sMENBQUUsSUFBSSwwQ0FBRSxJQUFJLEVBQUU7Z0JBQzVCLE1BQU0sR0FBRyxNQUFBLE1BQUEsSUFBSSxDQUFDLE9BQU8sMENBQUUsSUFBSSwwQ0FBRSxJQUFJLENBQUM7YUFDbkM7aUJBQU07Z0JBQ0wsTUFBTSxZQUFZLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FDaEMsQ0FBQSxNQUFBLE1BQUEsSUFBSSxDQUFDLE9BQU8sMENBQUUsS0FBSywwQ0FBRSxPQUFPLEtBQUksRUFBRSxDQUNuQyxDQUFDLE1BQU0sQ0FDTixDQUFDLENBQUMsRUFBRSxFQUFFLFdBQUMsT0FBQSxDQUFDLENBQUMsT0FBTyxNQUFLLE1BQUEsSUFBSSxDQUFDLGlCQUFpQixDQUFDLFVBQVUsQ0FBQyxJQUFJLDBDQUFFLEVBQUUsQ0FBQSxDQUFBLEVBQUEsQ0FDaEUsQ0FBQztnQkFDRixJQUFJLFlBQVksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO29CQUM3QixNQUFNLEdBQUcsQ0FBQSxNQUFBLFlBQVksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLDBDQUFFLElBQUksTUFBSSxNQUFBLFlBQVksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLDBDQUFFLEVBQUUsQ0FBQSxJQUFJLEVBQUUsQ0FBQztpQkFDdkU7cUJBQU07b0JBQ0wsTUFBTSxHQUFHLEdBQUcsQ0FBQztpQkFDZDthQUNGO1NBQ0Y7UUFFRCxPQUFPLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQ2hDLENBQUM7OzRHQXhEVSxlQUFlO2dHQUFmLGVBQWUseUxDakI1Qiwrd0NBb0NBOzJGRG5CYSxlQUFlO2tCQUwzQixTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxlQUFlO29CQUN6QixXQUFXLEVBQUUseUJBQXlCO29CQUN0QyxTQUFTLEVBQUUsQ0FBQyx5QkFBeUIsQ0FBQztpQkFDdkM7d0dBS1UsSUFBSTtzQkFBWixLQUFLO2dCQUlHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBSUcsSUFBSTtzQkFBWixLQUFLO2dCQUlHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBSUcsT0FBTztzQkFBZixLQUFLO2dCQUlHLElBQUk7c0JBQVosS0FBSztnQkFJRyxJQUFJO3NCQUFaLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDaGFubmVsLCBVc2VyIH0gZnJvbSAnc3RyZWFtLWNoYXQnO1xuaW1wb3J0IHsgQ2hhdENsaWVudFNlcnZpY2UgfSBmcm9tICcuLi9jaGF0LWNsaWVudC5zZXJ2aWNlJztcbmltcG9ydCB7XG4gIEF2YXRhckxvY2F0aW9uLFxuICBBdmF0YXJUeXBlLFxuICBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzLFxufSBmcm9tICcuLi90eXBlcyc7XG5cbi8qKlxuICogVGhlIGBBdmF0YXJgIGNvbXBvbmVudCBkaXNwbGF5cyB0aGUgcHJvdmlkZWQgaW1hZ2UsIHdpdGggZmFsbGJhY2sgdG8gdGhlIGZpcnN0IGxldHRlciBvZiB0aGUgb3B0aW9uYWwgbmFtZSBpbnB1dC5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc3RyZWFtLWF2YXRhcicsXG4gIHRlbXBsYXRlVXJsOiAnLi9hdmF0YXIuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9hdmF0YXIuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgQXZhdGFyQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIEFuIG9wdGlvbmFsIG5hbWUgb2YgdGhlIGltYWdlLCB1c2VkIGZvciBmYWxsYmFjayBpbWFnZSBvciBpbWFnZSB0aXRsZSAoaWYgYGltYWdlVXJsYCBpcyBwcm92aWRlZClcbiAgICovXG4gIEBJbnB1dCgpIG5hbWU6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgLyoqXG4gICAqIFRoZSBVUkwgb2YgdGhlIGltYWdlIHRvIGJlIGRpc3BsYXllZC4gSWYgdGhlIGltYWdlIGNhbid0IGJlIGRpc3BsYXllZCB0aGUgZmlyc3QgbGV0dGVyIG9mIHRoZSBuYW1lIGlucHV0IGlzIGRpc3BsYXllZC5cbiAgICovXG4gIEBJbnB1dCgpIGltYWdlVXJsOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gIC8qKlxuICAgKiBUaGUgc2l6ZSBpbiBwaXhlbHMgb2YgdGhlIGF2YXRhciBpbWFnZS5cbiAgICovXG4gIEBJbnB1dCgpIHNpemUgPSAzMjtcbiAgLyoqXG4gICAqIFRoZSBsb2NhdGlvbiB0aGUgYXZhdGFyIHdpbGwgYmUgZGlzcGxheWVkIGluXG4gICAqL1xuICBASW5wdXQoKSBsb2NhdGlvbjogQXZhdGFyTG9jYXRpb24gfCB1bmRlZmluZWQ7XG4gIC8qKlxuICAgKiBUaGUgY2hhbm5lbCB0aGUgYXZhdGFyIGJlbG9uZ3MgdG8gKGlmIGF2YXRhciBvZiBhIGNoYW5uZWwgaXMgZGlzcGxheWVkKVxuICAgKi9cbiAgQElucHV0KCkgY2hhbm5lbD86IENoYW5uZWw8RGVmYXVsdFN0cmVhbUNoYXRHZW5lcmljcz47XG4gIC8qKlxuICAgKiBUaGUgdXNlciB0aGUgYXZhdGFyIGJlbG9uZ3MgdG8gKGlmIGF2YXRhciBvZiBhIHVzZXIgaXMgZGlzcGxheWVkKVxuICAgKi9cbiAgQElucHV0KCkgdXNlcj86IFVzZXI8RGVmYXVsdFN0cmVhbUNoYXRHZW5lcmljcz47XG4gIC8qKlxuICAgKiBUaGUgdHlwZSBvZiB0aGUgYXZhdGFyOiBjaGFubmVsIGlmIGNoYW5uZWwgYXZhdGFyIGlzIGRpc3BsYXllZCwgdXNlciBpZiB1c2VyIGF2YXRhciBpcyBkaXNwbGF5ZWRcbiAgICovXG4gIEBJbnB1dCgpIHR5cGU6IEF2YXRhclR5cGUgfCB1bmRlZmluZWQ7XG4gIGlzTG9hZGVkID0gZmFsc2U7XG4gIGlzRXJyb3IgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIGNoYXRDbGllbnRTZXJ2aWNlOiBDaGF0Q2xpZW50U2VydmljZSkge31cblxuICBnZXQgaW5pdGlhbHMoKSB7XG4gICAgbGV0IHJlc3VsdDogc3RyaW5nID0gJyc7XG4gICAgaWYgKHRoaXMudHlwZSA9PT0gJ3VzZXInKSB7XG4gICAgICByZXN1bHQgPSB0aGlzLm5hbWU/LnRvU3RyaW5nKCkgfHwgJyc7XG4gICAgfSBlbHNlIGlmICh0aGlzLnR5cGUgPT09ICdjaGFubmVsJykge1xuICAgICAgaWYgKHRoaXMuY2hhbm5lbD8uZGF0YT8ubmFtZSkge1xuICAgICAgICByZXN1bHQgPSB0aGlzLmNoYW5uZWw/LmRhdGE/Lm5hbWU7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCBvdGhlck1lbWJlcnMgPSBPYmplY3QudmFsdWVzKFxuICAgICAgICAgIHRoaXMuY2hhbm5lbD8uc3RhdGU/Lm1lbWJlcnMgfHwge31cbiAgICAgICAgKS5maWx0ZXIoXG4gICAgICAgICAgKG0pID0+IG0udXNlcl9pZCAhPT0gdGhpcy5jaGF0Q2xpZW50U2VydmljZS5jaGF0Q2xpZW50LnVzZXI/LmlkXG4gICAgICAgICk7XG4gICAgICAgIGlmIChvdGhlck1lbWJlcnMubGVuZ3RoID09PSAxKSB7XG4gICAgICAgICAgcmVzdWx0ID0gb3RoZXJNZW1iZXJzWzBdLnVzZXI/Lm5hbWUgfHwgb3RoZXJNZW1iZXJzWzBdLnVzZXI/LmlkIHx8ICcnO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJlc3VsdCA9ICcjJztcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiByZXN1bHQuY2hhckF0KDApIHx8ICcnO1xuICB9XG59XG4iLCI8ZGl2XG4gIGNsYXNzPVwic3RyLWNoYXRfX2F2YXRhciBzdHItY2hhdF9fYXZhdGFyLS1jaXJjbGUgc3RyZWFtLWNoYXRfX2F2YXRhci0te3tcbiAgICBsb2NhdGlvblxuICB9fVwiXG4gIHRpdGxlPVwie3sgbmFtZSB9fVwiXG4gIFtzdHlsZV09XCJ7XG4gICAgZmxleEJhc2lzOiAnY2FsYyh2YXIoLS1zdHItY2hhdF9fc3BhY2luZy1weCwgMXB4KSAqICcgKyBzaXplICsgJyknLFxuICAgIGZvbnRTaXplOiAnY2FsYyh2YXIoLS1zdHItY2hhdF9fc3BhY2luZy1weCwgMXB4KSAqICcgKyBzaXplIC8gMiArICcpJyxcbiAgICBoZWlnaHQ6ICdjYWxjKHZhcigtLXN0ci1jaGF0X19zcGFjaW5nLXB4LCAxcHgpICogJyArIHNpemUgKyAnKScsXG4gICAgbGluZUhlaWdodDogJ2NhbGModmFyKC0tc3RyLWNoYXRfX3NwYWNpbmctcHgsIDFweCkgKiAnICsgc2l6ZSArICcpJyxcbiAgICB3aWR0aDogJ2NhbGModmFyKC0tc3RyLWNoYXRfX3NwYWNpbmctcHgsIDFweCkgKiAnICsgc2l6ZSArICcpJ1xuICB9XCJcbj5cbiAgPGltZ1xuICAgICpuZ0lmPVwiaW1hZ2VVcmwgJiYgIWlzRXJyb3I7IGVsc2UgZmFsbGJhY2tcIlxuICAgIGNsYXNzPVwic3RyLWNoYXRfX2F2YXRhci1pbWFnZSBzdHItY2hhdF9fYXZhdGFyLWltYWdle3tcbiAgICAgIGlzTG9hZGVkID8gJyBzdHItY2hhdF9fYXZhdGFyLWltYWdlLS1sb2FkZWQnIDogJydcbiAgICB9fVwiXG4gICAgc3JjPVwie3sgaW1hZ2VVcmwgfX1cIlxuICAgIGFsdD1cInt7IGluaXRpYWxzIH19XCJcbiAgICBkYXRhLXRlc3RpZD1cImF2YXRhci1pbWdcIlxuICAgIChsb2FkKT1cImlzTG9hZGVkID0gdHJ1ZVwiXG4gICAgKGVycm9yKT1cImlzRXJyb3IgPSB0cnVlXCJcbiAgICBbc3R5bGVdPVwie1xuICAgICAgZmxleEJhc2lzOiAnY2FsYyh2YXIoLS1zdHItY2hhdF9fc3BhY2luZy1weCwgMXB4KSAqICcgKyBzaXplICsgJyknLFxuICAgICAgaGVpZ2h0OiAnY2FsYyh2YXIoLS1zdHItY2hhdF9fc3BhY2luZy1weCwgMXB4KSAqICcgKyBzaXplICsgJyknLFxuICAgICAgb2JqZWN0Rml0OiAnY292ZXInLFxuICAgICAgd2lkdGg6ICdjYWxjKHZhcigtLXN0ci1jaGF0X19zcGFjaW5nLXB4LCAxcHgpICogJyArIHNpemUgKyAnKSdcbiAgICB9XCJcbiAgLz5cbiAgPG5nLXRlbXBsYXRlICNmYWxsYmFjaz5cbiAgICA8ZGl2IGRhdGEtdGVzdGlkPVwiZmFsbGJhY2staW1nXCIgY2xhc3M9XCJzdHItY2hhdF9fYXZhdGFyLWZhbGxiYWNrXCI+XG4gICAgICB7eyBpbml0aWFscyB9fVxuICAgIDwvZGl2PlxuICA8L25nLXRlbXBsYXRlPlxuPC9kaXY+XG4iXX0=
89
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/avatar/avatar.component.ts","../../../../../projects/stream-chat-angular/src/lib/avatar/avatar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;;;;AASjD;;GAEG;AAMH,MAAM,OAAO,eAAe;IAgC1B,YAAoB,iBAAoC;QAApC,sBAAiB,GAAjB,iBAAiB,CAAmB;QAvBxD;;WAEG;QACM,SAAI,GAAG,EAAE,CAAC;QAiBnB,aAAQ,GAAG,KAAK,CAAC;QACjB,YAAO,GAAG,KAAK,CAAC;IAE2C,CAAC;IAE5D,IAAI,QAAQ;;QACV,IAAI,MAAM,GAAW,EAAE,CAAC;QACxB,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,EAAE;YACxB,MAAM,GAAG,CAAA,MAAA,IAAI,CAAC,IAAI,0CAAE,QAAQ,EAAE,KAAI,EAAE,CAAC;SACtC;aAAM,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE;YAClC,IAAI,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,IAAI,EAAE;gBAC5B,MAAM,GAAG,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,IAAI,0CAAE,IAAI,CAAC;aACnC;iBAAM;gBACL,MAAM,WAAW,GAAG,IAAI,CAAC,+BAA+B,EAAE,CAAC;gBAC3D,IAAI,WAAW,EAAE;oBACf,MAAM,GAAG,WAAW,CAAC,IAAI,IAAI,WAAW,CAAC,EAAE,IAAI,EAAE,CAAC;iBACnD;qBAAM;oBACL,MAAM,GAAG,GAAG,CAAC;iBACd;aACF;SACF;QAED,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;IAChC,CAAC;IAED,IAAI,oBAAoB;QACtB,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE;YAC3B,OAAO,SAAS,CAAC;SAClB;aAAM;YACL,MAAM,WAAW,GAAG,IAAI,CAAC,+BAA+B,EAAE,CAAC;YAC3D,IAAI,WAAW,EAAE;gBACf,OAAO,WAAW,CAAC,KAAK,CAAC;aAC1B;iBAAM;gBACL,OAAO,SAAS,CAAC;aAClB;SACF;IACH,CAAC;IAEO,+BAA+B;;QACrC,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAChC,CAAA,MAAA,MAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,0CAAE,OAAO,KAAI,EAAE,CACnC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,WAAC,OAAA,CAAC,CAAC,OAAO,MAAK,MAAA,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAI,0CAAE,EAAE,CAAA,CAAA,EAAA,CAAC,CAAC;QAC1E,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;YAC7B,OAAO,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;SAC7B;aAAM;YACL,OAAO,SAAS,CAAC;SAClB;IACH,CAAC;;4GA5EU,eAAe;gGAAf,eAAe,yLCjB5B,i0CAoCA;2FDnBa,eAAe;kBAL3B,SAAS;mBAAC;oBACT,QAAQ,EAAE,eAAe;oBACzB,WAAW,EAAE,yBAAyB;oBACtC,SAAS,EAAE,CAAC,yBAAyB,CAAC;iBACvC;wGAKU,IAAI;sBAAZ,KAAK;gBAIG,QAAQ;sBAAhB,KAAK;gBAIG,IAAI;sBAAZ,KAAK;gBAIG,QAAQ;sBAAhB,KAAK;gBAIG,OAAO;sBAAf,KAAK;gBAIG,IAAI;sBAAZ,KAAK;gBAIG,IAAI;sBAAZ,KAAK","sourcesContent":["import { Component, Input } from '@angular/core';\nimport { Channel, User } from 'stream-chat';\nimport { ChatClientService } from '../chat-client.service';\nimport {\n  AvatarLocation,\n  AvatarType,\n  DefaultStreamChatGenerics,\n} from '../types';\n\n/**\n * The `Avatar` component displays the provided image, with fallback to the first letter of the optional name input.\n */\n@Component({\n  selector: 'stream-avatar',\n  templateUrl: './avatar.component.html',\n  styleUrls: ['./avatar.component.scss'],\n})\nexport class AvatarComponent {\n  /**\n   * An optional name of the image, used for fallback image or image title (if `imageUrl` is provided)\n   */\n  @Input() name: string | undefined;\n  /**\n   * The URL of the image to be displayed. If the image can't be displayed the first letter of the name input is displayed.\n   */\n  @Input() imageUrl: string | undefined;\n  /**\n   * The size in pixels of the avatar image.\n   */\n  @Input() size = 32;\n  /**\n   * The location the avatar will be displayed in\n   */\n  @Input() location: AvatarLocation | undefined;\n  /**\n   * The channel the avatar belongs to (if avatar of a channel is displayed)\n   */\n  @Input() channel?: Channel<DefaultStreamChatGenerics>;\n  /**\n   * The user the avatar belongs to (if avatar of a user is displayed)\n   */\n  @Input() user?: User<DefaultStreamChatGenerics>;\n  /**\n   * The type of the avatar: channel if channel avatar is displayed, user if user avatar is displayed\n   */\n  @Input() type: AvatarType | undefined;\n  isLoaded = false;\n  isError = false;\n\n  constructor(private chatClientService: ChatClientService) {}\n\n  get initials() {\n    let result: string = '';\n    if (this.type === 'user') {\n      result = this.name?.toString() || '';\n    } else if (this.type === 'channel') {\n      if (this.channel?.data?.name) {\n        result = this.channel?.data?.name;\n      } else {\n        const otherMember = this.getOtherMemberIfOneToOneChannel();\n        if (otherMember) {\n          result = otherMember.name || otherMember.id || '';\n        } else {\n          result = '#';\n        }\n      }\n    }\n\n    return result.charAt(0) || '';\n  }\n\n  get fallbackChannelImage() {\n    if (this.type !== 'channel') {\n      return undefined;\n    } else {\n      const otherMember = this.getOtherMemberIfOneToOneChannel();\n      if (otherMember) {\n        return otherMember.image;\n      } else {\n        return undefined;\n      }\n    }\n  }\n\n  private getOtherMemberIfOneToOneChannel() {\n    const otherMembers = Object.values(\n      this.channel?.state?.members || {}\n    ).filter((m) => m.user_id !== this.chatClientService.chatClient.user?.id);\n    if (otherMembers.length === 1) {\n      return otherMembers[0].user;\n    } else {\n      return undefined;\n    }\n  }\n}\n","<div\n  class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n    location\n  }}\"\n  title=\"{{ name }}\"\n  [style]=\"{\n    flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    fontSize: 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')',\n    height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n    width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n  }\"\n>\n  <img\n    *ngIf=\"(imageUrl || fallbackChannelImage) && !isError; else fallback\"\n    class=\"str-chat__avatar-image str-chat__avatar-image{{\n      isLoaded ? ' str-chat__avatar-image--loaded' : ''\n    }}\"\n    src=\"{{ imageUrl || fallbackChannelImage }}\"\n    alt=\"{{ initials }}\"\n    data-testid=\"avatar-img\"\n    (load)=\"isLoaded = true\"\n    (error)=\"isError = true\"\n    [style]=\"{\n      flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n      height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n      objectFit: 'cover',\n      width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n    }\"\n  />\n  <ng-template #fallback>\n    <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n      {{ initials }}\n    </div>\n  </ng-template>\n</div>\n"]}
@@ -19,7 +19,7 @@ import transliterate from '@stream-io/transliterate';
19
19
  import * as i8 from 'angular-mentions';
20
20
  import { MentionModule } from 'angular-mentions';
21
21
 
22
- const version = '4.2.0';
22
+ const version = '4.4.0';
23
23
 
24
24
  /**
25
25
  * The `NotificationService` can be used to add or remove notifications. By default the [`NotificationList`](../components/NotificationListComponent.mdx) component displays the currently active notifications.
@@ -1684,7 +1684,8 @@ class AttachmentConfigurationService {
1684
1684
  attachment.thumb_url ||
1685
1685
  attachment.image_url ||
1686
1686
  ''));
1687
- const { sizeRestriction, height } = this.getSizingRestrictions(url, element);
1687
+ const displayWarning = location === 'gallery' || location === 'single';
1688
+ const { sizeRestriction, height } = this.getSizingRestrictions(url, element, displayWarning);
1688
1689
  if (sizeRestriction) {
1689
1690
  // Apply 2x for retina displays
1690
1691
  sizeRestriction.height *= 2;
@@ -1710,7 +1711,8 @@ class AttachmentConfigurationService {
1710
1711
  let thumbUrl = undefined;
1711
1712
  if (attachment.thumb_url && this.shouldGenerateVideoThumbnail) {
1712
1713
  const url = new URL(attachment.thumb_url);
1713
- const { sizeRestriction, height } = this.getSizingRestrictions(url, element);
1714
+ const displayWarning = true;
1715
+ const { sizeRestriction, height } = this.getSizingRestrictions(url, element, displayWarning);
1714
1716
  if (sizeRestriction) {
1715
1717
  sizeRestriction.height *= 2;
1716
1718
  sizeRestriction.width *= 2;
@@ -1764,7 +1766,7 @@ class AttachmentConfigurationService {
1764
1766
  url.searchParams.set('h', sizeRestriction.height.toString());
1765
1767
  url.searchParams.set('w', sizeRestriction.width.toString());
1766
1768
  }
1767
- getSizingRestrictions(url, htmlElement) {
1769
+ getSizingRestrictions(url, htmlElement, displayWarning = false) {
1768
1770
  const urlParams = url.searchParams;
1769
1771
  const originalHeight = Number(urlParams.get('oh')) || 1;
1770
1772
  const originalWidth = Number(urlParams.get('ow')) || 1;
@@ -1787,6 +1789,9 @@ class AttachmentConfigurationService {
1787
1789
  }
1788
1790
  else {
1789
1791
  sizeRestriction = undefined;
1792
+ if (displayWarning) {
1793
+ console.warn(`Invalid value set for height/max-height and/or max-width for HTML element, this can cause scrolling issues inside the message list, more info https://getstream.io/chat/docs/sdk/angular/components/AttachmentListComponent/#image-and-video-sizing`);
1794
+ }
1790
1795
  }
1791
1796
  return { sizeRestriction, height };
1792
1797
  }
@@ -1951,7 +1956,7 @@ class AvatarComponent {
1951
1956
  this.isError = false;
1952
1957
  }
1953
1958
  get initials() {
1954
- var _a, _b, _c, _d, _e, _f, _g, _h, _j;
1959
+ var _a, _b, _c, _d, _e;
1955
1960
  let result = '';
1956
1961
  if (this.type === 'user') {
1957
1962
  result = ((_a = this.name) === null || _a === void 0 ? void 0 : _a.toString()) || '';
@@ -1961,9 +1966,9 @@ class AvatarComponent {
1961
1966
  result = (_e = (_d = this.channel) === null || _d === void 0 ? void 0 : _d.data) === null || _e === void 0 ? void 0 : _e.name;
1962
1967
  }
1963
1968
  else {
1964
- const otherMembers = Object.values(((_g = (_f = this.channel) === null || _f === void 0 ? void 0 : _f.state) === null || _g === void 0 ? void 0 : _g.members) || {}).filter((m) => { var _a; return m.user_id !== ((_a = this.chatClientService.chatClient.user) === null || _a === void 0 ? void 0 : _a.id); });
1965
- if (otherMembers.length === 1) {
1966
- result = ((_h = otherMembers[0].user) === null || _h === void 0 ? void 0 : _h.name) || ((_j = otherMembers[0].user) === null || _j === void 0 ? void 0 : _j.id) || '';
1969
+ const otherMember = this.getOtherMemberIfOneToOneChannel();
1970
+ if (otherMember) {
1971
+ result = otherMember.name || otherMember.id || '';
1967
1972
  }
1968
1973
  else {
1969
1974
  result = '#';
@@ -1972,9 +1977,33 @@ class AvatarComponent {
1972
1977
  }
1973
1978
  return result.charAt(0) || '';
1974
1979
  }
1980
+ get fallbackChannelImage() {
1981
+ if (this.type !== 'channel') {
1982
+ return undefined;
1983
+ }
1984
+ else {
1985
+ const otherMember = this.getOtherMemberIfOneToOneChannel();
1986
+ if (otherMember) {
1987
+ return otherMember.image;
1988
+ }
1989
+ else {
1990
+ return undefined;
1991
+ }
1992
+ }
1993
+ }
1994
+ getOtherMemberIfOneToOneChannel() {
1995
+ var _a, _b;
1996
+ const otherMembers = Object.values(((_b = (_a = this.channel) === null || _a === void 0 ? void 0 : _a.state) === null || _b === void 0 ? void 0 : _b.members) || {}).filter((m) => { var _a; return m.user_id !== ((_a = this.chatClientService.chatClient.user) === null || _a === void 0 ? void 0 : _a.id); });
1997
+ if (otherMembers.length === 1) {
1998
+ return otherMembers[0].user;
1999
+ }
2000
+ else {
2001
+ return undefined;
2002
+ }
2003
+ }
1975
2004
  }
1976
2005
  AvatarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, deps: [{ token: ChatClientService }], target: i0.ɵɵFactoryTarget.Component });
1977
- AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AvatarComponent, selector: "stream-avatar", inputs: { name: "name", imageUrl: "imageUrl", size: "size", location: "location", channel: "channel", user: "user", type: "type" }, ngImport: i0, template: "<div\n class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n location\n }}\"\n title=\"{{ name }}\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n fontSize: 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n>\n <img\n *ngIf=\"imageUrl && !isError; else fallback\"\n class=\"str-chat__avatar-image str-chat__avatar-image{{\n isLoaded ? ' str-chat__avatar-image--loaded' : ''\n }}\"\n src=\"{{ imageUrl }}\"\n alt=\"{{ initials }}\"\n data-testid=\"avatar-img\"\n (load)=\"isLoaded = true\"\n (error)=\"isError = true\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n objectFit: 'cover',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n />\n <ng-template #fallback>\n <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n {{ initials }}\n </div>\n </ng-template>\n</div>\n", styles: [""], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
2006
+ AvatarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: AvatarComponent, selector: "stream-avatar", inputs: { name: "name", imageUrl: "imageUrl", size: "size", location: "location", channel: "channel", user: "user", type: "type" }, ngImport: i0, template: "<div\n class=\"str-chat__avatar str-chat__avatar--circle stream-chat__avatar--{{\n location\n }}\"\n title=\"{{ name }}\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n fontSize: 'calc(var(--str-chat__spacing-px, 1px) * ' + size / 2 + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n lineHeight: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n>\n <img\n *ngIf=\"(imageUrl || fallbackChannelImage) && !isError; else fallback\"\n class=\"str-chat__avatar-image str-chat__avatar-image{{\n isLoaded ? ' str-chat__avatar-image--loaded' : ''\n }}\"\n src=\"{{ imageUrl || fallbackChannelImage }}\"\n alt=\"{{ initials }}\"\n data-testid=\"avatar-img\"\n (load)=\"isLoaded = true\"\n (error)=\"isError = true\"\n [style]=\"{\n flexBasis: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n height: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')',\n objectFit: 'cover',\n width: 'calc(var(--str-chat__spacing-px, 1px) * ' + size + ')'\n }\"\n />\n <ng-template #fallback>\n <div data-testid=\"fallback-img\" class=\"str-chat__avatar-fallback\">\n {{ initials }}\n </div>\n </ng-template>\n</div>\n", styles: [""], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
1978
2007
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: AvatarComponent, decorators: [{
1979
2008
  type: Component,
1980
2009
  args: [{