@flywheel-io/vision 2.7.0-beta.0 → 2.7.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -16,7 +16,7 @@ export declare class FwAvatarListComponent implements OnInit {
16
16
  applyBorderToChildAvatars: import("@angular/core").EffectRef;
17
17
  childrenChanged: import("@angular/core").EffectRef;
18
18
  hiddenAvatars: import("@angular/core").WritableSignal<any[]>;
19
- calcChipOverflow(): void;
19
+ calcOverflow(): void;
20
20
  static ɵfac: i0.ɵɵFactoryDeclaration<FwAvatarListComponent, never>;
21
21
  static ɵcmp: i0.ɵɵComponentDeclaration<FwAvatarListComponent, "fw-avatar-list", never, { "resizeDebounceMs": { "alias": "resizeDebounceMs"; "required": false; "isSignal": true; }; }, {}, ["avatars", "avatarElementRefs"], ["fw-avatar"], true, never>;
22
22
  }
@@ -26,7 +26,7 @@ export class FwAvatarListComponent {
26
26
  });
27
27
  this.childrenChanged = effect(() => {
28
28
  this.avatarElementRefs();
29
- this.calcChipOverflow();
29
+ this.calcOverflow();
30
30
  }, { allowSignalWrites: true });
31
31
  this.hiddenAvatars = signal([]);
32
32
  }
@@ -38,32 +38,25 @@ export class FwAvatarListComponent {
38
38
  resize$.next(resizeEntries);
39
39
  });
40
40
  resizeObserver.observe(this.hostElement.nativeElement, {});
41
- recalcOnResize$.subscribe(() => this.calcChipOverflow());
41
+ recalcOnResize$.subscribe(() => this.calcOverflow());
42
42
  }
43
- calcChipOverflow() {
43
+ calcOverflow() {
44
44
  const hostRect = this.hostElement.nativeElement.getBoundingClientRect();
45
45
  const avatarElements = this.avatarElementRefs().map(ref => ref.nativeElement);
46
46
  const moreTextElement = this.moreText().nativeElement;
47
- avatarElements.forEach(chip => chip.classList.remove('hidden'));
47
+ avatarElements.forEach(avatar => avatar.classList.remove('hidden'));
48
48
  moreTextElement.classList.remove('hidden');
49
- const visibleChips = avatarElements.filter(chip => {
50
- const chipRect = chip.getBoundingClientRect();
51
- return !(chipRect.right > hostRect.right || chipRect.bottom > hostRect.bottom);
49
+ const visibleAvatars = avatarElements.filter(avatar => {
50
+ const avatarRect = avatar.getBoundingClientRect();
51
+ return !(avatarRect.right > hostRect.right || avatarRect.bottom > hostRect.bottom);
52
52
  });
53
- const overflowingAvatars = avatarElements.toSpliced(0, visibleChips.length);
53
+ const overflowingAvatars = avatarElements.toSpliced(0, visibleAvatars.length);
54
54
  if (overflowingAvatars.length > 0) {
55
- const lastVisibleAvatar = visibleChips[visibleChips.length - 1];
55
+ const lastVisibleAvatar = visibleAvatars[visibleAvatars.length - 1];
56
56
  const enoughRoomForMoreText = (hostRect.right - lastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);
57
57
  if (!enoughRoomForMoreText) {
58
58
  overflowingAvatars.push(lastVisibleAvatar);
59
59
  }
60
- // avatars are less wide than the text so we might need to hide 2 to make room for the text
61
- // there's probably a fancy recursive solution here but ehhh
62
- const secondLastVisibleAvatar = visibleChips[visibleChips.length - 2];
63
- const needEvenMoreSpace = (hostRect.right - secondLastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);
64
- if (!needEvenMoreSpace) {
65
- overflowingAvatars.push(secondLastVisibleAvatar);
66
- }
67
60
  this.hiddenAvatars.set(overflowingAvatars);
68
61
  overflowingAvatars.forEach(hiddenChip => {
69
62
  hiddenChip.classList.add('hidden');
@@ -84,4 +77,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
84
77
  FwTooltipModule,
85
78
  ], template: "<ng-content select=\"fw-avatar\"></ng-content>\n<fw-tooltip> <!--TODO: figure out how to get a tooltip-->\n <p #moreText> +{{hiddenAvatars().length}} </p>\n</fw-tooltip>\n", styles: [":host{display:flex;align-items:center;flex-basis:max-content;flex-flow:row nowrap;justify-content:flex-start;max-height:36px;overflow:hidden}:host ::ng-deep fw-avatar:not(:first-child){margin-left:-4px}:host p{line-height:100%;cursor:default;margin:0;padding-left:8px;text-wrap:nowrap;color:var(--typography-muted)}:host::ng-deep .hidden{display:none}\n"] }]
86
79
  }] });
