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.

104 lines 19.1 kB
import { ConnectionPositionPair, Overlay, } from '@angular/cdk/overlay'; import { ComponentPortal } from '@angular/cdk/portal'; import { Injectable, Injector } from '@angular/core'; import { getUuid, isNil } from '@bimeister/utilities'; import { take } from 'rxjs/operators'; import { DROPDOWN_CONTAINER_DATA_TOKEN } from '../declarations/tokens/dropdown-container-data.token'; import { DropdownContainerComponent } from '../components/dropdown/components/dropdown-container/dropdown-container.component'; import { DropdownTemplateComponent } from '../components/dropdown/components/dropdown-template/dropdown-template.component'; import '../declarations/classes/abstract/dropdown-component-base.abstract'; import { DropdownRef } from '../declarations/classes/dropdown-ref.class'; import { OpenedDropdown } from '../declarations/classes/opened-dropdown.class'; import '../declarations/interfaces/dropdown-config.interface'; import '../declarations/interfaces/dropdown-container-data.interface'; import '../declarations/types/utility-types/dropdown-data.utility-type'; import { OVERLAY_VIEWPORT_MARGIN_PX, Theme } from '@bimeister/pupakit.common'; import * as i0 from "@angular/core"; import * as i1 from "@angular/cdk/overlay"; const HORIZONTAL_POSITIONS = ['center', 'end', 'start']; const VERTICAL_POSITIONS = ['top', 'bottom']; export class DropdownsService { constructor(overlay, injector) { this.overlay = overlay; this.injector = injector; this.dropdownStore = new Map(); } open(config) { const dropdownId = getUuid(); const overlayRef = this.getOverlayRef(config); const dropdownRef = new DropdownRef(overlayRef, config); const containerPortal = this.getComponentPortal(config, dropdownRef); dropdownRef.open(containerPortal); this.dropdownStore.set(dropdownId, dropdownRef); dropdownRef.closed$.pipe(take(1)).subscribe(() => { this.dropdownStore.delete(dropdownId); overlayRef.dispose(); }); return new OpenedDropdown(dropdownId, dropdownRef); } closeById(id) { const dropdownRef = this.dropdownStore.get(id); if (isNil(dropdownRef)) { return; } dropdownRef.close(); } closeAll() { this.dropdownStore.forEach((dropdownRef) => dropdownRef.close()); } isOpen(id) { return this.dropdownStore.has(id); } getComponentPortal(config, dropdownRef) { const injector = Injector.create({ parent: config.injector ?? this.injector, providers: [{ provide: DropdownRef, useValue: dropdownRef }], }); const componentPortal = new ComponentPortal(config.component ?? DropdownTemplateComponent, null, injector); const containerData = { componentPortal, theme: config.theme ?? Theme.Light, }; return new ComponentPortal(DropdownContainerComponent, null, Injector.create({ providers: [{ provide: DROPDOWN_CONTAINER_DATA_TOKEN, useValue: containerData }], })); } getOverlayRef(config) { return this.overlay.create({ width: this.getOverlayWidth(config), positionStrategy: this.getTargetPosition(config), }); } getTargetPosition(config) { const target = config.target; return this.overlay .position() .flexibleConnectedTo(target instanceof HTMLElement ? target : { x: target[0], y: target[1] }) .withFlexibleDimensions(false) .withPositions(this.getOverlayPositionsByHorizontalPosition(config.horizontalPosition)) .withViewportMargin(OVERLAY_VIEWPORT_MARGIN_PX); } getOverlayWidth(config) { const target = config.target; if (target instanceof HTMLElement && config.widthType === 'by-trigger') { const { width } = target.getBoundingClientRect(); return width; } return 'auto'; } getOverlayPositionsByHorizontalPosition(currentHorizontalPos) { const sortedHorizontalPositions = HORIZONTAL_POSITIONS.sort((horizontalPos) => (horizontalPos === currentHorizontalPos ? -1 : 1)); const overlayPositions = VERTICAL_POSITIONS.flatMap((verticalPos) => sortedHorizontalPositions.map((horizontalPos) => this.getConnectionPositionPair(horizontalPos, verticalPos))); return overlayPositions; } getConnectionPositionPair(overlayX, overlayY) { return new ConnectionPositionPair({ originX: overlayX, originY: 'bottom' }, { overlayX, overlayY }); } } DropdownsService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: DropdownsService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable }); DropdownsService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: DropdownsService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: DropdownsService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: function () { return [{ type: i1.Overlay }, { type: i0.Injector }]; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdowns.service.js","sourceRoot":"","sources":["../../../src/services/dropdowns.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConnectionPositionPair,\n  FlexibleConnectedPositionStrategy,\n  HorizontalConnectionPos,\n  Overlay,\n  OverlayRef,\n  VerticalConnectionPos,\n} from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { Injectable, Injector } from '@angular/core';\nimport { getUuid, isNil, Uuid } from '@bimeister/utilities';\nimport { take } from 'rxjs/operators';\nimport { DROPDOWN_CONTAINER_DATA_TOKEN } from '../declarations/tokens/dropdown-container-data.token';\nimport { DropdownContainerComponent } from '../components/dropdown/components/dropdown-container/dropdown-container.component';\nimport { DropdownTemplateComponent } from '../components/dropdown/components/dropdown-template/dropdown-template.component';\nimport { DropdownComponentBase } from '../declarations/classes/abstract/dropdown-component-base.abstract';\nimport { DropdownRef } from '../declarations/classes/dropdown-ref.class';\nimport { OpenedDropdown } from '../declarations/classes/opened-dropdown.class';\nimport { DropdownConfig } from '../declarations/interfaces/dropdown-config.interface';\nimport { DropdownContainerData } from '../declarations/interfaces/dropdown-container-data.interface';\nimport { DropdownDataType } from '../declarations/types/utility-types/dropdown-data.utility-type';\nimport { OVERLAY_VIEWPORT_MARGIN_PX, Position, Theme } from '@bimeister/pupakit.common';\n\nconst HORIZONTAL_POSITIONS: HorizontalConnectionPos[] = ['center', 'end', 'start'];\nconst VERTICAL_POSITIONS: VerticalConnectionPos[] = ['top', 'bottom'];\n\n@Injectable({ providedIn: 'root' })\nexport class DropdownsService {\n  private readonly dropdownStore: Map<string, DropdownRef> = new Map();\n\n  constructor(protected readonly overlay: Overlay, protected readonly injector: Injector) {}\n\n  public open<TComponent extends DropdownComponentBase<unknown>>(\n    config: DropdownConfig<TComponent, DropdownDataType<TComponent>>\n  ): OpenedDropdown {\n    const dropdownId: Uuid = getUuid();\n    const overlayRef: OverlayRef = this.getOverlayRef(config);\n    const dropdownRef: DropdownRef<DropdownDataType<TComponent>> = new DropdownRef(overlayRef, config);\n    const containerPortal: ComponentPortal<DropdownContainerComponent> = this.getComponentPortal(config, dropdownRef);\n\n    dropdownRef.open(containerPortal);\n\n    this.dropdownStore.set(dropdownId, dropdownRef);\n\n    dropdownRef.closed$.pipe(take(1)).subscribe(() => {\n      this.dropdownStore.delete(dropdownId);\n      overlayRef.dispose();\n    });\n\n    return new OpenedDropdown(dropdownId, dropdownRef);\n  }\n\n  public closeById(id: string): void {\n    const dropdownRef: DropdownRef = this.dropdownStore.get(id);\n\n    if (isNil(dropdownRef)) {\n      return;\n    }\n\n    dropdownRef.close();\n  }\n\n  public closeAll(): void {\n    this.dropdownStore.forEach((dropdownRef: DropdownRef) => dropdownRef.close());\n  }\n\n  public isOpen(id: string): boolean {\n    return this.dropdownStore.has(id);\n  }\n\n  private getComponentPortal<TComponent extends DropdownComponentBase<unknown>>(\n    config: DropdownConfig<TComponent, DropdownDataType<TComponent>>,\n    dropdownRef: DropdownRef<DropdownDataType<TComponent>>\n  ): ComponentPortal<DropdownContainerComponent> {\n    const injector: Injector = Injector.create({\n      parent: config.injector ?? this.injector,\n      providers: [{ provide: DropdownRef, useValue: dropdownRef }],\n    });\n\n    const componentPortal: ComponentPortal<unknown> = new ComponentPortal<unknown>(\n      config.component ?? DropdownTemplateComponent,\n      null,\n      injector\n    );\n\n    const containerData: DropdownContainerData<unknown> = {\n      componentPortal,\n      theme: config.theme ?? Theme.Light,\n    };\n\n    return new ComponentPortal(\n      DropdownContainerComponent,\n      null,\n      Injector.create({\n        providers: [{ provide: DROPDOWN_CONTAINER_DATA_TOKEN, useValue: containerData }],\n      })\n    );\n  }\n\n  private getOverlayRef<TComponent extends DropdownComponentBase<unknown>>(\n    config: DropdownConfig<TComponent, DropdownDataType<TComponent>>\n  ): OverlayRef {\n    return this.overlay.create({\n      width: this.getOverlayWidth(config),\n      positionStrategy: this.getTargetPosition(config),\n    });\n  }\n\n  private getTargetPosition<TComponent extends DropdownComponentBase<unknown>>(\n    config: DropdownConfig<TComponent, DropdownDataType<TComponent>>\n  ): FlexibleConnectedPositionStrategy {\n    const target: HTMLElement | Position = config.target;\n\n    return this.overlay\n      .position()\n      .flexibleConnectedTo(target instanceof HTMLElement ? target : { x: target[0], y: target[1] })\n      .withFlexibleDimensions(false)\n      .withPositions(this.getOverlayPositionsByHorizontalPosition(config.horizontalPosition))\n      .withViewportMargin(OVERLAY_VIEWPORT_MARGIN_PX);\n  }\n\n  private getOverlayWidth<TComponent extends DropdownComponentBase<unknown>>(\n    config: DropdownConfig<TComponent, DropdownDataType<TComponent>>\n  ): string | number {\n    const target: HTMLElement | Position = config.target;\n\n    if (target instanceof HTMLElement && config.widthType === 'by-trigger') {\n      const { width } = target.getBoundingClientRect();\n      return width;\n    }\n\n    return 'auto';\n  }\n\n  private getOverlayPositionsByHorizontalPosition(\n    currentHorizontalPos: HorizontalConnectionPos\n  ): ConnectionPositionPair[] {\n    const sortedHorizontalPositions: HorizontalConnectionPos[] = HORIZONTAL_POSITIONS.sort(\n      (horizontalPos: HorizontalConnectionPos) => (horizontalPos === currentHorizontalPos ? -1 : 1)\n    );\n\n    const overlayPositions: ConnectionPositionPair[] = VERTICAL_POSITIONS.flatMap(\n      (verticalPos: VerticalConnectionPos) =>\n        sortedHorizontalPositions.map((horizontalPos: HorizontalConnectionPos) =>\n          this.getConnectionPositionPair(horizontalPos, verticalPos)\n        )\n    );\n\n    return overlayPositions;\n  }\n\n  private getConnectionPositionPair(\n    overlayX: HorizontalConnectionPos,\n    overlayY: VerticalConnectionPos\n  ): ConnectionPositionPair {\n    return new ConnectionPositionPair({ originX: overlayX, originY: 'bottom' }, { overlayX, overlayY });\n  }\n}\n"]}