@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.
- package/CHANGELOG.md +6 -0
- package/esm2022/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.mjs +193 -263
- package/esm2022/lib/components/forms/components/select/components/select/directives/select/select.directive.mjs +7 -1
- package/esm2022/lib/components/forms/components/select/components/select-field/select-field.component.mjs +8 -5
- package/esm2022/lib/components/forms/components/select/directives/index.mjs +2 -0
- package/esm2022/lib/components/forms/components/select/directives/public-api.mjs +2 -0
- package/esm2022/lib/components/forms/components/select/directives/select-field/public-api.mjs +2 -0
- package/esm2022/lib/components/forms/components/select/directives/select-field/select-field.directive.mjs +21 -0
- package/esm2022/lib/components/forms/components/select/public-api.mjs +2 -1
- package/fesm2022/ethlete-cdk.mjs +216 -264
- package/fesm2022/ethlete-cdk.mjs.map +1 -1
- package/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.d.ts +10 -31
- package/lib/components/forms/components/select/components/select/directives/select/select.directive.d.ts +1 -0
- package/lib/components/forms/components/select/components/select-field/select-field.component.d.ts +5 -4
- package/lib/components/forms/components/select/directives/index.d.ts +1 -0
- package/lib/components/forms/components/select/directives/public-api.d.ts +1 -0
- package/lib/components/forms/components/select/directives/select-field/public-api.d.ts +1 -0
- package/lib/components/forms/components/select/directives/select-field/select-field.directive.d.ts +8 -0
- package/lib/components/forms/components/select/public-api.d.ts +1 -0
- 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,
|
|
6
|
-
import { BehaviorSubject, catchError, combineLatest, debounceTime, distinctUntilChanged, map,
|
|
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.
|
|
53
|
+
return this._selectionModel.getFilteredOptions();
|
|
99
54
|
}
|
|
100
55
|
set options(value) {
|
|
101
|
-
this.
|
|
102
|
-
}
|
|
103
|
-
get initialValue() {
|
|
104
|
-
return this._initialValue$.value;
|
|
56
|
+
this._selectionModel.setOptions(value);
|
|
105
57
|
}
|
|
106
58
|
set initialValue(value) {
|
|
107
|
-
this.
|
|
108
|
-
|
|
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.
|
|
133
|
-
}
|
|
134
|
-
get bindLabel() {
|
|
135
|
-
return this._bindLabel$.value;
|
|
93
|
+
this._selectionModel.setAllowMultiple(coerceBooleanProperty(value));
|
|
136
94
|
}
|
|
137
95
|
set bindLabel(value) {
|
|
138
|
-
this.
|
|
139
|
-
}
|
|
140
|
-
get bindValue() {
|
|
141
|
-
return this._bindValue$.value;
|
|
96
|
+
this._selectionModel.setLabelBinding(value);
|
|
142
97
|
}
|
|
143
98
|
set bindValue(value) {
|
|
144
|
-
this.
|
|
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.
|
|
171
|
-
this.
|
|
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
|
-
|
|
188
|
-
|
|
189
|
-
this.
|
|
190
|
-
this.
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
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.
|
|
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
|
|
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.
|
|
223
|
+
const isMultiple = this._selectionModel.allowMultiple;
|
|
327
224
|
const canAddCustomValue = this.allowCustomValues;
|
|
328
225
|
const value = event.target.value;
|
|
329
|
-
const
|
|
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
|
|
332
|
-
|
|
333
|
-
|
|
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
|
-
|
|
338
|
-
|
|
233
|
+
else {
|
|
234
|
+
if (isMultiple) {
|
|
235
|
+
// TODO: Toggle the focused option
|
|
236
|
+
}
|
|
237
|
+
else {
|
|
238
|
+
// TODO: Select the focused option
|
|
239
|
+
}
|
|
339
240
|
}
|
|
340
|
-
|
|
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
|
-
|
|
262
|
+
result.overlayOperation = 'close';
|
|
345
263
|
}
|
|
346
|
-
else {
|
|
347
|
-
|
|
348
|
-
|
|
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
|
-
|
|
356
|
-
return;
|
|
271
|
+
result.overlayOperation = 'close';
|
|
272
|
+
return this._interpretKeyHandlerResult(result);
|
|
357
273
|
}
|
|
358
274
|
if (!isOpen) {
|
|
359
|
-
|
|
275
|
+
result.overlayOperation = 'open';
|
|
360
276
|
}
|
|
361
277
|
if (keyCode === DOWN_ARROW) {
|
|
362
|
-
|
|
278
|
+
result.focusAction = 'next';
|
|
363
279
|
}
|
|
364
280
|
if (keyCode === UP_ARROW) {
|
|
365
|
-
|
|
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.
|
|
308
|
+
if (this._selectionModel.allowMultiple)
|
|
376
309
|
return;
|
|
377
310
|
if (this._currentFilter === '') {
|
|
378
|
-
this.
|
|
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
|
-
|
|
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.
|
|
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.
|
|
347
|
+
const option = this._selectionModel.getOptionByValue(value);
|
|
483
348
|
if (!option)
|
|
484
349
|
return;
|
|
485
|
-
const label = this.
|
|
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.
|
|
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$),
|
|
496
|
-
|
|
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(
|
|
501
|
-
throw comboboxError(1,
|
|
428
|
+
if (!isObjectArray(options)) {
|
|
429
|
+
throw comboboxError(1, options);
|
|
502
430
|
}
|
|
503
431
|
}
|
|
504
432
|
else if (expectedOptionType === ComboboxOptionType.Primitive) {
|
|
505
|
-
if (!isPrimitiveArray(
|
|
506
|
-
throw comboboxError(2,
|
|
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"]}
|