@ethlete/cdk 2.5.0 → 2.5.1

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.
Files changed (20) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/esm2022/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.mjs +193 -263
  3. package/esm2022/lib/components/forms/components/select/components/select/directives/select/select.directive.mjs +7 -1
  4. package/esm2022/lib/components/forms/components/select/components/select-field/select-field.component.mjs +8 -5
  5. package/esm2022/lib/components/forms/components/select/directives/index.mjs +2 -0
  6. package/esm2022/lib/components/forms/components/select/directives/public-api.mjs +2 -0
  7. package/esm2022/lib/components/forms/components/select/directives/select-field/public-api.mjs +2 -0
  8. package/esm2022/lib/components/forms/components/select/directives/select-field/select-field.directive.mjs +21 -0
  9. package/esm2022/lib/components/forms/components/select/public-api.mjs +2 -1
  10. package/fesm2022/ethlete-cdk.mjs +216 -264
  11. package/fesm2022/ethlete-cdk.mjs.map +1 -1
  12. package/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.d.ts +10 -31
  13. package/lib/components/forms/components/select/components/select/directives/select/select.directive.d.ts +1 -0
  14. package/lib/components/forms/components/select/components/select-field/select-field.component.d.ts +5 -4
  15. package/lib/components/forms/components/select/directives/index.d.ts +1 -0
  16. package/lib/components/forms/components/select/directives/public-api.d.ts +1 -0
  17. package/lib/components/forms/components/select/directives/select-field/public-api.d.ts +1 -0
  18. package/lib/components/forms/components/select/directives/select-field/select-field.directive.d.ts +8 -0
  19. package/lib/components/forms/components/select/public-api.d.ts +1 -0
  20. package/package.json +2 -2
@@ -1,42 +1,17 @@
1
1
  import { coerceBooleanProperty } from '@angular/cdk/coercion';
2
- import { DOWN_ARROW, ENTER, ESCAPE, TAB, UP_ARROW } from '@angular/cdk/keycodes';
2
+ import { A, DOWN_ARROW, END, ENTER, ESCAPE, HOME, PAGE_DOWN, PAGE_UP, SPACE, TAB, UP_ARROW, } from '@angular/cdk/keycodes';
3
3
  import { AsyncPipe, NgFor, NgIf } from '@angular/common';
4
4
  import { ChangeDetectionStrategy, Component, EventEmitter, InjectionToken, Input, Output, ViewEncapsulation, inject, isDevMode, } from '@angular/core';
5
- import { AnimatedOverlayDirective, LetDirective, clone } from '@ethlete/core';
6
- import { BehaviorSubject, catchError, combineLatest, debounceTime, distinctUntilChanged, map, of, shareReplay, skip, takeUntil, tap, throwError, } from 'rxjs';
5
+ import { AnimatedOverlayDirective, LetDirective, RuntimeError, SelectionModel } from '@ethlete/core';
6
+ import { BehaviorSubject, catchError, combineLatest, debounceTime, distinctUntilChanged, map, skip, skipWhile, takeUntil, tap, throwError, } from 'rxjs';
7
7
  import { ChevronIconComponent } from '../../../../../../../icons';
8
8
  import { INPUT_TOKEN, InputDirective, NativeInputRefDirective } from '../../../../../../directives';
9
9
  import { DecoratedInputBase } from '../../../../../../utils';
10
+ import { SELECT_FIELD_TOKEN } from '../../../../directives';
10
11
  import { ComboboxBodyComponent } from '../../partials';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "../../../../../../directives/input/input.directive";
13
14
  import * as i2 from "@ethlete/core";
