@sebgroup/green-angular 1.8.4 → 1.8.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,6 +1,7 @@
1
1
  import { ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Inject, Input, Optional, Output, ViewChild, } from '@angular/core';
2
2
  import { Subject } from 'rxjs';
3
3
  import { ON_SCROLL_TOKEN } from '../shared/on-scroll.directive';
4
+ import { CONTEXT_MENU_LEFT, CONTEXT_MENU_TOP } from './context-menu.constants';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/common";
6
7
  import * as i2 from "rxjs";
@@ -16,15 +17,14 @@ export class NggContextMenuComponent {
16
17
  this.closeOnScroll = false;
17
18
  this.contextMenuItemClicked = new EventEmitter();
18
19
  this.isActive = false;
19
- this.top = '0px';
20
- this.left = '0px';
20
+ this.top = CONTEXT_MENU_TOP;
21
+ this.left = CONTEXT_MENU_LEFT;
21
22
  }
22
23
  onDocumentClick(target) {
23
24
  if (!this.isActive) {
24
25
  return;
25
26
  }
26
- const contextMenuElement = this.elementRef.nativeElement;
27
- if (!contextMenuElement.contains(target)) {
27
+ if (!this.elementRef.nativeElement.contains(target)) {
28
28
  this.close();
29
29
  }
30
30
  }
@@ -48,8 +48,7 @@ export class NggContextMenuComponent {
48
48
  this.close();
49
49
  return;
50
50
  }
51
- const anchor = this.anchor?.nativeElement;
52
- const buttonRect = anchor.getBoundingClientRect();
51
+ const buttonRect = this.anchor?.nativeElement.getBoundingClientRect();
53
52
  const gapBetweenButtonAndPopover = 3;
54
53
  const left = this.calculateLeft(this.direction, buttonRect);
55
54
  const top = buttonRect.bottom + gapBetweenButtonAndPopover;
@@ -59,6 +58,8 @@ export class NggContextMenuComponent {
59
58
  }
60
59
  close() {
61
60
  this.isActive = false;
61
+ this.top = CONTEXT_MENU_TOP;
62
+ this.left = CONTEXT_MENU_LEFT;
62
63
  this.changeDetectorRef.markForCheck();
63
64
  }
64
65
  onItemClick(item) {
@@ -124,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
124
125
  type: HostListener,
125
126
  args: ['document:click', ['$event.target']]
126
127
  }] } });
