@seniorsistemas/angular-components 19.0.6 → 19.0.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/kanban/lib/kanban/kanban.component.mjs +3 -3
- package/esm2022/select/lib/select/select.component.mjs +38 -17
- package/fesm2022/seniorsistemas-angular-components-kanban.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-kanban.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-select.mjs +37 -16
- package/fesm2022/seniorsistemas-angular-components-select.mjs.map +1 -1
- package/package.json +7 -7
- package/select/lib/select/select.component.d.ts +6 -0
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { OverlayModule } from '@angular/cdk/overlay';
|
|
2
2
|
import { ScrollingModule } from '@angular/cdk/scrolling';
|
|
3
3
|
import { CommonModule } from '@angular/common';
|
|
4
|
-
import { Component, computed, forwardRef, inject, input, model, signal, viewChild, } from '@angular/core';
|
|
4
|
+
import { Component, computed, forwardRef, inject, input, model, signal, viewChild, effect, } from '@angular/core';
|
|
5
5
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
6
6
|
import { FormControl, FormGroup, NG_VALUE_ACCESSOR, ReactiveFormsModule } from '@angular/forms';
|
|
7
7
|
import { TranslateModule, TranslateService } from '@ngx-translate/core';
|
|
@@ -69,6 +69,7 @@ export class SelectComponent {
|
|
|
69
69
|
values = signal([]);
|
|
70
70
|
filterValue = signal('');
|
|
71
71
|
focusedIndex = signal(-1);
|
|
72
|
+
_pendingValue = null;
|
|
72
73
|
componentId = `select${SelectComponent.nextId++}`;
|
|
73
74
|
/**
|
|
74
75
|
* Computed que indica se todas as opções filtradas estão selecionadas.
|
|
@@ -200,6 +201,13 @@ export class SelectComponent {
|
|
|
200
201
|
.subscribe((value) => {
|
|
201
202
|
this.filterValue.set(value);
|
|
202
203
|
});
|
|
204
|
+
// Efeito para tentar aplicar valor pendente quando options mudar
|
|
205
|
+
effect(() => {
|
|
206
|
+
const opts = this.options();
|
|
207
|
+
if (opts && opts.length && this._pendingValue !== null) {
|
|
208
|
+
this._applyPendingValue();
|
|
209
|
+
}
|
|
210
|
+
});
|
|
203
211
|
}
|
|
204
212
|
ngOnInit() {
|
|
205
213
|
this._validateInputs();
|
|
@@ -213,38 +221,51 @@ export class SelectComponent {
|
|
|
213
221
|
* garantindo que apenas opções válidas sejam selecionadas conforme o modo de seleção (único ou múltiplo).
|
|
214
222
|
*/
|
|
215
223
|
writeValue(value) {
|
|
224
|
+
// Armazena valor pendente e tenta aplicar
|
|
225
|
+
this._pendingValue = value;
|
|
226
|
+
this._applyPendingValue();
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Tenta aplicar o valor pendente (_pendingValue) ao estado do componente,
|
|
230
|
+
* mapeando para InternalDropdownOption<T> conforme options disponíveis.
|
|
231
|
+
*/
|
|
232
|
+
_applyPendingValue() {
|
|
233
|
+
const value = this._pendingValue;
|
|
234
|
+
const opts = this.internalOptions();
|
|
235
|
+
const optionValue = this.optionValue();
|
|
216
236
|
if (value && this.multiple()) {
|
|
217
237
|
if (Array.isArray(value)) {
|
|
218
238
|
const mapped = value
|
|
219
|
-
.map((
|
|
220
|
-
|
|
221
|
-
return
|
|
222
|
-
}
|
|
239
|
+
.map((v) => {
|
|
240
|
+
if (optionValue) {
|
|
241
|
+
return opts.find((opt) => opt.data[optionValue] === v) || null;
|
|
242
|
+
}
|
|
243
|
+
else {
|
|
244
|
+
return opts.find((opt) => JSON.stringify(opt.data) === JSON.stringify(v)) || null;
|
|
245
|
+
}
|
|
223
246
|
})
|
|
224
|
-
.filter((
|
|
247
|
+
.filter((opt) => !!opt);
|
|
225
248
|
this.values.set(mapped);
|
|
226
249
|
}
|
|
227
250
|
else {
|
|
228
|
-
|
|
251
|
+
this.values.set([]);
|
|
229
252
|
}
|
|
230
253
|
}
|
|
231
254
|
else {
|
|
232
|
-
const optionValue = this.optionValue();
|
|
233
255
|
let mapped = null;
|
|
234
256
|
if (Array.isArray(value)) {
|
|
235
|
-
|
|
257
|
+
mapped = null;
|
|
236
258
|
}
|
|
237
|
-
if (optionValue && value) {
|
|
238
|
-
|
|
239
|
-
|
|
259
|
+
else if (optionValue && value != null) {
|
|
260
|
+
const compareValue = typeof value === 'object' && value !== null && optionValue in value ? value[optionValue] : value;
|
|
261
|
+
mapped = opts.find((opt) => opt.data[optionValue] === compareValue) || null;
|
|
240
262
|
}
|
|
241
|
-
else {
|
|
242
|
-
mapped =
|
|
243
|
-
this.internalOptions().find((option) => JSON.stringify(option.data) === JSON.stringify(value)) ??
|
|
244
|
-
null;
|
|
263
|
+
else if (value != null) {
|
|
264
|
+
mapped = opts.find((opt) => JSON.stringify(opt.data) === JSON.stringify(value)) || null;
|
|
245
265
|
}
|
|
246
266
|
this.value.set(mapped ?? null);
|
|
247
267
|
}
|
|
268
|
+
this._pendingValue = null;
|
|
248
269
|
}
|
|
249
270
|
/**
|
|
250
271
|
* Registra uma função de callback que será chamada sempre que o valor do componente mudar.
|
|
@@ -676,4 +697,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
676
697
|
},
|
|
677
698
|
], 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\n", styles: [":host.ng-dirty.ng-invalid{.dropdown-container,.select-container{border-color:#c13018;outline:none}}\n"] }]
|
|
678
699
|
}], ctorParameters: () => [] });
|
|
679
|
-
//# 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/**\n * @description Componente de seleção dropdown genérico com suporte a seleção simples e múltipla,\n * filtro, agrupamento, virtual scroll e acessibilidade (ARIA combobox). Implementa\n * {@link ControlValueAccessor} para integração com Reactive Forms e Template-driven Forms.\n *\n * @example\n * ```html\n * <s-select\n *   formControlName=\"categoria\"\n *   placeholder=\"Selecione uma categoria\"\n *   [options]=\"categorias\"\n *   optionLabel=\"nome\"\n *   optionValue=\"id\"\n *   [filter]=\"true\"\n *   [showClear]=\"true\" />\n * ```\n *\n * @category Inputs\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    /** @description Texto exibido quando nenhuma opção está selecionada. @default '' */\n    public placeholder = input('');\n\n    /** @description Habilita a seleção de múltiplas opções simultâneas. @default false */\n    public multiple = input(false);\n\n    /** @description Lista de opções disponíveis para seleção. Campo obrigatório. */\n    public options = input.required<T[]>();\n\n    /** @description Chave do objeto de opção usada como label de exibição. */\n    public optionLabel = input<keyof T>();\n\n    /** @description Chave do objeto de opção usada como valor a ser emitido ao formulário. */\n    public optionValue = input<keyof T>();\n\n    /** @description Exibe um botão para limpar a seleção atual. @default false */\n    public showClear = input(false);\n\n    /** @description Habilita um campo de filtro textual para pesquisa nas opções. @default false */\n    public filter = input(false);\n\n    /** @description Exibe um ícone de checkmark ao lado da opção selecionada. @default false */\n    public checkmark = input(false);\n\n    /** @description Chave(s) do objeto usada(s) para filtragem textual. Quando não informado, usa `optionLabel`. */\n    public filterBy = input<keyof T | (keyof T)[]>();\n\n    /** @description Habilita agrupamento de opções. As opções devem conter itens filhos conforme a estrutura de grupo suportada. @default false */\n    public group = input(false);\n\n    /** @description Habilita virtualização da lista para melhor performance com grandes volumes de opções. @default false */\n    public virtualScroll = input(false);\n\n    /** @description Altura fixa de cada item da lista em pixels, necessária para o cálculo do virtual scroll. @default 37 */\n    public virtualScrollItemSize = input(37);\n\n    /** @description Mensagem exibida quando nenhuma opção corresponde ao filtro aplicado. @default null */\n    public emptyMessage = input<string | null>(null);\n\n    /** @description Controla o estado desabilitado do componente via two-way binding. @default false */\n    public disabled = model(false);\n\n    /** @description Controla a visibilidade do painel de opções via two-way binding. @default 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(() => {\n        const index = this.focusedIndex();\n        if (index < 0) return null;\n        return this.filteredOptions()[index]?.data ?? null;\n    });\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     * Ao abrir, move o foco para o item atualmente selecionado.\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            this._focusSelectedOption();\n        }\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(): void {\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     * Define o foco no item atualmente selecionado ao abrir o dropdown.\n     * No modo simples, foca o item selecionado. No modo múltiplo, foca o primeiro item selecionado.\n     * Após renderização do overlay, rola a lista até o item focado.\n     * @private\n     */\n    private _focusSelectedOption(): void {\n        const options = this.filteredOptions();\n        let selectedIndex = -1;\n\n        if (this.multiple()) {\n            const firstSelected = this.values()[0];\n            if (firstSelected) {\n                selectedIndex = options.findIndex((o) => o.id === firstSelected.id);\n            }\n        } else {\n            const selected = this.value();\n            if (selected) {\n                selectedIndex = options.findIndex((o) => o.id === selected.id);\n            }\n        }\n\n        this.focusedIndex.set(selectedIndex);\n\n        if (selectedIndex >= 0) {\n            setTimeout(() => this._scrollToFocusedOption());\n        }\n    }\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\n"]}
|
|
700
|
+
//# 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    effect,\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/**\n * @description Componente de seleção dropdown genérico com suporte a seleção simples e múltipla,\n * filtro, agrupamento, virtual scroll e acessibilidade (ARIA combobox). Implementa\n * {@link ControlValueAccessor} para integração com Reactive Forms e Template-driven Forms.\n *\n * @example\n * ```html\n * <s-select\n *   formControlName=\"categoria\"\n *   placeholder=\"Selecione uma categoria\"\n *   [options]=\"categorias\"\n *   optionLabel=\"nome\"\n *   optionValue=\"id\"\n *   [filter]=\"true\"\n *   [showClear]=\"true\" />\n * ```\n *\n * @category Inputs\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    /** @description Texto exibido quando nenhuma opção está selecionada. @default '' */\n    public placeholder = input('');\n\n    /** @description Habilita a seleção de múltiplas opções simultâneas. @default false */\n    public multiple = input(false);\n\n    /** @description Lista de opções disponíveis para seleção. Campo obrigatório. */\n    public options = input.required<T[]>();\n\n    /** @description Chave do objeto de opção usada como label de exibição. */\n    public optionLabel = input<keyof T>();\n\n    /** @description Chave do objeto de opção usada como valor a ser emitido ao formulário. */\n    public optionValue = input<keyof T>();\n\n    /** @description Exibe um botão para limpar a seleção atual. @default false */\n    public showClear = input(false);\n\n    /** @description Habilita um campo de filtro textual para pesquisa nas opções. @default false */\n    public filter = input(false);\n\n    /** @description Exibe um ícone de checkmark ao lado da opção selecionada. @default false */\n    public checkmark = input(false);\n\n    /** @description Chave(s) do objeto usada(s) para filtragem textual. Quando não informado, usa `optionLabel`. */\n    public filterBy = input<keyof T | (keyof T)[]>();\n\n    /** @description Habilita agrupamento de opções. As opções devem conter itens filhos conforme a estrutura de grupo suportada. @default false */\n    public group = input(false);\n\n    /** @description Habilita virtualização da lista para melhor performance com grandes volumes de opções. @default false */\n    public virtualScroll = input(false);\n\n    /** @description Altura fixa de cada item da lista em pixels, necessária para o cálculo do virtual scroll. @default 37 */\n    public virtualScrollItemSize = input(37);\n\n    /** @description Mensagem exibida quando nenhuma opção corresponde ao filtro aplicado. @default null */\n    public emptyMessage = input<string | null>(null);\n\n    /** @description Controla o estado desabilitado do componente via two-way binding. @default false */\n    public disabled = model(false);\n\n    /** @description Controla a visibilidade do painel de opções via two-way binding. @default 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    private _pendingValue: T | T[] | null = null;\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(() => {\n        const index = this.focusedIndex();\n        if (index < 0) return null;\n        return this.filteredOptions()[index]?.data ?? null;\n    });\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        // Efeito para tentar aplicar valor pendente quando options mudar\n        effect(() => {\n            const opts = this.options();\n            if (opts && opts.length && this._pendingValue !== null) {\n                this._applyPendingValue();\n            }\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        // Armazena valor pendente e tenta aplicar\n        this._pendingValue = value;\n        this._applyPendingValue();\n    }\n\n    /**\n     * Tenta aplicar o valor pendente (_pendingValue) ao estado do componente,\n     * mapeando para InternalDropdownOption<T> conforme options disponíveis.\n     */\n    private _applyPendingValue(): void {\n        const value = this._pendingValue;\n        const opts = this.internalOptions();\n        const optionValue = this.optionValue();\n\n        if (value && this.multiple()) {\n            if (Array.isArray(value)) {\n                const mapped = value\n                    .map((v) => {\n                        if (optionValue) {\n                            return opts.find((opt) => opt.data[optionValue] === v) || null;\n                        } else {\n                            return opts.find((opt) => JSON.stringify(opt.data) === JSON.stringify(v)) || null;\n                        }\n                    })\n                    .filter((opt): opt is InternalDropdownOption<T> => !!opt);\n                this.values.set(mapped);\n            } else {\n                this.values.set([]);\n            }\n        } else {\n            let mapped: InternalDropdownOption<T> | null = null;\n            if (Array.isArray(value)) {\n                mapped = null;\n            } else if (optionValue && value != null) {\n                const compareValue =\n                    typeof value === 'object' && value !== null && optionValue in value ? value[optionValue] : value;\n                mapped = opts.find((opt) => opt.data[optionValue] === compareValue) || null;\n            } else if (value != null) {\n                mapped = opts.find((opt) => JSON.stringify(opt.data) === JSON.stringify(value)) || null;\n            }\n            this.value.set(mapped ?? null);\n        }\n        this._pendingValue = null;\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     * Ao abrir, move o foco para o item atualmente selecionado.\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            this._focusSelectedOption();\n        }\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(): void {\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     * Define o foco no item atualmente selecionado ao abrir o dropdown.\n     * No modo simples, foca o item selecionado. No modo múltiplo, foca o primeiro item selecionado.\n     * Após renderização do overlay, rola a lista até o item focado.\n     * @private\n     */\n    private _focusSelectedOption(): void {\n        const options = this.filteredOptions();\n        let selectedIndex = -1;\n\n        if (this.multiple()) {\n            const firstSelected = this.values()[0];\n            if (firstSelected) {\n                selectedIndex = options.findIndex((o) => o.id === firstSelected.id);\n            }\n        } else {\n            const selected = this.value();\n            if (selected) {\n                selectedIndex = options.findIndex((o) => o.id === selected.id);\n            }\n        }\n\n        this.focusedIndex.set(selectedIndex);\n\n        if (selectedIndex >= 0) {\n            setTimeout(() => this._scrollToFocusedOption());\n        }\n    }\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\n"]}
|
|
@@ -402,11 +402,11 @@ class KanbanComponent {
|
|
|
402
402
|
this.itemFooterTemplate = this.getItemFooterTemplate();
|
|
403
403
|
}
|
|
404
404
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: KanbanComponent, deps: [{ token: KanbanEventService }], target: i0.ɵɵFactoryTarget.Component });
|
|
405
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: KanbanComponent, selector: "s-kanban", inputs: { data: "data", showItemCheckboxes: "showItemCheckboxes", showColumnCheckboxes: "showColumnCheckboxes" }, outputs: { itemsMoved: "itemsMoved", dataUpdated: "dataUpdated", itemsSelected: "itemsSelected" }, queries: [{ propertyName: "templates", predicate: TemplateDirective }], ngImport: i0, template: "<div class=\"kanban\">\n
|
|
405
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: KanbanComponent, selector: "s-kanban", inputs: { data: "data", showItemCheckboxes: "showItemCheckboxes", showColumnCheckboxes: "showColumnCheckboxes" }, outputs: { itemsMoved: "itemsMoved", dataUpdated: "dataUpdated", itemsSelected: "itemsSelected" }, queries: [{ propertyName: "templates", predicate: TemplateDirective }], ngImport: i0, template: "<div class=\"kanban\">\n <s-kanban-column\n *ngFor=\"let column of data.columns\"\n [data]=\"column\"\n [showCheckbox]=\"showColumnCheckboxes\"\n [headerTemplate]=\"columnHeaderTemplate\"\n >\n <div\n [id]=\"column.id + ''\"\n style=\"height: 100%; width: 100%\"\n cdkDropList\n #dynamicList=\"cdkDropList\"\n [cdkDropListData]=\"column.items\"\n [cdkDropListConnectedTo]=\"getLinkedColumns(column)\"\n (cdkDropListDropped)=\"drop($event)\"\n >\n @if (column.items.length === 0) {\n @if (!columnEmptyMessageTemplate) {\n <div class=\"empty-message\">\n <p class=\"text\">\n <span class=\"fas fa-clock\"></span> \n <span>{{\n 'platform.angular_components.count_items_in_target'\n | translate: { count: column.items.length, target: column.title }\n }}</span>\n </p>\n </div>\n } @else {\n <ng-container\n *ngTemplateOutlet=\"columnEmptyMessageTemplate; context: { $implicit: column }\"\n ></ng-container>\n }\n }\n\n <div\n *ngFor=\"let item of column.items\"\n cdkDrag\n [cdkDragData]=\"item\"\n [cdkDragDisabled]=\"item.disabled || item.frozen\"\n (cdkDragStarted)=\"dragStarted()\"\n (cdkDragReleased)=\"dragReleased()\"\n (click)=\"selectItem($event, item, column)\"\n >\n <ng-container *ngTemplateOutlet=\"itemTemplate\"></ng-container>\n\n <ng-container *cdkDragPreview>\n @if (selectedItems.size > 1) {\n <s-kanban-item-dragging [quantityItems]=\"selectedItems.size\"></s-kanban-item-dragging>\n } @else {\n <div class=\"max-w-96\">\n <ng-container *ngTemplateOutlet=\"itemTemplate\"></ng-container>\n </div>\n }\n </ng-container>\n\n <ng-template #itemTemplate>\n <s-kanban-item\n [item]=\"item\"\n [selected]=\"selectedItems.has(item)\"\n [showCheckbox]=\"showItemCheckboxes\"\n [headerTemplate]=\"itemHeaderTemplate\"\n [bodyTemplate]=\"itemBodyTemplate\"\n [footerTemplate]=\"itemFooterTemplate\"\n >\n </s-kanban-item>\n </ng-template>\n\n <div *cdkDragPlaceholder>\n <div class=\"placeholder\">\n <div class=\"placeholder-line\"></div>\n </div>\n </div>\n </div>\n </div>\n </s-kanban-column>\n</div>\n\n", styles: [".kanban{display:flex;gap:16px;overflow:auto;width:100%}.kanban .empty-message{display:flex;justify-content:center;margin:16px}.kanban .empty-message .text{overflow:hidden;text-align:center;text-overflow:ellipsis;width:100%;white-space:nowrap}@media screen and (max-width: 600px){.kanban{flex-direction:column}}s-kanban-column{display:flex;flex-grow:1;min-width:292px}.placeholder{display:flex;justify-content:center}.placeholder .placeholder-line{background-color:#a5a5b2;height:1px;margin:6px 0;width:50%}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3$2.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep", "cdkDropListElementContainer"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i3$2.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer", "cdkDragScale"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "directive", type: i3$2.CdkDragPreview, selector: "ng-template[cdkDragPreview]", inputs: ["data", "matchSize"] }, { kind: "directive", type: i3$2.CdkDragPlaceholder, selector: "ng-template[cdkDragPlaceholder]", inputs: ["data"] }, { kind: "component", type: KanbanItemComponent, selector: "s-kanban-item", inputs: ["item", "selected", "showCheckbox", "headerTemplate", "bodyTemplate", "footerTemplate"] }, { kind: "component", type: KanbanColumnComponent, selector: "s-kanban-column", inputs: ["data", "showCheckbox", "headerTemplate"] }, { kind: "component", type: KanbanItemDraggingComponent, selector: "s-kanban-item-dragging", inputs: ["quantityItems"] }, { kind: "pipe", type: i3$1.TranslatePipe, name: "translate" }] });
|
|
406
406
|
}
|
|
407
407
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: KanbanComponent, decorators: [{
|
|
408
408
|
type: Component,
|
|
409
|
-
args: [{ selector: 's-kanban', template: "<div class=\"kanban\">\n
|
|
409
|
+
args: [{ selector: 's-kanban', template: "<div class=\"kanban\">\n <s-kanban-column\n *ngFor=\"let column of data.columns\"\n [data]=\"column\"\n [showCheckbox]=\"showColumnCheckboxes\"\n [headerTemplate]=\"columnHeaderTemplate\"\n >\n <div\n [id]=\"column.id + ''\"\n style=\"height: 100%; width: 100%\"\n cdkDropList\n #dynamicList=\"cdkDropList\"\n [cdkDropListData]=\"column.items\"\n [cdkDropListConnectedTo]=\"getLinkedColumns(column)\"\n (cdkDropListDropped)=\"drop($event)\"\n >\n @if (column.items.length === 0) {\n @if (!columnEmptyMessageTemplate) {\n <div class=\"empty-message\">\n <p class=\"text\">\n <span class=\"fas fa-clock\"></span> \n <span>{{\n 'platform.angular_components.count_items_in_target'\n | translate: { count: column.items.length, target: column.title }\n }}</span>\n </p>\n </div>\n } @else {\n <ng-container\n *ngTemplateOutlet=\"columnEmptyMessageTemplate; context: { $implicit: column }\"\n ></ng-container>\n }\n }\n\n <div\n *ngFor=\"let item of column.items\"\n cdkDrag\n [cdkDragData]=\"item\"\n [cdkDragDisabled]=\"item.disabled || item.frozen\"\n (cdkDragStarted)=\"dragStarted()\"\n (cdkDragReleased)=\"dragReleased()\"\n (click)=\"selectItem($event, item, column)\"\n >\n <ng-container *ngTemplateOutlet=\"itemTemplate\"></ng-container>\n\n <ng-container *cdkDragPreview>\n @if (selectedItems.size > 1) {\n <s-kanban-item-dragging [quantityItems]=\"selectedItems.size\"></s-kanban-item-dragging>\n } @else {\n <div class=\"max-w-96\">\n <ng-container *ngTemplateOutlet=\"itemTemplate\"></ng-container>\n </div>\n }\n </ng-container>\n\n <ng-template #itemTemplate>\n <s-kanban-item\n [item]=\"item\"\n [selected]=\"selectedItems.has(item)\"\n [showCheckbox]=\"showItemCheckboxes\"\n [headerTemplate]=\"itemHeaderTemplate\"\n [bodyTemplate]=\"itemBodyTemplate\"\n [footerTemplate]=\"itemFooterTemplate\"\n >\n </s-kanban-item>\n </ng-template>\n\n <div *cdkDragPlaceholder>\n <div class=\"placeholder\">\n <div class=\"placeholder-line\"></div>\n </div>\n </div>\n </div>\n </div>\n </s-kanban-column>\n</div>\n\n", styles: [".kanban{display:flex;gap:16px;overflow:auto;width:100%}.kanban .empty-message{display:flex;justify-content:center;margin:16px}.kanban .empty-message .text{overflow:hidden;text-align:center;text-overflow:ellipsis;width:100%;white-space:nowrap}@media screen and (max-width: 600px){.kanban{flex-direction:column}}s-kanban-column{display:flex;flex-grow:1;min-width:292px}.placeholder{display:flex;justify-content:center}.placeholder .placeholder-line{background-color:#a5a5b2;height:1px;margin:6px 0;width:50%}\n"] }]
|
|
410
410
|
}], ctorParameters: () => [{ type: KanbanEventService }], propDecorators: { data: [{
|
|
411
411
|
type: Input,
|
|
412
412
|
args: [{ required: true }]
|