14
- export class RuntimeError extends Error {
15
- constructor(code, message, data = '__ET_NO_DATA__') {
16
- super(formatRuntimeError(code, message));
17
- this.code = code;
18
- this.data = data;
19
- if (data !== '__ET_NO_DATA__') {
20
- try {
21
- const _data = clone(data);
22
- setTimeout(() => {
23
- console.error(_data);
24
- }, 1);
25
- }
26
- catch {
27
- setTimeout(() => {
28
- console.error(data);
29
- }, 1);
30
- }
31
- }
32
- }
33
- }
34
- export function formatRuntimeError(code, message) {
35
- // prefix code with zeros if it's less than 100
36
- const codeWithZeros = code < 10 ? `00${code}` : code < 100 ? `0${code}` : code;
37
- const fullCode = `ET${codeWithZeros}`;
38
- return `${fullCode}${message ? ': ' + message : ''}`;
39
- }
40
15
  const COMBOBOX_ERRORS = {
41
16
  1: 'Expected options to be an array of objects. This is due to "bindLabel" and "bindValue" being set.',
42
17
  2: 'Expected options to be an array of primitives. This is due to "bindLabel" and "bindValue" not being set or "allowCustomValues" being set to true.',
@@ -66,46 +41,35 @@ const isObjectArray = (value) => {
66
41
  const isEmptyArray = (value) => {
67
42
  return Array.isArray(value) && value.length === 0;
68
43
  };
69
- const isObject = (value) => {
70
- return typeof value === 'object' && value !== null;
71
- };
72
44
  const ComboboxOptionType = {
73
45
  Primitive: 'primitive',
74
46
  Object: 'object',
75
47
  };
76
- const ComboboxInputValueType = {
77
- Single: 'single',
78
- Array: 'array',
79
- };
80
- const getObjectProperty = (obj, prop) => {
81
- const hasDotNotation = prop.includes('.');
82
- if (!hasDotNotation)
83
- return obj[prop];
84
- const props = prop.split('.');
85
- let value = obj;
86
- for (const prop of props) {
87
- if (!isObject(value))
88
- return undefined;
89
- value = value[prop];
90
- }
91
- return value;
92
- };
93
48
  export const COMBOBOX_TOKEN = new InjectionToken('ET_COMBOBOX_TOKEN');
94
49
  class ComboboxComponent extends DecoratedInputBase {
95
50
  //#endregion
96
51
  //#region Inputs
97
52
  get options() {
98
- return this._options$.value;
53
+ return this._selectionModel.getFilteredOptions();
99
54
  }
100
55
  set options(value) {
101
- this._options$.next(value);
102
- }
103
- get initialValue() {
104
- return this._initialValue$.value;
56
+ this._selectionModel.setOptions(value);
105
57
  }
106
58
  set initialValue(value) {
107
- this._initialValue$.next(value);
108
- this._updateSelectedOptionsViaInitialValue();
59
+ this._selectionModel.setSelection(value);
60
+ }
61
+ get filterInternal() {
62
+ return this._filterInternal$.value;
63
+ }
64
+ set filterInternal(value) {
65
+ const val = coerceBooleanProperty(value);
66
+ this._filterInternal$.next(val);
67
+ if (!val) {
68
+ this._selectionModel.setFilter('');
69
+ }
70
+ else {
71
+ this._selectionModel.setFilter(this._currentFilter);
72
+ }
109
73
  }
110
74
  get loading() {
111
75
  return this._loading$.value;
@@ -125,23 +89,14 @@ class ComboboxComponent extends DecoratedInputBase {
125
89
  set placeholder(value) {
126
90
  this._placeholder$.next(value);
127
91
  }
128
- get multiple() {
129
- return this._multiple$.value;
130
- }
131
92
  set multiple(value) {
132
- this._multiple$.next(coerceBooleanProperty(value));
133
- }
134
- get bindLabel() {
135
- return this._bindLabel$.value;
93
+ this._selectionModel.setAllowMultiple(coerceBooleanProperty(value));
136
94
  }
137
95
  set bindLabel(value) {
138
- this._bindLabel$.next(value);
139
- }
140
- get bindValue() {
141
- return this._bindValue$.value;
96
+ this._selectionModel.setLabelBinding(value);
142
97
  }
143
98
  set bindValue(value) {
144
- this._bindValue$.next(value);
99
+ this._selectionModel.setValueBinding(value);
145
100
  }
146
101
  get allowCustomValues() {
147
102
  return this._allowCustomValues$.value;
@@ -151,9 +106,6 @@ class ComboboxComponent extends DecoratedInputBase {
151
106
  }
152
107
  //#endregion
153
108
  //#region Members
154
- get _rawSelectedOptions() {
155
- return this._rawSelectedOptions$.value;
156
- }
157
109
  get _currentFilter() {
158
110
  return this._currentFilter$.value;
159
111
  }
@@ -161,67 +113,30 @@ class ComboboxComponent extends DecoratedInputBase {
161
113
  return this._animatedOverlay.isMounted;
162
114
  }
163
115
  //#endregion
116
+ //#region Computes
117
+ //#endregion
164
118
  //#region Lifecycle
165
119
  constructor() {
166
120
  super();
167
121
  //#region Injects
168
122
  this._input = inject(INPUT_TOKEN);
169
123
  this._animatedOverlay = inject(AnimatedOverlayDirective);
170
- this._options$ = new BehaviorSubject([]);
171
- this.options$ = this._options$.asObservable();
172
- this._initialValue$ = new BehaviorSubject(null);
124
+ this._selectField = inject(SELECT_FIELD_TOKEN);
125
+ this._filterInternal$ = new BehaviorSubject(true);
173
126
  this._loading$ = new BehaviorSubject(false);
174
127
  this.loading$ = this._loading$.asObservable();
175
128
  this._error$ = new BehaviorSubject(null);
176
129
  this._placeholder$ = new BehaviorSubject(null);
177
- this._multiple$ = new BehaviorSubject(false);
178
- this._bindLabel$ = new BehaviorSubject(null);
179
- this._bindValue$ = new BehaviorSubject(null);
180
130
  this._allowCustomValues$ = new BehaviorSubject(false);
181
131
  //#endregion
182
132
  //#region Outputs
183
133
  this.filterChange = new EventEmitter();
184
- this._rawSelectedOptions$ = new BehaviorSubject([]);
185
134
  this._currentFilter$ = new BehaviorSubject('');
186
135
  this._isOpen$ = this._animatedOverlay.isMounted$;
187
- //#endregion
188
- //#region Computes
189
- this._allOptions$ = combineLatest([this._options$, this._rawSelectedOptions$]).pipe(map(([options, rawSelectedOptions]) => [...options, ...rawSelectedOptions]), shareReplay({ bufferSize: 1, refCount: true }));
190
- this.expectedOptionType$ = combineLatest([
191
- this._bindLabel$,
192
- this._bindValue$,
193
- this._allowCustomValues$,
194
- ]).pipe(map(([bindLabel, bindValue, allowCustomValues]) => {
195
- const shouldBeObjects = bindLabel && bindValue && !allowCustomValues;
196
- if (shouldBeObjects) {
197
- return ComboboxOptionType.Object;
198
- }
199
- return ComboboxOptionType.Primitive;
200
- }));
201
- this.expectedInputValueType$ = this._multiple$.pipe(map((multiple) => (multiple ? ComboboxInputValueType.Array : ComboboxInputValueType.Single)));
202
- this.selectedOptions$ = combineLatest([
203
- this._input.value$,
204
- this.expectedInputValueType$,
205
- this._allOptions$,
206
- ]).pipe(map(([inputValue, expectedInputValueType, allOptions]) => {
207
- if (expectedInputValueType === ComboboxInputValueType.Single) {
208
- if (Array.isArray(inputValue)) {
209
- return [];
210
- }
211
- const option = this._findOptionByValue(inputValue, allOptions);
212
- return [option];
213
- }
214
- else if (expectedInputValueType === ComboboxInputValueType.Array) {
215
- if (!Array.isArray(inputValue)) {
216
- return [];
217
- }
218
- const options = inputValue.map((value) => this._findOptionByValue(value, allOptions)).filter((v) => !!v);
219
- if (!options.length)
220
- return [];
221
- return options;
222
- }
223
- return [];
224
- }), map((options) => options.filter((v) => !!v)));
136
+ this._selectionModel = new SelectionModel();
137
+ this.selectedOptions$ = this._selectionModel.selection$;
138
+ this.multiple$ = this._selectionModel.allowMultiple$;
139
+ this.options$ = this._selectionModel.filteredOptions$;
225
140
  this._bindings.push({
226
141
  attribute: 'class.et-combobox--loading',
227
142
  observable: this._loading$,
@@ -234,6 +149,10 @@ class ComboboxComponent extends DecoratedInputBase {
234
149
  attribute: 'class.et-combobox--open',
235
150
  observable: this._isOpen$,
236
151
  });
152
+ this._selectField._bindings.push({
153
+ attribute: 'class.et-select-field--open',
154
+ observable: this._isOpen$,
155
+ });
237
156
  }
238
157
  ngOnInit() {
239
158
  this._initDispatchFilterChanges();
@@ -241,42 +160,24 @@ class ComboboxComponent extends DecoratedInputBase {
241
160
  if (isDevMode()) {
242
161
  this._debugValidateComboboxConfig();
243
162
  }
163
+ this._selectionModel.value$
164
+ .pipe(takeUntil(this._destroy$), tap((value) => {
165
+ this._input._updateValue(value);
166
+ this._setFilterFromInputValue();
167
+ }))
168
+ .subscribe();
244
169
  }
245
170
  //#endregion
246
171
  //#region Public Methods
247
172
  getOptionLabel(option) {
248
- if (!this.bindLabel || !isObject(option)) {
249
- return of(option);
250
- }
251
- return this._bindLabel$.pipe(map((label) => {
252
- if (!label)
253
- return option;
254
- return getObjectProperty(option, label) ?? option;
255
- }));
173
+ return this._selectionModel.getLabel$(option);
256
174
  }
257
175
  getOptionValue(option) {
258
- if (!this.bindValue || !isObject(option)) {
259
- return of(option);
260
- }
261
- return this._bindValue$.pipe(map((value) => {
262
- if (!value)
263
- return option;
264
- return getObjectProperty(option, value) ?? option;
265
- }));
176
+ return this._selectionModel.getValue$(option);
266
177
  }
267
178
  removeSelectedOption(option) {
179
+ this._selectionModel.removeSelectedOption(option);
268
180
  this.input._markAsTouched();
269
- const selectedOptions = this._rawSelectedOptions;
270
- const optionValue = this._getOptionValueSync(option);
271
- const index = selectedOptions.findIndex((v) => {
272
- const val = this._getOptionValueSync(v);
273
- return val === optionValue;
274
- });
275
- if (index === -1)
276
- return;
277
- selectedOptions.splice(index, 1);
278
- this._rawSelectedOptions$.next(selectedOptions);
279
- this._removeOrClearInputValue(optionValue);
280
181
  }
281
182
  open() {
282
183
  // if (!this._selectBodyConfig) return;
@@ -300,70 +201,102 @@ class ComboboxComponent extends DecoratedInputBase {
300
201
  }
301
202
  writeValueFromOption(option) {
302
203
  this.input._markAsTouched();
303
- const optionValue = this._getOptionValueSync(option);
304
- if (this._findOptionByValue(optionValue, this._rawSelectedOptions)) {
305
- if (this.multiple) {
306
- this.removeSelectedOption(option);
307
- this._removeOrClearInputValue(optionValue);
308
- }
309
- return;
204
+ if (this._selectionModel.allowMultiple) {
205
+ this._selectionModel.toggleSelectedOption(option);
206
+ }
207
+ else {
208
+ this._selectionModel.addSelectedOption(option);
310
209
  }
311
- this._setOrAddToInputValue(optionValue);
312
- this._addSelectedOption(option);
313
- if (!this.multiple) {
210
+ if (!this._selectionModel.allowMultiple) {
314
211
  this.close();
315
212
  this._setFilterFromInputValue();
316
213
  }
317
214
  }
318
215
  isOptionSelected(option) {
319
- return combineLatest([this._rawSelectedOptions$, this.getOptionValue(option)]).pipe(map(([selectedOptions, value]) => !!this._findOptionByValue(value, selectedOptions)));
216
+ return this._selectionModel.isSelected$(option);
320
217
  }
321
218
  //#endregion
322
219
  //#region Protected Methods
323
220
  processKeydownEvent(event) {
324
221
  const keyCode = event.keyCode;
325
222
  const isOpen = this._isOpen;
326
- const isMultiple = this.multiple;
223
+ const isMultiple = this._selectionModel.allowMultiple;
327
224
  const canAddCustomValue = this.allowCustomValues;
328
225
  const value = event.target.value;
329
- const hasValue = !!value;
226
+ const hasFilterValue = !!value;
227
+ const result = {};
330
228
  // The user typed a custom value and pressed enter. Add it to the selected options.
331
- if (keyCode === ENTER && canAddCustomValue && hasValue) {
332
- // Don't add the value if it already exists.
333
- if (!this._findOptionByValue(value, this._rawSelectedOptions)) {
334
- this._addSelectedOption(value);
335
- this._setOrAddToInputValue(value);
229
+ if (keyCode === ENTER) {
230
+ if (canAddCustomValue && hasFilterValue) {
231
+ result.optionAction = { type: 'add', option: value };
336
232
  }
337
- if (this.multiple) {
338
- this._updateFilter('');
233
+ else {
234
+ if (isMultiple) {
235
+ // TODO: Toggle the focused option
236
+ }
237
+ else {
238
+ // TODO: Select the focused option
239
+ }
339
240
  }
340
- return;
241
+ if (isMultiple) {
242
+ result.setFilter = '';
243
+ }
244
+ else {
245
+ result.overlayOperation = 'close';
246
+ }
247
+ return this._interpretKeyHandlerResult(result);
248
+ }
249
+ if (keyCode === SPACE) {
250
+ if (isMultiple) {
251
+ result.setFilter = '';
252
+ // TODO: Toggle the focused option
253
+ }
254
+ else {
255
+ result.overlayOperation = 'close';
256
+ // TODO: Select the focused option
257
+ }
258
+ return this._interpretKeyHandlerResult(result);
341
259
  }
342
260
  if (keyCode === ESCAPE) {
343
261
  if (isOpen) {
344
- this.close();
262
+ result.overlayOperation = 'close';
345
263
  }
346
- else {
347
- if (!isMultiple) {
348
- this._removeOrClearInputValue(null);
349
- this._updateFilter('');
350
- return;
351
- }
264
+ else if (!isMultiple) {
265
+ result.setFilter = '';
266
+ result.optionAction = 'clear';
352
267
  }
268
+ return this._interpretKeyHandlerResult(result);
353
269
  }
354
270
  if (keyCode === TAB) {
355
- this.close();
356
- return;
271
+ result.overlayOperation = 'close';
272
+ return this._interpretKeyHandlerResult(result);
357
273
  }
358
274
  if (!isOpen) {
359
- this.open();
275
+ result.overlayOperation = 'open';
360
276
  }
361
277
  if (keyCode === DOWN_ARROW) {
362
- // TODO: Implement
278
+ result.focusAction = 'next';
363
279
  }
364
280
  if (keyCode === UP_ARROW) {
365
- // TODO: Implement
281
+ result.focusAction = 'previous';
282
+ }
283
+ if (keyCode === PAGE_UP) {
284
+ result.focusAction = { type: 'offset', offset: -10 };
285
+ }
286
+ if (keyCode === PAGE_DOWN) {
287
+ result.focusAction = { type: 'offset', offset: 10 };
288
+ }
289
+ if (keyCode === HOME) {
290
+ result.focusAction = 'first';
291
+ }
292
+ if (keyCode === END) {
293
+ result.focusAction = 'last';
366
294
  }
295
+ if (keyCode === A && event.ctrlKey && isMultiple) {
296
+ result.optionAction = 'toggleAll';
297
+ event.preventDefault();
298
+ }
299
+ return this._interpretKeyHandlerResult(result);
367
300
  }
368
301
  processInputEvent(event) {
369
302
  const value = event.target.value;
@@ -372,10 +305,10 @@ class ComboboxComponent extends DecoratedInputBase {
372
305
  handleBlurEvent() {
373
306
  this.input._markAsTouched();
374
307
  this.input._setShouldDisplayError(true);
375
- if (this.multiple)
308
+ if (this._selectionModel.allowMultiple)
376
309
  return;
377
310
  if (this._currentFilter === '') {
378
- this._removeOrClearInputValue(null);
311
+ this._selectionModel.clearSelectedOptions();
379
312
  return;
380
313
  }
381
314
  this._setFilterFromInputValue();
@@ -384,7 +317,7 @@ class ComboboxComponent extends DecoratedInputBase {
384
317
  //#region Private Methods
385
318
  _initDispatchFilterChanges() {
386
319
  this._currentFilter$
387
- .pipe(debounceTime(300), distinctUntilChanged(), takeUntil(this._destroy$), tap((v) => this.filterChange.emit(v)))
320
+ .pipe(skipWhile(() => this.filterInternal), debounceTime(300), distinctUntilChanged(), takeUntil(this._destroy$), tap((v) => this.filterChange.emit(v)))
388
321
  .subscribe();
389
322
  }
390
323
  _initRepositionOnValueChanges() {
@@ -392,65 +325,6 @@ class ComboboxComponent extends DecoratedInputBase {
392
325
  .pipe(takeUntil(this._destroy$), debounceTime(0), tap(() => this._animatedOverlay._reposition()))
393
326
  .subscribe();
394
327
  }
395
- _updateSelectedOptionsViaInitialValue() {
396
- if (this.initialValue !== null && this.initialValue !== undefined) {
397
- if (Array.isArray(this.initialValue)) {
398
- this._addSelectedOptions(this.initialValue);
399
- }
400
- else {
401
- this._addSelectedOption(this.initialValue);
402
- }
403
- }
404
- }
405
- _addSelectedOption(option) {
406
- if (this.multiple) {
407
- this._rawSelectedOptions$.next([...this._rawSelectedOptions, option]);
408
- }
409
- else {
410
- this._rawSelectedOptions$.next([option]);
411
- }
412
- }
413
- _addSelectedOptions(options) {
414
- if (!this.multiple) {
415
- return;
416
- }
417
- this._rawSelectedOptions$.next([...this._rawSelectedOptions, ...options]);
418
- }
419
- _findOptionByValue(value, allOptions) {
420
- const option = allOptions.find((option) => {
421
- if (isObject(option) && this.bindValue) {
422
- return getObjectProperty(option, this.bindValue) === value;
423
- }
424
- return option === value;
425
- });
426
- return option;
427
- }
428
- _setOrAddToInputValue(value) {
429
- if (this.multiple) {
430
- if (Array.isArray(this._input.value)) {
431
- this._input._updateValue([...this._input.value, value]);
432
- }
433
- else {
434
- this._input._updateValue([value]);
435
- }
436
- }
437
- else {
438
- this._input._updateValue(value);
439
- }
440
- }
441
- _removeOrClearInputValue(value) {
442
- if (this.multiple) {
443
- if (Array.isArray(this._input.value)) {
444
- this._input._updateValue(this._input.value.filter((v) => v !== value));
445
- }
446
- else {
447
- this._input._updateValue([]);
448
- }
449
- }
450
- else {
451
- this._input._updateValue(null);
452
- }
453
- }
454
328
  _updateFilter(value) {
455
329
  if (this.input.nativeInputRef && this.input.nativeInputRef.element.nativeElement.value !== value) {
456
330
  this.input.nativeInputRef.element.nativeElement.value = value;
@@ -458,52 +332,106 @@ class ComboboxComponent extends DecoratedInputBase {
458
332
  if (this._currentFilter === value)
459
333
  return;
460
334
  this._currentFilter$.next(value);
461
- }
462
- _getOptionValueSync(option) {
463
- if (!this.bindValue || !isObject(option)) {
464
- return option;
335
+ if (this.filterInternal) {
336
+ this._selectionModel.setFilter(value);
465
337
  }
466
- return getObjectProperty(option, this.bindValue) ?? option;
467
- }
468
- _getOptionLabelSync(option) {
469
- if (!this.bindLabel || !isObject(option)) {
470
- return option;
471
- }
472
- return getObjectProperty(option, this.bindLabel) ?? option;
473
338
  }
474
339
  _setFilterFromInputValue() {
475
- if (this.multiple)
340
+ if (this._selectionModel.allowMultiple)
476
341
  return;
477
342
  const value = this.input.value;
478
343
  if (!value || Array.isArray(value)) {
479
344
  this._updateFilter('');
480
345
  return;
481
346
  }
482
- const option = this._findOptionByValue(value, this._rawSelectedOptions);
347
+ const option = this._selectionModel.getOptionByValue(value);
483
348
  if (!option)
484
349
  return;
485
- const label = this._getOptionLabelSync(option);
350
+ const label = this._selectionModel.getLabel(option);
486
351
  if (typeof label !== 'string')
487
352
  return;
488
353
  this._updateFilter(label);
489
354
  }
355
+ _interpretKeyHandlerResult(result) {
356
+ if (result.overlayOperation === 'close') {
357
+ this.close();
358
+ }
359
+ else if (result.overlayOperation === 'open') {
360
+ this.open();
361
+ }
362
+ if (result.setFilter !== undefined) {
363
+ this._updateFilter(result.setFilter);
364
+ }
365
+ if (result.optionAction) {
366
+ if (typeof result.optionAction === 'string') {
367
+ if (result.optionAction === 'clear') {
368
+ this._selectionModel.clearSelectedOptions();
369
+ }
370
+ else if (result.optionAction === 'toggleAll') {
371
+ this._selectionModel.toggleAllSelectedOptions();
372
+ }
373
+ }
374
+ else {
375
+ const { type, option } = result.optionAction;
376
+ if (type === 'add') {
377
+ this._selectionModel.addSelectedOption(option);
378
+ }
379
+ if (type === 'remove') {
380
+ this._selectionModel.removeSelectedOption(option);
381
+ }
382
+ if (type === 'toggle') {
383
+ this._selectionModel.toggleSelectedOption(option);
384
+ }
385
+ }
386
+ }
387
+ if (result.focusAction) {
388
+ if (typeof result.focusAction === 'string') {
389
+ if (result.focusAction === 'first') {
390
+ // TODO: Implement
391
+ }
392
+ if (result.focusAction === 'last') {
393
+ // TODO: Implement
394
+ }
395
+ if (result.focusAction === 'next') {
396
+ // TODO: Implement
397
+ }
398
+ if (result.focusAction === 'previous') {
399
+ // TODO: Implement
400
+ }
401
+ }
402
+ else {
403
+ const { type } = result.focusAction;
404
+ if (type === 'offset') {
405
+ const { offset } = result.focusAction;
406
+ // TODO: Implement
407
+ }
408
+ }
409
+ }
410
+ }
490
411
  //#endregion
491
412
  //#region Dev mode
492
413
  _debugValidateComboboxConfig(isRetry = false) {
493
- this.expectedOptionType$
414
+ combineLatest([this._selectionModel.labelBinding$, this._selectionModel.valueBinding$, this._allowCustomValues$])
494
415
  .pipe(skip(isRetry ? 1 : 0), // Skip if retrying to avoid infinite loop
495
- debounceTime(0), takeUntil(this._destroy$), tap((expectedOptionType) => {
496
- if (isEmptyArray(this.options)) {
416
+ debounceTime(0), takeUntil(this._destroy$), map(([bindLabel, bindValue, allowCustomValues]) => {
417
+ const shouldBeObjects = bindLabel && bindValue && !allowCustomValues;
418
+ if (shouldBeObjects) {
419
+ return ComboboxOptionType.Object;
420
+ }
421
+ return ComboboxOptionType.Primitive;
422
+ }), tap((expectedOptionType) => {
423
+ const options = this._selectionModel.options;
424
+ if (isEmptyArray(options)) {
497
425
  return;
498
426
  }
499
427
  if (expectedOptionType === ComboboxOptionType.Object) {
500
- if (!isObjectArray(this.options)) {
501
- throw comboboxError(1, this.options);
428
+ if (!isObjectArray(options)) {
429
+ throw comboboxError(1, options);
502
430
  }
503
431
  }
504
432
  else if (expectedOptionType === ComboboxOptionType.Primitive) {
505
- if (!isPrimitiveArray(this.options)) {
506
- throw comboboxError(2, this.options);
433
+ if (!isPrimitiveArray(options)) {
434
+ throw comboboxError(2, options);
507
435
  }
508
436
  }
509
437
  }), catchError((e) => {
@@ -513,12 +441,12 @@ class ComboboxComponent extends DecoratedInputBase {
513
441
  .subscribe();
514
442
  }
515
443
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.2", ngImport: i0, type: ComboboxComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
516
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.2", type: ComboboxComponent, isStandalone: true, selector: "et-combobox", inputs: { options: "options", initialValue: "initialValue", loading: "loading", error: "error", placeholder: "placeholder", multiple: "multiple", bindLabel: "bindLabel", bindValue: "bindValue", allowCustomValues: "allowCustomValues" }, outputs: { filterChange: "filterChange" }, host: { classAttribute: "et-combobox" }, providers: [
444
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.2", type: ComboboxComponent, isStandalone: true, selector: "et-combobox", inputs: { options: "options", initialValue: "initialValue", filterInternal: "filterInternal", loading: "loading", error: "error", placeholder: "placeholder", multiple: "multiple", bindLabel: "bindLabel", bindValue: "bindValue", allowCustomValues: "allowCustomValues" }, outputs: { filterChange: "filterChange" }, host: { classAttribute: "et-combobox" }, providers: [
517
445
  {
518
446
  provide: COMBOBOX_TOKEN,
519
447
  useExisting: ComboboxComponent,
520
448
  },
521
- ], usesInheritance: true, hostDirectives: [{ directive: i1.InputDirective }, { directive: i2.AnimatedOverlayDirective }], ngImport: i0, template: "<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n <ul *ngIf=\"multiple\" class=\"et-combobox-selected-options\">\n <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n {{ getOptionLabel(item) | async }}\n <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n </li>\n </ul>\n\n <input\n (keydown)=\"processKeydownEvent($event)\"\n (click)=\"open()\"\n (blur)=\"handleBlurEvent()\"\n (input)=\"processInputEvent($event)\"\n type=\"text\"\n etNativeInputRef\n />\n\n <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n <ng-content />\n</ng-template>\n", styles: [".et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-from,.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-to{opacity:0;transform:scaleY(0)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-active{transition:transform 125ms var(--ease-out-5),opacity 125ms var(--ease-out-5)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-active{transition:transform 50ms var(--ease-in-5),opacity 50ms var(--ease-in-5)}.et-combobox-body.et-with-default-animation[data-popper-placement^=top] .et-combobox-body-container{transform-origin:bottom}.et-combobox-body.et-with-default-animation[data-popper-placement^=bottom] .et-combobox-body-container{transform-origin:top}.et-combobox-body-container{display:block;background-color:#3d3d3d;color:#fff;max-height:300px;overflow:auto}@supports (overflow: overlay){.et-combobox-body-container{overflow:overlay}}.et-combobox-body{width:100%;display:grid}.et-combobox-option{display:block}.et-combobox{display:block;border:1px solid #ccc;padding:15px;position:relative}.et-combobox-chevron{transform:rotate(180deg);display:block;inline-size:15px;block-size:15px;position:absolute;inset-inline-end:15px;inset-block-start:7px}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NativeInputRefDirective, selector: "input[etNativeInputRef], textarea[etNativeInputRef], select[etNativeInputRef], button[etNativeInputRef]" }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }, { kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
449
+ ], usesInheritance: true, hostDirectives: [{ directive: i1.InputDirective }, { directive: i2.AnimatedOverlayDirective }], ngImport: i0, template: "<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n <ul *ngIf=\"multiple$ | async\" class=\"et-combobox-selected-options\">\n <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n {{ getOptionLabel(item) | async }}\n <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n </li>\n </ul>\n\n <input\n (keydown)=\"processKeydownEvent($event)\"\n (click)=\"open()\"\n (blur)=\"handleBlurEvent()\"\n (input)=\"processInputEvent($event)\"\n type=\"text\"\n etNativeInputRef\n />\n\n <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n <ng-content />\n</ng-template>\n", styles: [".et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-from,.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-to{opacity:0;transform:scaleY(0)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-active{transition:transform 125ms var(--ease-out-5),opacity 125ms var(--ease-out-5)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-active{transition:transform 50ms var(--ease-in-5),opacity 50ms var(--ease-in-5)}.et-combobox-body.et-with-default-animation[data-popper-placement^=top] .et-combobox-body-container{transform-origin:bottom}.et-combobox-body.et-with-default-animation[data-popper-placement^=bottom] .et-combobox-body-container{transform-origin:top}.et-combobox-body-container{display:block;background-color:#3d3d3d;color:#fff;max-height:300px;overflow:auto}@supports (overflow: overlay){.et-combobox-body-container{overflow:overlay}}.et-combobox-body{width:100%;display:grid}.et-combobox-option{display:block}.et-combobox{display:block;border:1px solid #ccc;padding:15px;position:relative}.et-combobox-chevron{transform:rotate(180deg);display:block;inline-size:15px;block-size:15px;position:absolute;inset-inline-end:15px;inset-block-start:7px}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NativeInputRefDirective, selector: "input[etNativeInputRef], textarea[etNativeInputRef], select[etNativeInputRef], button[etNativeInputRef]" }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }, { kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
522
450
  }
523
451
  export { ComboboxComponent };
524
452
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.2", ngImport: i0, type: ComboboxComponent, decorators: [{
@@ -530,12 +458,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.2", ngImpor
530
458
  provide: COMBOBOX_TOKEN,
531
459
  useExisting: ComboboxComponent,
532
460
  },
533
- ], template: "<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n <ul *ngIf=\"multiple\" class=\"et-combobox-selected-options\">\n <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n {{ getOptionLabel(item) | async }}\n <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n </li>\n </ul>\n\n <input\n (keydown)=\"processKeydownEvent($event)\"\n (click)=\"open()\"\n (blur)=\"handleBlurEvent()\"\n (input)=\"processInputEvent($event)\"\n type=\"text\"\n etNativeInputRef\n />\n\n <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n <ng-content />\n</ng-template>\n", styles: [".et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-from,.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-to{opacity:0;transform:scaleY(0)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-active{transition:transform 125ms var(--ease-out-5),opacity 125ms var(--ease-out-5)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-active{transition:transform 50ms var(--ease-in-5),opacity 50ms var(--ease-in-5)}.et-combobox-body.et-with-default-animation[data-popper-placement^=top] .et-combobox-body-container{transform-origin:bottom}.et-combobox-body.et-with-default-animation[data-popper-placement^=bottom] .et-combobox-body-container{transform-origin:top}.et-combobox-body-container{display:block;background-color:#3d3d3d;color:#fff;max-height:300px;overflow:auto}@supports (overflow: overlay){.et-combobox-body-container{overflow:overlay}}.et-combobox-body{width:100%;display:grid}.et-combobox-option{display:block}.et-combobox{display:block;border:1px solid #ccc;padding:15px;position:relative}.et-combobox-chevron{transform:rotate(180deg);display:block;inline-size:15px;block-size:15px;position:absolute;inset-inline-end:15px;inset-block-start:7px}\n"] }]
461
+ ], template: "<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n <ul *ngIf=\"multiple$ | async\" class=\"et-combobox-selected-options\">\n <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n {{ getOptionLabel(item) | async }}\n <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n </li>\n </ul>\n\n <input\n (keydown)=\"processKeydownEvent($event)\"\n (click)=\"open()\"\n (blur)=\"handleBlurEvent()\"\n (input)=\"processInputEvent($event)\"\n type=\"text\"\n etNativeInputRef\n />\n\n <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n <ng-content />\n</ng-template>\n", styles: [".et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-from,.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-to{opacity:0;transform:scaleY(0)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-enter-active{transition:transform 125ms var(--ease-out-5),opacity 125ms var(--ease-out-5)}.et-combobox-body.et-with-default-animation .et-combobox-body-container.et-animation-leave-active{transition:transform 50ms var(--ease-in-5),opacity 50ms var(--ease-in-5)}.et-combobox-body.et-with-default-animation[data-popper-placement^=top] .et-combobox-body-container{transform-origin:bottom}.et-combobox-body.et-with-default-animation[data-popper-placement^=bottom] .et-combobox-body-container{transform-origin:top}.et-combobox-body-container{display:block;background-color:#3d3d3d;color:#fff;max-height:300px;overflow:auto}@supports (overflow: overlay){.et-combobox-body-container{overflow:overlay}}.et-combobox-body{width:100%;display:grid}.et-combobox-option{display:block}.et-combobox{display:block;border:1px solid #ccc;padding:15px;position:relative}.et-combobox-chevron{transform:rotate(180deg);display:block;inline-size:15px;block-size:15px;position:absolute;inset-inline-end:15px;inset-block-start:7px}\n"] }]
534
462
  }], ctorParameters: function () { return []; }, propDecorators: { options: [{
535
463
  type: Input,
536
464
  args: [{ required: true }]
537
465
  }], initialValue: [{
538
466
  type: Input
467
+ }], filterInternal: [{
468
+ type: Input
539
469
  }], loading: [{
540
470
  type: Input
541
471
  }], error: [{
@@ -553,4 +483,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.2", ngImpor
553
483
  }], filterChange: [{
554
484
  type: Output
555
485
  }] } });
556
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"combobox.component.js","sourceRoot":"","sources":["../../../../../../../../../../../../../libs/cdk/src/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.ts","../../../../../../../../../../../../../libs/cdk/src/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.html"],"names":[],"mappings":"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","sourcesContent":["import { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { DOWN_ARROW, ENTER, ESCAPE, TAB, UP_ARROW } from '@angular/cdk/keycodes';\nimport { AsyncPipe, NgFor, NgIf } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  InjectionToken,\n  Input,\n  OnInit,\n  Output,\n  ViewEncapsulation,\n  inject,\n  isDevMode,\n} from '@angular/core';\nimport { AnimatedOverlayDirective, LetDirective, clone } from '@ethlete/core';\nimport {\n  BehaviorSubject,\n  catchError,\n  combineLatest,\n  debounceTime,\n  distinctUntilChanged,\n  map,\n  of,\n  shareReplay,\n  skip,\n  takeUntil,\n  tap,\n  throwError,\n} from 'rxjs';\nimport { ChevronIconComponent } from '../../../../../../../icons';\nimport { INPUT_TOKEN, InputDirective, NativeInputRefDirective } from '../../../../../../directives';\nimport { DecoratedInputBase } from '../../../../../../utils';\nimport { ComboboxBodyComponent } from '../../partials';\n\nexport class RuntimeError<T extends number> extends Error {\n  constructor(public code: T, message: null | false | string, public data: unknown = '__ET_NO_DATA__') {\n    super(formatRuntimeError<T>(code, message));\n\n    if (data !== '__ET_NO_DATA__') {\n      try {\n        const _data = clone(data);\n\n        setTimeout(() => {\n          console.error(_data);\n        }, 1);\n      } catch {\n        setTimeout(() => {\n          console.error(data);\n        }, 1);\n      }\n    }\n  }\n}\n\nexport function formatRuntimeError<T extends number>(code: T, message: null | false | string): string {\n  // prefix code with zeros if it's less than 100\n  const codeWithZeros = code < 10 ? `00${code}` : code < 100 ? `0${code}` : code;\n\n  const fullCode = `ET${codeWithZeros}`;\n\n  return `${fullCode}${message ? ': ' + message : ''}`;\n}\n\nconst COMBOBOX_ERRORS = {\n  1: 'Expected options to be an array of objects. This is due to \"bindLabel\" and \"bindValue\" being set.',\n  2: 'Expected options to be an array of primitives. This is due to \"bindLabel\" and \"bindValue\" not being set or \"allowCustomValues\" being set to true.',\n} as const;\n\nconst comboboxError = (code: keyof typeof COMBOBOX_ERRORS, data: unknown) => {\n  const message = `<et-combobox>: ${COMBOBOX_ERRORS[code]}`;\n\n  throw new RuntimeError(code, message, data);\n};\n\nconst isPrimitiveArray = (value: unknown): value is Array<string | number | boolean> => {\n  if (!Array.isArray(value)) return false;\n\n  const first = value[0];\n  const last = value[value.length - 1];\n\n  if (!first || !last) return false;\n\n  return typeof first !== 'object' && typeof last !== 'object';\n};\n\nconst isObjectArray = (value: unknown): value is Array<Record<string, unknown>> => {\n  if (!Array.isArray(value)) return false;\n\n  const first = value[0];\n  const last = value[value.length - 1];\n\n  if (!first || !last) return false;\n\n  return typeof first === 'object' && typeof last === 'object';\n};\n\nconst isEmptyArray = (value: unknown): value is [] => {\n  return Array.isArray(value) && value.length === 0;\n};\n\nconst isObject = (value: unknown): value is Record<string, unknown> => {\n  return typeof value === 'object' && value !== null;\n};\n\nconst ComboboxOptionType = {\n  Primitive: 'primitive',\n  Object: 'object',\n} as const;\n\nconst ComboboxInputValueType = {\n  Single: 'single',\n  Array: 'array',\n} as const;\n\nconst getObjectProperty = (obj: Record<string, unknown>, prop: string) => {\n  const hasDotNotation = prop.includes('.');\n\n  if (!hasDotNotation) return obj[prop];\n\n  const props = prop.split('.');\n\n  let value: unknown = obj;\n\n  for (const prop of props) {\n    if (!isObject(value)) return undefined;\n\n    value = value[prop];\n  }\n\n  return value;\n};\n\nexport const COMBOBOX_TOKEN = new InjectionToken<ComboboxComponent>('ET_COMBOBOX_TOKEN');\n\n@Component({\n  selector: 'et-combobox',\n  templateUrl: './combobox.component.html',\n  styleUrls: ['./combobox.component.scss'],\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    class: 'et-combobox',\n  },\n  imports: [NgIf, NativeInputRefDirective, AsyncPipe, ChevronIconComponent, LetDirective, NgFor],\n  hostDirectives: [{ directive: InputDirective }, AnimatedOverlayDirective],\n  providers: [\n    {\n      provide: COMBOBOX_TOKEN,\n      useExisting: ComboboxComponent,\n    },\n  ],\n})\nexport class ComboboxComponent extends DecoratedInputBase implements OnInit {\n  //#region Injects\n\n  private readonly _input = inject(INPUT_TOKEN);\n  private readonly _animatedOverlay = inject<AnimatedOverlayDirective<ComboboxBodyComponent>>(AnimatedOverlayDirective);\n\n  //#endregion\n\n  //#region Inputs\n\n  @Input({ required: true })\n  get options() {\n    return this._options$.value;\n  }\n  set options(value: unknown[]) {\n    this._options$.next(value);\n  }\n  private _options$ = new BehaviorSubject<unknown[]>([]);\n  readonly options$ = this._options$.asObservable();\n\n  @Input()\n  get initialValue() {\n    return this._initialValue$.value;\n  }\n  set initialValue(value: unknown) {\n    this._initialValue$.next(value);\n\n    this._updateSelectedOptionsViaInitialValue();\n  }\n  private _initialValue$ = new BehaviorSubject<unknown>(null);\n\n  @Input()\n  get loading(): boolean {\n    return this._loading$.value;\n  }\n  set loading(value: BooleanInput) {\n    this._loading$.next(coerceBooleanProperty(value));\n  }\n  private _loading$ = new BehaviorSubject(false);\n  readonly loading$ = this._loading$.asObservable();\n\n  @Input()\n  get error() {\n    return this._error$.value;\n  }\n  set error(value: unknown) {\n    this._error$.next(value);\n  }\n  private _error$ = new BehaviorSubject<unknown>(null);\n\n  @Input()\n  get placeholder() {\n    return this._placeholder$.value;\n  }\n  set placeholder(value: string | null) {\n    this._placeholder$.next(value);\n  }\n  private _placeholder$ = new BehaviorSubject<string | null>(null);\n\n  @Input()\n  get multiple(): boolean {\n    return this._multiple$.value;\n  }\n  set multiple(value: BooleanInput) {\n    this._multiple$.next(coerceBooleanProperty(value));\n  }\n  private _multiple$ = new BehaviorSubject(false);\n\n  @Input()\n  get bindLabel() {\n    return this._bindLabel$.value;\n  }\n  set bindLabel(value: string | null) {\n    this._bindLabel$.next(value);\n  }\n  private _bindLabel$ = new BehaviorSubject<string | null>(null);\n\n  @Input()\n  get bindValue() {\n    return this._bindValue$.value;\n  }\n  set bindValue(value: string | null) {\n    this._bindValue$.next(value);\n  }\n  private _bindValue$ = new BehaviorSubject<string | null>(null);\n\n  @Input()\n  get allowCustomValues(): boolean {\n    return this._allowCustomValues$.value;\n  }\n  set allowCustomValues(value: BooleanInput) {\n    this._allowCustomValues$.next(coerceBooleanProperty(value));\n  }\n  private _allowCustomValues$ = new BehaviorSubject(false);\n\n  //#endregion\n\n  //#region Outputs\n\n  @Output()\n  protected readonly filterChange = new EventEmitter<string>();\n\n  //#endregion\n\n  //#region Members\n\n  private get _rawSelectedOptions() {\n    return this._rawSelectedOptions$.value;\n  }\n  private readonly _rawSelectedOptions$ = new BehaviorSubject<unknown[]>([]);\n\n  private get _currentFilter() {\n    return this._currentFilter$.value;\n  }\n  private readonly _currentFilter$ = new BehaviorSubject<string>('');\n\n  private get _isOpen() {\n    return this._animatedOverlay.isMounted;\n  }\n  private readonly _isOpen$ = this._animatedOverlay.isMounted$;\n\n  //#endregion\n\n  //#region Computes\n\n  private readonly _allOptions$ = combineLatest([this._options$, this._rawSelectedOptions$]).pipe(\n    map(([options, rawSelectedOptions]) => [...options, ...rawSelectedOptions]),\n    shareReplay({ bufferSize: 1, refCount: true }),\n  );\n\n  protected readonly expectedOptionType$ = combineLatest([\n    this._bindLabel$,\n    this._bindValue$,\n    this._allowCustomValues$,\n  ]).pipe(\n    map(([bindLabel, bindValue, allowCustomValues]) => {\n      const shouldBeObjects = bindLabel && bindValue && !allowCustomValues;\n\n      if (shouldBeObjects) {\n        return ComboboxOptionType.Object;\n      }\n\n      return ComboboxOptionType.Primitive;\n    }),\n  );\n\n  protected readonly expectedInputValueType$ = this._multiple$.pipe(\n    map((multiple) => (multiple ? ComboboxInputValueType.Array : ComboboxInputValueType.Single)),\n  );\n\n  protected readonly selectedOptions$ = combineLatest([\n    this._input.value$,\n    this.expectedInputValueType$,\n    this._allOptions$,\n  ]).pipe(\n    map(([inputValue, expectedInputValueType, allOptions]) => {\n      if (expectedInputValueType === ComboboxInputValueType.Single) {\n        if (Array.isArray(inputValue)) {\n          return [];\n        }\n\n        const option = this._findOptionByValue(inputValue, allOptions);\n\n        return [option];\n      } else if (expectedInputValueType === ComboboxInputValueType.Array) {\n        if (!Array.isArray(inputValue)) {\n          return [];\n        }\n\n        const options = inputValue.map((value) => this._findOptionByValue(value, allOptions)).filter((v) => !!v);\n\n        if (!options.length) return [];\n\n        return options;\n      }\n\n      return [];\n    }),\n    map((options) => options.filter((v) => !!v)),\n  );\n\n  //#endregion\n\n  //#region Lifecycle\n\n  constructor() {\n    super();\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--loading',\n      observable: this._loading$,\n    });\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--error',\n      observable: this._error$.pipe(map((v) => !!v)),\n    });\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--open',\n      observable: this._isOpen$,\n    });\n  }\n\n  ngOnInit(): void {\n    this._initDispatchFilterChanges();\n    this._initRepositionOnValueChanges();\n\n    if (isDevMode()) {\n      this._debugValidateComboboxConfig();\n    }\n  }\n\n  //#endregion\n\n  //#region Public Methods\n\n  getOptionLabel(option: unknown) {\n    if (!this.bindLabel || !isObject(option)) {\n      return of(option);\n    }\n\n    return this._bindLabel$.pipe(\n      map((label) => {\n        if (!label) return option;\n\n        return getObjectProperty(option, label) ?? option;\n      }),\n    );\n  }\n\n  getOptionValue(option: unknown) {\n    if (!this.bindValue || !isObject(option)) {\n      return of(option);\n    }\n\n    return this._bindValue$.pipe(\n      map((value) => {\n        if (!value) return option;\n\n        return getObjectProperty(option, value) ?? option;\n      }),\n    );\n  }\n\n  removeSelectedOption(option: unknown) {\n    this.input._markAsTouched();\n\n    const selectedOptions = this._rawSelectedOptions;\n    const optionValue = this._getOptionValueSync(option);\n\n    const index = selectedOptions.findIndex((v) => {\n      const val = this._getOptionValueSync(v);\n\n      return val === optionValue;\n    });\n\n    if (index === -1) return;\n\n    selectedOptions.splice(index, 1);\n\n    this._rawSelectedOptions$.next(selectedOptions);\n\n    this._removeOrClearInputValue(optionValue);\n  }\n\n  open() {\n    // if (!this._selectBodyConfig) return;\n\n    if (this._isOpen || this.input.disabled) return;\n\n    // this._setSelectedOptionActive();\n\n    const instance = this._animatedOverlay.mount({\n      component: ComboboxBodyComponent,\n      mirrorWidth: true,\n      // data: { _bodyTemplate: this._selectBodyConfig.template },\n    });\n\n    if (!instance) return;\n\n    // this._selectBodyId$.next(instance.selectBody.id);\n  }\n\n  close() {\n    if (!this._isOpen) return;\n\n    this._animatedOverlay.unmount();\n\n    // this._selectBodyId$.next(null);\n  }\n\n  writeValueFromOption(option: unknown) {\n    this.input._markAsTouched();\n\n    const optionValue = this._getOptionValueSync(option);\n\n    if (this._findOptionByValue(optionValue, this._rawSelectedOptions)) {\n      if (this.multiple) {\n        this.removeSelectedOption(option);\n        this._removeOrClearInputValue(optionValue);\n      }\n\n      return;\n    }\n\n    this._setOrAddToInputValue(optionValue);\n    this._addSelectedOption(option);\n\n    if (!this.multiple) {\n      this.close();\n      this._setFilterFromInputValue();\n    }\n  }\n\n  isOptionSelected(option: unknown) {\n    return combineLatest([this._rawSelectedOptions$, this.getOptionValue(option)]).pipe(\n      map(([selectedOptions, value]) => !!this._findOptionByValue(value, selectedOptions)),\n    );\n  }\n\n  //#endregion\n\n  //#region Protected Methods\n\n  protected processKeydownEvent(event: KeyboardEvent) {\n    const keyCode = event.keyCode;\n    const isOpen = this._isOpen;\n    const isMultiple = this.multiple;\n    const canAddCustomValue = this.allowCustomValues;\n    const value = (event.target as HTMLInputElement).value;\n    const hasValue = !!value;\n\n    // The user typed a custom value and pressed enter. Add it to the selected options.\n    if (keyCode === ENTER && canAddCustomValue && hasValue) {\n      // Don't add the value if it already exists.\n      if (!this._findOptionByValue(value, this._rawSelectedOptions)) {\n        this._addSelectedOption(value);\n        this._setOrAddToInputValue(value);\n      }\n\n      if (this.multiple) {\n        this._updateFilter('');\n      }\n\n      return;\n    }\n\n    if (keyCode === ESCAPE) {\n      if (isOpen) {\n        this.close();\n      } else {\n        if (!isMultiple) {\n          this._removeOrClearInputValue(null);\n          this._updateFilter('');\n          return;\n        }\n      }\n    }\n\n    if (keyCode === TAB) {\n      this.close();\n      return;\n    }\n\n    if (!isOpen) {\n      this.open();\n    }\n\n    if (keyCode === DOWN_ARROW) {\n      // TODO: Implement\n    }\n\n    if (keyCode === UP_ARROW) {\n      // TODO: Implement\n    }\n  }\n\n  protected processInputEvent(event: Event) {\n    const value = (event.target as HTMLInputElement).value;\n\n    this._updateFilter(value);\n  }\n\n  protected handleBlurEvent() {\n    this.input._markAsTouched();\n    this.input._setShouldDisplayError(true);\n\n    if (this.multiple) return;\n\n    if (this._currentFilter === '') {\n      this._removeOrClearInputValue(null);\n      return;\n    }\n\n    this._setFilterFromInputValue();\n  }\n\n  //#endregion\n\n  //#region Private Methods\n\n  private _initDispatchFilterChanges() {\n    this._currentFilter$\n      .pipe(\n        debounceTime(300),\n        distinctUntilChanged(),\n        takeUntil(this._destroy$),\n        tap((v) => this.filterChange.emit(v)),\n      )\n      .subscribe();\n  }\n\n  private _initRepositionOnValueChanges() {\n    this.input.valueChange$\n      .pipe(\n        takeUntil(this._destroy$),\n        debounceTime(0),\n        tap(() => this._animatedOverlay._reposition()),\n      )\n      .subscribe();\n  }\n\n  private _updateSelectedOptionsViaInitialValue() {\n    if (this.initialValue !== null && this.initialValue !== undefined) {\n      if (Array.isArray(this.initialValue)) {\n        this._addSelectedOptions(this.initialValue);\n      } else {\n        this._addSelectedOption(this.initialValue);\n      }\n    }\n  }\n\n  private _addSelectedOption(option: unknown) {\n    if (this.multiple) {\n      this._rawSelectedOptions$.next([...this._rawSelectedOptions, option]);\n    } else {\n      this._rawSelectedOptions$.next([option]);\n    }\n  }\n\n  private _addSelectedOptions(options: unknown[]) {\n    if (!this.multiple) {\n      return;\n    }\n\n    this._rawSelectedOptions$.next([...this._rawSelectedOptions, ...options]);\n  }\n\n  private _findOptionByValue(value: unknown, allOptions: unknown[]) {\n    const option = allOptions.find((option) => {\n      if (isObject(option) && this.bindValue) {\n        return getObjectProperty(option, this.bindValue) === value;\n      }\n\n      return option === value;\n    });\n\n    return option;\n  }\n\n  private _setOrAddToInputValue(value: unknown) {\n    if (this.multiple) {\n      if (Array.isArray(this._input.value)) {\n        this._input._updateValue([...this._input.value, value]);\n      } else {\n        this._input._updateValue([value]);\n      }\n    } else {\n      this._input._updateValue(value);\n    }\n  }\n\n  private _removeOrClearInputValue(value: unknown) {\n    if (this.multiple) {\n      if (Array.isArray(this._input.value)) {\n        this._input._updateValue(this._input.value.filter((v) => v !== value));\n      } else {\n        this._input._updateValue([]);\n      }\n    } else {\n      this._input._updateValue(null);\n    }\n  }\n\n  private _updateFilter(value: string) {\n    if (this.input.nativeInputRef && this.input.nativeInputRef.element.nativeElement.value !== value) {\n      this.input.nativeInputRef.element.nativeElement.value = value;\n    }\n\n    if (this._currentFilter === value) return;\n\n    this._currentFilter$.next(value);\n  }\n\n  private _getOptionValueSync(option: unknown) {\n    if (!this.bindValue || !isObject(option)) {\n      return option;\n    }\n\n    return getObjectProperty(option, this.bindValue) ?? option;\n  }\n\n  private _getOptionLabelSync(option: unknown) {\n    if (!this.bindLabel || !isObject(option)) {\n      return option;\n    }\n\n    return getObjectProperty(option, this.bindLabel) ?? option;\n  }\n\n  private _setFilterFromInputValue() {\n    if (this.multiple) return;\n\n    const value = this.input.value;\n\n    if (!value || Array.isArray(value)) {\n      this._updateFilter('');\n      return;\n    }\n\n    const option = this._findOptionByValue(value, this._rawSelectedOptions);\n\n    if (!option) return;\n\n    const label = this._getOptionLabelSync(option);\n\n    if (typeof label !== 'string') return;\n\n    this._updateFilter(label);\n  }\n\n  //#endregion\n\n  //#region Dev mode\n\n  private _debugValidateComboboxConfig(isRetry = false) {\n    this.expectedOptionType$\n      .pipe(\n        skip(isRetry ? 1 : 0), // Skip if retrying to avoid infinite loop\n        debounceTime(0),\n        takeUntil(this._destroy$),\n        tap((expectedOptionType) => {\n          if (isEmptyArray(this.options)) {\n            return;\n          }\n\n          if (expectedOptionType === ComboboxOptionType.Object) {\n            if (!isObjectArray(this.options)) {\n              throw comboboxError(1, this.options);\n            }\n          } else if (expectedOptionType === ComboboxOptionType.Primitive) {\n            if (!isPrimitiveArray(this.options)) {\n              throw comboboxError(2, this.options);\n            }\n          }\n        }),\n        catchError((e) => {\n          this._debugValidateComboboxConfig(true);\n          return throwError(() => e);\n        }),\n      )\n      .subscribe();\n  }\n\n  //#endregion\n}\n","<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n  <ul *ngIf=\"multiple\" class=\"et-combobox-selected-options\">\n    <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n      {{ getOptionLabel(item) | async }}\n      <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n    </li>\n  </ul>\n\n  <input\n    (keydown)=\"processKeydownEvent($event)\"\n    (click)=\"open()\"\n    (blur)=\"handleBlurEvent()\"\n    (input)=\"processInputEvent($event)\"\n    type=\"text\"\n    etNativeInputRef\n  />\n\n  <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n  <ng-content />\n</ng-template>\n"]}
486
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"combobox.component.js","sourceRoot":"","sources":["../../../../../../../../../../../../../libs/cdk/src/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.ts","../../../../../../../../../../../../../libs/cdk/src/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.html"],"names":[],"mappings":"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","sourcesContent":["import { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport {\n  A,\n  DOWN_ARROW,\n  END,\n  ENTER,\n  ESCAPE,\n  HOME,\n  PAGE_DOWN,\n  PAGE_UP,\n  SPACE,\n  TAB,\n  UP_ARROW,\n} from '@angular/cdk/keycodes';\nimport { AsyncPipe, NgFor, NgIf } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  InjectionToken,\n  Input,\n  OnInit,\n  Output,\n  ViewEncapsulation,\n  inject,\n  isDevMode,\n} from '@angular/core';\nimport { AnimatedOverlayDirective, LetDirective, RuntimeError, SelectionModel } from '@ethlete/core';\nimport {\n  BehaviorSubject,\n  catchError,\n  combineLatest,\n  debounceTime,\n  distinctUntilChanged,\n  map,\n  skip,\n  skipWhile,\n  takeUntil,\n  tap,\n  throwError,\n} from 'rxjs';\nimport { ChevronIconComponent } from '../../../../../../../icons';\nimport { INPUT_TOKEN, InputDirective, NativeInputRefDirective } from '../../../../../../directives';\nimport { DecoratedInputBase } from '../../../../../../utils';\nimport { SELECT_FIELD_TOKEN } from '../../../../directives';\nimport { ComboboxBodyComponent } from '../../partials';\n\nconst COMBOBOX_ERRORS = {\n  1: 'Expected options to be an array of objects. This is due to \"bindLabel\" and \"bindValue\" being set.',\n  2: 'Expected options to be an array of primitives. This is due to \"bindLabel\" and \"bindValue\" not being set or \"allowCustomValues\" being set to true.',\n} as const;\n\nconst comboboxError = (code: keyof typeof COMBOBOX_ERRORS, data: unknown) => {\n  const message = `<et-combobox>: ${COMBOBOX_ERRORS[code]}`;\n\n  throw new RuntimeError(code, message, data);\n};\n\nconst isPrimitiveArray = (value: unknown): value is Array<string | number | boolean> => {\n  if (!Array.isArray(value)) return false;\n\n  const first = value[0];\n  const last = value[value.length - 1];\n\n  if (!first || !last) return false;\n\n  return typeof first !== 'object' && typeof last !== 'object';\n};\n\nconst isObjectArray = (value: unknown): value is Array<Record<string, unknown>> => {\n  if (!Array.isArray(value)) return false;\n\n  const first = value[0];\n  const last = value[value.length - 1];\n\n  if (!first || !last) return false;\n\n  return typeof first === 'object' && typeof last === 'object';\n};\n\nconst isEmptyArray = (value: unknown): value is [] => {\n  return Array.isArray(value) && value.length === 0;\n};\n\nconst ComboboxOptionType = {\n  Primitive: 'primitive',\n  Object: 'object',\n} as const;\n\nexport const COMBOBOX_TOKEN = new InjectionToken<ComboboxComponent>('ET_COMBOBOX_TOKEN');\n\ninterface KeyHandlerResult {\n  setFilter?: string;\n  overlayOperation?: 'open' | 'close';\n  optionAction?:\n    | {\n        type: 'add';\n        option: unknown;\n      }\n    | {\n        type: 'remove';\n        option: unknown;\n      }\n    | {\n        type: 'toggle';\n        option: unknown;\n      }\n    | 'clear'\n    | 'toggleAll';\n  focusAction?: 'first' | 'last' | 'next' | 'previous' | { type: 'offset'; offset: number };\n}\n\n@Component({\n  selector: 'et-combobox',\n  templateUrl: './combobox.component.html',\n  styleUrls: ['./combobox.component.scss'],\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    class: 'et-combobox',\n  },\n  imports: [NgIf, NativeInputRefDirective, AsyncPipe, ChevronIconComponent, LetDirective, NgFor],\n  hostDirectives: [{ directive: InputDirective }, AnimatedOverlayDirective],\n  providers: [\n    {\n      provide: COMBOBOX_TOKEN,\n      useExisting: ComboboxComponent,\n    },\n  ],\n})\nexport class ComboboxComponent extends DecoratedInputBase implements OnInit {\n  //#region Injects\n\n  private readonly _input = inject(INPUT_TOKEN);\n  private readonly _animatedOverlay = inject<AnimatedOverlayDirective<ComboboxBodyComponent>>(AnimatedOverlayDirective);\n  private readonly _selectField = inject(SELECT_FIELD_TOKEN);\n\n  //#endregion\n\n  //#region Inputs\n\n  @Input({ required: true })\n  get options() {\n    return this._selectionModel.getFilteredOptions();\n  }\n  set options(value: unknown[]) {\n    this._selectionModel.setOptions(value);\n  }\n\n  @Input()\n  set initialValue(value: unknown) {\n    this._selectionModel.setSelection(value);\n  }\n\n  @Input()\n  get filterInternal(): boolean {\n    return this._filterInternal$.value;\n  }\n  set filterInternal(value: BooleanInput) {\n    const val = coerceBooleanProperty(value);\n    this._filterInternal$.next(val);\n\n    if (!val) {\n      this._selectionModel.setFilter('');\n    } else {\n      this._selectionModel.setFilter(this._currentFilter);\n    }\n  }\n  private _filterInternal$ = new BehaviorSubject(true);\n\n  @Input()\n  get loading(): boolean {\n    return this._loading$.value;\n  }\n  set loading(value: BooleanInput) {\n    this._loading$.next(coerceBooleanProperty(value));\n  }\n  private _loading$ = new BehaviorSubject(false);\n  readonly loading$ = this._loading$.asObservable();\n\n  @Input()\n  get error() {\n    return this._error$.value;\n  }\n  set error(value: unknown) {\n    this._error$.next(value);\n  }\n  private _error$ = new BehaviorSubject<unknown>(null);\n\n  @Input()\n  get placeholder() {\n    return this._placeholder$.value;\n  }\n  set placeholder(value: string | null) {\n    this._placeholder$.next(value);\n  }\n  private _placeholder$ = new BehaviorSubject<string | null>(null);\n\n  @Input()\n  set multiple(value: BooleanInput) {\n    this._selectionModel.setAllowMultiple(coerceBooleanProperty(value));\n  }\n\n  @Input()\n  set bindLabel(value: string | null) {\n    this._selectionModel.setLabelBinding(value);\n  }\n\n  @Input()\n  set bindValue(value: string | null) {\n    this._selectionModel.setValueBinding(value);\n  }\n\n  @Input()\n  get allowCustomValues(): boolean {\n    return this._allowCustomValues$.value;\n  }\n  set allowCustomValues(value: BooleanInput) {\n    this._allowCustomValues$.next(coerceBooleanProperty(value));\n  }\n  private _allowCustomValues$ = new BehaviorSubject(false);\n\n  //#endregion\n\n  //#region Outputs\n\n  @Output()\n  protected readonly filterChange = new EventEmitter<string>();\n\n  //#endregion\n\n  //#region Members\n\n  private get _currentFilter() {\n    return this._currentFilter$.value;\n  }\n  private readonly _currentFilter$ = new BehaviorSubject<string>('');\n\n  private get _isOpen() {\n    return this._animatedOverlay.isMounted;\n  }\n  private readonly _isOpen$ = this._animatedOverlay.isMounted$;\n\n  private readonly _selectionModel = new SelectionModel();\n\n  readonly selectedOptions$ = this._selectionModel.selection$;\n  readonly multiple$ = this._selectionModel.allowMultiple$;\n  readonly options$ = this._selectionModel.filteredOptions$;\n\n  //#endregion\n\n  //#region Computes\n\n  //#endregion\n\n  //#region Lifecycle\n\n  constructor() {\n    super();\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--loading',\n      observable: this._loading$,\n    });\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--error',\n      observable: this._error$.pipe(map((v) => !!v)),\n    });\n\n    this._bindings.push({\n      attribute: 'class.et-combobox--open',\n      observable: this._isOpen$,\n    });\n\n    this._selectField._bindings.push({\n      attribute: 'class.et-select-field--open',\n      observable: this._isOpen$,\n    });\n  }\n\n  ngOnInit(): void {\n    this._initDispatchFilterChanges();\n    this._initRepositionOnValueChanges();\n\n    if (isDevMode()) {\n      this._debugValidateComboboxConfig();\n    }\n\n    this._selectionModel.value$\n      .pipe(\n        takeUntil(this._destroy$),\n        tap((value) => {\n          this._input._updateValue(value);\n          this._setFilterFromInputValue();\n        }),\n      )\n      .subscribe();\n  }\n\n  //#endregion\n\n  //#region Public Methods\n\n  getOptionLabel(option: unknown) {\n    return this._selectionModel.getLabel$(option);\n  }\n\n  getOptionValue(option: unknown) {\n    return this._selectionModel.getValue$(option);\n  }\n\n  removeSelectedOption(option: unknown) {\n    this._selectionModel.removeSelectedOption(option);\n\n    this.input._markAsTouched();\n  }\n\n  open() {\n    // if (!this._selectBodyConfig) return;\n\n    if (this._isOpen || this.input.disabled) return;\n\n    // this._setSelectedOptionActive();\n\n    const instance = this._animatedOverlay.mount({\n      component: ComboboxBodyComponent,\n      mirrorWidth: true,\n      // data: { _bodyTemplate: this._selectBodyConfig.template },\n    });\n\n    if (!instance) return;\n\n    // this._selectBodyId$.next(instance.selectBody.id);\n  }\n\n  close() {\n    if (!this._isOpen) return;\n\n    this._animatedOverlay.unmount();\n\n    // this._selectBodyId$.next(null);\n  }\n\n  writeValueFromOption(option: unknown) {\n    this.input._markAsTouched();\n\n    if (this._selectionModel.allowMultiple) {\n      this._selectionModel.toggleSelectedOption(option);\n    } else {\n      this._selectionModel.addSelectedOption(option);\n    }\n\n    if (!this._selectionModel.allowMultiple) {\n      this.close();\n      this._setFilterFromInputValue();\n    }\n  }\n\n  isOptionSelected(option: unknown) {\n    return this._selectionModel.isSelected$(option);\n  }\n\n  //#endregion\n\n  //#region Protected Methods\n\n  protected processKeydownEvent(event: KeyboardEvent) {\n    const keyCode = event.keyCode;\n    const isOpen = this._isOpen;\n    const isMultiple = this._selectionModel.allowMultiple;\n    const canAddCustomValue = this.allowCustomValues;\n    const value = (event.target as HTMLInputElement).value;\n    const hasFilterValue = !!value;\n\n    const result: KeyHandlerResult = {};\n\n    // The user typed a custom value and pressed enter. Add it to the selected options.\n    if (keyCode === ENTER) {\n      if (canAddCustomValue && hasFilterValue) {\n        result.optionAction = { type: 'add', option: value };\n      } else {\n        if (isMultiple) {\n          // TODO: Toggle the focused option\n        } else {\n          // TODO: Select the focused option\n        }\n      }\n\n      if (isMultiple) {\n        result.setFilter = '';\n      } else {\n        result.overlayOperation = 'close';\n      }\n\n      return this._interpretKeyHandlerResult(result);\n    }\n\n    if (keyCode === SPACE) {\n      if (isMultiple) {\n        result.setFilter = '';\n        // TODO: Toggle the focused option\n      } else {\n        result.overlayOperation = 'close';\n        // TODO: Select the focused option\n      }\n\n      return this._interpretKeyHandlerResult(result);\n    }\n\n    if (keyCode === ESCAPE) {\n      if (isOpen) {\n        result.overlayOperation = 'close';\n      } else if (!isMultiple) {\n        result.setFilter = '';\n        result.optionAction = 'clear';\n      }\n\n      return this._interpretKeyHandlerResult(result);\n    }\n\n    if (keyCode === TAB) {\n      result.overlayOperation = 'close';\n      return this._interpretKeyHandlerResult(result);\n    }\n\n    if (!isOpen) {\n      result.overlayOperation = 'open';\n    }\n\n    if (keyCode === DOWN_ARROW) {\n      result.focusAction = 'next';\n    }\n\n    if (keyCode === UP_ARROW) {\n      result.focusAction = 'previous';\n    }\n\n    if (keyCode === PAGE_UP) {\n      result.focusAction = { type: 'offset', offset: -10 };\n    }\n\n    if (keyCode === PAGE_DOWN) {\n      result.focusAction = { type: 'offset', offset: 10 };\n    }\n\n    if (keyCode === HOME) {\n      result.focusAction = 'first';\n    }\n\n    if (keyCode === END) {\n      result.focusAction = 'last';\n    }\n\n    if (keyCode === A && event.ctrlKey && isMultiple) {\n      result.optionAction = 'toggleAll';\n      event.preventDefault();\n    }\n\n    return this._interpretKeyHandlerResult(result);\n  }\n\n  protected processInputEvent(event: Event) {\n    const value = (event.target as HTMLInputElement).value;\n\n    this._updateFilter(value);\n  }\n\n  protected handleBlurEvent() {\n    this.input._markAsTouched();\n    this.input._setShouldDisplayError(true);\n\n    if (this._selectionModel.allowMultiple) return;\n\n    if (this._currentFilter === '') {\n      this._selectionModel.clearSelectedOptions();\n      return;\n    }\n\n    this._setFilterFromInputValue();\n  }\n\n  //#endregion\n\n  //#region Private Methods\n\n  private _initDispatchFilterChanges() {\n    this._currentFilter$\n      .pipe(\n        skipWhile(() => this.filterInternal),\n        debounceTime(300),\n        distinctUntilChanged(),\n        takeUntil(this._destroy$),\n        tap((v) => this.filterChange.emit(v)),\n      )\n      .subscribe();\n  }\n\n  private _initRepositionOnValueChanges() {\n    this.input.valueChange$\n      .pipe(\n        takeUntil(this._destroy$),\n        debounceTime(0),\n        tap(() => this._animatedOverlay._reposition()),\n      )\n      .subscribe();\n  }\n\n  private _updateFilter(value: string) {\n    if (this.input.nativeInputRef && this.input.nativeInputRef.element.nativeElement.value !== value) {\n      this.input.nativeInputRef.element.nativeElement.value = value;\n    }\n\n    if (this._currentFilter === value) return;\n\n    this._currentFilter$.next(value);\n\n    if (this.filterInternal) {\n      this._selectionModel.setFilter(value);\n    }\n  }\n\n  private _setFilterFromInputValue() {\n    if (this._selectionModel.allowMultiple) return;\n\n    const value = this.input.value;\n\n    if (!value || Array.isArray(value)) {\n      this._updateFilter('');\n      return;\n    }\n\n    const option = this._selectionModel.getOptionByValue(value);\n\n    if (!option) return;\n\n    const label = this._selectionModel.getLabel(option);\n\n    if (typeof label !== 'string') return;\n\n    this._updateFilter(label);\n  }\n\n  private _interpretKeyHandlerResult(result: KeyHandlerResult) {\n    if (result.overlayOperation === 'close') {\n      this.close();\n    } else if (result.overlayOperation === 'open') {\n      this.open();\n    }\n\n    if (result.setFilter !== undefined) {\n      this._updateFilter(result.setFilter);\n    }\n\n    if (result.optionAction) {\n      if (typeof result.optionAction === 'string') {\n        if (result.optionAction === 'clear') {\n          this._selectionModel.clearSelectedOptions();\n        } else if (result.optionAction === 'toggleAll') {\n          this._selectionModel.toggleAllSelectedOptions();\n        }\n      } else {\n        const { type, option } = result.optionAction;\n\n        if (type === 'add') {\n          this._selectionModel.addSelectedOption(option);\n        }\n\n        if (type === 'remove') {\n          this._selectionModel.removeSelectedOption(option);\n        }\n\n        if (type === 'toggle') {\n          this._selectionModel.toggleSelectedOption(option);\n        }\n      }\n    }\n\n    if (result.focusAction) {\n      if (typeof result.focusAction === 'string') {\n        if (result.focusAction === 'first') {\n          // TODO: Implement\n        }\n\n        if (result.focusAction === 'last') {\n          // TODO: Implement\n        }\n\n        if (result.focusAction === 'next') {\n          // TODO: Implement\n        }\n\n        if (result.focusAction === 'previous') {\n          // TODO: Implement\n        }\n      } else {\n        const { type } = result.focusAction;\n\n        if (type === 'offset') {\n          const { offset } = result.focusAction;\n\n          // TODO: Implement\n        }\n      }\n    }\n  }\n\n  //#endregion\n\n  //#region Dev mode\n\n  private _debugValidateComboboxConfig(isRetry = false) {\n    combineLatest([this._selectionModel.labelBinding$, this._selectionModel.valueBinding$, this._allowCustomValues$])\n      .pipe(\n        skip(isRetry ? 1 : 0), // Skip if retrying to avoid infinite loop\n        debounceTime(0),\n        takeUntil(this._destroy$),\n        map(([bindLabel, bindValue, allowCustomValues]) => {\n          const shouldBeObjects = bindLabel && bindValue && !allowCustomValues;\n\n          if (shouldBeObjects) {\n            return ComboboxOptionType.Object;\n          }\n\n          return ComboboxOptionType.Primitive;\n        }),\n        tap((expectedOptionType) => {\n          const options = this._selectionModel.options;\n\n          if (isEmptyArray(options)) {\n            return;\n          }\n\n          if (expectedOptionType === ComboboxOptionType.Object) {\n            if (!isObjectArray(options)) {\n              throw comboboxError(1, options);\n            }\n          } else if (expectedOptionType === ComboboxOptionType.Primitive) {\n            if (!isPrimitiveArray(options)) {\n              throw comboboxError(2, options);\n            }\n          }\n        }),\n        catchError((e) => {\n          this._debugValidateComboboxConfig(true);\n          return throwError(() => e);\n        }),\n      )\n      .subscribe();\n  }\n\n  //#endregion\n}\n","<!-- eslint-disable @angular-eslint/template/click-events-have-key-events -->\n<!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n<ng-content select=\"[etInputPrefix]\" />\n\n<div class=\"et-combobox-wrapper\">\n  <ul *ngIf=\"multiple$ | async\" class=\"et-combobox-selected-options\">\n    <li *ngFor=\"let item of selectedOptions$ | async\" class=\"et-combobox-selected-option\">\n      {{ getOptionLabel(item) | async }}\n      <span (click)=\"removeSelectedOption(item)\" class=\"et-combobox-selected-option-remove\" tabindex=\"-1\"> x </span>\n    </li>\n  </ul>\n\n  <input\n    (keydown)=\"processKeydownEvent($event)\"\n    (click)=\"open()\"\n    (blur)=\"handleBlurEvent()\"\n    (input)=\"processInputEvent($event)\"\n    type=\"text\"\n    etNativeInputRef\n  />\n\n  <et-chevron-icon class=\"et-combobox-chevron\" />\n</div>\n\n<ng-content select=\"[etInputSuffix]\" />\n\n<ng-template>\n  <ng-content />\n</ng-template>\n"]}