127
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../../libs/angular/src/lib/context-menu/context-menu.component.ts","../../../../../../libs/angular/src/lib/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core'\nimport { DropdownOption } from '@sebgroup/extract'\nimport { Subject, Subscription } from 'rxjs'\nimport { ON_SCROLL_TOKEN } from '../shared/on-scroll.directive'\n\n@Component({\n  selector: 'ngg-context-menu',\n  templateUrl: './context-menu.component.html',\n})\nexport class NggContextMenuComponent\n  implements AfterViewInit, OnDestroy, OnInit\n{\n  @Input() direction: 'ltr' | 'rtl' = 'ltr'\n  @Input() menuItems: DropdownOption[] = []\n  @Input() menuItemTemplate: TemplateRef<unknown> | null = null\n  @Input() menuAnchorTemplate: TemplateRef<unknown> | null = null\n  @Input() closeOnScroll = false\n\n  @Output() contextMenuItemClicked: EventEmitter<DropdownOption> =\n    new EventEmitter<DropdownOption>()\n\n  @ViewChild('contextMenuPopover') popover: ElementRef | undefined\n\n  @ViewChild('contextMenuAnchor') anchor: ElementRef | undefined\n\n  isActive = false\n  top = '0px'\n  left = '0px'\n\n  resizeObserver?: ResizeObserver\n  menuCloseSubscription?: Subscription\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    private elementRef: ElementRef,\n    @Optional()\n    @Inject(ON_SCROLL_TOKEN)\n    public closeContextMenu: Subject<void>\n  ) {}\n\n  @HostListener('document:click', ['$event.target'])\n  onDocumentClick(target: HTMLElement): void {\n    if (!this.isActive) {\n      return\n    }\n    const contextMenuElement = this.elementRef.nativeElement as HTMLElement\n\n    if (!contextMenuElement.contains(target)) {\n      this.close()\n    }\n  }\n\n  public ngOnInit(): void {\n    this.resizeObserver = new ResizeObserver(() => {\n      this.close()\n    })\n    this.resizeObserver.observe(document.body)\n  }\n\n  public ngAfterViewInit(): void {\n    if (this.closeOnScroll) {\n      this.menuCloseSubscription = this.closeContextMenu?.subscribe(() =>\n        this.close()\n      )\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this.resizeObserver?.unobserve(document.body)\n    this.menuCloseSubscription?.unsubscribe()\n  }\n\n  open(): void {\n    if (this.isActive) {\n      this.close()\n      return\n    }\n\n    const anchor = this.anchor?.nativeElement as HTMLElement\n    const buttonRect = anchor.getBoundingClientRect()\n\n    const gapBetweenButtonAndPopover = 3\n    const left = this.calculateLeft(this.direction, buttonRect)\n    const top = buttonRect.bottom + gapBetweenButtonAndPopover;\n\n    this.left = `${left}px`\n    this.top = `${top}px`\n    this.isActive = true\n  }\n\n  close(): void {\n    this.isActive = false\n    this.changeDetectorRef.markForCheck()\n  }\n\n  onItemClick(item: DropdownOption): void {\n    this.contextMenuItemClicked.emit(item)\n    this.close()\n  }\n\n  onMenuItemKeyDown(event: KeyboardEvent, menuItem: DropdownOption): void {\n    switch (event.key) {\n      case 'Enter':\n      case ' ':\n        event.preventDefault()\n        this.onItemClick(menuItem)\n        break\n      default:\n        break\n    }\n  }\n\n  calculateLeft(direction: string, buttonRect: DOMRect): number {\n    const popover = this.popover?.nativeElement as HTMLElement\n    const popoverWidth = popover?.offsetWidth || 0\n\n    switch (direction) {\n      case 'rtl':\n        return popoverWidth <= buttonRect.left\n          ? buttonRect.right - popoverWidth\n          : buttonRect.left\n      case 'ltr':\n      default:\n        return buttonRect.right + popoverWidth <= window.innerWidth\n          ? buttonRect.left\n          : buttonRect.right - popoverWidth\n    }\n  }\n}\n","<button\n  #contextMenuAnchor\n  class=\"ghost small px-3\"\n  [class.active]=\"isActive\"\n  (click)=\"open()\"\n>\n  <ng-container\n    [ngTemplateOutlet]=\"menuAnchorTemplate ?? defaultButtonTemplate\"\n  >\n  </ng-container>\n</button>\n\n<div class=\"context-menu-overlay\">\n  <div\n    class=\"popover popover-context-menu\"\n    [class.active]=\"isActive\"\n    [style.top]=\"top\"\n    [style.left]=\"left\"\n    #contextMenuPopover\n  >\n    <ul role=\"listbox\">\n      <li\n        *ngFor=\"let menuItem of menuItems\"\n        (click)=\"onItemClick(menuItem)\"\n        tabindex=\"0\"\n        (keydown)=\"onMenuItemKeyDown($event, menuItem)\"\n      >\n        <ng-container\n          [ngTemplateOutlet]=\"menuItemTemplate ?? defaultMenuItemTemplate\"\n          [ngTemplateOutletContext]=\"{ $implicit: menuItem }\"\n        >\n        </ng-container>\n      </li>\n    </ul>\n  </div>\n</div>\n\n<ng-template #defaultMenuItemTemplate let-menuItem>\n  <span>{{ menuItem.label }}</span>\n</ng-template>\n\n<ng-template #defaultButtonTemplate>\n  <i class=\"sg-icon sg-icon-ellipsis\">Open context menu</i>\n</ng-template>\n"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../../libs/angular/src/lib/context-menu/context-menu.component.ts","../../../../../../libs/angular/src/lib/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core'\nimport { DropdownOption } from '@sebgroup/extract'\nimport { Subject, Subscription } from 'rxjs'\nimport { ON_SCROLL_TOKEN } from '../shared/on-scroll.directive'\nimport { CONTEXT_MENU_LEFT, CONTEXT_MENU_TOP } from './context-menu.constants'\n\n@Component({\n  selector: 'ngg-context-menu',\n  templateUrl: './context-menu.component.html',\n})\nexport class NggContextMenuComponent\n  implements AfterViewInit, OnDestroy, OnInit\n{\n  @Input() direction: 'ltr' | 'rtl' = 'ltr'\n  @Input() menuItems: DropdownOption[] = []\n  @Input() menuItemTemplate: TemplateRef<unknown> | null = null\n  @Input() menuAnchorTemplate: TemplateRef<unknown> | null = null\n  @Input() closeOnScroll = false\n\n  @Output() contextMenuItemClicked: EventEmitter<DropdownOption> =\n    new EventEmitter<DropdownOption>()\n\n  @ViewChild('contextMenuPopover') popover!: ElementRef<HTMLElement>\n\n  @ViewChild('contextMenuAnchor') anchor!: ElementRef<HTMLElement>\n\n  isActive = false\n  top = CONTEXT_MENU_TOP\n  left = CONTEXT_MENU_LEFT\n\n  resizeObserver?: ResizeObserver\n  menuCloseSubscription?: Subscription\n\n  constructor(\n    private changeDetectorRef: ChangeDetectorRef,\n    private elementRef: ElementRef<HTMLElement>,\n    @Optional()\n    @Inject(ON_SCROLL_TOKEN)\n    public closeContextMenu: Subject<void>\n  ) {}\n\n  @HostListener('document:click', ['$event.target'])\n  onDocumentClick(target: HTMLElement): void {\n    if (!this.isActive) {\n      return\n    }\n\n    if (!this.elementRef.nativeElement.contains(target)) {\n      this.close()\n    }\n  }\n\n  public ngOnInit(): void {\n    this.resizeObserver = new ResizeObserver(() => {\n      this.close()\n    })\n    this.resizeObserver.observe(document.body)\n  }\n\n  public ngAfterViewInit(): void {\n    if (this.closeOnScroll) {\n      this.menuCloseSubscription = this.closeContextMenu?.subscribe(() =>\n        this.close()\n      )\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this.resizeObserver?.unobserve(document.body)\n    this.menuCloseSubscription?.unsubscribe()\n  }\n\n  open(): void {\n    if (this.isActive) {\n      this.close()\n      return\n    }\n\n    const buttonRect = this.anchor?.nativeElement.getBoundingClientRect()\n\n    const gapBetweenButtonAndPopover = 3\n    const left = this.calculateLeft(this.direction, buttonRect)\n    const top = buttonRect.bottom + gapBetweenButtonAndPopover;\n\n    this.left = `${left}px`\n    this.top = `${top}px`\n    this.isActive = true\n  }\n\n  close(): void {\n    this.isActive = false\n    this.top = CONTEXT_MENU_TOP\n    this.left = CONTEXT_MENU_LEFT\n    this.changeDetectorRef.markForCheck()\n  }\n\n  onItemClick(item: DropdownOption): void {\n    this.contextMenuItemClicked.emit(item)\n    this.close()\n  }\n\n  onMenuItemKeyDown(event: KeyboardEvent, menuItem: DropdownOption): void {\n    switch (event.key) {\n      case 'Enter':\n      case ' ':\n        event.preventDefault()\n        this.onItemClick(menuItem)\n        break\n      default:\n        break\n    }\n  }\n\n  calculateLeft(direction: string, buttonRect: DOMRect): number {\n    const popover = this.popover?.nativeElement as HTMLElement\n    const popoverWidth = popover?.offsetWidth || 0\n\n    switch (direction) {\n      case 'rtl':\n        return popoverWidth <= buttonRect.left\n          ? buttonRect.right - popoverWidth\n          : buttonRect.left\n      case 'ltr':\n      default:\n        return buttonRect.right + popoverWidth <= window.innerWidth\n          ? buttonRect.left\n          : buttonRect.right - popoverWidth\n    }\n  }\n}\n","<button\n  #contextMenuAnchor\n  class=\"ghost small px-3\"\n  [class.active]=\"isActive\"\n  (click)=\"open()\"\n>\n  <ng-container\n    [ngTemplateOutlet]=\"menuAnchorTemplate ?? defaultButtonTemplate\"\n  >\n  </ng-container>\n</button>\n\n<div class=\"context-menu-overlay\">\n  <div\n    class=\"popover popover-context-menu\"\n    [class.active]=\"isActive\"\n    [style.top]=\"top\"\n    [style.left]=\"left\"\n    #contextMenuPopover\n  >\n    <ul role=\"listbox\">\n      <li\n        *ngFor=\"let menuItem of menuItems\"\n        (click)=\"onItemClick(menuItem)\"\n        tabindex=\"0\"\n        (keydown)=\"onMenuItemKeyDown($event, menuItem)\"\n      >\n        <ng-container\n          [ngTemplateOutlet]=\"menuItemTemplate ?? defaultMenuItemTemplate\"\n          [ngTemplateOutletContext]=\"{ $implicit: menuItem }\"\n        >\n        </ng-container>\n      </li>\n    </ul>\n  </div>\n</div>\n\n<ng-template #defaultMenuItemTemplate let-menuItem>\n  <span>{{ menuItem.label }}</span>\n</ng-template>\n\n<ng-template #defaultButtonTemplate>\n  <i class=\"sg-icon sg-icon-ellipsis\">Open context menu</i>\n</ng-template>\n"]}
@@ -0,0 +1,3 @@
1
+ export const CONTEXT_MENU_TOP = '0px';
2
+ export const CONTEXT_MENU_LEFT = '0px';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC1tZW51LmNvbnN0YW50cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYW5ndWxhci9zcmMvbGliL2NvbnRleHQtbWVudS9jb250ZXh0LW1lbnUuY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHLEtBQUssQ0FBQztBQUV0QyxNQUFNLENBQUMsTUFBTSxpQkFBaUIsR0FBRyxLQUFLLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgQ09OVEVYVF9NRU5VX1RPUCA9ICcwcHgnO1xuXG5leHBvcnQgY29uc3QgQ09OVEVYVF9NRU5VX0xFRlQgPSAnMHB4JzsiXX0=
@@ -37,7 +37,12 @@ export class NggModalComponent {
37
37
  if (this.trapFocus) {
38
38
  this.enableFocusTrap();
39
39
  }
40
- disableBodyScroll(this.ref.nativeElement);
40
+ disableBodyScroll(this.ref.nativeElement, {
41
+ allowTouchMove: (el) => {
42
+ // Allow touchmove for elements inside modal, its required for scroll to work on iOS devices
43
+ return this.ref.nativeElement.contains(el);
44
+ }
45
+ });
41
46
  }
42
47
  else {
43
48
  this.disableFocusTrap();
@@ -193,4 +198,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
193
198
  }], confirm: [{
194
199
  type: Output
195
200
  }] } });
