@danske/sapphire-angular 2.2.0 → 2.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.
Files changed (69) hide show
  1. package/esm2020/lib/checkbox/src/checkbox.component.mjs +20 -4
  2. package/esm2020/lib/checkbox/src/checkbox.module.mjs +4 -3
  3. package/esm2020/lib/field/src/field-control.mjs +1 -1
  4. package/esm2020/lib/field/src/field.component.mjs +24 -9
  5. package/esm2020/lib/listbox/src/listbox.component.mjs +2 -2
  6. package/esm2020/lib/menu/src/menu-trigger.directive.mjs +10 -4
  7. package/esm2020/lib/menu/src/menu.component.mjs +2 -2
  8. package/esm2020/lib/modal/src/dialog/dialog-trigger.directive.mjs +3 -8
  9. package/esm2020/lib/modal/src/modal-trigger.directive.mjs +13 -11
  10. package/esm2020/lib/modal/src/modal.service.mjs +5 -4
  11. package/esm2020/lib/modal/src/panel/panel-trigger.directive.mjs +3 -8
  12. package/esm2020/lib/popover/src/popover-trigger.directive.mjs +11 -11
  13. package/esm2020/lib/popover/src/popover.component.mjs +2 -2
  14. package/esm2020/lib/radio/public_api.mjs +3 -1
  15. package/esm2020/lib/radio/src/radio-group.component.mjs +22 -159
  16. package/esm2020/lib/radio/src/radio.component.mjs +8 -267
  17. package/esm2020/lib/radio/src/radio.module.mjs +40 -6
  18. package/esm2020/lib/radio/src/segmented/segmented-radio-group.component.mjs +130 -0
  19. package/esm2020/lib/radio/src/segmented/segmented-radio.component.mjs +25 -0
  20. package/esm2020/lib/radio/src/shared/radio-base.mjs +276 -0
  21. package/esm2020/lib/radio/src/shared/radio-group-base.mjs +166 -0
  22. package/esm2020/lib/segmented-tabs/src/segmented-tab.component.mjs +2 -2
  23. package/esm2020/lib/segmented-tabs/src/segmented-tabs.component.mjs +19 -12
  24. package/esm2020/lib/select/src/basic-select/basic-select.component.mjs +2 -2
  25. package/esm2020/lib/select/src/select/select.component.mjs +4 -3
  26. package/esm2020/lib/table/public_api.mjs +2 -1
  27. package/esm2020/lib/table/src/cdk-virtual-scroll-viewport-fix.directive.mjs +61 -0
  28. package/esm2020/lib/table/src/table.module.mjs +7 -2
  29. package/esm2020/lib/text-field/src/text-field.component.mjs +4 -3
  30. package/esm2020/lib/theme/public_api.mjs +2 -2
  31. package/esm2020/lib/theme/src/sapphire-overlay.service.mjs +62 -0
  32. package/esm2020/lib/theme/src/theme-base.directive.mjs +6 -46
  33. package/esm2020/lib/theme/src/theme.module.mjs +3 -29
  34. package/esm2020/lib/theme/src/themes.mjs +4 -4
  35. package/esm2020/lib/tooltip/src/tooltip.directive.mjs +4 -4
  36. package/fesm2015/danske-sapphire-angular.mjs +645 -377
  37. package/fesm2015/danske-sapphire-angular.mjs.map +1 -1
  38. package/fesm2020/danske-sapphire-angular.mjs +643 -377
  39. package/fesm2020/danske-sapphire-angular.mjs.map +1 -1
  40. package/lib/checkbox/src/checkbox.component.d.ts +6 -1
  41. package/lib/checkbox/src/checkbox.module.d.ts +4 -3
  42. package/lib/field/src/field-control.d.ts +1 -0
  43. package/lib/field/src/field.component.d.ts +11 -3
  44. package/lib/modal/src/dialog/dialog-trigger.directive.d.ts +0 -3
  45. package/lib/modal/src/modal-trigger.directive.d.ts +3 -2
  46. package/lib/modal/src/modal.service.d.ts +10 -2
  47. package/lib/modal/src/panel/panel-trigger.directive.d.ts +0 -3
  48. package/lib/popover/src/popover-trigger.directive.d.ts +4 -3
  49. package/lib/radio/public_api.d.ts +2 -0
  50. package/lib/radio/src/radio-group.component.d.ts +5 -60
  51. package/lib/radio/src/radio.component.d.ts +4 -96
  52. package/lib/radio/src/radio.module.d.ts +8 -5
  53. package/lib/radio/src/segmented/segmented-radio-group.component.d.ts +40 -0
  54. package/lib/radio/src/segmented/segmented-radio.component.d.ts +8 -0
  55. package/lib/radio/src/shared/radio-base.d.ts +103 -0
  56. package/lib/radio/src/shared/radio-group-base.d.ts +63 -0
  57. package/lib/segmented-tabs/src/segmented-tabs.component.d.ts +3 -2
  58. package/lib/select/src/select/select.component.d.ts +1 -0
  59. package/lib/table/public_api.d.ts +1 -0
  60. package/lib/table/src/cdk-virtual-scroll-viewport-fix.directive.d.ts +25 -0
  61. package/lib/table/src/table-cell.directive.d.ts +1 -1
  62. package/lib/table/src/table.module.d.ts +3 -2
  63. package/lib/text-field/src/text-field.component.d.ts +1 -0
  64. package/lib/theme/public_api.d.ts +1 -1
  65. package/lib/theme/src/sapphire-overlay.service.d.ts +34 -0
  66. package/lib/tooltip/src/tooltip.directive.d.ts +2 -2
  67. package/package.json +3 -3
  68. package/esm2020/lib/theme/src/sapphire-overlay-container.service.mjs +0 -37
  69. package/lib/theme/src/sapphire-overlay-container.service.d.ts +0 -14
@@ -7,16 +7,16 @@ import { takeUntil } from 'rxjs/operators';
7
7
  import { flip, toConnectedPosition } from '../../common/placement';
8
8
  import { CoerceBoolean } from '../../common/coerce-boolean.decorator';
9
9
  import * as i0 from "@angular/core";
10
- import * as i1 from "@angular/cdk/overlay";
10
+ import * as i1 from "../../theme/src/sapphire-overlay.service";
11
11
  export const POPOVER_TRIGGER = new InjectionToken('sp-popover-trigger');
12
12
  // TODO(UC-2728): improve event handling when reusable press utility is added
13
13
  export class PopoverTriggerDirective {
14
14
  get getPopoverId() {
15
15
  return this.overlayRef?.hostElement.querySelector('[role="dialog"]')?.id;
16
16
  }
17
- constructor(changeDetectorRef, cdkOverlay, viewContainerRef, elementRef, injector) {
17
+ constructor(changeDetectorRef, overlay, viewContainerRef, elementRef, injector) {
18
18
  this.changeDetectorRef = changeDetectorRef;
19
- this.cdkOverlay = cdkOverlay;
19
+ this.overlay = overlay;
20
20
  this.viewContainerRef = viewContainerRef;
21
21
  this.elementRef = elementRef;
22
22
  this.injector = injector;
@@ -69,7 +69,7 @@ export class PopoverTriggerDirective {
69
69
  if (!this.isOpen() && this.templateRef != null && !this.disabled) {
70
70
  this.opened.next();
71
71
  if (!this.overlayRef) {
72
- this.overlayRef = this.cdkOverlay.create(this.getOverlayConfig());
72
+ this.overlayRef = this.overlay.create(this.getOverlayConfig());
73
73
  this.overlayRef.updatePositionStrategy(this.getOverlayPositionStrategy());
74
74
  this.overlayRef.detachments().subscribe(() => {
75
75
  this.closed.next();
@@ -180,9 +180,9 @@ export class PopoverTriggerDirective {
180
180
  return new OverlayConfig({
181
181
  scrollStrategy: this.nonModal
182
182
  ? this.closeOnScroll
183
- ? this.cdkOverlay.scrollStrategies.close()
184
- : this.cdkOverlay.scrollStrategies.reposition()
185
- : this.cdkOverlay.scrollStrategies.block(),
183
+ ? this.overlay.scrollStrategies.close()
184
+ : this.overlay.scrollStrategies.reposition()
185
+ : this.overlay.scrollStrategies.block(),
186
186
  // Including a transparent backdrop according to the latest decision of making popovers modal.
187
187
  hasBackdrop: !this.nonModal,
188
188
  backdropClass: 'cdk-overlay-transparent-backdrop',
@@ -199,7 +199,7 @@ export class PopoverTriggerDirective {
199
199
  }
200
200
  getOverlayPositionStrategy() {
201
201
  const placement = this.placement || 'bottom';
202
- return (this.cdkOverlay
202
+ return (this.overlay
203
203
  .position()
204
204
  .flexibleConnectedTo(this.elementRef)
205
205
  .withLockedPosition()
@@ -228,7 +228,7 @@ export class PopoverTriggerDirective {
228
228
  .withViewportMargin(placement.includes('end') ? 0 : 6));
229
229
  }
230
230
  }
231
- PopoverTriggerDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: PopoverTriggerDirective, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.Overlay }, { token: i0.ViewContainerRef }, { token: i0.ElementRef }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Directive });
231
+ PopoverTriggerDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: PopoverTriggerDirective, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.SapphireOverlay }, { token: i0.ViewContainerRef }, { token: i0.ElementRef }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Directive });
232
232
  PopoverTriggerDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: PopoverTriggerDirective, selector: "[spPopoverTriggerFor]", inputs: { placement: ["spPopoverPlacement", "placement"], templateRef: ["spPopoverTriggerFor", "templateRef"], nonModal: ["spPopoverNonModal", "nonModal"], openOnPressStart: ["spPopoverTriggerOpenOnPressStart", "openOnPressStart"], closeOnScroll: ["spPopoverTriggerCloseOnScroll", "closeOnScroll"], disabled: ["spPopoverTriggerDisabled", "disabled"], disablePushInsideViewport: ["spPopoverDisablePushInsideViewport", "disablePushInsideViewport"] }, outputs: { opened: "spPopoverTriggerOpened", closed: "spPopoverTriggerClosed" }, host: { listeners: { "mousedown": "handleMousedown($event)", "click": "handleClick($event)", "keydown.tab": "handleTab()", "focusout": "onFocusout($event)", "keydown": "handleKeyDown($event)" }, properties: { "class.is-active": "isOpen()", "attr.aria-expanded": "isOpen()", "attr.aria-controls": "this.getPopoverId" } }, providers: [
233
233
  { provide: POPOVER_TRIGGER, useExisting: PopoverTriggerDirective },
234
234
  ], exportAs: ["spPopoverTrigger"], ngImport: i0 });
@@ -260,7 +260,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
260
260
  },
261
261
  exportAs: 'spPopoverTrigger',
262
262
  }]
263
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.Overlay }, { type: i0.ViewContainerRef }, { type: i0.ElementRef }, { type: i0.Injector }]; }, propDecorators: { placement: [{
263
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.SapphireOverlay }, { type: i0.ViewContainerRef }, { type: i0.ElementRef }, { type: i0.Injector }]; }, propDecorators: { placement: [{
264
264
  type: Input,
265
265
  args: ['spPopoverPlacement']
266
266
  }], templateRef: [{
@@ -306,4 +306,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
306
306
  type: HostListener,
307
307
  args: ['keydown', ['$event']]
308
308
  }] } });
