@cuby-ui/core 0.0.104 → 0.0.106
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/context-menu/context-menu.component.d.ts +5 -5
- package/components/root/index.d.ts +1 -0
- package/components/root/root.options.d.ts +1 -0
- package/components/select/select.component.d.ts +6 -2
- package/esm2020/components/context-menu/context-menu.component.mjs +24 -23
- package/esm2020/components/root/index.mjs +2 -1
- package/esm2020/components/root/root.component.mjs +3 -2
- package/esm2020/components/root/root.options.mjs +2 -0
- package/esm2020/components/select/select.component.mjs +38 -6
- package/fesm2015/cuby-ui-core.mjs +252 -219
- package/fesm2015/cuby-ui-core.mjs.map +1 -1
- package/fesm2020/cuby-ui-core.mjs +251 -218
- package/fesm2020/cuby-ui-core.mjs.map +1 -1
- package/package.json +3 -3
- package/styles/theme.scss +4 -0
- package/styles/variables/colors.scss +2 -0
|
@@ -1,10 +1,10 @@
|
|
|
1
|
-
import type { OnInit, AfterViewInit } from '@angular/core';
|
|
1
|
+
import type { OnInit, AfterViewInit, TemplateRef } from '@angular/core';
|
|
2
2
|
import type { CuiContextMenuItem } from '../../interfaces';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
import * as i1 from "@cuby-ui/cdk";
|
|
5
5
|
export declare class CuiContextMenuComponent implements OnInit, AfterViewInit {
|
|
6
6
|
private readonly changeDetectorRef;
|
|
7
|
-
private readonly
|
|
7
|
+
private readonly viewContainerRef;
|
|
8
8
|
private readonly document;
|
|
9
9
|
private readonly documentElement;
|
|
10
10
|
private readonly window;
|
|
@@ -15,11 +15,10 @@ export declare class CuiContextMenuComponent implements OnInit, AfterViewInit {
|
|
|
15
15
|
protected readonly EXTRA_OFFSET_TOP = 0;
|
|
16
16
|
protected isVisible: boolean;
|
|
17
17
|
protected targetEventListener: (event: MouseEvent) => void;
|
|
18
|
+
protected node?: HTMLDivElement;
|
|
18
19
|
items: CuiContextMenuItem[];
|
|
19
20
|
target: HTMLElement;
|
|
20
|
-
|
|
21
|
-
protected clientX?: number;
|
|
22
|
-
protected clientY?: number;
|
|
21
|
+
protected readonly list: TemplateRef<any>;
|
|
23
22
|
ngOnInit(): void;
|
|
24
23
|
ngAfterViewInit(): void;
|
|
25
24
|
protected onSelect(item: CuiContextMenuItem): void;
|
|
@@ -31,6 +30,7 @@ export declare class CuiContextMenuComponent implements OnInit, AfterViewInit {
|
|
|
31
30
|
private initHidingSubscription;
|
|
32
31
|
private initHidingFromWindowEvent;
|
|
33
32
|
private close;
|
|
33
|
+
private renderList;
|
|
34
34
|
private changePosition;
|
|
35
35
|
static ɵfac: i0.ɵɵFactoryDeclaration<CuiContextMenuComponent, never>;
|
|
36
36
|
static ɵcmp: i0.ɵɵComponentDeclaration<CuiContextMenuComponent, "cui-context-menu[items][target]", never, { "items": "items"; "target": "target"; }, {}, never, never, false, [{ directive: typeof i1.CuiClickOutsideDirective; inputs: {}; outputs: {}; }, { directive: typeof i1.CuiTargetDirective; inputs: { "ccTarget": "target"; }; outputs: {}; }]>;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const CUI_ROOT_SELECTOR = "cui-root";
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type { OnInit } from '@angular/core';
|
|
1
|
+
import type { OnInit, TemplateRef } from '@angular/core';
|
|
2
2
|
import { ElementRef } from '@angular/core';
|
|
3
3
|
import { ControlValueAccessor } from '@angular/forms';
|
|
4
4
|
import type { CuiOnChange, CuiOnTouched, CuiNullable } from '@cuby-ui/cdk';
|
|
@@ -10,6 +10,7 @@ export declare class CuiSelectComponent implements ControlValueAccessor, OnInit
|
|
|
10
10
|
private readonly element;
|
|
11
11
|
private readonly document;
|
|
12
12
|
private readonly changeDetectorRef;
|
|
13
|
+
private readonly viewContainerRef;
|
|
13
14
|
private readonly destroy$;
|
|
14
15
|
private readonly cuiTextFieldController;
|
|
15
16
|
private readonly cuiClickOutsideDirective;
|
|
@@ -23,7 +24,9 @@ export declare class CuiSelectComponent implements ControlValueAccessor, OnInit
|
|
|
23
24
|
protected selectedOption?: CuiOption;
|
|
24
25
|
options: CuiOption[];
|
|
25
26
|
defaultOptionText?: string;
|
|
26
|
-
protected
|
|
27
|
+
protected readonly button: ElementRef<HTMLButtonElement>;
|
|
28
|
+
protected readonly getOptionsListHeightContainer: ElementRef<HTMLUListElement>;
|
|
29
|
+
protected readonly optionsWrapper: TemplateRef<unknown>;
|
|
27
30
|
protected gap: string;
|
|
28
31
|
protected get id(): string | undefined;
|
|
29
32
|
protected get size(): CuiSizeSm | CuiSizeMd;
|
|
@@ -38,6 +41,7 @@ export declare class CuiSelectComponent implements ControlValueAccessor, OnInit
|
|
|
38
41
|
protected onSelect(option: CuiOption): void;
|
|
39
42
|
protected onSwitch(): void;
|
|
40
43
|
protected onClose(): void;
|
|
44
|
+
protected onWindowResize(): void;
|
|
41
45
|
private initClickOutsideSubscription;
|
|
42
46
|
private open;
|
|
43
47
|
private close;
|
|
@@ -1,7 +1,8 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, ChangeDetectorRef, Component,
|
|
1
|
+
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostListener, inject, Input, ViewContainerRef, ViewChild, } from '@angular/core';
|
|
2
2
|
import { DOCUMENT } from '@angular/common';
|
|
3
3
|
import { fromEvent, take, takeUntil } from 'rxjs';
|
|
4
4
|
import { CuiDestroyService, CuiClickOutsideDirective, CuiTargetDirective, CUI_WINDOW } from '@cuby-ui/cdk';
|
|
5
|
+
import { CUI_ROOT_SELECTOR } from '../root';
|
|
5
6
|
import * as i0 from "@angular/core";
|
|
6
7
|
import * as i1 from "@cuby-ui/cdk";
|
|
7
8
|
import * as i2 from "@angular/common";
|
|
@@ -9,7 +10,7 @@ import * as i3 from "../svg/svg.component";
|
|
|
9
10
|
export class CuiContextMenuComponent {
|
|
10
11
|
constructor() {
|
|
11
12
|
this.changeDetectorRef = inject(ChangeDetectorRef);
|
|
12
|
-
this.
|
|
13
|
+
this.viewContainerRef = inject(ViewContainerRef);
|
|
13
14
|
this.document = inject(DOCUMENT);
|
|
14
15
|
this.documentElement = this.document.documentElement;
|
|
15
16
|
this.window = inject(CUI_WINDOW);
|
|
@@ -19,7 +20,6 @@ export class CuiContextMenuComponent {
|
|
|
19
20
|
this.EXTRA_OFFSET_LEFT = 0;
|
|
20
21
|
this.EXTRA_OFFSET_TOP = 0;
|
|
21
22
|
this.isVisible = false;
|
|
22
|
-
this.isHidden = false;
|
|
23
23
|
}
|
|
24
24
|
ngOnInit() {
|
|
25
25
|
this.initClickOutsideSubscription();
|
|
@@ -28,7 +28,7 @@ export class CuiContextMenuComponent {
|
|
|
28
28
|
this.initTargetElementListener();
|
|
29
29
|
}
|
|
30
30
|
onSelect(item) {
|
|
31
|
-
this.
|
|
31
|
+
this.close();
|
|
32
32
|
item.command?.();
|
|
33
33
|
}
|
|
34
34
|
onClose() {
|
|
@@ -47,7 +47,7 @@ export class CuiContextMenuComponent {
|
|
|
47
47
|
if (this.target.contains(target)) {
|
|
48
48
|
return;
|
|
49
49
|
}
|
|
50
|
-
this.
|
|
50
|
+
this.close();
|
|
51
51
|
});
|
|
52
52
|
}
|
|
53
53
|
initTargetElementListener() {
|
|
@@ -55,7 +55,9 @@ export class CuiContextMenuComponent {
|
|
|
55
55
|
event.stopPropagation();
|
|
56
56
|
this.document.body.click();
|
|
57
57
|
this.isVisible = !this.isVisible;
|
|
58
|
+
this.viewContainerRef.clear();
|
|
58
59
|
if (this.isVisible) {
|
|
60
|
+
this.renderList();
|
|
59
61
|
this.changePosition(event.clientX, event.clientY);
|
|
60
62
|
this.initHidingSubscription();
|
|
61
63
|
}
|
|
@@ -79,13 +81,19 @@ export class CuiContextMenuComponent {
|
|
|
79
81
|
}
|
|
80
82
|
close() {
|
|
81
83
|
this.isVisible = false;
|
|
84
|
+
this.viewContainerRef.clear();
|
|
85
|
+
}
|
|
86
|
+
renderList() {
|
|
87
|
+
const embeddedViewRef = this.viewContainerRef.createEmbeddedView(this.list);
|
|
88
|
+
this.node = embeddedViewRef.rootNodes[0];
|
|
89
|
+
embeddedViewRef.detectChanges();
|
|
90
|
+
this.document.querySelector(CUI_ROOT_SELECTOR).appendChild(this.node);
|
|
82
91
|
}
|
|
83
92
|
changePosition(x, y) {
|
|
84
|
-
this.isHidden = true;
|
|
85
93
|
setTimeout(() => {
|
|
86
94
|
const { clientWidth, clientHeight } = this.documentElement;
|
|
87
|
-
const width = this.
|
|
88
|
-
const height = this.
|
|
95
|
+
const width = this.node.offsetWidth;
|
|
96
|
+
const height = this.node.offsetHeight;
|
|
89
97
|
let left = x;
|
|
90
98
|
let top = y;
|
|
91
99
|
if (left + width > clientWidth) {
|
|
@@ -96,15 +104,14 @@ export class CuiContextMenuComponent {
|
|
|
96
104
|
}
|
|
97
105
|
left = Math.max(left, this.EXTRA_OFFSET_LEFT);
|
|
98
106
|
top = Math.max(top, this.EXTRA_OFFSET_TOP);
|
|
99
|
-
this.
|
|
100
|
-
this.
|
|
101
|
-
this.clientY = top;
|
|
107
|
+
this.node.style.left = left + 'px';
|
|
108
|
+
this.node.style.top = top + 'px';
|
|
102
109
|
this.changeDetectorRef.markForCheck();
|
|
103
110
|
});
|
|
104
111
|
}
|
|
105
112
|
}
|
|
106
113
|
CuiContextMenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CuiContextMenuComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
107
|
-
CuiContextMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CuiContextMenuComponent, selector: "cui-context-menu[items][target]", inputs: { items: "items", target: "target" }, host: { listeners: { "window:resize": "onClose()", "click": "onStopClickPropagation($event)" },
|
|
114
|
+
CuiContextMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CuiContextMenuComponent, selector: "cui-context-menu[items][target]", inputs: { items: "items", target: "target" }, host: { listeners: { "window:resize": "onClose()", "click": "onStopClickPropagation($event)" } }, providers: [CuiDestroyService], viewQueries: [{ propertyName: "list", first: true, predicate: ["list"], descendants: true }], hostDirectives: [{ directive: i1.CuiClickOutsideDirective }, { directive: i1.CuiTargetDirective, inputs: ["ccTarget", "target"] }], ngImport: i0, template: "<ng-template #list>\n <div class=\"c-container\">\n <ul\n *ngIf=\"isVisible\"\n class=\"c-ul\"\n >\n <li\n *ngFor=\"let item of items; trackBy: trackByFn\"\n [style.color]=\"item.color\"\n >\n <button\n type=\"button\"\n class=\"c-button\"\n (click)=\"onSelect(item)\"\n >\n <cui-svg\n *ngIf=\"item.icon\"\n [icon]=\"item.icon\"\n [color]=\"item.color\"\n ></cui-svg>\n <span>{{ item.label }}</span>\n </button>\n </li>\n </ul>\n </div>\n</ng-template>\n", styles: [":host{display:none}.c-container{position:fixed;width:268px;border-radius:8px}.c-ul{padding:7px 0;margin:0;list-style:none;font-weight:400;font-size:14px;line-height:20px;box-shadow:0 1px 4px #0000000a,0 1px 4px #0000000a;border:1px solid var(--cui-base-200);border-radius:inherit;background:var(--cui-base-0);color:var(--cui-base-900);font-family:var(--cui-main-font)}.c-button{padding:8px 11px 8px 0;border:0;flex-shrink:0;outline:none;cursor:pointer;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:8px 11px;width:100%;display:flex;align-items:center;gap:12px}.c-button:active{background:var(--cui-base-50)}@media (hover: hover){.c-button:hover{background:var(--cui-base-50)}}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.CuiSvgComponent, selector: "cui-svg[icon]", inputs: ["width", "height", "strokeWidth", "color", "icon"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
108
115
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CuiContextMenuComponent, decorators: [{
|
|
109
116
|
type: Component,
|
|
110
117
|
args: [{ selector: 'cui-context-menu[items][target]', changeDetection: ChangeDetectionStrategy.OnPush, providers: [CuiDestroyService], hostDirectives: [
|
|
@@ -113,20 +120,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
113
120
|
directive: CuiTargetDirective,
|
|
114
121
|
inputs: ['ccTarget: target']
|
|
115
122
|
}
|
|
116
|
-
], template: "<ul\n
|
|
123
|
+
], template: "<ng-template #list>\n <div class=\"c-container\">\n <ul\n *ngIf=\"isVisible\"\n class=\"c-ul\"\n >\n <li\n *ngFor=\"let item of items; trackBy: trackByFn\"\n [style.color]=\"item.color\"\n >\n <button\n type=\"button\"\n class=\"c-button\"\n (click)=\"onSelect(item)\"\n >\n <cui-svg\n *ngIf=\"item.icon\"\n [icon]=\"item.icon\"\n [color]=\"item.color\"\n ></cui-svg>\n <span>{{ item.label }}</span>\n </button>\n </li>\n </ul>\n </div>\n</ng-template>\n", styles: [":host{display:none}.c-container{position:fixed;width:268px;border-radius:8px}.c-ul{padding:7px 0;margin:0;list-style:none;font-weight:400;font-size:14px;line-height:20px;box-shadow:0 1px 4px #0000000a,0 1px 4px #0000000a;border:1px solid var(--cui-base-200);border-radius:inherit;background:var(--cui-base-0);color:var(--cui-base-900);font-family:var(--cui-main-font)}.c-button{padding:8px 11px 8px 0;border:0;flex-shrink:0;outline:none;cursor:pointer;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:8px 11px;width:100%;display:flex;align-items:center;gap:12px}.c-button:active{background:var(--cui-base-50)}@media (hover: hover){.c-button:hover{background:var(--cui-base-50)}}\n"] }]
|
|
117
124
|
}], propDecorators: { items: [{
|
|
118
125
|
type: Input
|
|
119
126
|
}], target: [{
|
|
120
127
|
type: Input
|
|
121
|
-
}],
|
|
122
|
-
type:
|
|
123
|
-
args: ['
|
|
124
|
-
}], clientX: [{
|
|
125
|
-
type: HostBinding,
|
|
126
|
-
args: ['style.left.px']
|
|
127
|
-
}], clientY: [{
|
|
128
|
-
type: HostBinding,
|
|
129
|
-
args: ['style.top.px']
|
|
128
|
+
}], list: [{
|
|
129
|
+
type: ViewChild,
|
|
130
|
+
args: ['list']
|
|
130
131
|
}], onClose: [{
|
|
131
132
|
type: HostListener,
|
|
132
133
|
args: ['window:resize']
|
|
@@ -134,4 +135,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
134
135
|
type: HostListener,
|
|
135
136
|
args: ['click', ['$event']]
|
|
136
137
|
}] } });
|
|
137
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../projects/core/components/context-menu/context-menu.component.ts","../../../../../projects/core/components/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import type { OnInit, AfterViewInit } from '@angular/core';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  HostBinding,\n  HostListener,\n  inject,\n  Input\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\nimport { fromEvent, take, takeUntil } from 'rxjs';\n\nimport { CuiDestroyService, CuiClickOutsideDirective, CuiTargetDirective, CUI_WINDOW } from '@cuby-ui/cdk';\n\nimport type { CuiContextMenuItem } from '../../interfaces';\n\n@Component({\n  selector: 'cui-context-menu[items][target]',\n  templateUrl: './context-menu.component.html',\n  styleUrls: ['./context-menu.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [CuiDestroyService],\n  hostDirectives: [\n    CuiClickOutsideDirective,\n    {\n      directive: CuiTargetDirective,\n      inputs: ['ccTarget: target']\n    }\n  ],\n})\nexport class CuiContextMenuComponent implements OnInit, AfterViewInit {\n  private readonly changeDetectorRef = inject(ChangeDetectorRef);\n  private readonly element = inject(ElementRef).nativeElement;\n  private readonly document = inject(DOCUMENT);\n  private readonly documentElement = this.document.documentElement;\n  private readonly window = inject(CUI_WINDOW);\n  private readonly destroy$ = inject(CuiDestroyService, { self: true });\n  private readonly cuiClickOutsideDirective = inject(CuiClickOutsideDirective, { self: true });\n  private readonly cuiTargetDirective = inject(CuiTargetDirective, { self: true });\n\n  protected readonly EXTRA_OFFSET_LEFT = 0;\n  protected readonly EXTRA_OFFSET_TOP = 0;\n\n  protected isVisible = false;\n  protected targetEventListener!: (event: MouseEvent) => void;\n\n  @Input()\n  public items!: CuiContextMenuItem[];\n\n  @Input()\n  public target!: HTMLElement;\n\n  @HostBinding('class._hidden')\n  isHidden: boolean = false;\n\n  @HostBinding('style.left.px')\n  protected clientX?: number;\n\n  @HostBinding('style.top.px')\n  protected clientY?: number;\n\n  public ngOnInit(): void {\n    this.initClickOutsideSubscription();\n  }\n\n  public ngAfterViewInit(): void {\n    this.initTargetElementListener();\n  }\n\n  protected onSelect(item: CuiContextMenuItem): void {\n    this.isVisible = false;\n\n    item.command?.();\n  }\n\n  @HostListener('window:resize')\n  protected onClose(): void {\n    this.close();\n  }\n\n  @HostListener('click', ['$event'])\n  protected onStopClickPropagation(event: MouseEvent): void {\n    event.stopPropagation();\n  }\n\n  protected trackByFn(_: number, item: CuiContextMenuItem): string {\n    return item.label;\n  }\n\n  private initClickOutsideSubscription(): void {\n    this.cuiClickOutsideDirective.cuiClickOutside\n      .pipe(takeUntil(this.destroy$))\n      .subscribe((target) => {\n        if (this.target.contains(target)) {\n          return;\n        }\n\n        this.isVisible = false;\n      })\n  }\n\n  private initTargetElementListener(): void {\n    this.targetEventListener = (event) => {\n      event.stopPropagation();\n      this.document.body.click();\n\n      this.isVisible = !this.isVisible;\n\n      if (this.isVisible) {\n        this.changePosition(event.clientX, event.clientY);\n        this.initHidingSubscription();\n      }\n\n      this.changeDetectorRef.markForCheck();\n    };\n\n    this.cuiTargetDirective.clicked\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(this.targetEventListener);\n  }\n\n  private initHidingSubscription(): void {\n    this.initHidingFromWindowEvent('scroll');\n    this.initHidingFromWindowEvent('resize');\n  }\n\n  private initHidingFromWindowEvent(eventName: keyof WindowEventMap): void {\n    fromEvent(this.window, eventName)\n      .pipe(take(1))\n      .subscribe(() => {\n        this.close();\n        this.changeDetectorRef.markForCheck();\n      });\n  }\n\n  private close(): void {\n    this.isVisible = false;\n  }\n\n  private changePosition(x: number, y: number): void {\n    this.isHidden = true;\n\n    setTimeout(() => {\n      const { clientWidth, clientHeight } = this.documentElement;\n      const width = this.element.offsetWidth;\n      const height = this.element.offsetHeight;\n      let left = x;\n      let top = y;\n\n      if (left + width > clientWidth) {\n        left -= width;\n      }\n\n      if (top + height > clientHeight) {\n        top -= height;\n      }\n\n      left = Math.max(left, this.EXTRA_OFFSET_LEFT);\n      top = Math.max(top, this.EXTRA_OFFSET_TOP);\n\n      this.isHidden = false;\n      this.clientX = left;\n      this.clientY = top;\n\n      this.changeDetectorRef.markForCheck();\n    });\n  }\n}\n","<ul\n  *ngIf=\"isVisible\"\n  class=\"c-ul\"\n>\n  <li\n    *ngFor=\"let item of items; trackBy: trackByFn\"\n    [style.color]=\"item.color\"\n  >\n    <button\n      class=\"c-button\"\n      (click)=\"onSelect(item)\"\n    >\n      <cui-svg\n        *ngIf=\"item.icon\"\n        [icon]=\"item.icon\"\n        [color]=\"item.color\"\n      ></cui-svg>\n      <span>{{ item.label }}</span>\n    </button>\n  </li>\n</ul>\n"]}
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../projects/core/components/context-menu/context-menu.component.ts","../../../../../projects/core/components/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import type { OnInit, AfterViewInit, TemplateRef } from '@angular/core';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  HostListener,\n  inject,\n  Input,\n  ViewContainerRef,\n  ViewChild,\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\nimport { fromEvent, take, takeUntil } from 'rxjs';\n\nimport { CuiDestroyService, CuiClickOutsideDirective, CuiTargetDirective, CUI_WINDOW } from '@cuby-ui/cdk';\n\nimport type { CuiContextMenuItem } from '../../interfaces';\nimport { CUI_ROOT_SELECTOR } from '../root';\n\n@Component({\n  selector: 'cui-context-menu[items][target]',\n  templateUrl: './context-menu.component.html',\n  styleUrls: ['./context-menu.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [CuiDestroyService],\n  hostDirectives: [\n    CuiClickOutsideDirective,\n    {\n      directive: CuiTargetDirective,\n      inputs: ['ccTarget: target']\n    }\n  ],\n})\nexport class CuiContextMenuComponent implements OnInit, AfterViewInit {\n  private readonly changeDetectorRef = inject(ChangeDetectorRef);\n  private readonly viewContainerRef = inject(ViewContainerRef);\n  private readonly document = inject(DOCUMENT);\n  private readonly documentElement = this.document.documentElement;\n  private readonly window = inject(CUI_WINDOW);\n  private readonly destroy$ = inject(CuiDestroyService, { self: true });\n  private readonly cuiClickOutsideDirective = inject(CuiClickOutsideDirective, { self: true });\n  private readonly cuiTargetDirective = inject(CuiTargetDirective, { self: true });\n\n  protected readonly EXTRA_OFFSET_LEFT = 0;\n  protected readonly EXTRA_OFFSET_TOP = 0;\n\n  protected isVisible = false;\n  protected targetEventListener!: (event: MouseEvent) => void;\n  protected node?: HTMLDivElement;\n\n  @Input()\n  public items!: CuiContextMenuItem[];\n\n  @Input()\n  public target!: HTMLElement;\n\n  @ViewChild('list')\n  protected readonly list!: TemplateRef<any>;\n\n  public ngOnInit(): void {\n    this.initClickOutsideSubscription();\n  }\n\n  public ngAfterViewInit(): void {\n    this.initTargetElementListener();\n  }\n\n  protected onSelect(item: CuiContextMenuItem): void {\n    this.close();\n\n    item.command?.();\n  }\n\n  @HostListener('window:resize')\n  protected onClose(): void {\n    this.close();\n  }\n\n  @HostListener('click', ['$event'])\n  protected onStopClickPropagation(event: MouseEvent): void {\n    event.stopPropagation();\n  }\n\n  protected trackByFn(_: number, item: CuiContextMenuItem): string {\n    return item.label;\n  }\n\n  private initClickOutsideSubscription(): void {\n    this.cuiClickOutsideDirective.cuiClickOutside\n      .pipe(takeUntil(this.destroy$))\n      .subscribe((target) => {\n        if (this.target.contains(target)) {\n          return;\n        }\n\n        this.close();\n      })\n  }\n\n  private initTargetElementListener(): void {\n    this.targetEventListener = (event) => {\n      event.stopPropagation();\n      this.document.body.click();\n\n      this.isVisible = !this.isVisible;\n      this.viewContainerRef.clear();\n\n      if (this.isVisible) {\n        this.renderList();\n        this.changePosition(event.clientX, event.clientY);\n        this.initHidingSubscription();\n      }\n\n      this.changeDetectorRef.markForCheck();\n    };\n\n    this.cuiTargetDirective.clicked\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(this.targetEventListener);\n  }\n\n  private initHidingSubscription(): void {\n    this.initHidingFromWindowEvent('scroll');\n    this.initHidingFromWindowEvent('resize');\n  }\n\n  private initHidingFromWindowEvent(eventName: keyof WindowEventMap): void {\n    fromEvent(this.window, eventName)\n      .pipe(take(1))\n      .subscribe(() => {\n        this.close();\n        this.changeDetectorRef.markForCheck();\n      });\n  }\n\n  private close(): void {\n    this.isVisible = false;\n    this.viewContainerRef.clear();\n  }\n\n  private renderList(): void {\n    const embeddedViewRef = this.viewContainerRef.createEmbeddedView(this.list);\n\n    this.node = embeddedViewRef.rootNodes[0];\n\n    embeddedViewRef.detectChanges();\n\n    this.document.querySelector(CUI_ROOT_SELECTOR)!.appendChild(this.node!);\n  }\n\n  private changePosition(x: number, y: number): void {\n    setTimeout(() => {\n      const { clientWidth, clientHeight } = this.documentElement;\n      const width = this.node!.offsetWidth;\n      const height = this.node!.offsetHeight;\n      let left = x;\n      let top = y;\n\n      if (left + width > clientWidth) {\n        left -= width;\n      }\n\n      if (top + height > clientHeight) {\n        top -= height;\n      }\n\n      left = Math.max(left, this.EXTRA_OFFSET_LEFT);\n      top = Math.max(top, this.EXTRA_OFFSET_TOP);\n\n      this.node!.style.left = left + 'px';\n      this.node!.style.top = top + 'px';\n\n      this.changeDetectorRef.markForCheck();\n    });\n  }\n}\n","<ng-template #list>\n  <div class=\"c-container\">\n    <ul\n      *ngIf=\"isVisible\"\n      class=\"c-ul\"\n    >\n      <li\n        *ngFor=\"let item of items; trackBy: trackByFn\"\n        [style.color]=\"item.color\"\n      >\n        <button\n          type=\"button\"\n          class=\"c-button\"\n          (click)=\"onSelect(item)\"\n        >\n          <cui-svg\n            *ngIf=\"item.icon\"\n            [icon]=\"item.icon\"\n            [color]=\"item.color\"\n          ></cui-svg>\n          <span>{{ item.label }}</span>\n        </button>\n      </li>\n    </ul>\n  </div>\n</ng-template>\n"]}
|
|
@@ -1,3 +1,4 @@
|
|
|
1
1
|
export * from './root.component';
|
|
2
2
|
export * from './root.module';
|
|
3
|
-
|
|
3
|
+
export * from './root.options';
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL2NvbXBvbmVudHMvcm9vdC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsZ0JBQWdCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3Jvb3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcm9vdC5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9yb290Lm9wdGlvbnMnO1xuIl19
|
|
@@ -3,6 +3,7 @@ import { DOCUMENT } from '@angular/common';
|
|
|
3
3
|
import { CuiDestroyService } from '@cuby-ui/cdk';
|
|
4
4
|
import { takeUntil } from 'rxjs';
|
|
5
5
|
import { CuiThemeService } from '../../services';
|
|
6
|
+
import { CUI_ROOT_SELECTOR } from './root.options';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "../dialog/dialogs.component";
|
|
8
9
|
import * as i2 from "../alert/alerts.component";
|
|
@@ -25,6 +26,6 @@ CuiRootComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version
|
|
|
25
26
|
CuiRootComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CuiRootComponent, selector: "cui-root", providers: [CuiDestroyService], ngImport: i0, template: "<div class=\"c-content\">\n <ng-content />\n</div>\n<cui-dialogs />\n<cui-alerts />\n", styles: [".c-content{isolation:isolate}\n"], dependencies: [{ kind: "component", type: i1.CuiDialogsComponent, selector: "cui-dialogs" }, { kind: "component", type: i2.CuiAlertsComponent, selector: "cui-alerts" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
26
27
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CuiRootComponent, decorators: [{
|
|
27
28
|
type: Component,
|
|
28
|
-
args: [{ selector:
|
|
29
|
+
args: [{ selector: CUI_ROOT_SELECTOR, changeDetection: ChangeDetectionStrategy.OnPush, providers: [CuiDestroyService], template: "<div class=\"c-content\">\n <ng-content />\n</div>\n<cui-dialogs />\n<cui-alerts />\n", styles: [".c-content{isolation:isolate}\n"] }]
|
|
29
30
|
}] });
|
|
30
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm9vdC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL2NvbXBvbmVudHMvcm9vdC9yb290LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvcmUvY29tcG9uZW50cy9yb290L3Jvb3QudGVtcGxhdGUuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBQ2pELE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFFakMsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ2pELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7O0FBU25ELE1BQU0sT0FBTyxnQkFBZ0I7SUFQN0I7UUFRbUIsYUFBUSxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM1QixXQUFNLEdBQUcsTUFBTSxDQUFDLGVBQWUsQ0FBQyxDQUFDO1FBQ2pDLGFBQVEsR0FBRyxNQUFNLENBQUMsaUJBQWlCLENBQUMsQ0FBQztLQVd2RDtJQVRRLFFBQVE7UUFDYixJQUFJLENBQUMscUJBQXFCLEVBQUUsQ0FBQztJQUMvQixDQUFDO0lBRU8scUJBQXFCO1FBQzNCLElBQUksQ0FBQyxNQUFNO2FBQ1IsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7YUFDOUIsU0FBUyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxlQUFlLENBQUMsWUFBWSxDQUFDLFVBQVUsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBQ3ZGLENBQUM7OzhHQWJVLGdCQUFnQjtrR0FBaEIsZ0JBQWdCLG1DQUZoQixDQUFDLGlCQUFpQixDQUFDLDBCQ2RoQyx3RkFLQTs0RkRXYSxnQkFBZ0I7a0JBUDVCLFNBQVM7K0JBQ0UsaUJBQWlCLG1CQUdWLHVCQUF1QixDQUFDLE1BQU0sYUFDcEMsQ0FBQyxpQkFBaUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IERPQ1VNRU5UIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEN1aURlc3Ryb3lTZXJ2aWNlIH0gZnJvbSAnQGN1YnktdWkvY2RrJztcbmltcG9ydCB7IHRha2VVbnRpbCB9IGZyb20gJ3J4anMnO1xuXG5pbXBvcnQgeyBDdWlUaGVtZVNlcnZpY2UgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcyc7XG5pbXBvcnQgeyBDVUlfUk9PVF9TRUxFQ1RPUiB9IGZyb20gJy4vcm9vdC5vcHRpb25zJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiBDVUlfUk9PVF9TRUxFQ1RPUixcbiAgdGVtcGxhdGVVcmw6ICcuL3Jvb3QudGVtcGxhdGUuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3Jvb3Quc3R5bGUuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgcHJvdmlkZXJzOiBbQ3VpRGVzdHJveVNlcnZpY2VdXG59KVxuZXhwb3J0IGNsYXNzIEN1aVJvb3RDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICBwcml2YXRlIHJlYWRvbmx5IGRvY3VtZW50ID0gaW5qZWN0KERPQ1VNRU5UKTtcbiAgcHJpdmF0ZSByZWFkb25seSB0aGVtZSQgPSBpbmplY3QoQ3VpVGhlbWVTZXJ2aWNlKTtcbiAgcHJpdmF0ZSByZWFkb25seSBkZXN0cm95JCA9IGluamVjdChDdWlEZXN0cm95U2VydmljZSk7XG5cbiAgcHVibGljIG5nT25Jbml0KCk6IHZvaWQge1xuICAgIHRoaXMuaW5pdFRoZW1lU3Vic2NyaXB0aW9uKCk7XG4gIH1cblxuICBwcml2YXRlIGluaXRUaGVtZVN1YnNjcmlwdGlvbigpOiB2b2lkIHtcbiAgICB0aGlzLnRoZW1lJFxuICAgICAgLnBpcGUodGFrZVVudGlsKHRoaXMuZGVzdHJveSQpKVxuICAgICAgLnN1YnNjcmliZSh0aGVtZSA9PiB0aGlzLmRvY3VtZW50LmRvY3VtZW50RWxlbWVudC5zZXRBdHRyaWJ1dGUoJ2N1aVRoZW1lJywgdGhlbWUpKTtcbiAgfVxufVxuIiwiPGRpdiBjbGFzcz1cImMtY29udGVudFwiPlxuICA8bmctY29udGVudCAvPlxuPC9kaXY+XG48Y3VpLWRpYWxvZ3MgLz5cbjxjdWktYWxlcnRzIC8+XG4iXX0=
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export const CUI_ROOT_SELECTOR = 'cui-root';
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm9vdC5vcHRpb25zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29yZS9jb21wb25lbnRzL3Jvb3Qvcm9vdC5vcHRpb25zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLGlCQUFpQixHQUFHLFVBQVUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBDVUlfUk9PVF9TRUxFQ1RPUiA9ICdjdWktcm9vdCc7Il19
|
|
@@ -1,10 +1,11 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, inject, Input, ElementRef, ViewChild, HostBinding } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, inject, Input, ElementRef, ViewChild, HostBinding, ViewContainerRef, HostListener } from '@angular/core';
|
|
2
2
|
import { DOCUMENT } from '@angular/common';
|
|
3
3
|
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
4
4
|
import { CuiDestroyService } from '@cuby-ui/cdk';
|
|
5
5
|
import { cuiProvide, CuiClickOutsideDirective } from '@cuby-ui/cdk';
|
|
6
6
|
import { takeUntil } from 'rxjs';
|
|
7
7
|
import { CUI_TEXT_FIELD_CONTROLLER, CUI_TEXT_FILED_CONTROLLER_PROVIDER } from '../../directives';
|
|
8
|
+
import { CUI_ROOT_SELECTOR } from '../root';
|
|
8
9
|
import * as i0 from "@angular/core";
|
|
9
10
|
import * as i1 from "@cuby-ui/cdk";
|
|
10
11
|
import * as i2 from "@angular/common";
|
|
@@ -14,6 +15,7 @@ export class CuiSelectComponent {
|
|
|
14
15
|
this.element = inject(ElementRef).nativeElement;
|
|
15
16
|
this.document = inject(DOCUMENT);
|
|
16
17
|
this.changeDetectorRef = inject(ChangeDetectorRef);
|
|
18
|
+
this.viewContainerRef = inject(ViewContainerRef);
|
|
17
19
|
this.destroy$ = inject(CuiDestroyService, { self: true });
|
|
18
20
|
this.cuiTextFieldController = inject(CUI_TEXT_FIELD_CONTROLLER);
|
|
19
21
|
this.cuiClickOutsideDirective = inject(CuiClickOutsideDirective, { self: true });
|
|
@@ -75,6 +77,10 @@ export class CuiSelectComponent {
|
|
|
75
77
|
onClose() {
|
|
76
78
|
this.close();
|
|
77
79
|
}
|
|
80
|
+
onWindowResize() {
|
|
81
|
+
this.button.nativeElement.blur();
|
|
82
|
+
this.close();
|
|
83
|
+
}
|
|
78
84
|
initClickOutsideSubscription() {
|
|
79
85
|
this.cuiClickOutsideDirective.cuiClickOutside
|
|
80
86
|
.pipe(takeUntil(this.destroy$))
|
|
@@ -87,37 +93,63 @@ export class CuiSelectComponent {
|
|
|
87
93
|
}
|
|
88
94
|
open() {
|
|
89
95
|
this.isOpened = true;
|
|
90
|
-
const { bottom: elementRectBottom } = this.element.getBoundingClientRect();
|
|
96
|
+
const { top: elementRectTop, left: elementRectLeft, bottom: elementRectBottom } = this.element.getBoundingClientRect();
|
|
91
97
|
const distanceToBottom = this.document.documentElement.clientHeight
|
|
92
98
|
- (elementRectBottom + this.SPACE_BETWEEN_BUTTON_AND_OPTIONS);
|
|
99
|
+
const embeddedViewRef = this.viewContainerRef.createEmbeddedView(this.optionsWrapper);
|
|
100
|
+
const node = embeddedViewRef.rootNodes[0];
|
|
101
|
+
const nodeStyles = node.style;
|
|
93
102
|
this.isOptionsListAbove = distanceToBottom < this.getOptionsListHeightContainer.nativeElement.offsetHeight;
|
|
103
|
+
embeddedViewRef.detectChanges();
|
|
104
|
+
nodeStyles.width = this.element.offsetWidth + 'px';
|
|
105
|
+
nodeStyles.left = window.scrollX + elementRectLeft + 'px';
|
|
106
|
+
if (this.isOptionsListAbove) {
|
|
107
|
+
nodeStyles.top = window.scrollY + elementRectTop
|
|
108
|
+
- this.getOptionsListHeightContainer.nativeElement.offsetHeight
|
|
109
|
+
- this.SPACE_BETWEEN_BUTTON_AND_OPTIONS
|
|
110
|
+
+ 'px';
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
nodeStyles.top = window.scrollY + elementRectBottom + this.SPACE_BETWEEN_BUTTON_AND_OPTIONS + 'px';
|
|
114
|
+
}
|
|
115
|
+
this.document.querySelector(CUI_ROOT_SELECTOR).appendChild(embeddedViewRef.rootNodes[0]);
|
|
94
116
|
}
|
|
95
117
|
close() {
|
|
96
118
|
this.isOpened = false;
|
|
119
|
+
this.viewContainerRef.clear();
|
|
97
120
|
}
|
|
98
121
|
}
|
|
99
122
|
CuiSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CuiSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
100
|
-
CuiSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CuiSelectComponent, selector: "cui-select[options]", inputs: { options: "options", defaultOptionText: "defaultOptionText" }, host: { properties: { "style.--c-gap": "this.gap" } }, providers: [
|
|
123
|
+
CuiSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CuiSelectComponent, selector: "cui-select[options]", inputs: { options: "options", defaultOptionText: "defaultOptionText" }, host: { listeners: { "window:resize": "onWindowResize()" }, properties: { "style.--c-gap": "this.gap" } }, providers: [
|
|
101
124
|
CUI_TEXT_FILED_CONTROLLER_PROVIDER,
|
|
102
125
|
cuiProvide(NG_VALUE_ACCESSOR, CuiSelectComponent, true),
|
|
103
126
|
CuiDestroyService
|
|
104
|
-
], viewQueries: [{ propertyName: "getOptionsListHeightContainer", first: true, predicate: ["getOptionsListHeightContainer"], descendants: true }], hostDirectives: [{ directive: i1.CuiClickOutsideDirective }], ngImport: i0, template: "<button\n type=\"button\"\n [attr.id]=\"id\"\n [attr.data-size]=\"size\"\n [disabled]=\"isDisabled\"\n class=\"c-button\"\n [class.c-button_with-error]=\"isError\"\n (click)=\"onSwitch()\"\n>\n <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n {{ selectedOption.label }}\n </ng-container>\n <ng-template #placeholderTemplate>\n <span class=\"c-placeholder\">{{ placeholder }}</span>\n </ng-template>\n <cui-svg\n icon=\"cuiIconChevronDown\"\n color=\"var(--cui-base-500)\"\n />\n</button>\n<
|
|
127
|
+
], viewQueries: [{ propertyName: "button", first: true, predicate: ["button"], descendants: true }, { propertyName: "getOptionsListHeightContainer", first: true, predicate: ["getOptionsListHeightContainer"], descendants: true }, { propertyName: "optionsWrapper", first: true, predicate: ["optionsWrapper"], descendants: true }], hostDirectives: [{ directive: i1.CuiClickOutsideDirective }], ngImport: i0, template: "<button\n #button\n type=\"button\"\n [attr.id]=\"id\"\n [attr.data-size]=\"size\"\n [disabled]=\"isDisabled\"\n class=\"c-button\"\n [class.c-button_active]=\"isOpened\"\n [class.c-button_with-error]=\"isError\"\n (click)=\"onSwitch()\"\n>\n <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n {{ selectedOption.label }}\n </ng-container>\n <ng-template #placeholderTemplate>\n <span class=\"c-placeholder\">{{ placeholder }}</span>\n </ng-template>\n <cui-svg\n icon=\"cuiIconChevronDown\"\n color=\"var(--cui-base-500)\"\n />\n</button>\n<ng-template #optionsWrapper>\n <div\n class=\"c-options__wrapper\"\n [class.c-options__wrapper_top]=\"isOptionsListAbove\"\n >\n <ng-container *ngTemplateOutlet=\"optionsList\"/>\n </div>\n</ng-template>\n<div\n #getOptionsListHeightContainer\n class=\"c-get-options-list-height-container\"\n>\n <ng-container *ngTemplateOutlet=\"optionsList\"/>\n</div>\n<ng-template #optionsList>\n <ul\n class=\"c-options\"\n [class.c-options_top]=\"isOptionsListAbove\"\n >\n <ng-container *ngIf=\"options.length; else empty\">\n <li *ngFor=\"let option of options; trackBy: trackByFn\">\n <button\n type=\"button\"\n class=\"c-option-button\"\n [class.c-option-button_selected]=\"selectedOption === option\"\n (click)=\"onSelect(option)\"\n >\n <span>{{ option.label }}</span>\n <cui-svg\n *ngIf=\"selectedOption === option\"\n icon=\"cuiIconCheckSm\"\n color=\"var(--cui-info)\"\n class=\"c-option-button__icon\"\n />\n </button>\n </li>\n </ng-container>\n <ng-template #empty>\n <li *ngIf=\"defaultOptionText\">\n <button\n type=\"button\"\n class=\"c-option-button\"\n (click)=\"onClose()\"\n >\n {{ defaultOptionText }}\n </button>\n </li>\n </ng-template>\n </ul>\n</ng-template>\n", styles: [":host{display:block;font-family:var(--cui-main-font);position:relative}.c-button{border:0;flex-shrink:0;outline:none;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:0 13px;font-weight:400;font-size:14px;line-height:20px;display:flex;justify-content:space-between;align-items:center;gap:8px;width:100%;border:1px solid var(--cui-base-200);border-radius:8px;cursor:pointer;background:var(--cui-input);color:var(--cui-base-900);font-family:var(--cui-main-font)}.c-button:hover{border-color:var(--cui-base-300)}.c-button_active.c-button_active,.c-button:focus{box-shadow:0 0 0 2px var(--cui-focus);border-color:var(--cui-info)}.c-button:disabled{cursor:not-allowed;opacity:.5;background:var(--cui-base-50);border-color:var(--cui-base-200)}.c-button[data-size=sm]{padding-top:7px;padding-bottom:7px}.c-button[data-size=md]{padding-top:8px;padding-bottom:8px}.c-button_with-error{border-color:var(--cui-danger)}.c-button_with-error:focus{box-shadow:0 0 0 2px #d92d2040;border-color:var(--cui-danger)}.c-placeholder{color:var(--cui-base-400)}.c-options__wrapper{position:absolute;font-family:var(--cui-main-font)}.c-get-options-list-height-container{position:fixed;visibility:hidden}.c-options{padding:7px 0;margin:0;list-style:none;font-weight:400;font-size:14px;line-height:20px;box-shadow:0 1px 4px #0000000a,0 1px 4px #0000000a;overflow:auto;max-height:180px;border:1px solid var(--cui-base-200);border-radius:8px;background:var(--cui-base-0);color:var(--cui-base-900)}.c-option-button{padding:8px 11px 8px 0;border:0;flex-shrink:0;outline:none;cursor:pointer;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:8px 11px;width:100%;display:flex;align-items:center;gap:12px}.c-option-button:active{background:var(--cui-base-50)}@media (hover: hover){.c-option-button:hover{background:var(--cui-base-50)}}.c-option-button_selected{background:var(--cui-base-50)}.c-option-button__icon{margin-left:auto}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i3.CuiSvgComponent, selector: "cui-svg[icon]", inputs: ["width", "height", "strokeWidth", "color", "icon"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
105
128
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CuiSelectComponent, decorators: [{
|
|
106
129
|
type: Component,
|
|
107
130
|
args: [{ selector: 'cui-select[options]', changeDetection: ChangeDetectionStrategy.OnPush, providers: [
|
|
108
131
|
CUI_TEXT_FILED_CONTROLLER_PROVIDER,
|
|
109
132
|
cuiProvide(NG_VALUE_ACCESSOR, CuiSelectComponent, true),
|
|
110
133
|
CuiDestroyService
|
|
111
|
-
], hostDirectives: [CuiClickOutsideDirective], template: "<button\n type=\"button\"\n [attr.id]=\"id\"\n [attr.data-size]=\"size\"\n [disabled]=\"isDisabled\"\n class=\"c-button\"\n [class.c-button_with-error]=\"isError\"\n (click)=\"onSwitch()\"\n>\n <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n {{ selectedOption.label }}\n </ng-container>\n <ng-template #placeholderTemplate>\n <span class=\"c-placeholder\">{{ placeholder }}</span>\n </ng-template>\n <cui-svg\n icon=\"cuiIconChevronDown\"\n color=\"var(--cui-base-500)\"\n />\n</button>\n<
|
|
134
|
+
], hostDirectives: [CuiClickOutsideDirective], template: "<button\n #button\n type=\"button\"\n [attr.id]=\"id\"\n [attr.data-size]=\"size\"\n [disabled]=\"isDisabled\"\n class=\"c-button\"\n [class.c-button_active]=\"isOpened\"\n [class.c-button_with-error]=\"isError\"\n (click)=\"onSwitch()\"\n>\n <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n {{ selectedOption.label }}\n </ng-container>\n <ng-template #placeholderTemplate>\n <span class=\"c-placeholder\">{{ placeholder }}</span>\n </ng-template>\n <cui-svg\n icon=\"cuiIconChevronDown\"\n color=\"var(--cui-base-500)\"\n />\n</button>\n<ng-template #optionsWrapper>\n <div\n class=\"c-options__wrapper\"\n [class.c-options__wrapper_top]=\"isOptionsListAbove\"\n >\n <ng-container *ngTemplateOutlet=\"optionsList\"/>\n </div>\n</ng-template>\n<div\n #getOptionsListHeightContainer\n class=\"c-get-options-list-height-container\"\n>\n <ng-container *ngTemplateOutlet=\"optionsList\"/>\n</div>\n<ng-template #optionsList>\n <ul\n class=\"c-options\"\n [class.c-options_top]=\"isOptionsListAbove\"\n >\n <ng-container *ngIf=\"options.length; else empty\">\n <li *ngFor=\"let option of options; trackBy: trackByFn\">\n <button\n type=\"button\"\n class=\"c-option-button\"\n [class.c-option-button_selected]=\"selectedOption === option\"\n (click)=\"onSelect(option)\"\n >\n <span>{{ option.label }}</span>\n <cui-svg\n *ngIf=\"selectedOption === option\"\n icon=\"cuiIconCheckSm\"\n color=\"var(--cui-info)\"\n class=\"c-option-button__icon\"\n />\n </button>\n </li>\n </ng-container>\n <ng-template #empty>\n <li *ngIf=\"defaultOptionText\">\n <button\n type=\"button\"\n class=\"c-option-button\"\n (click)=\"onClose()\"\n >\n {{ defaultOptionText }}\n </button>\n </li>\n </ng-template>\n </ul>\n</ng-template>\n", styles: [":host{display:block;font-family:var(--cui-main-font);position:relative}.c-button{border:0;flex-shrink:0;outline:none;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:0 13px;font-weight:400;font-size:14px;line-height:20px;display:flex;justify-content:space-between;align-items:center;gap:8px;width:100%;border:1px solid var(--cui-base-200);border-radius:8px;cursor:pointer;background:var(--cui-input);color:var(--cui-base-900);font-family:var(--cui-main-font)}.c-button:hover{border-color:var(--cui-base-300)}.c-button_active.c-button_active,.c-button:focus{box-shadow:0 0 0 2px var(--cui-focus);border-color:var(--cui-info)}.c-button:disabled{cursor:not-allowed;opacity:.5;background:var(--cui-base-50);border-color:var(--cui-base-200)}.c-button[data-size=sm]{padding-top:7px;padding-bottom:7px}.c-button[data-size=md]{padding-top:8px;padding-bottom:8px}.c-button_with-error{border-color:var(--cui-danger)}.c-button_with-error:focus{box-shadow:0 0 0 2px #d92d2040;border-color:var(--cui-danger)}.c-placeholder{color:var(--cui-base-400)}.c-options__wrapper{position:absolute;font-family:var(--cui-main-font)}.c-get-options-list-height-container{position:fixed;visibility:hidden}.c-options{padding:7px 0;margin:0;list-style:none;font-weight:400;font-size:14px;line-height:20px;box-shadow:0 1px 4px #0000000a,0 1px 4px #0000000a;overflow:auto;max-height:180px;border:1px solid var(--cui-base-200);border-radius:8px;background:var(--cui-base-0);color:var(--cui-base-900)}.c-option-button{padding:8px 11px 8px 0;border:0;flex-shrink:0;outline:none;cursor:pointer;appearance:none;color:inherit;font:inherit;background:inherit;text-decoration:none;-webkit-tap-highlight-color:transparent;padding:8px 11px;width:100%;display:flex;align-items:center;gap:12px}.c-option-button:active{background:var(--cui-base-50)}@media (hover: hover){.c-option-button:hover{background:var(--cui-base-50)}}.c-option-button_selected{background:var(--cui-base-50)}.c-option-button__icon{margin-left:auto}\n"] }]
|
|
112
135
|
}], propDecorators: { options: [{
|
|
113
136
|
type: Input
|
|
114
137
|
}], defaultOptionText: [{
|
|
115
138
|
type: Input
|
|
139
|
+
}], button: [{
|
|
140
|
+
type: ViewChild,
|
|
141
|
+
args: ['button']
|
|
116
142
|
}], getOptionsListHeightContainer: [{
|
|
117
143
|
type: ViewChild,
|
|
118
144
|
args: ['getOptionsListHeightContainer']
|
|
145
|
+
}], optionsWrapper: [{
|
|
146
|
+
type: ViewChild,
|
|
147
|
+
args: ['optionsWrapper']
|
|
119
148
|
}], gap: [{
|
|
120
149
|
type: HostBinding,
|
|
121
150
|
args: ['style.--c-gap']
|
|
151
|
+
}], onWindowResize: [{
|
|
152
|
+
type: HostListener,
|
|
153
|
+
args: ['window:resize']
|
|
122
154
|
}] } });
|
|
123
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../projects/core/components/select/select.component.ts","../../../../../projects/core/components/select/select.template.html"],"names":[],"mappings":"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","sourcesContent":["import type { OnInit } from '@angular/core';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  inject,\n  Input,\n  ElementRef,\n  ViewChild,\n  HostBinding\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport type { CuiOnChange, CuiOnTouched, CuiNullable } from '@cuby-ui/cdk';\nimport { CuiDestroyService } from '@cuby-ui/cdk';\nimport { cuiProvide, CuiClickOutsideDirective } from '@cuby-ui/cdk';\nimport { takeUntil } from 'rxjs';\n\nimport type { CuiOption } from '../../interfaces';\nimport type { CuiSizeMd, CuiSizeSm } from '../../types';\nimport { CUI_TEXT_FIELD_CONTROLLER, CUI_TEXT_FILED_CONTROLLER_PROVIDER } from '../../directives';\n\n@Component({\n  selector: 'cui-select[options]',\n  templateUrl: './select.template.html',\n  styleUrls: ['./select.style.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    CUI_TEXT_FILED_CONTROLLER_PROVIDER,\n    cuiProvide(NG_VALUE_ACCESSOR, CuiSelectComponent, true),\n    CuiDestroyService\n  ],\n  hostDirectives: [CuiClickOutsideDirective]\n})\nexport class CuiSelectComponent implements ControlValueAccessor, OnInit {\n  private readonly element = inject(ElementRef).nativeElement;\n  private readonly document = inject(DOCUMENT);\n  private readonly changeDetectorRef = inject(ChangeDetectorRef);\n  private readonly destroy$ = inject(CuiDestroyService, { self: true });\n  private readonly cuiTextFieldController = inject(CUI_TEXT_FIELD_CONTROLLER);\n  private readonly cuiClickOutsideDirective = inject(CuiClickOutsideDirective, { self: true });\n\n  protected readonly SPACE_BETWEEN_BUTTON_AND_OPTIONS = 3;\n\n  protected isOpened = false;\n  protected isOptionsListAbove = false;\n  protected value!: CuiNullable<unknown>;\n  protected onChange!: CuiOnChange<unknown>;\n  protected onTouched!: CuiOnTouched;\n  protected isDisabled!: boolean;\n  protected selectedOption?: CuiOption;\n\n  @Input()\n  public options!: CuiOption[];\n\n  @Input()\n  public defaultOptionText?: string;\n\n  @ViewChild('getOptionsListHeightContainer')\n  protected getOptionsListHeightContainer!: ElementRef<HTMLUListElement>;\n\n  @HostBinding('style.--c-gap')\n  protected gap = `${this.SPACE_BETWEEN_BUTTON_AND_OPTIONS}px`;\n\n  protected get id(): string | undefined {\n    return this.cuiTextFieldController.id;\n  }\n\n  protected get size(): CuiSizeSm | CuiSizeMd {\n    return this.cuiTextFieldController.size;\n  }\n\n  protected get placeholder(): string | undefined {\n    return this.cuiTextFieldController.placeholder;\n  }\n\n  protected get isError(): boolean {\n    return this.cuiTextFieldController.isError;\n  }\n\n  protected trackByFn(_: number, item: CuiOption): string {\n    return item.label;\n  }\n\n  public writeValue(value: unknown): void {\n    this.value = value;\n    this.selectedOption = value as CuiOption;\n\n    this.changeDetectorRef.markForCheck();\n  }\n\n  public registerOnChange(fn: CuiOnChange<unknown>): void {\n    this.onChange = fn;\n  }\n\n  public registerOnTouched(fn: CuiOnTouched): void {\n    this.onTouched = fn;\n  }\n\n  public ngOnInit(): void {\n    this.initClickOutsideSubscription();\n  }\n\n  public setDisabledState(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n\n    this.changeDetectorRef.markForCheck();\n  }\n\n  protected onSelect(option: CuiOption): void {\n    this.value = option.value;\n    this.selectedOption = option;\n\n    this.onChange(this.value);\n    this.onSwitch();\n    this.onTouched();\n  }\n\n  protected onSwitch(): void {\n    if (!this.options.length && !this.defaultOptionText) {\n      return;\n    }\n\n    if (!this.isOpened) {\n      this.open();\n\n      return;\n    }\n\n    this.close();\n  }\n\n  protected onClose(): void {\n    this.close();\n  }\n\n  private initClickOutsideSubscription(): void {\n    this.cuiClickOutsideDirective.cuiClickOutside\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(() => {\n        if (this.isOpened) {\n          this.onTouched();\n        }\n\n        this.close();\n      });\n  }\n\n  private open(): void {\n    this.isOpened = true;\n\n    const { bottom: elementRectBottom } = this.element.getBoundingClientRect();\n    const distanceToBottom = this.document.documentElement.clientHeight\n      - (elementRectBottom + this.SPACE_BETWEEN_BUTTON_AND_OPTIONS);\n\n    this.isOptionsListAbove = distanceToBottom < this.getOptionsListHeightContainer.nativeElement.offsetHeight;\n  }\n\n  private close(): void {\n    this.isOpened = false;\n  }\n}\n","<button\n  type=\"button\"\n  [attr.id]=\"id\"\n  [attr.data-size]=\"size\"\n  [disabled]=\"isDisabled\"\n  class=\"c-button\"\n  [class.c-button_with-error]=\"isError\"\n  (click)=\"onSwitch()\"\n>\n  <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n    {{ selectedOption.label }}\n  </ng-container>\n  <ng-template #placeholderTemplate>\n    <span class=\"c-placeholder\">{{ placeholder }}</span>\n  </ng-template>\n  <cui-svg\n    icon=\"cuiIconChevronDown\"\n    color=\"var(--cui-base-500)\"\n  />\n</button>\n<div\n  *ngIf=\"isOpened\"\n  class=\"c-options__wrapper\"\n  [class.c-options__wrapper_top]=\"isOptionsListAbove\"\n>\n  <ng-container *ngTemplateOutlet=\"optionsList\"/>\n</div>\n<div\n  #getOptionsListHeightContainer\n  class=\"c-get-options-list-height-container\"\n>\n  <ng-container *ngTemplateOutlet=\"optionsList\"/>\n</div>\n<ng-template #optionsList>\n  <ul\n    class=\"c-options\"\n    [class.c-options_top]=\"isOptionsListAbove\"\n  >\n    <ng-container *ngIf=\"options.length; else empty\">\n      <li *ngFor=\"let option of options; trackBy: trackByFn\">\n        <button\n          class=\"c-option-button\"\n          [class.c-option-button_selected]=\"selectedOption === option\"\n          (click)=\"onSelect(option)\"\n        >\n          <span>{{ option.label }}</span>\n          <cui-svg\n            *ngIf=\"selectedOption === option\"\n            icon=\"cuiIconCheckSm\"\n            color=\"var(--cui-info)\"\n            class=\"c-option-button__icon\"\n          />\n        </button>\n      </li>\n    </ng-container>\n    <ng-template #empty>\n      <li *ngIf=\"defaultOptionText\">\n        <button\n          class=\"c-option-button\"\n          (click)=\"onClose()\"\n        >\n          {{ defaultOptionText }}\n        </button>\n      </li>\n    </ng-template>\n  </ul>\n</ng-template>\n"]}
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../projects/core/components/select/select.component.ts","../../../../../projects/core/components/select/select.template.html"],"names":[],"mappings":"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","sourcesContent":["import type { OnInit, TemplateRef } from '@angular/core';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  inject,\n  Input,\n  ElementRef,\n  ViewChild,\n  HostBinding,\n  ViewContainerRef,\n  HostListener\n} from '@angular/core';\nimport { DOCUMENT } from '@angular/common';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport type { CuiOnChange, CuiOnTouched, CuiNullable } from '@cuby-ui/cdk';\nimport { CuiDestroyService } from '@cuby-ui/cdk';\nimport { cuiProvide, CuiClickOutsideDirective } from '@cuby-ui/cdk';\nimport { takeUntil } from 'rxjs';\n\nimport type { CuiOption } from '../../interfaces';\nimport type { CuiSizeMd, CuiSizeSm } from '../../types';\nimport { CUI_TEXT_FIELD_CONTROLLER, CUI_TEXT_FILED_CONTROLLER_PROVIDER } from '../../directives';\nimport { CUI_ROOT_SELECTOR } from '../root';\n\n@Component({\n  selector: 'cui-select[options]',\n  templateUrl: './select.template.html',\n  styleUrls: ['./select.style.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    CUI_TEXT_FILED_CONTROLLER_PROVIDER,\n    cuiProvide(NG_VALUE_ACCESSOR, CuiSelectComponent, true),\n    CuiDestroyService\n  ],\n  hostDirectives: [CuiClickOutsideDirective]\n})\nexport class CuiSelectComponent implements ControlValueAccessor, OnInit {\n  private readonly element = inject(ElementRef).nativeElement;\n  private readonly document = inject(DOCUMENT);\n  private readonly changeDetectorRef = inject(ChangeDetectorRef);\n  private readonly viewContainerRef = inject(ViewContainerRef);\n  private readonly destroy$ = inject(CuiDestroyService, {self: true});\n  private readonly cuiTextFieldController = inject(CUI_TEXT_FIELD_CONTROLLER);\n  private readonly cuiClickOutsideDirective = inject(CuiClickOutsideDirective, {self: true});\n\n  protected readonly SPACE_BETWEEN_BUTTON_AND_OPTIONS = 3;\n\n  protected isOpened = false;\n  protected isOptionsListAbove = false;\n  protected value!: CuiNullable<unknown>;\n  protected onChange!: CuiOnChange<unknown>;\n  protected onTouched!: CuiOnTouched;\n  protected isDisabled!: boolean;\n  protected selectedOption?: CuiOption;\n\n  @Input()\n  public options!: CuiOption[];\n\n  @Input()\n  public defaultOptionText?: string;\n\n  @ViewChild('button')\n  protected readonly button!: ElementRef<HTMLButtonElement>;\n\n  @ViewChild('getOptionsListHeightContainer')\n  protected readonly getOptionsListHeightContainer!: ElementRef<HTMLUListElement>;\n\n  @ViewChild('optionsWrapper')\n  protected readonly optionsWrapper!: TemplateRef<unknown>;\n\n  @HostBinding('style.--c-gap')\n  protected gap = `${this.SPACE_BETWEEN_BUTTON_AND_OPTIONS}px`;\n\n  protected get id(): string | undefined {\n    return this.cuiTextFieldController.id;\n  }\n\n  protected get size(): CuiSizeSm | CuiSizeMd {\n    return this.cuiTextFieldController.size;\n  }\n\n  protected get placeholder(): string | undefined {\n    return this.cuiTextFieldController.placeholder;\n  }\n\n  protected get isError(): boolean {\n    return this.cuiTextFieldController.isError;\n  }\n\n  protected trackByFn(_: number, item: CuiOption): string {\n    return item.label;\n  }\n\n  public writeValue(value: unknown): void {\n    this.value = value;\n    this.selectedOption = value as CuiOption;\n\n    this.changeDetectorRef.markForCheck();\n  }\n\n  public registerOnChange(fn: CuiOnChange<unknown>): void {\n    this.onChange = fn;\n  }\n\n  public registerOnTouched(fn: CuiOnTouched): void {\n    this.onTouched = fn;\n  }\n\n  public ngOnInit(): void {\n    this.initClickOutsideSubscription();\n  }\n\n  public setDisabledState(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n\n    this.changeDetectorRef.markForCheck();\n  }\n\n  protected onSelect(option: CuiOption): void {\n    this.value = option.value;\n    this.selectedOption = option;\n\n    this.onChange(this.value);\n    this.onSwitch();\n    this.onTouched();\n  }\n\n  protected onSwitch(): void {\n    if (!this.options.length && !this.defaultOptionText) {\n      return;\n    }\n\n    if (!this.isOpened) {\n      this.open();\n\n      return;\n    }\n\n    this.close();\n  }\n\n  protected onClose(): void {\n    this.close();\n  }\n\n  @HostListener('window:resize')\n  protected onWindowResize(): void {\n    this.button.nativeElement.blur();\n    this.close();\n  }\n\n  private initClickOutsideSubscription(): void {\n    this.cuiClickOutsideDirective.cuiClickOutside\n    .pipe(takeUntil(this.destroy$))\n    .subscribe(() => {\n      if (this.isOpened) {\n        this.onTouched();\n      }\n\n      this.close();\n    });\n  }\n\n  private open(): void {\n    this.isOpened = true;\n\n    const {\n      top: elementRectTop,\n      left: elementRectLeft,\n      bottom: elementRectBottom\n    } = this.element.getBoundingClientRect();\n    const distanceToBottom = this.document.documentElement.clientHeight\n      - (elementRectBottom + this.SPACE_BETWEEN_BUTTON_AND_OPTIONS);\n    const embeddedViewRef = this.viewContainerRef.createEmbeddedView(this.optionsWrapper);\n    const node = embeddedViewRef.rootNodes[0];\n    const nodeStyles = node.style;\n\n    this.isOptionsListAbove = distanceToBottom < this.getOptionsListHeightContainer.nativeElement.offsetHeight;\n\n    embeddedViewRef.detectChanges();\n\n    nodeStyles.width = this.element.offsetWidth + 'px';\n    nodeStyles.left = window.scrollX + elementRectLeft + 'px';\n\n    if (this.isOptionsListAbove) {\n      nodeStyles.top = window.scrollY + elementRectTop\n        - this.getOptionsListHeightContainer.nativeElement.offsetHeight\n        - this.SPACE_BETWEEN_BUTTON_AND_OPTIONS\n        + 'px';\n    } else {\n      nodeStyles.top = window.scrollY + elementRectBottom + this.SPACE_BETWEEN_BUTTON_AND_OPTIONS + 'px';\n    }\n\n    this.document.querySelector(CUI_ROOT_SELECTOR)!.appendChild(embeddedViewRef.rootNodes[0]);\n  }\n\n  private close(): void {\n    this.isOpened = false;\n\n    this.viewContainerRef.clear();\n  }\n}\n","<button\n  #button\n  type=\"button\"\n  [attr.id]=\"id\"\n  [attr.data-size]=\"size\"\n  [disabled]=\"isDisabled\"\n  class=\"c-button\"\n  [class.c-button_active]=\"isOpened\"\n  [class.c-button_with-error]=\"isError\"\n  (click)=\"onSwitch()\"\n>\n  <ng-container *ngIf=\"selectedOption; else placeholderTemplate\">\n    {{ selectedOption.label }}\n  </ng-container>\n  <ng-template #placeholderTemplate>\n    <span class=\"c-placeholder\">{{ placeholder }}</span>\n  </ng-template>\n  <cui-svg\n    icon=\"cuiIconChevronDown\"\n    color=\"var(--cui-base-500)\"\n  />\n</button>\n<ng-template #optionsWrapper>\n  <div\n      class=\"c-options__wrapper\"\n      [class.c-options__wrapper_top]=\"isOptionsListAbove\"\n  >\n    <ng-container *ngTemplateOutlet=\"optionsList\"/>\n  </div>\n</ng-template>\n<div\n  #getOptionsListHeightContainer\n  class=\"c-get-options-list-height-container\"\n>\n  <ng-container *ngTemplateOutlet=\"optionsList\"/>\n</div>\n<ng-template #optionsList>\n  <ul\n    class=\"c-options\"\n    [class.c-options_top]=\"isOptionsListAbove\"\n  >\n    <ng-container *ngIf=\"options.length; else empty\">\n      <li *ngFor=\"let option of options; trackBy: trackByFn\">\n        <button\n          type=\"button\"\n          class=\"c-option-button\"\n          [class.c-option-button_selected]=\"selectedOption === option\"\n          (click)=\"onSelect(option)\"\n        >\n          <span>{{ option.label }}</span>\n          <cui-svg\n            *ngIf=\"selectedOption === option\"\n            icon=\"cuiIconCheckSm\"\n            color=\"var(--cui-info)\"\n            class=\"c-option-button__icon\"\n          />\n        </button>\n      </li>\n    </ng-container>\n    <ng-template #empty>\n      <li *ngIf=\"defaultOptionText\">\n        <button\n          type=\"button\"\n          class=\"c-option-button\"\n          (click)=\"onClose()\"\n        >\n          {{ defaultOptionText }}\n        </button>\n      </li>\n    </ng-template>\n  </ul>\n</ng-template>\n"]}
|