UNPKG

@taiga-ui/kit

Version:

Taiga UI Angular main components kit

149 lines 21.9 kB
import { Directive, inject, Input } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { tuiAsControl, TuiControl } from '@taiga-ui/cdk/classes'; import { TuiActiveZone } from '@taiga-ui/cdk/directives/active-zone'; import { TuiNativeValidator } from '@taiga-ui/cdk/directives/native-validator'; import { TUI_IS_MOBILE, tuiFallbackValueProvider } from '@taiga-ui/cdk/tokens'; import { tuiGetClipboardDataText, tuiInjectElement } from '@taiga-ui/cdk/utils/dom'; import { tuiDirectiveBinding, tuiSanitizeText } from '@taiga-ui/cdk/utils/miscellaneous'; import { tuiAsTextfieldAccessor, TuiTextfieldBase, TuiTextfieldMultiComponent, } from '@taiga-ui/core/components/textfield'; import { TuiDropdownDirective, TuiDropdownOpen, tuiDropdownOpen, } from '@taiga-ui/core/directives/dropdown'; import { TUI_ITEMS_HANDLERS, } from '@taiga-ui/core/directives/items-handlers'; import { filter } from 'rxjs'; import { TUI_INPUT_CHIP_OPTIONS } from './input-chip.options'; import * as i0 from "@angular/core"; import * as i1 from "@taiga-ui/cdk/directives/native-validator"; import * as i2 from "@taiga-ui/core/components/textfield"; // TODO(v5): remove base component after angular update class TuiInputChipBaseDirective extends TuiControl { constructor() { super(...arguments); this.options = inject(TUI_INPUT_CHIP_OPTIONS); this.mobile = inject(TUI_IS_MOBILE); this.dropdown = inject(TuiDropdownDirective); this.textfield = inject(TuiTextfieldMultiComponent); this.open = tuiDropdownOpen(); this.handlers = inject(TUI_ITEMS_HANDLERS); this.enabled = tuiDirectiveBinding(TuiDropdownOpen, 'tuiDropdownEnabled', this.interactive, {}); this.sub = inject(TuiActiveZone) .tuiActiveZoneChange.pipe(filter((active) => !active && !this.el.matches('select')), takeUntilDestroyed()) .subscribe(() => { this.onEnter(); this.textfield.value.set(''); }); this.separator = this.options.separator; this.unique = this.options.unique; this.el = tuiInjectElement(); } setValue(value) { this.textfield.value.set(''); this.onChange(this.unique ? Array.from(new Set(value.reverse())).reverse() : value); } onEnter(rawValue = this.textfield.value()) { const value = rawValue.trim(); const items = this.separator ? value.split(this.separator) : [value]; const valid = items .map((item) => tuiSanitizeText(item)) .filter((item) => item && !this.handlers.disabledItemHandler()(item) && this.handlers.stringify()(item)); if (!value || !valid.length) { return; } this.setValue([...this.value(), ...valid]); this.scrollTo(); } onInput() { this.open.set(!!this.dropdown.content); if (this.separator && this.textfield.value().match(this.separator)) { this.onEnter(); } else { this.scrollTo(); } } onPaste(event) { const value = 'dataTransfer' in event ? event.dataTransfer?.getData('text/plain') || '' : tuiGetClipboardDataText(event); if (this.textfield.input?.nativeElement) { this.textfield.input.nativeElement.value = value; } this.onEnter(value); } onBackspace(key) { // (keydown.backspace) doesn't emit event on empty input in ios safari if (key === 'Backspace' && !this.textfield.value() && this.interactive()) { if (this.mobile || !this.textfield.item) { this.onChange(this.value().slice(0, -1)); } else { this.el.dispatchEvent(new KeyboardEvent('keydown', { key: 'ArrowLeft', bubbles: true, })); } } } scrollTo() { const sign = this.textfield.el.matches('[dir="rtl"] :scope') ? -1 : 1; // Allow change detection to run and add new tag to DOM setTimeout(() => { this.textfield.el.scrollTo({ left: sign * Number.MAX_SAFE_INTEGER, top: Number.MAX_SAFE_INTEGER, }); }, 100); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputChipBaseDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: TuiInputChipBaseDirective, isStandalone: true, inputs: { separator: "separator", unique: "unique" }, host: { attributes: { "enterkeyhint": "enter" }, listeners: { "keydown.enter.prevent": "onEnter()", "keydown.zoneless": "onBackspace($event.key)", "input": "onInput()", "paste.prevent": "onPaste($event)", "drop.prevent": "onPaste($event)" }, properties: { "disabled": "disabled()" } }, usesInheritance: true, ngImport: i0 }); } } export { TuiInputChipBaseDirective }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputChipBaseDirective, decorators: [{ type: Directive, args: [{ standalone: true, host: { enterkeyhint: 'enter', '[disabled]': 'disabled()', '(keydown.enter.prevent)': 'onEnter()', '(keydown.zoneless)': 'onBackspace($event.key)', '(input)': 'onInput()', '(paste.prevent)': 'onPaste($event)', '(drop.prevent)': 'onPaste($event)', }, }] }], propDecorators: { separator: [{ type: Input }], unique: [{ type: Input }] } }); class TuiInputChipDirective extends TuiInputChipBaseDirective { static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputChipDirective, deps: null, target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: TuiInputChipDirective, isStandalone: true, selector: "input[tuiInputChip]", providers: [ tuiAsControl(TuiInputChipDirective), tuiFallbackValueProvider([]), tuiAsTextfieldAccessor(TuiInputChipDirective), ], usesInheritance: true, hostDirectives: [{ directive: i1.TuiNativeValidator }, { directive: i2.TuiTextfieldBase, inputs: ["invalid", "invalid", "focused", "focused", "readOnly", "readOnly", "state", "state"] }], ngImport: i0 }); } } export { TuiInputChipDirective }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputChipDirective, decorators: [{ type: Directive, args: [{ standalone: true, selector: 'input[tuiInputChip]', providers: [ tuiAsControl(TuiInputChipDirective), tuiFallbackValueProvider([]), tuiAsTextfieldAccessor(TuiInputChipDirective), ], hostDirectives: [ TuiNativeValidator, { directive: TuiTextfieldBase, inputs: ['invalid', 'focused', 'readOnly', 'state'], }, ], }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-chip.directive.js","sourceRoot":"","sources":["../../../../../projects/kit/components/input-chip/input-chip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, Input} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {tuiAsControl, TuiControl} from '@taiga-ui/cdk/classes';\nimport {TuiActiveZone} from '@taiga-ui/cdk/directives/active-zone';\nimport {TuiNativeValidator} from '@taiga-ui/cdk/directives/native-validator';\nimport {TUI_IS_MOBILE, tuiFallbackValueProvider} from '@taiga-ui/cdk/tokens';\nimport {tuiGetClipboardDataText, tuiInjectElement} from '@taiga-ui/cdk/utils/dom';\nimport {tuiDirectiveBinding, tuiSanitizeText} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {\n    tuiAsTextfieldAccessor,\n    type TuiTextfieldAccessor,\n    TuiTextfieldBase,\n    TuiTextfieldMultiComponent,\n} from '@taiga-ui/core/components/textfield';\nimport {\n    TuiDropdownDirective,\n    TuiDropdownOpen,\n    tuiDropdownOpen,\n} from '@taiga-ui/core/directives/dropdown';\nimport {\n    TUI_ITEMS_HANDLERS,\n    type TuiItemsHandlers,\n} from '@taiga-ui/core/directives/items-handlers';\nimport {filter} from 'rxjs';\n\nimport {TUI_INPUT_CHIP_OPTIONS} from './input-chip.options';\n\n// TODO(v5): remove base component after angular update\n@Directive({\n    standalone: true,\n    host: {\n        enterkeyhint: 'enter',\n        '[disabled]': 'disabled()',\n        '(keydown.enter.prevent)': 'onEnter()',\n        '(keydown.zoneless)': 'onBackspace($event.key)',\n        '(input)': 'onInput()',\n        '(paste.prevent)': 'onPaste($event)',\n        '(drop.prevent)': 'onPaste($event)',\n    },\n})\nexport class TuiInputChipBaseDirective<T>\n    extends TuiControl<T[]>\n    implements TuiTextfieldAccessor<T[]>\n{\n    private readonly options = inject(TUI_INPUT_CHIP_OPTIONS);\n    private readonly mobile = inject(TUI_IS_MOBILE);\n    private readonly dropdown = inject(TuiDropdownDirective);\n\n    protected readonly textfield = inject(TuiTextfieldMultiComponent);\n    protected readonly open = tuiDropdownOpen();\n    protected readonly handlers: TuiItemsHandlers<T> = inject(TUI_ITEMS_HANDLERS);\n    protected readonly enabled = tuiDirectiveBinding(\n        TuiDropdownOpen,\n        'tuiDropdownEnabled',\n        this.interactive,\n        {},\n    );\n\n    protected readonly sub = inject(TuiActiveZone)\n        .tuiActiveZoneChange.pipe(\n            filter((active) => !active && !this.el.matches('select')),\n            takeUntilDestroyed(),\n        )\n        .subscribe(() => {\n            this.onEnter();\n            this.textfield.value.set('');\n        });\n\n    @Input()\n    public separator = this.options.separator;\n\n    @Input()\n    public unique = this.options.unique;\n\n    public readonly el = tuiInjectElement<HTMLInputElement>();\n\n    public setValue(value: T[]): void {\n        this.textfield.value.set('');\n        this.onChange(\n            this.unique ? Array.from(new Set(value.reverse())).reverse() : value,\n        );\n    }\n\n    protected onEnter(rawValue = this.textfield.value()): void {\n        const value = rawValue.trim();\n        const items = this.separator ? value.split(this.separator) : [value];\n\n        const valid = items\n            .map((item) => tuiSanitizeText(item) as T)\n            .filter(\n                (item) =>\n                    item &&\n                    !this.handlers.disabledItemHandler()(item) &&\n                    this.handlers.stringify()(item),\n            );\n\n        if (!value || !valid.length) {\n            return;\n        }\n\n        this.setValue([...this.value(), ...valid]);\n        this.scrollTo();\n    }\n\n    protected onInput(): void {\n        this.open.set(!!this.dropdown.content);\n\n        if (this.separator && this.textfield.value().match(this.separator)) {\n            this.onEnter();\n        } else {\n            this.scrollTo();\n        }\n    }\n\n    protected onPaste(event: ClipboardEvent | DragEvent): void {\n        const value =\n            'dataTransfer' in event\n                ? event.dataTransfer?.getData('text/plain') || ''\n                : tuiGetClipboardDataText(event);\n\n        if (this.textfield.input?.nativeElement) {\n            this.textfield.input.nativeElement.value = value;\n        }\n\n        this.onEnter(value);\n    }\n\n    protected onBackspace(key: string): void {\n        // (keydown.backspace) doesn't emit event on empty input in ios safari\n        if (key === 'Backspace' && !this.textfield.value() && this.interactive()) {\n            if (this.mobile || !this.textfield.item) {\n                this.onChange(this.value().slice(0, -1));\n            } else {\n                this.el.dispatchEvent(\n                    new KeyboardEvent('keydown', {\n                        key: 'ArrowLeft',\n                        bubbles: true,\n                    }),\n                );\n            }\n        }\n    }\n\n    protected scrollTo(): void {\n        const sign = this.textfield.el.matches('[dir=\"rtl\"] :scope') ? -1 : 1;\n\n        // Allow change detection to run and add new tag to DOM\n        setTimeout(() => {\n            this.textfield.el.scrollTo({\n                left: sign * Number.MAX_SAFE_INTEGER,\n                top: Number.MAX_SAFE_INTEGER,\n            });\n        }, 100);\n    }\n}\n\n@Directive({\n    standalone: true,\n    selector: 'input[tuiInputChip]',\n    providers: [\n        tuiAsControl(TuiInputChipDirective),\n        tuiFallbackValueProvider([]),\n        tuiAsTextfieldAccessor(TuiInputChipDirective),\n    ],\n    hostDirectives: [\n        TuiNativeValidator,\n        {\n            directive: TuiTextfieldBase,\n            inputs: ['invalid', 'focused', 'readOnly', 'state'],\n        },\n    ],\n})\nexport class TuiInputChipDirective<T> extends TuiInputChipBaseDirective<T> {}\n"]}