309
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-trigger.directive.js","sourceRoot":"","sources":["../../../../../src/lib/popover/src/popover-trigger.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  FlexibleConnectedPositionStrategy,\n  Overlay,\n  OverlayConfig,\n  OverlayRef,\n} from '@angular/cdk/overlay';\nimport { TemplatePortal } from '@angular/cdk/portal';\nimport {\n  ChangeDetectorRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  InjectionToken,\n  Injector,\n  Input,\n  Output,\n  TemplateRef,\n  ViewContainerRef,\n} from '@angular/core';\nimport { merge, Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { flip, Placement, toConnectedPosition } from '../../common/placement';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport { CoerceBoolean } from '../../common/coerce-boolean.decorator';\n\nexport interface PopoverTrigger {\n  toggle(): void;\n  open(): void;\n  close(): void;\n  elementRef: ElementRef<HTMLElement>;\n}\nexport const POPOVER_TRIGGER = new InjectionToken<PopoverTrigger>(\n  'sp-popover-trigger'\n);\n\n// TODO(UC-2728): improve event handling when reusable press utility is added\n\n@Directive({\n  selector: '[spPopoverTriggerFor]',\n  providers: [\n    { provide: POPOVER_TRIGGER, useExisting: PopoverTriggerDirective },\n  ],\n  host: {\n    '[class.is-active]': 'isOpen()',\n    '[attr.aria-expanded]': 'isOpen()',\n  },\n  exportAs: 'spPopoverTrigger',\n})\nexport class PopoverTriggerDirective implements PopoverTrigger {\n  @Input('spPopoverPlacement')\n  public placement: Placement = 'bottom';\n\n  @Input('spPopoverTriggerFor')\n  templateRef: TemplateRef<unknown> | null = null;\n\n  /**\n   * Whether the popover should not block interaction with background.\n   */\n  @Input('spPopoverNonModal')\n  @CoerceBoolean\n  nonModal: BooleanInput = false;\n\n  /**\n   * Whether the popover should be opened on press start.\n   */\n  @Input('spPopoverTriggerOpenOnPressStart')\n  @CoerceBoolean\n  openOnPressStart: BooleanInput = false;\n\n  /**\n   * Whether the popover should be closed when the container is scrolled.\n   * @default false\n   */\n  @Input('spPopoverTriggerCloseOnScroll')\n  @CoerceBoolean\n  closeOnScroll: BooleanInput = false;\n\n  /**\n   * Can be used to disable toggling popover on clicks. Useful if `spPopoverTrigger`\n   * is applied on an element which doesn't support `disabled` property.\n   */\n  @Input('spPopoverTriggerDisabled')\n  @CoerceBoolean\n  disabled: BooleanInput = false;\n\n  /**\n   * If the popover should be pushed around whenever it doesn't fit the viewport\n   * with the provided placement.\n   */\n  @Input('spPopoverDisablePushInsideViewport')\n  @CoerceBoolean\n  disablePushInsideViewport: BooleanInput = false;\n\n  /** Emits when the attached popover is requested to open */\n  @Output('spPopoverTriggerOpened')\n  readonly opened: EventEmitter<void> = new EventEmitter();\n\n  /** Emits when the attached popover is requested to close */\n  @Output('spPopoverTriggerClosed')\n  readonly closed: EventEmitter<void> = new EventEmitter();\n\n  /** Emits when this trigger is destroyed. */\n  protected readonly destroyed: Subject<void> = new Subject();\n\n  /** A reference to the overlay which manages the triggered menu */\n  protected overlayRef: OverlayRef | null = null;\n\n  private isBeingPressed = false;\n\n  @HostBinding('attr.aria-controls')\n  public get getPopoverId() {\n    return this.overlayRef?.hostElement.querySelector('[role=\"dialog\"]')?.id;\n  }\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    private cdkOverlay: Overlay,\n    private viewContainerRef: ViewContainerRef,\n    public elementRef: ElementRef<HTMLElement>,\n    private injector: Injector\n  ) {}\n\n  ngOnDestroy() {\n    this.destroyOverlay();\n\n    this.destroyed.next();\n    this.destroyed.complete();\n  }\n\n  /** Toggle the attached popover. */\n  toggle() {\n    this.isOpen() ? this.close() : this.open();\n  }\n\n  /** Open the attached popover. */\n  open() {\n    if (!this.isOpen() && this.templateRef != null && !this.disabled) {\n      this.opened.next();\n      if (!this.overlayRef) {\n        this.overlayRef = this.cdkOverlay.create(this.getOverlayConfig());\n        this.overlayRef.updatePositionStrategy(\n          this.getOverlayPositionStrategy()\n        );\n        this.overlayRef.detachments().subscribe(() => {\n          this.closed.next();\n          this.changeDetectorRef.detectChanges();\n        });\n      }\n      this.overlayRef\n        .outsidePointerEvents()\n        .pipe(takeUntil(merge(this.closed, this.destroyed)))\n        .subscribe((event: MouseEvent) => {\n          if (\n            // If popover is opened on press start, an outside pointer event\n            // is emitted as the mouse is being released, which would immediately\n            // close the popover. Probably a bug in @angular/cdk/overlay.\n            !(this.openOnPressStart && this.isBeingPressed) &&\n            !this.elementRef.nativeElement.contains(event.target as Element)\n          ) {\n            this.close();\n          }\n        });\n      this.overlayRef.attach(this.getContentPortal());\n      this.changeDetectorRef.detectChanges();\n    }\n  }\n\n  /** Close the opened popover. */\n  close() {\n    if (this.isOpen()) {\n      this.overlayRef?.detach();\n      this.changeDetectorRef.markForCheck();\n    }\n  }\n\n  /** Whether the popover is open. */\n  isOpen() {\n    return !!this.overlayRef?.hasAttached();\n  }\n\n  @HostListener('mousedown', ['$event'])\n  private handleMousedown(event: MouseEvent) {\n    this.isBeingPressed = true;\n    if (this.openOnPressStart) {\n      if (event.currentTarget instanceof HTMLElement) {\n        // We need to focus the trigger before opening. Otherwise, the focus restoration logic will\n        // be messed up, since the wrong element would have the focus when the popover is mounted.\n        event.currentTarget.focus();\n        event.preventDefault();\n        document.addEventListener(\n          'mouseup',\n          () => {\n            setTimeout(() => {\n              this.isBeingPressed = false;\n            });\n          },\n          { once: true }\n        );\n      }\n      this.toggle();\n    }\n  }\n\n  @HostListener('click', ['$event'])\n  private handleClick() {\n    if (!this.openOnPressStart) {\n      this.toggle();\n    }\n  }\n\n  /**\n   * While onFocusout takes care of closing the popover when focus goes out, in the rare cases\n   * that there is no more focusable element between the trigger and the popover, the focus can\n   * go to the popover on pressing `Tab`. But the popover should be closed when tabbing on the\n   * trigger.\n   */\n  @HostListener('keydown.tab')\n  private handleTab() {\n    this.close();\n  }\n\n  /**\n   * In some use cases (e.g. in searchable select) popover trigger grabs the focus, right after\n   * the popover is opened. That requires popover trigger to detect when focus goes out of it\n   * and close the popover.\n   * Note: we can't use `blur` event since it won't bubble up, and the focus can be on an element\n   * inside the trigger, but not the trigger itself.\n   *\n   */\n  @HostListener('focusout', ['$event'])\n  private onFocusout(event: FocusEvent) {\n    if (\n      event.relatedTarget === null &&\n      document.activeElement === event.target\n    ) {\n      // This happens on tab switch, without focus being actually moved.\n      return;\n    }\n    const focusReceivingElement =\n      event.relatedTarget instanceof Element ? event.relatedTarget : null;\n    if (\n      // focus is going nowhere\n      !focusReceivingElement ||\n      // focus is going anywhere outside popover and popover trigger\n      !(\n        this.elementRef?.nativeElement?.contains(focusReceivingElement) ||\n        // This is to not have the popover closed as soon as mouse is down on something inside the\n        // popover, if the focus is currently on the trigger\n        this.overlayRef?.overlayElement?.contains(focusReceivingElement)\n      )\n    ) {\n      this.close();\n    }\n  }\n\n  @HostListener('keydown', ['$event'])\n  private handleKeyDown(event: KeyboardEvent) {\n    if (\n      event.target === this.elementRef.nativeElement &&\n      !(this.elementRef.nativeElement instanceof HTMLButtonElement) &&\n      (event.key === 'Enter' || event.key === ' ')\n    ) {\n      event.preventDefault();\n      this.toggle();\n    }\n  }\n\n  private destroyOverlay() {\n    if (this.overlayRef) {\n      this.overlayRef.dispose();\n      this.overlayRef = null;\n    }\n  }\n\n  /** Get the configuration object used to create the overlay. */\n  private getOverlayConfig() {\n    return new OverlayConfig({\n      scrollStrategy: this.nonModal\n        ? this.closeOnScroll\n          ? this.cdkOverlay.scrollStrategies.close()\n          : this.cdkOverlay.scrollStrategies.reposition()\n        : this.cdkOverlay.scrollStrategies.block(), // to match what we have in react\n\n      // Including a transparent backdrop according to the latest decision of making popovers modal.\n      hasBackdrop: !this.nonModal,\n      backdropClass: 'cdk-overlay-transparent-backdrop',\n    });\n  }\n  private getContentPortal(): TemplatePortal | null {\n    if (this.templateRef) {\n      return new TemplatePortal(\n        this.templateRef,\n        this.viewContainerRef,\n        null,\n        Injector.create({\n          providers: [{ provide: POPOVER_TRIGGER, useValue: this }],\n          parent: this.injector,\n        })\n      );\n    }\n    return null;\n  }\n\n  private getOverlayPositionStrategy(): FlexibleConnectedPositionStrategy {\n    const placement = this.placement || 'bottom';\n    return (\n      this.cdkOverlay\n        .position()\n        .flexibleConnectedTo(this.elementRef)\n        .withLockedPosition()\n        .withGrowAfterOpen()\n        /**\n         * Disabling pushing, to fix an issue in FlexibleConnectedPositionStrategy where the\n         * overlay's bottom grows past the top of the trigger element (aka origin).\n         * I.e., when the overlay is supposed to be placed above the trigger element, it can\n         * grow in height and cover the trigger element. This is more of an issue when the trigger\n         * element remains interactive after the overlay is opened, such as in searchable select.\n         * Note that this is not the case when the overlay is placed above the trigger element.\n         * In that case the overlay will not eat up all available viewport height, and is sized in\n         * a way that respects the preferred placement above the trigger element.\n         * Disabling push is not the ideal solution but prevents the issue.\n         * push is disabled only for overlays where the horizontal alignment is anchored to left\n         * or right of the trigger element, based on the fair assumption that the connected\n         * element is itself within the viewport, and we won't need to push the overlay horizontally\n         * to have it within the viewport. It should also be ok not to push vertically, since we\n         * allow for vertical flipping if needed.\n         **/\n        .withPush(!this.disablePushInsideViewport)\n        .withPositions([\n          toConnectedPosition(placement),\n          toConnectedPosition(flip(placement)),\n        ])\n        .withViewportMargin(placement.includes('end') ? 0 : 6)\n    );\n  }\n}\n"]}
309
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-trigger.directive.js","sourceRoot":"","sources":["../../../../../src/lib/popover/src/popover-trigger.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  FlexibleConnectedPositionStrategy,\n  OverlayConfig,\n  OverlayRef,\n} from '@angular/cdk/overlay';\nimport { TemplatePortal } from '@angular/cdk/portal';\nimport {\n  ChangeDetectorRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  InjectionToken,\n  Injector,\n  Input,\n  Output,\n  TemplateRef,\n  ViewContainerRef,\n} from '@angular/core';\nimport { merge, Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { flip, Placement, toConnectedPosition } from '../../common/placement';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport { CoerceBoolean } from '../../common/coerce-boolean.decorator';\nimport { SapphireOverlay } from '../../theme/src/sapphire-overlay.service';\n\nexport interface PopoverTrigger {\n  toggle(): void;\n  open(): void;\n  close(): void;\n  elementRef: ElementRef<HTMLElement>;\n}\nexport const POPOVER_TRIGGER = new InjectionToken<PopoverTrigger>(\n  'sp-popover-trigger'\n);\n\n// TODO(UC-2728): improve event handling when reusable press utility is added\n\n@Directive({\n  selector: '[spPopoverTriggerFor]',\n  providers: [\n    { provide: POPOVER_TRIGGER, useExisting: PopoverTriggerDirective },\n  ],\n  host: {\n    '[class.is-active]': 'isOpen()',\n    '[attr.aria-expanded]': 'isOpen()',\n  },\n  exportAs: 'spPopoverTrigger',\n})\nexport class PopoverTriggerDirective implements PopoverTrigger {\n  @Input('spPopoverPlacement')\n  public placement: Placement = 'bottom';\n\n  @Input('spPopoverTriggerFor')\n  templateRef: TemplateRef<unknown> | null = null;\n\n  /**\n   * Whether the popover should not block interaction with background.\n   */\n  @Input('spPopoverNonModal')\n  @CoerceBoolean\n  nonModal: BooleanInput = false;\n\n  /**\n   * Whether the popover should be opened on press start.\n   */\n  @Input('spPopoverTriggerOpenOnPressStart')\n  @CoerceBoolean\n  openOnPressStart: BooleanInput = false;\n\n  /**\n   * Whether the popover should be closed when the container is scrolled.\n   * @default false\n   */\n  @Input('spPopoverTriggerCloseOnScroll')\n  @CoerceBoolean\n  closeOnScroll: BooleanInput = false;\n\n  /**\n   * Can be used to disable toggling popover on clicks. Useful if `spPopoverTrigger`\n   * is applied on an element which doesn't support `disabled` property.\n   */\n  @Input('spPopoverTriggerDisabled')\n  @CoerceBoolean\n  disabled: BooleanInput = false;\n\n  /**\n   * If the popover should be pushed around whenever it doesn't fit the viewport\n   * with the provided placement.\n   */\n  @Input('spPopoverDisablePushInsideViewport')\n  @CoerceBoolean\n  disablePushInsideViewport: BooleanInput = false;\n\n  /** Emits when the attached popover is requested to open */\n  @Output('spPopoverTriggerOpened')\n  readonly opened: EventEmitter<void> = new EventEmitter();\n\n  /** Emits when the attached popover is requested to close */\n  @Output('spPopoverTriggerClosed')\n  readonly closed: EventEmitter<void> = new EventEmitter();\n\n  /** Emits when this trigger is destroyed. */\n  protected readonly destroyed: Subject<void> = new Subject();\n\n  /** A reference to the overlay which manages the triggered menu */\n  protected overlayRef: OverlayRef | null = null;\n\n  private isBeingPressed = false;\n\n  @HostBinding('attr.aria-controls')\n  public get getPopoverId() {\n    return this.overlayRef?.hostElement.querySelector('[role=\"dialog\"]')?.id;\n  }\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    private overlay: SapphireOverlay,\n    private viewContainerRef: ViewContainerRef,\n    public elementRef: ElementRef<HTMLElement>,\n    private injector: Injector\n  ) {}\n\n  ngOnDestroy() {\n    this.destroyOverlay();\n\n    this.destroyed.next();\n    this.destroyed.complete();\n  }\n\n  /** Toggle the attached popover. */\n  toggle() {\n    this.isOpen() ? this.close() : this.open();\n  }\n\n  /** Open the attached popover. */\n  open() {\n    if (!this.isOpen() && this.templateRef != null && !this.disabled) {\n      this.opened.next();\n      if (!this.overlayRef) {\n        this.overlayRef = this.overlay.create(this.getOverlayConfig());\n        this.overlayRef.updatePositionStrategy(\n          this.getOverlayPositionStrategy()\n        );\n        this.overlayRef.detachments().subscribe(() => {\n          this.closed.next();\n          this.changeDetectorRef.detectChanges();\n        });\n      }\n      this.overlayRef\n        .outsidePointerEvents()\n        .pipe(takeUntil(merge(this.closed, this.destroyed)))\n        .subscribe((event: MouseEvent) => {\n          if (\n            // If popover is opened on press start, an outside pointer event\n            // is emitted as the mouse is being released, which would immediately\n            // close the popover. Probably a bug in @angular/cdk/overlay.\n            !(this.openOnPressStart && this.isBeingPressed) &&\n            !this.elementRef.nativeElement.contains(event.target as Element)\n          ) {\n            this.close();\n          }\n        });\n      this.overlayRef.attach(this.getContentPortal());\n      this.changeDetectorRef.detectChanges();\n    }\n  }\n\n  /** Close the opened popover. */\n  close() {\n    if (this.isOpen()) {\n      this.overlayRef?.detach();\n      this.changeDetectorRef.markForCheck();\n    }\n  }\n\n  /** Whether the popover is open. */\n  isOpen() {\n    return !!this.overlayRef?.hasAttached();\n  }\n\n  @HostListener('mousedown', ['$event'])\n  private handleMousedown(event: MouseEvent) {\n    this.isBeingPressed = true;\n    if (this.openOnPressStart) {\n      if (event.currentTarget instanceof HTMLElement) {\n        // We need to focus the trigger before opening. Otherwise, the focus restoration logic will\n        // be messed up, since the wrong element would have the focus when the popover is mounted.\n        event.currentTarget.focus();\n        event.preventDefault();\n        document.addEventListener(\n          'mouseup',\n          () => {\n            setTimeout(() => {\n              this.isBeingPressed = false;\n            });\n          },\n          { once: true }\n        );\n      }\n      this.toggle();\n    }\n  }\n\n  @HostListener('click', ['$event'])\n  private handleClick() {\n    if (!this.openOnPressStart) {\n      this.toggle();\n    }\n  }\n\n  /**\n   * While onFocusout takes care of closing the popover when focus goes out, in the rare cases\n   * that there is no more focusable element between the trigger and the popover, the focus can\n   * go to the popover on pressing `Tab`. But the popover should be closed when tabbing on the\n   * trigger.\n   */\n  @HostListener('keydown.tab')\n  private handleTab() {\n    this.close();\n  }\n\n  /**\n   * In some use cases (e.g. in searchable select) popover trigger grabs the focus, right after\n   * the popover is opened. That requires popover trigger to detect when focus goes out of it\n   * and close the popover.\n   * Note: we can't use `blur` event since it won't bubble up, and the focus can be on an element\n   * inside the trigger, but not the trigger itself.\n   *\n   */\n  @HostListener('focusout', ['$event'])\n  private onFocusout(event: FocusEvent) {\n    if (\n      event.relatedTarget === null &&\n      document.activeElement === event.target\n    ) {\n      // This happens on tab switch, without focus being actually moved.\n      return;\n    }\n    const focusReceivingElement =\n      event.relatedTarget instanceof Element ? event.relatedTarget : null;\n    if (\n      // focus is going nowhere\n      !focusReceivingElement ||\n      // focus is going anywhere outside popover and popover trigger\n      !(\n        this.elementRef?.nativeElement?.contains(focusReceivingElement) ||\n        // This is to not have the popover closed as soon as mouse is down on something inside the\n        // popover, if the focus is currently on the trigger\n        this.overlayRef?.overlayElement?.contains(focusReceivingElement)\n      )\n    ) {\n      this.close();\n    }\n  }\n\n  @HostListener('keydown', ['$event'])\n  private handleKeyDown(event: KeyboardEvent) {\n    if (\n      event.target === this.elementRef.nativeElement &&\n      !(this.elementRef.nativeElement instanceof HTMLButtonElement) &&\n      (event.key === 'Enter' || event.key === ' ')\n    ) {\n      event.preventDefault();\n      this.toggle();\n    }\n  }\n\n  private destroyOverlay() {\n    if (this.overlayRef) {\n      this.overlayRef.dispose();\n      this.overlayRef = null;\n    }\n  }\n\n  /** Get the configuration object used to create the overlay. */\n  private getOverlayConfig() {\n    return new OverlayConfig({\n      scrollStrategy: this.nonModal\n        ? this.closeOnScroll\n          ? this.overlay.scrollStrategies.close()\n          : this.overlay.scrollStrategies.reposition()\n        : this.overlay.scrollStrategies.block(), // to match what we have in react\n\n      // Including a transparent backdrop according to the latest decision of making popovers modal.\n      hasBackdrop: !this.nonModal,\n      backdropClass: 'cdk-overlay-transparent-backdrop',\n    });\n  }\n  private getContentPortal(): TemplatePortal | null {\n    if (this.templateRef) {\n      return new TemplatePortal(\n        this.templateRef,\n        this.viewContainerRef,\n        null,\n        Injector.create({\n          providers: [{ provide: POPOVER_TRIGGER, useValue: this }],\n          parent: this.injector,\n        })\n      );\n    }\n    return null;\n  }\n\n  private getOverlayPositionStrategy(): FlexibleConnectedPositionStrategy {\n    const placement = this.placement || 'bottom';\n    return (\n      this.overlay\n        .position()\n        .flexibleConnectedTo(this.elementRef)\n        .withLockedPosition()\n        .withGrowAfterOpen()\n        /**\n         * Disabling pushing, to fix an issue in FlexibleConnectedPositionStrategy where the\n         * overlay's bottom grows past the top of the trigger element (aka origin).\n         * I.e., when the overlay is supposed to be placed above the trigger element, it can\n         * grow in height and cover the trigger element. This is more of an issue when the trigger\n         * element remains interactive after the overlay is opened, such as in searchable select.\n         * Note that this is not the case when the overlay is placed above the trigger element.\n         * In that case the overlay will not eat up all available viewport height, and is sized in\n         * a way that respects the preferred placement above the trigger element.\n         * Disabling push is not the ideal solution but prevents the issue.\n         * push is disabled only for overlays where the horizontal alignment is anchored to left\n         * or right of the trigger element, based on the fair assumption that the connected\n         * element is itself within the viewport, and we won't need to push the overlay horizontally\n         * to have it within the viewport. It should also be ok not to push vertically, since we\n         * allow for vertical flipping if needed.\n         **/\n        .withPush(!this.disablePushInsideViewport)\n        .withPositions([\n          toConnectedPosition(placement),\n          toConnectedPosition(flip(placement)),\n        ])\n        .withViewportMargin(placement.includes('end') ? 0 : 6)\n    );\n  }\n}\n"]}
@@ -75,7 +75,7 @@ export class PopoverComponent {
75
75
  }
76
76
  }