87
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar-list.component.js","sourceRoot":"","sources":["../../../../../src/components/avatar-list/avatar-list.component.ts","../../../../../src/components/avatar-list/avatar-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Component, contentChildren, effect, ElementRef, inject, input, OnInit, signal, viewChild } from '@angular/core';\nimport { BehaviorSubject, debounce, distinctUntilChanged, timer } from 'rxjs';\n\nimport { FwAvatarComponent } from '../avatar/avatar.component';\nimport { FwAvatarModule } from '../avatar/avatar.module';\nimport { FwTooltipModule } from '../tooltip/tooltip.module';\n\n// the nature of this component is to handle a complex layout issue so disabling this lint rule\n/* eslint-disable @rx-angular/prefer-no-layout-sensitive-apis */\n\n\n/**\n * Component for displaying and managing the overflow of a list of `fw-avatar`s\n * @see [Vision Docs](https://cdn.flywheel.io/docs/vision/master/?path=/docs/components-avatar-list--docs)\n */\n@Component({\n  selector: 'fw-avatar-list',\n  standalone: true,\n  imports: [\n    FwAvatarModule,\n    FwTooltipModule,\n  ],\n  templateUrl: './avatar-list.component.html',\n  styleUrl: './avatar-list.component.scss',\n})\nexport class FwAvatarListComponent implements OnInit {\n\n  hostElement = inject(ElementRef);\n  avatars = contentChildren<FwAvatarComponent>(FwAvatarComponent);\n  avatarElementRefs = contentChildren<FwAvatarComponent, ElementRef>(FwAvatarComponent, { read: ElementRef });\n  moreText = viewChild.required<ElementRef<HTMLParagraphElement>>('moreText');\n  changeDetector = inject(ChangeDetectorRef);\n  resizeDebounceMs = input(100);\n\n  ngOnInit(): void {\n    // wire up resize observer to rxjs and debouce\n    const resize$ = new BehaviorSubject<ResizeObserverEntry[]>(null);\n    const recalcOnResize$ = resize$.pipe(\n      distinctUntilChanged(),\n      debounce(() => timer(this.resizeDebounceMs())), // debounce otherwise it can run EVERY frame\n    );\n\n    const resizeObserver = new ResizeObserver((resizeEntries) => {\n      resize$.next(resizeEntries);\n    })\n    resizeObserver.observe(this.hostElement.nativeElement, { });\n    recalcOnResize$.subscribe(() => this.calcChipOverflow());\n  }\n\n  applyBorderToChildAvatars = effect(() => {\n    this.avatars().forEach(avatar => {\n      avatar.bordered = true\n    });\n  })\n\n  childrenChanged = effect(() => {\n    this.avatarElementRefs();\n    this.calcChipOverflow();\n  }, { allowSignalWrites: true })\n\n\n  hiddenAvatars = signal([]);\n\n  calcChipOverflow(): void {\n\n    const hostRect = (this.hostElement.nativeElement as HTMLElement).getBoundingClientRect();\n    const avatarElements = this.avatarElementRefs().map(ref => ref.nativeElement as HTMLElement);\n    const moreTextElement = this.moreText().nativeElement;\n\n    avatarElements.forEach(chip => chip.classList.remove('hidden'));\n    moreTextElement.classList.remove('hidden');\n\n    const visibleChips = avatarElements.filter(chip => {\n      const chipRect = chip.getBoundingClientRect();\n      return !(chipRect.right > hostRect.right || chipRect.bottom > hostRect.bottom);\n    })\n\n    const overflowingAvatars = avatarElements.toSpliced(0, visibleChips.length);\n\n    if (overflowingAvatars.length > 0) {\n      const lastVisibleAvatar = visibleChips[visibleChips.length - 1];\n      const enoughRoomForMoreText = (hostRect.right - lastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);\n\n      if (!enoughRoomForMoreText) {\n        overflowingAvatars.push(lastVisibleAvatar)\n      }\n\n      // avatars are less wide than the text so we might need to hide 2 to make room for the text\n      // there's probably a fancy recursive solution here but ehhh\n      const secondLastVisibleAvatar = visibleChips[visibleChips.length - 2];\n      const needEvenMoreSpace = (hostRect.right - secondLastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);\n\n      if (!needEvenMoreSpace) {\n        overflowingAvatars.push(secondLastVisibleAvatar)\n      }\n\n      this.hiddenAvatars.set(overflowingAvatars);\n      overflowingAvatars.forEach(hiddenChip => {\n        hiddenChip.classList.add('hidden');\n      })\n    } else {\n      this.moreText().nativeElement.classList.add('hidden');\n    }\n\n    this.changeDetector.detectChanges();\n  }\n\n}\n","<ng-content select=\"fw-avatar\"></ng-content>\n<fw-tooltip> <!--TODO: figure out how to get a tooltip-->\n  <p #moreText> +{{hiddenAvatars().length}} </p>\n</fw-tooltip>\n"]}
80
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar-list.component.js","sourceRoot":"","sources":["../../../../../src/components/avatar-list/avatar-list.component.ts","../../../../../src/components/avatar-list/avatar-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Component, contentChildren, effect, ElementRef, inject, input, OnInit, signal, viewChild } from '@angular/core';\nimport { BehaviorSubject, debounce, distinctUntilChanged, timer } from 'rxjs';\n\nimport { FwAvatarComponent } from '../avatar/avatar.component';\nimport { FwAvatarModule } from '../avatar/avatar.module';\nimport { FwTooltipModule } from '../tooltip/tooltip.module';\n\n// the nature of this component is to handle a complex layout issue so disabling this lint rule\n/* eslint-disable @rx-angular/prefer-no-layout-sensitive-apis */\n\n\n/**\n * Component for displaying and managing the overflow of a list of `fw-avatar`s\n * @see [Vision Docs](https://cdn.flywheel.io/docs/vision/master/?path=/docs/components-avatar-list--docs)\n */\n@Component({\n  selector: 'fw-avatar-list',\n  standalone: true,\n  imports: [\n    FwAvatarModule,\n    FwTooltipModule,\n  ],\n  templateUrl: './avatar-list.component.html',\n  styleUrl: './avatar-list.component.scss',\n})\nexport class FwAvatarListComponent implements OnInit {\n\n  hostElement = inject(ElementRef);\n  avatars = contentChildren<FwAvatarComponent>(FwAvatarComponent);\n  avatarElementRefs = contentChildren<FwAvatarComponent, ElementRef>(FwAvatarComponent, { read: ElementRef });\n  moreText = viewChild.required<ElementRef<HTMLParagraphElement>>('moreText');\n  changeDetector = inject(ChangeDetectorRef);\n  resizeDebounceMs = input(100);\n\n  ngOnInit(): void {\n    // wire up resize observer to rxjs and debouce\n    const resize$ = new BehaviorSubject<ResizeObserverEntry[]>(null);\n    const recalcOnResize$ = resize$.pipe(\n      distinctUntilChanged(),\n      debounce(() => timer(this.resizeDebounceMs())), // debounce otherwise it can run EVERY frame\n    );\n\n    const resizeObserver = new ResizeObserver((resizeEntries) => {\n      resize$.next(resizeEntries);\n    })\n    resizeObserver.observe(this.hostElement.nativeElement, { });\n    recalcOnResize$.subscribe(() => this.calcOverflow());\n  }\n\n  applyBorderToChildAvatars = effect(() => {\n    this.avatars().forEach(avatar => {\n      avatar.bordered = true\n    });\n  })\n\n  childrenChanged = effect(() => {\n    this.avatarElementRefs();\n    this.calcOverflow();\n  }, { allowSignalWrites: true })\n\n\n  hiddenAvatars = signal([]);\n\n  calcOverflow(): void {\n\n    const hostRect = (this.hostElement.nativeElement as HTMLElement).getBoundingClientRect();\n    const avatarElements = this.avatarElementRefs().map(ref => ref.nativeElement as HTMLElement);\n    const moreTextElement = this.moreText().nativeElement;\n\n    avatarElements.forEach(avatar => avatar.classList.remove('hidden'));\n    moreTextElement.classList.remove('hidden');\n\n    const visibleAvatars = avatarElements.filter(avatar => {\n      const avatarRect = avatar.getBoundingClientRect();\n      return !(avatarRect.right > hostRect.right || avatarRect.bottom > hostRect.bottom);\n    })\n\n    const overflowingAvatars = avatarElements.toSpliced(0, visibleAvatars.length);\n\n    if (overflowingAvatars.length > 0) {\n      const lastVisibleAvatar = visibleAvatars[visibleAvatars.length - 1];\n      const enoughRoomForMoreText = (hostRect.right - lastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);\n\n      if (!enoughRoomForMoreText) {\n        overflowingAvatars.push(lastVisibleAvatar)\n      }\n\n      this.hiddenAvatars.set(overflowingAvatars);\n      overflowingAvatars.forEach(hiddenChip => {\n        hiddenChip.classList.add('hidden');\n      })\n    } else {\n      this.moreText().nativeElement.classList.add('hidden');\n    }\n\n    this.changeDetector.detectChanges();\n  }\n\n}\n","<ng-content select=\"fw-avatar\"></ng-content>\n<fw-tooltip> <!--TODO: figure out how to get a tooltip-->\n  <p #moreText> +{{hiddenAvatars().length}} </p>\n</fw-tooltip>\n"]}
@@ -76,7 +76,7 @@ export class FwTooltipDirective {
76
76
  this.overlayRef.set(this.overlayService.create());
77
77
  }
