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.

136 lines 23 kB
import { Overlay, OverlayConfig, } from '@angular/cdk/overlay'; import { ComponentPortal } from '@angular/cdk/portal'; import { Injectable, Injector } from '@angular/core'; import { OVERLAY_VIEWPORT_MARGIN_PX } from '@bimeister/pupakit.common'; import { filterNotNil, shareReplayWithRefCount } from '@bimeister/utilities'; import { BehaviorSubject, combineLatest, Subscription } from 'rxjs'; import { debounceTime, distinctUntilChanged, filter, map, switchMap, take } from 'rxjs/operators'; import { TOOLTIP_SERVICE_TOKEN } from '../../../declarations/tokens/tooltip-service.token'; import '../../../declarations/interfaces/tooltip-service-declaration.interface'; import { TooltipContentComponent } from '../components/tooltip-content/tooltip-content.component'; import { OVERLAY_POSITIONS } from '../positions'; import * as i0 from "@angular/core"; import * as i1 from "@angular/cdk/overlay"; export class TooltipService { constructor(overlay) { this.overlay = overlay; this.subscription = new Subscription(); this.mouseOverTrigger$ = new BehaviorSubject(false); this.mouseOverContent$ = new BehaviorSubject(false); this.mouseOverTooltip$ = combineLatest([ this.mouseOverTrigger$.pipe(distinctUntilChanged()), this.mouseOverContent$.pipe(distinctUntilChanged()), ]).pipe(debounceTime(0), map(([mouseOverTrigger, mouseOverContent]) => mouseOverTrigger || mouseOverContent)); this.isOpenedState$ = new BehaviorSubject(false); this.isOpened$ = this.isOpenedState$.pipe(distinctUntilChanged()); this.isDisabledState$ = new BehaviorSubject(false); this.isDisabled$ = this.isDisabledState$.pipe(distinctUntilChanged()); this.tooltipContentState$ = new BehaviorSubject(null); this.tooltipContent$ = this.tooltipContentState$.pipe(distinctUntilChanged()); this.tooltipContentTemplateState$ = new BehaviorSubject(null); this.tooltipContentTemplate$ = this.tooltipContentTemplateState$.pipe(distinctUntilChanged()); this.tooltipHideOnHoverState$ = new BehaviorSubject(true); this.tooltipHideOnHoverHover$ = this.tooltipHideOnHoverState$.pipe(distinctUntilChanged()); this.overlayRef$ = new BehaviorSubject(null); this.triggerRef$ = new BehaviorSubject(null); this.tooltipPositionStrategy$ = new BehaviorSubject(null); this.tooltipPosition$ = this.tooltipPositionStrategy$.pipe(filterNotNil(), switchMap((positionStrategy) => positionStrategy.positionChanges), shareReplayWithRefCount()); this.subscription.add(this.processMouseOverTooltipChanges()); } ngOnDestroy() { this.close(); this.subscription.unsubscribe(); } registerTooltipTriggerRef(triggerRef) { this.triggerRef$.next(triggerRef); } setOpenedState(isOpened) { this.isOpenedState$ .pipe(take(1), filter((currentOpenedState) => currentOpenedState !== isOpened)) .subscribe(() => (isOpened ? this.open() : this.close())); } setDisabledState(isDisabled) { this.isDisabledState$.next(isDisabled); } setTooltipHideOnHoverState(hideOnTooltipHover) { this.tooltipHideOnHoverState$.next(hideOnTooltipHover); } setTooltipContentState(content) { this.tooltipContentState$.next(content); } setTooltipContentTemplateState(template) { this.tooltipContentTemplateState$.next(template); } processTriggerMouseEnter() { this.mouseOverTrigger$.next(true); } processTriggerMouseLeave() { this.mouseOverTrigger$.next(false); } processContentMouseEnter() { this.mouseOverContent$.next(true); } processContentMouseLeave() { this.mouseOverContent$.next(false); } processMouseOverTooltipChanges() { return this.mouseOverTooltip$.subscribe((mouseOverTooltip) => { this.setOpenedState(mouseOverTooltip); }); } createOverlay() { this.getPositionStrategy() .pipe(take(1), filterNotNil()) .subscribe((positionStrategy) => { const overlayConfig = new OverlayConfig({ positionStrategy, }); const overlayRef = this.overlay.create(overlayConfig); this.overlayRef$.next(overlayRef); this.tooltipPositionStrategy$.next(positionStrategy); }); } getPositionStrategy() { return this.triggerRef$.pipe(take(1), filterNotNil(), map((triggerRef) => { const positionStrategy = this.overlay .position() .flexibleConnectedTo(triggerRef) .withPositions(OVERLAY_POSITIONS) .withViewportMargin(OVERLAY_VIEWPORT_MARGIN_PX); return positionStrategy; })); } open() { this.createOverlay(); this.overlayRef$.pipe(filterNotNil(), take(1)).subscribe((overlayRef) => { overlayRef.attach(this.getComponentPortal()); overlayRef.updatePosition(); this.isOpenedState$.next(true); }); } close() { this.overlayRef$.pipe(filterNotNil(), take(1)).subscribe((overlayRef) => { this.isOpenedState$.next(false); overlayRef.dispose(); overlayRef.detach(); this.overlayRef$.next(null); }); } getComponentPortal() { const portalInjector = Injector.create({ providers: [ { provide: TOOLTIP_SERVICE_TOKEN, useValue: this, }, ], }); return new ComponentPortal(TooltipContentComponent, null, portalInjector); } } TooltipService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TooltipService, deps: [{ token: i1.Overlay }], target: i0.ɵɵFactoryTarget.Injectable }); TooltipService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TooltipService }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TooltipService, decorators: [{ type: Injectable }], ctorParameters: function () { return [{ type: i1.Overlay }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.service.js","sourceRoot":"","sources":["../../../../../src/components/tooltip/services/tooltip.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConnectedOverlayPositionChange,\n  FlexibleConnectedPositionStrategy,\n  Overlay,\n  OverlayConfig,\n  OverlayRef,\n} from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { ElementRef, Injectable, Injector, OnDestroy, TemplateRef } from '@angular/core';\nimport { OVERLAY_VIEWPORT_MARGIN_PX } from '@bimeister/pupakit.common';\nimport { filterNotNil, Nullable, shareReplayWithRefCount } from '@bimeister/utilities';\nimport { BehaviorSubject, combineLatest, Observable, Subscription } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, filter, map, switchMap, take } from 'rxjs/operators';\nimport { TOOLTIP_SERVICE_TOKEN } from '../../../declarations/tokens/tooltip-service.token';\nimport { TooltipServiceDeclaration } from '../../../declarations/interfaces/tooltip-service-declaration.interface';\nimport { TooltipContentComponent } from '../components/tooltip-content/tooltip-content.component';\nimport { OVERLAY_POSITIONS } from '../positions';\n\n@Injectable()\nexport class TooltipService implements OnDestroy, TooltipServiceDeclaration {\n  private readonly subscription: Subscription = new Subscription();\n\n  private readonly mouseOverTrigger$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  private readonly mouseOverContent$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\n  private readonly mouseOverTooltip$: Observable<boolean> = combineLatest([\n    this.mouseOverTrigger$.pipe(distinctUntilChanged()),\n    this.mouseOverContent$.pipe(distinctUntilChanged()),\n  ]).pipe(\n    debounceTime(0),\n    map(([mouseOverTrigger, mouseOverContent]: [boolean, boolean]) => mouseOverTrigger || mouseOverContent)\n  );\n\n  private readonly isOpenedState$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  public readonly isOpened$: Observable<boolean> = this.isOpenedState$.pipe(distinctUntilChanged());\n\n  private readonly isDisabledState$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  public readonly isDisabled$: Observable<boolean> = this.isDisabledState$.pipe(distinctUntilChanged());\n\n  private readonly tooltipContentState$: BehaviorSubject<Nullable<string>> = new BehaviorSubject<Nullable<string>>(\n    null\n  );\n  public readonly tooltipContent$: Observable<Nullable<string>> = this.tooltipContentState$.pipe(\n    distinctUntilChanged()\n  );\n\n  private readonly tooltipContentTemplateState$: BehaviorSubject<Nullable<TemplateRef<unknown>>> = new BehaviorSubject<\n    Nullable<TemplateRef<unknown>>\n  >(null);\n  public readonly tooltipContentTemplate$: Observable<Nullable<TemplateRef<unknown>>> =\n    this.tooltipContentTemplateState$.pipe(distinctUntilChanged());\n\n  private readonly tooltipHideOnHoverState$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(true);\n  public readonly tooltipHideOnHoverHover$: Observable<boolean> = this.tooltipHideOnHoverState$.pipe(\n    distinctUntilChanged()\n  );\n\n  private readonly overlayRef$: BehaviorSubject<Nullable<OverlayRef>> = new BehaviorSubject<Nullable<OverlayRef>>(null);\n  private readonly triggerRef$: BehaviorSubject<Nullable<ElementRef<HTMLElement>>> = new BehaviorSubject<\n    Nullable<ElementRef<HTMLElement>>\n  >(null);\n\n  private readonly tooltipPositionStrategy$: BehaviorSubject<Nullable<FlexibleConnectedPositionStrategy>> =\n    new BehaviorSubject<Nullable<FlexibleConnectedPositionStrategy>>(null);\n  public readonly tooltipPosition$: Observable<ConnectedOverlayPositionChange> = this.tooltipPositionStrategy$.pipe(\n    filterNotNil(),\n    switchMap((positionStrategy: FlexibleConnectedPositionStrategy) => positionStrategy.positionChanges),\n    shareReplayWithRefCount()\n  );\n\n  constructor(private readonly overlay: Overlay) {\n    this.subscription.add(this.processMouseOverTooltipChanges());\n  }\n\n  public ngOnDestroy(): void {\n    this.close();\n    this.subscription.unsubscribe();\n  }\n\n  public registerTooltipTriggerRef(triggerRef: ElementRef<HTMLElement>): void {\n    this.triggerRef$.next(triggerRef);\n  }\n\n  public setOpenedState(isOpened: boolean): void {\n    this.isOpenedState$\n      .pipe(\n        take(1),\n        filter((currentOpenedState: boolean) => currentOpenedState !== isOpened)\n      )\n      .subscribe(() => (isOpened ? this.open() : this.close()));\n  }\n\n  public setDisabledState(isDisabled: boolean): void {\n    this.isDisabledState$.next(isDisabled);\n  }\n\n  public setTooltipHideOnHoverState(hideOnTooltipHover: boolean): void {\n    this.tooltipHideOnHoverState$.next(hideOnTooltipHover);\n  }\n\n  public setTooltipContentState(content: string): void {\n    this.tooltipContentState$.next(content);\n  }\n\n  public setTooltipContentTemplateState(template: TemplateRef<unknown>): void {\n    this.tooltipContentTemplateState$.next(template);\n  }\n\n  public processTriggerMouseEnter(): void {\n    this.mouseOverTrigger$.next(true);\n  }\n\n  public processTriggerMouseLeave(): void {\n    this.mouseOverTrigger$.next(false);\n  }\n\n  public processContentMouseEnter(): void {\n    this.mouseOverContent$.next(true);\n  }\n\n  public processContentMouseLeave(): void {\n    this.mouseOverContent$.next(false);\n  }\n\n  private processMouseOverTooltipChanges(): Subscription {\n    return this.mouseOverTooltip$.subscribe((mouseOverTooltip: boolean) => {\n      this.setOpenedState(mouseOverTooltip);\n    });\n  }\n\n  private createOverlay(): void {\n    this.getPositionStrategy()\n      .pipe(take(1), filterNotNil())\n      .subscribe((positionStrategy: FlexibleConnectedPositionStrategy) => {\n        const overlayConfig: OverlayConfig = new OverlayConfig({\n          positionStrategy,\n        });\n        const overlayRef: OverlayRef = this.overlay.create(overlayConfig);\n        this.overlayRef$.next(overlayRef);\n        this.tooltipPositionStrategy$.next(positionStrategy);\n      });\n  }\n\n  private getPositionStrategy(): Observable<FlexibleConnectedPositionStrategy> {\n    return this.triggerRef$.pipe(\n      take(1),\n      filterNotNil(),\n      map((triggerRef: ElementRef<HTMLElement>) => {\n        const positionStrategy: FlexibleConnectedPositionStrategy = this.overlay\n          .position()\n          .flexibleConnectedTo(triggerRef)\n          .withPositions(OVERLAY_POSITIONS)\n          .withViewportMargin(OVERLAY_VIEWPORT_MARGIN_PX);\n\n        return positionStrategy;\n      })\n    );\n  }\n\n  private open(): void {\n    this.createOverlay();\n    this.overlayRef$.pipe(filterNotNil(), take(1)).subscribe((overlayRef: OverlayRef) => {\n      overlayRef.attach(this.getComponentPortal());\n      overlayRef.updatePosition();\n\n      this.isOpenedState$.next(true);\n    });\n  }\n\n  private close(): void {\n    this.overlayRef$.pipe(filterNotNil(), take(1)).subscribe((overlayRef: OverlayRef) => {\n      this.isOpenedState$.next(false);\n      overlayRef.dispose();\n      overlayRef.detach();\n      this.overlayRef$.next(null);\n    });\n  }\n\n  private getComponentPortal(): ComponentPortal<TooltipContentComponent> {\n    const portalInjector: Injector = Injector.create({\n      providers: [\n        {\n          provide: TOOLTIP_SERVICE_TOKEN,\n          useValue: this,\n        },\n      ],\n    });\n    return new ComponentPortal(TooltipContentComponent, null, portalInjector);\n  }\n}\n"]}