UNPKG

@taiga-ui/kit

Version:
247 lines • 34.3 kB
import { __decorate, __param } from "tslib"; import { ChangeDetectorRef, Directive, ElementRef, HostBinding, Inject, Input, ViewChild, } from '@angular/core'; import { NgControl } from '@angular/forms'; import { AbstractTuiControl, clamp, quantize, round, setNativeFocused, tuiDefaultProp, typedFromEvent, } from '@taiga-ui/cdk'; import { TUI_FLOATING_PRECISION } from '@taiga-ui/kit/constants'; import { TUI_FROM_TO_TEXTS } from '@taiga-ui/kit/tokens'; import { Observable, race, Subject } from 'rxjs'; import { map, switchMap, takeUntil } from 'rxjs/operators'; export const SLIDER_KEYBOARD_STEP = 0.05; export const DOT_WIDTH = { s: 8, m: 16, }; /** * @awful TODO: refactor * @internal * @dynamic */ let AbstractTuiSlider = class AbstractTuiSlider extends AbstractTuiControl { constructor(ngControl, changeDetectorRef, documentRef, fromToTexts$) { super(ngControl, changeDetectorRef); this.documentRef = documentRef; this.fromToTexts$ = fromToTexts$; this.min = 0; this.max = Infinity; this.segments = 0; this.steps = 0; this.pluralize = null; this.size = 'm'; this.keySteps = null; this.focusVisibleLeft = false; this.focusVisibleRight = false; // @bad TODO: handle pointer events instead of mouse and touch events this.pointerDown$ = new Subject(); } get segmented() { return this.segments > 0; } get discrete() { return this.steps > 0; } get length() { return this.max - this.min; } get isLeftFocusable() { return !this.disabled && this.focusable && this.right !== 100; } get isRightFocusable() { return !this.disabled && this.focusable && this.left !== 100; } ngOnInit() { super.ngOnInit(); const mouseMoves$ = typedFromEvent(this.documentRef, 'mousemove'); const mouseUps$ = typedFromEvent(this.documentRef, 'mouseup'); const touchMoves$ = typedFromEvent(this.documentRef, 'touchmove'); const touchEnds$ = typedFromEvent(this.documentRef, 'touchend'); let isPointerDownRight; this.pointerDown$ .pipe(map((event) => { const rect = event.currentTarget.getBoundingClientRect(); const clientX = event instanceof MouseEvent ? event.clientX : event.touches[0].clientX; const fraction = clamp(this.getFractionFromEvents(rect, clientX), 0, 1); const deltaLeft = fraction * 100 - this.left; const deltaRight = fraction * 100 - 100 + this.right; isPointerDownRight = Math.abs(deltaLeft) > Math.abs(deltaRight) || deltaRight > 0 || (this.left === 0 && this.right === 100); const calibratedFraction = clamp(this.getCalibratedFractionFromEvents(rect, clientX, isPointerDownRight), 0, 1); const value = this.getValueFromFraction(this.fractionGuard(calibratedFraction)); this.processValue(value, isPointerDownRight); this.processFocus(isPointerDownRight); return rect; }), switchMap(rect => race([touchMoves$, mouseMoves$]).pipe(map((event) => this.getCalibratedFractionFromEvents(rect, event instanceof MouseEvent ? event.clientX : event.touches[0].clientX, isPointerDownRight)), takeUntil(race([mouseUps$, touchEnds$])))), map(fraction => this.fractionGuard(fraction))) .subscribe(fraction => { this.processValue(this.getValueFromFraction(fraction), isPointerDownRight); }); } ngOnDestroy() { super.ngOnDestroy(); this.pointerDown$.complete(); } onMouseDown(event) { if (this.disabled) { return; } event.preventDefault(); this.pointerDown$.next(event); } onTouchStart(event) { if (this.disabled) { return; } event.preventDefault(); this.pointerDown$.next(event); } isPluralized(pluralize) { return pluralize !== null && pluralize.length === 3; } decrement(right) { this.processStep(false, right); } increment(right) { this.processStep(true, right); } getSegmentLabel(segment) { return round(this.getValueFromFraction(segment / this.segments), 2); } getSegmentPrefix(segment, texts) { if (this.segments !== 1) { return ''; } if (segment === 0) { return `${texts[0]} `; } return `${texts[1]} `; } onActiveZone(active) { this.updateFocused(active); } onLeftFocusVisible(focusVisible) { this.focusVisibleLeft = focusVisible; } onRightFocusVisible(focusVisible) { this.focusVisibleRight = focusVisible; } getFractionFromValue(value) { const fraction = (value - this.min) / this.length; return this.keySteps !== null ? this.fractionValueKeyStepConverter(value, false) : clamp(Number.isFinite(fraction) ? fraction : 1, 0, 1); } getValueFromFraction(fraction) { return this.keySteps !== null ? this.fractionValueKeyStepConverter(fraction, true) : round(this.fractionGuard(fraction) * this.length + this.min, TUI_FLOATING_PRECISION); } getCalibratedFractionFromEvents(rect, clientX, _) { return this.getFractionFromEvents(rect, clientX); } processFocus(right) { if (!this.focusable || !this.dotRight || !this.dotLeft) { return; } if (right) { setNativeFocused(this.dotRight.nativeElement); } else { setNativeFocused(this.dotLeft.nativeElement); } } /** * Function for converting the fullness of the slider to a value and vice versa * taking into account the steps of linear dependence. * * @param value passed value * @param isFraction translation is carried out from fullness to value */ fractionValueKeyStepConverter(value, isFraction) { const steps = [[0, this.min]].concat(this.keySteps, [ [100, this.max], ]); let prevFraction = 0; let nextFraction = 100; let prevValue = this.min; let nextValue = this.max; for (let i = 1; i < steps.length; i++) { if ((isFraction && steps[i][0] / 100 > value) || (!isFraction && steps[i][1] > value)) { prevFraction = steps[i - 1][0] || 0; nextFraction = steps[i][0]; prevValue = steps[i - 1][1]; nextValue = steps[i][1]; break; } } const deltaFraction = nextFraction - prevFraction; const deltaValue = nextValue - prevValue; return isFraction ? round(((value * 100 - prevFraction) / deltaFraction) * deltaValue + prevValue, TUI_FLOATING_PRECISION) : clamp(((value - prevValue) / deltaValue) * deltaFraction + prevFraction, 0, 100) / 100; } fractionGuard(fraction) { return this.discrete ? clamp(quantize(fraction, 1 / this.steps), 0, 1) : clamp(fraction, 0, 1); } getFractionFromEvents(rect, clientX) { const value = clientX - rect.left - DOT_WIDTH[this.size] / 2; const total = rect.width - DOT_WIDTH[this.size]; return round(value / total, TUI_FLOATING_PRECISION); } }; AbstractTuiSlider.ctorParameters = () => [ { type: NgControl }, { type: ChangeDetectorRef }, { type: Document }, { type: Observable, decorators: [{ type: Inject, args: [TUI_FROM_TO_TEXTS,] }] } ]; __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "min", void 0); __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "max", void 0); __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "segments", void 0); __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "steps", void 0); __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "pluralize", void 0); __decorate([ Input(), HostBinding('attr.data-tui-host-size'), tuiDefaultProp() ], AbstractTuiSlider.prototype, "size", void 0); __decorate([ Input(), tuiDefaultProp() ], AbstractTuiSlider.prototype, "keySteps", void 0); __decorate([ ViewChild('dotLeft') ], AbstractTuiSlider.prototype, "dotLeft", void 0); __decorate([ ViewChild('dotRight') ], AbstractTuiSlider.prototype, "dotRight", void 0); __decorate([ HostBinding('class._segmented') ], AbstractTuiSlider.prototype, "segmented", null); AbstractTuiSlider = __decorate([ Directive(), __param(3, Inject(TUI_FROM_TO_TEXTS)) ], AbstractTuiSlider); export { AbstractTuiSlider }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slider.js","sourceRoot":"ng://@taiga-ui/kit/abstract/","sources":["slider/slider.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectorRef,\n    Directive,\n    ElementRef,\n    HostBinding,\n    Inject,\n    Input,\n    ViewChild,\n} from '@angular/core';\nimport {NgControl} from '@angular/forms';\nimport {\n    AbstractTuiControl,\n    clamp,\n    quantize,\n    round,\n    setNativeFocused,\n    tuiDefaultProp,\n    TuiEventWith,\n    TuiNativeFocusableElement,\n    typedFromEvent,\n} from '@taiga-ui/cdk';\nimport {TuiPluralize, TuiSizeS, TuiWithOptionalMinMax} from '@taiga-ui/core';\nimport {TUI_FLOATING_PRECISION} from '@taiga-ui/kit/constants';\nimport {TUI_FROM_TO_TEXTS} from '@taiga-ui/kit/tokens';\nimport {TuiKeySteps} from '@taiga-ui/kit/types';\nimport {Observable, race, Subject} from 'rxjs';\nimport {map, switchMap, takeUntil} from 'rxjs/operators';\n\nexport const SLIDER_KEYBOARD_STEP = 0.05;\nexport const DOT_WIDTH: {[key: string]: number} = {\n    s: 8,\n    m: 16,\n};\n\n/**\n * @awful TODO: refactor\n * @internal\n * @dynamic\n */\n@Directive()\nexport abstract class AbstractTuiSlider<T>\n    extends AbstractTuiControl<T>\n    implements TuiWithOptionalMinMax<number> {\n    @Input()\n    @tuiDefaultProp()\n    min = 0;\n\n    @Input()\n    @tuiDefaultProp()\n    max = Infinity;\n\n    @Input()\n    @tuiDefaultProp()\n    segments = 0;\n\n    @Input()\n    @tuiDefaultProp()\n    steps = 0;\n\n    @Input()\n    @tuiDefaultProp()\n    pluralize: TuiPluralize | null = null;\n\n    @Input()\n    @HostBinding('attr.data-tui-host-size')\n    @tuiDefaultProp()\n    size: TuiSizeS = 'm';\n\n    @Input()\n    @tuiDefaultProp()\n    keySteps: TuiKeySteps | null = null;\n\n    focusVisibleLeft = false;\n\n    focusVisibleRight = false;\n\n    @ViewChild('dotLeft')\n    protected dotLeft?: ElementRef<TuiNativeFocusableElement>;\n\n    @ViewChild('dotRight')\n    protected dotRight?: ElementRef<TuiNativeFocusableElement>;\n\n    // @bad TODO: handle pointer events instead of mouse and touch events\n    private pointerDown$ = new Subject<\n        TuiEventWith<MouseEvent | TouchEvent, HTMLElement>\n    >();\n\n    protected constructor(\n        ngControl: NgControl | null,\n        changeDetectorRef: ChangeDetectorRef,\n        private readonly documentRef: Document,\n        @Inject(TUI_FROM_TO_TEXTS)\n        readonly fromToTexts$: Observable<[string, string]>,\n    ) {\n        super(ngControl, changeDetectorRef);\n    }\n\n    @HostBinding('class._segmented')\n    get segmented(): boolean {\n        return this.segments > 0;\n    }\n\n    get discrete(): boolean {\n        return this.steps > 0;\n    }\n\n    get length(): number {\n        return this.max - this.min;\n    }\n\n    get isLeftFocusable(): boolean {\n        return !this.disabled && this.focusable && this.right !== 100;\n    }\n\n    get isRightFocusable(): boolean {\n        return !this.disabled && this.focusable && this.left !== 100;\n    }\n\n    abstract get left(): number;\n\n    abstract get right(): number;\n\n    ngOnInit() {\n        super.ngOnInit();\n\n        const mouseMoves$ = typedFromEvent(this.documentRef, 'mousemove');\n        const mouseUps$ = typedFromEvent(this.documentRef, 'mouseup');\n        const touchMoves$ = typedFromEvent(this.documentRef, 'touchmove');\n        const touchEnds$ = typedFromEvent(this.documentRef, 'touchend');\n        let isPointerDownRight: boolean;\n\n        this.pointerDown$\n            .pipe(\n                map((event: MouseEvent | TouchEvent) => {\n                    const rect = (event.currentTarget as HTMLElement).getBoundingClientRect();\n                    const clientX =\n                        event instanceof MouseEvent\n                            ? event.clientX\n                            : event.touches[0].clientX;\n                    const fraction = clamp(\n                        this.getFractionFromEvents(rect, clientX),\n                        0,\n                        1,\n                    );\n                    const deltaLeft = fraction * 100 - this.left;\n                    const deltaRight = fraction * 100 - 100 + this.right;\n\n                    isPointerDownRight =\n                        Math.abs(deltaLeft) > Math.abs(deltaRight) ||\n                        deltaRight > 0 ||\n                        (this.left === 0 && this.right === 100);\n\n                    const calibratedFraction = clamp(\n                        this.getCalibratedFractionFromEvents(\n                            rect,\n                            clientX,\n                            isPointerDownRight,\n                        ),\n                        0,\n                        1,\n                    );\n                    const value = this.getValueFromFraction(\n                        this.fractionGuard(calibratedFraction),\n                    );\n\n                    this.processValue(value, isPointerDownRight);\n                    this.processFocus(isPointerDownRight);\n\n                    return rect;\n                }),\n                switchMap(rect =>\n                    race([touchMoves$, mouseMoves$]).pipe(\n                        map((event: any) =>\n                            this.getCalibratedFractionFromEvents(\n                                rect,\n                                event instanceof MouseEvent\n                                    ? event.clientX\n                                    : event.touches[0].clientX,\n                                isPointerDownRight,\n                            ),\n                        ),\n                        takeUntil(race([mouseUps$, touchEnds$])),\n                    ),\n                ),\n                map(fraction => this.fractionGuard(fraction)),\n            )\n            .subscribe(fraction => {\n                this.processValue(\n                    this.getValueFromFraction(fraction),\n                    isPointerDownRight,\n                );\n            });\n    }\n\n    ngOnDestroy() {\n        super.ngOnDestroy();\n        this.pointerDown$.complete();\n    }\n\n    onMouseDown(event: TuiEventWith<MouseEvent, HTMLElement>) {\n        if (this.disabled) {\n            return;\n        }\n\n        event.preventDefault();\n        this.pointerDown$.next(event);\n    }\n\n    onTouchStart(event: TuiEventWith<TouchEvent, HTMLElement>) {\n        if (this.disabled) {\n            return;\n        }\n\n        event.preventDefault();\n        this.pointerDown$.next(event);\n    }\n\n    isPluralized(pluralize: TuiPluralize | null): pluralize is TuiPluralize {\n        return pluralize !== null && pluralize.length === 3;\n    }\n\n    decrement(right: boolean) {\n        this.processStep(false, right);\n    }\n\n    increment(right: boolean) {\n        this.processStep(true, right);\n    }\n\n    getSegmentLabel(segment: number): number {\n        return round(this.getValueFromFraction(segment / this.segments), 2);\n    }\n\n    getSegmentPrefix(segment: number, texts: [string, string]): string {\n        if (this.segments !== 1) {\n            return '';\n        }\n\n        if (segment === 0) {\n            return `${texts[0]} `;\n        }\n\n        return `${texts[1]} `;\n    }\n\n    onActiveZone(active: boolean) {\n        this.updateFocused(active);\n    }\n\n    onLeftFocusVisible(focusVisible: boolean) {\n        this.focusVisibleLeft = focusVisible;\n    }\n\n    onRightFocusVisible(focusVisible: boolean) {\n        this.focusVisibleRight = focusVisible;\n    }\n\n    protected abstract processValue(value: number, right?: boolean): void;\n\n    protected abstract processStep(increment: boolean, right?: boolean): void;\n\n    protected getFractionFromValue(value: number): number {\n        const fraction = (value - this.min) / this.length;\n\n        return this.keySteps !== null\n            ? this.fractionValueKeyStepConverter(value, false)\n            : clamp(Number.isFinite(fraction) ? fraction : 1, 0, 1);\n    }\n\n    protected getValueFromFraction(fraction: number): number {\n        return this.keySteps !== null\n            ? this.fractionValueKeyStepConverter(fraction, true)\n            : round(\n                  this.fractionGuard(fraction) * this.length + this.min,\n                  TUI_FLOATING_PRECISION,\n              );\n    }\n\n    protected getCalibratedFractionFromEvents(\n        rect: ClientRect,\n        clientX: number,\n        _: boolean,\n    ): number {\n        return this.getFractionFromEvents(rect, clientX);\n    }\n\n    private processFocus(right: boolean) {\n        if (!this.focusable || !this.dotRight || !this.dotLeft) {\n            return;\n        }\n\n        if (right) {\n            setNativeFocused(this.dotRight.nativeElement);\n        } else {\n            setNativeFocused(this.dotLeft.nativeElement);\n        }\n    }\n\n    /**\n     * Function for converting the fullness of the slider to a value and vice versa\n     * taking into account the steps of linear dependence.\n     *\n     * @param value passed value\n     * @param isFraction translation is carried out from fullness to value\n     */\n    private fractionValueKeyStepConverter(value: number, isFraction: boolean): number {\n        const steps = [[0, this.min]].concat(this.keySteps as TuiKeySteps, [\n            [100, this.max],\n        ]);\n\n        let prevFraction = 0;\n        let nextFraction = 100;\n        let prevValue = this.min;\n        let nextValue = this.max;\n\n        for (let i = 1; i < steps.length; i++) {\n            if (\n                (isFraction && steps[i][0] / 100 > value) ||\n                (!isFraction && steps[i][1] > value)\n            ) {\n                prevFraction = steps[i - 1][0] || 0;\n                nextFraction = steps[i][0];\n                prevValue = steps[i - 1][1];\n                nextValue = steps[i][1];\n                break;\n            }\n        }\n\n        const deltaFraction = nextFraction - prevFraction;\n        const deltaValue = nextValue - prevValue;\n\n        return isFraction\n            ? round(\n                  ((value * 100 - prevFraction) / deltaFraction) * deltaValue + prevValue,\n                  TUI_FLOATING_PRECISION,\n              )\n            : clamp(\n                  ((value - prevValue) / deltaValue) * deltaFraction + prevFraction,\n                  0,\n                  100,\n              ) / 100;\n    }\n\n    private fractionGuard(fraction: number): number {\n        return this.discrete\n            ? clamp(quantize(fraction, 1 / this.steps), 0, 1)\n            : clamp(fraction, 0, 1);\n    }\n\n    private getFractionFromEvents(rect: ClientRect, clientX: number): number {\n        const value = clientX - rect.left - DOT_WIDTH[this.size] / 2;\n        const total = rect.width - DOT_WIDTH[this.size];\n\n        return round(value / total, TUI_FLOATING_PRECISION);\n    }\n}\n"]}