UNPKG

@hxui/angular

Version:

An Angular library based on the [HXUI design system](https://hxui.io).

281 lines 36.5 kB
import { Overlay } from '@angular/cdk/overlay'; import { TemplatePortal } from '@angular/cdk/portal'; import { ContentChild, Directive, ElementRef, EventEmitter, Input, Output, Renderer2, ViewContainerRef } from '@angular/core'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { DropdownMenuDirective } from './dropdown-menu.directive'; import { DropdownConfig } from './dropdown.config'; import * as i0 from "@angular/core"; import * as i1 from "@angular/cdk/overlay"; import * as i2 from "./dropdown.config"; export class DropdownDirective { constructor(_elementRef, _viewContainerRef, overlay, _config, renderer) { this._elementRef = _elementRef; this._viewContainerRef = _viewContainerRef; this.overlay = overlay; this._config = _config; this.renderer = renderer; this._destroyed = new Subject(); this.isOpen = false; this.placement = 'bottom'; this._autoClose = this._config.autoClose; this.isOpenChange = new EventEmitter(); this.onShown = new EventEmitter(); this.onHidden = new EventEmitter(); this.isDisabled = false; this.showDelay = this._config.showDelay; this.hideDelay = this._config.hideDelay; this.offsetY = 0; this.offsetX = 0; this.createClipPathMask = false; } set autoClose(value) { this._autoClose = value; } get autoClose() { return this._autoClose; } ngOnDestroy() { if (this._overlayRef) { this._overlayRef.dispose(); this._overlayRef = null; } this._destroyed.next(true); this._destroyed.complete(); } /** * Toggles an element’s popover. This is considered a “manual” triggering of * the popover. */ toggle(value) { if (this.isOpen || value === false) { return this.hide(); } return this.show(); } show(delay = this.showDelay) { if (this.isDisabled || this.isOpen) { return; } const overlayRef = this._createOverlay(); this._detach(); overlayRef.attach(this._portal); if (this.createClipPathMask) { this._addClipPathMaskStyles(); } this._setWidthsRelativeTo(overlayRef); this.isOpen = true; this.isOpenChange.emit(this.isOpen); this.onShown.emit(); } hide(delay = this.hideDelay) { this._detach(); this.isOpen = false; this.isOpenChange.emit(this.isOpen); this.onHidden.emit(); } _createOverlay() { if (this._overlayRef) { return this._overlayRef; } this._portal = new TemplatePortal(this.menu.templateRef, this._viewContainerRef); const positionStrategy = this.overlay .position() .flexibleConnectedTo(this._elementRef) .withFlexibleDimensions(false) .withDefaultOffsetX(this.offsetX) .withDefaultOffsetY(this.offsetY) .withPositions([ { originX: 'start', originY: 'top', overlayX: 'start', overlayY: 'top' } ]) .withTransformOriginOn('.hxa-dropdown-control'); this._overlayRef = this.overlay.create({ positionStrategy: positionStrategy, panelClass: [ 'hxui-reset', 'hxa-dropdown-panel', 'is-open', this.minWidthRelativeTo ? 'is-fluid-min-width' : 'not-fuild-min-width' ], hasBackdrop: true, backdropClass: 'cdk-overlay-transparent-backdrop' }); this._updatePosition(); this._overlayRef .detachments() .pipe(takeUntil(this._destroyed)) .subscribe(() => this._detach()); this._overlayRef.backdropClick().subscribe(() => this.hide()); const position = this._overlayRef.getConfig() .positionStrategy; position.positionChanges.pipe(takeUntil(this._destroyed)).subscribe(pos => { if (pos.connectionPair.originX === 'start') { this.placement = 'left'; } else if (pos.connectionPair.originX === 'end') { this.placement = 'right'; } else if (pos.connectionPair.originY === 'top') { this.placement = 'top'; } else if (pos.connectionPair.originY === 'bottom') { this.placement = 'bottom'; } }); return this._overlayRef; } _detach() { if (this._overlayRef && this._overlayRef.hasAttached()) { this._overlayRef.detach(); } } _setWidthsRelativeTo(overlayRef) { if (this.maxWidthRelativeTo && this.minWidthRelativeTo) { const elem = document.getElementById(this.maxWidthRelativeTo); overlayRef.updateSize({ minWidth: elem.clientWidth, maxWidth: elem.clientWidth }); } else if (this.maxWidthRelativeTo) { const elem = document.getElementById(this.maxWidthRelativeTo); overlayRef.updateSize({ maxWidth: elem.clientWidth }); } else if (this.minWidthRelativeTo) { const elem = document.getElementById(this.minWidthRelativeTo); overlayRef.updateSize({ minWidth: elem.clientWidth }); } } _updatePosition() { const position = this._overlayRef.getConfig() .positionStrategy; const origin = this._getOrigin(); const overlay = this._getOverlayPosition(); position.withPositions([ { ...origin.main, ...overlay.main }, { ...origin.fallback, ...overlay.fallback } ]); } /** * Returns the origin position and a fallback position based on the user's position preference. * The fallback position is the inverse of the origin (e.g. `'bottom' -> 'top'`). */ _getOrigin() { const placement = this.placement; let originPlacement; if (placement === 'top' || placement === 'bottom') { originPlacement = { originX: 'start', originY: placement === 'top' ? 'top' : 'bottom' }; } else if (placement === 'left') { originPlacement = { originX: 'start', originY: 'center' }; } else if (placement === 'right') { originPlacement = { originX: 'end', originY: 'center' }; } else { console.error('Position error', placement); } const { x, y } = this._invertPosition(originPlacement.originX, originPlacement.originY); return { main: originPlacement, fallback: { originX: x, originY: y } }; } /** Returns the overlay position and a fallback position based on the user's preference */ _getOverlayPosition() { const placement = this.placement; let overlayPlacement; if (placement === 'top') { overlayPlacement = { overlayX: 'start', overlayY: 'bottom' }; } else if (placement === 'bottom') { overlayPlacement = { overlayX: 'start', overlayY: 'top' }; } else if (placement === 'left') { overlayPlacement = { overlayX: 'end', overlayY: 'center' }; } else if (placement === 'right') { overlayPlacement = { overlayX: 'start', overlayY: 'center' }; } else { console.error('Could not find a position', placement); } const { x, y } = this._invertPosition(overlayPlacement.overlayX, overlayPlacement.overlayY); return { main: overlayPlacement, fallback: { overlayX: x, overlayY: y } }; } _invertPosition(x, y) { if (this.placement === 'top' || this.placement === 'bottom') { if (y === 'top') { y = 'bottom'; } else if (y === 'bottom') { y = 'top'; } } else { if (x === 'end') { x = 'start'; } else if (x === 'start') { x = 'end'; } } return { x, y }; } // Create a clip path mask in the backdrop. The mask is a rectangle, the size of the viewcontainer // This enables the user to interact with the contents of the viewcontainer without closing the dropdown _addClipPathMaskStyles() { const HTMLEl = this._overlayRef.backdropElement; const viewRefNativeEl = this._viewContainerRef.element.nativeElement; const elRect = viewRefNativeEl.getBoundingClientRect(); if (elRect) { const clipPathStyle = `polygon(0% 0%, 0% 100%, ${elRect.left}px 100%, ${elRect.left}px ${elRect.top}px, ${elRect.right}px ${elRect.top}px, ${elRect.right}px ${elRect.bottom}px, ${elRect.left}px ${elRect.bottom}px, ${elRect.left}px 100%, 100% 100%, 100% 0%)`; this.renderer.setStyle(HTMLEl, 'clip-path', clipPathStyle); } } } DropdownDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: DropdownDirective, deps: [{ token: i0.ElementRef }, { token: i0.ViewContainerRef }, { token: i1.Overlay }, { token: i2.DropdownConfig }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); DropdownDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.11", type: DropdownDirective, selector: "[hxaDropdown],[hxDropdown]", inputs: { placement: "placement", autoClose: "autoClose", isDisabled: "isDisabled", showDelay: "showDelay", hideDelay: "hideDelay", maxWidthRelativeTo: "maxWidthRelativeTo", minWidthRelativeTo: "minWidthRelativeTo", offsetY: "offsetY", offsetX: "offsetX", createClipPathMask: "createClipPathMask" }, outputs: { isOpenChange: "isOpenChange", onShown: "onShown", onHidden: "onHidden" }, queries: [{ propertyName: "menu", first: true, predicate: DropdownMenuDirective, descendants: true }], exportAs: ["hx-dropdown", "hxa-dropdown"], ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: DropdownDirective, decorators: [{ type: Directive, args: [{ selector: '[hxaDropdown],[hxDropdown]', exportAs: 'hx-dropdown, hxa-dropdown' }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ViewContainerRef }, { type: i1.Overlay }, { type: i2.DropdownConfig }, { type: i0.Renderer2 }]; }, propDecorators: { menu: [{ type: ContentChild, args: [DropdownMenuDirective] }], placement: [{ type: Input }], autoClose: [{ type: Input }], isOpenChange: [{ type: Output }], onShown: [{ type: Output }], onHidden: [{ type: Output }], isDisabled: [{ type: Input }], showDelay: [{ type: Input }], hideDelay: [{ type: Input }], maxWidthRelativeTo: [{ type: Input }], minWidthRelativeTo: [{ type: Input }], offsetY: [{ type: Input }], offsetX: [{ type: Input }], createClipPathMask: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.directive.js","sourceRoot":"","sources":["../../../../../projects/hx-ui/src/lib/dropdown/dropdown.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  FlexibleConnectedPositionStrategy,\r\n  HorizontalConnectionPos,\r\n  OriginConnectionPosition,\r\n  Overlay,\r\n  OverlayConnectionPosition,\r\n  OverlayRef,\r\n  VerticalConnectionPos\r\n} from '@angular/cdk/overlay';\r\nimport { TemplatePortal } from '@angular/cdk/portal';\r\nimport {\r\n  ContentChild,\r\n  Directive,\r\n  ElementRef,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  Output,\r\n  Renderer2,\r\n  ViewContainerRef\r\n} from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { DropdownMenuDirective } from './dropdown-menu.directive';\r\nimport { DropdownConfig } from './dropdown.config';\r\n\r\n@Directive({\r\n  selector: '[hxaDropdown],[hxDropdown]',\r\n  exportAs: 'hx-dropdown, hxa-dropdown'\r\n})\r\nexport class DropdownDirective implements OnDestroy {\r\n  @ContentChild(DropdownMenuDirective) menu: DropdownMenuDirective;\r\n\r\n  _overlayRef: OverlayRef | null;\r\n  private _portal: TemplatePortal;\r\n  private readonly _destroyed = new Subject();\r\n  public isOpen = false;\r\n\r\n  @Input()\r\n  placement: 'top' | 'bottom' | 'left' | 'right' = 'bottom';\r\n\r\n  private _autoClose = this._config.autoClose;\r\n  @Input()\r\n  set autoClose(value: boolean) {\r\n    this._autoClose = value;\r\n  }\r\n\r\n  get autoClose(): boolean {\r\n    return this._autoClose;\r\n  }\r\n\r\n  @Output() isOpenChange: EventEmitter<boolean> = new EventEmitter<boolean>();\r\n\r\n  @Output() onShown: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Output() onHidden: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  @Input()\r\n  isDisabled = false;\r\n\r\n  @Input()\r\n  showDelay = this._config.showDelay;\r\n\r\n  @Input()\r\n  hideDelay = this._config.hideDelay;\r\n\r\n  @Input()\r\n  maxWidthRelativeTo: string;\r\n\r\n  @Input()\r\n  minWidthRelativeTo: string;\r\n\r\n  @Input()\r\n  offsetY = 0;\r\n\r\n  @Input()\r\n  offsetX = 0;\r\n\r\n  @Input()\r\n  createClipPathMask = false;\r\n\r\n  constructor(\r\n    private _elementRef: ElementRef,\r\n    private _viewContainerRef: ViewContainerRef,\r\n    public overlay: Overlay,\r\n    public _config: DropdownConfig,\r\n    private renderer: Renderer2\r\n  ) {}\r\n\r\n  ngOnDestroy(): void {\r\n    if (this._overlayRef) {\r\n      this._overlayRef.dispose();\r\n      this._overlayRef = null;\r\n    }\r\n    this._destroyed.next(true);\r\n    this._destroyed.complete();\r\n  }\r\n\r\n  /**\r\n   * Toggles an element’s popover. This is considered a “manual” triggering of\r\n   * the popover.\r\n   */\r\n  toggle(value?: boolean): void {\r\n    if (this.isOpen || value === false) {\r\n      return this.hide();\r\n    }\r\n\r\n    return this.show();\r\n  }\r\n\r\n  show(delay: number = this.showDelay) {\r\n    if (this.isDisabled || this.isOpen) {\r\n      return;\r\n    }\r\n\r\n    const overlayRef = this._createOverlay();\r\n    this._detach();\r\n    overlayRef.attach(this._portal);\r\n\r\n    if (this.createClipPathMask) {\r\n      this._addClipPathMaskStyles();\r\n    }\r\n\r\n    this._setWidthsRelativeTo(overlayRef);\r\n    this.isOpen = true;\r\n    this.isOpenChange.emit(this.isOpen);\r\n    this.onShown.emit();\r\n  }\r\n\r\n  hide(delay: number = this.hideDelay) {\r\n    this._detach();\r\n    this.isOpen = false;\r\n    this.isOpenChange.emit(this.isOpen);\r\n    this.onHidden.emit();\r\n  }\r\n\r\n  private _createOverlay(): OverlayRef {\r\n    if (this._overlayRef) {\r\n      return this._overlayRef;\r\n    }\r\n\r\n    this._portal = new TemplatePortal(\r\n      this.menu.templateRef,\r\n      this._viewContainerRef\r\n    );\r\n\r\n    const positionStrategy = this.overlay\r\n      .position()\r\n      .flexibleConnectedTo(this._elementRef)\r\n      .withFlexibleDimensions(false)\r\n      .withDefaultOffsetX(this.offsetX)\r\n      .withDefaultOffsetY(this.offsetY)\r\n      .withPositions([\r\n        { originX: 'start', originY: 'top', overlayX: 'start', overlayY: 'top' }\r\n      ])\r\n      .withTransformOriginOn('.hxa-dropdown-control');\r\n\r\n    this._overlayRef = this.overlay.create({\r\n      positionStrategy: positionStrategy,\r\n      panelClass: [\r\n        'hxui-reset',\r\n        'hxa-dropdown-panel',\r\n        'is-open',\r\n        this.minWidthRelativeTo ? 'is-fluid-min-width' : 'not-fuild-min-width'\r\n      ],\r\n      hasBackdrop: true,\r\n      backdropClass: 'cdk-overlay-transparent-backdrop'\r\n    });\r\n\r\n    this._updatePosition();\r\n\r\n    this._overlayRef\r\n      .detachments()\r\n      .pipe(takeUntil(this._destroyed))\r\n      .subscribe(() => this._detach());\r\n\r\n    this._overlayRef.backdropClick().subscribe(() => this.hide());\r\n\r\n    const position = this._overlayRef.getConfig()\r\n      .positionStrategy as FlexibleConnectedPositionStrategy;\r\n    position.positionChanges.pipe(takeUntil(this._destroyed)).subscribe(pos => {\r\n      if (pos.connectionPair.originX === 'start') {\r\n        this.placement = 'left';\r\n      } else if (pos.connectionPair.originX === 'end') {\r\n        this.placement = 'right';\r\n      } else if (pos.connectionPair.originY === 'top') {\r\n        this.placement = 'top';\r\n      } else if (pos.connectionPair.originY === 'bottom') {\r\n        this.placement = 'bottom';\r\n      }\r\n    });\r\n\r\n    return this._overlayRef;\r\n  }\r\n\r\n  private _detach() {\r\n    if (this._overlayRef && this._overlayRef.hasAttached()) {\r\n      this._overlayRef.detach();\r\n    }\r\n  }\r\n\r\n  private _setWidthsRelativeTo(overlayRef: OverlayRef) {\r\n    if (this.maxWidthRelativeTo && this.minWidthRelativeTo) {\r\n      const elem: Element = document.getElementById(this.maxWidthRelativeTo);\r\n      overlayRef.updateSize({\r\n        minWidth: elem.clientWidth,\r\n        maxWidth: elem.clientWidth\r\n      });\r\n    } else if (this.maxWidthRelativeTo) {\r\n      const elem: Element = document.getElementById(this.maxWidthRelativeTo);\r\n      overlayRef.updateSize({ maxWidth: elem.clientWidth });\r\n    } else if (this.minWidthRelativeTo) {\r\n      const elem: Element = document.getElementById(this.minWidthRelativeTo);\r\n      overlayRef.updateSize({ minWidth: elem.clientWidth });\r\n    }\r\n  }\r\n\r\n  private _updatePosition() {\r\n    const position = this._overlayRef!.getConfig()\r\n      .positionStrategy as FlexibleConnectedPositionStrategy;\r\n    const origin = this._getOrigin();\r\n    const overlay = this._getOverlayPosition();\r\n\r\n    position.withPositions([\r\n      { ...origin.main, ...overlay.main },\r\n      { ...origin.fallback, ...overlay.fallback }\r\n    ]);\r\n  }\r\n\r\n  /**\r\n   * Returns the origin position and a fallback position based on the user's position preference.\r\n   * The fallback position is the inverse of the origin (e.g. `'bottom' -> 'top'`).\r\n   */\r\n  private _getOrigin(): {\r\n    main: OriginConnectionPosition;\r\n    fallback: OriginConnectionPosition;\r\n  } {\r\n    const placement = this.placement;\r\n    let originPlacement: OriginConnectionPosition;\r\n\r\n    if (placement === 'top' || placement === 'bottom') {\r\n      originPlacement = {\r\n        originX: 'start',\r\n        originY: placement === 'top' ? 'top' : 'bottom'\r\n      };\r\n    } else if (placement === 'left') {\r\n      originPlacement = { originX: 'start', originY: 'center' };\r\n    } else if (placement === 'right') {\r\n      originPlacement = { originX: 'end', originY: 'center' };\r\n    } else {\r\n      console.error('Position error', placement);\r\n    }\r\n\r\n    const { x, y } = this._invertPosition(\r\n      originPlacement.originX,\r\n      originPlacement.originY\r\n    );\r\n\r\n    return {\r\n      main: originPlacement,\r\n      fallback: { originX: x, originY: y }\r\n    };\r\n  }\r\n\r\n  /** Returns the overlay position and a fallback position based on the user's preference */\r\n  private _getOverlayPosition(): {\r\n    main: OverlayConnectionPosition;\r\n    fallback: OverlayConnectionPosition;\r\n  } {\r\n    const placement = this.placement;\r\n    let overlayPlacement: OverlayConnectionPosition;\r\n\r\n    if (placement === 'top') {\r\n      overlayPlacement = { overlayX: 'start', overlayY: 'bottom' };\r\n    } else if (placement === 'bottom') {\r\n      overlayPlacement = { overlayX: 'start', overlayY: 'top' };\r\n    } else if (placement === 'left') {\r\n      overlayPlacement = { overlayX: 'end', overlayY: 'center' };\r\n    } else if (placement === 'right') {\r\n      overlayPlacement = { overlayX: 'start', overlayY: 'center' };\r\n    } else {\r\n      console.error('Could not find a position', placement);\r\n    }\r\n\r\n    const { x, y } = this._invertPosition(\r\n      overlayPlacement.overlayX,\r\n      overlayPlacement.overlayY\r\n    );\r\n\r\n    return {\r\n      main: overlayPlacement,\r\n      fallback: { overlayX: x, overlayY: y }\r\n    };\r\n  }\r\n\r\n  private _invertPosition(\r\n    x: HorizontalConnectionPos,\r\n    y: VerticalConnectionPos\r\n  ) {\r\n    if (this.placement === 'top' || this.placement === 'bottom') {\r\n      if (y === 'top') {\r\n        y = 'bottom';\r\n      } else if (y === 'bottom') {\r\n        y = 'top';\r\n      }\r\n    } else {\r\n      if (x === 'end') {\r\n        x = 'start';\r\n      } else if (x === 'start') {\r\n        x = 'end';\r\n      }\r\n    }\r\n\r\n    return { x, y };\r\n  }\r\n\r\n  // Create a clip path mask in the backdrop. The mask is a rectangle, the size of the viewcontainer\r\n  // This enables the user to interact with the contents of the viewcontainer without closing the dropdown\r\n  private _addClipPathMaskStyles() {\r\n    const HTMLEl = this._overlayRef.backdropElement;\r\n    const viewRefNativeEl = this._viewContainerRef.element.nativeElement;\r\n    const elRect = viewRefNativeEl.getBoundingClientRect();\r\n    if (elRect) {\r\n      const clipPathStyle = `polygon(0% 0%, 0% 100%, ${elRect.left}px 100%, ${elRect.left}px ${elRect.top}px, ${elRect.right}px ${elRect.top}px, ${elRect.right}px ${elRect.bottom}px, ${elRect.left}px ${elRect.bottom}px, ${elRect.left}px 100%, 100% 100%, 100% 0%)`;\r\n      this.renderer.setStyle(HTMLEl, 'clip-path', clipPathStyle);\r\n    }\r\n  }\r\n}\r\n"]}