UNPKG

@taiga-ui/kit

Version:

Taiga UI Angular main components kit

126 lines 19.1 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 } 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"; class TuiInputChipDirective extends TuiControl { constructor() { super(...arguments); this.handlers = inject(TUI_ITEMS_HANDLERS); this.options = inject(TUI_INPUT_CHIP_OPTIONS); this.mobile = inject(TUI_IS_MOBILE); this.textfield = inject(TuiTextfieldMultiComponent); this.open = tuiDropdownOpen(); this.dropdown = inject(TuiDropdownDirective); 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() { const value = this.textfield.value().trim(); const items = this.separator ? value.split(this.separator) : [value]; const valid = items.filter((item) => item && !this.handlers.disabledItemHandler()(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); this.textfield.value.set(value); this.onEnter(); } onBackspace(key) { // (keydown.backspace) doesn't emit event on empty input in ios safari if (key === 'Backspace' && !this.textfield.value() && this.interactive() && (this.mobile || !this.textfield.item)) { this.onChange(this.value().slice(0, -1)); } } 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, }); }); } 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]", 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()" } }, 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'], }, ], 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 }] } }); //# 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} from '@taiga-ui/cdk/utils/miscellaneous';\nimport type {TuiTextfieldAccessor} from '@taiga-ui/core/components/textfield';\nimport {\n    tuiAsTextfieldAccessor,\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 type {TuiItemsHandlers} from '@taiga-ui/core/directives/items-handlers';\nimport {TUI_ITEMS_HANDLERS} from '@taiga-ui/core/directives/items-handlers';\nimport {filter} from 'rxjs';\n\nimport {TUI_INPUT_CHIP_OPTIONS} from './input-chip.options';\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    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 TuiInputChipDirective<T>\n    extends TuiControl<T[]>\n    implements TuiTextfieldAccessor<T[]>\n{\n    private readonly handlers: TuiItemsHandlers<T> = inject(TUI_ITEMS_HANDLERS);\n    private readonly options = inject(TUI_INPUT_CHIP_OPTIONS);\n    private readonly mobile = inject(TUI_IS_MOBILE);\n    private readonly textfield = inject(TuiTextfieldMultiComponent);\n    private readonly open = tuiDropdownOpen();\n    private readonly dropdown = inject(TuiDropdownDirective);\n\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(): void {\n        const value = this.textfield.value().trim();\n        const items: any[] = this.separator ? value.split(this.separator) : [value];\n        const valid = items.filter(\n            (item) => item && !this.handlers.disabledItemHandler()(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        this.textfield.value.set(value);\n        this.onEnter();\n    }\n\n    protected onBackspace(key: string): void {\n        // (keydown.backspace) doesn't emit event on empty input in ios safari\n        if (\n            key === 'Backspace' &&\n            !this.textfield.value() &&\n            this.interactive() &&\n            (this.mobile || !this.textfield.item)\n        ) {\n            this.onChange(this.value().slice(0, -1));\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        });\n    }\n}\n"]}