77
77
  PopoverComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: PopoverComponent, deps: [{ token: POPOVER_TRIGGER, optional: true }], target: i0.ɵɵFactoryTarget.Component });
78
- PopoverComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: PopoverComponent, selector: "sp-popover", inputs: { noPadding: "noPadding", noMaxWidth: "noMaxWidth", noAutoFocus: "noAutoFocus", trapFocus: "trapFocus" }, host: { attributes: { "role": "presentation" }, listeners: { "keydown.Escape": "handleEscape($event)" }, properties: { "class.sapphire-popover": "true", "class.sapphire-popover--padded": "!noPadding", "class.sapphire-popover--max-width": "!noMaxWidth" } }, queries: [{ propertyName: "title", first: true, predicate: PopoverTitleDirective, descendants: true }], viewQueries: [{ propertyName: "contentWrapper", first: true, predicate: ["contentWrapper"], descendants: true, read: ElementRef }], hostDirectives: [{ directive: i1.ThemeCheckDirective }, { directive: i2.UseComponentStylesOnHost }], ngImport: i0, template: "<!--\n NOTE: cdkTrapFocus is required for cdkTrapFocusAutoCapture to work, which would be similar to\n <FocusScope autoFocus /> in react-aria.\n\n It prevents focus from going out of popover by bringing it back to the first\n focusable element.\n\n We sometimes counter this in (cdkFocusChange) and sometimes don't, based on\n the \"trapFocus\" input.\n -->\n<div\n #contentWrapper\n [id]=\"ID\"\n role=\"dialog\"\n tabindex=\"-1\"\n cdkMonitorSubtreeFocus\n cdkTrapFocus\n [cdkTrapFocusAutoCapture]=\"!noAutoFocus\"\n class=\"dialog\"\n [attr.aria-labelledby]=\"title?.ID\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n>\n <ng-content></ng-content>\n</div>\n", styles: ["@keyframes fade-in{0%{opacity:0}to{opacity:1}}.sapphire-popover{box-sizing:border-box;background-color:var(--sapphire-semantic-color-background-popover-default);box-shadow:var(--sapphire-semantic-shadow-popover);animation:fade-in var(--sapphire-semantic-time-fade-quick) ease-in-out;border-radius:var(--sapphire-semantic-size-radius-popover);overflow:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.sapphire-popover--padded{padding:var(--sapphire-semantic-size-spacing-30)}.sapphire-popover--max-width{max-width:var(--sapphire-global-size-generic-1120)}.dialog{outline:none}\n"], dependencies: [{ kind: "directive", type: i3.CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "directive", type: i3.CdkMonitorFocus, selector: "[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]", outputs: ["cdkFocusChange"], exportAs: ["cdkMonitorFocus"] }] });
78
+ PopoverComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: PopoverComponent, selector: "sp-popover", inputs: { noPadding: "noPadding", noMaxWidth: "noMaxWidth", noAutoFocus: "noAutoFocus", trapFocus: "trapFocus" }, host: { attributes: { "role": "presentation" }, listeners: { "keydown.Escape": "handleEscape($event)" }, properties: { "class.sapphire-popover": "true", "class.sapphire-popover--padded": "!noPadding", "class.sapphire-popover--max-width": "!noMaxWidth" } }, queries: [{ propertyName: "title", first: true, predicate: PopoverTitleDirective, descendants: true }], viewQueries: [{ propertyName: "contentWrapper", first: true, predicate: ["contentWrapper"], descendants: true, read: ElementRef }], hostDirectives: [{ directive: i1.ThemeCheckDirective }, { directive: i2.UseComponentStylesOnHost }], ngImport: i0, template: "<!--\n NOTE: cdkTrapFocus is required for cdkTrapFocusAutoCapture to work, which would be similar to\n <FocusScope autoFocus /> in react-aria.\n\n It prevents focus from going out of popover by bringing it back to the first\n focusable element.\n\n We sometimes counter this in (cdkFocusChange) and sometimes don't, based on\n the \"trapFocus\" input.\n -->\n<div\n #contentWrapper\n [id]=\"ID\"\n role=\"dialog\"\n tabindex=\"-1\"\n cdkMonitorSubtreeFocus\n cdkTrapFocus\n [cdkTrapFocusAutoCapture]=\"!noAutoFocus\"\n class=\"dialog\"\n [attr.aria-labelledby]=\"title?.ID\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n>\n <ng-content></ng-content>\n</div>\n", styles: ["@keyframes fade-in{0%{opacity:0}to{opacity:1}}.sapphire-popover{box-sizing:border-box;background-color:var(--sapphire-semantic-color-background-popover-default);box-shadow:var(--sapphire-semantic-shadow-popover);animation:fade-in var(--sapphire-semantic-time-fade-quick) ease-in-out;border-radius:var(--sapphire-semantic-size-radius-popover);overflow:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.sapphire-popover--padded{padding:var(--sapphire-semantic-size-spacing-container-horizontal-sm)}.sapphire-popover--max-width{max-width:var(--sapphire-global-size-generic-1120)}.dialog{outline:none}\n"], dependencies: [{ kind: "directive", type: i3.CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "directive", type: i3.CdkMonitorFocus, selector: "[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]", outputs: ["cdkFocusChange"], exportAs: ["cdkMonitorFocus"] }] });
79
79
  __decorate([
80
80
  CoerceBoolean
81
81
  ], PopoverComponent.prototype, "noPadding", void 0);
