@seniorsistemas/components-ai 0.0.0-master-d4a804fe

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (76) hide show
  1. package/CHANGELOG.md +101 -0
  2. package/CONTRIBUTING.md +317 -0
  3. package/README.md +161 -0
  4. package/docs/API.md +486 -0
  5. package/docs/COMPONENTS.md +272 -0
  6. package/docs/EXAMPLES.md +559 -0
  7. package/docs/MIGRATION.md +367 -0
  8. package/esm2022/lib/angular-components.module.mjs +25 -0
  9. package/esm2022/lib/components/breadcrumb/breadcrumb.component.mjs +121 -0
  10. package/esm2022/lib/components/bulk-delete-dialog/bulk-delete-dialog.component.mjs +176 -0
  11. package/esm2022/lib/components/dynamic-form/dynamic-form.component.mjs +625 -0
  12. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-date.component.mjs +86 -0
  13. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-dropdown.component.mjs +103 -0
  14. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-lookup.component.mjs +599 -0
  15. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-number.component.mjs +92 -0
  16. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-text.component.mjs +163 -0
  17. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-textarea.component.mjs +81 -0
  18. package/esm2022/lib/components/dynamic-form/fields/dynamic-field-wrapper.component.mjs +93 -0
  19. package/esm2022/lib/components/dynamic-form/fields/index.mjs +8 -0
  20. package/esm2022/lib/components/dynamic-form/models/dynamic-form.models.mjs +2 -0
  21. package/esm2022/lib/components/export-dialog/export-dialog.component.mjs +219 -0
  22. package/esm2022/lib/config/translation.config.mjs +70 -0
  23. package/esm2022/lib/directives/cep-mask.directive.mjs +79 -0
  24. package/esm2022/lib/directives/document-mask.directive.mjs +62 -0
  25. package/esm2022/lib/directives/phone-mask.directive.mjs +59 -0
  26. package/esm2022/lib/interceptors/api.interceptor.mjs +55 -0
  27. package/esm2022/lib/models/base-entity.interface.mjs +2 -0
  28. package/esm2022/lib/models/entity-list.config.mjs +2 -0
  29. package/esm2022/lib/pipes/translate.pipe.mjs +74 -0
  30. package/esm2022/lib/services/auth.service.mjs +169 -0
  31. package/esm2022/lib/services/cookie.service.mjs +90 -0
  32. package/esm2022/lib/services/entity.service.mjs +208 -0
  33. package/esm2022/lib/services/mask.service.mjs +121 -0
  34. package/esm2022/lib/services/permission.service.mjs +180 -0
  35. package/esm2022/lib/services/senior-token.service.mjs +209 -0
  36. package/esm2022/lib/services/theme.service.mjs +85 -0
  37. package/esm2022/lib/services/translation.service.mjs +232 -0
  38. package/esm2022/public-api.mjs +90 -0
  39. package/esm2022/seniorsistemas-components-ai.mjs +5 -0
  40. package/fesm2022/seniorsistemas-components-ai.mjs +4006 -0
  41. package/fesm2022/seniorsistemas-components-ai.mjs.map +1 -0
  42. package/index.d.ts +5 -0
  43. package/lib/angular-components.module.d.ts +13 -0
  44. package/lib/components/breadcrumb/breadcrumb.component.d.ts +23 -0
  45. package/lib/components/bulk-delete-dialog/bulk-delete-dialog.component.d.ts +46 -0
  46. package/lib/components/dynamic-form/dynamic-form.component.d.ts +97 -0
  47. package/lib/components/dynamic-form/fields/dynamic-field-date.component.d.ts +16 -0
  48. package/lib/components/dynamic-form/fields/dynamic-field-dropdown.component.d.ts +17 -0
  49. package/lib/components/dynamic-form/fields/dynamic-field-lookup.component.d.ts +52 -0
  50. package/lib/components/dynamic-form/fields/dynamic-field-number.component.d.ts +16 -0
  51. package/lib/components/dynamic-form/fields/dynamic-field-text.component.d.ts +17 -0
  52. package/lib/components/dynamic-form/fields/dynamic-field-textarea.component.d.ts +16 -0
  53. package/lib/components/dynamic-form/fields/dynamic-field-wrapper.component.d.ts +20 -0
  54. package/lib/components/dynamic-form/fields/index.d.ts +7 -0
  55. package/lib/components/dynamic-form/models/dynamic-form.models.d.ts +178 -0
  56. package/lib/components/export-dialog/export-dialog.component.d.ts +56 -0
  57. package/lib/config/translation.config.d.ts +24 -0
  58. package/lib/directives/cep-mask.directive.d.ts +13 -0
  59. package/lib/directives/document-mask.directive.d.ts +19 -0
  60. package/lib/directives/phone-mask.directive.d.ts +11 -0
  61. package/lib/interceptors/api.interceptor.d.ts +2 -0
  62. package/lib/models/base-entity.interface.d.ts +7 -0
  63. package/lib/models/entity-list.config.d.ts +161 -0
  64. package/lib/pipes/translate.pipe.d.ts +21 -0
  65. package/lib/services/auth.service.d.ts +82 -0
  66. package/lib/services/cookie.service.d.ts +31 -0
  67. package/lib/services/entity.service.d.ts +99 -0
  68. package/lib/services/mask.service.d.ts +36 -0
  69. package/lib/services/permission.service.d.ts +91 -0
  70. package/lib/services/senior-token.service.d.ts +70 -0
  71. package/lib/services/theme.service.d.ts +16 -0
  72. package/lib/services/translation.service.d.ts +54 -0
  73. package/package.json +53 -0
  74. package/public-api.d.ts +17 -0
  75. package/src/lib/styles/entity-list.shared.scss +383 -0
  76. package/src/lib/styles/index.scss +10 -0
