UNPKG

@taiga-ui/kit

Version:

Taiga UI Angular main components kit

76 lines 16.4 kB
import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common'; import { ChangeDetectionStrategy, Component, computed, effect, inject, Input, signal, ViewChildren, } from '@angular/core'; import { tuiAsControl, TuiControl } from '@taiga-ui/cdk/classes'; import { EMPTY_QUERY, TUI_ALLOW_SIGNAL_WRITES } from '@taiga-ui/cdk/constants'; import { tuiIsPresent } from '@taiga-ui/cdk/utils/miscellaneous'; import { tuiAsTextfieldAccessor, TuiTextfieldDirective, TuiWithTextfield, } from '@taiga-ui/core/components/textfield'; import { TUI_ITEMS_HANDLERS, } from '@taiga-ui/core/directives/items-handlers'; import { tuiIsFlat } from '@taiga-ui/kit/utils'; import * as i0 from "@angular/core"; import * as i1 from "@taiga-ui/core/components/textfield"; class TuiNativeSelect extends TuiControl { constructor() { super(...arguments); this.textfield = inject(TuiTextfieldDirective); this.options = signal(EMPTY_QUERY); this.isFlat = tuiIsFlat; this.placeholder = signal(''); this.itemsHandlers = inject(TUI_ITEMS_HANDLERS); this.stringified = computed((value = this.value()) => tuiIsPresent(value) ? this.itemsHandlers.stringify()(value) : ''); this.isSelected = computed((value = this.value()) => (x) => tuiIsPresent(value) && this.itemsHandlers.identityMatcher()(x, value)); this.valueEffect = effect(() => { /** * Wait until all `<option>`-s are inside DOM. * Otherwise * ``` * document.querySelector('select').value = 'even upcoming valid value'; * // same as * document.querySelector('select').value = ''; * ``` * (it breaks `tuiValue` utility logic) */ if (this.options().length) { this.textfield.value.set(this.stringified()); } }, TUI_ALLOW_SIGNAL_WRITES); this.items = []; this.labels = []; } // TODO(v5): use signal inputs set placeholderSetter(x) { this.placeholder.set(x); } setValue(value) { this.onChange(value); } set optionsSetter(options) { this.options.set(options); } selectOption(index) { const items = (this.items?.flat() ?? []); const placeholderOffset = this.stringified() ? 0 : 1; this.onChange(items[index - placeholderOffset] ?? null); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiNativeSelect, deps: null, target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiNativeSelect, isStandalone: true, selector: "select[tuiSelect]", inputs: { items: "items", labels: "labels", placeholderSetter: ["placeholder", "placeholderSetter"] }, host: { listeners: { "change": "selectOption($event.target.options.selectedIndex)" }, properties: { "attr.aria-invalid": "invalid()", "disabled": "!interactive()" } }, providers: [tuiAsTextfieldAccessor(TuiNativeSelect), tuiAsControl(TuiNativeSelect)], viewQueries: [{ propertyName: "optionsSetter", predicate: ["optionRef"], descendants: true }], usesInheritance: true, hostDirectives: [{ directive: i1.TuiWithTextfield }], ngImport: i0, template: "<option\n *ngIf=\"!stringified()\"\n disabled\n selected\n value=\"\"\n>\n {{ placeholder() }}\n</option>\n\n<ng-container *ngIf=\"items && !isFlat(items); else flatItems\">\n <optgroup\n *ngFor=\"let group of labels; let index = index\"\n [label]=\"labels[index]\"\n >\n <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items[index]}\" />\n </optgroup>\n</ng-container>\n\n<ng-template #flatItems>\n <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items}\" />\n</ng-template>\n\n<ng-template\n #options\n let-items\n>\n <option\n *ngFor=\"let option of items\"\n #optionRef\n [disabled]=\"itemsHandlers.disabledItemHandler()(option)\"\n [selected]=\"isSelected()(option)\"\n [value]=\"itemsHandlers.stringify()(option)\"\n >\n {{ itemsHandlers.stringify()(option) }}\n </option>\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } export { TuiNativeSelect }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiNativeSelect, decorators: [{ type: Component, args: [{ standalone: true, selector: 'select[tuiSelect]', imports: [NgForOf, NgIf, NgTemplateOutlet], changeDetection: ChangeDetectionStrategy.OnPush, providers: [tuiAsTextfieldAccessor(TuiNativeSelect), tuiAsControl(TuiNativeSelect)], hostDirectives: [TuiWithTextfield], host: { '[attr.aria-invalid]': 'invalid()', '[disabled]': '!interactive()', '(change)': 'selectOption($event.target.options.selectedIndex)', }, template: "<option\n *ngIf=\"!stringified()\"\n disabled\n selected\n value=\"\"\n>\n {{ placeholder() }}\n</option>\n\n<ng-container *ngIf=\"items && !isFlat(items); else flatItems\">\n <optgroup\n *ngFor=\"let group of labels; let index = index\"\n [label]=\"labels[index]\"\n >\n <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items[index]}\" />\n </optgroup>\n</ng-container>\n\n<ng-template #flatItems>\n <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items}\" />\n</ng-template>\n\n<ng-template\n #options\n let-items\n>\n <option\n *ngFor=\"let option of items\"\n #optionRef\n [disabled]=\"itemsHandlers.disabledItemHandler()(option)\"\n [selected]=\"isSelected()(option)\"\n [value]=\"itemsHandlers.stringify()(option)\"\n >\n {{ itemsHandlers.stringify()(option) }}\n </option>\n</ng-template>\n" }] }], propDecorators: { items: [{ type: Input }], labels: [{ type: Input }], placeholderSetter: [{ type: Input, args: ['placeholder'] }], optionsSetter: [{ type: ViewChildren, args: ['optionRef'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"native-select.component.js","sourceRoot":"","sources":["../../../../../../projects/kit/components/select/native-select/native-select.component.ts","../../../../../../projects/kit/components/select/native-select/native-select.template.html"],"names":[],"mappings":"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","sourcesContent":["import {NgForOf, NgIf, NgTemplateOutlet} from '@angular/common';\nimport {\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    effect,\n    type ElementRef,\n    inject,\n    Input,\n    type QueryList,\n    signal,\n    ViewChildren,\n} from '@angular/core';\nimport {tuiAsControl, TuiControl} from '@taiga-ui/cdk/classes';\nimport {EMPTY_QUERY, TUI_ALLOW_SIGNAL_WRITES} from '@taiga-ui/cdk/constants';\nimport {tuiIsPresent} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {\n    tuiAsTextfieldAccessor,\n    type TuiTextfieldAccessor,\n    TuiTextfieldDirective,\n    TuiWithTextfield,\n} from '@taiga-ui/core/components/textfield';\nimport {\n    TUI_ITEMS_HANDLERS,\n    type TuiItemsHandlers,\n} from '@taiga-ui/core/directives/items-handlers';\nimport {tuiIsFlat} from '@taiga-ui/kit/utils';\n\n@Component({\n    standalone: true,\n    selector: 'select[tuiSelect]',\n    imports: [NgForOf, NgIf, NgTemplateOutlet],\n    templateUrl: './native-select.template.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    providers: [tuiAsTextfieldAccessor(TuiNativeSelect), tuiAsControl(TuiNativeSelect)],\n    hostDirectives: [TuiWithTextfield],\n    host: {\n        '[attr.aria-invalid]': 'invalid()',\n        '[disabled]': '!interactive()',\n        '(change)': 'selectOption($event.target.options.selectedIndex)',\n    },\n})\nexport class TuiNativeSelect<T>\n    extends TuiControl<T | null>\n    implements TuiTextfieldAccessor<T>\n{\n    private readonly textfield = inject(TuiTextfieldDirective);\n    private readonly options =\n        signal<QueryList<ElementRef<HTMLOptionElement>>>(EMPTY_QUERY);\n\n    protected readonly isFlat = tuiIsFlat;\n    protected readonly placeholder = signal('');\n    protected readonly itemsHandlers: TuiItemsHandlers<T> = inject(TUI_ITEMS_HANDLERS);\n\n    protected readonly stringified = computed((value = this.value()) =>\n        tuiIsPresent(value) ? this.itemsHandlers.stringify()(value) : '',\n    );\n\n    protected readonly isSelected = computed(\n        (value = this.value()) =>\n            (x: T) =>\n                tuiIsPresent(value) && this.itemsHandlers.identityMatcher()(x, value),\n    );\n\n    protected readonly valueEffect = effect(() => {\n        /**\n         * Wait until all `<option>`-s are inside DOM.\n         * Otherwise\n         * ```\n         * document.querySelector('select').value = 'even upcoming valid value';\n         * // same as\n         * document.querySelector('select').value = '';\n         * ```\n         * (it breaks `tuiValue` utility logic)\n         */\n        if (this.options().length) {\n            this.textfield.value.set(this.stringified());\n        }\n    }, TUI_ALLOW_SIGNAL_WRITES);\n\n    @Input()\n    public items: ReadonlyArray<readonly T[]> | readonly T[] | null = [];\n\n    @Input()\n    public labels: readonly string[] = [];\n\n    // TODO(v5): use signal inputs\n    @Input('placeholder')\n    public set placeholderSetter(x: string) {\n        this.placeholder.set(x);\n    }\n\n    public setValue(value: T | null): void {\n        this.onChange(value);\n    }\n\n    @ViewChildren('optionRef')\n    protected set optionsSetter(options: QueryList<ElementRef<HTMLOptionElement>>) {\n        this.options.set(options);\n    }\n\n    protected selectOption(index: number): void {\n        const items = (this.items?.flat() ?? []) as T[];\n        const placeholderOffset = this.stringified() ? 0 : 1;\n\n        this.onChange(items[index - placeholderOffset] ?? null);\n    }\n}\n","<option\n    *ngIf=\"!stringified()\"\n    disabled\n    selected\n    value=\"\"\n>\n    {{ placeholder() }}\n</option>\n\n<ng-container *ngIf=\"items && !isFlat(items); else flatItems\">\n    <optgroup\n        *ngFor=\"let group of labels; let index = index\"\n        [label]=\"labels[index]\"\n    >\n        <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items[index]}\" />\n    </optgroup>\n</ng-container>\n\n<ng-template #flatItems>\n    <ng-container *ngTemplateOutlet=\"options; context: {$implicit: items}\" />\n</ng-template>\n\n<ng-template\n    #options\n    let-items\n>\n    <option\n        *ngFor=\"let option of items\"\n        #optionRef\n        [disabled]=\"itemsHandlers.disabledItemHandler()(option)\"\n        [selected]=\"isSelected()(option)\"\n        [value]=\"itemsHandlers.stringify()(option)\"\n    >\n        {{ itemsHandlers.stringify()(option) }}\n    </option>\n</ng-template>\n"]}