@taiga-ui/core
Version:
Core library for creating Angular components and applications using Taiga UI
92 lines • 20.6 kB
JavaScript
import { ChangeDetectionStrategy, Component, computed, inject, } from '@angular/core';
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
import { EMPTY_CLIENT_RECT } from '@taiga-ui/cdk/constants';
import { TuiActiveZone } from '@taiga-ui/cdk/directives/active-zone';
import { TuiAnimated } from '@taiga-ui/cdk/directives/animated';
import { tuiInjectElement } from '@taiga-ui/cdk/utils/dom';
import { tuiClamp } from '@taiga-ui/cdk/utils/math';
import { tuiPx } from '@taiga-ui/cdk/utils/miscellaneous';
import { tuiPositionAccessorFor, TuiRectAccessor, tuiRectAccessorFor, } from '@taiga-ui/core/classes';
import { TuiScrollbar } from '@taiga-ui/core/components/scrollbar';
import { TuiPositionService, TuiVisualViewportService } from '@taiga-ui/core/services';
import { TUI_DARK_MODE, TUI_VIEWPORT } from '@taiga-ui/core/tokens';
import { PolymorpheusOutlet } from '@taiga-ui/polymorpheus';
import { map, takeWhile } from 'rxjs';
import { TuiDropdownDirective } from './dropdown.directive';
import { TUI_DROPDOWN_CONTEXT } from './dropdown.providers';
import { TUI_DROPDOWN_OPTIONS } from './dropdown-options.directive';
import { TuiDropdownPosition } from './dropdown-position.directive';
import * as i0 from "@angular/core";
import * as i1 from "@taiga-ui/cdk/directives/active-zone";
import * as i2 from "@taiga-ui/cdk/directives/animated";
/**
* @description:
* This component is used to show template in a portal
* using default style of white rounded box with a shadow
*/
class TuiDropdownComponent {
constructor() {
this.el = tuiInjectElement();
this.accessor = inject(TuiRectAccessor);
this.viewport = inject(TUI_VIEWPORT);
this.vvs = inject(TuiVisualViewportService);
this.styles$ = inject(TuiPositionService).pipe(takeWhile(() => this.directive.el.isConnected &&
!!this.directive.el.getBoundingClientRect().height), map((v) => (this.position === 'fixed' ? this.vvs.correct(v) : v)), map(([top, left]) => this.getStyles(left, top)), takeUntilDestroyed());
this.options = inject(TUI_DROPDOWN_OPTIONS);
this.directive = inject(TuiDropdownDirective);
this.context = inject(TUI_DROPDOWN_CONTEXT, { optional: true });
this.darkMode = inject(TUI_DARK_MODE);
this.position = this.directive.position;
this.theme = computed((_ = this.darkMode()) => this.directive.el.closest('[tuiTheme]')?.getAttribute('tuiTheme'));
this.close = () => this.directive.toggle(false);
}
ngAfterViewInit() {
this.styles$.subscribe({
next: (styles) => Object.assign(this.el.style, styles),
complete: () => this.close?.(),
});
}
getStyles(x, y) {
const { maxHeight, minHeight, offset, limitWidth } = this.options;
const parent = this.el.offsetParent?.getBoundingClientRect() || EMPTY_CLIENT_RECT;
const { left = 0, top = 0 } = this.position === 'fixed' ? {} : parent;
const rect = this.accessor.getClientRect();
const viewport = this.viewport.getClientRect();
const above = rect.top - viewport.top - 2 * offset;
const below = viewport.top + viewport.height - y - offset;
const available = y > rect.bottom ? below : above;
const height = this.el.getBoundingClientRect().right <= rect.left || x >= rect.right
? maxHeight
: tuiClamp(available, minHeight, maxHeight);
y -= top;
x -= left;
return {
position: this.position,
top: tuiPx(Math.round(Math.max(y, offset - top))),
left: tuiPx(Math.round(x)),
maxHeight: tuiPx(Math.round(height)),
width: limitWidth === 'fixed' ? tuiPx(Math.round(rect.width)) : '',
minWidth: limitWidth === 'min' ? tuiPx(Math.round(rect.width)) : '',
maxWidth: tuiPx(Math.round(viewport.width) - 16), // 8px min gap from each side
};
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiDropdownComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TuiDropdownComponent, isStandalone: true, selector: "tui-dropdown", host: { properties: { "attr.data-appearance": "options.appearance", "attr.tuiTheme": "theme()" } }, providers: [
TuiPositionService,
tuiPositionAccessorFor('dropdown', TuiDropdownPosition),
tuiRectAccessorFor('dropdown', TuiDropdownDirective),
], hostDirectives: [{ directive: i1.TuiActiveZone }, { directive: i2.TuiAnimated }], ngImport: i0, template: "<tui-scrollbar class=\"t-scroll\">\n <div\n *polymorpheusOutlet=\"directive._content() as text; context: {$implicit: close}\"\n class=\"t-primitive\"\n >\n {{ text }}\n </div>\n</tui-scrollbar>\n", styles: [":host{position:absolute;display:flex;box-shadow:var(--tui-shadow-medium);color:var(--tui-text-primary);background:var(--tui-background-elevation-3);border-radius:var(--tui-radius-m);overflow:hidden;border:1px solid var(--tui-border-normal);box-sizing:border-box;isolation:isolate;pointer-events:auto;--tui-from: translateY(-1rem)}:host.tui-enter,:host.tui-leave{animation-name:tuiFade,tuiSlide}:host:not([style*=top]){visibility:hidden}.t-scroll{flex-grow:1;max-inline-size:100%;inline-size:-webkit-max-content;inline-size:max-content;overscroll-behavior:none}.t-primitive{padding:1rem}\n"], dependencies: [{ kind: "directive", type: PolymorpheusOutlet, selector: "[polymorpheusOutlet]", inputs: ["polymorpheusOutlet", "polymorpheusOutletContext"] }, { kind: "component", type: TuiScrollbar, selector: "tui-scrollbar", inputs: ["hidden"] }], changeDetection: i0.ChangeDetectionStrategy.Default }); }
}
export { TuiDropdownComponent };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TuiDropdownComponent, decorators: [{
type: Component,
args: [{ standalone: true, selector: 'tui-dropdown', imports: [PolymorpheusOutlet, TuiScrollbar], changeDetection: ChangeDetectionStrategy.Default, providers: [
TuiPositionService,
tuiPositionAccessorFor('dropdown', TuiDropdownPosition),
tuiRectAccessorFor('dropdown', TuiDropdownDirective),
], hostDirectives: [TuiActiveZone, TuiAnimated], host: {
'[attr.data-appearance]': 'options.appearance',
'[attr.tuiTheme]': 'theme()',
}, template: "<tui-scrollbar class=\"t-scroll\">\n <div\n *polymorpheusOutlet=\"directive._content() as text; context: {$implicit: close}\"\n class=\"t-primitive\"\n >\n {{ text }}\n </div>\n</tui-scrollbar>\n", styles: [":host{position:absolute;display:flex;box-shadow:var(--tui-shadow-medium);color:var(--tui-text-primary);background:var(--tui-background-elevation-3);border-radius:var(--tui-radius-m);overflow:hidden;border:1px solid var(--tui-border-normal);box-sizing:border-box;isolation:isolate;pointer-events:auto;--tui-from: translateY(-1rem)}:host.tui-enter,:host.tui-leave{animation-name:tuiFade,tuiSlide}:host:not([style*=top]){visibility:hidden}.t-scroll{flex-grow:1;max-inline-size:100%;inline-size:-webkit-max-content;inline-size:max-content;overscroll-behavior:none}.t-primitive{padding:1rem}\n"] }]
}] });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown.component.js","sourceRoot":"","sources":["../../../../../projects/core/directives/dropdown/dropdown.component.ts","../../../../../projects/core/directives/dropdown/dropdown.template.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    type AfterViewInit,\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    inject,\n} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {EMPTY_CLIENT_RECT} from '@taiga-ui/cdk/constants';\nimport {TuiActiveZone} from '@taiga-ui/cdk/directives/active-zone';\nimport {TuiAnimated} from '@taiga-ui/cdk/directives/animated';\nimport {tuiInjectElement} from '@taiga-ui/cdk/utils/dom';\nimport {tuiClamp} from '@taiga-ui/cdk/utils/math';\nimport {tuiPx} from '@taiga-ui/cdk/utils/miscellaneous';\nimport {\n    tuiPositionAccessorFor,\n    TuiRectAccessor,\n    tuiRectAccessorFor,\n} from '@taiga-ui/core/classes';\nimport {TuiScrollbar} from '@taiga-ui/core/components/scrollbar';\nimport {TuiPositionService, TuiVisualViewportService} from '@taiga-ui/core/services';\nimport {TUI_DARK_MODE, TUI_VIEWPORT} from '@taiga-ui/core/tokens';\nimport {PolymorpheusOutlet} from '@taiga-ui/polymorpheus';\nimport {map, takeWhile} from 'rxjs';\n\nimport {TuiDropdownDirective} from './dropdown.directive';\nimport {TUI_DROPDOWN_CONTEXT} from './dropdown.providers';\nimport {TUI_DROPDOWN_OPTIONS} from './dropdown-options.directive';\nimport {TuiDropdownPosition} from './dropdown-position.directive';\n\n/**\n * @description:\n * This component is used to show template in a portal\n * using default style of white rounded box with a shadow\n */\n@Component({\n    standalone: true,\n    selector: 'tui-dropdown',\n    imports: [PolymorpheusOutlet, TuiScrollbar],\n    templateUrl: './dropdown.template.html',\n    styleUrls: ['./dropdown.style.less'],\n    // @bad TODO: OnPush\n    // eslint-disable-next-line @angular-eslint/prefer-on-push-component-change-detection\n    changeDetection: ChangeDetectionStrategy.Default,\n    providers: [\n        TuiPositionService,\n        tuiPositionAccessorFor('dropdown', TuiDropdownPosition),\n        tuiRectAccessorFor('dropdown', TuiDropdownDirective),\n    ],\n    hostDirectives: [TuiActiveZone, TuiAnimated],\n    host: {\n        '[attr.data-appearance]': 'options.appearance',\n        '[attr.tuiTheme]': 'theme()',\n    },\n})\nexport class TuiDropdownComponent implements AfterViewInit {\n    private readonly el = tuiInjectElement();\n    private readonly accessor = inject(TuiRectAccessor);\n    private readonly viewport = inject(TUI_VIEWPORT);\n    private readonly vvs = inject(TuiVisualViewportService);\n\n    private readonly styles$ = inject(TuiPositionService).pipe(\n        takeWhile(\n            () =>\n                this.directive.el.isConnected &&\n                !!this.directive.el.getBoundingClientRect().height,\n        ),\n        map((v) => (this.position === 'fixed' ? this.vvs.correct(v) : v)),\n        map(([top, left]) => this.getStyles(left, top)),\n        takeUntilDestroyed(),\n    );\n\n    protected readonly options = inject(TUI_DROPDOWN_OPTIONS);\n    protected readonly directive = inject(TuiDropdownDirective);\n    protected readonly context = inject(TUI_DROPDOWN_CONTEXT, {optional: true});\n    protected readonly darkMode = inject(TUI_DARK_MODE);\n    protected readonly position = this.directive.position;\n    protected readonly theme = computed((_ = this.darkMode()) =>\n        this.directive.el.closest('[tuiTheme]')?.getAttribute('tuiTheme'),\n    );\n\n    public ngAfterViewInit(): void {\n        this.styles$.subscribe({\n            next: (styles) => Object.assign(this.el.style, styles),\n            complete: () => this.close?.(),\n        });\n    }\n\n    protected readonly close = (): void => this.directive.toggle(false);\n\n    private getStyles(x: number, y: number): Record<string, string> {\n        const {maxHeight, minHeight, offset, limitWidth} = this.options;\n        const parent = this.el.offsetParent?.getBoundingClientRect() || EMPTY_CLIENT_RECT;\n        const {left = 0, top = 0} = this.position === 'fixed' ? {} : parent;\n        const rect = this.accessor.getClientRect();\n        const viewport = this.viewport.getClientRect();\n        const above = rect.top - viewport.top - 2 * offset;\n        const below = viewport.top + viewport.height - y - offset;\n        const available = y > rect.bottom ? below : above;\n        const height =\n            this.el.getBoundingClientRect().right <= rect.left || x >= rect.right\n                ? maxHeight\n                : tuiClamp(available, minHeight, maxHeight);\n\n        y -= top;\n        x -= left;\n\n        return {\n            position: this.position,\n            top: tuiPx(Math.round(Math.max(y, offset - top))),\n            left: tuiPx(Math.round(x)),\n            maxHeight: tuiPx(Math.round(height)),\n            width: limitWidth === 'fixed' ? tuiPx(Math.round(rect.width)) : '',\n            minWidth: limitWidth === 'min' ? tuiPx(Math.round(rect.width)) : '',\n            maxWidth: tuiPx(Math.round(viewport.width) - 16), // 8px min gap from each side\n        };\n    }\n}\n","<tui-scrollbar class=\"t-scroll\">\n    <div\n        *polymorpheusOutlet=\"directive._content() as text; context: {$implicit: close}\"\n        class=\"t-primitive\"\n    >\n        {{ text }}\n    </div>\n</tui-scrollbar>\n"]}