UNPKG

@bimeister/pupakit.kit

Version:

PupaKit is an open source collection of Angular components based on an atomic approach to building interfaces, which guarantees better performance and greater development flexibility.

120 lines 18.4 kB
import { DOCUMENT } from '@angular/common'; import { Directive, ElementRef, HostBinding, Inject, Input, NgZone, Optional, } from '@angular/core'; import '@bimeister/utilities'; import { BehaviorSubject, fromEvent, merge, Subscription } from 'rxjs'; import { switchMap, take } from 'rxjs/operators'; import '../../../declarations/classes/opened-dropdown.class'; import '../../../declarations/interfaces/dropdown-directive-params.interface'; import '../../../declarations/interfaces/dropdown-host.interface'; import '../components/dropdown-template/dropdown-template.component'; import { subscribeInsideAngular, subscribeOutsideAngular, ThemeService } from '@bimeister/pupakit.common'; import { DropdownsService } from '../../../services/dropdowns.service'; import { ThemeWrapperService } from '../../theme-wrapper/services/theme-wrapper.service'; import * as i0 from "@angular/core"; import * as i1 from "../../../services/dropdowns.service"; import * as i2 from "@bimeister/pupakit.common"; import * as i3 from "../../theme-wrapper/services/theme-wrapper.service"; const CURSOR_POINTER = 'pointer'; export class DropdownDirective { constructor(triggerRef, dropdownsService, ngZone, themeService, document, themeWrapperService) { this.triggerRef = triggerRef; this.dropdownsService = dropdownsService; this.ngZone = ngZone; this.themeService = themeService; this.document = document; this.themeWrapperService = themeWrapperService; this.pupaDropdownDisabled = false; this.cursorStyle = CURSOR_POINTER; this.opened$ = new BehaviorSubject(false); this.theme$ = this.themeWrapperService?.theme$ ?? this.themeService.theme$; this.subscription = new Subscription(); this.isTriggerTouched$ = new BehaviorSubject(false); this.dropdown = null; this.outsideTouchEventSubscription = null; this.params = null; } ngAfterViewInit() { this.subscription.add(this.handleTriggerClickEvents()); this.subscription.add(this.handleTriggerTouchEvents()); } ngOnDestroy() { this.subscription.unsubscribe(); } setDropdownParams(params) { this.params = params; } open() { if (this.params === null) { throw new Error('[DropdownDirective] dropdownParams has not set. You can set *pupaDropdownTemplate on element'); } if (this.pupaDropdownDisabled) { return; } this.theme$ .pipe(take(1), switchMap((theme) => { this.dropdown = this.dropdownsService.open({ target: this.pupaDropdownRealTriggerElement ?? this.triggerRef.nativeElement, widthType: this.params.widthType, horizontalPosition: this.params.horizontalPosition, theme, data: { templateRef: this.params.templateRef, }, }); this.opened$.next(true); this.isTriggerTouched$.next(false); this.outsideTouchEventSubscription = this.handleOutsideTriggerTouchEvents(); return this.dropdown.closed$; })) .subscribe(() => { this.outsideTouchEventSubscription?.unsubscribe(); this.opened$.next(false); this.dropdown = null; }); } close() { this.dropdown?.close(); } toggle() { this.opened$ .pipe(take(1), subscribeInsideAngular(this.ngZone)) .subscribe((opened) => (opened ? this.close() : this.open())); } handleTriggerClickEvents() { return fromEvent(this.triggerRef.nativeElement, 'click').subscribe(() => this.toggle()); } handleTriggerTouchEvents() { return merge(fromEvent(this.triggerRef.nativeElement, 'touchstart'), fromEvent(this.triggerRef.nativeElement, 'mousedown')) .pipe(subscribeOutsideAngular(this.ngZone)) .subscribe(() => { this.isTriggerTouched$.next(true); }); } handleOutsideTriggerTouchEvents() { return merge(fromEvent(this.document, 'mousedown'), fromEvent(this.document, 'touchstart')) .pipe(switchMap(() => this.isTriggerTouched$.pipe(take(1))), subscribeOutsideAngular(this.ngZone)) .subscribe((isTriggerTouched) => (isTriggerTouched ? this.isTriggerTouched$.next(false) : this.close())); } } DropdownDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: DropdownDirective, deps: [{ token: i0.ElementRef }, { token: i1.DropdownsService }, { token: i0.NgZone }, { token: i2.ThemeService }, { token: DOCUMENT }, { token: i3.ThemeWrapperService, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); DropdownDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.2.12", type: DropdownDirective, selector: "[pupaDropdown]", inputs: { pupaDropdownDisabled: "pupaDropdownDisabled", pupaDropdownRealTriggerElement: "pupaDropdownRealTriggerElement" }, host: { properties: { "style.cursor": "this.cursorStyle" } }, exportAs: ["pupaDropdown"], ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: DropdownDirective, decorators: [{ type: Directive, args: [{ selector: '[pupaDropdown]', exportAs: 'pupaDropdown', }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.DropdownsService }, { type: i0.NgZone }, { type: i2.ThemeService }, { type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: i3.ThemeWrapperService, decorators: [{ type: Optional }] }]; }, propDecorators: { pupaDropdownDisabled: [{ type: Input }], pupaDropdownRealTriggerElement: [{ type: Input }], cursorStyle: [{ type: HostBinding, args: ['style.cursor'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.directive.js","sourceRoot":"","sources":["../../../../../src/components/dropdown/directives/dropdown.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  AfterViewInit,\n  Directive,\n  ElementRef,\n  HostBinding,\n  Inject,\n  Input,\n  NgZone,\n  OnDestroy,\n  Optional,\n} from '@angular/core';\nimport { Nullable } from '@bimeister/utilities';\nimport { BehaviorSubject, fromEvent, merge, Observable, Subscription } from 'rxjs';\nimport { switchMap, take } from 'rxjs/operators';\nimport { OpenedDropdown } from '../../../declarations/classes/opened-dropdown.class';\nimport { DropdownDirectiveParams } from '../../../declarations/interfaces/dropdown-directive-params.interface';\nimport { DropdownHost } from '../../../declarations/interfaces/dropdown-host.interface';\nimport { DropdownTemplateComponent } from '../components/dropdown-template/dropdown-template.component';\nimport { subscribeInsideAngular, subscribeOutsideAngular, Theme, ThemeService } from '@bimeister/pupakit.common';\nimport { DropdownsService } from '../../../services/dropdowns.service';\nimport { ThemeWrapperService } from '../../theme-wrapper/services/theme-wrapper.service';\n\nconst CURSOR_POINTER: string = 'pointer';\n\n@Directive({\n  selector: '[pupaDropdown]',\n  exportAs: 'pupaDropdown',\n})\nexport class DropdownDirective implements AfterViewInit, OnDestroy, DropdownHost {\n  @Input() public pupaDropdownDisabled: boolean = false;\n  @Input() public pupaDropdownRealTriggerElement?: HTMLElement;\n\n  @HostBinding('style.cursor') public cursorStyle: string = CURSOR_POINTER;\n\n  public readonly opened$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\n  private readonly theme$: Observable<Theme> = this.themeWrapperService?.theme$ ?? this.themeService.theme$;\n\n  private readonly subscription: Subscription = new Subscription();\n  private readonly isTriggerTouched$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\n  private dropdown: Nullable<OpenedDropdown> = null;\n  private outsideTouchEventSubscription: Nullable<Subscription> = null;\n\n  private params: DropdownDirectiveParams | null = null;\n\n  constructor(\n    public readonly triggerRef: ElementRef<HTMLElement>,\n    private readonly dropdownsService: DropdownsService,\n    private readonly ngZone: NgZone,\n    private readonly themeService: ThemeService,\n    @Inject(DOCUMENT) private readonly document: Document,\n    @Optional() private readonly themeWrapperService?: ThemeWrapperService\n  ) {}\n\n  public ngAfterViewInit(): void {\n    this.subscription.add(this.handleTriggerClickEvents());\n    this.subscription.add(this.handleTriggerTouchEvents());\n  }\n\n  public ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  public setDropdownParams(params: DropdownDirectiveParams): void {\n    this.params = params;\n  }\n\n  public open(): void {\n    if (this.params === null) {\n      throw new Error('[DropdownDirective] dropdownParams has not set. You can set *pupaDropdownTemplate on element');\n    }\n\n    if (this.pupaDropdownDisabled) {\n      return;\n    }\n\n    this.theme$\n      .pipe(\n        take(1),\n        switchMap((theme: Theme) => {\n          this.dropdown = this.dropdownsService.open<DropdownTemplateComponent<unknown>>({\n            target: this.pupaDropdownRealTriggerElement ?? this.triggerRef.nativeElement,\n            widthType: this.params.widthType,\n            horizontalPosition: this.params.horizontalPosition,\n            theme,\n            data: {\n              templateRef: this.params.templateRef,\n            },\n          });\n\n          this.opened$.next(true);\n          this.isTriggerTouched$.next(false);\n          this.outsideTouchEventSubscription = this.handleOutsideTriggerTouchEvents();\n\n          return this.dropdown.closed$;\n        })\n      )\n      .subscribe(() => {\n        this.outsideTouchEventSubscription?.unsubscribe();\n        this.opened$.next(false);\n        this.dropdown = null;\n      });\n  }\n\n  public close(): void {\n    this.dropdown?.close();\n  }\n\n  public toggle(): void {\n    this.opened$\n      .pipe(take(1), subscribeInsideAngular(this.ngZone))\n      .subscribe((opened: boolean) => (opened ? this.close() : this.open()));\n  }\n\n  private handleTriggerClickEvents(): Subscription {\n    return fromEvent(this.triggerRef.nativeElement, 'click').subscribe(() => this.toggle());\n  }\n\n  private handleTriggerTouchEvents(): Subscription {\n    return merge(\n      fromEvent(this.triggerRef.nativeElement, 'touchstart'),\n      fromEvent(this.triggerRef.nativeElement, 'mousedown')\n    )\n      .pipe(subscribeOutsideAngular(this.ngZone))\n      .subscribe(() => {\n        this.isTriggerTouched$.next(true);\n      });\n  }\n\n  private handleOutsideTriggerTouchEvents(): Subscription {\n    return merge(fromEvent(this.document, 'mousedown'), fromEvent(this.document, 'touchstart'))\n      .pipe(\n        switchMap(() => this.isTriggerTouched$.pipe(take(1))),\n        subscribeOutsideAngular(this.ngZone)\n      )\n      .subscribe((isTriggerTouched: boolean) => (isTriggerTouched ? this.isTriggerTouched$.next(false) : this.close()));\n  }\n}\n"]}