UNPKG

@taiga-ui/core

Version:

Core library for creating Angular components and applications using Taiga UI

169 lines 26.1 kB
import { __decorate } from "tslib"; import { computed, ContentChild, Directive, ElementRef, EventEmitter, inject, Input, Output, } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { TuiActiveZone } from '@taiga-ui/cdk/directives/active-zone'; import { TuiObscured } from '@taiga-ui/cdk/directives/obscured'; import { tuiCloseWatcher, tuiIfMap, tuiWatch, tuiZonefull, } from '@taiga-ui/cdk/observables'; import { tuiGetActualTarget, tuiInjectElement, tuiIsElement, tuiIsElementEditable, tuiIsHTMLElement, } from '@taiga-ui/cdk/utils/dom'; import { tuiGetClosestFocusable, tuiIsNativeFocusedIn, tuiIsNativeKeyboardFocusable, } from '@taiga-ui/cdk/utils/focus'; import { tuiAsDriver } from '@taiga-ui/core/classes'; import { tuiIsEditingKey } from '@taiga-ui/core/utils/miscellaneous'; import { shouldCall } from '@taiga-ui/event-plugins'; import { filter, fromEvent, merge } from 'rxjs'; import { TuiDropdownDirective } from './dropdown.directive'; import { TuiDropdownDriver } from './dropdown.driver'; import * as i0 from "@angular/core"; import * as i1 from "@taiga-ui/cdk/directives/obscured"; import * as i2 from "@taiga-ui/cdk/directives/active-zone"; function shouldClose(event) { return ( // @ts-ignore typeof CloseWatcher === 'undefined' && // ?. for auto fill events event.key?.toLowerCase() === 'escape' && this.tuiDropdownEnabled && !!this.tuiDropdownOpen && !this['dropdown']()?.nextElementSibling); } class TuiDropdownOpen { constructor() { this.directive = inject(TuiDropdownDirective); this.el = tuiInjectElement(); this.obscured = inject(TuiObscured); this.activeZone = inject(TuiActiveZone); this.dropdown = computed(() => this.directive.ref()?.location.nativeElement); this.tuiDropdownEnabled = true; this.tuiDropdownOpen = false; this.tuiDropdownOpenChange = new EventEmitter(); // TODO: make it private when all legacy controls will be deleted from @taiga-ui/legacy (5.0) this.driver = inject(TuiDropdownDriver); this.sub = this.driver .pipe(tuiIfMap(() => merge(tuiCloseWatcher(), this.obscured.tuiObscured.pipe(filter(Boolean)), this.activeZone.tuiActiveZoneChange.pipe(filter((a) => !a)), fromEvent(this.el, 'focusin').pipe(filter((event) => !this.host.contains(tuiGetActualTarget(event)) || !this.directive.ref())))), tuiZonefull(), tuiWatch(), takeUntilDestroyed()) .subscribe(() => this.toggle(false)); this.sync = this.driver.pipe(takeUntilDestroyed()).subscribe((open) => { if (open !== this.tuiDropdownOpen) { this.update(open); } }); } ngOnChanges() { this.drive(!!this.tuiDropdownOpen); this.tuiDropdownOpenChange.emit(!!this.tuiDropdownOpen); } toggle(open) { if (this.focused && !open) { this.host.focus({ preventScroll: true }); } this.update(open); } onEsc(event) { event.preventDefault(); this.toggle(false); } onClick(target) { if (!this.editable && this.host.contains(target)) { this.update(!this.tuiDropdownOpen); } } onArrow(event, up) { if (!tuiIsElement(event.target) || !this.host.contains(event.target) || !this.tuiDropdownEnabled || !this.directive._content()) { return; } event.preventDefault(); this.focusDropdown(up); } onKeydown(event) { const target = tuiGetActualTarget(event); if (!event.defaultPrevented && tuiIsEditingKey(event.key) && this.editable && this.focused && tuiIsHTMLElement(target) && !tuiIsElementEditable(target)) { this.host.focus({ preventScroll: true }); } } get host() { const initial = this.dropdownHost?.nativeElement || this.el; const focusable = tuiIsNativeKeyboardFocusable(initial) ? initial : tuiGetClosestFocusable({ initial, root: this.el }); return this.dropdownHost?.nativeElement || focusable || this.el; } get editable() { return tuiIsElementEditable(this.host); } get focused() { return tuiIsNativeFocusedIn(this.host) || tuiIsNativeFocusedIn(this.dropdown()); } update(open) { if (open && !this.tuiDropdownEnabled) { return this.drive(); } this.tuiDropdownOpen = open; this.tuiDropdownOpenChange.emit(open); this.drive(); } drive(open = !!this.tuiDropdownOpen && this.tuiDropdownEnabled) { this.obscured.tuiObscuredEnabled = open; this.driver.next(open); } focusDropdown(previous) { const root = this.dropdown(); if (!root) { this.update(true); return; } const doc = this.el.ownerDocument; const child = root.appendChild(doc.createElement('div')); const initial = previous ? child : root; const focusable = tuiGetClosestFocusable({ initial, previous, root }); child.remove(); focusable?.focus(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiDropdownOpen, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: TuiDropdownOpen, isStandalone: true, selector: "[tuiDropdown][tuiDropdownOpen],[tuiDropdown][tuiDropdownOpenChange]", inputs: { tuiDropdownEnabled: "tuiDropdownEnabled", tuiDropdownOpen: "tuiDropdownOpen" }, outputs: { tuiDropdownOpenChange: "tuiDropdownOpenChange" }, host: { listeners: { "click": "onClick($event.target)", "keydown.arrowDown": "onArrow($event, false)", "keydown.arrowUp": "onArrow($event, true)", "document:keydown.zoneless.capture": "onEsc($event)", "document:keydown.zoneless": "onKeydown($event)", "tuiActiveZoneChange": "0" } }, providers: [TuiDropdownDriver, tuiAsDriver(TuiDropdownDriver)], queries: [{ propertyName: "dropdownHost", first: true, predicate: ["tuiDropdownHost"], descendants: true, read: ElementRef }], usesOnChanges: true, hostDirectives: [{ directive: i1.TuiObscured }, { directive: i2.TuiActiveZone, inputs: ["tuiActiveZoneParent", "tuiActiveZoneParent"], outputs: ["tuiActiveZoneChange", "tuiActiveZoneChange"] }], ngImport: i0 }); } } __decorate([ shouldCall(shouldClose) ], TuiDropdownOpen.prototype, "onEsc", null); export { TuiDropdownOpen }; i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiDropdownOpen, decorators: [{ type: Directive, args: [{ standalone: true, selector: '[tuiDropdown][tuiDropdownOpen],[tuiDropdown][tuiDropdownOpenChange]', providers: [TuiDropdownDriver, tuiAsDriver(TuiDropdownDriver)], hostDirectives: [ TuiObscured, { directive: TuiActiveZone, inputs: ['tuiActiveZoneParent'], outputs: ['tuiActiveZoneChange'], }, ], host: { '(click)': 'onClick($event.target)', '(keydown.arrowDown)': 'onArrow($event, false)', '(keydown.arrowUp)': 'onArrow($event, true)', '(document:keydown.zoneless.capture)': 'onEsc($event)', '(document:keydown.zoneless)': 'onKeydown($event)', // TODO: Necessary because startWith(false) + distinctUntilChanged() in TuiActiveZone, think of better solution '(tuiActiveZoneChange)': '0', }, }] }], propDecorators: { dropdownHost: [{ type: ContentChild, args: ['tuiDropdownHost', { descendants: true, read: ElementRef }] }], tuiDropdownEnabled: [{ type: Input }], tuiDropdownOpen: [{ type: Input }], tuiDropdownOpenChange: [{ type: Output }], onEsc: [] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-open.directive.js","sourceRoot":"","sources":["../../../../../projects/core/directives/dropdown/dropdown-open.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    computed,\n    ContentChild,\n    Directive,\n    ElementRef,\n    EventEmitter,\n    inject,\n    Input,\n    type OnChanges,\n    Output,\n} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {TuiActiveZone} from '@taiga-ui/cdk/directives/active-zone';\nimport {TuiObscured} from '@taiga-ui/cdk/directives/obscured';\nimport {\n    tuiCloseWatcher,\n    tuiIfMap,\n    tuiWatch,\n    tuiZonefull,\n} from '@taiga-ui/cdk/observables';\nimport {\n    tuiGetActualTarget,\n    tuiInjectElement,\n    tuiIsElement,\n    tuiIsElementEditable,\n    tuiIsHTMLElement,\n} from '@taiga-ui/cdk/utils/dom';\nimport {\n    tuiGetClosestFocusable,\n    tuiIsNativeFocusedIn,\n    tuiIsNativeKeyboardFocusable,\n} from '@taiga-ui/cdk/utils/focus';\nimport {tuiAsDriver} from '@taiga-ui/core/classes';\nimport {tuiIsEditingKey} from '@taiga-ui/core/utils/miscellaneous';\nimport {shouldCall} from '@taiga-ui/event-plugins';\nimport {filter, fromEvent, merge} from 'rxjs';\n\nimport {TuiDropdownDirective} from './dropdown.directive';\nimport {TuiDropdownDriver} from './dropdown.driver';\n\nfunction shouldClose(this: TuiDropdownOpen, event: KeyboardEvent): boolean {\n    return (\n        // @ts-ignore\n        typeof CloseWatcher === 'undefined' &&\n        // ?. for auto fill events\n        event.key?.toLowerCase() === 'escape' &&\n        this.tuiDropdownEnabled &&\n        !!this.tuiDropdownOpen &&\n        !this['dropdown']()?.nextElementSibling\n    );\n}\n\n@Directive({\n    standalone: true,\n    selector: '[tuiDropdown][tuiDropdownOpen],[tuiDropdown][tuiDropdownOpenChange]',\n    providers: [TuiDropdownDriver, tuiAsDriver(TuiDropdownDriver)],\n    hostDirectives: [\n        TuiObscured,\n        {\n            directive: TuiActiveZone,\n            inputs: ['tuiActiveZoneParent'],\n            outputs: ['tuiActiveZoneChange'],\n        },\n    ],\n    host: {\n        '(click)': 'onClick($event.target)',\n        '(keydown.arrowDown)': 'onArrow($event, false)',\n        '(keydown.arrowUp)': 'onArrow($event, true)',\n        '(document:keydown.zoneless.capture)': 'onEsc($event)',\n        '(document:keydown.zoneless)': 'onKeydown($event)',\n        // TODO: Necessary because startWith(false) + distinctUntilChanged() in TuiActiveZone, think of better solution\n        '(tuiActiveZoneChange)': '0',\n    },\n})\nexport class TuiDropdownOpen implements OnChanges {\n    @ContentChild('tuiDropdownHost', {descendants: true, read: ElementRef})\n    private readonly dropdownHost?: ElementRef<HTMLElement>;\n\n    private readonly directive = inject(TuiDropdownDirective);\n    private readonly el = tuiInjectElement();\n    private readonly obscured = inject(TuiObscured);\n    private readonly activeZone = inject(TuiActiveZone);\n\n    private readonly dropdown = computed(\n        () => this.directive.ref()?.location.nativeElement,\n    );\n\n    @Input()\n    public tuiDropdownEnabled = true;\n\n    @Input()\n    public tuiDropdownOpen: boolean | '' = false;\n\n    @Output()\n    public readonly tuiDropdownOpenChange = new EventEmitter<boolean>();\n\n    // TODO: make it private when all legacy controls will be deleted from @taiga-ui/legacy (5.0)\n    public readonly driver = inject(TuiDropdownDriver);\n    public readonly sub = this.driver\n        .pipe(\n            tuiIfMap(() =>\n                merge(\n                    tuiCloseWatcher(),\n                    this.obscured.tuiObscured.pipe(filter(Boolean)),\n                    this.activeZone.tuiActiveZoneChange.pipe(filter((a) => !a)),\n                    fromEvent(this.el, 'focusin').pipe(\n                        filter(\n                            (event) =>\n                                !this.host.contains(tuiGetActualTarget(event)) ||\n                                !this.directive.ref(),\n                        ),\n                    ),\n                ),\n            ),\n            tuiZonefull(),\n            tuiWatch(),\n            takeUntilDestroyed(),\n        )\n        .subscribe(() => this.toggle(false));\n\n    public readonly sync = this.driver.pipe(takeUntilDestroyed()).subscribe((open) => {\n        if (open !== this.tuiDropdownOpen) {\n            this.update(open);\n        }\n    });\n\n    public ngOnChanges(): void {\n        this.drive(!!this.tuiDropdownOpen);\n        this.tuiDropdownOpenChange.emit(!!this.tuiDropdownOpen);\n    }\n\n    public toggle(open: boolean): void {\n        if (this.focused && !open) {\n            this.host.focus({preventScroll: true});\n        }\n\n        this.update(open);\n    }\n\n    @shouldCall(shouldClose)\n    protected onEsc(event: KeyboardEvent): void {\n        event.preventDefault();\n        this.toggle(false);\n    }\n\n    protected onClick(target: HTMLElement): void {\n        if (!this.editable && this.host.contains(target)) {\n            this.update(!this.tuiDropdownOpen);\n        }\n    }\n\n    protected onArrow(event: KeyboardEvent, up: boolean): void {\n        if (\n            !tuiIsElement(event.target) ||\n            !this.host.contains(event.target) ||\n            !this.tuiDropdownEnabled ||\n            !this.directive._content()\n        ) {\n            return;\n        }\n\n        event.preventDefault();\n        this.focusDropdown(up);\n    }\n\n    protected onKeydown(event: KeyboardEvent): void {\n        const target = tuiGetActualTarget(event);\n\n        if (\n            !event.defaultPrevented &&\n            tuiIsEditingKey(event.key) &&\n            this.editable &&\n            this.focused &&\n            tuiIsHTMLElement(target) &&\n            !tuiIsElementEditable(target)\n        ) {\n            this.host.focus({preventScroll: true});\n        }\n    }\n\n    private get host(): HTMLElement {\n        const initial = this.dropdownHost?.nativeElement || this.el;\n        const focusable = tuiIsNativeKeyboardFocusable(initial)\n            ? initial\n            : tuiGetClosestFocusable({initial, root: this.el});\n\n        return this.dropdownHost?.nativeElement || focusable || this.el;\n    }\n\n    private get editable(): boolean {\n        return tuiIsElementEditable(this.host);\n    }\n\n    private get focused(): boolean {\n        return tuiIsNativeFocusedIn(this.host) || tuiIsNativeFocusedIn(this.dropdown());\n    }\n\n    private update(open: boolean): void {\n        if (open && !this.tuiDropdownEnabled) {\n            return this.drive();\n        }\n\n        this.tuiDropdownOpen = open;\n        this.tuiDropdownOpenChange.emit(open);\n        this.drive();\n    }\n\n    private drive(open = !!this.tuiDropdownOpen && this.tuiDropdownEnabled): void {\n        this.obscured.tuiObscuredEnabled = open;\n        this.driver.next(open);\n    }\n\n    private focusDropdown(previous: boolean): void {\n        const root = this.dropdown();\n\n        if (!root) {\n            this.update(true);\n\n            return;\n        }\n\n        const doc = this.el.ownerDocument;\n        const child = root.appendChild(doc.createElement('div'));\n        const initial = previous ? child : root;\n        const focusable = tuiGetClosestFocusable({initial, previous, root});\n\n        child.remove();\n        focusable?.focus();\n    }\n}\n"]}