@@ -95,7 +95,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
95
95
  '[class.sapphire-popover--padded]': '!noPadding',
96
96
  '[class.sapphire-popover--max-width]': '!noMaxWidth',
97
97
  role: 'presentation',
98
- }, hostDirectives: [ThemeCheckDirective, UseComponentStylesOnHost], template: "<!--\n NOTE: cdkTrapFocus is required for cdkTrapFocusAutoCapture to work, which would be similar to\n <FocusScope autoFocus /> in react-aria.\n\n It prevents focus from going out of popover by bringing it back to the first\n focusable element.\n\n We sometimes counter this in (cdkFocusChange) and sometimes don't, based on\n the \"trapFocus\" input.\n -->\n<div\n #contentWrapper\n [id]=\"ID\"\n role=\"dialog\"\n tabindex=\"-1\"\n cdkMonitorSubtreeFocus\n cdkTrapFocus\n [cdkTrapFocusAutoCapture]=\"!noAutoFocus\"\n class=\"dialog\"\n [attr.aria-labelledby]=\"title?.ID\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n>\n <ng-content></ng-content>\n</div>\n", styles: ["@keyframes fade-in{0%{opacity:0}to{opacity:1}}.sapphire-popover{box-sizing:border-box;background-color:var(--sapphire-semantic-color-background-popover-default);box-shadow:var(--sapphire-semantic-shadow-popover);animation:fade-in var(--sapphire-semantic-time-fade-quick) ease-in-out;border-radius:var(--sapphire-semantic-size-radius-popover);overflow:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.sapphire-popover--padded{padding:var(--sapphire-semantic-size-spacing-30)}.sapphire-popover--max-width{max-width:var(--sapphire-global-size-generic-1120)}.dialog{outline:none}\n"] }]
98
+ }, hostDirectives: [ThemeCheckDirective, UseComponentStylesOnHost], template: "<!--\n NOTE: cdkTrapFocus is required for cdkTrapFocusAutoCapture to work, which would be similar to\n <FocusScope autoFocus /> in react-aria.\n\n It prevents focus from going out of popover by bringing it back to the first\n focusable element.\n\n We sometimes counter this in (cdkFocusChange) and sometimes don't, based on\n the \"trapFocus\" input.\n -->\n<div\n #contentWrapper\n [id]=\"ID\"\n role=\"dialog\"\n tabindex=\"-1\"\n cdkMonitorSubtreeFocus\n cdkTrapFocus\n [cdkTrapFocusAutoCapture]=\"!noAutoFocus\"\n class=\"dialog\"\n [attr.aria-labelledby]=\"title?.ID\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n>\n <ng-content></ng-content>\n</div>\n", styles: ["@keyframes fade-in{0%{opacity:0}to{opacity:1}}.sapphire-popover{box-sizing:border-box;background-color:var(--sapphire-semantic-color-background-popover-default);box-shadow:var(--sapphire-semantic-shadow-popover);animation:fade-in var(--sapphire-semantic-time-fade-quick) ease-in-out;border-radius:var(--sapphire-semantic-size-radius-popover);overflow:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.sapphire-popover--padded{padding:var(--sapphire-semantic-size-spacing-container-horizontal-sm)}.sapphire-popover--max-width{max-width:var(--sapphire-global-size-generic-1120)}.dialog{outline:none}\n"] }]
99
99
  }], ctorParameters: function () { return [{ type: undefined, decorators: [{
100
100
  type: Optional
101
101
  }, {
@@ -5,4 +5,6 @@
5
5
  export * from './src/radio.component';
6
6
  export * from './src/radio-group.component';
7
7
  export * from './src/radio.module';
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcmFkaW8vcHVibGljX2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw0RUFBNEU7QUFDNUUsNkVBQTZFO0FBQzdFLHNEQUFzRDtBQUN0RCxXQUFXO0FBQ1gsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsb0JBQW9CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBXZSBoYXZlIHNlcGFyYXRlIHB1YmxpY19hcGkudHMgZmlsZSBmb3IgZWFjaCBhbmd1bGFyIG1vZHVsZSwgdG8gaGF2ZSB0aGVtXG4vLyBleHBvc2VkIGFzIHNlcGFyYXRlIGVudHJ5IHBvaW50cy4gVGhlIGJ1aWxkIGlzIHN0aWxsIG5vdCBlbWl0dGluZyBzZXBhcmF0ZVxuLy8gZW50cnkgcG9pbnRzLCB3aGljaCB3aWxsIGJlIGFkZHJlc3NlZCBpbiB0aGlzIHRhc2s6XG4vLyBXQ0NKLTgzOFxuZXhwb3J0ICogZnJvbSAnLi9zcmMvcmFkaW8uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3JhZGlvLWdyb3VwLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NyYy9yYWRpby5tb2R1bGUnO1xuIl19
8
+ export * from './src/segmented/segmented-radio.component';
9
+ export * from './src/segmented/segmented-radio-group.component';
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcmFkaW8vcHVibGljX2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw0RUFBNEU7QUFDNUUsNkVBQTZFO0FBQzdFLHNEQUFzRDtBQUN0RCxXQUFXO0FBQ1gsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLGlEQUFpRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gV2UgaGF2ZSBzZXBhcmF0ZSBwdWJsaWNfYXBpLnRzIGZpbGUgZm9yIGVhY2ggYW5ndWxhciBtb2R1bGUsIHRvIGhhdmUgdGhlbVxuLy8gZXhwb3NlZCBhcyBzZXBhcmF0ZSBlbnRyeSBwb2ludHMuIFRoZSBidWlsZCBpcyBzdGlsbCBub3QgZW1pdHRpbmcgc2VwYXJhdGVcbi8vIGVudHJ5IHBvaW50cywgd2hpY2ggd2lsbCBiZSBhZGRyZXNzZWQgaW4gdGhpcyB0YXNrOlxuLy8gV0NDSi04MzhcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3JhZGlvLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NyYy9yYWRpby1ncm91cC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zcmMvcmFkaW8ubW9kdWxlJztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3NlZ21lbnRlZC9zZWdtZW50ZWQtcmFkaW8uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc3JjL3NlZ21lbnRlZC9zZWdtZW50ZWQtcmFkaW8tZ3JvdXAuY29tcG9uZW50JztcbiJdfQ==
@@ -1,147 +1,24 @@
1
- import { __decorate } from "tslib";
2
- import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, forwardRef, Input, Optional, Output, QueryList, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, forwardRef, Input, } from '@angular/core';
3
2
  import { ThemeCheckDirective } from '../../theme/src/theme-check.directive';
4
- import { RadioComponent, SapphireRadioChange } from './radio.component';
5
3
  import { NG_VALUE_ACCESSOR } from '@angular/forms';
6
- import { coerceBooleanProperty } from '@angular/cdk/coercion';
7
- import { AutoId } from '../../common/auto-id.decorator';
8
4
  import { FieldControl } from '../../field/src/field-control';
5
+ import { RadioGroupBase } from './shared/radio-group-base';
9
6
  import * as i0 from "@angular/core";
10
- import * as i1 from "../../field/src/field.component";
11
- import * as i2 from "../../theme/src/theme-check.directive";
12
- import * as i3 from "@angular/cdk/a11y";
13
- export class RadioGroupComponent {
14
- /** Name of the radio button group. All radio buttons inside this group will use this name. */
15
- get name() {
16
- return this._name;
17
- }
18
- set name(value) {
19
- this._name = value;
20
- this.updateRadioButtonNames();
21
- }
22
- get value() {
23
- return this._value;
24
- }
25
- set value(newValue) {
26
- if (this._value !== newValue) {
27
- this._value = newValue;
28
- this.updateSelectedRadioFromValue();
29
- this._checkSelectedRadioButton();
30
- }
31
- }
32
- /**
33
- * The currently selected radio button. If set to a new radio button, the radio group value
34
- * will be updated to match the new selected button. */
35
- get selected() {
36
- return this._selected;
37
- }
38
- set selected(selected) {
39
- this._selected = selected;
40
- this.value = selected ? selected.value : null;
41
- this._checkSelectedRadioButton();
42
- }
43
- /** Whether the radio group is disabled */
44
- get disabled() {
45
- return this._disabled;
46
- }
47
- set disabled(value) {
48
- this._disabled = coerceBooleanProperty(value);
49
- }
50
- /** Whether the radio group is readonly */
51
- get readonly() {
52
- return this._readonly;
53
- }
54
- set readonly(value) {
55
- this._readonly = coerceBooleanProperty(value);
56
- }
57
- /** Whether the radio group is required */
58
- get required() {
59
- return this._required;
60
- }
61
- set required(value) {
62
- this._required = coerceBooleanProperty(value);
63
- }
64
- constructor(changeDetectorRef, _field) {
65
- this.changeDetectorRef = changeDetectorRef;
66
- this._field = _field;
67
- this.uniqueId = '';
7
+ import * as i1 from "../../theme/src/theme-check.directive";
8
+ import * as i2 from "@angular/cdk/a11y";
9
+ export class RadioGroupComponent extends RadioGroupBase {
10
+ constructor() {
11
+ super(...arguments);
68
12
  this.orientation = 'vertical';
69
- this._name = `sapphire-radio-group-${this.uniqueId}`;
70
- /**
71
- * Event emitted when the group value changes.
72
- * Change events are only emitted when the value changes due to user interaction with
73
- * the radio button (the same behavior as `<input type-"radio">`).
74
- */
75
- this.change = new EventEmitter();
76
- this._value = '';
77
- this._selected = null;
78
- this._disabled = false;
79
- this._readonly = false;
80
- this._required = false;
81
- /** Child radio buttons. */
82
- this.radios = new QueryList();
83
- this.onTouched = () => { };
84
- this._controlValueAccessorChangeFn = () => { };
85
- }
86
- _checkSelectedRadioButton() {
87
- if (this._selected && !this._selected.checked) {
88
- this._selected.checked = true;
89
- }
90
- }
91
- /** Dispatch change event with current selection and group value. */
92
- _emitChangeEvent() {
93
- this.change.emit(new SapphireRadioChange(this._selected, this._value));
94
- }
95
- // Implemented as part of ControlValueAccessor.
96
- writeValue(value) {
97
- this.value = value;
98
- }
99
- // Implemented as part of ControlValueAccessor.
100
- registerOnChange(fn) {
101
- this._controlValueAccessorChangeFn = fn;
102
- }
103
- // Implemented as part of ControlValueAccessor.
104
- registerOnTouched(fn) {
105
- this.onTouched = fn;
106
13
  }
107
- // Implemented as part of ControlValueAccessor.
108
- setDisabledState(disabled) {
109
- this.disabled = disabled;
110
- this.changeDetectorRef.markForCheck();
111
- }
112
- // Implementing FieldControl interface
113
- isDisabled() {
114
- return this.disabled;
115
- }
116
- _contentFocusChanged(focusOrigin) {
14
+ _onFocusChange(focusOrigin) {
117
15
  if (!focusOrigin) {
118
16
  this.onTouched();
119
17
  }
120
18
  }
121
- /** Updates the `selected` radio button from the internal _value state. */
122
- updateSelectedRadioFromValue() {
123
- // If the value already matches the selected radio, do nothing.
124
- const alreadySelected = this._selected !== null && this._selected.value === this._value;
125
- if (this.radios && !alreadySelected) {
126
- this._selected = null;
127
- this.radios.forEach((radio) => {
128
- radio.checked = this.value === radio.value;
129
- if (radio.checked) {
130
- this._selected = radio;
131
- }
132
- });
133
- }
134
- }
135
- updateRadioButtonNames() {
136
- if (this.radios) {
137
- this.radios.forEach((radio) => {
138
- radio.name = this.name;
139
- });
140
- }
141
- }
142
19
  }
143
- RadioGroupComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: RadioGroupComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.FieldComponent, optional: true }], target: i0.ɵɵFactoryTarget.Component });
144
- RadioGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: RadioGroupComponent, selector: "sp-radio-group", inputs: { orientation: "orientation", name: "name", value: "value", selected: "selected", disabled: "disabled", readonly: "readonly", required: "required" }, outputs: { change: "change" }, host: { attributes: { "role": "radiogroup" } }, providers: [
20
+ RadioGroupComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: RadioGroupComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
21
+ RadioGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: RadioGroupComponent, selector: "sp-radio-group", inputs: { orientation: "orientation" }, host: { attributes: { "role": "radiogroup" } }, providers: [
145
22
  {
146
23
  provide: NG_VALUE_ACCESSOR,
147
24
  useExisting: forwardRef(() => RadioGroupComponent),
@@ -151,10 +28,11 @@ RadioGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", ve
151
28
  provide: FieldControl,
152
29
  useExisting: forwardRef(() => RadioGroupComponent),
153
30
  },
154
- ], queries: [{ propertyName: "radios", predicate: i0.forwardRef(function () { return RadioComponent; }), descendants: true }], hostDirectives: [{ directive: i2.ThemeCheckDirective }], ngImport: i0, template: "<div\n class=\"sapphire-field-group\"\n role=\"radiogroup\"\n [attr.aria-orientation]=\"orientation\"\n [attr.aria-labelledby]=\"_field?.label?.id\"\n [class.sapphire-field-group--vertical]=\"orientation === 'vertical'\"\n [class.sapphire-field-group--horizontal]=\"orientation === 'horizontal'\"\n [class.sapphire-field-group--md]=\"_field?.size === 'md'\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n cdkMonitorSubtreeFocus\n>\n <ng-content></ng-content>\n</div>\n", styles: [".sapphire-field-group{display:flex;flex-wrap:wrap;column-gap:var(--sapphire-semantic-size-spacing-50);row-gap:var(--sapphire-semantic-size-spacing-30)}.sapphire-field-group--horizontal{flex-direction:row}.sapphire-field-group--vertical{flex-direction:column}.sapphire-field-group--md{row-gap:var(--sapphire-semantic-size-spacing-20)}\n"], dependencies: [{ kind: "directive", type: i3.CdkMonitorFocus, selector: "[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]", outputs: ["cdkFocusChange"], exportAs: ["cdkMonitorFocus"] }], changeDetection: i0.ChangeDetectionStrategy.Default });
155
- __decorate([
156
- AutoId()
157
- ], RadioGroupComponent.prototype, "uniqueId", void 0);
31
+ {
32
+ provide: RadioGroupBase,
33
+ useExisting: forwardRef(() => RadioGroupComponent),
34
+ },
35
+ ], usesInheritance: true, hostDirectives: [{ directive: i1.ThemeCheckDirective }], ngImport: i0, template: "<div\n class=\"sapphire-field-group\"\n role=\"radiogroup\"\n [attr.aria-orientation]=\"orientation\"\n [attr.aria-labelledby]=\"_field?.label?.id\"\n [class.sapphire-field-group--vertical]=\"orientation === 'vertical'\"\n [class.sapphire-field-group--horizontal]=\"orientation === 'horizontal'\"\n [class.sapphire-field-group--md]=\"_field?.size === 'md'\"\n (cdkFocusChange)=\"_onFocusChange($event)\"\n cdkMonitorSubtreeFocus\n>\n <ng-content></ng-content>\n</div>\n", styles: [".sapphire-field-group{display:flex;flex-wrap:wrap;column-gap:var(--sapphire-semantic-size-spacing-50);row-gap:var(--sapphire-semantic-size-spacing-30)}.sapphire-field-group--horizontal{flex-direction:row}.sapphire-field-group--vertical{flex-direction:column}.sapphire-field-group--md{row-gap:var(--sapphire-semantic-size-spacing-20)}\n"], dependencies: [{ kind: "directive", type: i2.CdkMonitorFocus, selector: "[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]", outputs: ["cdkFocusChange"], exportAs: ["cdkMonitorFocus"] }], changeDetection: i0.ChangeDetectionStrategy.Default });
158
36
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: RadioGroupComponent, decorators: [{
159
37
  type: Component,
160
38
  args: [{ selector: 'sp-radio-group', changeDetection: ChangeDetectionStrategy.Default, providers: [
@@ -167,29 +45,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImpor
167
45
  provide: FieldControl,
168
46
  useExisting: forwardRef(() => RadioGroupComponent),
169
47
  },
48
+ {
49
+ provide: RadioGroupBase,
50
+ useExisting: forwardRef(() => RadioGroupComponent),
51
+ },
170
52
  ], host: {
171
53
  role: 'radiogroup',
172
- }, hostDirectives: [ThemeCheckDirective], template: "<div\n class=\"sapphire-field-group\"\n role=\"radiogroup\"\n [attr.aria-orientation]=\"orientation\"\n [attr.aria-labelledby]=\"_field?.label?.id\"\n [class.sapphire-field-group--vertical]=\"orientation === 'vertical'\"\n [class.sapphire-field-group--horizontal]=\"orientation === 'horizontal'\"\n [class.sapphire-field-group--md]=\"_field?.size === 'md'\"\n (cdkFocusChange)=\"_contentFocusChanged($event)\"\n cdkMonitorSubtreeFocus\n>\n <ng-content></ng-content>\n</div>\n", styles: [".sapphire-field-group{display:flex;flex-wrap:wrap;column-gap:var(--sapphire-semantic-size-spacing-50);row-gap:var(--sapphire-semantic-size-spacing-30)}.sapphire-field-group--horizontal{flex-direction:row}.sapphire-field-group--vertical{flex-direction:column}.sapphire-field-group--md{row-gap:var(--sapphire-semantic-size-spacing-20)}\n"] }]
173
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.FieldComponent, decorators: [{
174
- type: Optional
175
- }] }]; }, propDecorators: { uniqueId: [], orientation: [{
176
- type: Input
177
- }], name: [{
178
- type: Input
179
- }], change: [{
180
- type: Output
181
- }], value: [{
182
- type: Input
183
- }], selected: [{
184
- type: Input
185
- }], disabled: [{
186
- type: Input
187
- }], readonly: [{
188
- type: Input
189
- }], required: [{
54
+ }, hostDirectives: [ThemeCheckDirective], template: "<div\n class=\"sapphire-field-group\"\n role=\"radiogroup\"\n [attr.aria-orientation]=\"orientation\"\n [attr.aria-labelledby]=\"_field?.label?.id\"\n [class.sapphire-field-group--vertical]=\"orientation === 'vertical'\"\n [class.sapphire-field-group--horizontal]=\"orientation === 'horizontal'\"\n [class.sapphire-field-group--md]=\"_field?.size === 'md'\"\n (cdkFocusChange)=\"_onFocusChange($event)\"\n cdkMonitorSubtreeFocus\n>\n <ng-content></ng-content>\n</div>\n", styles: [".sapphire-field-group{display:flex;flex-wrap:wrap;column-gap:var(--sapphire-semantic-size-spacing-50);row-gap:var(--sapphire-semantic-size-spacing-30)}.sapphire-field-group--horizontal{flex-direction:row}.sapphire-field-group--vertical{flex-direction:column}.sapphire-field-group--md{row-gap:var(--sapphire-semantic-size-spacing-20)}\n"] }]
55
+ }], propDecorators: { orientation: [{
190
56
  type: Input
191
- }], radios: [{
192
- type: ContentChildren,
193
- args: [forwardRef(() => RadioComponent), { descendants: true }]
194
57
  }] } });
