@skyux/core 9.9.0 → 9.11.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.
@@ -76,7 +76,9 @@ export class SkyDynamicComponentService {
76
76
  if (!componentRef) {
77
77
  return;
78
78
  }
79
- this.#applicationRef.detachView(componentRef.hostView);
79
+ if (!this.#applicationRef.destroyed) {
80
+ this.#applicationRef.detachView(componentRef.hostView);
81
+ }
80
82
  componentRef.destroy();
81
83
  }
82
84
  #getRootNode(componentRef) {
@@ -108,4 +110,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.9", ngImpor
108
110
  providedIn: 'any',
109
111
  }]
110
112
  }] });
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-component.service.js","sourceRoot":"","sources":["../../../../../../../../libs/components/core/src/lib/modules/dynamic-component/dynamic-component.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ApplicationRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  EnvironmentInjector,\n  Injectable,\n  Renderer2,\n  RendererFactory2,\n  Type,\n  createComponent,\n  createEnvironmentInjector,\n  inject,\n} from '@angular/core';\n\nimport { SkyAppWindowRef } from '../window/window-ref';\n\nimport { SkyDynamicComponentLocation } from './dynamic-component-location';\nimport { SkyDynamicComponentOptions } from './dynamic-component-options';\n\n/**\n * Angular service for creating and rendering a dynamic component.\n * @internal\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class SkyDynamicComponentService {\n  #applicationRef: ApplicationRef;\n\n  #renderer: Renderer2;\n\n  #windowRef: SkyAppWindowRef;\n\n  #environmentInjector = inject(EnvironmentInjector);\n\n  constructor(\n    applicationRef: ApplicationRef,\n    windowRef: SkyAppWindowRef,\n    rendererFactory: RendererFactory2\n  ) {\n    this.#applicationRef = applicationRef;\n    this.#windowRef = windowRef;\n\n    // Based on suggestions from https://github.com/angular/angular/issues/17824\n    // for accessing an instance of Renderer2 in a service since Renderer2 can't\n    // be injected into a service.  Passing undefined for both parameters results\n    // in the default renderer which is what we want here.\n    this.#renderer = rendererFactory.createRenderer(undefined, null);\n  }\n\n  /**\n   * Creates an instance of the specified component and adds it to the specified location\n   * on the page.\n   */\n  public createComponent<T>(\n    componentType: Type<T>,\n    options?: SkyDynamicComponentOptions\n  ): ComponentRef<T> {\n    options ||= {\n      location: SkyDynamicComponentLocation.BodyBottom,\n    };\n\n    const environmentInjector = createEnvironmentInjector(\n      options.providers ?? [],\n      options.environmentInjector ?? this.#environmentInjector\n    );\n\n    let componentRef: ComponentRef<T>;\n\n    if (options.viewContainerRef) {\n      componentRef = options.viewContainerRef.createComponent(componentType, {\n        environmentInjector,\n      });\n    } else {\n      componentRef = createComponent<T>(componentType, {\n        environmentInjector,\n      });\n\n      this.#applicationRef.attachView(componentRef.hostView);\n\n      const el = this.#getRootNode(componentRef);\n\n      const bodyEl = this.#windowRef.nativeWindow.document.body;\n\n      switch (options.location) {\n        case SkyDynamicComponentLocation.BeforeElement:\n          if (!options.referenceEl) {\n            throw new Error(\n              '[SkyDynamicComponentService] Could not create a component at location `SkyDynamicComponentLocation.BeforeElement` because a reference element was not provided.'\n            );\n          }\n\n          this.#renderer.insertBefore(\n            options.referenceEl.parentElement,\n            el,\n            options.referenceEl\n          );\n          break;\n        case SkyDynamicComponentLocation.ElementTop:\n          if (!options.referenceEl) {\n            throw new Error(\n              '[SkyDynamicComponentService] Could not create a component at location `SkyDynamicComponentLocation.ElementTop` because a reference element was not provided.'\n            );\n          }\n\n          this.#renderer.insertBefore(\n            options.referenceEl,\n            el,\n            options.referenceEl.firstChild\n          );\n          break;\n        case SkyDynamicComponentLocation.ElementBottom:\n          this.#renderer.appendChild(options.referenceEl, el);\n          break;\n        case SkyDynamicComponentLocation.BodyTop:\n          this.#renderer.insertBefore(bodyEl, el, bodyEl.firstChild);\n          break;\n        default:\n          this.#renderer.appendChild(bodyEl, el);\n          break;\n      }\n    }\n\n    return componentRef;\n  }\n\n  /**\n   * Removes a component ref from the page\n   * @param componentRef Component ref for the component being removed\n   */\n  public removeComponent<T>(componentRef: ComponentRef<T> | undefined): void {\n    if (!componentRef) {\n      return;\n    }\n\n    this.#applicationRef.detachView(componentRef.hostView);\n    componentRef.destroy();\n  }\n\n  #getRootNode<T>(componentRef: ComponentRef<T>): any {\n    // Technique for retrieving the component's root node taken from here:\n    // https://malcoded.com/posts/angular-dynamic-components\n    return (componentRef.hostView as EmbeddedViewRef<T>).rootNodes[0];\n  }\n}\n\n/**\n * Angular service for creating and rendering a dynamic component.\n * @internal\n * @deprecated Use `SkyDynamicComponentService` to create a standalone component instead.\n */\n@Injectable({\n  providedIn: 'any',\n})\nexport class SkyDynamicComponentLegacyService extends SkyDynamicComponentService {}\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-component.service.js","sourceRoot":"","sources":["../../../../../../../../libs/components/core/src/lib/modules/dynamic-component/dynamic-component.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ApplicationRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  EnvironmentInjector,\n  Injectable,\n  Renderer2,\n  RendererFactory2,\n  Type,\n  createComponent,\n  createEnvironmentInjector,\n  inject,\n} from '@angular/core';\n\nimport { SkyAppWindowRef } from '../window/window-ref';\n\nimport { SkyDynamicComponentLocation } from './dynamic-component-location';\nimport { SkyDynamicComponentOptions } from './dynamic-component-options';\n\n/**\n * Angular service for creating and rendering a dynamic component.\n * @internal\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class SkyDynamicComponentService {\n  #applicationRef: ApplicationRef;\n\n  #renderer: Renderer2;\n\n  #windowRef: SkyAppWindowRef;\n\n  #environmentInjector = inject(EnvironmentInjector);\n\n  constructor(\n    applicationRef: ApplicationRef,\n    windowRef: SkyAppWindowRef,\n    rendererFactory: RendererFactory2\n  ) {\n    this.#applicationRef = applicationRef;\n    this.#windowRef = windowRef;\n\n    // Based on suggestions from https://github.com/angular/angular/issues/17824\n    // for accessing an instance of Renderer2 in a service since Renderer2 can't\n    // be injected into a service.  Passing undefined for both parameters results\n    // in the default renderer which is what we want here.\n    this.#renderer = rendererFactory.createRenderer(undefined, null);\n  }\n\n  /**\n   * Creates an instance of the specified component and adds it to the specified location\n   * on the page.\n   */\n  public createComponent<T>(\n    componentType: Type<T>,\n    options?: SkyDynamicComponentOptions\n  ): ComponentRef<T> {\n    options ||= {\n      location: SkyDynamicComponentLocation.BodyBottom,\n    };\n\n    const environmentInjector = createEnvironmentInjector(\n      options.providers ?? [],\n      options.environmentInjector ?? this.#environmentInjector\n    );\n\n    let componentRef: ComponentRef<T>;\n\n    if (options.viewContainerRef) {\n      componentRef = options.viewContainerRef.createComponent(componentType, {\n        environmentInjector,\n      });\n    } else {\n      componentRef = createComponent<T>(componentType, {\n        environmentInjector,\n      });\n\n      this.#applicationRef.attachView(componentRef.hostView);\n\n      const el = this.#getRootNode(componentRef);\n\n      const bodyEl = this.#windowRef.nativeWindow.document.body;\n\n      switch (options.location) {\n        case SkyDynamicComponentLocation.BeforeElement:\n          if (!options.referenceEl) {\n            throw new Error(\n              '[SkyDynamicComponentService] Could not create a component at location `SkyDynamicComponentLocation.BeforeElement` because a reference element was not provided.'\n            );\n          }\n\n          this.#renderer.insertBefore(\n            options.referenceEl.parentElement,\n            el,\n            options.referenceEl\n          );\n          break;\n        case SkyDynamicComponentLocation.ElementTop:\n          if (!options.referenceEl) {\n            throw new Error(\n              '[SkyDynamicComponentService] Could not create a component at location `SkyDynamicComponentLocation.ElementTop` because a reference element was not provided.'\n            );\n          }\n\n          this.#renderer.insertBefore(\n            options.referenceEl,\n            el,\n            options.referenceEl.firstChild\n          );\n          break;\n        case SkyDynamicComponentLocation.ElementBottom:\n          this.#renderer.appendChild(options.referenceEl, el);\n          break;\n        case SkyDynamicComponentLocation.BodyTop:\n          this.#renderer.insertBefore(bodyEl, el, bodyEl.firstChild);\n          break;\n        default:\n          this.#renderer.appendChild(bodyEl, el);\n          break;\n      }\n    }\n\n    return componentRef;\n  }\n\n  /**\n   * Removes a component ref from the page\n   * @param componentRef Component ref for the component being removed\n   */\n  public removeComponent<T>(componentRef: ComponentRef<T> | undefined): void {\n    if (!componentRef) {\n      return;\n    }\n\n    if (!this.#applicationRef.destroyed) {\n      this.#applicationRef.detachView(componentRef.hostView);\n    }\n    componentRef.destroy();\n  }\n\n  #getRootNode<T>(componentRef: ComponentRef<T>): any {\n    // Technique for retrieving the component's root node taken from here:\n    // https://malcoded.com/posts/angular-dynamic-components\n    return (componentRef.hostView as EmbeddedViewRef<T>).rootNodes[0];\n  }\n}\n\n/**\n * Angular service for creating and rendering a dynamic component.\n * @internal\n * @deprecated Use `SkyDynamicComponentService` to create a standalone component instead.\n */\n@Injectable({\n  providedIn: 'any',\n})\nexport class SkyDynamicComponentLegacyService extends SkyDynamicComponentService {}\n"]}
@@ -15,5 +15,5 @@ export class Version {
15
15
  /**
16
16
  * Represents the version of @skyux/core.
17
17
  */
18
- export const VERSION = new Version('9.9.0');
18
+ export const VERSION = new Version('9.11.0');
19
19
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvY29tcG9uZW50cy9jb3JlL3NyYy92ZXJzaW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLHdDQUF3QztBQUN4QyxtRkFBbUY7QUFFbkY7OztHQUdHO0FBQ0gsTUFBTSxPQUFPLE9BQU87SUFLbEIsWUFBNEIsSUFBWTtRQUFaLFNBQUksR0FBSixJQUFJLENBQVE7UUFDdEMsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2hDLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNoQyxJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNsRCxDQUFDO0NBQ0Y7QUFFRDs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxJQUFJLE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gVGFrZW4gZnJvbSBBbmd1bGFyJ3MgdmVyc2lvbi50cyBmaWxlLlxuLy8gU2VlOiBodHRwczovL2dpdGh1Yi5jb20vYW5ndWxhci9hbmd1bGFyL2Jsb2IvMTYuMi54L3BhY2thZ2VzL2NvcmUvc3JjL3ZlcnNpb24udHNcblxuLyoqXG4gKiBSZXByZXNlbnRzIHRoZSB2ZXJzaW9uIG9mIGEgcGFja2FnZS5cbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgY2xhc3MgVmVyc2lvbiB7XG4gIHB1YmxpYyByZWFkb25seSBtYWpvcjogc3RyaW5nO1xuICBwdWJsaWMgcmVhZG9ubHkgbWlub3I6IHN0cmluZztcbiAgcHVibGljIHJlYWRvbmx5IHBhdGNoOiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IGZ1bGw6IHN0cmluZykge1xuICAgIHRoaXMubWFqb3IgPSBmdWxsLnNwbGl0KCcuJylbMF07XG4gICAgdGhpcy5taW5vciA9IGZ1bGwuc3BsaXQoJy4nKVsxXTtcbiAgICB0aGlzLnBhdGNoID0gZnVsbC5zcGxpdCgnLicpLnNsaWNlKDIpLmpvaW4oJy4nKTtcbiAgfVxufVxuXG4vKipcbiAqIFJlcHJlc2VudHMgdGhlIHZlcnNpb24gb2YgQHNreXV4L2NvcmUuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gbmV3IFZlcnNpb24oJzAuMC4wLVBMQUNFSE9MREVSJyk7XG4iXX0=
@@ -1,7 +1,7 @@
1
1
  import * as i0 from '@angular/core';
2
2
  import { NgModule, Injectable, inject, RendererFactory2, NgZone, EventEmitter, Directive, Input, Output, EnvironmentInjector, createEnvironmentInjector, createComponent, ChangeDetectorRef, ElementRef, ViewContainerRef, Component, ChangeDetectionStrategy, ViewChild, InjectionToken, Optional, Inject, Pipe, HostBinding, ApplicationRef, Renderer2 } from '@angular/core';
3
3
  import * as i1$1 from '@angular/common';
4
- import { CommonModule, DOCUMENT } from '@angular/common';
4
+ import { DOCUMENT, CommonModule } from '@angular/common';
5
5
  import { Subject, Subscription, ReplaySubject, fromEvent, BehaviorSubject, Observable, of, concat, animationFrameScheduler } from 'rxjs';
6
6
  import { takeUntil, debounceTime, finalize, switchMap, map } from 'rxjs/operators';
7
7
  import { ViewportRuler } from '@angular/cdk/overlay';
@@ -310,19 +310,11 @@ function getElementOffset(element, bufferOffset) {
310
310
  let left;
311
311
  let right;
312
312
  let bottom;
313
- if (element === document.body) {
314
- left = 0;
315
- top = 0;
316
- right = document.documentElement.clientWidth;
317
- bottom = document.documentElement.clientHeight;
318
- }
319
- else {
320
- const clientRect = element.getBoundingClientRect();
321
- left = clientRect.left;
322
- top = clientRect.top;
323
- right = clientRect.right;
324
- bottom = clientRect.bottom;
325
- }
313
+ const clientRect = element.getBoundingClientRect();
314
+ left = clientRect.left;
315
+ top = clientRect.top;
316
+ right = clientRect.right;
317
+ bottom = clientRect.bottom;
326
318
  bottom -= bufferOffsetBottom;
327
319
  left += bufferOffsetLeft;
328
320
  right -= bufferOffsetRight;
@@ -334,16 +326,53 @@ function getElementOffset(element, bufferOffset) {
334
326
  top,
335
327
  };
336
328
  }
329
+ /**
330
+ * Returns an AffixRect that represents the outer dimensions of a given element.
331
+ */
332
+ function getOuterRect(element) {
333
+ const rect = element.getBoundingClientRect();
334
+ const computedStyle = window.getComputedStyle(element, undefined);
335
+ const marginTop = parseFloat(computedStyle.marginTop);
336
+ const marginLeft = parseFloat(computedStyle.marginLeft);
337
+ const marginRight = parseFloat(computedStyle.marginRight);
338
+ const marginBottom = parseFloat(computedStyle.marginBottom);
339
+ return {
340
+ top: rect.top - marginTop,
341
+ left: rect.left - marginLeft,
342
+ bottom: rect.top + rect.height + marginBottom,
343
+ right: rect.left + rect.width + marginLeft + marginRight,
344
+ width: rect.width + marginLeft + marginRight,
345
+ height: rect.height + marginTop + marginBottom,
346
+ };
347
+ }
348
+ /**
349
+ * Returns the visible rect for a given element.
350
+ */
351
+ function getVisibleRectForElement(viewportRuler, element) {
352
+ const elementRect = getOuterRect(element);
353
+ const viewportRect = viewportRuler.getViewportRect();
354
+ const visibleRect = {
355
+ top: Math.max(elementRect.top, 0),
356
+ left: Math.max(elementRect.left, 0),
357
+ bottom: Math.min(elementRect.bottom, viewportRect.bottom),
358
+ right: Math.min(elementRect.right, viewportRect.right),
359
+ };
360
+ return {
361
+ ...visibleRect,
362
+ width: visibleRect.right - visibleRect.left,
363
+ height: visibleRect.bottom - visibleRect.top,
364
+ };
365
+ }
337
366
  function getOverflowParents(child) {
338
367
  const bodyElement = window.document.body;
339
368
  const results = [];
340
369
  let parentElement = child?.parentNode;
341
- while (parentElement !== undefined &&
342
- parentElement !== bodyElement &&
343
- parentElement instanceof HTMLElement) {
344
- const overflowY = window
345
- .getComputedStyle(parentElement, undefined)
346
- .overflowY.toLowerCase();
370
+ while (parentElement !== undefined && parentElement instanceof HTMLElement) {
371
+ const computedStyle = window.getComputedStyle(parentElement, undefined);
372
+ const overflowY = computedStyle.overflowY.toLowerCase();
373
+ if (computedStyle.position === 'fixed' || parentElement.matches('body')) {
374
+ break;
375
+ }
347
376
  if (overflowY === 'auto' ||
348
377
  overflowY === 'hidden' ||
349
378
  overflowY === 'scroll') {
@@ -356,18 +385,33 @@ function getOverflowParents(child) {
356
385
  }
357
386
  /**
358
387
  * Confirms offset is fully visible within a parent element.
359
- * @param parent
360
- * @param offset
361
388
  */
362
- function isOffsetFullyVisibleWithinParent(parent, offset, bufferOffset) {
363
- const parentOffset = getElementOffset(parent, bufferOffset);
364
- return !(parentOffset.top > offset.top ||
365
- parentOffset.right < offset.right ||
366
- parentOffset.bottom < offset.bottom ||
367
- parentOffset.left > offset.left);
368
- }
369
- function isOffsetPartiallyVisibleWithinParent(parent, offset, bufferOffset) {
370
- const parentOffset = getElementOffset(parent, bufferOffset);
389
+ function isOffsetFullyVisibleWithinParent(viewportRuler, parent, offset, bufferOffset) {
390
+ let parentOffset;
391
+ if (parent.matches('body')) {
392
+ const viewportRect = viewportRuler.getViewportRect();
393
+ parentOffset = {
394
+ top: 0,
395
+ left: 0,
396
+ right: viewportRect.right,
397
+ bottom: viewportRect.bottom,
398
+ };
399
+ }
400
+ else if (bufferOffset) {
401
+ parentOffset = getElementOffset(parent, bufferOffset);
402
+ }
403
+ else {
404
+ parentOffset = getVisibleRectForElement(viewportRuler, parent);
405
+ }
406
+ return (parentOffset.top <= offset.top &&
407
+ parentOffset.right >= offset.right &&
408
+ parentOffset.bottom >= offset.bottom &&
409
+ parentOffset.left <= offset.left);
410
+ }
411
+ function isOffsetPartiallyVisibleWithinParent(viewportRuler, parent, offset, bufferOffset) {
412
+ const parentOffset = bufferOffset
413
+ ? getElementOffset(parent, bufferOffset)
414
+ : getVisibleRectForElement(viewportRuler, parent);
371
415
  return !(parentOffset.top >= offset.bottom ||
372
416
  parentOffset.right <= offset.left ||
373
417
  parentOffset.bottom <= offset.top ||
@@ -424,6 +468,7 @@ class SkyAffixer {
424
468
  #baseElement;
425
469
  #currentOffset;
426
470
  #currentPlacement;
471
+ #layoutViewport;
427
472
  #offsetChange;
428
473
  #offsetChangeObs;
429
474
  #overflowParents = [];
@@ -438,9 +483,10 @@ class SkyAffixer {
438
483
  #zone;
439
484
  #_config = DEFAULT_AFFIX_CONFIG;
440
485
  #scrollChangeListener = () => this.#scrollChange.next();
441
- constructor(affixedElement, renderer, viewportRuler, zone) {
486
+ constructor(affixedElement, renderer, viewportRuler, zone, layoutViewport) {
442
487
  this.#affixedElement = affixedElement;
443
488
  this.#renderer = renderer;
489
+ this.#layoutViewport = layoutViewport;
444
490
  this.#viewportRuler = viewportRuler;
445
491
  this.#zone = zone;
446
492
  this.#offsetChange = new Subject();
@@ -488,12 +534,33 @@ class SkyAffixer {
488
534
  }
489
535
  #affix() {
490
536
  const offset = this.#getOffset();
537
+ const offsetParentRect = this.#getOffsetParentRect();
538
+ offset.top = offset.top - offsetParentRect.top;
539
+ offset.left = offset.left - offsetParentRect.left;
540
+ offset.bottom = offset.bottom - offsetParentRect.top;
541
+ offset.right = offset.right - offsetParentRect.left;
491
542
  if (this.#isNewOffset(offset)) {
492
543
  this.#renderer.setStyle(this.#affixedElement, 'top', `${offset.top}px`);
493
544
  this.#renderer.setStyle(this.#affixedElement, 'left', `${offset.left}px`);
494
545
  this.#offsetChange.next({ offset });
495
546
  }
496
547
  }
548
+ #getOffsetParentRect() {
549
+ if (this.#affixedElement.offsetParent) {
550
+ return getOuterRect(this.#affixedElement.offsetParent);
551
+ }
552
+ else {
553
+ const layoutRect = getOuterRect(this.#layoutViewport);
554
+ return {
555
+ top: layoutRect.top,
556
+ left: layoutRect.left,
557
+ height: layoutRect.height,
558
+ width: layoutRect.width,
559
+ bottom: layoutRect.top - layoutRect.height,
560
+ right: layoutRect.left - layoutRect.width,
561
+ };
562
+ }
563
+ }
497
564
  #getOffset() {
498
565
  const parent = this.#getAutoFitContextParent();
499
566
  const maxAttempts = 4;
@@ -501,22 +568,9 @@ class SkyAffixer {
501
568
  let isAffixedElementFullyVisible = false;
502
569
  let offset;
503
570
  let placement = this.#config.placement;
504
- const autoFitOverflowOffset = this.#config.autoFitOverflowOffset || {
505
- bottom: 0,
506
- left: 0,
507
- right: 0,
508
- top: 0,
509
- };
510
- if (this.#config.position === 'absolute') {
511
- const { top, left } = this.#viewportRuler.getViewportScrollPosition();
512
- autoFitOverflowOffset.top = (autoFitOverflowOffset.top || 0) + top;
513
- autoFitOverflowOffset.left = (autoFitOverflowOffset.left || 0) + left;
514
- autoFitOverflowOffset.bottom = (autoFitOverflowOffset.bottom || 0) + top;
515
- autoFitOverflowOffset.right = (autoFitOverflowOffset.right || 0) + left;
516
- }
517
571
  do {
518
572
  offset = this.#getPreferredOffset(placement);
519
- isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(parent, offset, autoFitOverflowOffset);
573
+ isAffixedElementFullyVisible = isOffsetFullyVisibleWithinParent(this.#viewportRuler, parent, offset, this.#config.autoFitOverflowOffset);
520
574
  if (!this.#config.enableAutoFit) {
521
575
  break;
522
576
  }
@@ -543,31 +597,12 @@ class SkyAffixer {
543
597
  // No suitable placement was found, so revert to preferred placement.
544
598
  return this.#getPreferredOffset(this.#config.placement);
545
599
  }
546
- #getRect(baseElement) {
547
- const baseDomRect = baseElement.getBoundingClientRect();
548
- const baseRect = {
549
- top: baseDomRect.top,
550
- bottom: baseDomRect.bottom,
551
- left: baseDomRect.left,
552
- right: baseDomRect.right,
553
- width: baseDomRect.width,
554
- height: baseDomRect.height,
555
- };
556
- if (this.#config.position === 'absolute') {
557
- const { left, top } = this.#viewportRuler.getViewportScrollPosition();
558
- baseRect.top += top;
559
- baseRect.left += left;
560
- baseRect.bottom += top;
561
- baseRect.right += left;
562
- }
563
- return baseRect;
564
- }
565
600
  #getPreferredOffset(placement) {
566
601
  if (!this.#baseElement) {
567
602
  return { top: 0, left: 0, bottom: 0, right: 0 };
568
603
  }
569
- const affixedRect = this.#getRect(this.#affixedElement);
570
- const baseRect = this.#getRect(this.#baseElement);
604
+ const affixedRect = getOuterRect(this.#affixedElement);
605
+ const baseRect = this.#baseElement.getBoundingClientRect();
571
606
  const horizontalAlignment = this.#config.horizontalAlignment;
572
607
  const verticalAlignment = this.#config.verticalAlignment;
573
608
  const enableAutoFit = this.#config.enableAutoFit;
@@ -650,10 +685,33 @@ class SkyAffixer {
650
685
  * the affixed element would otherwise be clipped.
651
686
  */
652
687
  #adjustOffsetToOverflowParent(offset, placement, baseElement) {
688
+ const affixedRect = getOuterRect(this.#affixedElement);
689
+ const baseRect = baseElement.getBoundingClientRect();
653
690
  const parent = this.#getAutoFitContextParent();
654
- const parentOffset = getElementOffset(parent, this.#config.autoFitOverflowOffset);
655
- const affixedRect = this.#getRect(this.#affixedElement);
656
- const baseRect = this.#getRect(baseElement);
691
+ let parentOffset;
692
+ if (this.#config.autoFitContext === SkyAffixAutoFitContext.OverflowParent) {
693
+ if (this.#config.autoFitOverflowOffset) {
694
+ // When the config contains a specific offset.
695
+ parentOffset = getElementOffset(parent, this.#config.autoFitOverflowOffset);
696
+ }
697
+ else if (isOffsetFullyVisibleWithinParent(this.#viewportRuler, parent, baseRect)) {
698
+ // When the base element is fully visible within the parent, aim for the visible portion of the parent element.
699
+ parentOffset = getVisibleRectForElement(this.#viewportRuler, parent);
700
+ }
701
+ else {
702
+ // Anywhere in the parent element.
703
+ parentOffset = getOuterRect(parent);
704
+ }
705
+ }
706
+ else {
707
+ const viewportRect = this.#viewportRuler.getViewportRect();
708
+ parentOffset = {
709
+ top: -viewportRect.top,
710
+ left: -viewportRect.left,
711
+ bottom: -viewportRect.bottom,
712
+ right: -viewportRect.right,
713
+ };
714
+ }
657
715
  // A pixel value representing the leeway between the edge of the overflow parent and the edge
658
716
  // of the base element before it disappears from view.
659
717
  // If the visible portion of the base element is less than this pixel value, the auto-fit
@@ -739,11 +797,13 @@ class SkyAffixer {
739
797
  return true;
740
798
  }
741
799
  #isBaseElementVisible() {
800
+ // Can't get here if the base element is undefined.
801
+ /* istanbul ignore if */
742
802
  if (!this.#baseElement) {
743
803
  return false;
744
804
  }
745
805
  const baseRect = this.#baseElement.getBoundingClientRect();
746
- return isOffsetPartiallyVisibleWithinParent(this.#getImmediateOverflowParent(), {
806
+ return isOffsetPartiallyVisibleWithinParent(this.#viewportRuler, this.#getImmediateOverflowParent(), {
747
807
  top: baseRect.top,
748
808
  left: baseRect.left,
749
809
  right: baseRect.right,
@@ -782,12 +842,34 @@ class SkyAffixService {
782
842
  #renderer = inject(RendererFactory2).createRenderer(undefined, null);
783
843
  #viewportRuler = inject(ViewportRuler);
784
844
  #zone = inject(NgZone);
845
+ #layoutViewport = this.#createLayoutViewportShim(inject(DOCUMENT));
846
+ ngOnDestroy() {
847
+ this.#renderer.removeChild(this.#layoutViewport.parentNode, this.#layoutViewport);
848
+ }
785
849
  /**
786
850
  * Creates an instance of [[SkyAffixer]].
787
851
  * @param affixed The element to be affixed.
788
852
  */
789
853
  createAffixer(affixed) {
790
- return new SkyAffixer(affixed.nativeElement, this.#renderer, this.#viewportRuler, this.#zone);
854
+ return new SkyAffixer(affixed.nativeElement, this.#renderer, this.#viewportRuler, this.#zone, this.#layoutViewport);
855
+ }
856
+ /**
857
+ * Create a layout viewport element that can be used to determine the relative position
858
+ * of the visual viewport. Inspired by
859
+ * https://github.com/WICG/visual-viewport/blob/gh-pages/examples/fixed-to-viewport.html
860
+ */
861
+ #createLayoutViewportShim(doc) {
862
+ const layoutViewportElement = this.#renderer.createElement('div');
863
+ this.#renderer.addClass(layoutViewportElement, 'sky-affix-layout-viewport-shim');
864
+ this.#renderer.setStyle(layoutViewportElement, 'width', '100%');
865
+ this.#renderer.setStyle(layoutViewportElement, 'height', '100%');
866
+ this.#renderer.setStyle(layoutViewportElement, 'position', 'fixed');
867
+ this.#renderer.setStyle(layoutViewportElement, 'top', '0');
868
+ this.#renderer.setStyle(layoutViewportElement, 'left', '0');
869
+ this.#renderer.setStyle(layoutViewportElement, 'visibility', 'hidden');
870
+ this.#renderer.setStyle(layoutViewportElement, 'pointerEvents', 'none');
871
+ this.#renderer.appendChild(doc.body, layoutViewportElement);
872
+ return layoutViewportElement;
791
873
  }
792
874
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.9", ngImport: i0, type: SkyAffixService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
793
875
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.9", ngImport: i0, type: SkyAffixService, providedIn: 'root' }); }
@@ -1163,7 +1245,9 @@ class SkyDynamicComponentService {
1163
1245
  if (!componentRef) {
1164
1246
  return;
1165
1247
  }
1166
- this.#applicationRef.detachView(componentRef.hostView);
1248
+ if (!this.#applicationRef.destroyed) {
1249
+ this.#applicationRef.detachView(componentRef.hostView);
1250
+ }
1167
1251
  componentRef.destroy();
1168
1252
  }
1169
1253
  #getRootNode(componentRef) {
@@ -4001,7 +4085,7 @@ class Version {
4001
4085
  /**
4002
4086
  * Represents the version of @skyux/core.
4003
4087
  */
4004
- const VERSION = new Version('9.9.0');
4088
+ const VERSION = new Version('9.11.0');
4005
4089
 
4006
4090
  /**
4007
4091
  * Generated bundle index. Do not edit.