@spartan-ng/brain 0.0.1-alpha.411 → 0.0.1-alpha.413

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.
@@ -1,41 +1,56 @@
1
- import { CdkListbox, CdkListboxModule, CdkOption } from '@angular/cdk/listbox';
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, output, signal, viewChild, } from '@angular/core';
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, combineLatest, of } from 'rxjs';
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 { BrnSelectService } from './brn-select.service';
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
- _selectService = inject(BrnSelectService);
19
- triggerWidth = this._selectService.triggerWidth;
20
- multiple = input(false);
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
- _disabledFromSetDisabledState = signal(false);
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
- options = contentChildren(CdkOption, { descendants: true });
29
- options$ = toObservable(this.options);
30
- optionsAndIndex$ = this.options$.pipe(map((options, index) => [options, index]));
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
- closeDelay = input(100);
34
- isExpanded = this._selectService.isExpanded;
35
- _delayedExpanded = toSignal(toObservable(this.isExpanded).pipe(switchMap((expanded) => (!expanded ? of(expanded).pipe(delay(this.closeDelay())) : of(expanded))), takeUntilDestroyed()), { initialValue: false });
36
- state = computed(() => (this.isExpanded() ? 'open' : 'closed'));
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
- backupLabelId = computed(() => this._selectService.labelId());
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.isExpanded()) {
156
- this.close();
111
+ if (this.open()) {
112
+ this.hide();
157
113
  }
158
114
  else {
159
- this.open();
115
+ this.show();
160
116
  }
161
117
  }
162
- open() {
163
- if (!this._canOpen())
118
+ show() {
119
+ if (this.open() || this.disabled() || this._formDisabled() || this.options()?.length == 0) {
164
120
  return;
165
- this._selectService.state.update((state) => ({
166
- ...state,
167
- isExpanded: true,
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
- close() {
173
- if (!this.isExpanded())
125
+ hide() {
126
+ if (!this.open())
174
127
  return;
175
- if (this._selectService.selectTrigger) {
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
- _canOpen() {
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.writeValue$.next(value);
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._disabledFromSetDisabledState.set(isDisabled);
143
+ this._formDisabled.set(isDisabled);
204
144
  }
205
- /**
206
- * Once writeValue is called and options are available we can handle setting the initial options
207
- * @private
208
- */
209
- handleInitialOptionSelect() {
210
- // Write value cannot be handled until options are available, so we wait until both are available with a combineLatest
211
- combineLatest([this.writeValue$, this.options$])
212
- .pipe(map((values, index) => [...values, index]), takeUntilDestroyed())
213
- .subscribe(([value, _, index]) => {
214
- this._shouldEmitValueChange.set(false);
215
- this._selectService.setInitialSelectedOptions(value);
216
- // the first time this observable emits a value we are simply setting the initial state
217
- // this change should not count as changing the state of the select, so we need to mark as pristine
218
- if (index === 0) {
219
- this.ngControl?.control?.markAsPristine();
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
- * When options change, our current selected options may become invalid
225
- * Here we will automatically update our current selected options so that they are always inline with the possibleOptions
226
- * @private
227
- */
228
- handleOptionChanges() {
229
- this.optionsAndIndex$.pipe(takeUntilDestroyed()).subscribe(([options, index]) => {
230
- if (index > 0) {
231
- this.handleInvalidOptions(options);
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
- * Check that our "selectedOptions" are still valid when "possibleOptions" is about to be updated
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
- const selectedOption = selectedOptions[0] ?? null;
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: { openedChange: "openedChange", valueChange: "valueChange" }, providers: [
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: CdkOption, descendants: true, isSignal: true }], viewQueries: [{ propertyName: "_overlayDir", first: true, predicate: CdkConnectedOverlay, descendants: true, isSignal: true }], ngImport: i0, template: `
275
- @if (!labelProvided() && placeholder()) {
276
- <label class="hidden" [attr.id]="backupLabelId()">{{ placeholder() }}</label>
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)="close()"
294
- (detach)="close()"
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 (!labelProvided() && placeholder()) {
320
- <label class="hidden" [attr.id]="backupLabelId()">{{ placeholder() }}</label>
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)="close()"
338
- (detach)="close()"
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":"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","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==