@taiga-ui/layout
Version:
A package with Taiga UI layout components
96 lines • 18.3 kB
JavaScript
import { ChangeDetectionStrategy, Component, EventEmitter, inject, Input, Output, TemplateRef, ViewChild, } from '@angular/core';
import { toSignal } from '@angular/core/rxjs-interop';
import { tuiContainsOrAfter, tuiInjectElement, tuiIsElement, } from '@taiga-ui/cdk/utils/dom';
import { tuiGetClosestFocusable } from '@taiga-ui/cdk/utils/focus';
import { tuiDirectiveBinding } from '@taiga-ui/cdk/utils/miscellaneous';
import { TuiTextfieldComponent, TuiWithTextfield, } from '@taiga-ui/core/components/textfield';
import { TuiIcons } from '@taiga-ui/core/directives/icons';
import { TuiPopupService } from '@taiga-ui/core/directives/popup';
import { TUI_COMMON_ICONS } from '@taiga-ui/core/tokens';
import { tuiCellOptionsProvider } from '@taiga-ui/layout/components/cell';
import { TUI_INPUT_SEARCH } from '@taiga-ui/layout/tokens';
import { PolymorpheusOutlet } from '@taiga-ui/polymorpheus';
import * as i0 from "@angular/core";
import * as i1 from "@taiga-ui/core/components/textfield";
class TuiInputSearch {
constructor() {
this.el = tuiInjectElement();
this.service = inject(TuiPopupService);
this.textfield = inject(TuiTextfieldComponent);
this.i18n = toSignal(inject(TUI_INPUT_SEARCH));
this.parent = this.textfield.el.parentElement;
this.neighbor = this.textfield.el.nextSibling;
this.placeholder = '';
this.icon = tuiDirectiveBinding(TuiIcons, 'iconStart', inject(TUI_COMMON_ICONS).search, {});
this.tuiInputSearchOpen = false;
this.tuiInputSearchOpenChange = new EventEmitter();
}
ngOnChanges() {
if (this.tuiInputSearchOpen) {
this.open();
}
else {
this.close();
}
}
open() {
if (this.ref?.destroyed === false || !this.template) {
return;
}
this.placeholder = this.el.placeholder;
this.parent = this.textfield.el.parentElement;
this.neighbor = this.textfield.el.nextSibling;
this.ref = this.service.addTemplate(this.template);
this.ref.rootNodes[0]?.insertAdjacentElement('afterbegin', this.textfield.el);
this.el.focus({ preventScroll: true });
this.el.placeholder = this.i18n()?.placeholder || this.el.placeholder;
this.tuiInputSearchOpen = true;
this.tuiInputSearchOpenChange.emit(true);
}
close() {
this.el.placeholder = this.placeholder || this.el.placeholder;
this.parent?.insertBefore(this.textfield.el, this.neighbor);
this.ref?.destroy();
this.tuiInputSearchOpen = false;
this.tuiInputSearchOpenChange.emit(false);
}
onArrow() {
tuiGetClosestFocusable({
initial: this.container?.nativeElement || this.el,
root: this.container?.nativeElement || this.el,
})?.focus();
}
onFocus({ target }) {
if (this.container &&
target !== this.el &&
tuiIsElement(target) &&
!tuiContainsOrAfter(this.container.nativeElement, target)) {
this.close();
}
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputSearch, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiInputSearch, isStandalone: true, selector: "input[tuiInputSearch]", inputs: { tuiInputSearch: "tuiInputSearch", tuiInputSearchOpen: "tuiInputSearchOpen" }, outputs: { tuiInputSearchOpenChange: "tuiInputSearchOpenChange" }, host: { attributes: { "ngSkipHydration": "true" }, listeners: { "focus": "open()", "keydown.tab.prevent": "0", "keydown.arrowDown.prevent": "onArrow()" } }, providers: [tuiCellOptionsProvider({ size: 'm' })], viewQueries: [{ propertyName: "template", first: true, predicate: TemplateRef, descendants: true }, { propertyName: "container", first: true, predicate: ["container"], descendants: true }], usesOnChanges: true, hostDirectives: [{ directive: i1.TuiWithTextfield }], ngImport: i0, template: "<ng-template>\n <div\n tuiTheme=\"dark\"\n class=\"t-container\"\n (document:focusin)=\"onFocus($event)\"\n (keydown.esc)=\"close()\"\n (pointerdown.self)=\"close()\"\n >\n <div\n #container\n class=\"t-content\"\n >\n <ng-container *polymorpheusOutlet=\"tuiInputSearch\" />\n </div>\n </div>\n</ng-template>\n", styles: [".t-container{position:fixed;top:0;left:0;bottom:0;right:0;display:grid;gap:.25rem;grid-template:-webkit-min-content / minmax(auto,50rem);grid-template:min-content / minmax(auto,50rem);place-content:start center;padding:.5rem;background:var(--tui-service-backdrop);box-shadow:0 0 0 5rem var(--tui-service-backdrop)}.t-container ::ng-deep>tui-textfield{background:var(--tui-background-base-alt)}.t-content{overflow:hidden}\n"], dependencies: [{ kind: "directive", type: PolymorpheusOutlet, selector: "[polymorpheusOutlet]", inputs: ["polymorpheusOutlet", "polymorpheusOutletContext"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
export { TuiInputSearch };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiInputSearch, decorators: [{
type: Component,
args: [{ standalone: true, selector: 'input[tuiInputSearch]', imports: [PolymorpheusOutlet], changeDetection: ChangeDetectionStrategy.OnPush, providers: [tuiCellOptionsProvider({ size: 'm' })], hostDirectives: [TuiWithTextfield], host: {
ngSkipHydration: 'true',
'(focus)': 'open()',
'(keydown.tab.prevent)': '0',
'(keydown.arrowDown.prevent)': 'onArrow()',
}, template: "<ng-template>\n <div\n tuiTheme=\"dark\"\n class=\"t-container\"\n (document:focusin)=\"onFocus($event)\"\n (keydown.esc)=\"close()\"\n (pointerdown.self)=\"close()\"\n >\n <div\n #container\n class=\"t-content\"\n >\n <ng-container *polymorpheusOutlet=\"tuiInputSearch\" />\n </div>\n </div>\n</ng-template>\n", styles: [".t-container{position:fixed;top:0;left:0;bottom:0;right:0;display:grid;gap:.25rem;grid-template:-webkit-min-content / minmax(auto,50rem);grid-template:min-content / minmax(auto,50rem);place-content:start center;padding:.5rem;background:var(--tui-service-backdrop);box-shadow:0 0 0 5rem var(--tui-service-backdrop)}.t-container ::ng-deep>tui-textfield{background:var(--tui-background-base-alt)}.t-content{overflow:hidden}\n"] }]
}], propDecorators: { template: [{
type: ViewChild,
args: [TemplateRef]
}], container: [{
type: ViewChild,
args: ['container']
}], tuiInputSearch: [{
type: Input
}], tuiInputSearchOpen: [{
type: Input
}], tuiInputSearchOpenChange: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-search.component.js","sourceRoot":"","sources":["../../../../../projects/layout/components/input-search/input-search.component.ts","../../../../../projects/layout/components/input-search/input-search.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    Component,\n    type ElementRef,\n    type EmbeddedViewRef,\n    EventEmitter,\n    inject,\n    Input,\n    type OnChanges,\n    Output,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {toSignal} from '@angular/core/rxjs-interop';\nimport {\n    tuiContainsOrAfter,\n    tuiInjectElement,\n    tuiIsElement,\n} from '@taiga-ui/cdk/utils/dom';\nimport {tuiGetClosestFocusable} from '@taiga-ui/cdk/utils/focus';\nimport {tuiDirectiveBinding} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {\n    TuiTextfieldComponent,\n    TuiWithTextfield,\n} from '@taiga-ui/core/components/textfield';\nimport {TuiIcons} from '@taiga-ui/core/directives/icons';\nimport {TuiPopupService} from '@taiga-ui/core/directives/popup';\nimport {TUI_COMMON_ICONS} from '@taiga-ui/core/tokens';\nimport {tuiCellOptionsProvider} from '@taiga-ui/layout/components/cell';\nimport {TUI_INPUT_SEARCH} from '@taiga-ui/layout/tokens';\nimport {type PolymorpheusContent, PolymorpheusOutlet} from '@taiga-ui/polymorpheus';\n\n@Component({\n    standalone: true,\n    selector: 'input[tuiInputSearch]',\n    imports: [PolymorpheusOutlet],\n    templateUrl: './input-search.component.html',\n    styleUrls: ['./input-search.component.less'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    providers: [tuiCellOptionsProvider({size: 'm'})],\n    hostDirectives: [TuiWithTextfield],\n    host: {\n        ngSkipHydration: 'true',\n        '(focus)': 'open()',\n        '(keydown.tab.prevent)': '0',\n        '(keydown.arrowDown.prevent)': 'onArrow()',\n    },\n})\nexport class TuiInputSearch implements OnChanges {\n    @ViewChild(TemplateRef)\n    private readonly template?: TemplateRef<any>;\n\n    @ViewChild('container')\n    private readonly container?: ElementRef<HTMLElement>;\n\n    private readonly el = tuiInjectElement<HTMLInputElement>();\n    private readonly service = inject(TuiPopupService);\n    private readonly textfield = inject(TuiTextfieldComponent);\n    private readonly i18n = toSignal(inject(TUI_INPUT_SEARCH));\n    private parent = this.textfield.el.parentElement;\n    private neighbor = this.textfield.el.nextSibling;\n    private placeholder = '';\n    private ref?: EmbeddedViewRef<unknown>;\n\n    protected readonly icon = tuiDirectiveBinding(\n        TuiIcons,\n        'iconStart',\n        inject(TUI_COMMON_ICONS).search,\n        {},\n    );\n\n    @Input()\n    public tuiInputSearch: PolymorpheusContent;\n\n    @Input()\n    public tuiInputSearchOpen = false;\n\n    @Output()\n    public readonly tuiInputSearchOpenChange = new EventEmitter<boolean>();\n\n    public ngOnChanges(): void {\n        if (this.tuiInputSearchOpen) {\n            this.open();\n        } else {\n            this.close();\n        }\n    }\n\n    public open(): void {\n        if (this.ref?.destroyed === false || !this.template) {\n            return;\n        }\n\n        this.placeholder = this.el.placeholder;\n        this.parent = this.textfield.el.parentElement;\n        this.neighbor = this.textfield.el.nextSibling;\n        this.ref = this.service.addTemplate(this.template);\n        this.ref.rootNodes[0]?.insertAdjacentElement('afterbegin', this.textfield.el);\n        this.el.focus({preventScroll: true});\n        this.el.placeholder = this.i18n()?.placeholder || this.el.placeholder;\n        this.tuiInputSearchOpen = true;\n        this.tuiInputSearchOpenChange.emit(true);\n    }\n\n    public close(): void {\n        this.el.placeholder = this.placeholder || this.el.placeholder;\n        this.parent?.insertBefore(this.textfield.el, this.neighbor);\n        this.ref?.destroy();\n        this.tuiInputSearchOpen = false;\n        this.tuiInputSearchOpenChange.emit(false);\n    }\n\n    protected onArrow(): void {\n        tuiGetClosestFocusable({\n            initial: this.container?.nativeElement || this.el,\n            root: this.container?.nativeElement || this.el,\n        })?.focus();\n    }\n\n    protected onFocus({target}: Event): void {\n        if (\n            this.container &&\n            target !== this.el &&\n            tuiIsElement(target) &&\n            !tuiContainsOrAfter(this.container.nativeElement, target)\n        ) {\n            this.close();\n        }\n    }\n}\n","<ng-template>\n    <div\n        tuiTheme=\"dark\"\n        class=\"t-container\"\n        (document:focusin)=\"onFocus($event)\"\n        (keydown.esc)=\"close()\"\n        (pointerdown.self)=\"close()\"\n    >\n        <div\n            #container\n            class=\"t-content\"\n        >\n            <ng-container *polymorpheusOutlet=\"tuiInputSearch\" />\n        </div>\n    </div>\n</ng-template>\n"]}