@@ -0,0 +1,367 @@
1
+ # Guia de Migração
2
+
3
+ ## 📦 Migrando para @seniorsistemas/components-ia
4
+
5
+ Este guia ajuda na migração de projetos que usavam componentes antigos ou outras bibliotecas.
6
+
7
+ ---
8
+
9
+ ## 🔄 Migração de @senior-gestao-relacionamento/angular-components
10
+
11
+ Se você está migrando do pacote antigo `@senior-gestao-relacionamento/angular-components`:
12
+
13
+ ### 1. Atualizar Dependência
14
+
15
+ ```bash
16
+ # Remover pacote antigo
17
+ npm uninstall @senior-gestao-relacionamento/angular-components
18
+
19
+ # Instalar novo pacote
20
+ npm install @seniorsistemas/components-ia
21
+ ```
22
+
23
+ ### 2. Atualizar Imports
24
+
25
+ Substituir todos os imports:
26
+
27
+ ```typescript
28
+ // Antes
29
+ import { DynamicFormComponent } from '@senior-gestao-relacionamento/angular-components';
30
+
31
+ // Depois
32
+ import { DynamicFormComponent } from '@seniorsistemas/components-ia';
33
+ ```
34
+
35
+ **Script para substituição em massa (Linux/Mac):**
36
+ ```bash
37
+ find src -type f -name "*.ts" -exec sed -i 's/@senior-gestao-relacionamento\/angular-components/@seniorsistemas\/components-ia/g' {} +
38
+ ```
39
+
40
+ **Script para substituição em massa (Windows PowerShell):**
41
+ ```powershell
42
+ Get-ChildItem -Path src -Filter *.ts -Recurse | ForEach-Object {
43
+ (Get-Content $_.FullName) -replace '@senior-gestao-relacionamento/angular-components', '@seniorsistemas/components-ia' |
44
+ Set-Content $_.FullName
45
+ }
46
+ ```
47
+
48
+ ### 3. Atualizar Prefixos de Componentes
49
+
50
+ Os componentes agora usam o prefixo `sia-` ao invés de `sgr-`:
51
+
52
+ ```html
53
+ <!-- Antes -->
54
+ <sgr-dynamic-form></sgr-dynamic-form>
55
+ <sgr-export-dialog></sgr-export-dialog>
56
+ <sgr-breadcrumb></sgr-breadcrumb>
57
+
58
+ <!-- Depois -->
59
+ <sia-dynamic-form></sia-dynamic-form>
60
+ <sia-export-dialog></sia-export-dialog>
61
+ <sia-breadcrumb></sia-breadcrumb>
62
+ ```
63
+
64
+ **Script para substituição em massa (Linux/Mac):**
65
+ ```bash
66
+ find src -type f -name "*.html" -exec sed -i 's/<sgr-/<sia-/g' {} +
67
+ find src -type f -name "*.html" -exec sed -i 's/<\/sgr-/<\/sia-/g' {} +
68
+ ```
69
+
70
+ **Script para substituição em massa (Windows PowerShell):**
71
+ ```powershell
72
+ Get-ChildItem -Path src -Filter *.html -Recurse | ForEach-Object {
73
+ (Get-Content $_.FullName) -replace '<sgr-', '<sia-' -replace '</sgr-', '</sia-' |
74
+ Set-Content $_.FullName
75
+ }
76
+ ```
77
+
78
+ ### 4. Recompilar
79
+
80
+ ```bash
81
+ npm run build
82
+ ```
83
+
84
+ ---
85
+
86
+ ## 🔄 Migração de Componentes Legados
87
+
88
+ ### De EntityFormDialogComponent para DynamicFormComponent
89
+
90
+ **Antes:**
91
+ ```typescript
92
+ <app-entity-form-dialog
93
+ [(visible)]="showDialog"
94
+ [entity]="selectedEntity"
95
+ [formFields]="formFields"
96
+ [service]="entityService"
97
+ [entityName]="'Product'"
98
+ (onSave)="loadData()"
99
+ ></app-entity-form-dialog>
100
+ ```
101
+
102
+ **Depois:**
103
+ ```typescript
104
+ <sia-dynamic-form
105
+ [(visible)]="showDialog"
106
+ [entity]="selectedEntity"
107
+ [formFields]="formFields"
108
+ [mode]="selectedEntity ? 'edit' : 'create'"
109
+ [displayMode]="'dialog'"
110
+ [dialogHeader]="selectedEntity ? 'Editar Produto' : 'Novo Produto'"
111
+ (onSubmit)="handleSubmit($event)"
112
+ ></sia-dynamic-form>
113
+ ```
114
+
115
+ **Mudanças no TypeScript:**
116
+ ```typescript
117
+ // Antes
118
+ onSave(): void {
119
+ this.loadData();
120
+ }
121
+
122
+ // Depois
123
+ handleSubmit(formValue: any): void {
124
+ const operation = this.selectedEntity
125
+ ? this.entityService.update(this.selectedEntity.id, formValue)
126
+ : this.entityService.insert(formValue);
127
+
128
+ operation.subscribe({
129
+ next: () => {
130
+ this.messageService.add({
131
+ severity: 'success',
132
+ summary: 'Sucesso',
133
+ detail: 'Registro salvo com sucesso'
134
+ });
135
+ this.showDialog = false;
136
+ this.loadData();
137
+ },
138
+ error: (error: any) => {
139
+ this.messageService.add({
140
+ severity: 'error',
141
+ summary: 'Erro',
142
+ detail: 'Erro ao salvar registro'
143
+ });
144
+ }
145
+ });
146
+ }
147
+ ```
148
+
149
+ ---
150
+
151
+ ### De FormFilterComponent para DynamicFormComponent
152
+
153
+ **Antes:**
154
+ ```typescript
155
+ <app-form-filter
156
+ [filterFields]="filterFields"
157
+ (onFilter)="applyFilter($event)"
158
+ ></app-form-filter>
159
+ ```
160
+
161
+ **Depois:**
162
+ ```typescript
163
+ <sia-dynamic-form
164
+ [formFields]="filterFields"
165
+ [mode]="'filter'"
166
+ [submitButtonLabel]="'Filtrar'"
167
+ (onSubmit)="applyFilter($event)"
168
+ ></sia-dynamic-form>
169
+ ```
170
+
171
+ **Mudanças no TypeScript:**
172
+ ```typescript
173
+ // Antes
174
+ applyFilter(filters: any): void {
175
+ this.filters = filters;
176
+ this.loadData();
177
+ }
178
+
179
+ // Depois
180
+ applyFilter(filterValues: any): void {
181
+ // Remover valores vazios
182
+ const filters = Object.entries(filterValues)
183
+ .filter(([_, value]) => value !== null && value !== undefined && value !== '')
184
+ .reduce((acc, [key, value]) => ({ ...acc, [key]: value }), {});
185
+
186
+ this.filters = filters;
187
+ this.loadData();
188
+ }
189
+ ```
190
+
191
+ ---
192
+
193
+ ### De EditInlineComponent para DynamicFormComponent
194
+
195
+ **Antes:**
196
+ ```typescript
197
+ <app-edit-inline
198
+ [fields]="editFields"
199
+ [entity]="entity"
200
+ (onSave)="handleSave($event)"
201
+ ></app-edit-inline>
202
+ ```
203
+
204
+ **Depois:**
205
+ ```typescript
206
+ <sia-dynamic-form
207
+ [formFields]="editFields"
208
+ [entity]="entity"
209
+ [mode]="'inline-edit'"
210
+ (onSubmit)="handleSave($event)"
211
+ ></sia-dynamic-form>
212
+ ```
213
+
214
+ ---
215
+
216
+ ## 🆕 Novos Recursos
217
+
218
+ ### Suporte a Seções
219
+
220
+ O DynamicFormComponent agora suporta seções colapsáveis:
221
+
222
+ ```typescript
223
+ formSections: DynamicFormSection[] = [
224
+ {
225
+ title: 'Dados Básicos',
226
+ fields: [/* campos */]
227
+ },
228
+ {
229
+ title: 'Endereço',
230
+ collapsible: true,
231
+ collapsed: true,
232
+ fields: [/* campos */]
233
+ }
234
+ ];
235
+ ```
236
+
237
+ ```html
238
+ <sia-dynamic-form
239
+ [formSections]="formSections"
240
+ [mode]="'create'"
241
+ (onSubmit)="handleSubmit($event)"
242
+ ></sia-dynamic-form>
243
+ ```
244
+
245
+ ### Modo Inline Edit
246
+
247
+ Novo modo para edição inline sem botões de ação:
248
+
249
+ ```typescript
250
+ <sia-dynamic-form
251
+ [formFields]="fields"
252
+ [entity]="entity"
253
+ [mode]="'inline-edit'"
254
+ (onSubmit)="handleSubmit($event)"
255
+ ></sia-dynamic-form>
256
+ ```
257
+
258
+ ### Display Modes
259
+
260
+ Três modos de exibição:
261
+
262
+ - `inline`: Formulário inline (padrão)
263
+ - `dialog`: Formulário em dialog modal
264
+ - `drawer`: Formulário em drawer lateral
265
+
266
+ ---
267
+
268
+ ## ⚠️ Breaking Changes
269
+
270
+ ### 1. Método create() renomeado para insert()
271
+
272
+ ```typescript
273
+ // Antes
274
+ this.entityService.create(entity);
275
+
276
+ // Depois
277
+ this.entityService.insert(entity);
278
+ ```
279
+
280
+ ### 2. Evento onSave renomeado para onSubmit
281
+
282
+ ```typescript
283
+ // Antes
284
+ (onSave)="handleSave($event)"
285
+
286
+ // Depois
287
+ (onSubmit)="handleSubmit($event)"
288
+ ```
289
+
290
+ ### 3. Prefixo de componentes alterado
291
+
292
+ Todos os componentes agora usam `sia-` ao invés de `sgr-`.
293
+
294
+ ### 4. Estrutura de FormFields
295
+
296
+ O campo `cols` agora é obrigatório para controle de layout:
297
+
298
+ ```typescript
299
+ // Antes
300
+ {
301
+ name: 'name',
302
+ label: 'Nome',
303
+ type: FieldType.TEXT
304
+ }
305
+
306
+ // Depois
307
+ {
308
+ name: 'name',
309
+ label: 'Nome',
310
+ type: FieldType.TEXT,
311
+ cols: 12 // 1-12 (grid de 12 colunas)
312
+ }
313
+ ```
314
+
315
+ ---
316
+
317
+ ## 📋 Checklist de Migração
318
+
319
+ - [ ] Atualizar dependência no package.json
320
+ - [ ] Substituir imports em arquivos TypeScript
321
+ - [ ] Atualizar prefixos de componentes em templates HTML
322
+ - [ ] Substituir `create()` por `insert()` nos serviços
323
+ - [ ] Atualizar eventos `onSave` para `onSubmit`
324
+ - [ ] Adicionar propriedade `cols` nos formFields
325
+ - [ ] Testar compilação
326
+ - [ ] Testar funcionalidades em desenvolvimento
327
+ - [ ] Executar testes automatizados
328
+ - [ ] Atualizar documentação do projeto
329
+
330
+ ---
331
+
332
+ ## 🆘 Problemas Comuns
333
+
334
+ ### Erro: Module not found '@senior-gestao-relacionamento/angular-components'
335
+
336
+ **Solução:** Certifique-se de ter atualizado todos os imports. Use os scripts de substituição em massa fornecidos acima.
337
+
338
+ ### Erro: Property 'create' does not exist on type 'EntityService'
339
+
340
+ **Solução:** Substitua `create()` por `insert()`.
341
+
342
+ ### Componentes não aparecem no template
343
+
344
+ **Solução:** Verifique se atualizou os prefixos de `sgr-` para `sia-`.
345
+
346
+ ### Erro de compilação em formFields
347
+
348
+ **Solução:** Adicione a propriedade `cols` em todos os campos.
349
+
350
+ ---
351
+
352
+ ## 📞 Suporte
353
+
354
+ Se encontrar problemas durante a migração:
355
+
356
+ 1. Consulte a [documentação completa](./COMPONENTS.md)
357
+ 2. Veja os [exemplos práticos](./EXAMPLES.md)
358
+ 3. Abra uma issue no repositório
359
+ 4. Entre em contato com a equipe de desenvolvimento
360
+
361
+ ---
362
+
363
+ ## 🔗 Veja Também
364
+
365
+ - [Documentação de Componentes](./COMPONENTS.md)
366
+ - [Exemplos Práticos](./EXAMPLES.md)
367
+ - [Referência da API](./API.md)
@@ -0,0 +1,25 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Módulo principal da biblioteca @seniorsistemas/components-ia
6
+ *
7
+ * Este módulo será populado conforme os componentes forem migrados.
8
+ * Por enquanto, é um módulo vazio para permitir o build inicial.
9
+ */
10
+ export class AngularComponentsModule {
11
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AngularComponentsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
12
+ static ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.14", ngImport: i0, type: AngularComponentsModule, imports: [CommonModule] });
13
+ static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AngularComponentsModule, imports: [CommonModule] });
14
+ }
15
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AngularComponentsModule, decorators: [{
16
+ type: NgModule,
17
+ args: [{
18
+ declarations: [],
19
+ imports: [
20
+ CommonModule
21
+ ],
22
+ exports: []
23
+ }]
24
+ }] });
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYW5ndWxhci1jb21wb25lbnRzLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMtYWkvc3JjL2xpYi9hbmd1bGFyLWNvbXBvbmVudHMubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDOztBQUUvQzs7Ozs7R0FLRztBQVFILE1BQU0sT0FBTyx1QkFBdUI7d0dBQXZCLHVCQUF1Qjt5R0FBdkIsdUJBQXVCLFlBSmhDLFlBQVk7eUdBSUgsdUJBQXVCLFlBSmhDLFlBQVk7OzRGQUlILHVCQUF1QjtrQkFQbkMsUUFBUTttQkFBQztvQkFDUixZQUFZLEVBQUUsRUFBRTtvQkFDaEIsT0FBTyxFQUFFO3dCQUNQLFlBQVk7cUJBQ2I7b0JBQ0QsT0FBTyxFQUFFLEVBQUU7aUJBQ1oiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcblxuLyoqXG4gKiBNw7NkdWxvIHByaW5jaXBhbCBkYSBiaWJsaW90ZWNhIEBzZW5pb3JzaXN0ZW1hcy9jb21wb25lbnRzLWlhXG4gKiBcbiAqIEVzdGUgbcOzZHVsbyBzZXLDoSBwb3B1bGFkbyBjb25mb3JtZSBvcyBjb21wb25lbnRlcyBmb3JlbSBtaWdyYWRvcy5cbiAqIFBvciBlbnF1YW50bywgw6kgdW0gbcOzZHVsbyB2YXppbyBwYXJhIHBlcm1pdGlyIG8gYnVpbGQgaW5pY2lhbC5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZVxuICBdLFxuICBleHBvcnRzOiBbXVxufSlcbmV4cG9ydCBjbGFzcyBBbmd1bGFyQ29tcG9uZW50c01vZHVsZSB7IH1cbiJdfQ==
@@ -0,0 +1,121 @@
1
+ import { Component, Optional, Inject, Input } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { RouterModule, NavigationEnd } from '@angular/router';
4
+ import { BreadcrumbModule } from 'primeng/breadcrumb';
5
+ import { filter } from 'rxjs/operators';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/router";
8
+ import * as i2 from "primeng/breadcrumb";
9
+ export class BreadcrumbComponent {
10
+ router;
11
+ activatedRoute;
12
+ translationService;
13
+ translationPrefix = 'crmx.business';
14
+ isProduction = false;
15
+ breadcrumbItems = [];
16
+ constructor(router, activatedRoute, translationService) {
17
+ this.router = router;
18
+ this.activatedRoute = activatedRoute;
19
+ this.translationService = translationService;
20
+ }
21
+ ngOnInit() {
22
+ this.buildBreadcrumbs();
23
+ // Atualiza breadcrumbs quando a navegação muda
24
+ this.router.events
25
+ .pipe(filter(event => event instanceof NavigationEnd))
26
+ .subscribe(() => this.buildBreadcrumbs());
27
+ }
28
+ get homeItem() {
29
+ // Só mostra o link do home em desenvolvimento
30
+ if (!this.isProduction) {
31
+ return {
32
+ icon: 'pi pi-home',
33
+ routerLink: '/main',
34
+ title: this.t('breadcrumb_home')
35
+ };
36
+ }
37
+ // Em produção, o home não tem link (apenas ícone)
38
+ return {
39
+ icon: 'pi pi-home',
40
+ title: this.t('breadcrumb_home'),
41
+ disabled: true
42
+ };
43
+ }
44
+ buildBreadcrumbs() {
45
+ const breadcrumbs = [];
46
+ let currentRoute = this.activatedRoute.root;
47
+ let url = '';
48
+ let lastBreadcrumbLabel = '';
49
+ // Percorre toda a árvore de rotas
50
+ while (currentRoute) {
51
+ // Verifica se há segmentos de URL
52
+ if (currentRoute.snapshot.url.length > 0) {
53
+ const routeURL = currentRoute.snapshot.url.map(segment => segment.path).join('/');
54
+ url += `/${routeURL}`;
55
+ }
56
+ // Verifica se a rota tem dados de breadcrumb
57
+ const breadcrumbData = currentRoute.snapshot.data['breadcrumb'];
58
+ // Adiciona o breadcrumb se existir e não for duplicado
59
+ if (breadcrumbData) {
60
+ const label = this.t(breadcrumbData.title);
61
+ // Só adiciona se não for o mesmo label do anterior (evita duplicação)
62
+ if (label !== lastBreadcrumbLabel) {
63
+ breadcrumbs.push({
64
+ label: label,
65
+ icon: breadcrumbData.icon,
66
+ routerLink: url || undefined
67
+ });
68
+ lastBreadcrumbLabel = label;
69
+ }
70
+ }
71
+ // Move para a próxima rota filha
72
+ currentRoute = currentRoute.firstChild;
73
+ }
74
+ // O último item não deve ter link (está desabilitado)
75
+ if (breadcrumbs.length > 0) {
76
+ const lastItem = breadcrumbs[breadcrumbs.length - 1];
77
+ lastItem.routerLink = undefined;
78
+ lastItem.disabled = true;
79
+ }
80
+ this.breadcrumbItems = breadcrumbs;
81
+ }
82
+ t(key) {
83
+ if (this.translationService) {
84
+ // Se a chave já tem o prefixo, usa diretamente
85
+ if (key.includes('.')) {
86
+ return this.translationService.translate(key);
87
+ }
88
+ // Caso contrário, adiciona o prefixo
89
+ return this.translationService.translate(`${this.translationPrefix}.${key}`);
90
+ }
91
+ return key;
92
+ }
93
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: BreadcrumbComponent, deps: [{ token: i1.Router }, { token: i1.ActivatedRoute }, { token: 'TranslationService', optional: true }], target: i0.ɵɵFactoryTarget.Component });
94
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: BreadcrumbComponent, isStandalone: true, selector: "sia-breadcrumb", inputs: { translationPrefix: "translationPrefix", isProduction: "isProduction" }, ngImport: i0, template: `
95
+ <p-breadcrumb
96
+ [model]="breadcrumbItems"
97
+ [home]="homeItem"
98
+ styleClass="custom-breadcrumb"
99
+ ></p-breadcrumb>
100
+ `, isInline: true, styles: ["::ng-deep .p-breadcrumb{background:transparent;border:none;padding:0!important}::ng-deep .custom-breadcrumb{background:transparent;border:none;padding:0!important;.p-breadcrumb-list{background:transparent;padding:0!important;.p-menuitem{.p-menuitem-link{color:var(--text-secondary);text-decoration:none;font-size:14px;font-weight:500;padding:8px 12px;border-radius:6px;transition:all .2s ease;display:flex;align-items:center;&:hover{background:var(--p-primary-50);color:var(--p-primary-600)}.p-menuitem-icon{margin-right:8px;font-size:16px;display:inline-flex;align-items:center}}&:last-child .p-menuitem-link{color:var(--p-primary-600);font-weight:600;background:var(--p-primary-50)}}.p-breadcrumb-separator{color:var(--text-muted);margin:0 8px;font-size:14px}}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: RouterModule }, { kind: "ngmodule", type: BreadcrumbModule }, { kind: "component", type: i2.Breadcrumb, selector: "p-breadcrumb", inputs: ["model", "style", "styleClass", "home", "homeAriaLabel"], outputs: ["onItemClick"] }] });
101
+ }
102
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: BreadcrumbComponent, decorators: [{
103
+ type: Component,
104
+ args: [{ selector: 'sia-breadcrumb', standalone: true, imports: [CommonModule, RouterModule, BreadcrumbModule], template: `
105
+ <p-breadcrumb
106
+ [model]="breadcrumbItems"
107
+ [home]="homeItem"
108
+ styleClass="custom-breadcrumb"
109
+ ></p-breadcrumb>
110
+ `, styles: ["::ng-deep .p-breadcrumb{background:transparent;border:none;padding:0!important}::ng-deep .custom-breadcrumb{background:transparent;border:none;padding:0!important;.p-breadcrumb-list{background:transparent;padding:0!important;.p-menuitem{.p-menuitem-link{color:var(--text-secondary);text-decoration:none;font-size:14px;font-weight:500;padding:8px 12px;border-radius:6px;transition:all .2s ease;display:flex;align-items:center;&:hover{background:var(--p-primary-50);color:var(--p-primary-600)}.p-menuitem-icon{margin-right:8px;font-size:16px;display:inline-flex;align-items:center}}&:last-child .p-menuitem-link{color:var(--p-primary-600);font-weight:600;background:var(--p-primary-50)}}.p-breadcrumb-separator{color:var(--text-muted);margin:0 8px;font-size:14px}}}\n"] }]
111
+ }], ctorParameters: () => [{ type: i1.Router }, { type: i1.ActivatedRoute }, { type: undefined, decorators: [{
112
+ type: Optional
113
+ }, {
114
+ type: Inject,
115
+ args: ['TranslationService']
116
+ }] }], propDecorators: { translationPrefix: [{
117
+ type: Input
118
+ }], isProduction: [{
119
+ type: Input
120
+ }] } });
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"breadcrumb.component.js","sourceRoot":"","sources":["../../../../../projects/components-ai/src/lib/components/breadcrumb/breadcrumb.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, Optional, Inject, Input } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { Router, RouterModule, ActivatedRoute, NavigationEnd } from '@angular/router';\nimport { BreadcrumbModule } from 'primeng/breadcrumb';\nimport { MenuItem } from 'primeng/api';\nimport { filter } from 'rxjs/operators';\n\ninterface BreadcrumbData {\n  title: string;\n  icon?: string;\n}\n\ninterface TranslationService {\n  translate(key: string, params?: any): string;\n}\n\n@Component({\n  selector: 'sia-breadcrumb',\n  standalone: true,\n  imports: [CommonModule, RouterModule, BreadcrumbModule],\n  template: `\n    <p-breadcrumb \n      [model]=\"breadcrumbItems\" \n      [home]=\"homeItem\"\n      styleClass=\"custom-breadcrumb\"\n    ></p-breadcrumb>\n  `,\n  styles: [`\n    ::ng-deep .p-breadcrumb {\n      background: transparent;\n      border: none;\n      padding: 0 !important;\n    }\n    \n    ::ng-deep .custom-breadcrumb {\n      background: transparent;\n      border: none;\n      padding: 0 !important;\n      \n      .p-breadcrumb-list {\n        background: transparent;\n        padding: 0 !important;\n        \n        .p-menuitem {\n          .p-menuitem-link {\n            color: var(--text-secondary);\n            text-decoration: none;\n            font-size: 14px;\n            font-weight: 500;\n            padding: 8px 12px;\n            border-radius: 6px;\n            transition: all 0.2s ease;\n            display: flex;\n            align-items: center;\n            \n            &:hover {\n              background: var(--p-primary-50);\n              color: var(--p-primary-600);\n            }\n            \n            .p-menuitem-icon {\n              margin-right: 8px;\n              font-size: 16px;\n              display: inline-flex;\n              align-items: center;\n            }\n          }\n          \n          &:last-child .p-menuitem-link {\n            color: var(--p-primary-600);\n            font-weight: 600;\n            background: var(--p-primary-50);\n          }\n        }\n        \n        .p-breadcrumb-separator {\n          color: var(--text-muted);\n          margin: 0 8px;\n          font-size: 14px;\n        }\n      }\n    }\n  `]\n})\nexport class BreadcrumbComponent implements OnInit {\n  @Input() translationPrefix: string = 'crmx.business';\n  @Input() isProduction: boolean = false;\n  \n  breadcrumbItems: MenuItem[] = [];\n\n  constructor(\n    private router: Router,\n    private activatedRoute: ActivatedRoute,\n    @Optional() @Inject('TranslationService') private translationService?: TranslationService\n  ) {}\n\n  ngOnInit(): void {\n    this.buildBreadcrumbs();\n    \n    // Atualiza breadcrumbs quando a navegação muda\n    this.router.events\n      .pipe(filter(event => event instanceof NavigationEnd))\n      .subscribe(() => this.buildBreadcrumbs());\n  }\n\n  get homeItem(): MenuItem {\n    // Só mostra o link do home em desenvolvimento\n    if (!this.isProduction) {\n      return {\n        icon: 'pi pi-home',\n        routerLink: '/main',\n        title: this.t('breadcrumb_home')\n      };\n    }\n    \n    // Em produção, o home não tem link (apenas ícone)\n    return {\n      icon: 'pi pi-home',\n      title: this.t('breadcrumb_home'),\n      disabled: true\n    };\n  }\n\n  private buildBreadcrumbs(): void {\n    const breadcrumbs: MenuItem[] = [];\n    let currentRoute: ActivatedRoute | null = this.activatedRoute.root;\n    let url = '';\n    let lastBreadcrumbLabel = '';\n\n    // Percorre toda a árvore de rotas\n    while (currentRoute) {\n      // Verifica se há segmentos de URL\n      if (currentRoute.snapshot.url.length > 0) {\n        const routeURL = currentRoute.snapshot.url.map(segment => segment.path).join('/');\n        url += `/${routeURL}`;\n      }\n      \n      // Verifica se a rota tem dados de breadcrumb\n      const breadcrumbData = currentRoute.snapshot.data['breadcrumb'] as BreadcrumbData;\n      \n      // Adiciona o breadcrumb se existir e não for duplicado\n      if (breadcrumbData) {\n        const label = this.t(breadcrumbData.title);\n        \n        // Só adiciona se não for o mesmo label do anterior (evita duplicação)\n        if (label !== lastBreadcrumbLabel) {\n          breadcrumbs.push({\n            label: label,\n            icon: breadcrumbData.icon,\n            routerLink: url || undefined\n          });\n          lastBreadcrumbLabel = label;\n        }\n      }\n\n      // Move para a próxima rota filha\n      currentRoute = currentRoute.firstChild;\n    }\n\n    // O último item não deve ter link (está desabilitado)\n    if (breadcrumbs.length > 0) {\n      const lastItem = breadcrumbs[breadcrumbs.length - 1];\n      lastItem.routerLink = undefined;\n      lastItem.disabled = true;\n    }\n\n    this.breadcrumbItems = breadcrumbs;\n  }\n\n  private t(key: string): string {\n    if (this.translationService) {\n      // Se a chave já tem o prefixo, usa diretamente\n      if (key.includes('.')) {\n        return this.translationService.translate(key);\n      }\n      // Caso contrário, adiciona o prefixo\n      return this.translationService.translate(`${this.translationPrefix}.${key}`);\n    }\n    return key;\n  }\n}\n"]}