196
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../libs/angular/src/lib/modal/modal.component.ts","../../../../../../libs/angular/src/lib/modal/modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, HostBinding, Input, OnDestroy, OnInit, Output, ViewChild } from '@angular/core';\nimport { ModalType, Size } from '@sebgroup/extract';\nimport { ConfigurableFocusTrap, ConfigurableFocusTrapFactory } from '@angular/cdk/a11y';\nimport {\n    disableBodyScroll,\n    enableBodyScroll,\n} from \"body-scroll-lock\";\n\n@Component({\n    selector: 'ngg-modal',\n    styleUrls: ['./modal.component.scss'],\n    templateUrl: './modal.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class NggModalComponent implements OnDestroy, OnInit {\n    @Input() public modalType?: ModalType\n    @Input() public header?: string\n    @Input() public confirmLabel?: string\n    @Input() public dismissLabel?: string\n    @Input() public size?: Size\n    @Input() public get trapFocus(): boolean | undefined {\n        return this._trapFocus;\n    }\n    public set trapFocus(value: boolean | undefined) {\n        this._trapFocus = value;\n        \n        if (value) {\n            if (this._isOpen) {\n                this.enableFocusTrap();\n            }\n        } else {\n            this.disableFocusTrap();\n        }\n    }\n\n    @Input()\n    public get isOpen(): boolean | undefined {\n        return this._isOpen;\n    }\n    public set isOpen(value: boolean | undefined) {\n        this._isOpen = value;\n\n        if (value) {\n            if (this.trapFocus) {\n                this.enableFocusTrap();\n            }\n\n            disableBodyScroll(this.ref.nativeElement);\n        } else {\n            this.disableFocusTrap();\n            enableBodyScroll(this.ref.nativeElement);\n        }\n    }\n\n    @Output() public isOpenChange: EventEmitter<boolean> = new EventEmitter<boolean>();\n    @Output() public closed: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n    @Output() public confirm: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n    @Output() public dismiss: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n\n    @HostBinding('class.open') get open() { return this.isOpen; }\n    @ViewChild('backdrop') private backdropRef?: ElementRef<HTMLInputElement>;\n    private _isOpen?: boolean;\n    private _trapFocus?: boolean;\n    private configurableFocusTrap: ConfigurableFocusTrap;\n\n    constructor(private ref: ElementRef<HTMLElement>, private configurableFocusTrapFactory: ConfigurableFocusTrapFactory) {\n        this.configurableFocusTrap = this.configurableFocusTrapFactory.create(this.ref.nativeElement);\n    }\n\n    ngOnInit(): void {\n        if (this._isOpen && this.trapFocus) \n           this.enableFocusTrap(); \n        else\n            this.disableFocusTrap();\n    }\n\n    public handleCloseClick(event: MouseEvent) {\n        this.closeModal(event);\n    }\n\n    public handleBackdropClick(event: MouseEvent) {\n        if (event.target == this.backdropRef?.nativeElement)\n            this.closeModal(event);\n    }\n\n    public handleDismiss(event: MouseEvent) {\n        this.dismiss.emit(event);\n    }\n\n    public handleConfirm(event: MouseEvent) {\n        this.confirm.emit(event);\n    }\n\n    private closeModal(event: MouseEvent) {\n        if (this.closed.observers.length > 0) {\n            this.closed.emit(event);\n        }\n        else {\n            this.isOpen = false;\n            this.isOpenChange.emit(this.isOpen);\n        }\n    }\n\n    private enableFocusTrap() {\n        if (this.configurableFocusTrap) {\n            this.configurableFocusTrap.enabled = true;\n            this.configurableFocusTrap.focusInitialElementWhenReady();\n        }\n    }\n    \n    private disableFocusTrap() {\n        if (this.configurableFocusTrap) {\n            this.configurableFocusTrap.enabled = false;\n        }\n    }\n\n    ngOnDestroy(): void {\n        this.configurableFocusTrap?.destroy();\n        enableBodyScroll(this.ref.nativeElement);\n    }\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-header]',\n    styleUrls: ['./modal.component.scss'],\n    template: `\n    <h3 data-testid=\"modal-header-text\">{{header}}</h3>\n    <button data-testid=\"modal-close-button\" class=\"close\" (click)=\"this.handleClose($event)\">\n        <span className=\"sr-only\">Close</span>\n        <i></i>\n    </button>\n    `\n})\nexport class NggModalHeaderComponent {\n    @Input() header?: string;\n    @Output() closed: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n\n    handleClose(event: MouseEvent) {\n        this.closed.emit(event);\n    }\n\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-body]',\n    styleUrls: ['./modal.component.scss'],\n    template: `<ng-content></ng-content>`\n})\nexport class NggModalBodyComponent {\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-footer]',\n    styleUrls: ['./modal.component.scss'],\n    template: `\n    <button data-testid=\"modal-dismiss-button\" *ngIf=\"dismissLabel\" class=\"secondary\" (click)=\"this.handleDismiss($event)\">{{dismissLabel}}</button>\n    <button data-testid=\"modal-confirm-button\" *ngIf=\"confirmLabel\" class=\"primary\" (click)=\"this.handleConfirm($event)\">{{confirmLabel}}</button>\n    `\n})\nexport class NggModalFooterComponent {\n    @Input() dismissLabel?: string;\n    @Input() confirmLabel?: string;\n    @Output() dismiss: EventEmitter<MouseEvent> = new EventEmitter();\n    @Output() confirm: EventEmitter<MouseEvent> = new EventEmitter();\n\n    handleDismiss(event: MouseEvent) {\n        this.dismiss.emit(event);\n    }\n\n    handleConfirm(event: MouseEvent) {\n        this.confirm.emit(event);\n    }\n}","<ng-container *ngIf=\"isOpen\" [ngSwitch]=\"modalType\">\n    <aside data-testid=\"modal\" *ngSwitchCase=\"'slideout'\" role=\"dialog\" aria-modal=\"true\" [class.small]=\"size === 'sm'\" [class.medium]=\"size === 'md'\" [class.large]=\"size === 'lg'\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </aside>\n    <main data-testid=\"modal\" *ngSwitchCase=\"'takeover'\" role=\"dialog\" aria-modal=\"true\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </main>\n    <section data-testid=\"modal\" *ngSwitchDefault role=\"dialog\" aria-modal=\"true\" [class.small]=\"size === 'sm'\" [class.medium]=\"size === 'md'\" [class.large]=\"size === 'lg'\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </section>\n    <ng-template #contentTpl>\n            <ng-content></ng-content>\n    </ng-template>\n    <div #backdrop data-testid=\"modal-backdrop\" class=\"backdrop\" (click)=\"this.handleBackdropClick($event)\" [attr.aria-hidden]=\"true\"></div>\n</ng-container>"]}
201
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../libs/angular/src/lib/modal/modal.component.ts","../../../../../../libs/angular/src/lib/modal/modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, HostBinding, Input, OnDestroy, OnInit, Output, ViewChild } from '@angular/core';\nimport { ModalType, Size } from '@sebgroup/extract';\nimport { ConfigurableFocusTrap, ConfigurableFocusTrapFactory } from '@angular/cdk/a11y';\nimport {\n    disableBodyScroll,\n    enableBodyScroll,\n} from \"body-scroll-lock\";\n\n@Component({\n    selector: 'ngg-modal',\n    styleUrls: ['./modal.component.scss'],\n    templateUrl: './modal.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class NggModalComponent implements OnDestroy, OnInit {\n    @Input() public modalType?: ModalType\n    @Input() public header?: string\n    @Input() public confirmLabel?: string\n    @Input() public dismissLabel?: string\n    @Input() public size?: Size\n    @Input() public get trapFocus(): boolean | undefined {\n        return this._trapFocus;\n    }\n    public set trapFocus(value: boolean | undefined) {\n        this._trapFocus = value;\n\n        if (value) {\n            if (this._isOpen) {\n                this.enableFocusTrap();\n            }\n        } else {\n            this.disableFocusTrap();\n        }\n    }\n\n    @Input()\n    public get isOpen(): boolean | undefined {\n        return this._isOpen;\n    }\n    public set isOpen(value: boolean | undefined) {\n        this._isOpen = value;\n\n        if (value) {\n            if (this.trapFocus) {\n                this.enableFocusTrap();\n            }\n\n            disableBodyScroll(this.ref.nativeElement, {\n                allowTouchMove: (el) => {\n                    // Allow touchmove for elements inside modal, its required for scroll to work on iOS devices\n                    return this.ref.nativeElement.contains(el);\n                }\n            });\n        } else {\n            this.disableFocusTrap();\n            enableBodyScroll(this.ref.nativeElement);\n        }\n    }\n\n    @Output() public isOpenChange: EventEmitter<boolean> = new EventEmitter<boolean>();\n    @Output() public closed: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n    @Output() public confirm: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n    @Output() public dismiss: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n\n    @HostBinding('class.open') get open() { return this.isOpen; }\n    @ViewChild('backdrop') private backdropRef?: ElementRef<HTMLInputElement>;\n    private _isOpen?: boolean;\n    private _trapFocus?: boolean;\n    private configurableFocusTrap: ConfigurableFocusTrap;\n\n    constructor(private ref: ElementRef<HTMLElement>, private configurableFocusTrapFactory: ConfigurableFocusTrapFactory) {\n        this.configurableFocusTrap = this.configurableFocusTrapFactory.create(this.ref.nativeElement);\n    }\n\n    ngOnInit(): void {\n        if (this._isOpen && this.trapFocus)\n            this.enableFocusTrap();\n        else\n            this.disableFocusTrap();\n    }\n\n    public handleCloseClick(event: MouseEvent) {\n        this.closeModal(event);\n    }\n\n    public handleBackdropClick(event: MouseEvent) {\n        if (event.target == this.backdropRef?.nativeElement)\n            this.closeModal(event);\n    }\n\n    public handleDismiss(event: MouseEvent) {\n        this.dismiss.emit(event);\n    }\n\n    public handleConfirm(event: MouseEvent) {\n        this.confirm.emit(event);\n    }\n\n    private closeModal(event: MouseEvent) {\n        if (this.closed.observers.length > 0) {\n            this.closed.emit(event);\n        }\n        else {\n            this.isOpen = false;\n            this.isOpenChange.emit(this.isOpen);\n        }\n    }\n\n    private enableFocusTrap() {\n        if (this.configurableFocusTrap) {\n            this.configurableFocusTrap.enabled = true;\n            this.configurableFocusTrap.focusInitialElementWhenReady();\n        }\n    }\n\n    private disableFocusTrap() {\n        if (this.configurableFocusTrap) {\n            this.configurableFocusTrap.enabled = false;\n        }\n    }\n\n    ngOnDestroy(): void {\n        this.configurableFocusTrap?.destroy();\n        enableBodyScroll(this.ref.nativeElement);\n    }\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-header]',\n    styleUrls: ['./modal.component.scss'],\n    template: `\n    <h3 data-testid=\"modal-header-text\">{{header}}</h3>\n    <button data-testid=\"modal-close-button\" class=\"close\" (click)=\"this.handleClose($event)\">\n        <span className=\"sr-only\">Close</span>\n        <i></i>\n    </button>\n    `\n})\nexport class NggModalHeaderComponent {\n    @Input() header?: string;\n    @Output() closed: EventEmitter<MouseEvent> = new EventEmitter<MouseEvent>();\n\n    handleClose(event: MouseEvent) {\n        this.closed.emit(event);\n    }\n\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-body]',\n    styleUrls: ['./modal.component.scss'],\n    template: `<ng-content></ng-content>`\n})\nexport class NggModalBodyComponent {\n}\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: '[ngg-modal-footer]',\n    styleUrls: ['./modal.component.scss'],\n    template: `\n    <button data-testid=\"modal-dismiss-button\" *ngIf=\"dismissLabel\" class=\"secondary\" (click)=\"this.handleDismiss($event)\">{{dismissLabel}}</button>\n    <button data-testid=\"modal-confirm-button\" *ngIf=\"confirmLabel\" class=\"primary\" (click)=\"this.handleConfirm($event)\">{{confirmLabel}}</button>\n    `\n})\nexport class NggModalFooterComponent {\n    @Input() dismissLabel?: string;\n    @Input() confirmLabel?: string;\n    @Output() dismiss: EventEmitter<MouseEvent> = new EventEmitter();\n    @Output() confirm: EventEmitter<MouseEvent> = new EventEmitter();\n\n    handleDismiss(event: MouseEvent) {\n        this.dismiss.emit(event);\n    }\n\n    handleConfirm(event: MouseEvent) {\n        this.confirm.emit(event);\n    }\n}\n","<ng-container *ngIf=\"isOpen\" [ngSwitch]=\"modalType\">\n    <aside data-testid=\"modal\" *ngSwitchCase=\"'slideout'\" role=\"dialog\" aria-modal=\"true\" [class.small]=\"size === 'sm'\" [class.medium]=\"size === 'md'\" [class.large]=\"size === 'lg'\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </aside>\n    <main data-testid=\"modal\" *ngSwitchCase=\"'takeover'\" role=\"dialog\" aria-modal=\"true\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </main>\n    <section data-testid=\"modal\" *ngSwitchDefault role=\"dialog\" aria-modal=\"true\" [class.small]=\"size === 'sm'\" [class.medium]=\"size === 'md'\" [class.large]=\"size === 'lg'\">\n        <header ngg-modal-header data-testid=\"modal-header\" [header]=\"header\" (closed)=\"this.handleCloseClick($event)\"></header>\n        <div ngg-modal-body data-testid=\"modal-body\" class=\"body\">\n            <ng-container *ngTemplateOutlet=\"contentTpl\"></ng-container> \n        </div>\n        <footer ngg-modal-footer data-testid=\"modal-footer\" *ngIf=\"dismissLabel || confirmLabel\" [dismissLabel]=\"dismissLabel\" [confirmLabel]=\"confirmLabel\" (dismiss)=\"this.handleDismiss($event)\" (confirm)=\"this.handleConfirm($event)\"></footer>\n    </section>\n    <ng-template #contentTpl>\n            <ng-content></ng-content>\n    </ng-template>\n    <div #backdrop data-testid=\"modal-backdrop\" class=\"backdrop\" (click)=\"this.handleBackdropClick($event)\" [attr.aria-hidden]=\"true\"></div>\n</ng-container>"]}
@@ -316,6 +316,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
316
316
  }]
