@spartan-ng/brain 0.0.1-alpha.412 → 0.0.1-alpha.414
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/esm2022/select/index.mjs +15 -5
- package/esm2022/select/lib/brn-select-content.component.mjs +64 -45
- package/esm2022/select/lib/brn-select-content.token.mjs +9 -0
- package/esm2022/select/lib/brn-select-option.directive.mjs +49 -45
- package/esm2022/select/lib/brn-select-placeholder.directive.mjs +16 -0
- package/esm2022/select/lib/brn-select-trigger.directive.mjs +67 -0
- package/esm2022/select/lib/brn-select-value.component.mjs +58 -36
- package/esm2022/select/lib/brn-select-value.directive.mjs +16 -0
- package/esm2022/select/lib/brn-select.component.mjs +104 -178
- package/esm2022/select/lib/brn-select.token.mjs +9 -0
- package/fesm2022/spartan-ng-brain-select.mjs +378 -497
- package/fesm2022/spartan-ng-brain-select.mjs.map +1 -1
- package/package.json +1 -1
- package/select/index.d.ts +11 -5
- package/select/lib/brn-select-content.component.d.ts +17 -19
- package/select/lib/brn-select-content.token.d.ts +4 -0
- package/select/lib/brn-select-option.directive.d.ts +19 -19
- package/select/lib/brn-select-placeholder.directive.d.ts +8 -0
- package/select/lib/brn-select-trigger.directive.d.ts +20 -0
- package/select/lib/brn-select-value.component.d.ts +13 -6
- package/select/lib/brn-select-value.directive.d.ts +11 -0
- package/select/lib/brn-select.component.d.ts +38 -49
- package/select/lib/brn-select.token.d.ts +4 -0
- package/esm2022/select/lib/brn-select.service.mjs +0 -207
- package/select/lib/brn-select.service.d.ts +0 -71
|
@@ -1,41 +1,56 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { CdkListboxModule } from '@angular/cdk/listbox';
|
|
2
2
|
import { CdkConnectedOverlay, OverlayModule, } from '@angular/cdk/overlay';
|
|
3
|
-
import { ChangeDetectionStrategy, Component, computed, contentChild, contentChildren, inject, input,
|
|
3
|
+
import { ChangeDetectionStrategy, Component, Injector, afterNextRender, booleanAttribute, computed, contentChild, contentChildren, inject, input, model, numberAttribute, signal, viewChild, } from '@angular/core';
|
|
4
4
|
import { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
5
5
|
import { FormGroupDirective, NgControl, NgForm } from '@angular/forms';
|
|
6
6
|
import { provideExposedSideProviderExisting, provideExposesStateProviderExisting, } from '@spartan-ng/brain/core';
|
|
7
7
|
import { BrnFormFieldControl } from '@spartan-ng/brain/form-field';
|
|
8
8
|
import { ErrorStateMatcher, ErrorStateTracker } from '@spartan-ng/brain/forms';
|
|
9
9
|
import { BrnLabelDirective } from '@spartan-ng/brain/label';
|
|
10
|
-
import { Subject,
|
|
10
|
+
import { Subject, of } from 'rxjs';
|
|
11
11
|
import { delay, map, switchMap } from 'rxjs/operators';
|
|
12
12
|
import { BrnSelectContentComponent } from './brn-select-content.component';
|
|
13
|
-
import {
|
|
13
|
+
import { BrnSelectOptionDirective } from './brn-select-option.directive';
|
|
14
|
+
import { provideBrnSelect } from './brn-select.token';
|
|
14
15
|
import * as i0 from "@angular/core";
|
|
15
16
|
import * as i1 from "@angular/cdk/overlay";
|
|
16
17
|
let nextId = 0;
|
|
17
18
|
export class BrnSelectComponent {
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
19
|
+
_defaultErrorStateMatcher = inject(ErrorStateMatcher);
|
|
20
|
+
_parentForm = inject(NgForm, { optional: true });
|
|
21
|
+
_injector = inject(Injector);
|
|
22
|
+
_parentFormGroup = inject(FormGroupDirective, { optional: true });
|
|
23
|
+
ngControl = inject(NgControl, { optional: true, self: true });
|
|
24
|
+
id = input(`brn-select-${nextId++}`);
|
|
25
|
+
multiple = input(false, {
|
|
26
|
+
transform: booleanAttribute,
|
|
27
|
+
});
|
|
21
28
|
placeholder = input('');
|
|
22
|
-
disabled = input(false
|
|
29
|
+
disabled = input(false, {
|
|
30
|
+
transform: booleanAttribute,
|
|
31
|
+
});
|
|
23
32
|
dir = input('ltr');
|
|
24
|
-
|
|
33
|
+
closeDelay = input(100, {
|
|
34
|
+
transform: numberAttribute,
|
|
35
|
+
});
|
|
36
|
+
open = model(false);
|
|
37
|
+
value = model();
|
|
38
|
+
compareWith = input((o1, o2) => o1 === o2);
|
|
39
|
+
_formDisabled = signal(false);
|
|
40
|
+
/** Label provided by the consumer. */
|
|
25
41
|
selectLabel = contentChild(BrnLabelDirective, { descendants: false });
|
|
26
42
|
/** Overlay pane containing the options. */
|
|
27
43
|
selectContent = contentChild.required(BrnSelectContentComponent);
|
|
28
|
-
|
|
29
|
-
options
|
|
30
|
-
|
|
44
|
+
/** @internal */
|
|
45
|
+
options = contentChildren(BrnSelectOptionDirective, { descendants: true });
|
|
46
|
+
/** @internal Derive the selected options to filter out the unselected options */
|
|
47
|
+
selectedOptions = computed(() => this.options().filter((option) => option.selected()));
|
|
31
48
|
/** Overlay pane containing the options. */
|
|
32
|
-
_overlayDir = viewChild(CdkConnectedOverlay);
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
_delayedExpanded = toSignal(toObservable(this.
|
|
36
|
-
state = computed(() => (this.
|
|
37
|
-
openedChange = output();
|
|
38
|
-
valueChange = output();
|
|
49
|
+
_overlayDir = viewChild.required(CdkConnectedOverlay);
|
|
50
|
+
trigger = signal(null);
|
|
51
|
+
triggerWidth = signal(0);
|
|
52
|
+
_delayedExpanded = toSignal(toObservable(this.open).pipe(switchMap((expanded) => (!expanded ? of(expanded).pipe(delay(this.closeDelay())) : of(expanded))), takeUntilDestroyed()), { initialValue: false });
|
|
53
|
+
state = computed(() => (this.open() ? 'open' : 'closed'));
|
|
39
54
|
_positionChanges$ = new Subject();
|
|
40
55
|
side = toSignal(this._positionChanges$.pipe(map((change) =>
|
|
41
56
|
// todo: better translation or adjusting hlm to take that into account
|
|
@@ -44,14 +59,11 @@ export class BrnSelectComponent {
|
|
|
44
59
|
? 'left'
|
|
45
60
|
: 'right'
|
|
46
61
|
: change.connectionPair.originY)), { initialValue: 'bottom' });
|
|
47
|
-
|
|
48
|
-
labelProvided = signal(false);
|
|
49
|
-
ngControl = inject(NgControl, { optional: true, self: true });
|
|
62
|
+
labelId = computed(() => this.selectLabel()?.id ?? `${this.id()}--label`);
|
|
50
63
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
51
64
|
_onChange = () => { };
|
|
52
65
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
|
53
66
|
_onTouched = () => { };
|
|
54
|
-
_shouldEmitValueChange = signal(false);
|
|
55
67
|
/*
|
|
56
68
|
* This position config ensures that the top "start" corner of the overlay
|
|
57
69
|
* is aligned with with the top "start" of the origin by default (overlapping
|
|
@@ -85,113 +97,41 @@ export class BrnSelectComponent {
|
|
|
85
97
|
},
|
|
86
98
|
];
|
|
87
99
|
errorStateTracker;
|
|
88
|
-
_defaultErrorStateMatcher = inject(ErrorStateMatcher);
|
|
89
|
-
_parentForm = inject(NgForm, { optional: true });
|
|
90
|
-
_parentFormGroup = inject(FormGroupDirective, { optional: true });
|
|
91
100
|
errorState = computed(() => this.errorStateTracker.errorState());
|
|
92
|
-
writeValue$ = new Subject();
|
|
93
101
|
constructor() {
|
|
94
|
-
this._selectService.state.update((state) => ({
|
|
95
|
-
...state,
|
|
96
|
-
multiple: this.multiple,
|
|
97
|
-
placeholder: this.placeholder,
|
|
98
|
-
disabled: this.disabled,
|
|
99
|
-
disabledBySetDisabled: this._disabledFromSetDisabledState,
|
|
100
|
-
dir: this.dir,
|
|
101
|
-
}));
|
|
102
|
-
this.handleOptionChanges();
|
|
103
|
-
this.handleInitialOptionSelect();
|
|
104
|
-
this._selectService.state.update((state) => ({
|
|
105
|
-
...state,
|
|
106
|
-
id: `brn-select-${nextId++}`,
|
|
107
|
-
}));
|
|
108
102
|
if (this.ngControl !== null) {
|
|
109
103
|
this.ngControl.valueAccessor = this;
|
|
110
104
|
}
|
|
111
|
-
// Watch for Listbox Selection Changes to trigger Collapse and Value Change
|
|
112
|
-
this._selectService.listBoxValueChangeEvent$.pipe(takeUntilDestroyed()).subscribe(() => {
|
|
113
|
-
if (!this.multiple()) {
|
|
114
|
-
this.close();
|
|
115
|
-
}
|
|
116
|
-
// we set shouldEmitValueChange to true because we want to propagate the value change
|
|
117
|
-
// as a result of user interaction
|
|
118
|
-
this._shouldEmitValueChange.set(true);
|
|
119
|
-
});
|
|
120
|
-
/**
|
|
121
|
-
* Listening to value changes in order to trigger forms api on change
|
|
122
|
-
* ShouldEmitValueChange simply ensures we only propagate value change when a user makes a selection
|
|
123
|
-
* we don't propagate changes made from outside the component (ex. patch value or initial value from form control)
|
|
124
|
-
*/
|
|
125
|
-
toObservable(this._selectService.value).subscribe((value) => {
|
|
126
|
-
if (this._shouldEmitValueChange()) {
|
|
127
|
-
this._onChange((value ?? null));
|
|
128
|
-
this.valueChange.emit((value ?? null));
|
|
129
|
-
}
|
|
130
|
-
this._shouldEmitValueChange.set(true);
|
|
131
|
-
});
|
|
132
105
|
this.errorStateTracker = new ErrorStateTracker(this._defaultErrorStateMatcher, this.ngControl, this._parentFormGroup, this._parentForm);
|
|
133
106
|
}
|
|
134
|
-
ngAfterContentInit() {
|
|
135
|
-
// Check if Label Directive Provided and pass to service
|
|
136
|
-
const label = this.selectLabel();
|
|
137
|
-
if (label) {
|
|
138
|
-
this.labelProvided.set(true);
|
|
139
|
-
this._selectService.state.update((state) => ({
|
|
140
|
-
...state,
|
|
141
|
-
labelId: label.id(),
|
|
142
|
-
}));
|
|
143
|
-
}
|
|
144
|
-
else if (this.placeholder()) {
|
|
145
|
-
this._selectService.state.update((state) => ({
|
|
146
|
-
...state,
|
|
147
|
-
labelId: `${state.id}--label`,
|
|
148
|
-
}));
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
107
|
ngDoCheck() {
|
|
152
108
|
this.errorStateTracker.updateErrorState();
|
|
153
109
|
}
|
|
154
110
|
toggle() {
|
|
155
|
-
if (this.
|
|
156
|
-
this.
|
|
111
|
+
if (this.open()) {
|
|
112
|
+
this.hide();
|
|
157
113
|
}
|
|
158
114
|
else {
|
|
159
|
-
this.
|
|
115
|
+
this.show();
|
|
160
116
|
}
|
|
161
117
|
}
|
|
162
|
-
|
|
163
|
-
if (
|
|
118
|
+
show() {
|
|
119
|
+
if (this.open() || this.disabled() || this._formDisabled() || this.options()?.length == 0) {
|
|
164
120
|
return;
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
}));
|
|
169
|
-
this.openedChange.emit(true);
|
|
170
|
-
this._moveFocusToCDKList();
|
|
121
|
+
}
|
|
122
|
+
this.open.set(true);
|
|
123
|
+
afterNextRender(() => this.selectContent().focusList(), { injector: this._injector });
|
|
171
124
|
}
|
|
172
|
-
|
|
173
|
-
if (!this.
|
|
125
|
+
hide() {
|
|
126
|
+
if (!this.open())
|
|
174
127
|
return;
|
|
175
|
-
|
|
176
|
-
this._selectService.selectTrigger.focus();
|
|
177
|
-
}
|
|
178
|
-
this.openedChange.emit(false);
|
|
179
|
-
this._selectService.state.update((state) => ({
|
|
180
|
-
...state,
|
|
181
|
-
isExpanded: false,
|
|
182
|
-
}));
|
|
128
|
+
this.open.set(false);
|
|
183
129
|
this._onTouched();
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
return !this.isExpanded() && !this.disabled() && this.options()?.length > 0;
|
|
187
|
-
}
|
|
188
|
-
_moveFocusToCDKList() {
|
|
189
|
-
setTimeout(() => {
|
|
190
|
-
this.selectContent()?.focusList();
|
|
191
|
-
});
|
|
130
|
+
// restore focus to the trigger
|
|
131
|
+
this.trigger()?.focus();
|
|
192
132
|
}
|
|
193
133
|
writeValue(value) {
|
|
194
|
-
this.
|
|
134
|
+
this.value.set(value);
|
|
195
135
|
}
|
|
196
136
|
registerOnChange(fn) {
|
|
197
137
|
this._onChange = fn;
|
|
@@ -200,80 +140,65 @@ export class BrnSelectComponent {
|
|
|
200
140
|
this._onTouched = fn;
|
|
201
141
|
}
|
|
202
142
|
setDisabledState(isDisabled) {
|
|
203
|
-
this.
|
|
143
|
+
this._formDisabled.set(isDisabled);
|
|
204
144
|
}
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
.
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
}
|
|
221
|
-
});
|
|
145
|
+
selectOption(value) {
|
|
146
|
+
// if this is a multiple select we need to add the value to the array
|
|
147
|
+
if (this.multiple()) {
|
|
148
|
+
const currentValue = this.value();
|
|
149
|
+
const newValue = currentValue ? [...currentValue, value] : [value];
|
|
150
|
+
this.value.set(newValue);
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
this.value.set(value);
|
|
154
|
+
}
|
|
155
|
+
this._onChange?.(this.value());
|
|
156
|
+
// if this is single select close the dropdown
|
|
157
|
+
if (!this.multiple()) {
|
|
158
|
+
this.hide();
|
|
159
|
+
}
|
|
222
160
|
}
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
this._selectService.updatePossibleOptions(options);
|
|
234
|
-
});
|
|
161
|
+
deselectOption(value) {
|
|
162
|
+
if (this.multiple()) {
|
|
163
|
+
const currentValue = this.value();
|
|
164
|
+
const newValue = currentValue.filter((val) => val !== value);
|
|
165
|
+
this.value.set(newValue);
|
|
166
|
+
}
|
|
167
|
+
else {
|
|
168
|
+
this.value.set(null);
|
|
169
|
+
}
|
|
170
|
+
this._onChange?.(this.value());
|
|
235
171
|
}
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
handleInvalidOptions(options) {
|
|
240
|
-
const selectedOptions = this._selectService.selectedOptions();
|
|
241
|
-
const availableOptionSet = new Set(options);
|
|
242
|
-
if (this._selectService.multiple()) {
|
|
243
|
-
const filteredOptions = selectedOptions.filter((o) => availableOptionSet.has(o));
|
|
244
|
-
if (selectedOptions.length !== filteredOptions.length) {
|
|
245
|
-
const value = filteredOptions.map((o) => o?.value ?? '');
|
|
246
|
-
this._selectService.state.update((state) => ({
|
|
247
|
-
...state,
|
|
248
|
-
selectedOptions: filteredOptions,
|
|
249
|
-
value: value,
|
|
250
|
-
}));
|
|
251
|
-
}
|
|
172
|
+
toggleSelect(value) {
|
|
173
|
+
if (this.isSelected(value)) {
|
|
174
|
+
this.deselectOption(value);
|
|
252
175
|
}
|
|
253
176
|
else {
|
|
254
|
-
|
|
255
|
-
if (selectedOption !== null && !availableOptionSet.has(selectedOption)) {
|
|
256
|
-
this._selectService.state.update((state) => ({
|
|
257
|
-
...state,
|
|
258
|
-
selectedOptions: [],
|
|
259
|
-
value: '',
|
|
260
|
-
}));
|
|
261
|
-
}
|
|
177
|
+
this.selectOption(value);
|
|
262
178
|
}
|
|
263
179
|
}
|
|
180
|
+
isSelected(value) {
|
|
181
|
+
const selection = this.value();
|
|
182
|
+
if (Array.isArray(selection)) {
|
|
183
|
+
return selection.some((val) => this.compareWith()(val, value));
|
|
184
|
+
}
|
|
185
|
+
else if (value !== undefined) {
|
|
186
|
+
return this.compareWith()(selection, value);
|
|
187
|
+
}
|
|
188
|
+
return false;
|
|
189
|
+
}
|
|
264
190
|
/** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.5", ngImport: i0, type: BrnSelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
265
|
-
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.5", type: BrnSelectComponent, isStandalone: true, selector: "brn-select, hlm-select", inputs: { multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: true, isRequired: false, transformFunction: null }, placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, dir: { classPropertyName: "dir", publicName: "dir", isSignal: true, isRequired: false, transformFunction: null }, closeDelay: { classPropertyName: "closeDelay", publicName: "closeDelay", isSignal: true, isRequired: false, transformFunction: null } }, outputs: {
|
|
266
|
-
BrnSelectService,
|
|
267
|
-
CdkListbox,
|
|
191
|
+
/** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.5", type: BrnSelectComponent, isStandalone: true, selector: "brn-select, hlm-select", inputs: { id: { classPropertyName: "id", publicName: "id", isSignal: true, isRequired: false, transformFunction: null }, multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: true, isRequired: false, transformFunction: null }, placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, dir: { classPropertyName: "dir", publicName: "dir", isSignal: true, isRequired: false, transformFunction: null }, closeDelay: { classPropertyName: "closeDelay", publicName: "closeDelay", isSignal: true, isRequired: false, transformFunction: null }, open: { classPropertyName: "open", publicName: "open", isSignal: true, isRequired: false, transformFunction: null }, value: { classPropertyName: "value", publicName: "value", isSignal: true, isRequired: false, transformFunction: null }, compareWith: { classPropertyName: "compareWith", publicName: "compareWith", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { open: "openChange", value: "valueChange" }, providers: [
|
|
268
192
|
provideExposedSideProviderExisting((() => BrnSelectComponent)),
|
|
269
193
|
provideExposesStateProviderExisting((() => BrnSelectComponent)),
|
|
194
|
+
provideBrnSelect(BrnSelectComponent),
|
|
270
195
|
{
|
|
271
196
|
provide: BrnFormFieldControl,
|
|
272
197
|
useExisting: BrnSelectComponent,
|
|
273
198
|
},
|
|
274
|
-
], queries: [{ propertyName: "selectLabel", first: true, predicate: BrnLabelDirective, isSignal: true }, { propertyName: "selectContent", first: true, predicate: BrnSelectContentComponent, descendants: true, isSignal: true }, { propertyName: "options", predicate:
|
|
275
|
-
@if (!
|
|
276
|
-
<label class="hidden" [attr.id]="
|
|
199
|
+
], queries: [{ propertyName: "selectLabel", first: true, predicate: BrnLabelDirective, isSignal: true }, { propertyName: "selectContent", first: true, predicate: BrnSelectContentComponent, descendants: true, isSignal: true }, { propertyName: "options", predicate: BrnSelectOptionDirective, descendants: true, isSignal: true }], viewQueries: [{ propertyName: "_overlayDir", first: true, predicate: CdkConnectedOverlay, descendants: true, isSignal: true }], ngImport: i0, template: `
|
|
200
|
+
@if (!selectLabel() && placeholder()) {
|
|
201
|
+
<label class="hidden" [attr.id]="labelId()">{{ placeholder() }}</label>
|
|
277
202
|
} @else {
|
|
278
203
|
<ng-content select="label[hlmLabel],label[brnLabel]" />
|
|
279
204
|
}
|
|
@@ -281,6 +206,7 @@ export class BrnSelectComponent {
|
|
|
281
206
|
<div cdk-overlay-origin (click)="toggle()" #trigger="cdkOverlayOrigin">
|
|
282
207
|
<ng-content select="hlm-select-trigger,[brnSelectTrigger]" />
|
|
283
208
|
</div>
|
|
209
|
+
|
|
284
210
|
<ng-template
|
|
285
211
|
cdk-connected-overlay
|
|
286
212
|
cdkConnectedOverlayLockPosition
|
|
@@ -290,8 +216,8 @@ export class BrnSelectComponent {
|
|
|
290
216
|
[cdkConnectedOverlayOpen]="_delayedExpanded()"
|
|
291
217
|
[cdkConnectedOverlayPositions]="_positions"
|
|
292
218
|
[cdkConnectedOverlayWidth]="triggerWidth() > 0 ? triggerWidth() : 'auto'"
|
|
293
|
-
(backdropClick)="
|
|
294
|
-
(detach)="
|
|
219
|
+
(backdropClick)="hide()"
|
|
220
|
+
(detach)="hide()"
|
|
295
221
|
(positionChange)="_positionChanges$.next($event)"
|
|
296
222
|
>
|
|
297
223
|
<ng-content />
|
|
@@ -306,18 +232,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
306
232
|
imports: [OverlayModule, CdkListboxModule],
|
|
307
233
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
308
234
|
providers: [
|
|
309
|
-
BrnSelectService,
|
|
310
|
-
CdkListbox,
|
|
311
235
|
provideExposedSideProviderExisting((() => BrnSelectComponent)),
|
|
312
236
|
provideExposesStateProviderExisting((() => BrnSelectComponent)),
|
|
237
|
+
provideBrnSelect(BrnSelectComponent),
|
|
313
238
|
{
|
|
314
239
|
provide: BrnFormFieldControl,
|
|
315
240
|
useExisting: BrnSelectComponent,
|
|
316
241
|
},
|
|
317
242
|
],
|
|
318
243
|
template: `
|
|
319
|
-
@if (!
|
|
320
|
-
<label class="hidden" [attr.id]="
|
|
244
|
+
@if (!selectLabel() && placeholder()) {
|
|
245
|
+
<label class="hidden" [attr.id]="labelId()">{{ placeholder() }}</label>
|
|
321
246
|
} @else {
|
|
322
247
|
<ng-content select="label[hlmLabel],label[brnLabel]" />
|
|
323
248
|
}
|
|
@@ -325,6 +250,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
325
250
|
<div cdk-overlay-origin (click)="toggle()" #trigger="cdkOverlayOrigin">
|
|
326
251
|
<ng-content select="hlm-select-trigger,[brnSelectTrigger]" />
|
|
327
252
|
</div>
|
|
253
|
+
|
|
328
254
|
<ng-template
|
|
329
255
|
cdk-connected-overlay
|
|
330
256
|
cdkConnectedOverlayLockPosition
|
|
@@ -334,8 +260,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
334
260
|
[cdkConnectedOverlayOpen]="_delayedExpanded()"
|
|
335
261
|
[cdkConnectedOverlayPositions]="_positions"
|
|
336
262
|
[cdkConnectedOverlayWidth]="triggerWidth() > 0 ? triggerWidth() : 'auto'"
|
|
337
|
-
(backdropClick)="
|
|
338
|
-
(detach)="
|
|
263
|
+
(backdropClick)="hide()"
|
|
264
|
+
(detach)="hide()"
|
|
339
265
|
(positionChange)="_positionChanges$.next($event)"
|
|
340
266
|
>
|
|
341
267
|
<ng-content />
|
|
@@ -343,4 +269,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.5", ngImpor
|
|
|
343
269
|
`,
|
|
344
270
|
}]
|
|
345
271
|
}], ctorParameters: () => [] });
|
|
346
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"brn-select.component.js","sourceRoot":"","sources":["../../../../../../libs/brain/select/src/lib/brn-select.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CdkListbox, CdkListboxModule, CdkOption } from '@angular/cdk/listbox';\nimport {\n\tCdkConnectedOverlay,\n\ttype ConnectedOverlayPositionChange,\n\ttype ConnectedPosition,\n\tOverlayModule,\n} from '@angular/cdk/overlay';\nimport {\n\tAfterContentInit,\n\tChangeDetectionStrategy,\n\tComponent,\n\ttype DoCheck,\n\ttype Signal,\n\tcomputed,\n\tcontentChild,\n\tcontentChildren,\n\tinject,\n\tinput,\n\toutput,\n\tsignal,\n\tviewChild,\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { type ControlValueAccessor, FormGroupDirective, NgControl, NgForm } from '@angular/forms';\nimport {\n\ttype ExposesSide,\n\ttype ExposesState,\n\tprovideExposedSideProviderExisting,\n\tprovideExposesStateProviderExisting,\n} from '@spartan-ng/brain/core';\nimport { BrnFormFieldControl } from '@spartan-ng/brain/form-field';\nimport { ChangeFn, ErrorStateMatcher, ErrorStateTracker, TouchFn } from '@spartan-ng/brain/forms';\nimport { BrnLabelDirective } from '@spartan-ng/brain/label';\nimport { Subject, combineLatest, of } from 'rxjs';\nimport { delay, map, switchMap } from 'rxjs/operators';\nimport { BrnSelectContentComponent } from './brn-select-content.component';\nimport { BrnSelectService } from './brn-select.service';\n\nexport type BrnReadDirection = 'ltr' | 'rtl';\n\nlet nextId = 0;\n\n@Component({\n\tselector: 'brn-select, hlm-select',\n\tstandalone: true,\n\timports: [OverlayModule, CdkListboxModule],\n\tchangeDetection: ChangeDetectionStrategy.OnPush,\n\tproviders: [\n\t\tBrnSelectService,\n\t\tCdkListbox,\n\t\tprovideExposedSideProviderExisting(() => BrnSelectComponent),\n\t\tprovideExposesStateProviderExisting(() => BrnSelectComponent),\n\t\t{\n\t\t\tprovide: BrnFormFieldControl,\n\t\t\tuseExisting: BrnSelectComponent,\n\t\t},\n\t],\n\ttemplate: `\n\t\t@if (!labelProvided() && placeholder()) {\n\t\t\t<label class=\"hidden\" [attr.id]=\"backupLabelId()\">{{ placeholder() }}</label>\n\t\t} @else {\n\t\t\t<ng-content select=\"label[hlmLabel],label[brnLabel]\" />\n\t\t}\n\n\t\t<div cdk-overlay-origin (click)=\"toggle()\" #trigger=\"cdkOverlayOrigin\">\n\t\t\t<ng-content select=\"hlm-select-trigger,[brnSelectTrigger]\" />\n\t\t</div>\n\t\t<ng-template\n\t\t\tcdk-connected-overlay\n\t\t\tcdkConnectedOverlayLockPosition\n\t\t\tcdkConnectedOverlayHasBackdrop\n\t\t\tcdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n\t\t\t[cdkConnectedOverlayOrigin]=\"trigger\"\n\t\t\t[cdkConnectedOverlayOpen]=\"_delayedExpanded()\"\n\t\t\t[cdkConnectedOverlayPositions]=\"_positions\"\n\t\t\t[cdkConnectedOverlayWidth]=\"triggerWidth() > 0 ? triggerWidth() : 'auto'\"\n\t\t\t(backdropClick)=\"close()\"\n\t\t\t(detach)=\"close()\"\n\t\t\t(positionChange)=\"_positionChanges$.next($event)\"\n\t\t>\n\t\t\t<ng-content />\n\t\t</ng-template>\n\t`,\n})\nexport class BrnSelectComponent<T = unknown>\n\timplements ControlValueAccessor, AfterContentInit, DoCheck, ExposesSide, ExposesState, BrnFormFieldControl\n{\n\tprivate readonly _selectService = inject(BrnSelectService);\n\n\tpublic readonly triggerWidth = this._selectService.triggerWidth;\n\n\tpublic readonly multiple = input<boolean>(false);\n\tpublic readonly placeholder = input<string>('');\n\tpublic readonly disabled = input<boolean>(false);\n\tpublic readonly dir = input<BrnReadDirection>('ltr');\n\tprivate readonly _disabledFromSetDisabledState = signal(false);\n\n\tprotected selectLabel = contentChild(BrnLabelDirective, { descendants: false });\n\t/** Overlay pane containing the options. */\n\tprotected selectContent = contentChild.required(BrnSelectContentComponent);\n\n\tprotected options = contentChildren(CdkOption, { descendants: true });\n\tprotected options$ = toObservable(this.options);\n\tprotected optionsAndIndex$ = this.options$.pipe(map((options, index) => [options, index] as const));\n\n\t/** Overlay pane containing the options. */\n\tprotected _overlayDir = viewChild(CdkConnectedOverlay);\n\n\tpublic readonly closeDelay = input<number>(100);\n\tpublic readonly isExpanded = this._selectService.isExpanded;\n\tprotected readonly _delayedExpanded = toSignal(\n\t\ttoObservable(this.isExpanded).pipe(\n\t\t\tswitchMap((expanded) => (!expanded ? of(expanded).pipe(delay(this.closeDelay())) : of(expanded))),\n\t\t\ttakeUntilDestroyed(),\n\t\t),\n\t\t{ initialValue: false },\n\t);\n\tpublic readonly state = computed(() => (this.isExpanded() ? 'open' : 'closed'));\n\n\tpublic readonly openedChange = output<boolean>();\n\tpublic readonly valueChange = output<T>();\n\n\tprotected readonly _positionChanges$ = new Subject<ConnectedOverlayPositionChange>();\n\tpublic readonly side: Signal<'top' | 'bottom' | 'left' | 'right'> = toSignal(\n\t\tthis._positionChanges$.pipe(\n\t\t\tmap<ConnectedOverlayPositionChange, 'top' | 'bottom' | 'left' | 'right'>((change) =>\n\t\t\t\t// todo: better translation or adjusting hlm to take that into account\n\t\t\t\tchange.connectionPair.originY === 'center'\n\t\t\t\t\t? change.connectionPair.originX === 'start'\n\t\t\t\t\t\t? 'left'\n\t\t\t\t\t\t: 'right'\n\t\t\t\t\t: change.connectionPair.originY,\n\t\t\t),\n\t\t),\n\t\t{ initialValue: 'bottom' },\n\t);\n\n\tpublic readonly backupLabelId = computed(() => this._selectService.labelId());\n\tpublic readonly labelProvided = signal(false);\n\n\tpublic readonly ngControl = inject(NgControl, { optional: true, self: true });\n\n\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\tprivate _onChange: ChangeFn<T> = () => {};\n\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\tprivate _onTouched: TouchFn = () => {};\n\n\tprivate readonly _shouldEmitValueChange = signal(false);\n\n\t/*\n\t * This position config ensures that the top \"start\" corner of the overlay\n\t * is aligned with with the top \"start\" of the origin by default (overlapping\n\t * the trigger completely). If the panel cannot fit below the trigger, it\n\t * will fall back to a position above the trigger.\n\t */\n\tprotected _positions: ConnectedPosition[] = [\n\t\t{\n\t\t\toriginX: 'start',\n\t\t\toriginY: 'bottom',\n\t\t\toverlayX: 'start',\n\t\t\toverlayY: 'top',\n\t\t},\n\t\t{\n\t\t\toriginX: 'end',\n\t\t\toriginY: 'bottom',\n\t\t\toverlayX: 'end',\n\t\t\toverlayY: 'top',\n\t\t},\n\t\t{\n\t\t\toriginX: 'start',\n\t\t\toriginY: 'top',\n\t\t\toverlayX: 'start',\n\t\t\toverlayY: 'bottom',\n\t\t},\n\t\t{\n\t\t\toriginX: 'end',\n\t\t\toriginY: 'top',\n\t\t\toverlayX: 'end',\n\t\t\toverlayY: 'bottom',\n\t\t},\n\t];\n\n\tpublic errorStateTracker: ErrorStateTracker;\n\n\tprivate readonly _defaultErrorStateMatcher = inject(ErrorStateMatcher);\n\tprivate readonly _parentForm = inject(NgForm, { optional: true });\n\tprivate readonly _parentFormGroup = inject(FormGroupDirective, { optional: true });\n\n\tpublic errorState = computed(() => this.errorStateTracker.errorState());\n\n\tpublic writeValue$ = new Subject<T>();\n\n\tconstructor() {\n\t\tthis._selectService.state.update((state) => ({\n\t\t\t...state,\n\t\t\tmultiple: this.multiple,\n\t\t\tplaceholder: this.placeholder,\n\t\t\tdisabled: this.disabled,\n\t\t\tdisabledBySetDisabled: this._disabledFromSetDisabledState,\n\t\t\tdir: this.dir,\n\t\t}));\n\t\tthis.handleOptionChanges();\n\t\tthis.handleInitialOptionSelect();\n\n\t\tthis._selectService.state.update((state) => ({\n\t\t\t...state,\n\t\t\tid: `brn-select-${nextId++}`,\n\t\t}));\n\t\tif (this.ngControl !== null) {\n\t\t\tthis.ngControl.valueAccessor = this;\n\t\t}\n\n\t\t// Watch for Listbox Selection Changes to trigger Collapse and Value Change\n\t\tthis._selectService.listBoxValueChangeEvent$.pipe(takeUntilDestroyed()).subscribe(() => {\n\t\t\tif (!this.multiple()) {\n\t\t\t\tthis.close();\n\t\t\t}\n\n\t\t\t// we set shouldEmitValueChange to true because we want to propagate the value change\n\t\t\t// as a result of user interaction\n\t\t\tthis._shouldEmitValueChange.set(true);\n\t\t});\n\n\t\t/**\n\t\t * Listening to value changes in order to trigger forms api on change\n\t\t * ShouldEmitValueChange simply ensures we only propagate value change when a user makes a selection\n\t\t * we don't propagate changes made from outside the component (ex. patch value or initial value from form control)\n\t\t */\n\t\ttoObservable(this._selectService.value).subscribe((value) => {\n\t\t\tif (this._shouldEmitValueChange()) {\n\t\t\t\tthis._onChange((value ?? null) as T);\n\t\t\t\tthis.valueChange.emit((value ?? null) as T);\n\t\t\t}\n\t\t\tthis._shouldEmitValueChange.set(true);\n\t\t});\n\n\t\tthis.errorStateTracker = new ErrorStateTracker(\n\t\t\tthis._defaultErrorStateMatcher,\n\t\t\tthis.ngControl,\n\t\t\tthis._parentFormGroup,\n\t\t\tthis._parentForm,\n\t\t);\n\t}\n\n\tpublic ngAfterContentInit(): void {\n\t\t// Check if Label Directive Provided and pass to service\n\t\tconst label = this.selectLabel();\n\t\tif (label) {\n\t\t\tthis.labelProvided.set(true);\n\t\t\tthis._selectService.state.update((state) => ({\n\t\t\t\t...state,\n\t\t\t\tlabelId: label.id(),\n\t\t\t}));\n\t\t} else if (this.placeholder()) {\n\t\t\tthis._selectService.state.update((state) => ({\n\t\t\t\t...state,\n\t\t\t\tlabelId: `${state.id}--label`,\n\t\t\t}));\n\t\t}\n\t}\n\n\tngDoCheck() {\n\t\tthis.errorStateTracker.updateErrorState();\n\t}\n\n\tpublic toggle(): void {\n\t\tif (this.isExpanded()) {\n\t\t\tthis.close();\n\t\t} else {\n\t\t\tthis.open();\n\t\t}\n\t}\n\n\tpublic open(): void {\n\t\tif (!this._canOpen()) return;\n\t\tthis._selectService.state.update((state) => ({\n\t\t\t...state,\n\t\t\tisExpanded: true,\n\t\t}));\n\t\tthis.openedChange.emit(true);\n\t\tthis._moveFocusToCDKList();\n\t}\n\n\tpublic close(): void {\n\t\tif (!this.isExpanded()) return;\n\n\t\tif (this._selectService.selectTrigger) {\n\t\t\tthis._selectService.selectTrigger.focus();\n\t\t}\n\n\t\tthis.openedChange.emit(false);\n\t\tthis._selectService.state.update((state) => ({\n\t\t\t...state,\n\t\t\tisExpanded: false,\n\t\t}));\n\t\tthis._onTouched();\n\t}\n\n\tprotected _canOpen(): boolean {\n\t\treturn !this.isExpanded() && !this.disabled() && this.options()?.length > 0;\n\t}\n\n\tprivate _moveFocusToCDKList(): void {\n\t\tsetTimeout(() => {\n\t\t\tthis.selectContent()?.focusList();\n\t\t});\n\t}\n\n\tpublic writeValue(value: T): void {\n\t\tthis.writeValue$.next(value);\n\t}\n\n\tpublic registerOnChange(fn: ChangeFn<T>): void {\n\t\tthis._onChange = fn;\n\t}\n\n\tpublic registerOnTouched(fn: TouchFn): void {\n\t\tthis._onTouched = fn;\n\t}\n\n\tpublic setDisabledState(isDisabled: boolean) {\n\t\tthis._disabledFromSetDisabledState.set(isDisabled);\n\t}\n\n\t/**\n\t * Once writeValue is called and options are available we can handle setting the initial options\n\t * @private\n\t */\n\tprivate handleInitialOptionSelect() {\n\t\t// Write value cannot be handled until options are available, so we wait until both are available with a combineLatest\n\t\tcombineLatest([this.writeValue$, this.options$])\n\t\t\t.pipe(\n\t\t\t\tmap((values, index) => [...values, index]),\n\t\t\t\ttakeUntilDestroyed(),\n\t\t\t)\n\t\t\t.subscribe(([value, _, index]) => {\n\t\t\t\tthis._shouldEmitValueChange.set(false);\n\t\t\t\tthis._selectService.setInitialSelectedOptions(value);\n\t\t\t\t// the first time this observable emits a value we are simply setting the initial state\n\t\t\t\t// this change should not count as changing the state of the select, so we need to mark as pristine\n\t\t\t\tif (index === 0) {\n\t\t\t\t\tthis.ngControl?.control?.markAsPristine();\n\t\t\t\t}\n\t\t\t});\n\t}\n\n\t/**\n\t * When options change, our current selected options may become invalid\n\t * Here we will automatically update our current selected options so that they are always inline with the possibleOptions\n\t * @private\n\t */\n\tprivate handleOptionChanges() {\n\t\tthis.optionsAndIndex$.pipe(takeUntilDestroyed()).subscribe(([options, index]) => {\n\t\t\tif (index > 0) {\n\t\t\t\tthis.handleInvalidOptions(options);\n\t\t\t}\n\t\t\tthis._selectService.updatePossibleOptions(options);\n\t\t});\n\t}\n\n\t/**\n\t * Check that our \"selectedOptions\" are still valid when \"possibleOptions\" is about to be updated\n\t */\n\tprivate handleInvalidOptions(options: readonly CdkOption[]) {\n\t\tconst selectedOptions = this._selectService.selectedOptions();\n\t\tconst availableOptionSet = new Set<CdkOption | null>(options);\n\t\tif (this._selectService.multiple()) {\n\t\t\tconst filteredOptions = selectedOptions.filter((o) => availableOptionSet.has(o));\n\t\t\tif (selectedOptions.length !== filteredOptions.length) {\n\t\t\t\tconst value = filteredOptions.map((o) => (o?.value as string) ?? '');\n\t\t\t\tthis._selectService.state.update((state) => ({\n\t\t\t\t\t...state,\n\t\t\t\t\tselectedOptions: filteredOptions,\n\t\t\t\t\tvalue: value,\n\t\t\t\t}));\n\t\t\t}\n\t\t} else {\n\t\t\tconst selectedOption = selectedOptions[0] ?? null;\n\t\t\tif (selectedOption !== null && !availableOptionSet.has(selectedOption)) {\n\t\t\t\tthis._selectService.state.update((state) => ({\n\t\t\t\t\t...state,\n\t\t\t\t\tselectedOptions: [],\n\t\t\t\t\tvalue: '',\n\t\t\t\t}));\n\t\t\t}\n\t\t}\n\t}\n}\n"]}
|
|
272
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"brn-select.component.js","sourceRoot":"","sources":["../../../../../../libs/brain/select/src/lib/brn-select.component.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EACN,mBAAmB,EAGnB,aAAa,GACb,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EACN,uBAAuB,EACvB,SAAS,EAET,QAAQ,EAER,eAAe,EACf,gBAAgB,EAChB,QAAQ,EACR,YAAY,EACZ,eAAe,EACf,MAAM,EACN,KAAK,EACL,KAAK,EACL,eAAe,EACf,MAAM,EACN,SAAS,GACT,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACxF,OAAO,EAA6B,kBAAkB,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAClG,OAAO,EAGN,kCAAkC,EAClC,mCAAmC,GACnC,MAAM,wBAAwB,CAAC;AAChC,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAY,iBAAiB,EAAE,iBAAiB,EAAW,MAAM,yBAAyB,CAAC;AAClG,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAC5D,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,EAAE,KAAK,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACvD,OAAO,EAAE,yBAAyB,EAAE,MAAM,gCAAgC,CAAC;AAC3E,OAAO,EAAE,wBAAwB,EAAE,MAAM,+BAA+B,CAAC;AAEzE,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;;;AAItD,IAAI,MAAM,GAAG,CAAC,CAAC;AA4Cf,MAAM,OAAO,kBAAkB;IAGb,yBAAyB,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;IACtD,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IACjD,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;IAC7B,gBAAgB,GAAG,MAAM,CAAC,kBAAkB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IACnE,SAAS,GAAG,MAAM,CAAC,SAAS,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;IAE9D,EAAE,GAAG,KAAK,CAAS,cAAc,MAAM,EAAE,EAAE,CAAC,CAAC;IAC7C,QAAQ,GAAG,KAAK,CAAwB,KAAK,EAAE;QAC9D,SAAS,EAAE,gBAAgB;KAC3B,CAAC,CAAC;IACa,WAAW,GAAG,KAAK,CAAS,EAAE,CAAC,CAAC;IAChC,QAAQ,GAAG,KAAK,CAAwB,KAAK,EAAE;QAC9D,SAAS,EAAE,gBAAgB;KAC3B,CAAC,CAAC;IACa,GAAG,GAAG,KAAK,CAAmB,KAAK,CAAC,CAAC;IACrC,UAAU,GAAG,KAAK,CAAsB,GAAG,EAAE;QAC5D,SAAS,EAAE,eAAe;KAC1B,CAAC,CAAC;IAEa,IAAI,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;IAC7B,KAAK,GAAG,KAAK,EAAW,CAAC;IACzB,WAAW,GAAG,KAAK,CAA4B,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACtE,aAAa,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAE9C,sCAAsC;IACnB,WAAW,GAAG,YAAY,CAAC,iBAAiB,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC;IAEzF,2CAA2C;IACxB,aAAa,GAAG,YAAY,CAAC,QAAQ,CAAC,yBAAyB,CAAC,CAAC;IAEpF,gBAAgB;IACA,OAAO,GAAG,eAAe,CAAC,wBAAwB,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC;IAE3F,iFAAiF;IACjE,eAAe,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IAEvG,2CAA2C;IACxB,WAAW,GAAG,SAAS,CAAC,QAAQ,CAAC,mBAAmB,CAAC,CAAC;IACzD,OAAO,GAAG,MAAM,CAAsC,IAAI,CAAC,CAAC;IAC5D,YAAY,GAAG,MAAM,CAAS,CAAC,CAAC,CAAC;IAE9B,gBAAgB,GAAG,QAAQ,CAC7C,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CAC3B,SAAS,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,EACjG,kBAAkB,EAAE,CACpB,EACD,EAAE,YAAY,EAAE,KAAK,EAAE,CACvB,CAAC;IAEc,KAAK,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC;IAEvD,iBAAiB,GAAG,IAAI,OAAO,EAAkC,CAAC;IAErE,IAAI,GAAgD,QAAQ,CAC3E,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC1B,GAAG,CAAsE,CAAC,MAAM,EAAE,EAAE;IACnF,sEAAsE;IACtE,MAAM,CAAC,cAAc,CAAC,OAAO,KAAK,QAAQ;QACzC,CAAC,CAAC,MAAM,CAAC,cAAc,CAAC,OAAO,KAAK,OAAO;YAC1C,CAAC,CAAC,MAAM;YACR,CAAC,CAAC,OAAO;QACV,CAAC,CAAC,MAAM,CAAC,cAAc,CAAC,OAAO,CAChC,CACD,EACD,EAAE,YAAY,EAAE,QAAQ,EAAE,CAC1B,CAAC;IAEc,OAAO,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,EAAE,EAAE,EAAE,IAAI,GAAG,IAAI,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC;IAE1F,gEAAgE;IACxD,SAAS,GAAsB,GAAG,EAAE,GAAE,CAAC,CAAC;IAChD,gEAAgE;IACxD,UAAU,GAAY,GAAG,EAAE,GAAE,CAAC,CAAC;IAEvC;;;;;OAKG;IACO,UAAU,GAAwB;QAC3C;YACC,OAAO,EAAE,OAAO;YAChB,OAAO,EAAE,QAAQ;YACjB,QAAQ,EAAE,OAAO;YACjB,QAAQ,EAAE,KAAK;SACf;QACD;YACC,OAAO,EAAE,KAAK;YACd,OAAO,EAAE,QAAQ;YACjB,QAAQ,EAAE,KAAK;YACf,QAAQ,EAAE,KAAK;SACf;QACD;YACC,OAAO,EAAE,OAAO;YAChB,OAAO,EAAE,KAAK;YACd,QAAQ,EAAE,OAAO;YACjB,QAAQ,EAAE,QAAQ;SAClB;QACD;YACC,OAAO,EAAE,KAAK;YACd,OAAO,EAAE,KAAK;YACd,QAAQ,EAAE,KAAK;YACf,QAAQ,EAAE,QAAQ;SAClB;KACD,CAAC;IAEK,iBAAiB,CAAoB;IAE5B,UAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,CAAC,CAAC;IAEjF;QACC,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;YAC7B,IAAI,CAAC,SAAS,CAAC,aAAa,GAAG,IAAI,CAAC;QACrC,CAAC;QAED,IAAI,CAAC,iBAAiB,GAAG,IAAI,iBAAiB,CAC7C,IAAI,CAAC,yBAAyB,EAC9B,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,gBAAgB,EACrB,IAAI,CAAC,WAAW,CAChB,CAAC;IACH,CAAC;IAED,SAAS;QACR,IAAI,CAAC,iBAAiB,CAAC,gBAAgB,EAAE,CAAC;IAC3C,CAAC;IAEM,MAAM;QACZ,IAAI,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;YACjB,IAAI,CAAC,IAAI,EAAE,CAAC;QACb,CAAC;aAAM,CAAC;YACP,IAAI,CAAC,IAAI,EAAE,CAAC;QACb,CAAC;IACF,CAAC;IAEM,IAAI;QACV,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,IAAI,CAAC,QAAQ,EAAE,IAAI,IAAI,CAAC,aAAa,EAAE,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,MAAM,IAAI,CAAC,EAAE,CAAC;YAC3F,OAAO;QACR,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACpB,eAAe,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,SAAS,EAAE,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;IACvF,CAAC;IAEM,IAAI;QACV,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE;YAAE,OAAO;QAEzB,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACrB,IAAI,CAAC,UAAU,EAAE,CAAC;QAElB,+BAA+B;QAC/B,IAAI,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,CAAC;IACzB,CAAC;IAEM,UAAU,CAAC,KAAQ;QACzB,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IACvB,CAAC;IAEM,gBAAgB,CAAC,EAAqB;QAC5C,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACrB,CAAC;IAEM,iBAAiB,CAAC,EAAW;QACnC,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;IACtB,CAAC;IAEM,gBAAgB,CAAC,UAAmB;QAC1C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IACpC,CAAC;IAED,YAAY,CAAC,KAAQ;QACpB,qEAAqE;QACrE,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YACrB,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,EAAS,CAAC;YACzC,MAAM,QAAQ,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC,GAAG,YAAY,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YACnE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC1B,CAAC;aAAM,CAAC;YACP,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACvB,CAAC;QAED,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,CAAC,KAAK,EAAa,CAAC,CAAC;QAE1C,8CAA8C;QAC9C,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YACtB,IAAI,CAAC,IAAI,EAAE,CAAC;QACb,CAAC;IACF,CAAC;IAED,cAAc,CAAC,KAAQ;QACtB,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YACrB,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,EAAS,CAAC;YACzC,MAAM,QAAQ,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,KAAK,KAAK,CAAC,CAAC;YAC7D,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC1B,CAAC;aAAM,CAAC;YACP,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAS,CAAC,CAAC;QAC3B,CAAC;QAED,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,CAAC,KAAK,EAAa,CAAC,CAAC;IAC3C,CAAC;IAED,YAAY,CAAC,KAAQ;QACpB,IAAI,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;YAC5B,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;aAAM,CAAC;YACP,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC1B,CAAC;IACF,CAAC;IAED,UAAU,CAAC,KAAQ;QAClB,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;QAE/B,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YAC9B,OAAO,SAAS,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC;QAChE,CAAC;aAAM,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YAChC,OAAO,IAAI,CAAC,WAAW,EAAE,CAAC,SAAc,EAAE,KAAK,CAAC,CAAC;QAClD,CAAC;QAED,OAAO,KAAK,CAAC;IACd,CAAC;0HA9NW,kBAAkB;8GAAlB,kBAAkB,mvCArCnB;YACV,kCAAkC,EAAC,GAAG,EAAE,CAAC,kBAAkB,EAAC;YAC5D,mCAAmC,EAAC,GAAG,EAAE,CAAC,kBAAkB,EAAC;YAC7D,gBAAgB,CAAC,kBAAkB,CAAC;YACpC;gBACC,OAAO,EAAE,mBAAmB;gBAC5B,WAAW,EAAE,kBAAkB;aAC/B;SACD,mEAyD6C,iBAAiB,6EAGN,yBAAyB,6EAGxC,wBAAwB,6GAMd,mBAAmB,gEApE7D;;;;;;;;;;;;;;;;;;;;;;;;;;EA0BT,2DArCS,aAAa,0rCAAE,gBAAgB;;2FAuC7B,kBAAkB;kBA1C9B,SAAS;mBAAC;oBACV,QAAQ,EAAE,wBAAwB;oBAClC,UAAU,EAAE,IAAI;oBAChB,OAAO,EAAE,CAAC,aAAa,EAAE,gBAAgB,CAAC;oBAC1C,eAAe,EAAE,uBAAuB,CAAC,MAAM;oBAC/C,SAAS,EAAE;wBACV,kCAAkC,EAAC,GAAG,EAAE,mBAAmB,EAAC;wBAC5D,mCAAmC,EAAC,GAAG,EAAE,mBAAmB,EAAC;wBAC7D,gBAAgB,oBAAoB;wBACpC;4BACC,OAAO,EAAE,mBAAmB;4BAC5B,WAAW,oBAAoB;yBAC/B;qBACD;oBACD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;EA0BT;iBACD","sourcesContent":["import { BooleanInput, NumberInput } from '@angular/cdk/coercion';\nimport { CdkListboxModule } from '@angular/cdk/listbox';\nimport {\n\tCdkConnectedOverlay,\n\ttype ConnectedOverlayPositionChange,\n\ttype ConnectedPosition,\n\tOverlayModule,\n} from '@angular/cdk/overlay';\nimport {\n\tChangeDetectionStrategy,\n\tComponent,\n\ttype DoCheck,\n\tInjector,\n\ttype Signal,\n\tafterNextRender,\n\tbooleanAttribute,\n\tcomputed,\n\tcontentChild,\n\tcontentChildren,\n\tinject,\n\tinput,\n\tmodel,\n\tnumberAttribute,\n\tsignal,\n\tviewChild,\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { type ControlValueAccessor, FormGroupDirective, NgControl, NgForm } from '@angular/forms';\nimport {\n\ttype ExposesSide,\n\ttype ExposesState,\n\tprovideExposedSideProviderExisting,\n\tprovideExposesStateProviderExisting,\n} from '@spartan-ng/brain/core';\nimport { BrnFormFieldControl } from '@spartan-ng/brain/form-field';\nimport { ChangeFn, ErrorStateMatcher, ErrorStateTracker, TouchFn } from '@spartan-ng/brain/forms';\nimport { BrnLabelDirective } from '@spartan-ng/brain/label';\nimport { Subject, of } from 'rxjs';\nimport { delay, map, switchMap } from 'rxjs/operators';\nimport { BrnSelectContentComponent } from './brn-select-content.component';\nimport { BrnSelectOptionDirective } from './brn-select-option.directive';\nimport { BrnSelectTriggerDirective } from './brn-select-trigger.directive';\nimport { provideBrnSelect } from './brn-select.token';\n\nexport type BrnReadDirection = 'ltr' | 'rtl';\n\nlet nextId = 0;\n\n@Component({\n\tselector: 'brn-select, hlm-select',\n\tstandalone: true,\n\timports: [OverlayModule, CdkListboxModule],\n\tchangeDetection: ChangeDetectionStrategy.OnPush,\n\tproviders: [\n\t\tprovideExposedSideProviderExisting(() => BrnSelectComponent),\n\t\tprovideExposesStateProviderExisting(() => BrnSelectComponent),\n\t\tprovideBrnSelect(BrnSelectComponent),\n\t\t{\n\t\t\tprovide: BrnFormFieldControl,\n\t\t\tuseExisting: BrnSelectComponent,\n\t\t},\n\t],\n\ttemplate: `\n\t\t@if (!selectLabel() && placeholder()) {\n\t\t\t<label class=\"hidden\" [attr.id]=\"labelId()\">{{ placeholder() }}</label>\n\t\t} @else {\n\t\t\t<ng-content select=\"label[hlmLabel],label[brnLabel]\" />\n\t\t}\n\n\t\t<div cdk-overlay-origin (click)=\"toggle()\" #trigger=\"cdkOverlayOrigin\">\n\t\t\t<ng-content select=\"hlm-select-trigger,[brnSelectTrigger]\" />\n\t\t</div>\n\n\t\t<ng-template\n\t\t\tcdk-connected-overlay\n\t\t\tcdkConnectedOverlayLockPosition\n\t\t\tcdkConnectedOverlayHasBackdrop\n\t\t\tcdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n\t\t\t[cdkConnectedOverlayOrigin]=\"trigger\"\n\t\t\t[cdkConnectedOverlayOpen]=\"_delayedExpanded()\"\n\t\t\t[cdkConnectedOverlayPositions]=\"_positions\"\n\t\t\t[cdkConnectedOverlayWidth]=\"triggerWidth() > 0 ? triggerWidth() : 'auto'\"\n\t\t\t(backdropClick)=\"hide()\"\n\t\t\t(detach)=\"hide()\"\n\t\t\t(positionChange)=\"_positionChanges$.next($event)\"\n\t\t>\n\t\t\t<ng-content />\n\t\t</ng-template>\n\t`,\n})\nexport class BrnSelectComponent<T = unknown>\n\timplements ControlValueAccessor, DoCheck, ExposesSide, ExposesState, BrnFormFieldControl\n{\n\tprivate readonly _defaultErrorStateMatcher = inject(ErrorStateMatcher);\n\tprivate readonly _parentForm = inject(NgForm, { optional: true });\n\tprivate readonly _injector = inject(Injector);\n\tprivate readonly _parentFormGroup = inject(FormGroupDirective, { optional: true });\n\tpublic readonly ngControl = inject(NgControl, { optional: true, self: true });\n\n\tpublic readonly id = input<string>(`brn-select-${nextId++}`);\n\tpublic readonly multiple = input<boolean, BooleanInput>(false, {\n\t\ttransform: booleanAttribute,\n\t});\n\tpublic readonly placeholder = input<string>('');\n\tpublic readonly disabled = input<boolean, BooleanInput>(false, {\n\t\ttransform: booleanAttribute,\n\t});\n\tpublic readonly dir = input<BrnReadDirection>('ltr');\n\tpublic readonly closeDelay = input<number, NumberInput>(100, {\n\t\ttransform: numberAttribute,\n\t});\n\n\tpublic readonly open = model<boolean>(false);\n\tpublic readonly value = model<T | T[]>();\n\tpublic readonly compareWith = input<(o1: T, o2: T) => boolean>((o1, o2) => o1 === o2);\n\tpublic readonly _formDisabled = signal(false);\n\n\t/** Label provided by the consumer. */\n\tprotected readonly selectLabel = contentChild(BrnLabelDirective, { descendants: false });\n\n\t/** Overlay pane containing the options. */\n\tprotected readonly selectContent = contentChild.required(BrnSelectContentComponent);\n\n\t/** @internal */\n\tpublic readonly options = contentChildren(BrnSelectOptionDirective, { descendants: true });\n\n\t/** @internal Derive the selected options to filter out the unselected options */\n\tpublic readonly selectedOptions = computed(() => this.options().filter((option) => option.selected()));\n\n\t/** Overlay pane containing the options. */\n\tprotected readonly _overlayDir = viewChild.required(CdkConnectedOverlay);\n\tpublic readonly trigger = signal<BrnSelectTriggerDirective<T> | null>(null);\n\tpublic readonly triggerWidth = signal<number>(0);\n\n\tprotected readonly _delayedExpanded = toSignal(\n\t\ttoObservable(this.open).pipe(\n\t\t\tswitchMap((expanded) => (!expanded ? of(expanded).pipe(delay(this.closeDelay())) : of(expanded))),\n\t\t\ttakeUntilDestroyed(),\n\t\t),\n\t\t{ initialValue: false },\n\t);\n\n\tpublic readonly state = computed(() => (this.open() ? 'open' : 'closed'));\n\n\tprotected readonly _positionChanges$ = new Subject<ConnectedOverlayPositionChange>();\n\n\tpublic readonly side: Signal<'top' | 'bottom' | 'left' | 'right'> = toSignal(\n\t\tthis._positionChanges$.pipe(\n\t\t\tmap<ConnectedOverlayPositionChange, 'top' | 'bottom' | 'left' | 'right'>((change) =>\n\t\t\t\t// todo: better translation or adjusting hlm to take that into account\n\t\t\t\tchange.connectionPair.originY === 'center'\n\t\t\t\t\t? change.connectionPair.originX === 'start'\n\t\t\t\t\t\t? 'left'\n\t\t\t\t\t\t: 'right'\n\t\t\t\t\t: change.connectionPair.originY,\n\t\t\t),\n\t\t),\n\t\t{ initialValue: 'bottom' },\n\t);\n\n\tpublic readonly labelId = computed(() => this.selectLabel()?.id ?? `${this.id()}--label`);\n\n\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\tprivate _onChange: ChangeFn<T | T[]> = () => {};\n\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\tprivate _onTouched: TouchFn = () => {};\n\n\t/*\n\t * This position config ensures that the top \"start\" corner of the overlay\n\t * is aligned with with the top \"start\" of the origin by default (overlapping\n\t * the trigger completely). If the panel cannot fit below the trigger, it\n\t * will fall back to a position above the trigger.\n\t */\n\tprotected _positions: ConnectedPosition[] = [\n\t\t{\n\t\t\toriginX: 'start',\n\t\t\toriginY: 'bottom',\n\t\t\toverlayX: 'start',\n\t\t\toverlayY: 'top',\n\t\t},\n\t\t{\n\t\t\toriginX: 'end',\n\t\t\toriginY: 'bottom',\n\t\t\toverlayX: 'end',\n\t\t\toverlayY: 'top',\n\t\t},\n\t\t{\n\t\t\toriginX: 'start',\n\t\t\toriginY: 'top',\n\t\t\toverlayX: 'start',\n\t\t\toverlayY: 'bottom',\n\t\t},\n\t\t{\n\t\t\toriginX: 'end',\n\t\t\toriginY: 'top',\n\t\t\toverlayX: 'end',\n\t\t\toverlayY: 'bottom',\n\t\t},\n\t];\n\n\tpublic errorStateTracker: ErrorStateTracker;\n\n\tpublic readonly errorState = computed(() => this.errorStateTracker.errorState());\n\n\tconstructor() {\n\t\tif (this.ngControl !== null) {\n\t\t\tthis.ngControl.valueAccessor = this;\n\t\t}\n\n\t\tthis.errorStateTracker = new ErrorStateTracker(\n\t\t\tthis._defaultErrorStateMatcher,\n\t\t\tthis.ngControl,\n\t\t\tthis._parentFormGroup,\n\t\t\tthis._parentForm,\n\t\t);\n\t}\n\n\tngDoCheck() {\n\t\tthis.errorStateTracker.updateErrorState();\n\t}\n\n\tpublic toggle(): void {\n\t\tif (this.open()) {\n\t\t\tthis.hide();\n\t\t} else {\n\t\t\tthis.show();\n\t\t}\n\t}\n\n\tpublic show(): void {\n\t\tif (this.open() || this.disabled() || this._formDisabled() || this.options()?.length == 0) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.open.set(true);\n\t\tafterNextRender(() => this.selectContent().focusList(), { injector: this._injector });\n\t}\n\n\tpublic hide(): void {\n\t\tif (!this.open()) return;\n\n\t\tthis.open.set(false);\n\t\tthis._onTouched();\n\n\t\t// restore focus to the trigger\n\t\tthis.trigger()?.focus();\n\t}\n\n\tpublic writeValue(value: T): void {\n\t\tthis.value.set(value);\n\t}\n\n\tpublic registerOnChange(fn: ChangeFn<T | T[]>): void {\n\t\tthis._onChange = fn;\n\t}\n\n\tpublic registerOnTouched(fn: TouchFn): void {\n\t\tthis._onTouched = fn;\n\t}\n\n\tpublic setDisabledState(isDisabled: boolean) {\n\t\tthis._formDisabled.set(isDisabled);\n\t}\n\n\tselectOption(value: T): void {\n\t\t// if this is a multiple select we need to add the value to the array\n\t\tif (this.multiple()) {\n\t\t\tconst currentValue = this.value() as T[];\n\t\t\tconst newValue = currentValue ? [...currentValue, value] : [value];\n\t\t\tthis.value.set(newValue);\n\t\t} else {\n\t\t\tthis.value.set(value);\n\t\t}\n\n\t\tthis._onChange?.(this.value() as T | T[]);\n\n\t\t// if this is single select close the dropdown\n\t\tif (!this.multiple()) {\n\t\t\tthis.hide();\n\t\t}\n\t}\n\n\tdeselectOption(value: T): void {\n\t\tif (this.multiple()) {\n\t\t\tconst currentValue = this.value() as T[];\n\t\t\tconst newValue = currentValue.filter((val) => val !== value);\n\t\t\tthis.value.set(newValue);\n\t\t} else {\n\t\t\tthis.value.set(null as T);\n\t\t}\n\n\t\tthis._onChange?.(this.value() as T | T[]);\n\t}\n\n\ttoggleSelect(value: T): void {\n\t\tif (this.isSelected(value)) {\n\t\t\tthis.deselectOption(value);\n\t\t} else {\n\t\t\tthis.selectOption(value);\n\t\t}\n\t}\n\n\tisSelected(value: T): boolean {\n\t\tconst selection = this.value();\n\n\t\tif (Array.isArray(selection)) {\n\t\t\treturn selection.some((val) => this.compareWith()(val, value));\n\t\t} else if (value !== undefined) {\n\t\t\treturn this.compareWith()(selection as T, value);\n\t\t}\n\n\t\treturn false;\n\t}\n}\n"]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { inject, InjectionToken } from '@angular/core';
|
|
2
|
+
const BrnSelectToken = new InjectionToken('BrnSelectToken');
|
|
3
|
+
export function injectBrnSelect() {
|
|
4
|
+
return inject(BrnSelectToken);
|
|
5
|
+
}
|
|
6
|
+
export function provideBrnSelect(select) {
|
|
7
|
+
return { provide: BrnSelectToken, useExisting: select };
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnJuLXNlbGVjdC50b2tlbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvYnJhaW4vc2VsZWN0L3NyYy9saWIvYnJuLXNlbGVjdC50b2tlbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQW9CLE1BQU0sRUFBRSxjQUFjLEVBQVEsTUFBTSxlQUFlLENBQUM7QUFHL0UsTUFBTSxjQUFjLEdBQUcsSUFBSSxjQUFjLENBQXFCLGdCQUFnQixDQUFDLENBQUM7QUFFaEYsTUFBTSxVQUFVLGVBQWU7SUFDOUIsT0FBTyxNQUFNLENBQUMsY0FBYyxDQUEwQixDQUFDO0FBQ3hELENBQUM7QUFFRCxNQUFNLFVBQVUsZ0JBQWdCLENBQUMsTUFBZ0M7SUFDaEUsT0FBTyxFQUFFLE9BQU8sRUFBRSxjQUFjLEVBQUUsV0FBVyxFQUFFLE1BQU0sRUFBRSxDQUFDO0FBQ3pELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBFeGlzdGluZ1Byb3ZpZGVyLCBpbmplY3QsIEluamVjdGlvblRva2VuLCBUeXBlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgdHlwZSB7IEJyblNlbGVjdENvbXBvbmVudCB9IGZyb20gJy4vYnJuLXNlbGVjdC5jb21wb25lbnQnO1xuXG5jb25zdCBCcm5TZWxlY3RUb2tlbiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxCcm5TZWxlY3RDb21wb25lbnQ+KCdCcm5TZWxlY3RUb2tlbicpO1xuXG5leHBvcnQgZnVuY3Rpb24gaW5qZWN0QnJuU2VsZWN0PFQ+KCk6IEJyblNlbGVjdENvbXBvbmVudDxUPiB7XG5cdHJldHVybiBpbmplY3QoQnJuU2VsZWN0VG9rZW4pIGFzIEJyblNlbGVjdENvbXBvbmVudDxUPjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVCcm5TZWxlY3Qoc2VsZWN0OiBUeXBlPEJyblNlbGVjdENvbXBvbmVudD4pOiBFeGlzdGluZ1Byb3ZpZGVyIHtcblx0cmV0dXJuIHsgcHJvdmlkZTogQnJuU2VsZWN0VG9rZW4sIHVzZUV4aXN0aW5nOiBzZWxlY3QgfTtcbn1cbiJdfQ==
|