@taiga-ui/kit
Version:
Taiga UI Angular main components kit
118 lines • 20.5 kB
JavaScript
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 { ResizeObserverService } 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(ResizeObserverService, { 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: [ResizeObserverService], 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: [ResizeObserverService], 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 type {OnChanges, QueryList} from '@angular/core';\nimport {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    DestroyRef,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    inject,\n    Input,\n    Output,\n} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {ResizeObserverService} 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 type {Observable} from 'rxjs';\nimport {delay} 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: [ResizeObserverService],\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(ResizeObserverService, {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"]}