317
317
  }], ctorParameters: function () { return [{ type: i0.ElementRef }]; } });
318
318
 
319
+ const CONTEXT_MENU_TOP = '0px';
320
+ const CONTEXT_MENU_LEFT = '0px';
321
+
319
322
  class NggContextMenuComponent {
320
323
  constructor(changeDetectorRef, elementRef, closeContextMenu) {
321
324
  this.changeDetectorRef = changeDetectorRef;
@@ -328,15 +331,14 @@ class NggContextMenuComponent {
328
331
  this.closeOnScroll = false;
329
332
  this.contextMenuItemClicked = new EventEmitter();
330
333
  this.isActive = false;
331
- this.top = '0px';
332
- this.left = '0px';
334
+ this.top = CONTEXT_MENU_TOP;
335
+ this.left = CONTEXT_MENU_LEFT;
333
336
  }
334
337
  onDocumentClick(target) {
335
338
  if (!this.isActive) {
336
339
  return;
337
340
  }
338
- const contextMenuElement = this.elementRef.nativeElement;
339
- if (!contextMenuElement.contains(target)) {
341
+ if (!this.elementRef.nativeElement.contains(target)) {
340
342
  this.close();
341
343
  }
342
344
  }
@@ -363,8 +365,7 @@ class NggContextMenuComponent {
363
365
  this.close();
364
366
  return;
365
367
  }
366
- const anchor = (_a = this.anchor) === null || _a === void 0 ? void 0 : _a.nativeElement;
367
- const buttonRect = anchor.getBoundingClientRect();
368
+ const buttonRect = (_a = this.anchor) === null || _a === void 0 ? void 0 : _a.nativeElement.getBoundingClientRect();
368
369
  const gapBetweenButtonAndPopover = 3;
369
370
  const left = this.calculateLeft(this.direction, buttonRect);
370
371
  const top = buttonRect.bottom + gapBetweenButtonAndPopover;
@@ -374,6 +375,8 @@ class NggContextMenuComponent {
374
375
  }
375
376
  close() {
376
377
  this.isActive = false;
378
+ this.top = CONTEXT_MENU_TOP;
379
+ this.left = CONTEXT_MENU_LEFT;
377
380
  this.changeDetectorRef.markForCheck();
378
381
  }
379
382
  onItemClick(item) {
@@ -999,7 +1002,12 @@ class NggModalComponent {
999
1002
  if (this.trapFocus) {
1000
1003
  this.enableFocusTrap();
1001
1004
  }
1002
- disableBodyScroll(this.ref.nativeElement);
1005
+ disableBodyScroll(this.ref.nativeElement, {
1006
+ allowTouchMove: (el) => {
1007
+ // Allow touchmove for elements inside modal, its required for scroll to work on iOS devices
1008
+ return this.ref.nativeElement.contains(el);
1009
+ }
1010
+ });
1003
1011
  }
1004
1012
  else {
1005
1013
  this.disableFocusTrap();