UNPKG

@doku-dev/doku-fragment

Version:

A new Angular UI library that moving away from Bootstrap and built from scratch.

101 lines 13.7 kB
import { DOCUMENT } from '@angular/common'; import { ContentChild, Directive, Inject, Input, } from '@angular/core'; import { ReplaySubject, fromEvent, takeUntil } from 'rxjs'; import { getClickType } from '../../utils/get-click-type'; import { DOKU_DROPDOWN, DOKU_DROPDOWN_MENU, DOKU_DROPDOWN_TOGGLE } from './dropdown.token'; import * as i0 from "@angular/core"; export class DokuDropdown { constructor(elementRef, renderer, ngZone, document) { this.elementRef = elementRef; this.renderer = renderer; this.ngZone = ngZone; this.document = document; /** * The position of the dropdown menu while opened based on the toggler/anchor element. * @default 'bottom-start' */ this.placement = 'bottom-start'; this.destroy$ = new ReplaySubject(); } get dropdownToggleElement() { return this.dropdownToggle?.['elementRef']?.nativeElement; } get dropdownMenuElement() { return this.dropdownMenu?.['elementRef']?.nativeElement; } ngAfterViewInit() { this.closeDropdownMenuEventHandler(); } ngOnDestroy() { this.destroy$.next(true); this.destroy$.complete(); } toggleDropdownMenu() { this.dropdownMenu?.toggle(); } showDropdownMenu() { this.dropdownMenu?.show(); } hideDropdownMenu() { this.dropdownMenu?.hide(); } /** * It will create a container for dropdown and append it to the body element. */ applyContainer() { this.resetContainer(); const bodyContainer = (this.bodyContainer = this.renderer.createElement('div')); bodyContainer.id = 'd-dropdown-portal'; this.originalDropdownMenu = this.dropdownMenuElement; this.renderer.appendChild(bodyContainer, this.originalDropdownMenu); this.renderer.appendChild(this.document.body, bodyContainer); } /** * Remove body container when dropdown closed and add back the dropdown menu to its original element position. */ resetContainer() { if (this.originalDropdownMenu) { this.renderer.appendChild(this.elementRef.nativeElement, this.originalDropdownMenu); } if (this.bodyContainer) { this.renderer.removeChild(this.document.body, this.bodyContainer); this.bodyContainer = undefined; } } closeDropdownMenuEventHandler() { this.ngZone.runOutsideAngular(() => { if (!this.dropdownToggleElement || !this.dropdownMenuElement) return; fromEvent(window, 'click', { capture: true }) .pipe(takeUntil(this.destroy$)) .subscribe((event) => { const { clickTrigger, clickOutside } = getClickType(event, [this.dropdownToggleElement], [this.dropdownMenuElement]); if (clickOutside && !clickTrigger) this.dropdownMenu?.hide(); }); }); } } DokuDropdown.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuDropdown, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.NgZone }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Directive }); DokuDropdown.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: DokuDropdown, isStandalone: true, selector: "[doku-dropdown]", inputs: { placement: "placement" }, providers: [{ provide: DOKU_DROPDOWN, useExisting: DokuDropdown }], queries: [{ propertyName: "dropdownToggle", first: true, predicate: DOKU_DROPDOWN_TOGGLE, descendants: true }, { propertyName: "dropdownMenu", first: true, predicate: DOKU_DROPDOWN_MENU, descendants: true }], exportAs: ["dokuDropdown"], ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuDropdown, decorators: [{ type: Directive, args: [{ selector: '[doku-dropdown]', exportAs: 'dokuDropdown', standalone: true, providers: [{ provide: DOKU_DROPDOWN, useExisting: DokuDropdown }], }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }]; }, propDecorators: { placement: [{ type: Input }], dropdownToggle: [{ type: ContentChild, args: [DOKU_DROPDOWN_TOGGLE] }], dropdownMenu: [{ type: ContentChild, args: [DOKU_DROPDOWN_MENU] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.directive.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/dropdown/dropdown.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  AfterViewInit,\n  ContentChild,\n  Directive,\n  ElementRef,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  Renderer2,\n} from '@angular/core';\nimport { ReplaySubject, fromEvent, takeUntil } from 'rxjs';\nimport { getClickType } from '../../utils/get-click-type';\nimport { DokuDropdownMenu } from './dropdown-menu.directive';\nimport { DokuDropdownToggle } from './dropdown-toggle.directive';\nimport { DokuDropdownPlacement } from './dropdown.interface';\nimport { DOKU_DROPDOWN, DOKU_DROPDOWN_MENU, DOKU_DROPDOWN_TOGGLE } from './dropdown.token';\n\n@Directive({\n  selector: '[doku-dropdown]',\n  exportAs: 'dokuDropdown',\n  standalone: true,\n  providers: [{ provide: DOKU_DROPDOWN, useExisting: DokuDropdown }],\n})\nexport class DokuDropdown implements OnDestroy, AfterViewInit {\n  /**\n   * The position of the dropdown menu while opened based on the toggler/anchor element.\n   * @default 'bottom-start'\n   */\n  @Input() placement: DokuDropdownPlacement = 'bottom-start';\n\n  @ContentChild(DOKU_DROPDOWN_TOGGLE) private dropdownToggle?: DokuDropdownToggle;\n  @ContentChild(DOKU_DROPDOWN_MENU) private dropdownMenu?: DokuDropdownMenu;\n\n  private bodyContainer?: HTMLElement;\n  private originalDropdownMenu?: HTMLElement;\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    protected elementRef: ElementRef,\n    private renderer: Renderer2,\n    private ngZone: NgZone,\n    @Inject(DOCUMENT) private document: Document\n  ) {}\n\n  private get dropdownToggleElement(): HTMLElement | undefined {\n    return this.dropdownToggle?.['elementRef']?.nativeElement;\n  }\n\n  private get dropdownMenuElement(): HTMLElement | undefined {\n    return this.dropdownMenu?.['elementRef']?.nativeElement;\n  }\n\n  ngAfterViewInit(): void {\n    this.closeDropdownMenuEventHandler();\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  toggleDropdownMenu() {\n    this.dropdownMenu?.toggle();\n  }\n\n  showDropdownMenu() {\n    this.dropdownMenu?.show();\n  }\n\n  hideDropdownMenu() {\n    this.dropdownMenu?.hide();\n  }\n\n  /**\n   * It will create a container for dropdown and append it to the body element.\n   */\n  protected applyContainer() {\n    this.resetContainer();\n    const bodyContainer: HTMLElement = (this.bodyContainer = this.renderer.createElement('div'));\n    bodyContainer.id = 'd-dropdown-portal';\n    this.originalDropdownMenu = this.dropdownMenuElement;\n    this.renderer.appendChild(bodyContainer, this.originalDropdownMenu);\n    this.renderer.appendChild(this.document.body, bodyContainer);\n  }\n\n  /**\n   * Remove body container when dropdown closed and add back the dropdown menu to its original element position.\n   */\n  protected resetContainer() {\n    if (this.originalDropdownMenu) {\n      this.renderer.appendChild(this.elementRef.nativeElement, this.originalDropdownMenu);\n    }\n    if (this.bodyContainer) {\n      this.renderer.removeChild(this.document.body, this.bodyContainer);\n      this.bodyContainer = undefined;\n    }\n  }\n\n  private closeDropdownMenuEventHandler() {\n    this.ngZone.runOutsideAngular(() => {\n      if (!this.dropdownToggleElement || !this.dropdownMenuElement) return;\n      fromEvent(window, 'click', { capture: true })\n        .pipe(takeUntil(this.destroy$))\n        .subscribe((event) => {\n          const { clickTrigger, clickOutside } = getClickType(\n            event,\n            [this.dropdownToggleElement],\n            [this.dropdownMenuElement]\n          );\n\n          if (clickOutside && !clickTrigger) this.dropdownMenu?.hide();\n        });\n    });\n  }\n}\n"]}