195
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-group.component.js","sourceRoot":"","sources":["../../../../../src/lib/radio/src/radio-group.component.ts","../../../../../src/lib/radio/src/radio-group.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  EventEmitter,\n  forwardRef,\n  Input,\n  Optional,\n  Output,\n  QueryList,\n} from '@angular/core';\nimport { ThemeCheckDirective } from '../../theme/src/theme-check.directive';\nimport { RadioComponent, SapphireRadioChange } from './radio.component';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { FocusOrigin } from '@angular/cdk/a11y';\nimport { AutoId } from '../../common/auto-id.decorator';\nimport { FieldControl } from '../../field/src/field-control';\nimport { FieldComponent } from '../../field/src/field.component';\n\n@Component({\n  selector: 'sp-radio-group',\n  templateUrl: './radio-group.component.html',\n  styleUrls: ['./radio-group.component.scss'],\n  changeDetection: ChangeDetectionStrategy.Default,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => RadioGroupComponent),\n      multi: true,\n    },\n    {\n      provide: FieldControl,\n      useExisting: forwardRef(() => RadioGroupComponent),\n    },\n  ],\n  host: {\n    role: 'radiogroup',\n  },\n  hostDirectives: [ThemeCheckDirective],\n})\nexport class RadioGroupComponent implements ControlValueAccessor, FieldControl {\n  @AutoId()\n  public uniqueId = '';\n\n  @Input() orientation: 'vertical' | 'horizontal' = 'vertical';\n\n  /** Name of the radio button group. All radio buttons inside this group will use this name. */\n  @Input()\n  get name(): string {\n    return this._name;\n  }\n  set name(value: string) {\n    this._name = value;\n    this.updateRadioButtonNames();\n  }\n  private _name: string = `sapphire-radio-group-${this.uniqueId}`;\n\n  /**\n   * Event emitted when the group value changes.\n   * Change events are only emitted when the value changes due to user interaction with\n   * the radio button (the same behavior as `<input type-\"radio\">`).\n   */\n  @Output() readonly change: EventEmitter<SapphireRadioChange> =\n    new EventEmitter<SapphireRadioChange>();\n\n  @Input()\n  get value(): any {\n    return this._value;\n  }\n  set value(newValue: any) {\n    if (this._value !== newValue) {\n      this._value = newValue;\n\n      this.updateSelectedRadioFromValue();\n      this._checkSelectedRadioButton();\n    }\n  }\n  private _value: any = '';\n\n  /**\n   * The currently selected radio button. If set to a new radio button, the radio group value\n   * will be updated to match the new selected button. */\n  @Input()\n  get selected() {\n    return this._selected;\n  }\n  set selected(selected: RadioComponent | null) {\n    this._selected = selected;\n    this.value = selected ? selected.value : null;\n    this._checkSelectedRadioButton();\n  }\n  private _selected: RadioComponent | null = null;\n\n  /** Whether the radio group is disabled */\n  @Input()\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: BooleanInput) {\n    this._disabled = coerceBooleanProperty(value);\n  }\n\n  private _disabled: boolean = false;\n\n  /** Whether the radio group is readonly */\n  @Input()\n  get readonly(): boolean {\n    return this._readonly;\n  }\n  set readonly(value: BooleanInput) {\n    this._readonly = coerceBooleanProperty(value);\n  }\n\n  private _readonly: boolean = false;\n\n  /** Whether the radio group is required */\n  @Input()\n  get required(): boolean {\n    return this._required;\n  }\n  set required(value: BooleanInput) {\n    this._required = coerceBooleanProperty(value);\n  }\n\n  private _required: boolean = false;\n\n  /** Child radio buttons. */\n  @ContentChildren(forwardRef(() => RadioComponent), { descendants: true })\n  private radios: QueryList<RadioComponent> = new QueryList<RadioComponent>();\n\n  private onTouched: () => any = () => {};\n  _controlValueAccessorChangeFn: (value: any) => void = () => {};\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    @Optional() public _field: FieldComponent\n  ) {}\n\n  _checkSelectedRadioButton() {\n    if (this._selected && !this._selected.checked) {\n      this._selected.checked = true;\n    }\n  }\n\n  /** Dispatch change event with current selection and group value. */\n  _emitChangeEvent(): void {\n    this.change.emit(new SapphireRadioChange(this._selected!, this._value));\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  writeValue(value: any) {\n    this.value = value;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnChange(fn: (value: any) => void) {\n    this._controlValueAccessorChangeFn = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  registerOnTouched(fn: any) {\n    this.onTouched = fn;\n  }\n\n  // Implemented as part of ControlValueAccessor.\n  setDisabledState(disabled: boolean) {\n    this.disabled = disabled;\n    this.changeDetectorRef.markForCheck();\n  }\n\n  // Implementing FieldControl interface\n  isDisabled(): boolean {\n    return this.disabled;\n  }\n\n  _contentFocusChanged(focusOrigin: FocusOrigin) {\n    if (!focusOrigin) {\n      this.onTouched();\n    }\n  }\n\n  /** Updates the `selected` radio button from the internal _value state. */\n  private updateSelectedRadioFromValue(): void {\n    // If the value already matches the selected radio, do nothing.\n    const alreadySelected =\n      this._selected !== null && this._selected.value === this._value;\n\n    if (this.radios && !alreadySelected) {\n      this._selected = null;\n      this.radios.forEach((radio) => {\n        radio.checked = this.value === radio.value;\n        if (radio.checked) {\n          this._selected = radio;\n        }\n      });\n    }\n  }\n\n  private updateRadioButtonNames(): void {\n    if (this.radios) {\n      this.radios.forEach((radio) => {\n        radio.name = this.name;\n      });\n    }\n  }\n}\n","<div\n  class=\"sapphire-field-group\"\n  role=\"radiogroup\"\n  [attr.aria-orientation]=\"orientation\"\n  [attr.aria-labelledby]=\"_field?.label?.id\"\n  [class.sapphire-field-group--vertical]=\"orientation === 'vertical'\"\n  [class.sapphire-field-group--horizontal]=\"orientation === 'horizontal'\"\n  [class.sapphire-field-group--md]=\"_field?.size === 'md'\"\n  (cdkFocusChange)=\"_contentFocusChanged($event)\"\n  cdkMonitorSubtreeFocus\n>\n  <ng-content></ng-content>\n</div>\n"]}
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFkaW8tZ3JvdXAuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9yYWRpby9zcmMvcmFkaW8tZ3JvdXAuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9yYWRpby9zcmMvcmFkaW8tZ3JvdXAuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLHVCQUF1QixFQUN2QixTQUFTLEVBQ1QsVUFBVSxFQUNWLEtBQUssR0FDTixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUM1RSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNuRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDN0QsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDJCQUEyQixDQUFDOzs7O0FBNEIzRCxNQUFNLE9BQU8sbUJBQW9CLFNBQVEsY0FBYztJQXpCdkQ7O1FBMEJXLGdCQUFXLEdBQThCLFVBQVUsQ0FBQztLQU85RDtJQUxDLGNBQWMsQ0FBQyxXQUF3QjtRQUNyQyxJQUFJLENBQUMsV0FBVyxFQUFFO1lBQ2hCLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztTQUNsQjtJQUNILENBQUM7O2dIQVBVLG1CQUFtQjtvR0FBbkIsbUJBQW1CLGlJQXBCbkI7UUFDVDtZQUNFLE9BQU8sRUFBRSxpQkFBaUI7WUFDMUIsV0FBVyxFQUFFLFVBQVUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQztZQUNsRCxLQUFLLEVBQUUsSUFBSTtTQUNaO1FBQ0Q7WUFDRSxPQUFPLEVBQUUsWUFBWTtZQUNyQixXQUFXLEVBQUUsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLG1CQUFtQixDQUFDO1NBQ25EO1FBQ0Q7WUFDRSxPQUFPLEVBQUUsY0FBYztZQUN2QixXQUFXLEVBQUUsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLG1CQUFtQixDQUFDO1NBQ25EO0tBQ0YsMEdDL0JILGllQWFBOzJGRHdCYSxtQkFBbUI7a0JBekIvQixTQUFTOytCQUNFLGdCQUFnQixtQkFHVCx1QkFBdUIsQ0FBQyxPQUFPLGFBQ3JDO3dCQUNUOzRCQUNFLE9BQU8sRUFBRSxpQkFBaUI7NEJBQzFCLFdBQVcsRUFBRSxVQUFVLENBQUMsR0FBRyxFQUFFLG9CQUFvQixDQUFDOzRCQUNsRCxLQUFLLEVBQUUsSUFBSTt5QkFDWjt3QkFDRDs0QkFDRSxPQUFPLEVBQUUsWUFBWTs0QkFDckIsV0FBVyxFQUFFLFVBQVUsQ0FBQyxHQUFHLEVBQUUsb0JBQW9CLENBQUM7eUJBQ25EO3dCQUNEOzRCQUNFLE9BQU8sRUFBRSxjQUFjOzRCQUN2QixXQUFXLEVBQUUsVUFBVSxDQUFDLEdBQUcsRUFBRSxvQkFBb0IsQ0FBQzt5QkFDbkQ7cUJBQ0YsUUFDSzt3QkFDSixJQUFJLEVBQUUsWUFBWTtxQkFDbkIsa0JBQ2UsQ0FBQyxtQkFBbUIsQ0FBQzs4QkFHNUIsV0FBVztzQkFBbkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBDb21wb25lbnQsXG4gIGZvcndhcmRSZWYsXG4gIElucHV0LFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFRoZW1lQ2hlY2tEaXJlY3RpdmUgfSBmcm9tICcuLi8uLi90aGVtZS9zcmMvdGhlbWUtY2hlY2suZGlyZWN0aXZlJztcbmltcG9ydCB7IE5HX1ZBTFVFX0FDQ0VTU09SIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgRmllbGRDb250cm9sIH0gZnJvbSAnLi4vLi4vZmllbGQvc3JjL2ZpZWxkLWNvbnRyb2wnO1xuaW1wb3J0IHsgUmFkaW9Hcm91cEJhc2UgfSBmcm9tICcuL3NoYXJlZC9yYWRpby1ncm91cC1iYXNlJztcbmltcG9ydCB7IEZvY3VzT3JpZ2luIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL2ExMXknO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzcC1yYWRpby1ncm91cCcsXG4gIHRlbXBsYXRlVXJsOiAnLi9yYWRpby1ncm91cC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3JhZGlvLWdyb3VwLmNvbXBvbmVudC5zY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuRGVmYXVsdCxcbiAgcHJvdmlkZXJzOiBbXG4gICAge1xuICAgICAgcHJvdmlkZTogTkdfVkFMVUVfQUNDRVNTT1IsXG4gICAgICB1c2VFeGlzdGluZzogZm9yd2FyZFJlZigoKSA9PiBSYWRpb0dyb3VwQ29tcG9uZW50KSxcbiAgICAgIG11bHRpOiB0cnVlLFxuICAgIH0sXG4gICAge1xuICAgICAgcHJvdmlkZTogRmllbGRDb250cm9sLFxuICAgICAgdXNlRXhpc3Rpbmc6IGZvcndhcmRSZWYoKCkgPT4gUmFkaW9Hcm91cENvbXBvbmVudCksXG4gICAgfSxcbiAgICB7XG4gICAgICBwcm92aWRlOiBSYWRpb0dyb3VwQmFzZSxcbiAgICAgIHVzZUV4aXN0aW5nOiBmb3J3YXJkUmVmKCgpID0+IFJhZGlvR3JvdXBDb21wb25lbnQpLFxuICAgIH0sXG4gIF0sXG4gIGhvc3Q6IHtcbiAgICByb2xlOiAncmFkaW9ncm91cCcsXG4gIH0sXG4gIGhvc3REaXJlY3RpdmVzOiBbVGhlbWVDaGVja0RpcmVjdGl2ZV0sXG59KVxuZXhwb3J0IGNsYXNzIFJhZGlvR3JvdXBDb21wb25lbnQgZXh0ZW5kcyBSYWRpb0dyb3VwQmFzZSB7XG4gIEBJbnB1dCgpIG9yaWVudGF0aW9uOiAndmVydGljYWwnIHwgJ2hvcml6b250YWwnID0gJ3ZlcnRpY2FsJztcblxuICBfb25Gb2N1c0NoYW5nZShmb2N1c09yaWdpbjogRm9jdXNPcmlnaW4pIHtcbiAgICBpZiAoIWZvY3VzT3JpZ2luKSB7XG4gICAgICB0aGlzLm9uVG91Y2hlZCgpO1xuICAgIH1cbiAgfVxufVxuIiwiPGRpdlxuICBjbGFzcz1cInNhcHBoaXJlLWZpZWxkLWdyb3VwXCJcbiAgcm9sZT1cInJhZGlvZ3JvdXBcIlxuICBbYXR0ci5hcmlhLW9yaWVudGF0aW9uXT1cIm9yaWVudGF0aW9uXCJcbiAgW2F0dHIuYXJpYS1sYWJlbGxlZGJ5XT1cIl9maWVsZD8ubGFiZWw/LmlkXCJcbiAgW2NsYXNzLnNhcHBoaXJlLWZpZWxkLWdyb3VwLS12ZXJ0aWNhbF09XCJvcmllbnRhdGlvbiA9PT0gJ3ZlcnRpY2FsJ1wiXG4gIFtjbGFzcy5zYXBwaGlyZS1maWVsZC1ncm91cC0taG9yaXpvbnRhbF09XCJvcmllbnRhdGlvbiA9PT0gJ2hvcml6b250YWwnXCJcbiAgW2NsYXNzLnNhcHBoaXJlLWZpZWxkLWdyb3VwLS1tZF09XCJfZmllbGQ/LnNpemUgPT09ICdtZCdcIlxuICAoY2RrRm9jdXNDaGFuZ2UpPVwiX29uRm9jdXNDaGFuZ2UoJGV2ZW50KVwiXG4gIGNka01vbml0b3JTdWJ0cmVlRm9jdXNcbj5cbiAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuPC9kaXY+XG4iXX0=