@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
JavaScript
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"]}