UNPKG

@taiga-ui/kit

Version:

Taiga UI Angular main components kit

118 lines 20.5 kB
import { __decorate } from "tslib"; import { AsyncPipe, NgIf } from '@angular/common'; import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChildren, DestroyRef, ElementRef, EventEmitter, forwardRef, inject, Input, Output, } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { WaResizeObserverService } from '@ng-web-apis/resize-observer'; import { EMPTY_QUERY } from '@taiga-ui/cdk/constants'; import { tuiQueryListChanges } from '@taiga-ui/cdk/observables'; import { TuiScrollService } from '@taiga-ui/cdk/services'; import { tuiInjectElement, tuiIsElement } from '@taiga-ui/cdk/utils/dom'; import { tuiMoveFocus } from '@taiga-ui/cdk/utils/focus'; import { tuiGetOriginalArrayFromQueryList, tuiPure, } from '@taiga-ui/cdk/utils/miscellaneous'; import { TUI_ANIMATIONS_SPEED } from '@taiga-ui/core/tokens'; import { tuiGetDuration } from '@taiga-ui/core/utils/miscellaneous'; import { delay } from 'rxjs'; import { TuiStep } from './step.component'; import * as i0 from "@angular/core"; class TuiStepperComponent { constructor() { this.steps = EMPTY_QUERY; this.cdr = inject(ChangeDetectorRef); this.el = tuiInjectElement(); this.scrollService = inject(TuiScrollService); this.speed = inject(TUI_ANIMATIONS_SPEED); this.destroyRef = inject(DestroyRef); this.$ = inject(WaResizeObserverService, { self: true }) .pipe(takeUntilDestroyed()) .subscribe(() => this.scrollIntoView(this.activeItemIndex)); this.activeItemIndex = 0; this.orientation = 'horizontal'; this.activeItemIndexChange = new EventEmitter(); } ngOnChanges() { this.scrollIntoView(this.activeItemIndex); } indexOf(step) { const index = tuiGetOriginalArrayFromQueryList(this.steps).findIndex(({ nativeElement }) => nativeElement === step); return index < 0 ? NaN : index; } isActive(index) { return index === this.activeItemIndex; } activate(index) { if (this.activeItemIndex === index) { return; } this.activeItemIndex = index; this.activeItemIndexChange.emit(index); this.cdr.markForCheck(); this.scrollIntoView(index); } get changes$() { // Delay is required to trigger change detection after steps are rendered, // so they can update their "active" status return tuiQueryListChanges(this.steps).pipe(delay(0)); } onHorizontal(event, step) { if (this.orientation !== 'horizontal' || !event.target) { return; } event.preventDefault(); this.moveFocus(event.target, step); } onVertical(event, step) { if (this.orientation !== 'vertical' || !event.target) { return; } event.preventDefault(); this.moveFocus(event.target, step); } moveFocus(current, step) { if (!tuiIsElement(current)) { return; } const stepElements = this.steps.toArray().map(({ nativeElement }) => nativeElement); const index = stepElements.findIndex((element) => element === current); tuiMoveFocus(index, stepElements, step); } scrollIntoView(index) { const step = this.steps.get(index)?.nativeElement; if (!step) { return; } const { clientHeight, clientWidth, offsetTop, offsetLeft } = this.el; const { offsetHeight, offsetWidth, offsetTop: stepOffsetTop, offsetLeft: stepOffsetLeft, } = step; const top = stepOffsetTop - offsetTop - clientHeight / 2 + offsetHeight / 2; const left = stepOffsetLeft - offsetLeft - clientWidth / 2 + offsetWidth / 2; this.scrollService .scroll$(this.el, Math.max(0, top), Math.max(0, left), tuiGetDuration(this.speed) / 3) .pipe(takeUntilDestroyed(this.destroyRef)) .subscribe(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiStepperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiStepperComponent, isStandalone: true, selector: "tui-stepper, nav[tuiStepper]", inputs: { activeItemIndex: "activeItemIndex", orientation: "orientation" }, outputs: { activeItemIndexChange: "activeItemIndexChange" }, host: { listeners: { "keydown.arrowRight": "onHorizontal($event, 1)", "keydown.arrowLeft": "onHorizontal($event, -1)", "keydown.arrowDown": "onVertical($event, 1)", "keydown.arrowUp": "onVertical($event, -1)" }, properties: { "attr.data-orientation": "orientation" } }, providers: [WaResizeObserverService], queries: [{ propertyName: "steps", predicate: i0.forwardRef(function () { return TuiStep; }), read: ElementRef }], usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"changes$ | async\" />\n<ng-content />\n", styles: [":host{scrollbar-width:none;-ms-overflow-style:none;display:flex;scroll-behavior:var(--tui-scroll-behavior);overflow:auto;counter-reset:steps;font:var(--tui-font-text-m)}:host::-webkit-scrollbar,:host::-webkit-scrollbar-thumb{display:none}:host[data-orientation=vertical]{flex-direction:column}\n"], dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } __decorate([ tuiPure ], TuiStepperComponent.prototype, "changes$", null); export { TuiStepperComponent }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiStepperComponent, decorators: [{ type: Component, args: [{ standalone: true, selector: 'tui-stepper, nav[tuiStepper]', imports: [AsyncPipe, NgIf], changeDetection: ChangeDetectionStrategy.OnPush, providers: [WaResizeObserverService], host: { '[attr.data-orientation]': 'orientation', '(keydown.arrowRight)': 'onHorizontal($event, 1)', '(keydown.arrowLeft)': 'onHorizontal($event, -1)', '(keydown.arrowDown)': 'onVertical($event, 1)', '(keydown.arrowUp)': 'onVertical($event, -1)', }, template: "<ng-container *ngIf=\"changes$ | async\" />\n<ng-content />\n", styles: [":host{scrollbar-width:none;-ms-overflow-style:none;display:flex;scroll-behavior:var(--tui-scroll-behavior);overflow:auto;counter-reset:steps;font:var(--tui-font-text-m)}:host::-webkit-scrollbar,:host::-webkit-scrollbar-thumb{display:none}:host[data-orientation=vertical]{flex-direction:column}\n"] }] }], propDecorators: { steps: [{ type: ContentChildren, args: [forwardRef(() => TuiStep), { read: ElementRef }] }], activeItemIndex: [{ type: Input }], orientation: [{ type: Input }], activeItemIndexChange: [{ type: Output }], changes$: [] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stepper.component.js","sourceRoot":"","sources":["../../../../../projects/kit/components/stepper/stepper.component.ts","../../../../../projects/kit/components/stepper/stepper.template.html"],"names":[],"mappings":"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","sourcesContent":["import {AsyncPipe, NgIf} from '@angular/common';\nimport {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    DestroyRef,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    inject,\n    Input,\n    type OnChanges,\n    Output,\n    type QueryList,\n} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {WaResizeObserverService} from '@ng-web-apis/resize-observer';\nimport {EMPTY_QUERY} from '@taiga-ui/cdk/constants';\nimport {tuiQueryListChanges} from '@taiga-ui/cdk/observables';\nimport {TuiScrollService} from '@taiga-ui/cdk/services';\nimport {tuiInjectElement, tuiIsElement} from '@taiga-ui/cdk/utils/dom';\nimport {tuiMoveFocus} from '@taiga-ui/cdk/utils/focus';\nimport {\n    tuiGetOriginalArrayFromQueryList,\n    tuiPure,\n} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {TUI_ANIMATIONS_SPEED} from '@taiga-ui/core/tokens';\nimport {type TuiOrientation} from '@taiga-ui/core/types';\nimport {tuiGetDuration} from '@taiga-ui/core/utils/miscellaneous';\nimport {delay, type Observable} from 'rxjs';\n\nimport {TuiStep} from './step.component';\n\n@Component({\n    standalone: true,\n    selector: 'tui-stepper, nav[tuiStepper]',\n    imports: [AsyncPipe, NgIf],\n    templateUrl: './stepper.template.html',\n    styleUrls: ['./stepper.style.less'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    providers: [WaResizeObserverService],\n    host: {\n        '[attr.data-orientation]': 'orientation',\n        '(keydown.arrowRight)': 'onHorizontal($event, 1)',\n        '(keydown.arrowLeft)': 'onHorizontal($event, -1)',\n        '(keydown.arrowDown)': 'onVertical($event, 1)',\n        '(keydown.arrowUp)': 'onVertical($event, -1)',\n    },\n})\nexport class TuiStepperComponent implements OnChanges {\n    @ContentChildren(forwardRef(() => TuiStep), {read: ElementRef})\n    private readonly steps: QueryList<ElementRef<HTMLElement>> = EMPTY_QUERY;\n\n    private readonly cdr = inject(ChangeDetectorRef);\n    private readonly el = tuiInjectElement();\n    private readonly scrollService = inject(TuiScrollService);\n    private readonly speed = inject(TUI_ANIMATIONS_SPEED);\n    private readonly destroyRef = inject(DestroyRef);\n\n    protected readonly $ = inject(WaResizeObserverService, {self: true})\n        .pipe(takeUntilDestroyed())\n        .subscribe(() => this.scrollIntoView(this.activeItemIndex));\n\n    @Input()\n    public activeItemIndex = 0;\n\n    @Input()\n    public orientation: TuiOrientation = 'horizontal';\n\n    @Output()\n    public readonly activeItemIndexChange = new EventEmitter<number>();\n\n    public ngOnChanges(): void {\n        this.scrollIntoView(this.activeItemIndex);\n    }\n\n    public indexOf(step: HTMLElement): number {\n        const index = tuiGetOriginalArrayFromQueryList(this.steps).findIndex(\n            ({nativeElement}) => nativeElement === step,\n        );\n\n        return index < 0 ? NaN : index;\n    }\n\n    public isActive(index: number): boolean {\n        return index === this.activeItemIndex;\n    }\n\n    public activate(index: number): void {\n        if (this.activeItemIndex === index) {\n            return;\n        }\n\n        this.activeItemIndex = index;\n        this.activeItemIndexChange.emit(index);\n        this.cdr.markForCheck();\n        this.scrollIntoView(index);\n    }\n\n    @tuiPure\n    protected get changes$(): Observable<unknown> {\n        // Delay is required to trigger change detection after steps are rendered,\n        // so they can update their \"active\" status\n        return tuiQueryListChanges(this.steps).pipe(delay(0));\n    }\n\n    protected onHorizontal(event: Event, step: number): void {\n        if (this.orientation !== 'horizontal' || !event.target) {\n            return;\n        }\n\n        event.preventDefault();\n        this.moveFocus(event.target, step);\n    }\n\n    protected onVertical(event: Event, step: number): void {\n        if (this.orientation !== 'vertical' || !event.target) {\n            return;\n        }\n\n        event.preventDefault();\n        this.moveFocus(event.target, step);\n    }\n\n    private moveFocus(current: EventTarget, step: number): void {\n        if (!tuiIsElement(current)) {\n            return;\n        }\n\n        const stepElements = this.steps.toArray().map(({nativeElement}) => nativeElement);\n        const index = stepElements.findIndex((element) => element === current);\n\n        tuiMoveFocus(index, stepElements, step);\n    }\n\n    private scrollIntoView(index: number): void {\n        const step = this.steps.get(index)?.nativeElement;\n\n        if (!step) {\n            return;\n        }\n\n        const {clientHeight, clientWidth, offsetTop, offsetLeft} = this.el;\n        const {\n            offsetHeight,\n            offsetWidth,\n            offsetTop: stepOffsetTop,\n            offsetLeft: stepOffsetLeft,\n        } = step;\n        const top = stepOffsetTop - offsetTop - clientHeight / 2 + offsetHeight / 2;\n        const left = stepOffsetLeft - offsetLeft - clientWidth / 2 + offsetWidth / 2;\n\n        this.scrollService\n            .scroll$(\n                this.el,\n                Math.max(0, top),\n                Math.max(0, left),\n                tuiGetDuration(this.speed) / 3,\n            )\n            .pipe(takeUntilDestroyed(this.destroyRef))\n            .subscribe();\n    }\n}\n","<ng-container *ngIf=\"changes$ | async\" />\n<ng-content />\n"]}