@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
101 lines • 13.7 kB
JavaScript
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"]}