78
78
  showTooltip() {
79
- if (!this.enabled()) {
79
+ if (!this.enabled() || !this.title()) {
80
80
  return;
81
81
  }
82
82
  const overlayRef = this.overlayRef();
@@ -111,4 +111,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
111
111
  standalone: true,
112
112
  }]
113
113
  }] });
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../src/components/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { ConnectedPosition, Overlay, OverlayConfig, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { ComponentRef, computed, Directive, effect, ElementRef, inject, input, model, OnInit, signal, ViewContainerRef } from '@angular/core';\n\nimport { FwTooltipPanelComponent, SimplePosition } from './tooltip-panel/tooltip-panel.component';\n\ntype DelayLength = 'none' | 'short' | 'long';\ntype DelayMap = { [K in DelayLength]: number };\n\nconst positionMap: { [key: string]: ConnectedPosition } = {\n  'none': { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: 3 },\n  'above': { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: 3 },\n  'below': { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', offsetY: -3 },\n  'left': { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center' },\n  'right': { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center' },\n};\n\nconst delayMap: DelayMap = {\n  'none': 0,\n  'short': 400,\n  'long': 700,\n}\n\nconst delayNameToMs = (name: DelayLength): number => {\n  return delayMap[name];\n}\n\n@Directive({\n  selector: '[fwTooltip]',\n  standalone: true,\n})\nexport class FwTooltipDirective implements OnInit {\n\n  // External Context\n  private elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n  private viewContainerRef = inject(ViewContainerRef);\n  private overlayService = inject(Overlay);\n\n  // Internal refs\n  private overlayRef = signal<OverlayRef>(undefined);\n  private tooltipRef = signal<ComponentRef<FwTooltipPanelComponent>>(undefined);\n  private tooltipPortal: ComponentPortal<FwTooltipPanelComponent>;\n\n  // Inputs\n  delayMs = input(400, {\n    transform: delayNameToMs,\n    alias: 'fwTooltipDelay',\n  });\n\n  // a lot of these ideally wouldn't be models,\n  // but it makes it easier to interact with this from derivative directives\n\n  title = model<string>('undefined', {\n    alias: 'fwTooltip',\n  })\n\n  position = model<SimplePosition>('above', {\n    alias: 'fwTooltipPosition',\n  });\n\n  maxWidth = model<number>(200, {\n    alias: 'fwTooltipMaxWidthPx',\n  })\n\n  enabled = model(true, {\n    alias: 'fwTooltipEnabled'\n  });\n\n  // derived state\n  connectedPosition = computed(() => [positionMap[this.position()]]);\n\n  positionStrategy = computed(() => {\n    const positionBuilder = this.overlayService.position();\n    const withOrigin = positionBuilder.flexibleConnectedTo(this.elementRef);\n    const completedStrategy = withOrigin.withPositions(this.connectedPosition());\n    return completedStrategy;\n  });\n\n  overlayConfig = computed<OverlayConfig>(() => ({\n    positionStrategy: this.positionStrategy(),\n    maxWidth: this.maxWidth(),\n    ScrollStrategyOptions: this.overlayService.scrollStrategies.close(),\n    disposeOnNavigation: true,\n  }));\n\n  // effects and methods\n  addMouseEventListeners = effect(() => {\n\n    const nativeElement = this.elementRef.nativeElement;\n\n    nativeElement.addEventListener('mouseenter', () => {\n      this.showTooltip();\n    });\n\n    nativeElement.addEventListener('mouseleave', () => {\n      this.hideTooltip();\n    });\n  })\n\n  ngOnInit(): void {\n    this.tooltipPortal = new ComponentPortal(FwTooltipPanelComponent, this.viewContainerRef);\n\n    this.overlayRef.set(this.overlayService.create());\n  }\n\n\n  private openDelayTimer: number;\n  private closeDelayTimer: number;\n\n  showTooltip(): void {\n\n    if(!this.enabled()) {\n      return;\n    }\n\n    const overlayRef = this.overlayRef();\n    overlayRef.updatePositionStrategy(this.positionStrategy());\n\n    if (this.openDelayTimer) {\n      return;\n    }\n\n    this.openDelayTimer = window.setTimeout(() => {\n      const tooltipRef = overlayRef.attach(this.tooltipPortal);\n      this.tooltipRef.set(tooltipRef);\n\n      tooltipRef.setInput('title', this.title());\n      tooltipRef.setInput('position', this.position());\n      tooltipRef.changeDetectorRef.detectChanges();\n    }, this.delayMs());\n  }\n\n  hideTooltip(): void {\n\n    const animationDelayMs = 200;\n\n\n    this.closeDelayTimer = window.setTimeout(() => {\n      window.clearTimeout(this.openDelayTimer);\n      this.openDelayTimer = undefined;\n      this.overlayRef().detach();\n      this.tooltipRef.set(undefined);\n    }, animationDelayMs);\n  }\n\n}\n"]}
114
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../src/components/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { ConnectedPosition, Overlay, OverlayConfig, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { ComponentRef, computed, Directive, effect, ElementRef, inject, input, model, OnInit, signal, ViewContainerRef } from '@angular/core';\n\nimport { FwTooltipPanelComponent, SimplePosition } from './tooltip-panel/tooltip-panel.component';\n\ntype DelayLength = 'none' | 'short' | 'long';\ntype DelayMap = { [K in DelayLength]: number };\n\nconst positionMap: { [key: string]: ConnectedPosition } = {\n  'none': { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: 3 },\n  'above': { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: 3 },\n  'below': { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', offsetY: -3 },\n  'left': { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center' },\n  'right': { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center' },\n};\n\nconst delayMap: DelayMap = {\n  'none': 0,\n  'short': 400,\n  'long': 700,\n}\n\nconst delayNameToMs = (name: DelayLength): number => {\n  return delayMap[name];\n}\n\n@Directive({\n  selector: '[fwTooltip]',\n  standalone: true,\n})\nexport class FwTooltipDirective implements OnInit {\n\n  // External Context\n  private elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n  private viewContainerRef = inject(ViewContainerRef);\n  private overlayService = inject(Overlay);\n\n  // Internal refs\n  private overlayRef = signal<OverlayRef>(undefined);\n  private tooltipRef = signal<ComponentRef<FwTooltipPanelComponent>>(undefined);\n  private tooltipPortal: ComponentPortal<FwTooltipPanelComponent>;\n\n  // Inputs\n  delayMs = input(400, {\n    transform: delayNameToMs,\n    alias: 'fwTooltipDelay',\n  });\n\n  // a lot of these ideally wouldn't be models,\n  // but it makes it easier to interact with this from derivative directives\n\n  title = model<string>('undefined', {\n    alias: 'fwTooltip',\n  })\n\n  position = model<SimplePosition>('above', {\n    alias: 'fwTooltipPosition',\n  });\n\n  maxWidth = model<number>(200, {\n    alias: 'fwTooltipMaxWidthPx',\n  })\n\n  enabled = model(true, {\n    alias: 'fwTooltipEnabled'\n  });\n\n  // derived state\n  connectedPosition = computed(() => [positionMap[this.position()]]);\n\n  positionStrategy = computed(() => {\n    const positionBuilder = this.overlayService.position();\n    const withOrigin = positionBuilder.flexibleConnectedTo(this.elementRef);\n    const completedStrategy = withOrigin.withPositions(this.connectedPosition());\n    return completedStrategy;\n  });\n\n  overlayConfig = computed<OverlayConfig>(() => ({\n    positionStrategy: this.positionStrategy(),\n    maxWidth: this.maxWidth(),\n    ScrollStrategyOptions: this.overlayService.scrollStrategies.close(),\n    disposeOnNavigation: true,\n  }));\n\n  // effects and methods\n  addMouseEventListeners = effect(() => {\n\n    const nativeElement = this.elementRef.nativeElement;\n\n    nativeElement.addEventListener('mouseenter', () => {\n      this.showTooltip();\n    });\n\n    nativeElement.addEventListener('mouseleave', () => {\n      this.hideTooltip();\n    });\n  })\n\n  ngOnInit(): void {\n    this.tooltipPortal = new ComponentPortal(FwTooltipPanelComponent, this.viewContainerRef);\n\n    this.overlayRef.set(this.overlayService.create());\n  }\n\n\n  private openDelayTimer: number;\n  private closeDelayTimer: number;\n\n  showTooltip(): void {\n\n    if(!this.enabled() || !this.title()) {\n      return;\n    }\n\n    const overlayRef = this.overlayRef();\n    overlayRef.updatePositionStrategy(this.positionStrategy());\n\n    if (this.openDelayTimer) {\n      return;\n    }\n\n    this.openDelayTimer = window.setTimeout(() => {\n      const tooltipRef = overlayRef.attach(this.tooltipPortal);\n      this.tooltipRef.set(tooltipRef);\n\n      tooltipRef.setInput('title', this.title());\n      tooltipRef.setInput('position', this.position());\n      tooltipRef.changeDetectorRef.detectChanges();\n    }, this.delayMs());\n  }\n\n  hideTooltip(): void {\n\n    const animationDelayMs = 200;\n\n\n    this.closeDelayTimer = window.setTimeout(() => {\n      window.clearTimeout(this.openDelayTimer);\n      this.openDelayTimer = undefined;\n      this.overlayRef().detach();\n      this.tooltipRef.set(undefined);\n    }, animationDelayMs);\n  }\n\n}\n"]}
@@ -1068,7 +1068,7 @@ class FwTooltipDirective {
1068
1068
  this.overlayRef.set(this.overlayService.create());
1069
1069
  }
1070
1070
  showTooltip() {
1071
- if (!this.enabled()) {
1071
+ if (!this.enabled() || !this.title()) {
1072
1072
  return;
1073
1073
  }
1074
1074
  const overlayRef = this.overlayRef();
@@ -1220,7 +1220,7 @@ class FwAvatarListComponent {
1220
1220
  });
1221
1221
  this.childrenChanged = effect(() => {
1222
1222
  this.avatarElementRefs();
1223
- this.calcChipOverflow();
1223
+ this.calcOverflow();
1224
1224
  }, { allowSignalWrites: true });
1225
1225
  this.hiddenAvatars = signal([]);
1226
1226
  }
@@ -1232,32 +1232,25 @@ class FwAvatarListComponent {
1232
1232
  resize$.next(resizeEntries);
1233
1233
  });
1234
1234
  resizeObserver.observe(this.hostElement.nativeElement, {});
1235
- recalcOnResize$.subscribe(() => this.calcChipOverflow());
1235
+ recalcOnResize$.subscribe(() => this.calcOverflow());
1236
1236
  }
1237
- calcChipOverflow() {
1237
+ calcOverflow() {
1238
1238
  const hostRect = this.hostElement.nativeElement.getBoundingClientRect();
1239
1239
  const avatarElements = this.avatarElementRefs().map(ref => ref.nativeElement);
1240
1240
  const moreTextElement = this.moreText().nativeElement;
1241
- avatarElements.forEach(chip => chip.classList.remove('hidden'));
1241
+ avatarElements.forEach(avatar => avatar.classList.remove('hidden'));
1242
1242
  moreTextElement.classList.remove('hidden');
1243
- const visibleChips = avatarElements.filter(chip => {
1244
- const chipRect = chip.getBoundingClientRect();
1245
- return !(chipRect.right > hostRect.right || chipRect.bottom > hostRect.bottom);
1243
+ const visibleAvatars = avatarElements.filter(avatar => {
1244
+ const avatarRect = avatar.getBoundingClientRect();
1245
+ return !(avatarRect.right > hostRect.right || avatarRect.bottom > hostRect.bottom);
1246
1246
  });
1247
- const overflowingAvatars = avatarElements.toSpliced(0, visibleChips.length);
1247
+ const overflowingAvatars = avatarElements.toSpliced(0, visibleAvatars.length);
1248
1248
  if (overflowingAvatars.length > 0) {
1249
- const lastVisibleAvatar = visibleChips[visibleChips.length - 1];
1249
+ const lastVisibleAvatar = visibleAvatars[visibleAvatars.length - 1];
1250
1250
  const enoughRoomForMoreText = (hostRect.right - lastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);
1251
1251
  if (!enoughRoomForMoreText) {
1252
1252
  overflowingAvatars.push(lastVisibleAvatar);
1253
1253
  }
1254
- // avatars are less wide than the text so we might need to hide 2 to make room for the text
1255
- // there's probably a fancy recursive solution here but ehhh
1256
- const secondLastVisibleAvatar = visibleChips[visibleChips.length - 2];
1257
- const needEvenMoreSpace = (hostRect.right - secondLastVisibleAvatar.getBoundingClientRect().right) > (moreTextElement.offsetWidth);
1258
- if (!needEvenMoreSpace) {
1259
- overflowingAvatars.push(secondLastVisibleAvatar);
1260
- }
1261
1254
  this.hiddenAvatars.set(overflowingAvatars);
1262
1255
  overflowingAvatars.forEach(hiddenChip => {
1263
1256
  hiddenChip.classList.add('hidden');