UNPKG

@taiga-ui/kit

Version:

Taiga UI Angular main components kit

174 lines 30.5 kB
import { computed, Directive, effect, inject, Input, signal } from '@angular/core'; import { toSignal } from '@angular/core/rxjs-interop'; import { MaskitoDirective } from '@maskito/angular'; import { maskitoDateTimeOptionsGenerator, maskitoSelectionChangeHandler, } from '@maskito/kit'; import { tuiAsControl, tuiValueTransformerFrom } from '@taiga-ui/cdk/classes'; import { DATE_FILLER_LENGTH, MILLISECONDS_IN_DAY, TuiDay, TuiTime, } from '@taiga-ui/cdk/date-time'; import { tuiClamp, tuiSum } from '@taiga-ui/cdk/utils/math'; import { tuiDirectiveBinding } from '@taiga-ui/cdk/utils/miscellaneous'; import { tuiAsOptionContent } from '@taiga-ui/core/components/data-list'; import { tuiAsTextfieldAccessor, TuiWithTextfield, } from '@taiga-ui/core/components/textfield'; import { TuiDropdownAuto } from '@taiga-ui/core/directives/dropdown'; import { TuiItemsHandlersValidator } from '@taiga-ui/core/directives/items-handlers'; import { TUI_DATE_ADAPTER, TuiInputDateBase, tuiWithDateFiller, } from '@taiga-ui/kit/components/input-date'; import { TuiSelectOption } from '@taiga-ui/kit/components/select'; import { TUI_TIME_TEXTS } from '@taiga-ui/kit/tokens'; import { tuiMaskito } from '@taiga-ui/kit/utils'; import { noop } from 'rxjs'; import { TUI_INPUT_DATE_TIME_OPTIONS } from './input-date-time.options'; import * as i0 from "@angular/core"; import * as i1 from "@maskito/angular"; import * as i2 from "@taiga-ui/core/directives/dropdown"; import * as i3 from "@taiga-ui/core/directives/items-handlers"; import * as i4 from "@taiga-ui/core/components/textfield"; const MIN_TIME = new TuiTime(0, 0); const MAX_TIME = TuiTime.fromAbsoluteMilliseconds(MILLISECONDS_IN_DAY - 1); class TuiInputDateTimeDirective extends TuiInputDateBase { constructor() { super(...arguments); this.timeFillers = toSignal(inject(TUI_TIME_TEXTS)); this.options = inject(TUI_INPUT_DATE_TIME_OPTIONS); this.filler = tuiWithDateFiller((date) => { const time = this.timeFillers()?.[this.timeMode()] ?? ''; return `${date}${this.options.dateTimeSeparator}${time}`; }); this.valueEffect = effect(noop); this.identity = this.handlers.identityMatcher.set((a, b) => tuiSum(...a.map(Number)) === tuiSum(...b.map(Number))); this.disabledItemHandler = tuiDirectiveBinding(TuiItemsHandlersValidator, 'disabledItemHandler', (value) => Boolean(value && this.handlers.disabledItemHandler()(value))); this.mask = tuiMaskito(computed(() => this.computeMask({ dateMode: TUI_DATE_ADAPTER[this.format().mode], timeMode: this.timeMode(), min: this.toNativeDate([this.min(), this.minTime()]), max: this.toNativeDate([this.max(), this.maxTime()]), dateSeparator: this.format().separator, dateTimeSeparator: this.options.dateTimeSeparator, }))); this.timeMode = signal(this.options.timeMode); this.minTime = signal(MIN_TIME); this.maxTime = signal(MAX_TIME); } // TODO(v5): use signal inputs set timeModeSetter(x) { this.timeMode.set(x); } set minSetter(min) { const [date, time] = Array.isArray(min) ? min : [min, null]; this.min.set(date || this.options.min); this.minTime.set(time ?? MIN_TIME); } set maxSetter(max) { const [date, time] = Array.isArray(max) ? max : [max, null]; this.max.set(date || this.options.max); this.maxTime.set(time ?? MAX_TIME); } setValue(value) { this.onChange(value); this.textfield.value.set(this.stringify(value)); } setDate(newDate) { const [date, time] = this.clampTime([newDate, this.value()?.[1] ?? null]); this.setValue([date, time]); this.open.set(false); setTimeout((caretIndex = DATE_FILLER_LENGTH + this.options.dateTimeSeparator.length) => this.el.setSelectionRange(caretIndex, caretIndex)); } writeValue(value) { super.writeValue(value); this.textfield.value.set(this.stringify(this.value())); } processCalendar(calendar) { super.processCalendar(calendar); calendar.disabledItemHandler = (day) => this.handlers.disabledItemHandler()([day, null]); } onValueChange(value) { this.textfield.value.set(value); this.control?.control?.updateValueAndValidity({ emitEvent: false }); const [date = '', time = ''] = value.split(this.options.dateTimeSeparator); const parsedDate = date.length >= DATE_FILLER_LENGTH ? TuiDay.normalizeParse(date, this.format().mode) : null; const parsedTime = time.length === this.timeMode().length ? TuiTime.fromString(time) : null; if (!parsedDate || (time && !parsedTime)) { return this.onChange(null); } const [prevDate, prevTime = null] = this.value() ?? []; if (!prevDate?.daySame(parsedDate) || Number(parsedTime) !== Number(prevTime)) { this.onChange([parsedDate, parsedTime]); } } stringify(value) { const [date, time] = value ?? []; const dateString = date?.toString(this.format().mode, this.format().separator) ?? ''; const timeString = time?.toString(this.timeMode()); return timeString ? `${dateString}${this.options.dateTimeSeparator}${timeString}` : dateString; } clampTime([date, time]) { const min = date.daySame(this.min()) ? this.minTime().toAbsoluteMilliseconds() : -Infinity; const max = date.daySame(this.max()) ? this.maxTime().toAbsoluteMilliseconds() : Infinity; return [ date, time && TuiTime.fromAbsoluteMilliseconds(tuiClamp(time.toAbsoluteMilliseconds(), min, max)), ]; } computeMask(params) { const options = maskitoDateTimeOptionsGenerator(params); const { timeMode, dateMode, dateTimeSeparator } = params; const inputModeSwitchPlugin = maskitoSelectionChangeHandler((element) => { element.inputMode = element.selectionStart >= dateMode.length + dateTimeSeparator.length + timeMode.indexOf(' AA') ? 'text' : 'numeric'; }); return { ...options, plugins: options.plugins.concat(timeMode.includes('AA') ? inputModeSwitchPlugin : []), }; } toNativeDate([{ year, month, day }, { hours, minutes, seconds, ms }]) { return new Date(year, month, day, hours, minutes, seconds, ms); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputDateTimeDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: TuiInputDateTimeDirective, isStandalone: true, selector: "input[tuiInputDateTime]", inputs: { timeModeSetter: ["timeMode", "timeModeSetter"], minSetter: ["min", "minSetter"], maxSetter: ["max", "maxSetter"] }, providers: [ tuiAsOptionContent(TuiSelectOption), tuiAsControl(TuiInputDateTimeDirective), tuiAsTextfieldAccessor(TuiInputDateTimeDirective), tuiValueTransformerFrom(TUI_INPUT_DATE_TIME_OPTIONS), ], usesInheritance: true, hostDirectives: [{ directive: i1.MaskitoDirective }, { directive: i2.TuiDropdownAuto }, { directive: i3.TuiItemsHandlersValidator }, { directive: i4.TuiWithTextfield }], ngImport: i0 }); } } export { TuiInputDateTimeDirective }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputDateTimeDirective, decorators: [{ type: Directive, args: [{ standalone: true, selector: 'input[tuiInputDateTime]', providers: [ tuiAsOptionContent(TuiSelectOption), tuiAsControl(TuiInputDateTimeDirective), tuiAsTextfieldAccessor(TuiInputDateTimeDirective), tuiValueTransformerFrom(TUI_INPUT_DATE_TIME_OPTIONS), ], hostDirectives: [ MaskitoDirective, TuiDropdownAuto, TuiItemsHandlersValidator, TuiWithTextfield, ], }] }], propDecorators: { timeModeSetter: [{ type: Input, args: ['timeMode'] }], minSetter: [{ type: Input, args: ['min'] }], maxSetter: [{ type: Input, args: ['max'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-date-time.directive.js","sourceRoot":"","sources":["../../../../../projects/kit/components/input-date-time/input-date-time.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {computed, Directive, effect, inject, Input, signal} from '@angular/core';\nimport {toSignal} from '@angular/core/rxjs-interop';\nimport {MaskitoDirective} from '@maskito/angular';\nimport type {MaskitoOptions} from '@maskito/core';\nimport type {MaskitoDateTimeParams, MaskitoTimeMode} from '@maskito/kit';\nimport {\n    maskitoDateTimeOptionsGenerator,\n    maskitoSelectionChangeHandler,\n} from '@maskito/kit';\nimport {tuiAsControl, tuiValueTransformerFrom} from '@taiga-ui/cdk/classes';\nimport {\n    DATE_FILLER_LENGTH,\n    MILLISECONDS_IN_DAY,\n    TuiDay,\n    TuiTime,\n} from '@taiga-ui/cdk/date-time';\nimport {tuiClamp, tuiSum} from '@taiga-ui/cdk/utils/math';\nimport {tuiDirectiveBinding} from '@taiga-ui/cdk/utils/miscellaneous';\nimport type {TuiCalendar} from '@taiga-ui/core/components/calendar';\nimport {tuiAsOptionContent} from '@taiga-ui/core/components/data-list';\nimport type {TuiTextfieldAccessor} from '@taiga-ui/core/components/textfield';\nimport {\n    tuiAsTextfieldAccessor,\n    TuiWithTextfield,\n} from '@taiga-ui/core/components/textfield';\nimport {TuiDropdownAuto} from '@taiga-ui/core/directives/dropdown';\nimport {TuiItemsHandlersValidator} from '@taiga-ui/core/directives/items-handlers';\nimport {\n    TUI_DATE_ADAPTER,\n    TuiInputDateBase,\n    tuiWithDateFiller,\n} from '@taiga-ui/kit/components/input-date';\nimport {TuiSelectOption} from '@taiga-ui/kit/components/select';\nimport {TUI_TIME_TEXTS} from '@taiga-ui/kit/tokens';\nimport {tuiMaskito} from '@taiga-ui/kit/utils';\nimport {noop} from 'rxjs';\n\nimport {TUI_INPUT_DATE_TIME_OPTIONS} from './input-date-time.options';\n\nconst MIN_TIME = new TuiTime(0, 0);\nconst MAX_TIME = TuiTime.fromAbsoluteMilliseconds(MILLISECONDS_IN_DAY - 1);\n\n@Directive({\n    standalone: true,\n    selector: 'input[tuiInputDateTime]',\n    providers: [\n        tuiAsOptionContent(TuiSelectOption),\n        tuiAsControl(TuiInputDateTimeDirective),\n        tuiAsTextfieldAccessor(TuiInputDateTimeDirective),\n        tuiValueTransformerFrom(TUI_INPUT_DATE_TIME_OPTIONS),\n    ],\n    hostDirectives: [\n        MaskitoDirective,\n        TuiDropdownAuto,\n        TuiItemsHandlersValidator,\n        TuiWithTextfield,\n    ],\n})\nexport class TuiInputDateTimeDirective\n    extends TuiInputDateBase<readonly [TuiDay, TuiTime | null]>\n    implements TuiTextfieldAccessor<readonly [TuiDay, TuiTime | null]>\n{\n    private readonly timeFillers = toSignal(inject(TUI_TIME_TEXTS));\n\n    protected override readonly options = inject(TUI_INPUT_DATE_TIME_OPTIONS);\n\n    protected override readonly filler = tuiWithDateFiller((date) => {\n        const time = this.timeFillers()?.[this.timeMode()] ?? '';\n\n        return `${date}${this.options.dateTimeSeparator}${time}`;\n    });\n\n    protected override valueEffect = effect(noop);\n\n    protected readonly identity = this.handlers.identityMatcher.set(\n        (a, b) => tuiSum(...a.map(Number)) === tuiSum(...b.map(Number)),\n    );\n\n    protected readonly disabledItemHandler = tuiDirectiveBinding(\n        TuiItemsHandlersValidator,\n        'disabledItemHandler',\n        (value: readonly [TuiDay, TuiTime | null] | null) =>\n            Boolean(value && this.handlers.disabledItemHandler()(value)),\n    );\n\n    protected readonly mask = tuiMaskito(\n        computed(() =>\n            this.computeMask({\n                dateMode: TUI_DATE_ADAPTER[this.format().mode],\n                timeMode: this.timeMode(),\n                min: this.toNativeDate([this.min(), this.minTime()]),\n                max: this.toNativeDate([this.max(), this.maxTime()]),\n                dateSeparator: this.format().separator,\n                dateTimeSeparator: this.options.dateTimeSeparator,\n            }),\n        ),\n    );\n\n    public readonly timeMode = signal(this.options.timeMode);\n    public readonly minTime = signal(MIN_TIME);\n    public readonly maxTime = signal(MAX_TIME);\n\n    // TODO(v5): use signal inputs\n    @Input('timeMode')\n    public set timeModeSetter(x: MaskitoTimeMode) {\n        this.timeMode.set(x);\n    }\n\n    @Input('min')\n    public override set minSetter(\n        min: TuiDay | readonly [TuiDay, TuiTime | null] | null,\n    ) {\n        const [date, time] = Array.isArray(min) ? min : [min, null];\n\n        this.min.set(date || this.options.min);\n        this.minTime.set(time ?? MIN_TIME);\n    }\n\n    @Input('max')\n    public override set maxSetter(\n        max: TuiDay | readonly [TuiDay, TuiTime | null] | null,\n    ) {\n        const [date, time] = Array.isArray(max) ? max : [max, null];\n\n        this.max.set(date || this.options.max);\n        this.maxTime.set(time ?? MAX_TIME);\n    }\n\n    public setValue(value: readonly [TuiDay, TuiTime | null] | null): void {\n        this.onChange(value);\n        this.textfield.value.set(this.stringify(value));\n    }\n\n    public override setDate(newDate: TuiDay): void {\n        const [date, time] = this.clampTime([newDate, this.value()?.[1] ?? null]);\n\n        this.setValue([date, time]);\n        this.open.set(false);\n        setTimeout(\n            (caretIndex = DATE_FILLER_LENGTH + this.options.dateTimeSeparator.length) =>\n                this.el.setSelectionRange(caretIndex, caretIndex),\n        );\n    }\n\n    public override writeValue(value: [TuiDay, TuiTime | null] | null): void {\n        super.writeValue(value);\n        this.textfield.value.set(this.stringify(this.value()));\n    }\n\n    protected override processCalendar(calendar: TuiCalendar): void {\n        super.processCalendar(calendar);\n        calendar.disabledItemHandler = (day: TuiDay) =>\n            this.handlers.disabledItemHandler()([day, null]);\n    }\n\n    protected override onValueChange(value: string): void {\n        this.textfield.value.set(value);\n        this.control?.control?.updateValueAndValidity({emitEvent: false});\n\n        const [date = '', time = ''] = value.split(this.options.dateTimeSeparator);\n        const parsedDate =\n            date.length >= DATE_FILLER_LENGTH\n                ? TuiDay.normalizeParse(date, this.format().mode)\n                : null;\n        const parsedTime =\n            time.length === this.timeMode().length ? TuiTime.fromString(time) : null;\n\n        if (!parsedDate || (time && !parsedTime)) {\n            return this.onChange(null);\n        }\n\n        const [prevDate, prevTime = null] = this.value() ?? [];\n\n        if (!prevDate?.daySame(parsedDate) || Number(parsedTime) !== Number(prevTime)) {\n            this.onChange([parsedDate, parsedTime]);\n        }\n    }\n\n    protected override stringify(\n        value: readonly [TuiDay, TuiTime | null] | null,\n    ): string {\n        const [date, time] = value ?? [];\n\n        const dateString =\n            date?.toString(this.format().mode, this.format().separator) ?? '';\n        const timeString = time?.toString(this.timeMode());\n\n        return timeString\n            ? `${dateString}${this.options.dateTimeSeparator}${timeString}`\n            : dateString;\n    }\n\n    private clampTime([date, time]: [TuiDay, TuiTime | null]): [TuiDay, TuiTime | null] {\n        const min = date.daySame(this.min())\n            ? this.minTime().toAbsoluteMilliseconds()\n            : -Infinity;\n        const max = date.daySame(this.max())\n            ? this.maxTime().toAbsoluteMilliseconds()\n            : Infinity;\n\n        return [\n            date,\n            time &&\n                TuiTime.fromAbsoluteMilliseconds(\n                    tuiClamp(time.toAbsoluteMilliseconds(), min, max),\n                ),\n        ];\n    }\n\n    private computeMask(\n        params: Omit<Required<MaskitoDateTimeParams>, 'timeStep'>,\n    ): MaskitoOptions {\n        const options = maskitoDateTimeOptionsGenerator(params);\n        const {timeMode, dateMode, dateTimeSeparator} = params;\n        const inputModeSwitchPlugin = maskitoSelectionChangeHandler((element) => {\n            element.inputMode =\n                element.selectionStart! >=\n                dateMode.length + dateTimeSeparator.length + timeMode.indexOf(' AA')\n                    ? 'text'\n                    : 'numeric';\n        });\n\n        return {\n            ...options,\n            plugins: options.plugins.concat(\n                timeMode.includes('AA') ? inputModeSwitchPlugin : [],\n            ),\n        };\n    }\n\n    private toNativeDate([{year, month, day}, {hours, minutes, seconds, ms}]: readonly [\n        TuiDay,\n        TuiTime,\n    ]): Date {\n        return new Date(year, month, day, hours, minutes, seconds, ms);\n    }\n}\n"]}