@seniorsistemas/angular-components 17.27.1-feature-sds-110-1bb20c20 → 17.27.1-feature-sds-110-57186492
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/autocomplete/index.d.ts +5 -0
- package/autocomplete/lib/autocomplete/autocomplete.component.d.ts +250 -0
- package/autocomplete/lib/autocomplete/components/autocomplete-chip/autocomplete-chip.component.d.ts +9 -0
- package/autocomplete/public-api.d.ts +1 -0
- package/button/lib/button/button.component.d.ts +1 -1
- package/dynamic-form/lib/dynamic-form/components/lookup/lookup.component.d.ts +1 -2
- package/dynamic-form/lib/dynamic-form/dynamic-form.module.d.ts +3 -2
- package/dynamic-form/lib/dynamic-form/form-field/fields/currency/currency-field.component.d.ts +3 -2
- package/esm2022/autocomplete/lib/autocomplete/autocomplete.component.mjs +554 -0
- package/esm2022/autocomplete/lib/autocomplete/components/autocomplete-chip/autocomplete-chip.component.mjs +17 -0
- package/esm2022/autocomplete/public-api.mjs +2 -0
- package/esm2022/autocomplete/seniorsistemas-angular-components-autocomplete.mjs +5 -0
- package/esm2022/button/lib/button/button.component.mjs +4 -10
- package/esm2022/chips/lib/chips/chips/chips.component.mjs +2 -2
- package/esm2022/control-errors/lib/control-errors/control-errors.component.mjs +2 -2
- package/esm2022/country-phone-picker/lib/country-phone-picker/country-phone-picker.component.mjs +2 -2
- package/esm2022/dynamic-form/lib/dynamic-form/components/lookup/lookup.component.mjs +16 -10
- package/esm2022/dynamic-form/lib/dynamic-form/dynamic-form.module.mjs +5 -1
- package/esm2022/dynamic-form/lib/dynamic-form/form-field/fields/autocomplete/autocomplete-field.component.mjs +7 -6
- package/esm2022/dynamic-form/lib/dynamic-form/form-field/fields/currency/currency-field.component.mjs +18 -8
- package/esm2022/dynamic-form/lib/dynamic-form/form-field/fields/lookup/lookup-field.component.mjs +1 -1
- package/esm2022/dynamic-form/lib/dynamic-form/form-field/fields/password/password-field.component.mjs +2 -2
- package/esm2022/dynamic-form/lib/dynamic-form/form-field/fields/select/select-field.component.mjs +1 -1
- package/esm2022/inline-edit/lib/inline-edit/components/fields/inline-edit-lookup/inline-edit-lookup.component.mjs +1 -1
- package/esm2022/lib/locale/fallback.mjs +2 -1
- package/esm2022/object-card/lib/object-card/object-card.component.mjs +3 -3
- package/esm2022/paginator/lib/paginator/paginator.component.mjs +2 -2
- package/esm2022/panel/lib/panel/panel.component.mjs +3 -3
- package/esm2022/select/lib/select/select.component.mjs +234 -336
- package/esm2022/text-area/lib/text-area/text-area.component.mjs +4 -4
- package/esm2022/text-area-ia/lib/text-area-ia/text-area-ia.component.mjs +4 -4
- package/esm2022/toast/lib/toast/toast.component.mjs +13 -13
- package/esm2022/token-list/lib/token-list/token-list.component.mjs +3 -3
- package/fesm2022/seniorsistemas-angular-components-autocomplete.mjs +575 -0
- package/fesm2022/seniorsistemas-angular-components-autocomplete.mjs.map +1 -0
- package/fesm2022/seniorsistemas-angular-components-button.mjs +3 -9
- package/fesm2022/seniorsistemas-angular-components-button.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-chips.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-chips.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-control-errors.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-control-errors.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-country-phone-picker.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-country-phone-picker.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs +38 -19
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-object-card.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-object-card.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-paginator.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-paginator.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-panel.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-panel.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-select.mjs +232 -334
- package/fesm2022/seniorsistemas-angular-components-select.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-text-area-ia.mjs +3 -3
- package/fesm2022/seniorsistemas-angular-components-text-area-ia.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-text-area.mjs +3 -3
- package/fesm2022/seniorsistemas-angular-components-text-area.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-toast.mjs +12 -12
- package/fesm2022/seniorsistemas-angular-components-toast.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-token-list.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-token-list.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components.mjs +1 -0
- package/fesm2022/seniorsistemas-angular-components.mjs.map +1 -1
- package/package.json +19 -13
- package/select/lib/select/select.component.d.ts +148 -157
- package/styles.css +1 -0
- package/text-area-ia/lib/text-area-ia/text-area-ia.component.d.ts +1 -1
- package/toast/lib/toast/toast.component.d.ts +4 -4
- package/src/lib/styles/tailwind.scss +0 -3
- package/tailwind.css +0 -1167
|
@@ -1,19 +1,21 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { OverlayModule } from '@angular/cdk/overlay';
|
|
2
2
|
import { ScrollingModule } from '@angular/cdk/scrolling';
|
|
3
|
-
import { CommonModule
|
|
4
|
-
import { Component, computed,
|
|
3
|
+
import { CommonModule } from '@angular/common';
|
|
4
|
+
import { Component, computed, forwardRef, inject, input, model, signal, viewChild, } from '@angular/core';
|
|
5
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
5
6
|
import { FormControl, FormGroup, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';
|
|
6
7
|
import { TranslateModule, TranslateService } from '@ngx-translate/core';
|
|
7
|
-
import { debounceTime, distinctUntilChanged, takeUntil } from 'rxjs/operators';
|
|
8
|
-
import { SelectOptionComponent } from './components/select-option/select-option.component';
|
|
9
|
-
import { Subject } from 'rxjs';
|
|
10
8
|
import { CheckboxComponent } from '@seniorsistemas/angular-components/checkbox';
|
|
9
|
+
import { debounceTime, distinctUntilChanged } from 'rxjs/operators';
|
|
10
|
+
import { SelectOptionComponent } from './components/select-option/select-option.component';
|
|
11
11
|
import * as i0 from "@angular/core";
|
|
12
12
|
import * as i1 from "@angular/common";
|
|
13
13
|
import * as i2 from "@angular/forms";
|
|
14
14
|
import * as i3 from "@angular/cdk/scrolling";
|
|
15
15
|
import * as i4 from "@ngx-translate/core";
|
|
16
|
+
import * as i5 from "@angular/cdk/overlay";
|
|
16
17
|
export class SelectComponent {
|
|
18
|
+
static nextId = 0;
|
|
17
19
|
placeholder = input('');
|
|
18
20
|
multiple = input(false);
|
|
19
21
|
options = input.required();
|
|
@@ -27,65 +29,49 @@ export class SelectComponent {
|
|
|
27
29
|
virtualScroll = input(false);
|
|
28
30
|
virtualScrollItemSize = input(37);
|
|
29
31
|
emptyMessage = input(null);
|
|
30
|
-
useRawValue = input(false);
|
|
31
|
-
dataKey = input();
|
|
32
32
|
disabled = model(false);
|
|
33
|
-
showOptions =
|
|
33
|
+
showOptions = model(false);
|
|
34
34
|
value = signal(null);
|
|
35
35
|
values = signal([]);
|
|
36
36
|
filterValue = signal('');
|
|
37
37
|
focusedIndex = signal(-1);
|
|
38
|
-
|
|
39
|
-
dropLeft = 0;
|
|
40
|
-
dropWidth = 0;
|
|
41
|
-
_portalHost = null;
|
|
38
|
+
componentId = `select${SelectComponent.nextId++}`;
|
|
42
39
|
/**
|
|
43
|
-
*
|
|
44
|
-
*
|
|
45
|
-
*
|
|
46
|
-
*
|
|
47
|
-
* - Usa `_getCompareValueExtractor()` para comparar os valores de forma consistente.
|
|
48
|
-
*
|
|
49
|
-
* Esse estado é usado para controlar o checkbox de "Selecionar todos" quando todas as opções visíveis estão marcadas.
|
|
40
|
+
* Computed que indica se todas as opções filtradas estão selecionadas.
|
|
41
|
+
* Retorna `true` se todas as opções disponíveis (excluindo agrupadores) estiverem presentes
|
|
42
|
+
* na lista de valores selecionados, e `false` caso contrário.
|
|
43
|
+
* @returns {boolean} `true` se todas as opções filtradas estão selecionadas, `false` caso contrário.
|
|
50
44
|
*/
|
|
51
45
|
allSelected = computed(() => {
|
|
52
|
-
const
|
|
53
|
-
const
|
|
54
|
-
|
|
55
|
-
|
|
46
|
+
const options = this.filteredOptions().filter((option) => !option.grouper);
|
|
47
|
+
const selectedIds = this.values().map((option) => option.id);
|
|
48
|
+
const ret = options.length > 0 && options.every((option) => selectedIds.includes(option.id));
|
|
49
|
+
return ret;
|
|
56
50
|
});
|
|
51
|
+
/**
|
|
52
|
+
* Retorna o item atualmente focado na lista de opções filtradas.
|
|
53
|
+
* @returns O dado do item focado ou `null` caso nenhum item esteja focado.
|
|
54
|
+
*/
|
|
57
55
|
focusedItem = computed(() => this.filteredOptions().at(this.focusedIndex())?.data ?? null);
|
|
58
56
|
/**
|
|
59
|
-
*
|
|
60
|
-
*
|
|
61
|
-
*
|
|
62
|
-
* -
|
|
63
|
-
*
|
|
64
|
-
*
|
|
65
|
-
* Essa estrutura é usada internamente para renderização, filtragem e controle de seleção.
|
|
57
|
+
* Computa e retorna a lista interna de opções do dropdown.
|
|
58
|
+
* - Se as opções estiverem agrupadas (`group()` retorna verdadeiro),
|
|
59
|
+
* utiliza `_grouperToFlat` para transformar as opções agrupadas em uma lista plana.
|
|
60
|
+
* - Caso contrário, retorna as opções com IDs adicionados.
|
|
61
|
+
* @returns Um array de `InternalDropdownOption<T>` representando as opções disponíveis para o componente select.
|
|
66
62
|
*/
|
|
67
63
|
internalOptions = computed(() => {
|
|
68
64
|
const opts = this.options();
|
|
69
|
-
const optionsWithId = this.
|
|
70
|
-
? opts.map((value, index) => ({
|
|
71
|
-
id: `id_primitive_${index}`,
|
|
72
|
-
grouper: false,
|
|
73
|
-
data: value,
|
|
74
|
-
}))
|
|
75
|
-
: this._addIdToOptions(opts);
|
|
65
|
+
const optionsWithId = this._addIdToOptions(opts);
|
|
76
66
|
return this.group() ? this._grouperToFlat(optionsWithId) : optionsWithId;
|
|
77
67
|
});
|
|
78
68
|
/**
|
|
79
|
-
* Retorna
|
|
80
|
-
*
|
|
81
|
-
*
|
|
82
|
-
*
|
|
83
|
-
*
|
|
84
|
-
*
|
|
85
|
-
* - Exibe o label do valor selecionado. *
|
|
86
|
-
* - Se nenhum valor estiver selecionado, retorna uma string vazia.
|
|
87
|
-
*
|
|
88
|
-
* Essa lógica é usada para mostrar o conteúdo atual do campo de seleção.
|
|
69
|
+
* Retorna uma string representando a seleção atual do componente.
|
|
70
|
+
* - Se o modo múltiplo estiver ativado e mais de um item estiver selecionado,
|
|
71
|
+
* retorna uma mensagem traduzida indicando o número total de registros selecionados.
|
|
72
|
+
* - Se apenas um item estiver selecionado, retorna o rótulo desse item.
|
|
73
|
+
* - Se o modo múltiplo não estiver ativado, retorna o rótulo do item selecionado ou uma string vazia caso nenhum item esteja selecionado.
|
|
74
|
+
* @returns {string} Texto representando a seleção atual.
|
|
89
75
|
*/
|
|
90
76
|
print = computed(() => {
|
|
91
77
|
if (this.multiple()) {
|
|
@@ -96,46 +82,45 @@ export class SelectComponent {
|
|
|
96
82
|
});
|
|
97
83
|
}
|
|
98
84
|
else if (selected.length === 1) {
|
|
99
|
-
return this._getLabelFromOption(selected[0]);
|
|
85
|
+
return this._getLabelFromOption(selected[0].data);
|
|
100
86
|
}
|
|
101
87
|
}
|
|
102
|
-
else
|
|
103
|
-
|
|
88
|
+
else {
|
|
89
|
+
const selected = this.value();
|
|
90
|
+
return selected ? this._getLabelFromOption(selected.data) : '';
|
|
104
91
|
}
|
|
105
92
|
return '';
|
|
106
93
|
});
|
|
94
|
+
/**
|
|
95
|
+
* Indica se o seletor está "limpo", ou seja, sem nenhum valor selecionado.
|
|
96
|
+
* - No modo múltiplo (`multiple`), retorna `true` se não houver nenhum valor selecionado (`values` está vazio).
|
|
97
|
+
* - No modo simples, retorna `true` se não houver valor selecionado (`value` é falsy).
|
|
98
|
+
* @returns `true` se não houver seleção, `false` caso contrário.
|
|
99
|
+
*/
|
|
107
100
|
isClean = computed(() => {
|
|
108
101
|
return this.multiple() ? !this.values().length : !this.value();
|
|
109
102
|
});
|
|
110
|
-
TIsPrimitive = computed(() => {
|
|
111
|
-
const opts = this.options();
|
|
112
|
-
if (!opts.length)
|
|
113
|
-
return false;
|
|
114
|
-
return ['string', 'number', 'boolean'].includes(typeof opts[0]);
|
|
115
|
-
});
|
|
116
103
|
/**
|
|
117
|
-
*
|
|
118
|
-
*
|
|
119
|
-
*
|
|
120
|
-
*
|
|
121
|
-
* - Ignora opções marcadas como `grouper`.
|
|
122
|
-
*
|
|
123
|
-
* Esse estado é usado, por exemplo, para exibir o checkbox de "Selecionar todos" em modo indeterminado.
|
|
104
|
+
* Computed que indica se o estado de seleção "Selecionar Todos" está indeterminado.
|
|
105
|
+
* Retorna `true` quando nem todos os itens filtrados estão selecionados e nem todos estão desmarcados,
|
|
106
|
+
* ou seja, quando há uma seleção parcial dos itens disponíveis.
|
|
107
|
+
* @returns {boolean} `true` se a seleção for parcial, `false` caso contrário.
|
|
124
108
|
*/
|
|
125
109
|
selectAllIsIndeterminate = computed(() => {
|
|
126
|
-
const selected = this.
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
const
|
|
110
|
+
const selected = this.internalOptions().filter((option) => {
|
|
111
|
+
return this.values().some((o) => o.id === option.id);
|
|
112
|
+
});
|
|
113
|
+
const options = this.filteredOptions().filter((option) => !option.grouper);
|
|
114
|
+
const all = options.length > 0 && options.every((option) => selected.includes(option));
|
|
115
|
+
const none = options.every((option) => !selected.includes(option));
|
|
130
116
|
return !all && !none;
|
|
131
117
|
});
|
|
132
118
|
/**
|
|
133
|
-
* Retorna
|
|
134
|
-
*
|
|
135
|
-
* -
|
|
136
|
-
* -
|
|
137
|
-
*
|
|
138
|
-
* - Usa o extrator de label para comparar os valores com o texto buscado.
|
|
119
|
+
* Retorna uma lista de opções filtradas com base no valor de busca (`filterValue`) e nos critérios definidos.
|
|
120
|
+
* - Se o filtro estiver desabilitado ou o valor de busca estiver vazio, retorna todas as opções.
|
|
121
|
+
* - Caso contrário, filtra as opções utilizando o(s) campo(s) especificado(s) em `filterBy`.
|
|
122
|
+
* - Ignora opções que possuem a propriedade `grouper` definida.
|
|
123
|
+
* @returns As opções filtradas conforme o valor de busca e critérios de filtro.
|
|
139
124
|
*/
|
|
140
125
|
filteredOptions = computed(() => {
|
|
141
126
|
const allOptions = this.internalOptions();
|
|
@@ -148,10 +133,10 @@ export class SelectComponent {
|
|
|
148
133
|
if (option.grouper) {
|
|
149
134
|
return false;
|
|
150
135
|
}
|
|
151
|
-
const labelExtractor = this._getLabelExtractor();
|
|
152
136
|
let label = '';
|
|
137
|
+
const optionLabel = this.optionLabel();
|
|
153
138
|
if (!filterBy) {
|
|
154
|
-
label =
|
|
139
|
+
label = optionLabel ? String(option.data[optionLabel]) : String(option.data);
|
|
155
140
|
}
|
|
156
141
|
else if (Array.isArray(filterBy)) {
|
|
157
142
|
label = filterBy.map((by) => String(option.data[by] ?? '')).join(' ');
|
|
@@ -165,89 +150,88 @@ export class SelectComponent {
|
|
|
165
150
|
filterForm = new FormGroup({
|
|
166
151
|
filter: new FormControl('', { nonNullable: true }),
|
|
167
152
|
});
|
|
168
|
-
_elementRef = inject((ElementRef));
|
|
169
|
-
_dropdownTemplate = viewChild('dropdownTemplate');
|
|
170
153
|
_containerDiv = viewChild('containerDiv');
|
|
171
154
|
_onChange = () => { };
|
|
172
155
|
_onTouched = () => { };
|
|
173
|
-
platformId = inject(PLATFORM_ID);
|
|
174
|
-
_viewContainerRef = inject(ViewContainerRef);
|
|
175
156
|
_translateService = inject(TranslateService);
|
|
176
|
-
|
|
177
|
-
/**
|
|
178
|
-
* Inicializa o componente, validando os inputs e configurando o filtro com debounce.
|
|
179
|
-
*/
|
|
180
|
-
ngOnInit() {
|
|
181
|
-
this._validateInputs();
|
|
157
|
+
constructor() {
|
|
182
158
|
this.filterForm
|
|
183
159
|
.get('filter')
|
|
184
|
-
.valueChanges.pipe(debounceTime(300), distinctUntilChanged(),
|
|
160
|
+
.valueChanges.pipe(debounceTime(300), distinctUntilChanged(), takeUntilDestroyed())
|
|
185
161
|
.subscribe((value) => {
|
|
186
162
|
this.filterValue.set(value);
|
|
187
163
|
});
|
|
188
164
|
}
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
*/
|
|
192
|
-
ngOnDestroy() {
|
|
193
|
-
this._destroyDropdownPortal();
|
|
194
|
-
if (this._portalHost) {
|
|
195
|
-
this._portalHost.dispose();
|
|
196
|
-
this._portalHost = null;
|
|
197
|
-
}
|
|
198
|
-
this.destroy$.next();
|
|
199
|
-
this.destroy$.complete();
|
|
165
|
+
ngOnInit() {
|
|
166
|
+
this._validateInputs();
|
|
200
167
|
}
|
|
201
168
|
/**
|
|
202
|
-
* Define o valor
|
|
203
|
-
* @param value
|
|
169
|
+
* Define o valor selecionado do componente select.
|
|
170
|
+
* @param value O valor a ser definido. Pode ser um objeto do tipo `T`, um array de `T` (quando múltipla seleção está habilitada), ou `null`.
|
|
171
|
+
* @throws Se o modo múltiplo estiver ativado e o valor não for um array, lança um erro.
|
|
172
|
+
* @throws Se o modo múltiplo estiver desativado e o valor for um array, lança um erro.
|
|
173
|
+
* Este método mapeia o(s) valor(es) fornecido(s) para as opções internas do componente,
|
|
174
|
+
* garantindo que apenas opções válidas sejam selecionadas conforme o modo de seleção (único ou múltiplo).
|
|
204
175
|
*/
|
|
205
176
|
writeValue(value) {
|
|
206
|
-
if (this.multiple()) {
|
|
177
|
+
if (value && this.multiple()) {
|
|
207
178
|
if (Array.isArray(value)) {
|
|
208
|
-
const
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
179
|
+
const mapped = value
|
|
180
|
+
.map((val) => {
|
|
181
|
+
return this.internalOptions().find((option) => {
|
|
182
|
+
return JSON.stringify(option.data) === JSON.stringify(val);
|
|
183
|
+
});
|
|
184
|
+
})
|
|
185
|
+
.filter((option) => !!option);
|
|
186
|
+
this.values.set(mapped);
|
|
214
187
|
}
|
|
215
188
|
else {
|
|
216
|
-
|
|
189
|
+
throw new Error('Value must be an array when "multiple" is true.');
|
|
217
190
|
}
|
|
218
191
|
}
|
|
219
192
|
else {
|
|
220
|
-
const
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
193
|
+
const optionValue = this.optionValue();
|
|
194
|
+
let mapped = null;
|
|
195
|
+
if (Array.isArray(value)) {
|
|
196
|
+
throw new Error('Value must not be an array when "multiple" is false.');
|
|
197
|
+
}
|
|
198
|
+
if (optionValue && value) {
|
|
199
|
+
mapped =
|
|
200
|
+
this.internalOptions().find((option) => option.data[optionValue] === value[optionValue]) ?? null;
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
mapped =
|
|
204
|
+
this.internalOptions().find((option) => JSON.stringify(option.data) === JSON.stringify(value)) ??
|
|
205
|
+
null;
|
|
206
|
+
}
|
|
207
|
+
this.value.set(mapped ?? null);
|
|
226
208
|
}
|
|
227
209
|
}
|
|
228
210
|
/**
|
|
229
|
-
* Registra
|
|
230
|
-
* @param onChange Função callback.
|
|
211
|
+
* Registra uma função de callback que será chamada sempre que o valor do componente mudar.
|
|
212
|
+
* @param onChange Função de callback que recebe o novo valor selecionado ou null.
|
|
231
213
|
*/
|
|
232
214
|
registerOnChange(onChange) {
|
|
233
215
|
this._onChange = onChange;
|
|
234
216
|
}
|
|
235
217
|
/**
|
|
236
|
-
* Registra
|
|
237
|
-
* @param onTouched Função callback.
|
|
218
|
+
* Registra uma função de callback que será chamada quando o componente for tocado (perder o foco).
|
|
219
|
+
* @param onTouched Função de callback a ser executada quando o componente for marcado como "tocado".
|
|
238
220
|
*/
|
|
239
221
|
registerOnTouched(onTouched) {
|
|
240
222
|
this._onTouched = onTouched;
|
|
241
223
|
}
|
|
242
224
|
/**
|
|
243
225
|
* Define o estado de desabilitado do componente.
|
|
244
|
-
* @param disabled Indica se o componente
|
|
226
|
+
* @param disabled Indica se o componente deve ser desabilitado (`true`) ou habilitado (`false`).
|
|
245
227
|
*/
|
|
246
228
|
setDisabledState(disabled) {
|
|
247
229
|
this.disabled.set(disabled);
|
|
248
230
|
}
|
|
249
231
|
/**
|
|
250
|
-
* Alterna a exibição
|
|
232
|
+
* Alterna a exibição das opções do componente select.
|
|
233
|
+
* Se o componente estiver desabilitado, não realiza nenhuma ação.
|
|
234
|
+
* Caso contrário, inverte o estado de exibição das opções.
|
|
251
235
|
*/
|
|
252
236
|
toggle() {
|
|
253
237
|
if (this.disabled()) {
|
|
@@ -255,29 +239,24 @@ export class SelectComponent {
|
|
|
255
239
|
}
|
|
256
240
|
const shouldOpen = !this.showOptions();
|
|
257
241
|
this.showOptions.set(shouldOpen);
|
|
258
|
-
if (shouldOpen) {
|
|
259
|
-
setTimeout(() => {
|
|
260
|
-
this._createDropdownPortal();
|
|
261
|
-
});
|
|
262
|
-
}
|
|
263
|
-
else {
|
|
264
|
-
this._destroyDropdownPortal();
|
|
265
|
-
}
|
|
266
242
|
}
|
|
267
243
|
/**
|
|
268
|
-
* Seleciona
|
|
269
|
-
* @param item
|
|
244
|
+
* Seleciona um item no componente de seleção.
|
|
245
|
+
* @param item O item a ser selecionado ou desmarcado.
|
|
246
|
+
* Se o componente estiver desabilitado, a função retorna imediatamente.
|
|
247
|
+
* No modo múltiplo, adiciona ou remove o item da lista de valores selecionados.
|
|
248
|
+
* No modo simples, define o item como valor selecionado e fecha o dropdown.
|
|
249
|
+
* Sempre notifica as mudanças e marca o componente como tocado.
|
|
270
250
|
*/
|
|
271
251
|
selectItem(item) {
|
|
272
252
|
if (this.disabled()) {
|
|
273
253
|
return;
|
|
274
254
|
}
|
|
275
|
-
const
|
|
255
|
+
const optionValue = this.optionValue();
|
|
276
256
|
if (this.multiple()) {
|
|
277
257
|
const store = [...this.values()];
|
|
278
|
-
const itemValue = getCompareValue(item);
|
|
279
258
|
const index = store.findIndex((selectedItem) => {
|
|
280
|
-
return
|
|
259
|
+
return selectedItem === item;
|
|
281
260
|
});
|
|
282
261
|
if (index !== -1) {
|
|
283
262
|
store.splice(index, 1);
|
|
@@ -286,20 +265,25 @@ export class SelectComponent {
|
|
|
286
265
|
store.push(item);
|
|
287
266
|
}
|
|
288
267
|
this.values.set(store);
|
|
289
|
-
const result =
|
|
268
|
+
const result = optionValue ? store.map((item) => item.data[optionValue]) : store;
|
|
290
269
|
this._onChange(result);
|
|
291
270
|
}
|
|
292
271
|
else {
|
|
293
272
|
this.value.set(item);
|
|
294
|
-
const result =
|
|
273
|
+
const result = optionValue ? item.data[optionValue] : item.data;
|
|
295
274
|
this._onChange(result);
|
|
296
275
|
this._closeDropdown();
|
|
297
276
|
}
|
|
298
277
|
this._onTouched();
|
|
299
278
|
}
|
|
300
279
|
/**
|
|
301
|
-
* Limpa
|
|
302
|
-
*
|
|
280
|
+
* Limpa o valor selecionado no componente select.
|
|
281
|
+
* Se o componente estiver desabilitado, não executa nenhuma ação.
|
|
282
|
+
* Para seleção múltipla, remove todos os valores selecionados e notifica a alteração.
|
|
283
|
+
* Para seleção única, define o valor como nulo e notifica a alteração.
|
|
284
|
+
* Fecha o dropdown e marca o componente como "tocado".
|
|
285
|
+
* Impede a propagação do evento do mouse.
|
|
286
|
+
* @param event Evento do mouse que acionou a limpeza da seleção.
|
|
303
287
|
*/
|
|
304
288
|
clear(event) {
|
|
305
289
|
if (this.disabled()) {
|
|
@@ -307,7 +291,7 @@ export class SelectComponent {
|
|
|
307
291
|
}
|
|
308
292
|
if (this.multiple()) {
|
|
309
293
|
this.values.set([]);
|
|
310
|
-
this._onChange(this.
|
|
294
|
+
this._onChange(this.values());
|
|
311
295
|
}
|
|
312
296
|
else {
|
|
313
297
|
this.value.set(null);
|
|
@@ -318,25 +302,17 @@ export class SelectComponent {
|
|
|
318
302
|
event.stopPropagation();
|
|
319
303
|
}
|
|
320
304
|
/**
|
|
321
|
-
* Limpa o filtro
|
|
305
|
+
* Limpa o filtro do formulário, resetando o campo 'filter' do formulário de filtro.
|
|
306
|
+
* Este método redefine o valor do campo 'filter' para seu estado inicial,
|
|
307
|
+
* removendo qualquer texto ou valor previamente inserido pelo usuário.
|
|
322
308
|
*/
|
|
323
309
|
clearFilter() {
|
|
324
310
|
this.filterForm.get('filter')?.reset();
|
|
325
311
|
}
|
|
326
312
|
/**
|
|
327
|
-
*
|
|
328
|
-
*
|
|
329
|
-
|
|
330
|
-
onDocumentClick(event) {
|
|
331
|
-
const clickTarget = event.target;
|
|
332
|
-
const clickedInsideComponent = this._elementRef.nativeElement.contains(clickTarget);
|
|
333
|
-
const dropdownInBody = document.querySelector('.dropdown-body-class')?.contains(clickTarget);
|
|
334
|
-
if (!clickedInsideComponent && !dropdownInBody) {
|
|
335
|
-
this._closeDropdown();
|
|
336
|
-
}
|
|
337
|
-
}
|
|
338
|
-
/**
|
|
339
|
-
* Fecha o dropdown em eventos de redimensionamento de janela.
|
|
313
|
+
* Manipula o evento de redimensionamento da janela.
|
|
314
|
+
* Fecha o dropdown de opções caso ele esteja visível ao redimensionar a janela.
|
|
315
|
+
* @param _ Evento de redimensionamento da janela (não utilizado).
|
|
340
316
|
*/
|
|
341
317
|
onWindowResize(_) {
|
|
342
318
|
if (this.showOptions()) {
|
|
@@ -344,15 +320,24 @@ export class SelectComponent {
|
|
|
344
320
|
}
|
|
345
321
|
}
|
|
346
322
|
/**
|
|
347
|
-
*
|
|
323
|
+
* Manipula o evento de clique no elemento container da seleção.
|
|
324
|
+
* Este método alterna o estado de exibição do componente select e,
|
|
325
|
+
* em seguida, define o foco no elemento container correspondente.
|
|
348
326
|
*/
|
|
349
327
|
onContainerDivClick() {
|
|
350
328
|
this.toggle();
|
|
351
329
|
this._containerDiv()?.nativeElement.focus();
|
|
352
330
|
}
|
|
353
331
|
/**
|
|
354
|
-
*
|
|
355
|
-
*
|
|
332
|
+
* Manipula eventos de teclado para navegação e seleção de opções no componente select.
|
|
333
|
+
* - 'ArrowDown': Move o foco para a próxima opção disponível.
|
|
334
|
+
* - 'ArrowUp': Move o foco para a opção anterior.
|
|
335
|
+
* - 'Tab': Seleciona a opção atualmente focada e fecha o dropdown.
|
|
336
|
+
* - 'Enter': Seleciona a opção focada se o dropdown estiver aberto, ou alterna a abertura do dropdown.
|
|
337
|
+
* - 'Escape': Fecha o dropdown.
|
|
338
|
+
* - ' ': (barra de espaço) Seleciona a opção focada se o dropdown estiver aberto, ou alterna a abertura do dropdown.
|
|
339
|
+
* Previne o comportamento padrão do navegador quando necessário para garantir a navegação adequada pelo teclado.
|
|
340
|
+
* @param event O evento de teclado disparado pelo usuário.
|
|
356
341
|
*/
|
|
357
342
|
onKeyDown(event) {
|
|
358
343
|
const options = this.filteredOptions();
|
|
@@ -362,7 +347,7 @@ export class SelectComponent {
|
|
|
362
347
|
const _selectOption = () => {
|
|
363
348
|
if (this.focusedIndex() >= 0 && this.focusedIndex() < options.length) {
|
|
364
349
|
event.preventDefault();
|
|
365
|
-
this.selectItem(options[this.focusedIndex()]
|
|
350
|
+
this.selectItem(options[this.focusedIndex()]);
|
|
366
351
|
}
|
|
367
352
|
};
|
|
368
353
|
switch (event.key) {
|
|
@@ -390,166 +375,113 @@ export class SelectComponent {
|
|
|
390
375
|
this._closeDropdown();
|
|
391
376
|
break;
|
|
392
377
|
case ' ':
|
|
393
|
-
|
|
378
|
+
event.preventDefault();
|
|
379
|
+
if (this.showOptions()) {
|
|
380
|
+
_selectOption();
|
|
381
|
+
}
|
|
382
|
+
else {
|
|
383
|
+
this.toggle();
|
|
384
|
+
}
|
|
394
385
|
break;
|
|
395
386
|
}
|
|
396
387
|
}
|
|
397
388
|
/**
|
|
398
|
-
* Retorna o rótulo
|
|
399
|
-
* @param option
|
|
400
|
-
* @returns
|
|
389
|
+
* Retorna o rótulo (label) associado a uma opção fornecida.
|
|
390
|
+
* @param option - A opção do tipo T para a qual o rótulo deve ser obtido.
|
|
391
|
+
* @returns O rótulo da opção como uma string.
|
|
401
392
|
*/
|
|
402
393
|
getOptionLabel(option) {
|
|
403
394
|
return this._getLabelFromOption(option);
|
|
404
395
|
}
|
|
405
396
|
/**
|
|
406
|
-
* Função
|
|
407
|
-
*
|
|
408
|
-
* @param item
|
|
409
|
-
* @
|
|
397
|
+
* Função de trackBy utilizada em diretivas *ngFor para otimizar a renderização de listas.
|
|
398
|
+
* Retorna o identificador único (`id`) de cada item do tipo `InternalDropdownOption<T>`.
|
|
399
|
+
* @param _ - Índice do item na lista (não utilizado).
|
|
400
|
+
* @param item - O item atual da lista do tipo `InternalDropdownOption<T>`.
|
|
401
|
+
* @returns O identificador único do item como uma string.
|
|
410
402
|
*/
|
|
411
403
|
trackById(_, item) {
|
|
412
404
|
return item.id;
|
|
413
405
|
}
|
|
414
406
|
/**
|
|
415
|
-
* Verifica se
|
|
416
|
-
* @param option
|
|
417
|
-
* @returns
|
|
407
|
+
* Verifica se a opção fornecida está selecionada.
|
|
408
|
+
* @param option - A opção interna do dropdown a ser verificada.
|
|
409
|
+
* @returns `true` se a opção estiver selecionada, caso contrário `false`.
|
|
418
410
|
*/
|
|
419
411
|
isOptionSelected(option) {
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
return selectedValues.includes(getCompareValue(option));
|
|
424
|
-
}
|
|
425
|
-
else {
|
|
426
|
-
const currentValue = this.value();
|
|
427
|
-
return currentValue !== null && getCompareValue(currentValue) === getCompareValue(option);
|
|
428
|
-
}
|
|
412
|
+
return (this.values().find((o) => {
|
|
413
|
+
return o === option;
|
|
414
|
+
}) !== undefined);
|
|
429
415
|
}
|
|
430
416
|
/**
|
|
431
|
-
* Alterna a seleção de todos os itens
|
|
417
|
+
* Alterna a seleção de todos os itens disponíveis.
|
|
418
|
+
* Se o modo múltiplo estiver ativado, seleciona todos os itens filtrados que não são agrupadores
|
|
419
|
+
* caso nem todos estejam selecionados, ou limpa a seleção caso todos já estejam selecionados.
|
|
420
|
+
* Após a alteração, notifica a mudança de valor.
|
|
432
421
|
*/
|
|
433
422
|
toggleSelectAll() {
|
|
434
423
|
if (!this.multiple()) {
|
|
435
424
|
return;
|
|
436
425
|
}
|
|
437
|
-
const options = this.filteredOptions().filter((
|
|
426
|
+
const options = this.filteredOptions().filter((option) => !option.grouper);
|
|
438
427
|
if (this.allSelected()) {
|
|
439
428
|
this.values.set([]);
|
|
440
429
|
}
|
|
441
430
|
else {
|
|
442
|
-
this.values.set(options
|
|
431
|
+
this.values.set(options);
|
|
443
432
|
}
|
|
444
|
-
this._onChange(this.
|
|
433
|
+
this._onChange(this.values());
|
|
445
434
|
}
|
|
446
435
|
/**
|
|
447
|
-
*
|
|
448
|
-
*
|
|
449
|
-
*
|
|
450
|
-
* @
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
return this.internalOptions().find((opt) => opt.data === value)?.data ?? null;
|
|
454
|
-
}
|
|
455
|
-
/**
|
|
456
|
-
* Remove propriedades internas adicionadas pelo componente, retornando apenas os dados crus.
|
|
457
|
-
*
|
|
458
|
-
* Usado para garantir que apenas os objetos de dados reais sejam emitidos ao formulário,
|
|
459
|
-
* especialmente no modo múltiplo.
|
|
460
|
-
*
|
|
461
|
-
* @param value Valor armazenado internamente (com ou sem `InternalDropdownOption`).
|
|
462
|
-
* @returns Os dados "puros", sem propriedades adicionais.
|
|
463
|
-
*/
|
|
464
|
-
_removeInternalProperties(value) {
|
|
465
|
-
if (this.TIsPrimitive()) {
|
|
466
|
-
return value;
|
|
467
|
-
}
|
|
468
|
-
return Array.isArray(value)
|
|
469
|
-
? value.map((opt) => opt.data)
|
|
470
|
-
: value.data;
|
|
471
|
-
}
|
|
472
|
-
/**
|
|
473
|
-
* Retorna o texto (label) correspondente a uma determinada opção.
|
|
474
|
-
* Usa a propriedade definida em `optionLabel`, ou tenta usar uma propriedade padrão (`label`).
|
|
475
|
-
*
|
|
476
|
-
* @param option A opção a ser exibida.
|
|
477
|
-
* @returns O texto legível da opção para exibição no componente.
|
|
436
|
+
* Retorna o rótulo (label) de uma opção fornecida.
|
|
437
|
+
* Este método utiliza a propriedade definida por `optionLabel` para extrair o valor do rótulo da opção.
|
|
438
|
+
* Caso `optionLabel` não esteja definido, retorna a representação em string da própria opção.
|
|
439
|
+
* @private
|
|
440
|
+
* @param option A opção da qual o rótulo será extraído.
|
|
441
|
+
* @returns O rótulo da opção como uma string.
|
|
478
442
|
*/
|
|
479
443
|
_getLabelFromOption(option) {
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
}
|
|
483
|
-
const labelKey = this.optionLabel();
|
|
484
|
-
if (!labelKey) {
|
|
485
|
-
return String(option.label ?? '');
|
|
486
|
-
}
|
|
487
|
-
return String(option[labelKey]);
|
|
444
|
+
const optionLabel = this.optionLabel();
|
|
445
|
+
return optionLabel ? String(option[optionLabel]) : String(option);
|
|
488
446
|
}
|
|
489
447
|
/**
|
|
490
|
-
* Valida
|
|
448
|
+
* Valida as entradas do componente de seleção.
|
|
449
|
+
* Este método verifica se as opções fornecidas são válidas. Caso as opções sejam objetos,
|
|
450
|
+
* garante que a propriedade `optionLabel` esteja definida, lançando um erro caso contrário.
|
|
451
|
+
* Se não houver opções, a validação é ignorada.
|
|
452
|
+
* @private
|
|
453
|
+
* @throws {Error} Se as opções forem objetos e `optionLabel` não estiver definido.
|
|
491
454
|
*/
|
|
492
455
|
_validateInputs() {
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
if (!this.TIsPrimitive() && !this.optionValue() && !this.dataKey()) {
|
|
497
|
-
throw new Error('Either the optionValue or dataKey input must be provided for non-primitive options.');
|
|
498
|
-
}
|
|
499
|
-
if (!this.TIsPrimitive() && !this.optionLabel()) {
|
|
500
|
-
throw new Error(`The 'optionLabel' input must be provided for non-primitive options.`);
|
|
501
|
-
}
|
|
502
|
-
}
|
|
503
|
-
_findOptionByValue(value) {
|
|
504
|
-
if (this.useRawValue()) {
|
|
505
|
-
return this.options().find((option) => option === value) ?? null;
|
|
506
|
-
}
|
|
507
|
-
const getValue = this._getValueExtractor();
|
|
508
|
-
return this.options().find((option) => getValue(option) === value) ?? null;
|
|
509
|
-
}
|
|
510
|
-
/**
|
|
511
|
-
* Extrai o valor final de uma opção, com base em `optionValue`.
|
|
512
|
-
* @returns Função que retorna o valor da opção.
|
|
513
|
-
*/
|
|
514
|
-
_getValueExtractor() {
|
|
515
|
-
if (this.TIsPrimitive()) {
|
|
516
|
-
return (option) => option;
|
|
517
|
-
}
|
|
518
|
-
if (this.useRawValue()) {
|
|
519
|
-
return (option) => option;
|
|
520
|
-
}
|
|
521
|
-
const valueInput = this.optionValue();
|
|
522
|
-
if (valueInput) {
|
|
523
|
-
return (option) => option[valueInput];
|
|
524
|
-
}
|
|
525
|
-
throw new Error(`optionValue is required for non-primitive options.`);
|
|
526
|
-
}
|
|
527
|
-
/**
|
|
528
|
-
* Extrai o valor comparável de uma opção.
|
|
529
|
-
* @returns Função que retorna a chave de comparação.
|
|
530
|
-
*/
|
|
531
|
-
_getCompareValueExtractor() {
|
|
532
|
-
if (this.TIsPrimitive()) {
|
|
533
|
-
return (option) => option;
|
|
456
|
+
const options = this.options();
|
|
457
|
+
if (!options.length) {
|
|
458
|
+
return;
|
|
534
459
|
}
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
460
|
+
const hasObjectOptions = options.some((option) => typeof option === 'object' && option !== null);
|
|
461
|
+
if (hasObjectOptions) {
|
|
462
|
+
if (!this.optionLabel()) {
|
|
463
|
+
throw new Error('The "optionLabel" input is required when options are objects.');
|
|
464
|
+
}
|
|
538
465
|
}
|
|
539
|
-
return this._getValueExtractor();
|
|
540
466
|
}
|
|
541
467
|
/**
|
|
542
|
-
* Fecha o dropdown
|
|
468
|
+
* Fecha o dropdown de opções do componente select.
|
|
469
|
+
* Este método oculta as opções disponíveis e redefine o índice do item focado para -1,
|
|
470
|
+
* indicando que nenhum item está atualmente focado.
|
|
471
|
+
* @private
|
|
543
472
|
*/
|
|
544
473
|
_closeDropdown() {
|
|
545
474
|
this.showOptions.set(false);
|
|
546
|
-
this._destroyDropdownPortal();
|
|
547
475
|
this.focusedIndex.set(-1);
|
|
548
476
|
}
|
|
549
477
|
/**
|
|
550
|
-
* Adiciona
|
|
551
|
-
*
|
|
552
|
-
*
|
|
478
|
+
* Adiciona um identificador único a cada opção fornecida.
|
|
479
|
+
* Para cada item no array de opções, gera um objeto `InternalDropdownOption` contendo o dado original
|
|
480
|
+
* e um campo `id` único baseado no timestamp atual e no índice do item. Caso a opção seja um grupo
|
|
481
|
+
* (detectado pela presença de um array `items`), adiciona também a propriedade `grouper: true`.
|
|
482
|
+
* @private
|
|
483
|
+
* @param options Array de opções do tipo `T` a serem processadas.
|
|
484
|
+
* @returns Um array de objetos `InternalDropdownOption<T>` com identificadores únicos.
|
|
553
485
|
*/
|
|
554
486
|
_addIdToOptions(options) {
|
|
555
487
|
const timestamp = Date.now();
|
|
@@ -568,9 +500,14 @@ export class SelectComponent {
|
|
|
568
500
|
});
|
|
569
501
|
}
|
|
570
502
|
/**
|
|
571
|
-
* Converte uma
|
|
572
|
-
*
|
|
573
|
-
*
|
|
503
|
+
* Converte uma lista de opções possivelmente agrupadas em uma lista plana de opções.
|
|
504
|
+
* Para cada opção que possui um agrupador (`grouper`) e um array de itens em `data.items`,
|
|
505
|
+
* adiciona a opção do agrupador e, em seguida, adiciona cada item do grupo como uma nova opção
|
|
506
|
+
* individual, atribuindo um `id` único para cada item do grupo.
|
|
507
|
+
* Caso a opção não seja um agrupador, ela é adicionada diretamente ao resultado.
|
|
508
|
+
* @private
|
|
509
|
+
* @param options Lista de opções internas do dropdown, podendo conter agrupadores.
|
|
510
|
+
* @returns Lista plana de opções, incluindo agrupadores e seus itens expandidos.
|
|
574
511
|
*/
|
|
575
512
|
_grouperToFlat(options) {
|
|
576
513
|
const result = [];
|
|
@@ -590,8 +527,9 @@ export class SelectComponent {
|
|
|
590
527
|
return result;
|
|
591
528
|
}
|
|
592
529
|
/**
|
|
593
|
-
* Move o foco para a próxima opção.
|
|
594
|
-
* @
|
|
530
|
+
* Move o foco para a próxima opção disponível na lista, ignorando opções que sejam agrupadores.
|
|
531
|
+
* @private
|
|
532
|
+
* @param options Lista de opções internas do dropdown.
|
|
595
533
|
*/
|
|
596
534
|
_focusNextOption(options) {
|
|
597
535
|
let nextIndex = this.focusedIndex() + 1;
|
|
@@ -604,8 +542,9 @@ export class SelectComponent {
|
|
|
604
542
|
}
|
|
605
543
|
}
|
|
606
544
|
/**
|
|
607
|
-
* Move o foco para a opção anterior.
|
|
608
|
-
* @
|
|
545
|
+
* Move o foco para a opção anterior na lista, ignorando opções do tipo "grouper".
|
|
546
|
+
* @private
|
|
547
|
+
* @param options Lista de opções internas do dropdown.
|
|
609
548
|
*/
|
|
610
549
|
_focusPreviousOption(options) {
|
|
611
550
|
let prevIndex = this.focusedIndex() - 1;
|
|
@@ -618,7 +557,11 @@ export class SelectComponent {
|
|
|
618
557
|
}
|
|
619
558
|
}
|
|
620
559
|
/**
|
|
621
|
-
* Rola
|
|
560
|
+
* Rola a lista de opções para garantir que a opção atualmente focada esteja visível.
|
|
561
|
+
* Este método obtém o ID da opção atualmente focada a partir da lista de opções filtradas.
|
|
562
|
+
* Se um ID válido for encontrado, busca o elemento correspondente no DOM e utiliza
|
|
563
|
+
* `scrollIntoView` para rolar até a opção, alinhando-a ao bloco mais próximo.
|
|
564
|
+
* @private
|
|
622
565
|
*/
|
|
623
566
|
_scrollToFocusedOption() {
|
|
624
567
|
const id = this.filteredOptions()[this.focusedIndex()]?.id;
|
|
@@ -628,66 +571,14 @@ export class SelectComponent {
|
|
|
628
571
|
const el = document.getElementById(id);
|
|
629
572
|
el?.scrollIntoView({ block: 'nearest' });
|
|
630
573
|
}
|
|
631
|
-
/**
|
|
632
|
-
* Extrai o texto de uma opção, com base em `optionLabel`.
|
|
633
|
-
* @returns Função que retorna o label da opção.
|
|
634
|
-
*/
|
|
635
|
-
_getLabelExtractor() {
|
|
636
|
-
if (this.TIsPrimitive()) {
|
|
637
|
-
return (option) => String(option);
|
|
638
|
-
}
|
|
639
|
-
const labelInput = this.optionLabel();
|
|
640
|
-
if (labelInput) {
|
|
641
|
-
return (option) => String(option[labelInput] ?? '');
|
|
642
|
-
}
|
|
643
|
-
else {
|
|
644
|
-
throw new Error(`optionLabel is required for non-primitive options.`);
|
|
645
|
-
}
|
|
646
|
-
}
|
|
647
|
-
/**
|
|
648
|
-
* Cria e posiciona dinamicamente o dropdown como portal.
|
|
649
|
-
*/
|
|
650
|
-
_createDropdownPortal() {
|
|
651
|
-
if (!isPlatformBrowser(this.platformId)) {
|
|
652
|
-
return;
|
|
653
|
-
}
|
|
654
|
-
const container = this._containerDiv();
|
|
655
|
-
const template = this._dropdownTemplate();
|
|
656
|
-
if (!container || !template) {
|
|
657
|
-
return;
|
|
658
|
-
}
|
|
659
|
-
const dropElementRect = container.nativeElement.getBoundingClientRect();
|
|
660
|
-
this.dropTop = dropElementRect.bottom + window.scrollY;
|
|
661
|
-
this.dropLeft = dropElementRect.left + window.scrollX;
|
|
662
|
-
this.dropWidth = dropElementRect.width;
|
|
663
|
-
this._destroyDropdownPortal();
|
|
664
|
-
this._portalHost = new DomPortalOutlet(document.body);
|
|
665
|
-
const portal = new TemplatePortal(template, this._viewContainerRef);
|
|
666
|
-
this._portalHost.attach(portal);
|
|
667
|
-
const dropdownEl = document.querySelector('.dropdown-body-class');
|
|
668
|
-
if (dropdownEl) {
|
|
669
|
-
dropdownEl.style.position = 'absolute';
|
|
670
|
-
dropdownEl.style.top = `${this.dropTop + 4}px`;
|
|
671
|
-
dropdownEl.style.left = `${this.dropLeft}px`;
|
|
672
|
-
dropdownEl.style.width = `${this.dropWidth}px`;
|
|
673
|
-
}
|
|
674
|
-
}
|
|
675
|
-
/**
|
|
676
|
-
* Remove o portal do dropdown do DOM.
|
|
677
|
-
*/
|
|
678
|
-
_destroyDropdownPortal() {
|
|
679
|
-
if (this._portalHost?.hasAttached()) {
|
|
680
|
-
this._portalHost.detach();
|
|
681
|
-
}
|
|
682
|
-
}
|
|
683
574
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SelectComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
684
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: SelectComponent, isStandalone: true, selector: "s-select", inputs: { placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null }, multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: true, isRequired: false, transformFunction: null }, options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: true, transformFunction: null }, optionLabel: { classPropertyName: "optionLabel", publicName: "optionLabel", isSignal: true, isRequired: false, transformFunction: null }, optionValue: { classPropertyName: "optionValue", publicName: "optionValue", isSignal: true, isRequired: false, transformFunction: null }, showClear: { classPropertyName: "showClear", publicName: "showClear", isSignal: true, isRequired: false, transformFunction: null }, filter: { classPropertyName: "filter", publicName: "filter", isSignal: true, isRequired: false, transformFunction: null }, checkmark: { classPropertyName: "checkmark", publicName: "checkmark", isSignal: true, isRequired: false, transformFunction: null }, filterBy: { classPropertyName: "filterBy", publicName: "filterBy", isSignal: true, isRequired: false, transformFunction: null }, group: { classPropertyName: "group", publicName: "group", isSignal: true, isRequired: false, transformFunction: null }, virtualScroll: { classPropertyName: "virtualScroll", publicName: "virtualScroll", isSignal: true, isRequired: false, transformFunction: null }, virtualScrollItemSize: { classPropertyName: "virtualScrollItemSize", publicName: "virtualScrollItemSize", isSignal: true, isRequired: false, transformFunction: null }, emptyMessage: { classPropertyName: "emptyMessage", publicName: "emptyMessage", isSignal: true, isRequired: false, transformFunction: null },
|
|
575
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: SelectComponent, isStandalone: true, selector: "s-select", inputs: { placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null }, multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: true, isRequired: false, transformFunction: null }, options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: true, transformFunction: null }, optionLabel: { classPropertyName: "optionLabel", publicName: "optionLabel", isSignal: true, isRequired: false, transformFunction: null }, optionValue: { classPropertyName: "optionValue", publicName: "optionValue", isSignal: true, isRequired: false, transformFunction: null }, showClear: { classPropertyName: "showClear", publicName: "showClear", isSignal: true, isRequired: false, transformFunction: null }, filter: { classPropertyName: "filter", publicName: "filter", isSignal: true, isRequired: false, transformFunction: null }, checkmark: { classPropertyName: "checkmark", publicName: "checkmark", isSignal: true, isRequired: false, transformFunction: null }, filterBy: { classPropertyName: "filterBy", publicName: "filterBy", isSignal: true, isRequired: false, transformFunction: null }, group: { classPropertyName: "group", publicName: "group", isSignal: true, isRequired: false, transformFunction: null }, virtualScroll: { classPropertyName: "virtualScroll", publicName: "virtualScroll", isSignal: true, isRequired: false, transformFunction: null }, virtualScrollItemSize: { classPropertyName: "virtualScrollItemSize", publicName: "virtualScrollItemSize", isSignal: true, isRequired: false, transformFunction: null }, emptyMessage: { classPropertyName: "emptyMessage", publicName: "emptyMessage", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: true, isRequired: false, transformFunction: null }, showOptions: { classPropertyName: "showOptions", publicName: "showOptions", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { disabled: "disabledChange", showOptions: "showOptionsChange" }, host: { attributes: { "role": "combobox" }, listeners: { "window:resize": "onWindowResize($event)" }, properties: { "attr.aria-expanded": "showOptions()", "attr.aria-haspopup": "\"listbox\"", "attr.aria-owns": "showOptions() ? componentId + \"-listbox\" : null", "attr.aria-activedescendant": "focusedIndex() >= 0 ? filteredOptions()[focusedIndex()]?.id : null", "attr.aria-label": "placeholder() || \"Select option\"", "attr.aria-disabled": "disabled()" } }, providers: [
|
|
685
576
|
{
|
|
686
577
|
provide: NG_VALUE_ACCESSOR,
|
|
687
578
|
useExisting: forwardRef(() => SelectComponent),
|
|
688
579
|
multi: true,
|
|
689
580
|
},
|
|
690
|
-
], viewQueries: [{ propertyName: "
|
|
581
|
+
], viewQueries: [{ propertyName: "_containerDiv", first: true, predicate: ["containerDiv"], descendants: true, isSignal: true }], ngImport: i0, template: "<div\n #containerDiv\n cdkOverlayOrigin\n #trigger=\"cdkOverlayOrigin\"\n class=\"select-container group flex h-[35px] w-full overflow-hidden rounded-[3px] border outline-1 outline-primary focus:outline\"\n [ngClass]=\"{\n 'pointer-events-none border-grayscale-20 bg-grayscale-5': disabled(),\n 'pointer-events-auto border-grayscale-30 bg-grayscale-0': !disabled(),\n }\"\n [attr.aria-disabled]=\"disabled()\"\n [attr.aria-expanded]=\"showOptions()\"\n [attr.aria-haspopup]=\"'listbox'\"\n [attr.aria-owns]=\"'dropdown-container-' + componentId\"\n [attr.aria-labelledby]=\"componentId + '-label'\"\n role=\"combobox\"\n (click)=\"onContainerDivClick()\"\n (keydown)=\"onKeyDown($event)\"\n tabindex=\"0\"\n>\n <span\n class=\"flex flex-grow select-none px-3 py-[7px]\"\n [ngClass]=\"{\n 'text-grayscale-90': !disabled() && !isClean(),\n 'text-grayscale-60': !disabled() && isClean(),\n 'text-grayscale-30': disabled(),\n }\"\n >\n {{ print() || placeholder() }}\n </span>\n\n @if (!disabled() && showClear() && !isClean()) {\n <button\n class=\"mx-3\"\n (click)=\"clear($event)\"\n >\n <i class=\"fas fa-times flex items-center\"></i>\n </button>\n }\n\n <div\n class=\"dropdown-container flex items-center border-l px-3 group-hover:bg-grayscale-10 group-focus:border-primary\"\n [ngClass]=\"{ 'border-grayscale-20': disabled(), 'border-grayscale-30': !disabled() }\"\n >\n <i\n class=\"fas\"\n [ngClass]=\"{\n 'text-grayscale-30': disabled(),\n 'text-grayscale-90': !disabled(),\n 'fa-caret-down': !showOptions(),\n 'fa-caret-up': showOptions(),\n }\"\n ></i>\n </div>\n</div>\n\n<ng-template\n #dropdownTemplate\n cdkConnectedOverlay\n [cdkConnectedOverlayOrigin]=\"trigger\"\n [cdkConnectedOverlayOpen]=\"showOptions()\"\n [cdkConnectedOverlayHasBackdrop]=\"true\"\n (backdropClick)=\"showOptions.set(false)\"\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n (detach)=\"showOptions.set(false)\"\n>\n <div\n [id]=\"'dropdown-container-' + componentId\"\n class=\"dropdown-body-class z-[1000] rounded-[3px] bg-grayscale-0 py-1 shadow-md\"\n (click)=\"$event.stopPropagation()\"\n [style.width.px]=\"containerDiv.offsetWidth\"\n >\n @if (filter()) {\n <div class=\"flex w-full items-center gap-3 px-3 py-1.5\">\n @if (multiple()) {\n <s-checkbox\n #selectAllCheckbox\n [checked]=\"allSelected()\"\n [indeterminate]=\"selectAllIsIndeterminate()\"\n (checkedChange)=\"toggleSelectAll()\"\n ></s-checkbox>\n }\n\n <div class=\"relative flex h-[35px] grow\">\n <form\n class=\"flex grow\"\n [formGroup]=\"filterForm\"\n >\n <input\n class=\"w-full grow rounded-[3px] border border-grayscale-30 pl-2.5 pr-7 outline-1 outline-primary\"\n type=\"text\"\n formControlName=\"filter\"\n (click)=\"$event.stopPropagation()\"\n />\n </form>\n <i class=\"fas fa-search absolute right-2.5 top-2.5 text-grayscale-90\"></i>\n </div>\n <button (click)=\"clearFilter()\">\n <i class=\"fas fa-times flex items-center\"></i>\n </button>\n </div>\n }\n\n <!-- Virtual scroll -->\n @if (virtualScroll() && filteredOptions().length > 10) {\n <cdk-virtual-scroll-viewport\n [itemSize]=\"virtualScrollItemSize()\"\n class=\"h-52 overflow-auto\"\n >\n <ng-container *cdkVirtualFor=\"let option of filteredOptions(); trackBy: trackById\">\n <s-select-option\n [id]=\"option.id\"\n [label]=\"getOptionLabel(option.data)\"\n [multiple]=\"multiple()\"\n [checkmark]=\"checkmark()\"\n [isSelected]=\"isOptionSelected(option)\"\n (selected)=\"selectItem(option)\"\n [isGrouper]=\"option.grouper\"\n [isFocused]=\"option.data === focusedItem()\"\n ></s-select-option>\n </ng-container>\n </cdk-virtual-scroll-viewport>\n }\n\n <!-- Normal list -->\n @if (!virtualScroll() || filteredOptions().length <= 10) {\n <ul class=\"max-h-52 overflow-auto\">\n @for (option of filteredOptions(); track option.id) {\n <s-select-option\n [id]=\"option.id\"\n [label]=\"getOptionLabel(option.data)\"\n [multiple]=\"multiple()\"\n [checkmark]=\"checkmark()\"\n [isSelected]=\"isOptionSelected(option)\"\n (selected)=\"selectItem(option)\"\n [isGrouper]=\"option.grouper ?? false\"\n [isFocused]=\"option.data === focusedItem()\"\n ></s-select-option>\n }\n </ul>\n }\n\n @if (filteredOptions().length === 0) {\n <span class=\"m-3 text-grayscale-60\">{{\n emptyMessage() ?? 'platform.angular_components.no_records_found' | translate\n }}</span>\n }\n </div>\n</ng-template>\n", styles: [":host.ng-dirty.ng-invalid{.dropdown-container,.select-container{border-color:#c13018;outline:none}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: ScrollingModule }, { kind: "directive", type: i3.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i3.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i3.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }, { kind: "component", type: SelectOptionComponent, selector: "s-select-option", inputs: ["id", "label", "multiple", "checkmark", "isGrouper", "isFocused", "isSelected"], outputs: ["isSelectedChange", "selected"] }, { kind: "component", type: CheckboxComponent, selector: "s-checkbox", inputs: ["disabled", "checked", "indeterminate", "label"], outputs: ["disabledChange", "checkedChange", "indeterminateChange"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: OverlayModule }, { kind: "directive", type: i5.CdkConnectedOverlay, selector: "[cdk-connected-overlay], [connected-overlay], [cdkConnectedOverlay]", inputs: ["cdkConnectedOverlayOrigin", "cdkConnectedOverlayPositions", "cdkConnectedOverlayPositionStrategy", "cdkConnectedOverlayOffsetX", "cdkConnectedOverlayOffsetY", "cdkConnectedOverlayWidth", "cdkConnectedOverlayHeight", "cdkConnectedOverlayMinWidth", "cdkConnectedOverlayMinHeight", "cdkConnectedOverlayBackdropClass", "cdkConnectedOverlayPanelClass", "cdkConnectedOverlayViewportMargin", "cdkConnectedOverlayScrollStrategy", "cdkConnectedOverlayOpen", "cdkConnectedOverlayDisableClose", "cdkConnectedOverlayTransformOriginOn", "cdkConnectedOverlayHasBackdrop", "cdkConnectedOverlayLockPosition", "cdkConnectedOverlayFlexibleDimensions", "cdkConnectedOverlayGrowAfterOpen", "cdkConnectedOverlayPush", "cdkConnectedOverlayDisposeOnNavigation"], outputs: ["backdropClick", "positionChange", "attach", "detach", "overlayKeydown", "overlayOutsideClick"], exportAs: ["cdkConnectedOverlay"] }, { kind: "directive", type: i5.CdkOverlayOrigin, selector: "[cdk-overlay-origin], [overlay-origin], [cdkOverlayOrigin]", exportAs: ["cdkOverlayOrigin"] }] });
|
|
691
582
|
}
|
|
692
583
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SelectComponent, decorators: [{
|
|
693
584
|
type: Component,
|
|
@@ -698,15 +589,22 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
698
589
|
SelectOptionComponent,
|
|
699
590
|
CheckboxComponent,
|
|
700
591
|
TranslateModule,
|
|
592
|
+
OverlayModule,
|
|
701
593
|
], host: {
|
|
702
|
-
'(document:click)': 'onDocumentClick($event)',
|
|
703
594
|
'(window:resize)': 'onWindowResize($event)',
|
|
595
|
+
'[attr.aria-expanded]': 'showOptions()',
|
|
596
|
+
'[attr.aria-haspopup]': '"listbox"',
|
|
597
|
+
'[attr.aria-owns]': 'showOptions() ? componentId + "-listbox" : null',
|
|
598
|
+
'[attr.aria-activedescendant]': 'focusedIndex() >= 0 ? filteredOptions()[focusedIndex()]?.id : null',
|
|
599
|
+
'[attr.aria-label]': 'placeholder() || "Select option"',
|
|
600
|
+
'[attr.aria-disabled]': 'disabled()',
|
|
601
|
+
role: 'combobox',
|
|
704
602
|
}, providers: [
|
|
705
603
|
{
|
|
706
604
|
provide: NG_VALUE_ACCESSOR,
|
|
707
605
|
useExisting: forwardRef(() => SelectComponent),
|
|
708
606
|
multi: true,
|
|
709
607
|
},
|
|
710
|
-
], template: "<
|
|
711
|
-
}] });
|
|
712
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../projects/angular-components/select/src/lib/select/select.component.ts","../../../../../projects/angular-components/select/src/lib/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { DomPortalOutlet, TemplatePortal } from '@angular/cdk/portal';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport {\n    Component,\n    computed,\n    ElementRef,\n    forwardRef,\n    inject,\n    input,\n    model,\n    OnDestroy,\n    OnInit,\n    PLATFORM_ID,\n    signal,\n    TemplateRef,\n    viewChild,\n    ViewContainerRef,\n} from '@angular/core';\nimport { ControlValueAccessor, FormControl, FormGroup, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\n\nimport { TranslateModule, TranslateService } from '@ngx-translate/core';\nimport { debounceTime, distinctUntilChanged, takeUntil } from 'rxjs/operators';\n\nimport { SelectOptionComponent } from './components/select-option/select-option.component';\nimport { Subject } from 'rxjs';\nimport { CheckboxComponent } from '@seniorsistemas/angular-components/checkbox';\n\ninterface InternalDropdownOption<T> {\n    data: T;\n    id: string;\n    grouper?: boolean;\n}\n@Component({\n    selector: 's-select',\n    templateUrl: './select.component.html',\n    standalone: true,\n    imports: [\n        CommonModule,\n        ReactiveFormsModule,\n        ScrollingModule,\n        SelectOptionComponent,\n        CheckboxComponent,\n        TranslateModule,\n    ],\n    host: {\n        '(document:click)': 'onDocumentClick($event)',\n        '(window:resize)': 'onWindowResize($event)',\n    },\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => SelectComponent),\n            multi: true,\n        },\n    ],\n})\nexport class SelectComponent<T> implements ControlValueAccessor, OnInit, OnDestroy {\n    public placeholder = input('');\n    public multiple = input(false);\n    public options = input.required<T[]>();\n    public optionLabel = input<keyof T>();\n    public optionValue = input<keyof T>();\n    public showClear = input(false);\n    public filter = input(false);\n    public checkmark = input(false);\n    public filterBy = input<keyof T | (keyof T)[]>();\n    public group = input(false);\n    public virtualScroll = input(false);\n    public virtualScrollItemSize = input(37);\n    public emptyMessage = input<string | null>(null);\n    public useRawValue = input(false);\n    public dataKey = input<keyof T>();\n\n    public disabled = model(false);\n    public showOptions = signal(false);\n\n    public value = signal<T | null>(null);\n    public values = signal<T[]>([]);\n    public filterValue = signal('');\n    public focusedIndex = signal(-1);\n\n    public dropTop = 0;\n    public dropLeft = 0;\n    public dropWidth = 0;\n\n    private _portalHost: DomPortalOutlet | null = null;\n\n    /**\n     * Verifica se todas as opções visíveis (filtradas e não agrupadoras) estão selecionadas.\n     *\n     * - Retorna `true` se todas as opções filtradas estiverem presentes em `values()`.\n     * - Ignora opções marcadas como `grouper`.\n     * - Usa `_getCompareValueExtractor()` para comparar os valores de forma consistente.\n     *\n     * Esse estado é usado para controlar o checkbox de \"Selecionar todos\" quando todas as opções visíveis estão marcadas.\n     */\n    public allSelected = computed(() => {\n        const selected = this.values();\n        const options = this.filteredOptions().filter((o) => !o.grouper);\n        return (\n            options.length > 0 &&\n            options.every((option) =>\n                selected.some(\n                    (s) => this._getCompareValueExtractor()(s) === this._getCompareValueExtractor()(option as T),\n                ),\n            )\n        );\n    });\n\n    public focusedItem = computed(() => this.filteredOptions().at(this.focusedIndex())?.data ?? null);\n\n    /**\n     * Converte as opções externas (`options()`) em uma estrutura interna padronizada (`InternalDropdownOption<T>`).\n     *\n     * - Se os dados forem primitivos (string, number, etc.), gera opções com IDs baseados no índice.\n     * - Se os dados forem objetos, delega a criação das opções para `_addIdToOptions()`, que adiciona IDs e marca agrupadores.\n     * - Se o modo agrupado estiver ativo (`group()`), transforma as opções em uma lista achatada com agrupadores via `_grouperToFlat()`.\n     *\n     * Essa estrutura é usada internamente para renderização, filtragem e controle de seleção.\n     */\n    public internalOptions = computed<InternalDropdownOption<T>[]>(() => {\n        const opts = this.options();\n\n        const optionsWithId: InternalDropdownOption<T>[] = this.TIsPrimitive()\n            ? opts.map((value, index) => ({\n                id: `id_primitive_${index}`,\n                grouper: false,\n                data: value as T,\n            }))\n            : this._addIdToOptions(opts);\n\n        return this.group() ? this._grouperToFlat(optionsWithId) : optionsWithId;\n    });\n\n    /**\n     * Retorna o texto que deve ser exibido no componente com base nos valores selecionados.\n     *\n     * - No modo múltiplo:\n     *   - Se houver mais de um item selecionado, exibe uma mensagem traduzida com o total de registros.\n     *   - Se houver apenas um item selecionado, exibe o label correspondente.\n     * - No modo simples:\n     *   - Exibe o label do valor selecionado.     *\n     * - Se nenhum valor estiver selecionado, retorna uma string vazia.\n     *\n     * Essa lógica é usada para mostrar o conteúdo atual do campo de seleção.\n     */\n\n    public print = computed(() => {\n        if (this.multiple()) {\n            const selected = this.values();\n            if (selected.length > 1) {\n                return this._translateService.instant(`platform.angular_components.total_records_selected`, {\n                    count: selected.length,\n                });\n            } else if (selected.length === 1) {\n                return this._getLabelFromOption(selected[0]);\n            }\n        } else if (this.value()) {\n            return this._getLabelFromOption(this.value() as T);\n        }\n\n        return '';\n    });\n\n    public isClean = computed(() => {\n        return this.multiple() ? !this.values().length : !this.value();\n    });\n\n    public TIsPrimitive = computed(() => {\n        const opts = this.options();\n        if (!opts.length) return false;\n        return ['string', 'number', 'boolean'].includes(typeof opts[0]);\n    });\n\n    /**\n     * Indica se o estado de seleção está indeterminado.\n     *\n     * - Retorna `true` quando **algumas**, mas não todas, opções visíveis estão selecionadas.\n     * - Retorna `false` quando **todas** ou **nenhuma** das opções visíveis estão selecionadas.\n     * - Ignora opções marcadas como `grouper`.\n     *\n     * Esse estado é usado, por exemplo, para exibir o checkbox de \"Selecionar todos\" em modo indeterminado.\n     */\n\n    public selectAllIsIndeterminate = computed(() => {\n        const selected = this.values();\n        const options = this.filteredOptions().filter((o) => !o.grouper);\n\n        const all = options.length > 0 && options.every((o) => selected.includes(o.data));\n        const none = options.every((o) => !selected.includes(o.data));\n\n        return !all && !none;\n    });\n\n    /**\n     * Retorna a lista de opções filtradas com base no valor digitado pelo usuário.\n     *\n     * - Se o filtro estiver desativado ou vazio, retorna todas as opções.\n     * - Caso contrário, aplica o filtro usando o(s) campo(s) definidos em `filterBy`.\n     * - Ignora opções marcadas como `grouper`.\n     * - Usa o extrator de label para comparar os valores com o texto buscado.\n     */\n\n    public filteredOptions = computed(() => {\n        const allOptions = this.internalOptions();\n        const search = this.filterValue().toLowerCase().trim();\n\n        if (!this.filter() || !search) {\n            return allOptions;\n        }\n\n        const filterBy = this.filterBy();\n        return allOptions.filter((option) => {\n            if (option.grouper) {\n                return false;\n            }\n\n            const labelExtractor = this._getLabelExtractor();\n            let label = '';\n\n            if (!filterBy) {\n                label = labelExtractor(option.data);\n            } else if (Array.isArray(filterBy)) {\n                label = filterBy.map((by) => String(option.data[by] ?? '')).join(' ');\n            } else {\n                label = String(option.data[filterBy] ?? '');\n            }\n\n            return label.toLowerCase().includes(search);\n        });\n    });\n\n    public readonly filterForm = new FormGroup<{ filter: FormControl<string> }>({\n        filter: new FormControl<string>('', { nonNullable: true }),\n    });\n\n    private readonly _elementRef = inject(ElementRef<HTMLElement>);\n    private _dropdownTemplate = viewChild<TemplateRef<any>>('dropdownTemplate');\n    private _containerDiv = viewChild<ElementRef<HTMLDivElement>>('containerDiv');\n    private _onChange: (value: T | T[] | null) => void = () => { };\n    private _onTouched: () => void = () => { };\n\n    private readonly platformId = inject(PLATFORM_ID);\n    private readonly _viewContainerRef = inject(ViewContainerRef);\n    private readonly _translateService = inject(TranslateService);\n\n    private destroy$ = new Subject<void>();\n\n    /**\n     * Inicializa o componente, validando os inputs e configurando o filtro com debounce.\n     */\n    public ngOnInit(): void {\n        this._validateInputs();\n\n        this.filterForm\n            .get('filter')!\n            .valueChanges.pipe(debounceTime(300), distinctUntilChanged(), takeUntil(this.destroy$))\n            .subscribe((value) => {\n                this.filterValue.set(value);\n            });\n    }\n\n    /**\n     * Executa limpeza de recursos ao destruir o componente.\n     */\n    public ngOnDestroy(): void {\n        this._destroyDropdownPortal();\n\n        if (this._portalHost) {\n            this._portalHost.dispose();\n            this._portalHost = null;\n        }\n\n        this.destroy$.next();\n        this.destroy$.complete();\n    }\n\n    /**\n     * Define o valor no componente vindo do formulário externo.\n     * @param value Valor atribuído pelo Angular forms.\n     */\n    public writeValue(value: any): void {\n        if (this.multiple()) {\n            if (Array.isArray(value)) {\n                const items = this.TIsPrimitive()\n                    ? value.map((v) => this._findOptionByPrimitiveValue(v)).filter(Boolean)\n                    : this.useRawValue()\n                        ? value\n                        : value.map((v) => this._findOptionByValue(v)).filter(Boolean);\n\n                this.values.set(items as T[]);\n            } else {\n                this.values.set([]);\n            }\n        } else {\n            const item = this.TIsPrimitive()\n                ? this._findOptionByPrimitiveValue(value)\n                : this.useRawValue()\n                    ? value\n                    : this._findOptionByValue(value);\n\n            this.value.set(item ?? null);\n        }\n    }\n\n    /**\n     * Registra a função que será chamada quando o valor mudar.\n     * @param onChange Função callback.\n     */\n    public registerOnChange(onChange: (value: T | T[] | null) => void): void {\n        this._onChange = onChange;\n    }\n\n    /**\n     * Registra a função que será chamada quando o componente for \"tocado\".\n     * @param onTouched Função callback.\n     */\n    public registerOnTouched(onTouched: () => void): void {\n        this._onTouched = onTouched;\n    }\n\n    /**\n     * Define o estado de desabilitado do componente.\n     * @param disabled Indica se o componente está desabilitado.\n     */\n    public setDisabledState(disabled: boolean): void {\n        this.disabled.set(disabled);\n    }\n\n    /**\n     * Alterna a exibição da lista de opções (dropdown).\n     */\n    public toggle(): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        const shouldOpen = !this.showOptions();\n        this.showOptions.set(shouldOpen);\n\n        if (shouldOpen) {\n            setTimeout(() => {\n                this._createDropdownPortal();\n            });\n        } else {\n            this._destroyDropdownPortal();\n        }\n    }\n\n    /**\n     * Seleciona ou desseleciona um item na lista.\n     * @param item Item a ser selecionado ou desmarcado.\n     */\n    public selectItem(item: T): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        const getCompareValue = this._getCompareValueExtractor();\n\n        if (this.multiple()) {\n            const store = [...this.values()];\n            const itemValue = getCompareValue(item);\n\n            const index = store.findIndex((selectedItem) => {\n                return getCompareValue(selectedItem) === itemValue;\n            });\n\n            if (index !== -1) {\n                store.splice(index, 1);\n            } else {\n                store.push(item);\n            }\n\n            this.values.set(store);\n\n            const result = this.useRawValue() ? store : store.map((i) => this._getValueExtractor()(i));\n\n            this._onChange(result);\n        } else {\n            this.value.set(item);\n\n            const result = this.useRawValue() ? item : this._getValueExtractor()(item);\n\n            this._onChange(result);\n            this._closeDropdown();\n        }\n\n        this._onTouched();\n    }\n\n    /**\n     * Limpa a seleção atual.\n     * @param event Evento do clique.\n     */\n    public clear(event: MouseEvent): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        if (this.multiple()) {\n            this.values.set([]);\n            this._onChange(this._removeInternalProperties(this.values()));\n        } else {\n            this.value.set(null);\n            this._onChange(this.value());\n        }\n\n        this._closeDropdown();\n\n        this._onTouched();\n        event.stopPropagation();\n    }\n\n    /**\n     * Limpa o filtro aplicado à lista de opções.\n     */\n    public clearFilter(): void {\n        this.filterForm.get('filter')?.reset();\n    }\n\n    /**\n     * Fecha o dropdown se o clique ocorrer fora do componente.\n     * @param event Evento do clique.\n     */\n    public onDocumentClick(event: MouseEvent): void {\n        const clickTarget = event.target as Node;\n\n        const clickedInsideComponent = this._elementRef.nativeElement.contains(clickTarget);\n        const dropdownInBody = document.querySelector('.dropdown-body-class')?.contains(clickTarget);\n\n        if (!clickedInsideComponent && !dropdownInBody) {\n            this._closeDropdown();\n        }\n    }\n\n    /**\n     * Fecha o dropdown em eventos de redimensionamento de janela.\n     */\n    public onWindowResize(_: Event): void {\n        if (this.showOptions()) {\n            this._closeDropdown();\n        }\n    }\n\n    /**\n     * Dispara a abertura/fechamento do dropdown ao clicar no container.\n     */\n    public onContainerDivClick(): void {\n        this.toggle();\n        this._containerDiv()?.nativeElement.focus();\n    }\n\n    /**\n     * Trata eventos de teclado para navegação e seleção.\n     * @param event Evento do teclado.\n     */\n    public onKeyDown(event: KeyboardEvent): void {\n        const options = this.filteredOptions();\n\n        if (!options.length) {\n            return;\n        }\n\n        const _selectOption = () => {\n            if (this.focusedIndex() >= 0 && this.focusedIndex() < options.length) {\n                event.preventDefault();\n                this.selectItem(options[this.focusedIndex()].data);\n            }\n        };\n\n        switch (event.key) {\n            case 'ArrowDown':\n                event.preventDefault();\n                this._focusNextOption(options);\n                break;\n            case 'ArrowUp':\n                event.preventDefault();\n                this._focusPreviousOption(options);\n                break;\n            case 'Tab':\n                _selectOption();\n                this._closeDropdown();\n                break;\n            case 'Enter':\n                if (this.showOptions()) {\n                    _selectOption();\n                } else {\n                    this.toggle();\n                }\n                break;\n            case 'Escape':\n                this._closeDropdown();\n                break;\n            case ' ':\n                _selectOption();\n                break;\n        }\n    }\n\n    /**\n     * Retorna o rótulo de exibição de uma opção.\n     * @param option Item da lista de opções.\n     * @returns Texto que será exibido no dropdown.\n     */\n    public getOptionLabel(option: T): string {\n        return this._getLabelFromOption(option);\n    }\n\n    /**\n     * Função usada para o trackBy no for do template.\n     * @param _ Índice do item.\n     * @param item Item da lista.\n     * @returns ID único da opção.\n     */\n    public trackById(_: number, item: InternalDropdownOption<T>): string {\n        return item.id;\n    }\n\n    /**\n     * Verifica se uma opção está selecionada.\n     * @param option Opção a ser verificada.\n     * @returns Verdadeiro se a opção estiver selecionada.\n     */\n    public isOptionSelected(option: T): boolean {\n        const getCompareValue = this._getCompareValueExtractor();\n\n        if (this.multiple()) {\n            const selectedValues = this.values().map(getCompareValue);\n            return selectedValues.includes(getCompareValue(option));\n        } else {\n            const currentValue = this.value();\n            return currentValue !== null && getCompareValue(currentValue) === getCompareValue(option);\n        }\n    }\n\n    /**\n     * Alterna a seleção de todos os itens filtrados.\n     */\n    public toggleSelectAll(): void {\n        if (!this.multiple()) {\n            return;\n        }\n\n        const options = this.filteredOptions().filter((o) => !o.grouper);\n\n        if (this.allSelected()) {\n            this.values.set([]);\n        } else {\n            this.values.set(options.map((o) => o.data));\n        }\n\n        this._onChange(this._removeInternalProperties(this.values()));\n    }\n\n    /**\n     * Procura e retorna uma opção com valor primitivo igual ao informado.\n     *\n     * @param value Valor primitivo (string, number ou boolean).\n     * @returns A opção correspondente, ou `null` se não encontrada.\n     */\n    private _findOptionByPrimitiveValue(value: any): T | null {\n        return this.internalOptions().find((opt) => opt.data === value)?.data ?? null;\n    }\n\n    /**\n     * Remove propriedades internas adicionadas pelo componente, retornando apenas os dados crus.\n     *\n     * Usado para garantir que apenas os objetos de dados reais sejam emitidos ao formulário,\n     * especialmente no modo múltiplo.\n     *\n     * @param value Valor armazenado internamente (com ou sem `InternalDropdownOption`).\n     * @returns Os dados \"puros\", sem propriedades adicionais.\n     */\n    private _removeInternalProperties(value: any) {\n        if (this.TIsPrimitive()) {\n            return value;\n        }\n\n        return Array.isArray(value)\n            ? value.map((opt: InternalDropdownOption<T>) => opt.data)\n            : (value as InternalDropdownOption<T>).data;\n    }\n\n    /**\n     * Retorna o texto (label) correspondente a uma determinada opção.\n     * Usa a propriedade definida em `optionLabel`, ou tenta usar uma propriedade padrão (`label`).\n     *\n     * @param option A opção a ser exibida.\n     * @returns O texto legível da opção para exibição no componente.\n     */\n    private _getLabelFromOption(option: T): string {\n        if (['string', 'number', 'boolean'].includes(typeof option)) {\n            return String(option);\n        }\n\n        const labelKey = this.optionLabel();\n\n        if (!labelKey) {\n            return String((option as any).label ?? '');\n        }\n\n        return String(option[labelKey]);\n    }\n\n    /**\n     * Valida os inputs obrigatórios do componente.\n     */\n    private _validateInputs(): void {\n        if (!this.useRawValue() && !this.TIsPrimitive() && !this.optionValue()) {\n            throw new Error(\n                `The 'optionValue' input must be provided for non-primitive options when useRawValue is false.`,\n            );\n        }\n\n        if (!this.TIsPrimitive() && !this.optionValue() && !this.dataKey()) {\n            throw new Error('Either the optionValue or dataKey input must be provided for non-primitive options.');\n        }\n\n        if (!this.TIsPrimitive() && !this.optionLabel()) {\n            throw new Error(`The 'optionLabel' input must be provided for non-primitive options.`);\n        }\n    }\n\n    private _findOptionByValue(value: any): T | null {\n        if (this.useRawValue()) {\n            return this.options().find((option) => option === value) ?? null;\n        }\n\n        const getValue = this._getValueExtractor();\n        return this.options().find((option) => getValue(option) === value) ?? null;\n    }\n\n    /**\n     * Extrai o valor final de uma opção, com base em `optionValue`.\n     * @returns Função que retorna o valor da opção.\n     */\n    private _getValueExtractor(): (option: T) => any {\n        if (this.TIsPrimitive()) {\n            return (option: any) => option;\n        }\n\n        if (this.useRawValue()) {\n            return (option: T) => option;\n        }\n\n        const valueInput = this.optionValue();\n\n        if (valueInput) {\n            return (option: T) => option[valueInput];\n        }\n\n        throw new Error(`optionValue is required for non-primitive options.`);\n    }\n\n    /**\n     * Extrai o valor comparável de uma opção.\n     * @returns Função que retorna a chave de comparação.\n     */\n    private _getCompareValueExtractor(): (option: T) => any {\n        if (this.TIsPrimitive()) {\n            return (option: any) => option;\n        }\n\n        if (this.dataKey()) {\n            const key = this.dataKey()!;\n            return (option: T) => option[key];\n        }\n\n        return this._getValueExtractor();\n    }\n\n    /**\n     * Fecha o dropdown e redefine o foco.\n     */\n    private _closeDropdown(): void {\n        this.showOptions.set(false);\n        this._destroyDropdownPortal();\n        this.focusedIndex.set(-1);\n    }\n\n    /**\n     * Adiciona identificadores únicos às opções.\n     * @param options Lista original de opções.\n     * @returns Lista com IDs adicionados.\n     */\n    private _addIdToOptions(options: T[]): InternalDropdownOption<T>[] {\n        const timestamp = Date.now();\n\n        return options.map((option, index) => {\n            if (this.group() && Array.isArray((option as any).items)) {\n                return {\n                    data: option,\n                    id: `id_${timestamp}_${index++}`,\n                    grouper: true,\n                };\n            }\n\n            return {\n                data: option,\n                id: `id_${timestamp}_${index++}`,\n            };\n        });\n    }\n\n    /**\n     * Converte uma estrutura agrupada em uma lista achatada.\n     * @param options Lista de opções com agrupadores.\n     * @returns Lista achatada de opções e grupos.\n     */\n    private _grouperToFlat(options: InternalDropdownOption<T>[]): InternalDropdownOption<T>[] {\n        const result: InternalDropdownOption<T>[] = [];\n\n        options.forEach((option) => {\n            if (option.grouper && Array.isArray((option.data as any).items)) {\n                result.push(option);\n                const groupItems = (option.data as any).items.map((item: T, index: number) => ({\n                    data: item,\n                    id: `${option.id}_item_${index}`,\n                }));\n                result.push(...groupItems);\n            } else {\n                result.push(option);\n            }\n        });\n\n        return result;\n    }\n\n    /**\n     * Move o foco para a próxima opção.\n     * @param options Lista de opções visíveis.\n     */\n    private _focusNextOption(options: InternalDropdownOption<T>[]) {\n        let nextIndex = this.focusedIndex() + 1;\n\n        while (nextIndex < options.length && options[nextIndex].grouper) {\n            nextIndex++;\n        }\n\n        if (nextIndex < options.length) {\n            this.focusedIndex.set(nextIndex);\n            this._scrollToFocusedOption();\n        }\n    }\n\n    /**\n     * Move o foco para a opção anterior.\n     * @param options Lista de opções visíveis.\n     */\n    private _focusPreviousOption(options: InternalDropdownOption<T>[]) {\n        let prevIndex = this.focusedIndex() - 1;\n\n        while (prevIndex >= 0 && options[prevIndex].grouper) {\n            prevIndex--;\n        }\n\n        if (prevIndex >= 0) {\n            this.focusedIndex.set(prevIndex);\n            this._scrollToFocusedOption();\n        }\n    }\n\n    /**\n     * Rola o dropdown até a opção atualmente focada.\n     */\n    private _scrollToFocusedOption() {\n        const id = this.filteredOptions()[this.focusedIndex()]?.id;\n\n        if (!id) {\n            return;\n        }\n\n        const el = document.getElementById(id);\n        el?.scrollIntoView({ block: 'nearest' });\n    }\n\n    /**\n     * Extrai o texto de uma opção, com base em `optionLabel`.\n     * @returns Função que retorna o label da opção.\n     */\n    private _getLabelExtractor(): (option: T) => string {\n        if (this.TIsPrimitive()) {\n            return (option: any) => String(option);\n        }\n\n        const labelInput = this.optionLabel();\n\n        if (labelInput) {\n            return (option: T) => String(option[labelInput] ?? '');\n        } else {\n            throw new Error(`optionLabel is required for non-primitive options.`);\n        }\n    }\n\n    /**\n     * Cria e posiciona dinamicamente o dropdown como portal.\n     */\n    private _createDropdownPortal(): void {\n        if (!isPlatformBrowser(this.platformId)) {\n            return;\n        }\n\n        const container = this._containerDiv();\n        const template = this._dropdownTemplate();\n\n        if (!container || !template) {\n            return;\n        }\n\n        const dropElementRect = container.nativeElement.getBoundingClientRect();\n\n        this.dropTop = dropElementRect.bottom + window.scrollY;\n        this.dropLeft = dropElementRect.left + window.scrollX;\n        this.dropWidth = dropElementRect.width;\n\n        this._destroyDropdownPortal();\n\n        this._portalHost = new DomPortalOutlet(document.body);\n\n        const portal = new TemplatePortal(template, this._viewContainerRef);\n\n        this._portalHost.attach(portal);\n\n        const dropdownEl = document.querySelector('.dropdown-body-class') as HTMLElement | null;\n        if (dropdownEl) {\n            dropdownEl.style.position = 'absolute';\n            dropdownEl.style.top = `${this.dropTop + 4}px`;\n            dropdownEl.style.left = `${this.dropLeft}px`;\n            dropdownEl.style.width = `${this.dropWidth}px`;\n        }\n    }\n\n    /**\n     * Remove o portal do dropdown do DOM.\n     */\n    private _destroyDropdownPortal(): void {\n        if (this._portalHost?.hasAttached()) {\n            this._portalHost.detach();\n        }\n    }\n}\n","<ng-template #portalAnchor></ng-template>\n\n<div\n    #containerDiv\n    class=\"group flex w-full overflow-hidden rounded-[3px] border outline-1 outline-primary focus:outline\"\n    [ngClass]=\"{\n        'pointer-events-none border-grayscale-20 bg-grayscale-5': disabled(),\n        'pointer-events-auto border-grayscale-30 bg-grayscale-0': !disabled(),\n    }\"\n    [attr.aria-disabled]=\"disabled()\"\n    (click)=\"onContainerDivClick()\"\n    (keydown)=\"onKeyDown($event)\"\n    tabindex=\"0\"\n>\n    <span\n        class=\"flex flex-grow select-none px-3 py-[7px]\"\n        [ngClass]=\"{\n            'text-grayscale-90': !disabled() && !isClean(),\n            'text-grayscale-60': !disabled() && isClean(),\n            'text-grayscale-30': disabled(),\n        }\"\n    >\n        {{ print() || placeholder() }}\n    </span>\n\n    @if (!disabled() && showClear() && !isClean()) {\n        <button\n            class=\"mx-3\"\n            (click)=\"clear($event)\"\n        >\n            <i class=\"fas fa-times flex items-center\"></i>\n        </button>\n    }\n\n    <div\n        class=\"flex items-center border-l px-3 group-hover:bg-grayscale-10 group-focus:border-primary\"\n        [ngClass]=\"{ 'border-grayscale-20': disabled(), 'border-grayscale-30': !disabled() }\"\n    >\n        <i\n            class=\"fas\"\n            [ngClass]=\"{\n                'text-grayscale-30': disabled(),\n                'text-grayscale-90': !disabled(),\n                'fa-caret-down': !showOptions(),\n                'fa-caret-up': showOptions(),\n            }\"\n        ></i>\n    </div>\n</div>\n\n<ng-template #dropdownTemplate>\n    <div\n        class=\"dropdown-body-class z-[1000] rounded-[3px] bg-grayscale-0 py-1 shadow-md\"\n        [style.position]=\"'absolute'\"\n        [style.top.px]=\"dropTop\"\n        [style.left.px]=\"dropLeft\"\n        (click)=\"$event.stopPropagation()\"\n    >\n        @if (filter()) {\n            <div class=\"flex w-full items-center gap-3 px-3 py-1.5\">\n                @if (multiple()) {\n                    <s-checkbox\n                        #selectAllCheckbox\n                        [checked]=\"allSelected()\"\n                        [indeterminate]=\"selectAllIsIndeterminate()\"\n                        (checkedChange)=\"toggleSelectAll()\"\n                    ></s-checkbox>\n                }\n\n                <div class=\"relative flex h-[35px] grow\">\n                    <form\n                        class=\"flex grow\"\n                        [formGroup]=\"filterForm\"\n                    >\n                        <input\n                            class=\"w-full grow rounded-[3px] border border-grayscale-30 pl-2.5 pr-7 outline-1 outline-primary\"\n                            type=\"text\"\n                            formControlName=\"filter\"\n                            (click)=\"$event.stopPropagation()\"\n                        />\n                    </form>\n                    <i class=\"fas fa-search absolute right-2.5 top-2.5 text-grayscale-90\"></i>\n                </div>\n                <button (click)=\"clearFilter()\">\n                    <i class=\"fas fa-times flex items-center\"></i>\n                </button>\n            </div>\n        }\n\n        <!-- Virtual scroll -->\n        @if (virtualScroll() && filteredOptions().length > 10) {\n            <cdk-virtual-scroll-viewport\n                [itemSize]=\"virtualScrollItemSize()\"\n                class=\"h-52 overflow-auto\"\n            >\n                <ng-container *cdkVirtualFor=\"let option of filteredOptions(); trackBy: trackById\">\n                    <s-select-option\n                        [id]=\"option.id\"\n                        [label]=\"getOptionLabel(option.data)\"\n                        [multiple]=\"multiple()\"\n                        [checkmark]=\"checkmark()\"\n                        [isSelected]=\"isOptionSelected(option.data)\"\n                        (selected)=\"selectItem(option.data)\"\n                        [isGrouper]=\"option.grouper\"\n                        [isFocused]=\"option.data === focusedItem()\"\n                    ></s-select-option>\n                </ng-container>\n            </cdk-virtual-scroll-viewport>\n        }\n\n        <!-- Normal list -->\n        @if (!virtualScroll() || filteredOptions().length <= 10) {\n            <ul class=\"max-h-52 overflow-auto\">\n                @for (option of filteredOptions(); track option.id) {\n                    <s-select-option\n                        [id]=\"option.id\"\n                        [label]=\"getOptionLabel(option.data)\"\n                        [multiple]=\"multiple()\"\n                        [checkmark]=\"checkmark()\"\n                        [isSelected]=\"isOptionSelected(option.data)\"\n                        (selected)=\"selectItem(option.data)\"\n                        [isGrouper]=\"option.grouper ?? false\"\n                        [isFocused]=\"option.data === focusedItem()\"\n                    ></s-select-option>\n                }\n            </ul>\n        }\n\n        @if (filteredOptions().length === 0) {\n            <span class=\"m-3 text-grayscale-60\">{{\n                emptyMessage() ?? 'platform.angular_components.no_records_found' | translate\n            }}</span>\n        }\n    </div>\n</ng-template>\n"]}
|
|
608
|
+
], template: "<div\n #containerDiv\n cdkOverlayOrigin\n #trigger=\"cdkOverlayOrigin\"\n class=\"select-container group flex h-[35px] w-full overflow-hidden rounded-[3px] border outline-1 outline-primary focus:outline\"\n [ngClass]=\"{\n 'pointer-events-none border-grayscale-20 bg-grayscale-5': disabled(),\n 'pointer-events-auto border-grayscale-30 bg-grayscale-0': !disabled(),\n }\"\n [attr.aria-disabled]=\"disabled()\"\n [attr.aria-expanded]=\"showOptions()\"\n [attr.aria-haspopup]=\"'listbox'\"\n [attr.aria-owns]=\"'dropdown-container-' + componentId\"\n [attr.aria-labelledby]=\"componentId + '-label'\"\n role=\"combobox\"\n (click)=\"onContainerDivClick()\"\n (keydown)=\"onKeyDown($event)\"\n tabindex=\"0\"\n>\n <span\n class=\"flex flex-grow select-none px-3 py-[7px]\"\n [ngClass]=\"{\n 'text-grayscale-90': !disabled() && !isClean(),\n 'text-grayscale-60': !disabled() && isClean(),\n 'text-grayscale-30': disabled(),\n }\"\n >\n {{ print() || placeholder() }}\n </span>\n\n @if (!disabled() && showClear() && !isClean()) {\n <button\n class=\"mx-3\"\n (click)=\"clear($event)\"\n >\n <i class=\"fas fa-times flex items-center\"></i>\n </button>\n }\n\n <div\n class=\"dropdown-container flex items-center border-l px-3 group-hover:bg-grayscale-10 group-focus:border-primary\"\n [ngClass]=\"{ 'border-grayscale-20': disabled(), 'border-grayscale-30': !disabled() }\"\n >\n <i\n class=\"fas\"\n [ngClass]=\"{\n 'text-grayscale-30': disabled(),\n 'text-grayscale-90': !disabled(),\n 'fa-caret-down': !showOptions(),\n 'fa-caret-up': showOptions(),\n }\"\n ></i>\n </div>\n</div>\n\n<ng-template\n #dropdownTemplate\n cdkConnectedOverlay\n [cdkConnectedOverlayOrigin]=\"trigger\"\n [cdkConnectedOverlayOpen]=\"showOptions()\"\n [cdkConnectedOverlayHasBackdrop]=\"true\"\n (backdropClick)=\"showOptions.set(false)\"\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n (detach)=\"showOptions.set(false)\"\n>\n <div\n [id]=\"'dropdown-container-' + componentId\"\n class=\"dropdown-body-class z-[1000] rounded-[3px] bg-grayscale-0 py-1 shadow-md\"\n (click)=\"$event.stopPropagation()\"\n [style.width.px]=\"containerDiv.offsetWidth\"\n >\n @if (filter()) {\n <div class=\"flex w-full items-center gap-3 px-3 py-1.5\">\n @if (multiple()) {\n <s-checkbox\n #selectAllCheckbox\n [checked]=\"allSelected()\"\n [indeterminate]=\"selectAllIsIndeterminate()\"\n (checkedChange)=\"toggleSelectAll()\"\n ></s-checkbox>\n }\n\n <div class=\"relative flex h-[35px] grow\">\n <form\n class=\"flex grow\"\n [formGroup]=\"filterForm\"\n >\n <input\n class=\"w-full grow rounded-[3px] border border-grayscale-30 pl-2.5 pr-7 outline-1 outline-primary\"\n type=\"text\"\n formControlName=\"filter\"\n (click)=\"$event.stopPropagation()\"\n />\n </form>\n <i class=\"fas fa-search absolute right-2.5 top-2.5 text-grayscale-90\"></i>\n </div>\n <button (click)=\"clearFilter()\">\n <i class=\"fas fa-times flex items-center\"></i>\n </button>\n </div>\n }\n\n <!-- Virtual scroll -->\n @if (virtualScroll() && filteredOptions().length > 10) {\n <cdk-virtual-scroll-viewport\n [itemSize]=\"virtualScrollItemSize()\"\n class=\"h-52 overflow-auto\"\n >\n <ng-container *cdkVirtualFor=\"let option of filteredOptions(); trackBy: trackById\">\n <s-select-option\n [id]=\"option.id\"\n [label]=\"getOptionLabel(option.data)\"\n [multiple]=\"multiple()\"\n [checkmark]=\"checkmark()\"\n [isSelected]=\"isOptionSelected(option)\"\n (selected)=\"selectItem(option)\"\n [isGrouper]=\"option.grouper\"\n [isFocused]=\"option.data === focusedItem()\"\n ></s-select-option>\n </ng-container>\n </cdk-virtual-scroll-viewport>\n }\n\n <!-- Normal list -->\n @if (!virtualScroll() || filteredOptions().length <= 10) {\n <ul class=\"max-h-52 overflow-auto\">\n @for (option of filteredOptions(); track option.id) {\n <s-select-option\n [id]=\"option.id\"\n [label]=\"getOptionLabel(option.data)\"\n [multiple]=\"multiple()\"\n [checkmark]=\"checkmark()\"\n [isSelected]=\"isOptionSelected(option)\"\n (selected)=\"selectItem(option)\"\n [isGrouper]=\"option.grouper ?? false\"\n [isFocused]=\"option.data === focusedItem()\"\n ></s-select-option>\n }\n </ul>\n }\n\n @if (filteredOptions().length === 0) {\n <span class=\"m-3 text-grayscale-60\">{{\n emptyMessage() ?? 'platform.angular_components.no_records_found' | translate\n }}</span>\n }\n </div>\n</ng-template>\n", styles: [":host.ng-dirty.ng-invalid{.dropdown-container,.select-container{border-color:#c13018;outline:none}}\n"] }]
|
|
609
|
+
}], ctorParameters: () => [] });
|
|
610
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../projects/angular-components/select/src/lib/select/select.component.ts","../../../../../projects/angular-components/select/src/lib/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { OverlayModule } from '@angular/cdk/overlay';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\nimport { CommonModule } from '@angular/common';\nimport {\n    Component,\n    computed,\n    ElementRef,\n    forwardRef,\n    inject,\n    input,\n    model,\n    OnInit,\n    signal,\n    viewChild,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ControlValueAccessor, FormControl, FormGroup, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';\n\nimport { TranslateModule, TranslateService } from '@ngx-translate/core';\nimport { CheckboxComponent } from '@seniorsistemas/angular-components/checkbox';\nimport { debounceTime, distinctUntilChanged } from 'rxjs/operators';\n\nimport { SelectOptionComponent } from './components/select-option/select-option.component';\n\ninterface InternalDropdownOption<T> {\n    data: T;\n    id: string;\n    grouper?: boolean;\n}\n@Component({\n    selector: 's-select',\n    templateUrl: './select.component.html',\n    standalone: true,\n    imports: [\n        CommonModule,\n        ReactiveFormsModule,\n        ScrollingModule,\n        SelectOptionComponent,\n        CheckboxComponent,\n        TranslateModule,\n        OverlayModule,\n    ],\n    host: {\n        '(window:resize)': 'onWindowResize($event)',\n        '[attr.aria-expanded]': 'showOptions()',\n        '[attr.aria-haspopup]': '\"listbox\"',\n        '[attr.aria-owns]': 'showOptions() ? componentId + \"-listbox\" : null',\n        '[attr.aria-activedescendant]': 'focusedIndex() >= 0 ? filteredOptions()[focusedIndex()]?.id : null',\n        '[attr.aria-label]': 'placeholder() || \"Select option\"',\n        '[attr.aria-disabled]': 'disabled()',\n        role: 'combobox',\n    },\n    styles: [\n        `\n            :host.ng-dirty.ng-invalid {\n                .dropdown-container,\n                .select-container {\n                    border-color: #c13018;\n                    outline: none;\n                }\n            }\n        `,\n    ],\n\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => SelectComponent),\n            multi: true,\n        },\n    ],\n})\nexport class SelectComponent<T> implements ControlValueAccessor, OnInit {\n    private static nextId = 0;\n\n    public placeholder = input('');\n    public multiple = input(false);\n    public options = input.required<T[]>();\n    public optionLabel = input<keyof T>();\n    public optionValue = input<keyof T>();\n    public showClear = input(false);\n    public filter = input(false);\n    public checkmark = input(false);\n    public filterBy = input<keyof T | (keyof T)[]>();\n    public group = input(false);\n    public virtualScroll = input(false);\n    public virtualScrollItemSize = input(37);\n    public emptyMessage = input<string | null>(null);\n\n    public disabled = model(false);\n    public showOptions = model(false);\n\n    public value = signal<InternalDropdownOption<T> | null>(null);\n    public values = signal<InternalDropdownOption<T>[]>([]);\n    public filterValue = signal('');\n    public focusedIndex = signal(-1);\n\n    public componentId = `select${SelectComponent.nextId++}`;\n\n    /**\n     * Computed que indica se todas as opções filtradas estão selecionadas.\n     * Retorna `true` se todas as opções disponíveis (excluindo agrupadores) estiverem presentes\n     * na lista de valores selecionados, e `false` caso contrário.\n     * @returns {boolean} `true` se todas as opções filtradas estão selecionadas, `false` caso contrário.\n     */\n    public allSelected = computed(() => {\n        const options = this.filteredOptions().filter((option) => !option.grouper);\n        const selectedIds = this.values().map((option) => option.id);\n\n        const ret = options.length > 0 && options.every((option) => selectedIds.includes(option.id));\n        return ret;\n    });\n\n    /**\n     * Retorna o item atualmente focado na lista de opções filtradas.\n     * @returns O dado do item focado ou `null` caso nenhum item esteja focado.\n     */\n    public focusedItem = computed(() => this.filteredOptions().at(this.focusedIndex())?.data ?? null);\n\n    /**\n     * Computa e retorna a lista interna de opções do dropdown.\n     * - Se as opções estiverem agrupadas (`group()` retorna verdadeiro),\n     *   utiliza `_grouperToFlat` para transformar as opções agrupadas em uma lista plana.\n     * - Caso contrário, retorna as opções com IDs adicionados.\n     * @returns Um array de `InternalDropdownOption<T>` representando as opções disponíveis para o componente select.\n     */\n    public internalOptions = computed<InternalDropdownOption<T>[]>(() => {\n        const opts = this.options();\n        const optionsWithId = this._addIdToOptions(opts);\n        return this.group() ? this._grouperToFlat(optionsWithId) : optionsWithId;\n    });\n\n    /**\n     * Retorna uma string representando a seleção atual do componente.\n     * - Se o modo múltiplo estiver ativado e mais de um item estiver selecionado,\n     *   retorna uma mensagem traduzida indicando o número total de registros selecionados.\n     * - Se apenas um item estiver selecionado, retorna o rótulo desse item.\n     * - Se o modo múltiplo não estiver ativado, retorna o rótulo do item selecionado ou uma string vazia caso nenhum item esteja selecionado.\n     * @returns {string} Texto representando a seleção atual.\n     */\n    public print = computed(() => {\n        if (this.multiple()) {\n            const selected = this.values();\n            if (selected.length > 1) {\n                return this._translateService.instant(`platform.angular_components.total_records_selected`, {\n                    count: selected.length,\n                });\n            } else if (selected.length === 1) {\n                return this._getLabelFromOption(selected[0].data);\n            }\n        } else {\n            const selected = this.value();\n            return selected ? this._getLabelFromOption(selected.data) : '';\n        }\n\n        return '';\n    });\n\n    /**\n     * Indica se o seletor está \"limpo\", ou seja, sem nenhum valor selecionado.\n     * - No modo múltiplo (`multiple`), retorna `true` se não houver nenhum valor selecionado (`values` está vazio).\n     * - No modo simples, retorna `true` se não houver valor selecionado (`value` é falsy).\n     * @returns `true` se não houver seleção, `false` caso contrário.\n     */\n    public isClean = computed(() => {\n        return this.multiple() ? !this.values().length : !this.value();\n    });\n\n    /**\n     * Computed que indica se o estado de seleção \"Selecionar Todos\" está indeterminado.\n     * Retorna `true` quando nem todos os itens filtrados estão selecionados e nem todos estão desmarcados,\n     * ou seja, quando há uma seleção parcial dos itens disponíveis.\n     * @returns {boolean} `true` se a seleção for parcial, `false` caso contrário.\n     */\n    public selectAllIsIndeterminate = computed(() => {\n        const selected = this.internalOptions().filter((option) => {\n            return this.values().some((o) => o.id === option.id);\n        });\n        const options = this.filteredOptions().filter((option) => !option.grouper);\n        const all = options.length > 0 && options.every((option) => selected.includes(option));\n        const none = options.every((option) => !selected.includes(option));\n\n        return !all && !none;\n    });\n\n    /**\n     * Retorna uma lista de opções filtradas com base no valor de busca (`filterValue`) e nos critérios definidos.\n     * - Se o filtro estiver desabilitado ou o valor de busca estiver vazio, retorna todas as opções.\n     * - Caso contrário, filtra as opções utilizando o(s) campo(s) especificado(s) em `filterBy`.\n     * - Ignora opções que possuem a propriedade `grouper` definida.\n     * @returns As opções filtradas conforme o valor de busca e critérios de filtro.\n     */\n    public filteredOptions = computed(() => {\n        const allOptions = this.internalOptions();\n        const search = this.filterValue().toLowerCase().trim();\n\n        if (!this.filter() || !search) {\n            return allOptions;\n        }\n\n        const filterBy = this.filterBy();\n        return allOptions.filter((option) => {\n            if (option.grouper) {\n                return false;\n            }\n\n            let label = '';\n\n            const optionLabel = this.optionLabel();\n\n            if (!filterBy) {\n                label = optionLabel ? String(option.data[optionLabel]) : String(option.data);\n            } else if (Array.isArray(filterBy)) {\n                label = filterBy.map((by) => String(option.data[by] ?? '')).join(' ');\n            } else {\n                label = String(option.data[filterBy] ?? '');\n            }\n\n            return label.toLowerCase().includes(search);\n        });\n    });\n\n    public readonly filterForm = new FormGroup<{ filter: FormControl<string> }>({\n        filter: new FormControl<string>('', { nonNullable: true }),\n    });\n\n    private _containerDiv = viewChild<ElementRef<HTMLDivElement>>('containerDiv');\n    private _onChange: (value: any | any[] | null) => void = () => {};\n    private _onTouched: () => void = () => {};\n\n    private readonly _translateService = inject(TranslateService);\n\n    constructor() {\n        this.filterForm\n            .get('filter')!\n            .valueChanges.pipe(debounceTime(300), distinctUntilChanged(), takeUntilDestroyed())\n            .subscribe((value) => {\n                this.filterValue.set(value);\n            });\n    }\n\n    public ngOnInit(): void {\n        this._validateInputs();\n    }\n\n    /**\n     * Define o valor selecionado do componente select.\n     * @param value O valor a ser definido. Pode ser um objeto do tipo `T`, um array de `T` (quando múltipla seleção está habilitada), ou `null`.\n     * @throws Se o modo múltiplo estiver ativado e o valor não for um array, lança um erro.\n     * @throws Se o modo múltiplo estiver desativado e o valor for um array, lança um erro.\n     * Este método mapeia o(s) valor(es) fornecido(s) para as opções internas do componente,\n     * garantindo que apenas opções válidas sejam selecionadas conforme o modo de seleção (único ou múltiplo).\n     */\n    public writeValue(value: T | T[] | null): void {\n        if (value && this.multiple()) {\n            if (Array.isArray(value)) {\n                const mapped = value\n                    .map((val) => {\n                        return this.internalOptions().find((option) => {\n                            return JSON.stringify(option.data) === JSON.stringify(val);\n                        });\n                    })\n                    .filter((option): option is InternalDropdownOption<T> => !!option);\n\n                this.values.set(mapped);\n            } else {\n                throw new Error('Value must be an array when \"multiple\" is true.');\n            }\n        } else {\n            const optionValue = this.optionValue();\n            let mapped: InternalDropdownOption<T> | null = null;\n\n            if (Array.isArray(value)) {\n                throw new Error('Value must not be an array when \"multiple\" is false.');\n            }\n\n            if (optionValue && value) {\n                mapped =\n                    this.internalOptions().find((option) => option.data[optionValue] === value[optionValue]) ?? null;\n            } else {\n                mapped =\n                    this.internalOptions().find((option) => JSON.stringify(option.data) === JSON.stringify(value)) ??\n                    null;\n            }\n\n            this.value.set(mapped ?? null);\n        }\n    }\n\n    /**\n     * Registra uma função de callback que será chamada sempre que o valor do componente mudar.\n     * @param onChange Função de callback que recebe o novo valor selecionado ou null.\n     */\n    public registerOnChange(onChange: (value: T | T[] | null) => void): void {\n        this._onChange = onChange;\n    }\n\n    /**\n     * Registra uma função de callback que será chamada quando o componente for tocado (perder o foco).\n     * @param onTouched Função de callback a ser executada quando o componente for marcado como \"tocado\".\n     */\n    public registerOnTouched(onTouched: () => void): void {\n        this._onTouched = onTouched;\n    }\n\n    /**\n     * Define o estado de desabilitado do componente.\n     * @param disabled Indica se o componente deve ser desabilitado (`true`) ou habilitado (`false`).\n     */\n    public setDisabledState(disabled: boolean): void {\n        this.disabled.set(disabled);\n    }\n\n    /**\n     * Alterna a exibição das opções do componente select.\n     * Se o componente estiver desabilitado, não realiza nenhuma ação.\n     * Caso contrário, inverte o estado de exibição das opções.\n     */\n    public toggle(): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        const shouldOpen = !this.showOptions();\n        this.showOptions.set(shouldOpen);\n    }\n\n    /**\n     * Seleciona um item no componente de seleção.\n     * @param item O item a ser selecionado ou desmarcado.\n     * Se o componente estiver desabilitado, a função retorna imediatamente.\n     * No modo múltiplo, adiciona ou remove o item da lista de valores selecionados.\n     * No modo simples, define o item como valor selecionado e fecha o dropdown.\n     * Sempre notifica as mudanças e marca o componente como tocado.\n     */\n    public selectItem(item: InternalDropdownOption<T>): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        const optionValue = this.optionValue();\n\n        if (this.multiple()) {\n            const store = [...this.values()];\n            const index = store.findIndex((selectedItem) => {\n                return selectedItem === item;\n            });\n\n            if (index !== -1) {\n                store.splice(index, 1);\n            } else {\n                store.push(item);\n            }\n\n            this.values.set(store);\n            const result = optionValue ? store.map((item) => item.data[optionValue]) : store;\n            this._onChange(result);\n        } else {\n            this.value.set(item);\n            const result = optionValue ? item.data[optionValue] : item.data;\n            this._onChange(result);\n            this._closeDropdown();\n        }\n\n        this._onTouched();\n    }\n\n    /**\n     * Limpa o valor selecionado no componente select.\n     * Se o componente estiver desabilitado, não executa nenhuma ação.\n     * Para seleção múltipla, remove todos os valores selecionados e notifica a alteração.\n     * Para seleção única, define o valor como nulo e notifica a alteração.\n     * Fecha o dropdown e marca o componente como \"tocado\".\n     * Impede a propagação do evento do mouse.\n     * @param event Evento do mouse que acionou a limpeza da seleção.\n     */\n    public clear(event: MouseEvent): void {\n        if (this.disabled()) {\n            return;\n        }\n\n        if (this.multiple()) {\n            this.values.set([]);\n            this._onChange(this.values());\n        } else {\n            this.value.set(null);\n            this._onChange(this.value());\n        }\n\n        this._closeDropdown();\n\n        this._onTouched();\n        event.stopPropagation();\n    }\n\n    /**\n     * Limpa o filtro do formulário, resetando o campo 'filter' do formulário de filtro.\n     * Este método redefine o valor do campo 'filter' para seu estado inicial,\n     * removendo qualquer texto ou valor previamente inserido pelo usuário.\n     */\n    public clearFilter(): void {\n        this.filterForm.get('filter')?.reset();\n    }\n\n    /**\n     * Manipula o evento de redimensionamento da janela.\n     * Fecha o dropdown de opções caso ele esteja visível ao redimensionar a janela.\n     * @param _ Evento de redimensionamento da janela (não utilizado).\n     */\n    public onWindowResize(_: Event): void {\n        if (this.showOptions()) {\n            this._closeDropdown();\n        }\n    }\n\n    /**\n     * Manipula o evento de clique no elemento container da seleção.\n     * Este método alterna o estado de exibição do componente select e,\n     * em seguida, define o foco no elemento container correspondente.\n     */\n    public onContainerDivClick(): void {\n        this.toggle();\n        this._containerDiv()?.nativeElement.focus();\n    }\n\n    /**\n     * Manipula eventos de teclado para navegação e seleção de opções no componente select.\n     * - 'ArrowDown': Move o foco para a próxima opção disponível.\n     * - 'ArrowUp': Move o foco para a opção anterior.\n     * - 'Tab': Seleciona a opção atualmente focada e fecha o dropdown.\n     * - 'Enter': Seleciona a opção focada se o dropdown estiver aberto, ou alterna a abertura do dropdown.\n     * - 'Escape': Fecha o dropdown.\n     * - ' ': (barra de espaço) Seleciona a opção focada se o dropdown estiver aberto, ou alterna a abertura do dropdown.\n     * Previne o comportamento padrão do navegador quando necessário para garantir a navegação adequada pelo teclado.\n     * @param event O evento de teclado disparado pelo usuário.\n     */\n    public onKeyDown(event: KeyboardEvent): void {\n        const options = this.filteredOptions();\n\n        if (!options.length) {\n            return;\n        }\n\n        const _selectOption = () => {\n            if (this.focusedIndex() >= 0 && this.focusedIndex() < options.length) {\n                event.preventDefault();\n                this.selectItem(options[this.focusedIndex()]);\n            }\n        };\n\n        switch (event.key) {\n            case 'ArrowDown':\n                event.preventDefault();\n                this._focusNextOption(options);\n                break;\n            case 'ArrowUp':\n                event.preventDefault();\n                this._focusPreviousOption(options);\n                break;\n            case 'Tab':\n                _selectOption();\n                this._closeDropdown();\n                break;\n            case 'Enter':\n                if (this.showOptions()) {\n                    _selectOption();\n                } else {\n                    this.toggle();\n                }\n                break;\n            case 'Escape':\n                this._closeDropdown();\n                break;\n            case ' ':\n                event.preventDefault();\n                if (this.showOptions()) {\n                    _selectOption();\n                } else {\n                    this.toggle();\n                }\n                break;\n        }\n    }\n\n    /**\n     * Retorna o rótulo (label) associado a uma opção fornecida.\n     * @param option - A opção do tipo T para a qual o rótulo deve ser obtido.\n     * @returns O rótulo da opção como uma string.\n     */\n    public getOptionLabel(option: T): string {\n        return this._getLabelFromOption(option);\n    }\n\n    /**\n     * Função de trackBy utilizada em diretivas *ngFor para otimizar a renderização de listas.\n     * Retorna o identificador único (`id`) de cada item do tipo `InternalDropdownOption<T>`.\n     * @param _ - Índice do item na lista (não utilizado).\n     * @param item - O item atual da lista do tipo `InternalDropdownOption<T>`.\n     * @returns O identificador único do item como uma string.\n     */\n    public trackById(_: number, item: InternalDropdownOption<T>): string {\n        return item.id;\n    }\n\n    /**\n     * Verifica se a opção fornecida está selecionada.\n     * @param option - A opção interna do dropdown a ser verificada.\n     * @returns `true` se a opção estiver selecionada, caso contrário `false`.\n     */\n    public isOptionSelected(option: InternalDropdownOption<T>): boolean {\n        return (\n            this.values().find((o) => {\n                return o === option;\n            }) !== undefined\n        );\n    }\n\n    /**\n     * Alterna a seleção de todos os itens disponíveis.\n     * Se o modo múltiplo estiver ativado, seleciona todos os itens filtrados que não são agrupadores\n     * caso nem todos estejam selecionados, ou limpa a seleção caso todos já estejam selecionados.\n     * Após a alteração, notifica a mudança de valor.\n     */\n    public toggleSelectAll(): void {\n        if (!this.multiple()) {\n            return;\n        }\n\n        const options = this.filteredOptions().filter((option) => !option.grouper);\n\n        if (this.allSelected()) {\n            this.values.set([]);\n        } else {\n            this.values.set(options);\n        }\n\n        this._onChange(this.values());\n    }\n\n    /**\n     * Retorna o rótulo (label) de uma opção fornecida.\n     * Este método utiliza a propriedade definida por `optionLabel` para extrair o valor do rótulo da opção.\n     * Caso `optionLabel` não esteja definido, retorna a representação em string da própria opção.\n     * @private\n     * @param option A opção da qual o rótulo será extraído.\n     * @returns O rótulo da opção como uma string.\n     */\n    private _getLabelFromOption(option: T): string {\n        const optionLabel = this.optionLabel();\n        return optionLabel ? String(option[optionLabel]) : String(option);\n    }\n\n    /**\n     * Valida as entradas do componente de seleção.\n     * Este método verifica se as opções fornecidas são válidas. Caso as opções sejam objetos,\n     * garante que a propriedade `optionLabel` esteja definida, lançando um erro caso contrário.\n     * Se não houver opções, a validação é ignorada.\n     * @private\n     * @throws {Error} Se as opções forem objetos e `optionLabel` não estiver definido.\n     */\n    private _validateInputs(): void {\n        const options = this.options();\n\n        if (!options.length) {\n            return;\n        }\n\n        const hasObjectOptions = options.some((option) => typeof option === 'object' && option !== null);\n\n        if (hasObjectOptions) {\n            if (!this.optionLabel()) {\n                throw new Error('The \"optionLabel\" input is required when options are objects.');\n            }\n        }\n    }\n\n    /**\n     * Fecha o dropdown de opções do componente select.\n     * Este método oculta as opções disponíveis e redefine o índice do item focado para -1,\n     * indicando que nenhum item está atualmente focado.\n     * @private\n     */\n    private _closeDropdown(): void {\n        this.showOptions.set(false);\n        this.focusedIndex.set(-1);\n    }\n\n    /**\n     * Adiciona um identificador único a cada opção fornecida.\n     * Para cada item no array de opções, gera um objeto `InternalDropdownOption` contendo o dado original\n     * e um campo `id` único baseado no timestamp atual e no índice do item. Caso a opção seja um grupo\n     * (detectado pela presença de um array `items`), adiciona também a propriedade `grouper: true`.\n     * @private\n     * @param options Array de opções do tipo `T` a serem processadas.\n     * @returns Um array de objetos `InternalDropdownOption<T>` com identificadores únicos.\n     */\n    private _addIdToOptions(options: T[]): InternalDropdownOption<T>[] {\n        const timestamp = Date.now();\n\n        return options.map((option, index) => {\n            if (this.group() && Array.isArray((option as any).items)) {\n                return {\n                    data: option,\n                    id: `id_${timestamp}_${index++}`,\n                    grouper: true,\n                };\n            }\n\n            return {\n                data: option,\n                id: `id_${timestamp}_${index++}`,\n            };\n        });\n    }\n\n    /**\n     * Converte uma lista de opções possivelmente agrupadas em uma lista plana de opções.\n     * Para cada opção que possui um agrupador (`grouper`) e um array de itens em `data.items`,\n     * adiciona a opção do agrupador e, em seguida, adiciona cada item do grupo como uma nova opção\n     * individual, atribuindo um `id` único para cada item do grupo.\n     * Caso a opção não seja um agrupador, ela é adicionada diretamente ao resultado.\n     * @private\n     * @param options Lista de opções internas do dropdown, podendo conter agrupadores.\n     * @returns Lista plana de opções, incluindo agrupadores e seus itens expandidos.\n     */\n    private _grouperToFlat(options: InternalDropdownOption<T>[]): InternalDropdownOption<T>[] {\n        const result: InternalDropdownOption<T>[] = [];\n\n        options.forEach((option) => {\n            if (option.grouper && Array.isArray((option.data as any).items)) {\n                result.push(option);\n                const groupItems = (option.data as any).items.map((item: T, index: number) => ({\n                    data: item,\n                    id: `${option.id}_item_${index}`,\n                }));\n                result.push(...groupItems);\n            } else {\n                result.push(option);\n            }\n        });\n\n        return result;\n    }\n\n    /**\n     * Move o foco para a próxima opção disponível na lista, ignorando opções que sejam agrupadores.\n     * @private\n     * @param options Lista de opções internas do dropdown.\n     */\n    private _focusNextOption(options: InternalDropdownOption<T>[]) {\n        let nextIndex = this.focusedIndex() + 1;\n\n        while (nextIndex < options.length && options[nextIndex].grouper) {\n            nextIndex++;\n        }\n\n        if (nextIndex < options.length) {\n            this.focusedIndex.set(nextIndex);\n            this._scrollToFocusedOption();\n        }\n    }\n\n    /**\n     * Move o foco para a opção anterior na lista, ignorando opções do tipo \"grouper\".\n     * @private\n     * @param options Lista de opções internas do dropdown.\n     */\n    private _focusPreviousOption(options: InternalDropdownOption<T>[]) {\n        let prevIndex = this.focusedIndex() - 1;\n\n        while (prevIndex >= 0 && options[prevIndex].grouper) {\n            prevIndex--;\n        }\n\n        if (prevIndex >= 0) {\n            this.focusedIndex.set(prevIndex);\n            this._scrollToFocusedOption();\n        }\n    }\n\n    /**\n     * Rola a lista de opções para garantir que a opção atualmente focada esteja visível.\n     * Este método obtém o ID da opção atualmente focada a partir da lista de opções filtradas.\n     * Se um ID válido for encontrado, busca o elemento correspondente no DOM e utiliza\n     * `scrollIntoView` para rolar até a opção, alinhando-a ao bloco mais próximo.\n     * @private\n     */\n    private _scrollToFocusedOption() {\n        const id = this.filteredOptions()[this.focusedIndex()]?.id;\n\n        if (!id) {\n            return;\n        }\n\n        const el = document.getElementById(id);\n        el?.scrollIntoView({ block: 'nearest' });\n    }\n}\n","<div\n    #containerDiv\n    cdkOverlayOrigin\n    #trigger=\"cdkOverlayOrigin\"\n    class=\"select-container group flex h-[35px] w-full overflow-hidden rounded-[3px] border outline-1 outline-primary focus:outline\"\n    [ngClass]=\"{\n        'pointer-events-none border-grayscale-20 bg-grayscale-5': disabled(),\n        'pointer-events-auto border-grayscale-30 bg-grayscale-0': !disabled(),\n    }\"\n    [attr.aria-disabled]=\"disabled()\"\n    [attr.aria-expanded]=\"showOptions()\"\n    [attr.aria-haspopup]=\"'listbox'\"\n    [attr.aria-owns]=\"'dropdown-container-' + componentId\"\n    [attr.aria-labelledby]=\"componentId + '-label'\"\n    role=\"combobox\"\n    (click)=\"onContainerDivClick()\"\n    (keydown)=\"onKeyDown($event)\"\n    tabindex=\"0\"\n>\n    <span\n        class=\"flex flex-grow select-none px-3 py-[7px]\"\n        [ngClass]=\"{\n            'text-grayscale-90': !disabled() && !isClean(),\n            'text-grayscale-60': !disabled() && isClean(),\n            'text-grayscale-30': disabled(),\n        }\"\n    >\n        {{ print() || placeholder() }}\n    </span>\n\n    @if (!disabled() && showClear() && !isClean()) {\n        <button\n            class=\"mx-3\"\n            (click)=\"clear($event)\"\n        >\n            <i class=\"fas fa-times flex items-center\"></i>\n        </button>\n    }\n\n    <div\n        class=\"dropdown-container flex items-center border-l px-3 group-hover:bg-grayscale-10 group-focus:border-primary\"\n        [ngClass]=\"{ 'border-grayscale-20': disabled(), 'border-grayscale-30': !disabled() }\"\n    >\n        <i\n            class=\"fas\"\n            [ngClass]=\"{\n                'text-grayscale-30': disabled(),\n                'text-grayscale-90': !disabled(),\n                'fa-caret-down': !showOptions(),\n                'fa-caret-up': showOptions(),\n            }\"\n        ></i>\n    </div>\n</div>\n\n<ng-template\n    #dropdownTemplate\n    cdkConnectedOverlay\n    [cdkConnectedOverlayOrigin]=\"trigger\"\n    [cdkConnectedOverlayOpen]=\"showOptions()\"\n    [cdkConnectedOverlayHasBackdrop]=\"true\"\n    (backdropClick)=\"showOptions.set(false)\"\n    cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n    (detach)=\"showOptions.set(false)\"\n>\n    <div\n        [id]=\"'dropdown-container-' + componentId\"\n        class=\"dropdown-body-class z-[1000] rounded-[3px] bg-grayscale-0 py-1 shadow-md\"\n        (click)=\"$event.stopPropagation()\"\n        [style.width.px]=\"containerDiv.offsetWidth\"\n    >\n        @if (filter()) {\n            <div class=\"flex w-full items-center gap-3 px-3 py-1.5\">\n                @if (multiple()) {\n                    <s-checkbox\n                        #selectAllCheckbox\n                        [checked]=\"allSelected()\"\n                        [indeterminate]=\"selectAllIsIndeterminate()\"\n                        (checkedChange)=\"toggleSelectAll()\"\n                    ></s-checkbox>\n                }\n\n                <div class=\"relative flex h-[35px] grow\">\n                    <form\n                        class=\"flex grow\"\n                        [formGroup]=\"filterForm\"\n                    >\n                        <input\n                            class=\"w-full grow rounded-[3px] border border-grayscale-30 pl-2.5 pr-7 outline-1 outline-primary\"\n                            type=\"text\"\n                            formControlName=\"filter\"\n                            (click)=\"$event.stopPropagation()\"\n                        />\n                    </form>\n                    <i class=\"fas fa-search absolute right-2.5 top-2.5 text-grayscale-90\"></i>\n                </div>\n                <button (click)=\"clearFilter()\">\n                    <i class=\"fas fa-times flex items-center\"></i>\n                </button>\n            </div>\n        }\n\n        <!-- Virtual scroll -->\n        @if (virtualScroll() && filteredOptions().length > 10) {\n            <cdk-virtual-scroll-viewport\n                [itemSize]=\"virtualScrollItemSize()\"\n                class=\"h-52 overflow-auto\"\n            >\n                <ng-container *cdkVirtualFor=\"let option of filteredOptions(); trackBy: trackById\">\n                    <s-select-option\n                        [id]=\"option.id\"\n                        [label]=\"getOptionLabel(option.data)\"\n                        [multiple]=\"multiple()\"\n                        [checkmark]=\"checkmark()\"\n                        [isSelected]=\"isOptionSelected(option)\"\n                        (selected)=\"selectItem(option)\"\n                        [isGrouper]=\"option.grouper\"\n                        [isFocused]=\"option.data === focusedItem()\"\n                    ></s-select-option>\n                </ng-container>\n            </cdk-virtual-scroll-viewport>\n        }\n\n        <!-- Normal list -->\n        @if (!virtualScroll() || filteredOptions().length <= 10) {\n            <ul class=\"max-h-52 overflow-auto\">\n                @for (option of filteredOptions(); track option.id) {\n                    <s-select-option\n                        [id]=\"option.id\"\n                        [label]=\"getOptionLabel(option.data)\"\n                        [multiple]=\"multiple()\"\n                        [checkmark]=\"checkmark()\"\n                        [isSelected]=\"isOptionSelected(option)\"\n                        (selected)=\"selectItem(option)\"\n                        [isGrouper]=\"option.grouper ?? false\"\n                        [isFocused]=\"option.data === focusedItem()\"\n                    ></s-select-option>\n                }\n            </ul>\n        }\n\n        @if (filteredOptions().length === 0) {\n            <span class=\"m-3 text-grayscale-60\">{{\n                emptyMessage() ?? 'platform.angular_components.no_records_found' | translate\n            }}</span>\n        }\n    </div>\n</ng-template>\n"]}
|