@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,625 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { PanelModule } from 'primeng/panel';
5
+ import { DialogModule } from 'primeng/dialog';
6
+ import { DrawerModule } from 'primeng/drawer';
7
+ import { ButtonModule } from 'primeng/button';
8
+ import { TranslatePipe } from '../../pipes/translate.pipe';
9
+ import { DynamicFieldTextComponent } from './fields/dynamic-field-text.component';
10
+ import { DynamicFieldNumberComponent } from './fields/dynamic-field-number.component';
11
+ import { DynamicFieldDateComponent } from './fields/dynamic-field-date.component';
12
+ import { DynamicFieldDropdownComponent } from './fields/dynamic-field-dropdown.component';
13
+ import { DynamicFieldLookupComponent } from './fields/dynamic-field-lookup.component';
14
+ import { DynamicFieldTextareaComponent } from './fields/dynamic-field-textarea.component';
15
+ import * as i0 from "@angular/core";
16
+ import * as i1 from "@angular/forms";
17
+ import * as i2 from "../../services/translation.service";
18
+ import * as i3 from "@angular/common";
19
+ import * as i4 from "primeng/panel";
20
+ import * as i5 from "primeng/api";
21
+ import * as i6 from "primeng/dialog";
22
+ import * as i7 from "primeng/drawer";
23
+ import * as i8 from "primeng/button";
24
+ export class DynamicFormComponent {
25
+ fb;
26
+ translationService;
27
+ formFields = [];
28
+ formSections = [];
29
+ fixedFilters = []; // Filtros fixos (sempre visíveis no topo)
30
+ filterSections = []; // Seções de filtros (colapsáveis)
31
+ entityData = null;
32
+ mode = 'form';
33
+ // Display Mode Configuration
34
+ displayMode = 'inline';
35
+ visible = false;
36
+ dialogConfig = {};
37
+ drawerConfig = {};
38
+ // Button Configuration
39
+ showSubmitButton = false;
40
+ showCancelButton = false;
41
+ submitButtonLabel;
42
+ cancelButtonLabel;
43
+ submitButtonIcon = 'pi pi-check';
44
+ cancelButtonIcon = 'pi pi-times';
45
+ formReady = new EventEmitter();
46
+ formSubmit = new EventEmitter();
47
+ visibleChange = new EventEmitter();
48
+ onCancel = new EventEmitter();
49
+ fieldSave = new EventEmitter();
50
+ form;
51
+ loading = false;
52
+ // Inline edit mode state
53
+ editingField = null;
54
+ savingField = false;
55
+ get hasFields() {
56
+ return this.formFields.length > 0;
57
+ }
58
+ get hasSections() {
59
+ return this.formSections.length > 0;
60
+ }
61
+ get hasFixedFilters() {
62
+ return this.fixedFilters.length > 0;
63
+ }
64
+ get hasFilterSections() {
65
+ return this.filterSections.length > 0;
66
+ }
67
+ get allFields() {
68
+ // Retorna todos os campos (diretos + das seções + filtros fixos + seções de filtros)
69
+ const sectionFields = this.formSections.flatMap(section => section.fields);
70
+ const filterSectionFields = this.filterSections.flatMap(section => section.fields);
71
+ return [...this.formFields, ...sectionFields, ...this.fixedFilters, ...filterSectionFields];
72
+ }
73
+ get isInlineMode() {
74
+ return this.displayMode === 'inline';
75
+ }
76
+ get isInlineEditMode() {
77
+ return this.mode === 'inline-edit';
78
+ }
79
+ get fieldMode() {
80
+ return this.isInlineEditMode ? 'form' : this.mode;
81
+ }
82
+ get isDialogMode() {
83
+ return this.displayMode === 'dialog';
84
+ }
85
+ get isDrawerMode() {
86
+ return this.displayMode === 'drawer';
87
+ }
88
+ // Dialog configuration with defaults
89
+ get dialogHeader() {
90
+ return this.dialogConfig.header || this.translationService.translate('crmx.business.form');
91
+ }
92
+ get dialogWidth() {
93
+ return this.dialogConfig.width || '600px';
94
+ }
95
+ get dialogDraggable() {
96
+ return this.dialogConfig.draggable !== undefined ? this.dialogConfig.draggable : false;
97
+ }
98
+ get dialogResizable() {
99
+ return this.dialogConfig.resizable !== undefined ? this.dialogConfig.resizable : false;
100
+ }
101
+ get dialogMaximizable() {
102
+ return this.dialogConfig.maximizable !== undefined ? this.dialogConfig.maximizable : false;
103
+ }
104
+ get dialogCloseOnEscape() {
105
+ return this.dialogConfig.closeOnEscape !== undefined ? this.dialogConfig.closeOnEscape : true;
106
+ }
107
+ get dialogDismissableMask() {
108
+ return this.dialogConfig.dismissableMask !== undefined ? this.dialogConfig.dismissableMask : false;
109
+ }
110
+ // Drawer configuration with defaults
111
+ get drawerHeader() {
112
+ return this.drawerConfig.header || this.translationService.translate('crmx.business.form');
113
+ }
114
+ get drawerPosition() {
115
+ return this.drawerConfig.position || 'right';
116
+ }
117
+ get drawerShowCloseIcon() {
118
+ return this.drawerConfig.showCloseIcon !== undefined ? this.drawerConfig.showCloseIcon : true;
119
+ }
120
+ get drawerCloseOnEscape() {
121
+ return this.drawerConfig.closeOnEscape !== undefined ? this.drawerConfig.closeOnEscape : true;
122
+ }
123
+ get drawerDismissable() {
124
+ return this.drawerConfig.dismissable !== undefined ? this.drawerConfig.dismissable : true;
125
+ }
126
+ get drawerStyle() {
127
+ const position = this.drawerPosition;
128
+ const isHorizontal = position === 'left' || position === 'right';
129
+ // Se tem customWidth, usa ela
130
+ if (this.drawerConfig.customWidth) {
131
+ return isHorizontal
132
+ ? { width: this.drawerConfig.customWidth }
133
+ : { height: this.drawerConfig.customWidth };
134
+ }
135
+ // Usa o tamanho padronizado
136
+ const size = this.drawerConfig.size || 'medium';
137
+ const sizeMap = {
138
+ 'small': isHorizontal ? '25%' : '25vh',
139
+ 'medium': isHorizontal ? '40%' : '40vh',
140
+ 'large': isHorizontal ? '60%' : '60vh',
141
+ 'xlarge': isHorizontal ? '80%' : '80vh',
142
+ 'full': isHorizontal ? '100%' : '100vh'
143
+ };
144
+ const sizeValue = sizeMap[size];
145
+ return isHorizontal
146
+ ? { width: sizeValue }
147
+ : { height: sizeValue };
148
+ }
149
+ get submitLabel() {
150
+ return this.submitButtonLabel || this.translationService.translate('crmx.business.save');
151
+ }
152
+ get cancelLabel() {
153
+ return this.cancelButtonLabel || this.translationService.translate('crmx.business.cancel');
154
+ }
155
+ constructor(fb, translationService) {
156
+ this.fb = fb;
157
+ this.translationService = translationService;
158
+ }
159
+ ngOnInit() {
160
+ this.initForm();
161
+ this.setupConditionalFieldsWatcher();
162
+ this.formReady.emit(this.form);
163
+ }
164
+ ngOnChanges(changes) {
165
+ if (changes['entityData'] && this.form) {
166
+ if (this.entityData) {
167
+ this.loadEntityData();
168
+ }
169
+ else {
170
+ this.resetForm();
171
+ }
172
+ }
173
+ // Reset loading and form when dialog/drawer is opened
174
+ if (changes['visible'] && changes['visible'].currentValue === true) {
175
+ this.loading = false;
176
+ // If opening without entityData (new record), reset form
177
+ if (!this.entityData && this.form) {
178
+ this.resetForm();
179
+ }
180
+ }
181
+ }
182
+ initForm() {
183
+ const formControls = {};
184
+ // Processa todos os campos (diretos + das seções)
185
+ this.allFields.forEach(field => {
186
+ const defaultValue = this.getDefaultValue(field);
187
+ const validators = this.mode === 'filter' ? [] : (field.validators || []);
188
+ const control = this.fb.control({ value: defaultValue, disabled: field.disabled || false }, validators);
189
+ formControls[field.field] = control;
190
+ });
191
+ this.form = this.fb.group(formControls);
192
+ this.applyInitialDisabledState();
193
+ }
194
+ getDefaultValue(field) {
195
+ if (field.defaultValue !== undefined) {
196
+ return field.defaultValue;
197
+ }
198
+ switch (field.type) {
199
+ case 'number':
200
+ case 'date':
201
+ return null;
202
+ case 'dropdown':
203
+ case 'enum':
204
+ return this.mode === 'filter' ? null : '';
205
+ default:
206
+ return '';
207
+ }
208
+ }
209
+ applyInitialDisabledState() {
210
+ this.allFields.forEach(field => {
211
+ if (field.disabledWhen) {
212
+ const control = this.form.get(field.field);
213
+ if (control && this.isFieldDisabled(field)) {
214
+ control.disable();
215
+ }
216
+ }
217
+ });
218
+ }
219
+ loadEntityData() {
220
+ if (this.entityData && this.form) {
221
+ const processedData = this.processEntityData(this.entityData);
222
+ this.form.patchValue(processedData);
223
+ }
224
+ }
225
+ processEntityData(entity) {
226
+ const processedData = { ...entity };
227
+ this.allFields.forEach(field => {
228
+ if (field.type === 'lookup' && processedData[field.field]) {
229
+ const lookupValue = processedData[field.field];
230
+ // Keep the full object if it has display data, otherwise extract ID
231
+ if (typeof lookupValue === 'object' && lookupValue.id) {
232
+ const hasDisplayData = field.lookupDisplayFields
233
+ ? field.lookupDisplayFields.some(f => this.getNestedValue(lookupValue, f))
234
+ : lookupValue[field.lookupDisplayField || 'name'];
235
+ if (hasDisplayData) {
236
+ // Keep the full object to avoid unnecessary API calls
237
+ processedData[field.field] = lookupValue;
238
+ }
239
+ else {
240
+ // Only has ID, extract it
241
+ processedData[field.field] = lookupValue.id;
242
+ }
243
+ }
244
+ }
245
+ });
246
+ return processedData;
247
+ }
248
+ getNestedValue(obj, path) {
249
+ return path.split('.').reduce((current, key) => current?.[key], obj);
250
+ }
251
+ resetForm() {
252
+ if (this.form) {
253
+ const resetValues = {};
254
+ this.allFields.forEach(field => {
255
+ resetValues[field.field] = this.getDefaultValue(field);
256
+ });
257
+ this.form.reset(resetValues);
258
+ }
259
+ }
260
+ processFilterValue(formValue) {
261
+ const processedValue = { ...formValue };
262
+ this.allFields.forEach(field => {
263
+ const fieldValue = processedValue[field.field];
264
+ if (field.type === 'lookup') {
265
+ if (fieldValue && typeof fieldValue === 'object' && fieldValue.id) {
266
+ processedValue[field.field] = fieldValue;
267
+ }
268
+ else if (fieldValue && typeof fieldValue === 'string') {
269
+ processedValue[field.field] = { id: fieldValue };
270
+ }
271
+ else {
272
+ processedValue[field.field] = null;
273
+ }
274
+ }
275
+ });
276
+ return processedValue;
277
+ }
278
+ processFormValue(formValue) {
279
+ const processedValue = { ...formValue };
280
+ if (this.entityData?.id) {
281
+ processedValue.id = this.entityData.id;
282
+ }
283
+ this.allFields.forEach(field => {
284
+ const fieldValue = processedValue[field.field];
285
+ if (field.type === 'lookup') {
286
+ // If it's an object, extract the ID
287
+ if (fieldValue && typeof fieldValue === 'object') {
288
+ processedValue[field.field] = fieldValue.id || fieldValue[field.lookupValueField || 'id'] || null;
289
+ }
290
+ else if (fieldValue && fieldValue !== '') {
291
+ // If it's already an ID, keep it
292
+ processedValue[field.field] = { id: fieldValue };
293
+ }
294
+ else {
295
+ processedValue[field.field] = null;
296
+ }
297
+ }
298
+ else if (field.type === 'date') {
299
+ if (fieldValue instanceof Date) {
300
+ processedValue[field.field] = fieldValue.toISOString().split('T')[0];
301
+ }
302
+ else if (!fieldValue) {
303
+ processedValue[field.field] = null;
304
+ }
305
+ }
306
+ else if (['text', 'textarea', 'number', 'enum', 'dropdown'].includes(field.type)) {
307
+ if (fieldValue === '' || fieldValue === undefined) {
308
+ processedValue[field.field] = null;
309
+ }
310
+ }
311
+ });
312
+ return processedValue;
313
+ }
314
+ getFieldGridClass(field) {
315
+ const classes = [];
316
+ // Colspan base (padrão: 6)
317
+ let baseColspan = field.colspan || 6;
318
+ // Verifica condições de colspan dinâmico
319
+ if (field.colspanWhen && field.colspanWhen.length > 0) {
320
+ for (const condition of field.colspanWhen) {
321
+ const dependentFieldValue = this.form.get(condition.field)?.value;
322
+ if (dependentFieldValue === condition.value) {
323
+ baseColspan = condition.colspan;
324
+ break;
325
+ }
326
+ }
327
+ }
328
+ // Adiciona classe base
329
+ classes.push(`col-${baseColspan}`);
330
+ // Suporta tanto o objeto size quanto as propriedades antigas (para compatibilidade)
331
+ const sizeXs = field.size?.xs ?? field.colspanXs;
332
+ const sizeSm = field.size?.sm ?? field.colspanSm;
333
+ const sizeMd = field.size?.md ?? field.colspanMd;
334
+ const sizeLg = field.size?.lg ?? field.colspanLg;
335
+ const sizeXl = field.size?.xl ?? field.colspanXl;
336
+ // Adiciona classes responsivas se definidas
337
+ if (sizeXs !== undefined) {
338
+ classes.push(`col-xs-${sizeXs}`);
339
+ }
340
+ if (sizeSm !== undefined) {
341
+ classes.push(`col-sm-${sizeSm}`);
342
+ }
343
+ if (sizeMd !== undefined) {
344
+ classes.push(`col-md-${sizeMd}`);
345
+ }
346
+ if (sizeLg !== undefined) {
347
+ classes.push(`col-lg-${sizeLg}`);
348
+ }
349
+ if (sizeXl !== undefined) {
350
+ classes.push(`col-xl-${sizeXl}`);
351
+ }
352
+ return classes.join(' ');
353
+ }
354
+ isFieldVisible(field) {
355
+ if (this.mode === 'filter') {
356
+ return true;
357
+ }
358
+ if (!field.visibleWhen) {
359
+ return true;
360
+ }
361
+ const dependentFieldValue = this.form.get(field.visibleWhen.field)?.value;
362
+ return dependentFieldValue === field.visibleWhen.value;
363
+ }
364
+ isFieldDisabled(field) {
365
+ if (field.disabled) {
366
+ return true;
367
+ }
368
+ if (!('disabledWhen' in field) || !field.disabledWhen) {
369
+ return false;
370
+ }
371
+ const dependentFieldValue = this.form.get(field.disabledWhen.field)?.value;
372
+ if (field.disabledWhen.value === null) {
373
+ return !dependentFieldValue;
374
+ }
375
+ return dependentFieldValue === field.disabledWhen.value;
376
+ }
377
+ setupConditionalFieldsWatcher() {
378
+ const visibleDependentFields = this.allFields.filter(field => field.visibleWhen);
379
+ const visibleWatchedFields = [...new Set(visibleDependentFields.map(field => field.visibleWhen.field))];
380
+ const disabledDependentFields = this.allFields.filter(field => field.disabledWhen);
381
+ const disabledWatchedFields = [...new Set(disabledDependentFields.map(field => field.disabledWhen.field))];
382
+ const allWatchedFields = [...new Set([...visibleWatchedFields, ...disabledWatchedFields])];
383
+ allWatchedFields.forEach(fieldName => {
384
+ const control = this.form.get(fieldName);
385
+ if (control) {
386
+ control.valueChanges.subscribe(() => {
387
+ this.clearHiddenFields();
388
+ this.updateDisabledFields();
389
+ });
390
+ }
391
+ });
392
+ }
393
+ updateDisabledFields() {
394
+ this.allFields.forEach(field => {
395
+ if (field.disabledWhen) {
396
+ const control = this.form.get(field.field);
397
+ if (control) {
398
+ const shouldBeDisabled = this.isFieldDisabled(field);
399
+ if (shouldBeDisabled && control.enabled) {
400
+ control.setValue(null);
401
+ control.disable();
402
+ }
403
+ else if (!shouldBeDisabled && control.disabled && !field.disabled) {
404
+ control.enable();
405
+ }
406
+ }
407
+ }
408
+ });
409
+ }
410
+ clearHiddenFields() {
411
+ this.allFields.forEach(field => {
412
+ if (field.visibleWhen && !this.isFieldVisible(field)) {
413
+ const control = this.form.get(field.field);
414
+ if (control) {
415
+ const resetValue = this.getDefaultValue(field);
416
+ control.setValue(resetValue);
417
+ control.markAsUntouched();
418
+ control.markAsPristine();
419
+ }
420
+ }
421
+ });
422
+ }
423
+ getForm() {
424
+ return this.form;
425
+ }
426
+ clearFilters() {
427
+ if (this.mode === 'filter') {
428
+ this.resetForm();
429
+ }
430
+ }
431
+ getTypePersonValue() {
432
+ const typePersonValue = this.form?.get('typePerson')?.value;
433
+ return typePersonValue || 'JURIDICAL_PERSON';
434
+ }
435
+ // Display Mode Control Methods
436
+ open(entity) {
437
+ if (entity) {
438
+ this.entityData = entity;
439
+ }
440
+ else {
441
+ this.entityData = null;
442
+ this.resetForm(); // Reset form when opening for new entity
443
+ }
444
+ this.loading = false; // Reset loading state
445
+ this.visible = true;
446
+ this.visibleChange.emit(true);
447
+ }
448
+ close() {
449
+ this.loading = false; // Reset loading state
450
+ this.visible = false;
451
+ this.visibleChange.emit(false);
452
+ this.resetForm(); // Reset form when closing
453
+ this.onCancel.emit();
454
+ }
455
+ handleCancel() {
456
+ this.close();
457
+ }
458
+ handleSubmit() {
459
+ if (this.mode === 'form' && this.form.invalid) {
460
+ this.form.markAllAsTouched();
461
+ return;
462
+ }
463
+ this.loading = true;
464
+ const formValue = this.mode === 'filter'
465
+ ? this.processFilterValue(this.form.value)
466
+ : this.processFormValue(this.form.value);
467
+ this.formSubmit.emit(formValue);
468
+ }
469
+ // Inline Edit Mode Methods
470
+ startEditField(fieldName) {
471
+ if (this.isInlineEditMode) {
472
+ this.editingField = fieldName;
473
+ // Ensure the field has the current value from entityData
474
+ if (this.entityData) {
475
+ const control = this.form.get(fieldName);
476
+ if (control) {
477
+ const field = this.allFields.find(f => f.field === fieldName);
478
+ const value = this.entityData[fieldName];
479
+ // For lookup fields in inline-edit mode, keep the full object
480
+ // This prevents unnecessary API calls since we already have the data
481
+ if (field?.type === 'lookup' && value && typeof value === 'object') {
482
+ // Keep the full object - the lookup component will use it
483
+ control.setValue(value);
484
+ }
485
+ else {
486
+ control.setValue(value);
487
+ }
488
+ }
489
+ }
490
+ }
491
+ }
492
+ isFieldEditing(fieldName) {
493
+ return this.editingField === fieldName;
494
+ }
495
+ saveInlineField(fieldName) {
496
+ if (!this.isInlineEditMode)
497
+ return;
498
+ const control = this.form.get(fieldName);
499
+ if (!control || control.invalid) {
500
+ control?.markAsTouched();
501
+ return;
502
+ }
503
+ this.savingField = true;
504
+ let fieldValue = control.value;
505
+ // For lookup fields, extract the ID if it's an object
506
+ const field = this.allFields.find(f => f.field === fieldName);
507
+ if (field?.type === 'lookup' && fieldValue && typeof fieldValue === 'object') {
508
+ fieldValue = fieldValue.id || fieldValue[field.lookupValueField || 'id'];
509
+ }
510
+ this.fieldSave.emit({ field: fieldName, value: fieldValue });
511
+ }
512
+ cancelInlineEdit() {
513
+ if (this.isInlineEditMode && this.editingField) {
514
+ const control = this.form.get(this.editingField);
515
+ if (control && this.entityData) {
516
+ // Restore original value
517
+ const originalValue = this.entityData[this.editingField];
518
+ control.setValue(originalValue);
519
+ control.markAsPristine();
520
+ control.markAsUntouched();
521
+ }
522
+ this.editingField = null;
523
+ this.savingField = false;
524
+ }
525
+ }
526
+ // Method to be called by parent after save completes
527
+ completeInlineEdit() {
528
+ this.editingField = null;
529
+ this.savingField = false;
530
+ }
531
+ getFieldDisplayValue(fieldName) {
532
+ if (!this.entityData)
533
+ return '';
534
+ const field = this.allFields.find(f => f.field === fieldName);
535
+ const value = this.entityData[fieldName];
536
+ if (!value)
537
+ return '';
538
+ // For lookup fields, return the display name
539
+ if (field?.type === 'lookup' && typeof value === 'object' && value.name) {
540
+ return value.name;
541
+ }
542
+ // For number fields, format appropriately
543
+ if (field?.type === 'number' && typeof value === 'number') {
544
+ if (field.numberMode === 'currency') {
545
+ return new Intl.NumberFormat(field.locale || 'pt-BR', {
546
+ style: 'currency',
547
+ currency: field.currency || 'BRL'
548
+ }).format(value);
549
+ }
550
+ return new Intl.NumberFormat(field.locale || 'pt-BR', {
551
+ minimumFractionDigits: field.minFractionDigits || 0,
552
+ maximumFractionDigits: field.maxFractionDigits || 2
553
+ }).format(value);
554
+ }
555
+ // For date fields, format as date
556
+ if (field?.type === 'date' && value) {
557
+ const date = typeof value === 'string' ? new Date(value) : value;
558
+ return date.toLocaleDateString('pt-BR');
559
+ }
560
+ return String(value);
561
+ }
562
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DynamicFormComponent, deps: [{ token: i1.FormBuilder }, { token: i2.TranslationService }], target: i0.ɵɵFactoryTarget.Component });
563
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: DynamicFormComponent, isStandalone: true, selector: "sia-dynamic-form", inputs: { formFields: "formFields", formSections: "formSections", fixedFilters: "fixedFilters", filterSections: "filterSections", entityData: "entityData", mode: "mode", displayMode: "displayMode", visible: "visible", dialogConfig: "dialogConfig", drawerConfig: "drawerConfig", showSubmitButton: "showSubmitButton", showCancelButton: "showCancelButton", submitButtonLabel: "submitButtonLabel", cancelButtonLabel: "cancelButtonLabel", submitButtonIcon: "submitButtonIcon", cancelButtonIcon: "cancelButtonIcon" }, outputs: { formReady: "formReady", formSubmit: "formSubmit", visibleChange: "visibleChange", onCancel: "onCancel", fieldSave: "fieldSave" }, usesOnChanges: true, ngImport: i0, template: "<!-- Modo Inline (padr\u00E3o) -->\n<ng-container *ngIf=\"isInlineMode\">\n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n</ng-container>\n\n<!-- Modo Dialog -->\n<p-dialog\n *ngIf=\"isDialogMode\"\n [(visible)]=\"visible\"\n [header]=\"dialogHeader | translate\"\n [modal]=\"true\"\n [style]=\"{ width: dialogWidth }\"\n [draggable]=\"dialogDraggable\"\n [resizable]=\"dialogResizable\"\n [maximizable]=\"dialogMaximizable\"\n [closeOnEscape]=\"dialogCloseOnEscape\"\n [dismissableMask]=\"dialogDismissableMask\"\n (onHide)=\"close()\">\n \n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n \n <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n <div class=\"dialog-footer\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n </ng-template>\n</p-dialog>\n\n<!-- Modo Drawer -->\n<p-drawer\n *ngIf=\"isDrawerMode\"\n [(visible)]=\"visible\"\n [header]=\"drawerHeader | translate\"\n [position]=\"drawerPosition\"\n [style]=\"drawerStyle\"\n [showCloseIcon]=\"drawerShowCloseIcon\"\n [closeOnEscape]=\"drawerCloseOnEscape\"\n [modal]=\"drawerDismissable\"\n (onHide)=\"close()\">\n \n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n \n <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n <div class=\"drawer-footer\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n </ng-template>\n</p-drawer>\n\n<!-- Template do Formul\u00E1rio (reutiliz\u00E1vel) -->\n<ng-template #formContent>\n <form [formGroup]=\"form\" class=\"dynamic-form\">\n \n <!-- Filtros Fixos (apenas no modo filter) -->\n <div *ngIf=\"mode === 'filter' && hasFixedFilters\" class=\"fixed-filters-section\">\n <div class=\"grid\">\n <ng-container *ngFor=\"let field of fixedFilters\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Campos Diretos (sem se\u00E7\u00E3o) -->\n <div *ngIf=\"hasFields\">\n <!-- Modo Inline Edit -->\n <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n <ng-container *ngFor=\"let field of formFields\">\n <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n\n <!-- Modo Form Normal -->\n <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n <ng-container *ngFor=\"let field of formFields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Se\u00E7\u00F5es de Campos (modo form) -->\n <div *ngIf=\"hasSections\" class=\"sections-container\">\n <p-panel \n *ngFor=\"let section of formSections\"\n [toggleable]=\"section.toggleable !== false\"\n [collapsed]=\"section.collapsed || false\"\n [styleClass]=\"'section-panel'\">\n \n <ng-template pTemplate=\"header\">\n <div class=\"section-header\">\n <i *ngIf=\"section.icon\" [class]=\"section.icon + ' section-icon'\"></i>\n <span class=\"section-title\">{{ section.title }}</span>\n </div>\n </ng-template>\n\n <!-- Modo Inline Edit -->\n <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n\n <!-- Modo Form Normal -->\n <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </p-panel>\n </div>\n\n <!-- Se\u00E7\u00F5es de Filtros (apenas no modo filter) -->\n <div *ngIf=\"mode === 'filter' && hasFilterSections\" class=\"filter-sections-container\">\n <p-panel \n *ngFor=\"let section of filterSections\"\n [header]=\"section.title\"\n [toggleable]=\"true\"\n [collapsed]=\"true\">\n\n <div class=\"grid\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </p-panel>\n </div>\n\n <!-- Bot\u00F5es inline (apenas para modo inline) -->\n <div *ngIf=\"isInlineMode && (showSubmitButton || showCancelButton)\" class=\"form-actions\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n\n </form>\n</ng-template>\n\n<!-- Template de Campo (reutiliz\u00E1vel) -->\n<ng-template #fieldTemplate let-field=\"field\">\n <!-- Text Field -->\n <sia-dynamic-field-text\n *ngIf=\"field.type === 'text'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-text>\n\n <!-- Textarea Field -->\n <sia-dynamic-field-textarea\n *ngIf=\"field.type === 'textarea'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-textarea>\n\n <!-- Number Field -->\n <sia-dynamic-field-number\n *ngIf=\"field.type === 'number'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-number>\n\n <!-- Date Field -->\n <sia-dynamic-field-date\n *ngIf=\"field.type === 'date'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-date>\n\n <!-- Dropdown/Enum Field -->\n <sia-dynamic-field-dropdown\n *ngIf=\"field.type === 'dropdown' || field.type === 'enum'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-dropdown>\n\n <!-- Lookup Field -->\n <sia-dynamic-field-lookup\n *ngIf=\"field.type === 'lookup'\"\n [field]=\"field\"\n [form]=\"form\"\n [formGroup]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-lookup>\n</ng-template>\n\n\n<!-- Template de Campo Inline Edit -->\n<ng-template #inlineEditFieldTemplate let-field=\"field\">\n <div class=\"edit-inline-field\">\n <label class=\"field-label\">{{ field.label | translate }}</label>\n \n <!-- Display Mode -->\n <div class=\"field-display\" \n *ngIf=\"!isFieldEditing(field.field)\" \n (click)=\"startEditField(field.field)\"\n [class.disabled]=\"field.disabled\">\n <span class=\"display-value\" \n [attr.data-placeholder]=\"field.placeholder || ('crmx.business.not_informed' | translate)\">\n {{ getFieldDisplayValue(field.field) }}\n </span>\n <i class=\"pi pi-pencil edit-icon\" *ngIf=\"!field.disabled\"></i>\n </div>\n \n <!-- Edit Mode -->\n <div class=\"field-edit\" *ngIf=\"isFieldEditing(field.field)\">\n <div class=\"input-container\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n \n <!-- Action Buttons -->\n <div class=\"edit-actions\">\n <p-button \n type=\"button\" \n icon=\"pi pi-check\" \n [rounded]=\"true\"\n severity=\"primary\"\n (onClick)=\"saveInlineField(field.field)\" \n [loading]=\"savingField\"\n [disabled]=\"form.get(field.field)?.invalid\">\n </p-button>\n <p-button \n type=\"button\" \n icon=\"pi pi-times\" \n [rounded]=\"true\"\n severity=\"secondary\"\n (onClick)=\"cancelInlineEdit()\"\n [disabled]=\"savingField\">\n </p-button>\n </div>\n \n <!-- Error Message -->\n <div class=\"field-error\" *ngIf=\"form.get(field.field)?.invalid && form.get(field.field)?.touched\">\n <small class=\"p-error\">\n {{ form.get(field.field)?.errors?.['required'] ? ('crmx.business.required_field' | translate) : '' }}\n </small>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [".dynamic-form{width:100%}.dynamic-form .grid{display:grid;grid-template-columns:repeat(12,1fr);gap:1rem}.dynamic-form .grid>div.col-1{grid-column:span 1}.dynamic-form .grid>div.col-2{grid-column:span 2}.dynamic-form .grid>div.col-3{grid-column:span 3}.dynamic-form .grid>div.col-4{grid-column:span 4}.dynamic-form .grid>div.col-5{grid-column:span 5}.dynamic-form .grid>div.col-6{grid-column:span 6}.dynamic-form .grid>div.col-7{grid-column:span 7}.dynamic-form .grid>div.col-8{grid-column:span 8}.dynamic-form .grid>div.col-9{grid-column:span 9}.dynamic-form .grid>div.col-10{grid-column:span 10}.dynamic-form .grid>div.col-11{grid-column:span 11}.dynamic-form .grid>div.col-12{grid-column:span 12}@media (max-width: 575.98px){.dynamic-form .grid>div.col-xs-1{grid-column:span 1}.dynamic-form .grid>div.col-xs-2{grid-column:span 2}.dynamic-form .grid>div.col-xs-3{grid-column:span 3}.dynamic-form .grid>div.col-xs-4{grid-column:span 4}.dynamic-form .grid>div.col-xs-5{grid-column:span 5}.dynamic-form .grid>div.col-xs-6{grid-column:span 6}.dynamic-form .grid>div.col-xs-7{grid-column:span 7}.dynamic-form .grid>div.col-xs-8{grid-column:span 8}.dynamic-form .grid>div.col-xs-9{grid-column:span 9}.dynamic-form .grid>div.col-xs-10{grid-column:span 10}.dynamic-form .grid>div.col-xs-11{grid-column:span 11}.dynamic-form .grid>div.col-xs-12{grid-column:span 12}}@media (min-width: 576px){.dynamic-form .grid>div.col-sm-1{grid-column:span 1}.dynamic-form .grid>div.col-sm-2{grid-column:span 2}.dynamic-form .grid>div.col-sm-3{grid-column:span 3}.dynamic-form .grid>div.col-sm-4{grid-column:span 4}.dynamic-form .grid>div.col-sm-5{grid-column:span 5}.dynamic-form .grid>div.col-sm-6{grid-column:span 6}.dynamic-form .grid>div.col-sm-7{grid-column:span 7}.dynamic-form .grid>div.col-sm-8{grid-column:span 8}.dynamic-form .grid>div.col-sm-9{grid-column:span 9}.dynamic-form .grid>div.col-sm-10{grid-column:span 10}.dynamic-form .grid>div.col-sm-11{grid-column:span 11}.dynamic-form .grid>div.col-sm-12{grid-column:span 12}}@media (min-width: 768px){.dynamic-form .grid>div.col-md-1{grid-column:span 1}.dynamic-form .grid>div.col-md-2{grid-column:span 2}.dynamic-form .grid>div.col-md-3{grid-column:span 3}.dynamic-form .grid>div.col-md-4{grid-column:span 4}.dynamic-form .grid>div.col-md-5{grid-column:span 5}.dynamic-form .grid>div.col-md-6{grid-column:span 6}.dynamic-form .grid>div.col-md-7{grid-column:span 7}.dynamic-form .grid>div.col-md-8{grid-column:span 8}.dynamic-form .grid>div.col-md-9{grid-column:span 9}.dynamic-form .grid>div.col-md-10{grid-column:span 10}.dynamic-form .grid>div.col-md-11{grid-column:span 11}.dynamic-form .grid>div.col-md-12{grid-column:span 12}}@media (min-width: 992px){.dynamic-form .grid>div.col-lg-1{grid-column:span 1}.dynamic-form .grid>div.col-lg-2{grid-column:span 2}.dynamic-form .grid>div.col-lg-3{grid-column:span 3}.dynamic-form .grid>div.col-lg-4{grid-column:span 4}.dynamic-form .grid>div.col-lg-5{grid-column:span 5}.dynamic-form .grid>div.col-lg-6{grid-column:span 6}.dynamic-form .grid>div.col-lg-7{grid-column:span 7}.dynamic-form .grid>div.col-lg-8{grid-column:span 8}.dynamic-form .grid>div.col-lg-9{grid-column:span 9}.dynamic-form .grid>div.col-lg-10{grid-column:span 10}.dynamic-form .grid>div.col-lg-11{grid-column:span 11}.dynamic-form .grid>div.col-lg-12{grid-column:span 12}}@media (min-width: 1200px){.dynamic-form .grid>div.col-xl-1{grid-column:span 1}.dynamic-form .grid>div.col-xl-2{grid-column:span 2}.dynamic-form .grid>div.col-xl-3{grid-column:span 3}.dynamic-form .grid>div.col-xl-4{grid-column:span 4}.dynamic-form .grid>div.col-xl-5{grid-column:span 5}.dynamic-form .grid>div.col-xl-6{grid-column:span 6}.dynamic-form .grid>div.col-xl-7{grid-column:span 7}.dynamic-form .grid>div.col-xl-8{grid-column:span 8}.dynamic-form .grid>div.col-xl-9{grid-column:span 9}.dynamic-form .grid>div.col-xl-10{grid-column:span 10}.dynamic-form .grid>div.col-xl-11{grid-column:span 11}.dynamic-form .grid>div.col-xl-12{grid-column:span 12}}.dynamic-form .sections-container{display:flex;flex-direction:column;gap:1rem;margin-top:1rem}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header{background-color:var(--surface-50);border-color:var(--surface-200)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header{display:flex;align-items:center;gap:.5rem;width:100%}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header .section-icon{font-size:1.1rem;color:var(--primary-color)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header .section-title{font-weight:600;color:var(--text-color)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-content{padding:1.5rem}.dynamic-form .fixed-filters-section{margin-bottom:1rem;padding-bottom:1rem;border-bottom:1px solid var(--surface-200)}.dynamic-form .filter-sections-container{display:flex;flex-direction:column;gap:1rem;margin-top:1rem}.dynamic-form .form-actions{display:flex;justify-content:flex-end;gap:.5rem;margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--surface-200)}.dialog-footer{display:flex;justify-content:flex-end;gap:.5rem}.drawer-footer{display:flex;justify-content:flex-end;gap:.5rem;padding:1rem;border-top:1px solid var(--surface-200)}.inline-edit-fields{display:flex;flex-direction:column;gap:1rem}.inline-edit-fields .inline-edit-field{display:flex;flex-direction:column;gap:.25rem}.inline-edit-fields .inline-edit-field .field-label{font-weight:500;font-size:.875rem;color:var(--text-color-secondary);margin-bottom:.25rem}.inline-edit-fields .inline-edit-field .field-display{display:flex;align-items:center;justify-content:space-between;padding:.75rem;background-color:var(--surface-50);border:1px solid var(--surface-200);border-radius:var(--border-radius);cursor:pointer;transition:all .2s ease;min-height:42px}.inline-edit-fields .inline-edit-field .field-display:hover:not(.disabled){background-color:var(--surface-100);border-color:var(--primary-color)}.inline-edit-fields .inline-edit-field .field-display:hover:not(.disabled) .edit-icon{opacity:1}.inline-edit-fields .inline-edit-field .field-display.disabled{cursor:not-allowed;opacity:.6}.inline-edit-fields .inline-edit-field .field-display .display-value{flex:1;color:var(--text-color)}.inline-edit-fields .inline-edit-field .field-display .display-value:empty:before{content:attr(data-placeholder);color:var(--text-color-secondary);font-style:italic}.inline-edit-fields .inline-edit-field .field-display .edit-icon{color:var(--primary-color);font-size:.875rem;opacity:.5;transition:opacity .2s ease}.inline-edit-fields .inline-edit-field .field-edit{display:flex;flex-direction:column;gap:.5rem}.inline-edit-fields .inline-edit-field .field-edit .input-container{flex:1}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field{gap:0}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field label{display:none}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field small.p-error{display:none}.inline-edit-fields .inline-edit-field .field-edit .edit-actions{display:flex;gap:.5rem;justify-content:flex-end}.inline-edit-fields .inline-edit-field .field-edit .field-error small.p-error{color:var(--p-red-500);font-size:.875rem}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: PanelModule }, { kind: "component", type: i4.Panel, selector: "p-panel", inputs: ["toggleable", "header", "collapsed", "style", "styleClass", "iconPos", "expandIcon", "collapseIcon", "showHeader", "toggler", "transitionOptions", "toggleButtonProps"], outputs: ["collapsedChange", "onBeforeToggle", "onAfterToggle"] }, { kind: "directive", type: i5.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "ngmodule", type: DialogModule }, { kind: "component", type: i6.Dialog, selector: "p-dialog", inputs: ["header", "draggable", "resizable", "positionLeft", "positionTop", "contentStyle", "contentStyleClass", "modal", "closeOnEscape", "dismissableMask", "rtl", "closable", "responsive", "appendTo", "breakpoints", "styleClass", "maskStyleClass", "maskStyle", "showHeader", "breakpoint", "blockScroll", "autoZIndex", "baseZIndex", "minX", "minY", "focusOnShow", "maximizable", "keepInViewport", "focusTrap", "transitionOptions", "closeIcon", "closeAriaLabel", "closeTabindex", "minimizeIcon", "maximizeIcon", "closeButtonProps", "maximizeButtonProps", "visible", "style", "position", "role", "content", "contentTemplate", "footerTemplate", "closeIconTemplate", "maximizeIconTemplate", "minimizeIconTemplate", "headlessTemplate"], outputs: ["onShow", "onHide", "visibleChange", "onResizeInit", "onResizeEnd", "onDragEnd", "onMaximize"] }, { kind: "ngmodule", type: DrawerModule }, { kind: "component", type: i7.Drawer, selector: "p-drawer", inputs: ["appendTo", "blockScroll", "style", "styleClass", "ariaCloseLabel", "autoZIndex", "baseZIndex", "modal", "closeButtonProps", "dismissible", "showCloseIcon", "closeOnEscape", "transitionOptions", "visible", "position", "fullScreen", "header", "maskStyle", "closable"], outputs: ["onShow", "onHide", "visibleChange"] }, { kind: "ngmodule", type: ButtonModule }, { kind: "component", type: i8.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "raised", "rounded", "text", "plain", "severity", "outlined", "link", "tabindex", "size", "variant", "style", "styleClass", "badgeClass", "badgeSeverity", "ariaLabel", "autofocus", "fluid", "buttonProps"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "pipe", type: TranslatePipe, name: "translate" }, { kind: "component", type: DynamicFieldTextComponent, selector: "sia-dynamic-field-text", inputs: ["field", "form", "mode"] }, { kind: "component", type: DynamicFieldNumberComponent, selector: "sia-dynamic-field-number", inputs: ["field", "form", "mode"] }, { kind: "component", type: DynamicFieldDateComponent, selector: "sia-dynamic-field-date", inputs: ["field", "form", "mode"] }, { kind: "component", type: DynamicFieldDropdownComponent, selector: "sia-dynamic-field-dropdown", inputs: ["field", "form", "mode"] }, { kind: "component", type: DynamicFieldLookupComponent, selector: "sia-dynamic-field-lookup", inputs: ["field", "form", "formGroup", "mode"] }, { kind: "component", type: DynamicFieldTextareaComponent, selector: "sia-dynamic-field-textarea", inputs: ["field", "form", "mode"] }] });
564
+ }
565
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: DynamicFormComponent, decorators: [{
566
+ type: Component,
567
+ args: [{ selector: 'sia-dynamic-form', standalone: true, imports: [
568
+ CommonModule,
569
+ ReactiveFormsModule,
570
+ PanelModule,
571
+ DialogModule,
572
+ DrawerModule,
573
+ ButtonModule,
574
+ TranslatePipe,
575
+ DynamicFieldTextComponent,
576
+ DynamicFieldNumberComponent,
577
+ DynamicFieldDateComponent,
578
+ DynamicFieldDropdownComponent,
579
+ DynamicFieldLookupComponent,
580
+ DynamicFieldTextareaComponent
581
+ ], template: "<!-- Modo Inline (padr\u00E3o) -->\n<ng-container *ngIf=\"isInlineMode\">\n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n</ng-container>\n\n<!-- Modo Dialog -->\n<p-dialog\n *ngIf=\"isDialogMode\"\n [(visible)]=\"visible\"\n [header]=\"dialogHeader | translate\"\n [modal]=\"true\"\n [style]=\"{ width: dialogWidth }\"\n [draggable]=\"dialogDraggable\"\n [resizable]=\"dialogResizable\"\n [maximizable]=\"dialogMaximizable\"\n [closeOnEscape]=\"dialogCloseOnEscape\"\n [dismissableMask]=\"dialogDismissableMask\"\n (onHide)=\"close()\">\n \n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n \n <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n <div class=\"dialog-footer\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n </ng-template>\n</p-dialog>\n\n<!-- Modo Drawer -->\n<p-drawer\n *ngIf=\"isDrawerMode\"\n [(visible)]=\"visible\"\n [header]=\"drawerHeader | translate\"\n [position]=\"drawerPosition\"\n [style]=\"drawerStyle\"\n [showCloseIcon]=\"drawerShowCloseIcon\"\n [closeOnEscape]=\"drawerCloseOnEscape\"\n [modal]=\"drawerDismissable\"\n (onHide)=\"close()\">\n \n <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n \n <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n <div class=\"drawer-footer\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n </ng-template>\n</p-drawer>\n\n<!-- Template do Formul\u00E1rio (reutiliz\u00E1vel) -->\n<ng-template #formContent>\n <form [formGroup]=\"form\" class=\"dynamic-form\">\n \n <!-- Filtros Fixos (apenas no modo filter) -->\n <div *ngIf=\"mode === 'filter' && hasFixedFilters\" class=\"fixed-filters-section\">\n <div class=\"grid\">\n <ng-container *ngFor=\"let field of fixedFilters\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Campos Diretos (sem se\u00E7\u00E3o) -->\n <div *ngIf=\"hasFields\">\n <!-- Modo Inline Edit -->\n <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n <ng-container *ngFor=\"let field of formFields\">\n <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n\n <!-- Modo Form Normal -->\n <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n <ng-container *ngFor=\"let field of formFields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </div>\n\n <!-- Se\u00E7\u00F5es de Campos (modo form) -->\n <div *ngIf=\"hasSections\" class=\"sections-container\">\n <p-panel \n *ngFor=\"let section of formSections\"\n [toggleable]=\"section.toggleable !== false\"\n [collapsed]=\"section.collapsed || false\"\n [styleClass]=\"'section-panel'\">\n \n <ng-template pTemplate=\"header\">\n <div class=\"section-header\">\n <i *ngIf=\"section.icon\" [class]=\"section.icon + ' section-icon'\"></i>\n <span class=\"section-title\">{{ section.title }}</span>\n </div>\n </ng-template>\n\n <!-- Modo Inline Edit -->\n <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n\n <!-- Modo Form Normal -->\n <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </p-panel>\n </div>\n\n <!-- Se\u00E7\u00F5es de Filtros (apenas no modo filter) -->\n <div *ngIf=\"mode === 'filter' && hasFilterSections\" class=\"filter-sections-container\">\n <p-panel \n *ngFor=\"let section of filterSections\"\n [header]=\"section.title\"\n [toggleable]=\"true\"\n [collapsed]=\"true\">\n\n <div class=\"grid\">\n <ng-container *ngFor=\"let field of section.fields\">\n <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n </ng-container>\n </div>\n </p-panel>\n </div>\n\n <!-- Bot\u00F5es inline (apenas para modo inline) -->\n <div *ngIf=\"isInlineMode && (showSubmitButton || showCancelButton)\" class=\"form-actions\">\n <p-button\n *ngIf=\"showCancelButton\"\n [label]=\"cancelLabel | translate\"\n [icon]=\"cancelButtonIcon\"\n severity=\"secondary\"\n (onClick)=\"handleCancel()\"\n [disabled]=\"loading\">\n </p-button>\n <p-button\n *ngIf=\"showSubmitButton\"\n [label]=\"submitLabel | translate\"\n [icon]=\"submitButtonIcon\"\n (onClick)=\"handleSubmit()\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid\">\n </p-button>\n </div>\n\n </form>\n</ng-template>\n\n<!-- Template de Campo (reutiliz\u00E1vel) -->\n<ng-template #fieldTemplate let-field=\"field\">\n <!-- Text Field -->\n <sia-dynamic-field-text\n *ngIf=\"field.type === 'text'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-text>\n\n <!-- Textarea Field -->\n <sia-dynamic-field-textarea\n *ngIf=\"field.type === 'textarea'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-textarea>\n\n <!-- Number Field -->\n <sia-dynamic-field-number\n *ngIf=\"field.type === 'number'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-number>\n\n <!-- Date Field -->\n <sia-dynamic-field-date\n *ngIf=\"field.type === 'date'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-date>\n\n <!-- Dropdown/Enum Field -->\n <sia-dynamic-field-dropdown\n *ngIf=\"field.type === 'dropdown' || field.type === 'enum'\"\n [field]=\"field\"\n [form]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-dropdown>\n\n <!-- Lookup Field -->\n <sia-dynamic-field-lookup\n *ngIf=\"field.type === 'lookup'\"\n [field]=\"field\"\n [form]=\"form\"\n [formGroup]=\"form\"\n [mode]=\"fieldMode\">\n </sia-dynamic-field-lookup>\n</ng-template>\n\n\n<!-- Template de Campo Inline Edit -->\n<ng-template #inlineEditFieldTemplate let-field=\"field\">\n <div class=\"edit-inline-field\">\n <label class=\"field-label\">{{ field.label | translate }}</label>\n \n <!-- Display Mode -->\n <div class=\"field-display\" \n *ngIf=\"!isFieldEditing(field.field)\" \n (click)=\"startEditField(field.field)\"\n [class.disabled]=\"field.disabled\">\n <span class=\"display-value\" \n [attr.data-placeholder]=\"field.placeholder || ('crmx.business.not_informed' | translate)\">\n {{ getFieldDisplayValue(field.field) }}\n </span>\n <i class=\"pi pi-pencil edit-icon\" *ngIf=\"!field.disabled\"></i>\n </div>\n \n <!-- Edit Mode -->\n <div class=\"field-edit\" *ngIf=\"isFieldEditing(field.field)\">\n <div class=\"input-container\">\n <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n </div>\n \n <!-- Action Buttons -->\n <div class=\"edit-actions\">\n <p-button \n type=\"button\" \n icon=\"pi pi-check\" \n [rounded]=\"true\"\n severity=\"primary\"\n (onClick)=\"saveInlineField(field.field)\" \n [loading]=\"savingField\"\n [disabled]=\"form.get(field.field)?.invalid\">\n </p-button>\n <p-button \n type=\"button\" \n icon=\"pi pi-times\" \n [rounded]=\"true\"\n severity=\"secondary\"\n (onClick)=\"cancelInlineEdit()\"\n [disabled]=\"savingField\">\n </p-button>\n </div>\n \n <!-- Error Message -->\n <div class=\"field-error\" *ngIf=\"form.get(field.field)?.invalid && form.get(field.field)?.touched\">\n <small class=\"p-error\">\n {{ form.get(field.field)?.errors?.['required'] ? ('crmx.business.required_field' | translate) : '' }}\n </small>\n </div>\n </div>\n </div>\n</ng-template>\n", styles: [".dynamic-form{width:100%}.dynamic-form .grid{display:grid;grid-template-columns:repeat(12,1fr);gap:1rem}.dynamic-form .grid>div.col-1{grid-column:span 1}.dynamic-form .grid>div.col-2{grid-column:span 2}.dynamic-form .grid>div.col-3{grid-column:span 3}.dynamic-form .grid>div.col-4{grid-column:span 4}.dynamic-form .grid>div.col-5{grid-column:span 5}.dynamic-form .grid>div.col-6{grid-column:span 6}.dynamic-form .grid>div.col-7{grid-column:span 7}.dynamic-form .grid>div.col-8{grid-column:span 8}.dynamic-form .grid>div.col-9{grid-column:span 9}.dynamic-form .grid>div.col-10{grid-column:span 10}.dynamic-form .grid>div.col-11{grid-column:span 11}.dynamic-form .grid>div.col-12{grid-column:span 12}@media (max-width: 575.98px){.dynamic-form .grid>div.col-xs-1{grid-column:span 1}.dynamic-form .grid>div.col-xs-2{grid-column:span 2}.dynamic-form .grid>div.col-xs-3{grid-column:span 3}.dynamic-form .grid>div.col-xs-4{grid-column:span 4}.dynamic-form .grid>div.col-xs-5{grid-column:span 5}.dynamic-form .grid>div.col-xs-6{grid-column:span 6}.dynamic-form .grid>div.col-xs-7{grid-column:span 7}.dynamic-form .grid>div.col-xs-8{grid-column:span 8}.dynamic-form .grid>div.col-xs-9{grid-column:span 9}.dynamic-form .grid>div.col-xs-10{grid-column:span 10}.dynamic-form .grid>div.col-xs-11{grid-column:span 11}.dynamic-form .grid>div.col-xs-12{grid-column:span 12}}@media (min-width: 576px){.dynamic-form .grid>div.col-sm-1{grid-column:span 1}.dynamic-form .grid>div.col-sm-2{grid-column:span 2}.dynamic-form .grid>div.col-sm-3{grid-column:span 3}.dynamic-form .grid>div.col-sm-4{grid-column:span 4}.dynamic-form .grid>div.col-sm-5{grid-column:span 5}.dynamic-form .grid>div.col-sm-6{grid-column:span 6}.dynamic-form .grid>div.col-sm-7{grid-column:span 7}.dynamic-form .grid>div.col-sm-8{grid-column:span 8}.dynamic-form .grid>div.col-sm-9{grid-column:span 9}.dynamic-form .grid>div.col-sm-10{grid-column:span 10}.dynamic-form .grid>div.col-sm-11{grid-column:span 11}.dynamic-form .grid>div.col-sm-12{grid-column:span 12}}@media (min-width: 768px){.dynamic-form .grid>div.col-md-1{grid-column:span 1}.dynamic-form .grid>div.col-md-2{grid-column:span 2}.dynamic-form .grid>div.col-md-3{grid-column:span 3}.dynamic-form .grid>div.col-md-4{grid-column:span 4}.dynamic-form .grid>div.col-md-5{grid-column:span 5}.dynamic-form .grid>div.col-md-6{grid-column:span 6}.dynamic-form .grid>div.col-md-7{grid-column:span 7}.dynamic-form .grid>div.col-md-8{grid-column:span 8}.dynamic-form .grid>div.col-md-9{grid-column:span 9}.dynamic-form .grid>div.col-md-10{grid-column:span 10}.dynamic-form .grid>div.col-md-11{grid-column:span 11}.dynamic-form .grid>div.col-md-12{grid-column:span 12}}@media (min-width: 992px){.dynamic-form .grid>div.col-lg-1{grid-column:span 1}.dynamic-form .grid>div.col-lg-2{grid-column:span 2}.dynamic-form .grid>div.col-lg-3{grid-column:span 3}.dynamic-form .grid>div.col-lg-4{grid-column:span 4}.dynamic-form .grid>div.col-lg-5{grid-column:span 5}.dynamic-form .grid>div.col-lg-6{grid-column:span 6}.dynamic-form .grid>div.col-lg-7{grid-column:span 7}.dynamic-form .grid>div.col-lg-8{grid-column:span 8}.dynamic-form .grid>div.col-lg-9{grid-column:span 9}.dynamic-form .grid>div.col-lg-10{grid-column:span 10}.dynamic-form .grid>div.col-lg-11{grid-column:span 11}.dynamic-form .grid>div.col-lg-12{grid-column:span 12}}@media (min-width: 1200px){.dynamic-form .grid>div.col-xl-1{grid-column:span 1}.dynamic-form .grid>div.col-xl-2{grid-column:span 2}.dynamic-form .grid>div.col-xl-3{grid-column:span 3}.dynamic-form .grid>div.col-xl-4{grid-column:span 4}.dynamic-form .grid>div.col-xl-5{grid-column:span 5}.dynamic-form .grid>div.col-xl-6{grid-column:span 6}.dynamic-form .grid>div.col-xl-7{grid-column:span 7}.dynamic-form .grid>div.col-xl-8{grid-column:span 8}.dynamic-form .grid>div.col-xl-9{grid-column:span 9}.dynamic-form .grid>div.col-xl-10{grid-column:span 10}.dynamic-form .grid>div.col-xl-11{grid-column:span 11}.dynamic-form .grid>div.col-xl-12{grid-column:span 12}}.dynamic-form .sections-container{display:flex;flex-direction:column;gap:1rem;margin-top:1rem}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header{background-color:var(--surface-50);border-color:var(--surface-200)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header{display:flex;align-items:center;gap:.5rem;width:100%}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header .section-icon{font-size:1.1rem;color:var(--primary-color)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-header .section-header .section-title{font-weight:600;color:var(--text-color)}.dynamic-form .sections-container ::ng-deep .section-panel .p-panel-content{padding:1.5rem}.dynamic-form .fixed-filters-section{margin-bottom:1rem;padding-bottom:1rem;border-bottom:1px solid var(--surface-200)}.dynamic-form .filter-sections-container{display:flex;flex-direction:column;gap:1rem;margin-top:1rem}.dynamic-form .form-actions{display:flex;justify-content:flex-end;gap:.5rem;margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--surface-200)}.dialog-footer{display:flex;justify-content:flex-end;gap:.5rem}.drawer-footer{display:flex;justify-content:flex-end;gap:.5rem;padding:1rem;border-top:1px solid var(--surface-200)}.inline-edit-fields{display:flex;flex-direction:column;gap:1rem}.inline-edit-fields .inline-edit-field{display:flex;flex-direction:column;gap:.25rem}.inline-edit-fields .inline-edit-field .field-label{font-weight:500;font-size:.875rem;color:var(--text-color-secondary);margin-bottom:.25rem}.inline-edit-fields .inline-edit-field .field-display{display:flex;align-items:center;justify-content:space-between;padding:.75rem;background-color:var(--surface-50);border:1px solid var(--surface-200);border-radius:var(--border-radius);cursor:pointer;transition:all .2s ease;min-height:42px}.inline-edit-fields .inline-edit-field .field-display:hover:not(.disabled){background-color:var(--surface-100);border-color:var(--primary-color)}.inline-edit-fields .inline-edit-field .field-display:hover:not(.disabled) .edit-icon{opacity:1}.inline-edit-fields .inline-edit-field .field-display.disabled{cursor:not-allowed;opacity:.6}.inline-edit-fields .inline-edit-field .field-display .display-value{flex:1;color:var(--text-color)}.inline-edit-fields .inline-edit-field .field-display .display-value:empty:before{content:attr(data-placeholder);color:var(--text-color-secondary);font-style:italic}.inline-edit-fields .inline-edit-field .field-display .edit-icon{color:var(--primary-color);font-size:.875rem;opacity:.5;transition:opacity .2s ease}.inline-edit-fields .inline-edit-field .field-edit{display:flex;flex-direction:column;gap:.5rem}.inline-edit-fields .inline-edit-field .field-edit .input-container{flex:1}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field{gap:0}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field label{display:none}.inline-edit-fields .inline-edit-field .field-edit .input-container ::ng-deep .field small.p-error{display:none}.inline-edit-fields .inline-edit-field .field-edit .edit-actions{display:flex;gap:.5rem;justify-content:flex-end}.inline-edit-fields .inline-edit-field .field-edit .field-error small.p-error{color:var(--p-red-500);font-size:.875rem}\n"] }]
582
+ }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.TranslationService }], propDecorators: { formFields: [{
583
+ type: Input
584
+ }], formSections: [{
585
+ type: Input
586
+ }], fixedFilters: [{
587
+ type: Input
588
+ }], filterSections: [{
589
+ type: Input
590
+ }], entityData: [{
591
+ type: Input
592
+ }], mode: [{
593
+ type: Input
594
+ }], displayMode: [{
595
+ type: Input
596
+ }], visible: [{
597
+ type: Input
598
+ }], dialogConfig: [{
599
+ type: Input
600
+ }], drawerConfig: [{
601
+ type: Input
602
+ }], showSubmitButton: [{
603
+ type: Input
604
+ }], showCancelButton: [{
605
+ type: Input
606
+ }], submitButtonLabel: [{
607
+ type: Input
608
+ }], cancelButtonLabel: [{
609
+ type: Input
610
+ }], submitButtonIcon: [{
611
+ type: Input
612
+ }], cancelButtonIcon: [{
613
+ type: Input
614
+ }], formReady: [{
615
+ type: Output
616
+ }], formSubmit: [{
617
+ type: Output
618
+ }], visibleChange: [{
619
+ type: Output
620
+ }], onCancel: [{
621
+ type: Output
622
+ }], fieldSave: [{
623
+ type: Output
624
+ }] } });
625
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-form.component.js","sourceRoot":"","sources":["../../../../../projects/components-ai/src/lib/components/dynamic-form/dynamic-form.component.ts","../../../../../projects/components-ai/src/lib/components/dynamic-form/dynamic-form.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output, OnChanges, SimpleChanges } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormBuilder, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { PanelModule } from 'primeng/panel';\nimport { DialogModule } from 'primeng/dialog';\nimport { DrawerModule } from 'primeng/drawer';\nimport { ButtonModule } from 'primeng/button';\nimport { \n  DynamicFormFieldConfig, \n  DynamicFormSection,\n  DynamicFormDisplayMode,\n  DynamicFormDialogConfig,\n  DynamicFormDrawerConfig\n} from './models/dynamic-form.models';\nimport { TranslationService } from '../../services/translation.service';\nimport { TranslatePipe } from '../../pipes/translate.pipe';\nimport { DynamicFieldTextComponent } from './fields/dynamic-field-text.component';\nimport { DynamicFieldNumberComponent } from './fields/dynamic-field-number.component';\nimport { DynamicFieldDateComponent } from './fields/dynamic-field-date.component';\nimport { DynamicFieldDropdownComponent } from './fields/dynamic-field-dropdown.component';\nimport { DynamicFieldLookupComponent } from './fields/dynamic-field-lookup.component';\nimport { DynamicFieldTextareaComponent } from './fields/dynamic-field-textarea.component';\n\n@Component({\n  selector: 'sia-dynamic-form',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    PanelModule,\n    DialogModule,\n    DrawerModule,\n    ButtonModule,\n    TranslatePipe,\n    DynamicFieldTextComponent,\n    DynamicFieldNumberComponent,\n    DynamicFieldDateComponent,\n    DynamicFieldDropdownComponent,\n    DynamicFieldLookupComponent,\n    DynamicFieldTextareaComponent\n  ],\n  templateUrl: './dynamic-form.component.html',\n  styleUrl: './dynamic-form.component.scss'\n})\nexport class DynamicFormComponent implements OnInit, OnChanges {\n  @Input() formFields: DynamicFormFieldConfig[] = [];\n  @Input() formSections: DynamicFormSection[] = [];\n  @Input() fixedFilters: DynamicFormFieldConfig[] = []; // Filtros fixos (sempre visíveis no topo)\n  @Input() filterSections: DynamicFormSection[] = []; // Seções de filtros (colapsáveis)\n  @Input() entityData: any = null;\n  @Input() mode: 'form' | 'filter' | 'inline-edit' = 'form';\n  \n  // Display Mode Configuration\n  @Input() displayMode: DynamicFormDisplayMode = 'inline';\n  @Input() visible: boolean = false;\n  @Input() dialogConfig: DynamicFormDialogConfig = {};\n  @Input() drawerConfig: DynamicFormDrawerConfig = {};\n  \n  // Button Configuration\n  @Input() showSubmitButton: boolean = false;\n  @Input() showCancelButton: boolean = false;\n  @Input() submitButtonLabel?: string;\n  @Input() cancelButtonLabel?: string;\n  @Input() submitButtonIcon: string = 'pi pi-check';\n  @Input() cancelButtonIcon: string = 'pi pi-times';\n  \n  @Output() formReady = new EventEmitter<FormGroup>();\n  @Output() formSubmit = new EventEmitter<any>();\n  @Output() visibleChange = new EventEmitter<boolean>();\n  @Output() onCancel = new EventEmitter<void>();\n  @Output() fieldSave = new EventEmitter<{ field: string, value: any }>();\n\n  form!: FormGroup;\n  loading: boolean = false;\n  \n  // Inline edit mode state\n  editingField: string | null = null;\n  savingField: boolean = false;\n\n  get hasFields(): boolean {\n    return this.formFields.length > 0;\n  }\n\n  get hasSections(): boolean {\n    return this.formSections.length > 0;\n  }\n\n  get hasFixedFilters(): boolean {\n    return this.fixedFilters.length > 0;\n  }\n\n  get hasFilterSections(): boolean {\n    return this.filterSections.length > 0;\n  }\n\n  get allFields(): DynamicFormFieldConfig[] {\n    // Retorna todos os campos (diretos + das seções + filtros fixos + seções de filtros)\n    const sectionFields = this.formSections.flatMap(section => section.fields);\n    const filterSectionFields = this.filterSections.flatMap(section => section.fields);\n    return [...this.formFields, ...sectionFields, ...this.fixedFilters, ...filterSectionFields];\n  }\n\n  get isInlineMode(): boolean {\n    return this.displayMode === 'inline';\n  }\n\n  get isInlineEditMode(): boolean {\n    return this.mode === 'inline-edit';\n  }\n\n  get fieldMode(): 'form' | 'filter' {\n    return this.isInlineEditMode ? 'form' : this.mode as 'form' | 'filter';\n  }\n\n  get isDialogMode(): boolean {\n    return this.displayMode === 'dialog';\n  }\n\n  get isDrawerMode(): boolean {\n    return this.displayMode === 'drawer';\n  }\n\n  // Dialog configuration with defaults\n  get dialogHeader(): string {\n    return this.dialogConfig.header || this.translationService.translate('crmx.business.form');\n  }\n\n  get dialogWidth(): string {\n    return this.dialogConfig.width || '600px';\n  }\n\n  get dialogDraggable(): boolean {\n    return this.dialogConfig.draggable !== undefined ? this.dialogConfig.draggable : false;\n  }\n\n  get dialogResizable(): boolean {\n    return this.dialogConfig.resizable !== undefined ? this.dialogConfig.resizable : false;\n  }\n\n  get dialogMaximizable(): boolean {\n    return this.dialogConfig.maximizable !== undefined ? this.dialogConfig.maximizable : false;\n  }\n\n  get dialogCloseOnEscape(): boolean {\n    return this.dialogConfig.closeOnEscape !== undefined ? this.dialogConfig.closeOnEscape : true;\n  }\n\n  get dialogDismissableMask(): boolean {\n    return this.dialogConfig.dismissableMask !== undefined ? this.dialogConfig.dismissableMask : false;\n  }\n\n  // Drawer configuration with defaults\n  get drawerHeader(): string {\n    return this.drawerConfig.header || this.translationService.translate('crmx.business.form');\n  }\n\n  get drawerPosition(): 'left' | 'right' | 'top' | 'bottom' {\n    return this.drawerConfig.position || 'right';\n  }\n\n  get drawerShowCloseIcon(): boolean {\n    return this.drawerConfig.showCloseIcon !== undefined ? this.drawerConfig.showCloseIcon : true;\n  }\n\n  get drawerCloseOnEscape(): boolean {\n    return this.drawerConfig.closeOnEscape !== undefined ? this.drawerConfig.closeOnEscape : true;\n  }\n\n  get drawerDismissable(): boolean {\n    return this.drawerConfig.dismissable !== undefined ? this.drawerConfig.dismissable : true;\n  }\n\n  get drawerStyle(): { [key: string]: string } {\n    const position = this.drawerPosition;\n    const isHorizontal = position === 'left' || position === 'right';\n    \n    // Se tem customWidth, usa ela\n    if (this.drawerConfig.customWidth) {\n      return isHorizontal \n        ? { width: this.drawerConfig.customWidth }\n        : { height: this.drawerConfig.customWidth };\n    }\n    \n    // Usa o tamanho padronizado\n    const size = this.drawerConfig.size || 'medium';\n    const sizeMap = {\n      'small': isHorizontal ? '25%' : '25vh',\n      'medium': isHorizontal ? '40%' : '40vh',\n      'large': isHorizontal ? '60%' : '60vh',\n      'xlarge': isHorizontal ? '80%' : '80vh',\n      'full': isHorizontal ? '100%' : '100vh'\n    };\n    \n    const sizeValue = sizeMap[size];\n    return isHorizontal \n      ? { width: sizeValue }\n      : { height: sizeValue };\n  }\n\n  get submitLabel(): string {\n    return this.submitButtonLabel || this.translationService.translate('crmx.business.save');\n  }\n\n  get cancelLabel(): string {\n    return this.cancelButtonLabel || this.translationService.translate('crmx.business.cancel');\n  }\n\n  constructor(\n    private fb: FormBuilder,\n    private translationService: TranslationService\n  ) {}\n\n  ngOnInit(): void {\n    this.initForm();\n    this.setupConditionalFieldsWatcher();\n    this.formReady.emit(this.form);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['entityData'] && this.form) {\n      if (this.entityData) {\n        this.loadEntityData();\n      } else {\n        this.resetForm();\n      }\n    }\n    \n    // Reset loading and form when dialog/drawer is opened\n    if (changes['visible'] && changes['visible'].currentValue === true) {\n      this.loading = false;\n      \n      // If opening without entityData (new record), reset form\n      if (!this.entityData && this.form) {\n        this.resetForm();\n      }\n    }\n  }\n\n  initForm(): void {\n    const formControls: any = {};\n    \n    // Processa todos os campos (diretos + das seções)\n    this.allFields.forEach(field => {\n      const defaultValue = this.getDefaultValue(field);\n      const validators = this.mode === 'filter' ? [] : (field.validators || []);\n      \n      const control = this.fb.control(\n        { value: defaultValue, disabled: field.disabled || false },\n        validators\n      );\n      \n      formControls[field.field] = control;\n    });\n    \n    this.form = this.fb.group(formControls);\n    this.applyInitialDisabledState();\n  }\n\n  private getDefaultValue(field: DynamicFormFieldConfig): any {\n    if (field.defaultValue !== undefined) {\n      return field.defaultValue;\n    }\n\n    switch (field.type) {\n      case 'number':\n      case 'date':\n        return null;\n      case 'dropdown':\n      case 'enum':\n        return this.mode === 'filter' ? null : '';\n      default:\n        return '';\n    }\n  }\n\n  private applyInitialDisabledState(): void {\n    this.allFields.forEach(field => {\n      if (field.disabledWhen) {\n        const control = this.form.get(field.field);\n        if (control && this.isFieldDisabled(field)) {\n          control.disable();\n        }\n      }\n    });\n  }\n\n  loadEntityData(): void {\n    if (this.entityData && this.form) {\n      const processedData = this.processEntityData(this.entityData);\n      this.form.patchValue(processedData);\n    }\n  }\n\n  private processEntityData(entity: any): any {\n    const processedData = { ...entity };\n    \n    this.allFields.forEach(field => {\n      if (field.type === 'lookup' && processedData[field.field]) {\n        const lookupValue = processedData[field.field];\n        // Keep the full object if it has display data, otherwise extract ID\n        if (typeof lookupValue === 'object' && lookupValue.id) {\n          const hasDisplayData = field.lookupDisplayFields \n            ? field.lookupDisplayFields.some(f => this.getNestedValue(lookupValue, f))\n            : lookupValue[field.lookupDisplayField || 'name'];\n          \n          if (hasDisplayData) {\n            // Keep the full object to avoid unnecessary API calls\n            processedData[field.field] = lookupValue;\n          } else {\n            // Only has ID, extract it\n            processedData[field.field] = lookupValue.id;\n          }\n        }\n      }\n    });\n    \n    return processedData;\n  }\n\n  private getNestedValue(obj: any, path: string): any {\n    return path.split('.').reduce((current, key) => current?.[key], obj);\n  }\n\n  resetForm(): void {\n    if (this.form) {\n      const resetValues: any = {};\n      this.allFields.forEach(field => {\n        resetValues[field.field] = this.getDefaultValue(field);\n      });\n      this.form.reset(resetValues);\n    }\n  }\n\n  private processFilterValue(formValue: any): any {\n    const processedValue = { ...formValue };\n    \n    this.allFields.forEach(field => {\n      const fieldValue = processedValue[field.field];\n      \n      if (field.type === 'lookup') {\n        if (fieldValue && typeof fieldValue === 'object' && fieldValue.id) {\n          processedValue[field.field] = fieldValue;\n        } else if (fieldValue && typeof fieldValue === 'string') {\n          processedValue[field.field] = { id: fieldValue };\n        } else {\n          processedValue[field.field] = null;\n        }\n      }\n    });\n    \n    return processedValue;\n  }\n\n  private processFormValue(formValue: any): any {\n    const processedValue = { ...formValue };\n    \n    if (this.entityData?.id) {\n      processedValue.id = this.entityData.id;\n    }\n    \n    this.allFields.forEach(field => {\n      const fieldValue = processedValue[field.field];\n      \n      if (field.type === 'lookup') {\n        // If it's an object, extract the ID\n        if (fieldValue && typeof fieldValue === 'object') {\n          processedValue[field.field] = fieldValue.id || fieldValue[field.lookupValueField || 'id'] || null;\n        } else if (fieldValue && fieldValue !== '') {\n          // If it's already an ID, keep it\n          processedValue[field.field] = { id: fieldValue };\n        } else {\n          processedValue[field.field] = null;\n        }\n      } else if (field.type === 'date') {\n        if (fieldValue instanceof Date) {\n          processedValue[field.field] = fieldValue.toISOString().split('T')[0];\n        } else if (!fieldValue) {\n          processedValue[field.field] = null;\n        }\n      } else if (['text', 'textarea', 'number', 'enum', 'dropdown'].includes(field.type)) {\n        if (fieldValue === '' || fieldValue === undefined) {\n          processedValue[field.field] = null;\n        }\n      }\n    });\n    \n    return processedValue;\n  }\n\n  getFieldGridClass(field: DynamicFormFieldConfig): string {\n    const classes: string[] = [];\n    \n    // Colspan base (padrão: 6)\n    let baseColspan = field.colspan || 6;\n    \n    // Verifica condições de colspan dinâmico\n    if (field.colspanWhen && field.colspanWhen.length > 0) {\n      for (const condition of field.colspanWhen) {\n        const dependentFieldValue = this.form.get(condition.field)?.value;\n        if (dependentFieldValue === condition.value) {\n          baseColspan = condition.colspan;\n          break;\n        }\n      }\n    }\n    \n    // Adiciona classe base\n    classes.push(`col-${baseColspan}`);\n    \n    // Suporta tanto o objeto size quanto as propriedades antigas (para compatibilidade)\n    const sizeXs = field.size?.xs ?? field.colspanXs;\n    const sizeSm = field.size?.sm ?? field.colspanSm;\n    const sizeMd = field.size?.md ?? field.colspanMd;\n    const sizeLg = field.size?.lg ?? field.colspanLg;\n    const sizeXl = field.size?.xl ?? field.colspanXl;\n    \n    // Adiciona classes responsivas se definidas\n    if (sizeXs !== undefined) {\n      classes.push(`col-xs-${sizeXs}`);\n    }\n    if (sizeSm !== undefined) {\n      classes.push(`col-sm-${sizeSm}`);\n    }\n    if (sizeMd !== undefined) {\n      classes.push(`col-md-${sizeMd}`);\n    }\n    if (sizeLg !== undefined) {\n      classes.push(`col-lg-${sizeLg}`);\n    }\n    if (sizeXl !== undefined) {\n      classes.push(`col-xl-${sizeXl}`);\n    }\n    \n    return classes.join(' ');\n  }\n\n  isFieldVisible(field: DynamicFormFieldConfig): boolean {\n    if (this.mode === 'filter') {\n      return true;\n    }\n    \n    if (!field.visibleWhen) {\n      return true;\n    }\n\n    const dependentFieldValue = this.form.get(field.visibleWhen.field)?.value;\n    return dependentFieldValue === field.visibleWhen.value;\n  }\n\n  isFieldDisabled(field: DynamicFormFieldConfig): boolean {\n    if (field.disabled) {\n      return true;\n    }\n    \n    if (!('disabledWhen' in field) || !field.disabledWhen) {\n      return false;\n    }\n\n    const dependentFieldValue = this.form.get(field.disabledWhen!.field)?.value;\n    \n    if (field.disabledWhen!.value === null) {\n      return !dependentFieldValue;\n    }\n    \n    return dependentFieldValue === field.disabledWhen!.value;\n  }\n\n  setupConditionalFieldsWatcher(): void {\n    const visibleDependentFields = this.allFields.filter(field => field.visibleWhen);\n    const visibleWatchedFields = [...new Set(visibleDependentFields.map(field => field.visibleWhen!.field))];\n\n    const disabledDependentFields = this.allFields.filter(field => field.disabledWhen);\n    const disabledWatchedFields = [...new Set(disabledDependentFields.map(field => field.disabledWhen!.field))];\n\n    const allWatchedFields = [...new Set([...visibleWatchedFields, ...disabledWatchedFields])];\n\n    allWatchedFields.forEach(fieldName => {\n      const control = this.form.get(fieldName);\n      if (control) {\n        control.valueChanges.subscribe(() => {\n          this.clearHiddenFields();\n          this.updateDisabledFields();\n        });\n      }\n    });\n  }\n\n  updateDisabledFields(): void {\n    this.allFields.forEach(field => {\n      if (field.disabledWhen) {\n        const control = this.form.get(field.field);\n        if (control) {\n          const shouldBeDisabled = this.isFieldDisabled(field);\n          \n          if (shouldBeDisabled && control.enabled) {\n            control.setValue(null);\n            control.disable();\n          } else if (!shouldBeDisabled && control.disabled && !field.disabled) {\n            control.enable();\n          }\n        }\n      }\n    });\n  }\n\n  clearHiddenFields(): void {\n    this.allFields.forEach(field => {\n      if (field.visibleWhen && !this.isFieldVisible(field)) {\n        const control = this.form.get(field.field);\n        if (control) {\n          const resetValue = this.getDefaultValue(field);\n          control.setValue(resetValue);\n          control.markAsUntouched();\n          control.markAsPristine();\n        }\n      }\n    });\n  }\n\n  getForm(): FormGroup {\n    return this.form;\n  }\n\n  clearFilters(): void {\n    if (this.mode === 'filter') {\n      this.resetForm();\n    }\n  }\n\n  getTypePersonValue(): 'NATURAL_PERSON' | 'JURIDICAL_PERSON' {\n    const typePersonValue = this.form?.get('typePerson')?.value;\n    return typePersonValue || 'JURIDICAL_PERSON';\n  }\n\n  // Display Mode Control Methods\n  open(entity?: any): void {\n    if (entity) {\n      this.entityData = entity;\n    } else {\n      this.entityData = null;\n      this.resetForm(); // Reset form when opening for new entity\n    }\n    this.loading = false; // Reset loading state\n    this.visible = true;\n    this.visibleChange.emit(true);\n  }\n\n  close(): void {\n    this.loading = false; // Reset loading state\n    this.visible = false;\n    this.visibleChange.emit(false);\n    this.resetForm(); // Reset form when closing\n    this.onCancel.emit();\n  }\n\n  handleCancel(): void {\n    this.close();\n  }\n\n  handleSubmit(): void {\n    if (this.mode === 'form' && this.form.invalid) {\n      this.form.markAllAsTouched();\n      return;\n    }\n    \n    this.loading = true;\n    \n    const formValue = this.mode === 'filter' \n      ? this.processFilterValue(this.form.value)\n      : this.processFormValue(this.form.value);\n    \n    this.formSubmit.emit(formValue);\n  }\n\n  // Inline Edit Mode Methods\n  startEditField(fieldName: string): void {\n    if (this.isInlineEditMode) {\n      this.editingField = fieldName;\n      \n      // Ensure the field has the current value from entityData\n      if (this.entityData) {\n        const control = this.form.get(fieldName);\n        if (control) {\n          const field = this.allFields.find(f => f.field === fieldName);\n          const value = this.entityData[fieldName];\n          \n          // For lookup fields in inline-edit mode, keep the full object\n          // This prevents unnecessary API calls since we already have the data\n          if (field?.type === 'lookup' && value && typeof value === 'object') {\n            // Keep the full object - the lookup component will use it\n            control.setValue(value);\n          } else {\n            control.setValue(value);\n          }\n        }\n      }\n    }\n  }\n\n  isFieldEditing(fieldName: string): boolean {\n    return this.editingField === fieldName;\n  }\n\n  saveInlineField(fieldName: string): void {\n    if (!this.isInlineEditMode) return;\n\n    const control = this.form.get(fieldName);\n    if (!control || control.invalid) {\n      control?.markAsTouched();\n      return;\n    }\n\n    this.savingField = true;\n    \n    let fieldValue = control.value;\n    \n    // For lookup fields, extract the ID if it's an object\n    const field = this.allFields.find(f => f.field === fieldName);\n    if (field?.type === 'lookup' && fieldValue && typeof fieldValue === 'object') {\n      fieldValue = fieldValue.id || fieldValue[field.lookupValueField || 'id'];\n    }\n    \n    this.fieldSave.emit({ field: fieldName, value: fieldValue });\n  }\n\n  cancelInlineEdit(): void {\n    if (this.isInlineEditMode && this.editingField) {\n      const control = this.form.get(this.editingField);\n      if (control && this.entityData) {\n        // Restore original value\n        const originalValue = this.entityData[this.editingField];\n        control.setValue(originalValue);\n        control.markAsPristine();\n        control.markAsUntouched();\n      }\n      this.editingField = null;\n      this.savingField = false;\n    }\n  }\n\n  // Method to be called by parent after save completes\n  completeInlineEdit(): void {\n    this.editingField = null;\n    this.savingField = false;\n  }\n\n  getFieldDisplayValue(fieldName: string): string {\n    if (!this.entityData) return '';\n\n    const field = this.allFields.find(f => f.field === fieldName);\n    const value = this.entityData[fieldName];\n\n    if (!value) return '';\n\n    // For lookup fields, return the display name\n    if (field?.type === 'lookup' && typeof value === 'object' && value.name) {\n      return value.name;\n    }\n\n    // For number fields, format appropriately\n    if (field?.type === 'number' && typeof value === 'number') {\n      if (field.numberMode === 'currency') {\n        return new Intl.NumberFormat(field.locale || 'pt-BR', {\n          style: 'currency',\n          currency: field.currency || 'BRL'\n        }).format(value);\n      }\n      return new Intl.NumberFormat(field.locale || 'pt-BR', {\n        minimumFractionDigits: field.minFractionDigits || 0,\n        maximumFractionDigits: field.maxFractionDigits || 2\n      }).format(value);\n    }\n\n    // For date fields, format as date\n    if (field?.type === 'date' && value) {\n      const date = typeof value === 'string' ? new Date(value) : value;\n      return date.toLocaleDateString('pt-BR');\n    }\n\n    return String(value);\n  }\n}\n","<!-- Modo Inline (padrão) -->\n<ng-container *ngIf=\"isInlineMode\">\n  <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n</ng-container>\n\n<!-- Modo Dialog -->\n<p-dialog\n  *ngIf=\"isDialogMode\"\n  [(visible)]=\"visible\"\n  [header]=\"dialogHeader | translate\"\n  [modal]=\"true\"\n  [style]=\"{ width: dialogWidth }\"\n  [draggable]=\"dialogDraggable\"\n  [resizable]=\"dialogResizable\"\n  [maximizable]=\"dialogMaximizable\"\n  [closeOnEscape]=\"dialogCloseOnEscape\"\n  [dismissableMask]=\"dialogDismissableMask\"\n  (onHide)=\"close()\">\n  \n  <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n  \n  <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n    <div class=\"dialog-footer\">\n      <p-button\n        *ngIf=\"showCancelButton\"\n        [label]=\"cancelLabel | translate\"\n        [icon]=\"cancelButtonIcon\"\n        severity=\"secondary\"\n        (onClick)=\"handleCancel()\"\n        [disabled]=\"loading\">\n      </p-button>\n      <p-button\n        *ngIf=\"showSubmitButton\"\n        [label]=\"submitLabel | translate\"\n        [icon]=\"submitButtonIcon\"\n        (onClick)=\"handleSubmit()\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid\">\n      </p-button>\n    </div>\n  </ng-template>\n</p-dialog>\n\n<!-- Modo Drawer -->\n<p-drawer\n  *ngIf=\"isDrawerMode\"\n  [(visible)]=\"visible\"\n  [header]=\"drawerHeader | translate\"\n  [position]=\"drawerPosition\"\n  [style]=\"drawerStyle\"\n  [showCloseIcon]=\"drawerShowCloseIcon\"\n  [closeOnEscape]=\"drawerCloseOnEscape\"\n  [modal]=\"drawerDismissable\"\n  (onHide)=\"close()\">\n  \n  <ng-container *ngTemplateOutlet=\"formContent\"></ng-container>\n  \n  <ng-template pTemplate=\"footer\" *ngIf=\"showSubmitButton || showCancelButton\">\n    <div class=\"drawer-footer\">\n      <p-button\n        *ngIf=\"showCancelButton\"\n        [label]=\"cancelLabel | translate\"\n        [icon]=\"cancelButtonIcon\"\n        severity=\"secondary\"\n        (onClick)=\"handleCancel()\"\n        [disabled]=\"loading\">\n      </p-button>\n      <p-button\n        *ngIf=\"showSubmitButton\"\n        [label]=\"submitLabel | translate\"\n        [icon]=\"submitButtonIcon\"\n        (onClick)=\"handleSubmit()\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid\">\n      </p-button>\n    </div>\n  </ng-template>\n</p-drawer>\n\n<!-- Template do Formulário (reutilizável) -->\n<ng-template #formContent>\n  <form [formGroup]=\"form\" class=\"dynamic-form\">\n    \n    <!-- Filtros Fixos (apenas no modo filter) -->\n    <div *ngIf=\"mode === 'filter' && hasFixedFilters\" class=\"fixed-filters-section\">\n      <div class=\"grid\">\n        <ng-container *ngFor=\"let field of fixedFilters\">\n          <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n            <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n          </div>\n        </ng-container>\n      </div>\n    </div>\n\n    <!-- Campos Diretos (sem seção) -->\n    <div *ngIf=\"hasFields\">\n      <!-- Modo Inline Edit -->\n      <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n        <ng-container *ngFor=\"let field of formFields\">\n          <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n            <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n          </div>\n        </ng-container>\n      </div>\n\n      <!-- Modo Form Normal -->\n      <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n        <ng-container *ngFor=\"let field of formFields\">\n          <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n            <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n          </div>\n        </ng-container>\n      </div>\n    </div>\n\n    <!-- Seções de Campos (modo form) -->\n    <div *ngIf=\"hasSections\" class=\"sections-container\">\n      <p-panel \n        *ngFor=\"let section of formSections\"\n        [toggleable]=\"section.toggleable !== false\"\n        [collapsed]=\"section.collapsed || false\"\n        [styleClass]=\"'section-panel'\">\n        \n        <ng-template pTemplate=\"header\">\n          <div class=\"section-header\">\n            <i *ngIf=\"section.icon\" [class]=\"section.icon + ' section-icon'\"></i>\n            <span class=\"section-title\">{{ section.title }}</span>\n          </div>\n        </ng-template>\n\n        <!-- Modo Inline Edit -->\n        <div *ngIf=\"isInlineEditMode\" class=\"inline-edit-fields\">\n          <ng-container *ngFor=\"let field of section.fields\">\n            <div *ngIf=\"isFieldVisible(field)\" class=\"inline-edit-field\">\n              <ng-container *ngTemplateOutlet=\"inlineEditFieldTemplate; context: { field: field }\"></ng-container>\n            </div>\n          </ng-container>\n        </div>\n\n        <!-- Modo Form Normal -->\n        <div *ngIf=\"!isInlineEditMode\" class=\"grid\">\n          <ng-container *ngFor=\"let field of section.fields\">\n            <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n              <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n            </div>\n          </ng-container>\n        </div>\n      </p-panel>\n    </div>\n\n    <!-- Seções de Filtros (apenas no modo filter) -->\n    <div *ngIf=\"mode === 'filter' && hasFilterSections\" class=\"filter-sections-container\">\n      <p-panel \n        *ngFor=\"let section of filterSections\"\n        [header]=\"section.title\"\n        [toggleable]=\"true\"\n        [collapsed]=\"true\">\n\n        <div class=\"grid\">\n          <ng-container *ngFor=\"let field of section.fields\">\n            <div *ngIf=\"isFieldVisible(field)\" [ngClass]=\"getFieldGridClass(field)\">\n              <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n            </div>\n          </ng-container>\n        </div>\n      </p-panel>\n    </div>\n\n    <!-- Botões inline (apenas para modo inline) -->\n    <div *ngIf=\"isInlineMode && (showSubmitButton || showCancelButton)\" class=\"form-actions\">\n      <p-button\n        *ngIf=\"showCancelButton\"\n        [label]=\"cancelLabel | translate\"\n        [icon]=\"cancelButtonIcon\"\n        severity=\"secondary\"\n        (onClick)=\"handleCancel()\"\n        [disabled]=\"loading\">\n      </p-button>\n      <p-button\n        *ngIf=\"showSubmitButton\"\n        [label]=\"submitLabel | translate\"\n        [icon]=\"submitButtonIcon\"\n        (onClick)=\"handleSubmit()\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid\">\n      </p-button>\n    </div>\n\n  </form>\n</ng-template>\n\n<!-- Template de Campo (reutilizável) -->\n<ng-template #fieldTemplate let-field=\"field\">\n  <!-- Text Field -->\n  <sia-dynamic-field-text\n    *ngIf=\"field.type === 'text'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-text>\n\n  <!-- Textarea Field -->\n  <sia-dynamic-field-textarea\n    *ngIf=\"field.type === 'textarea'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-textarea>\n\n  <!-- Number Field -->\n  <sia-dynamic-field-number\n    *ngIf=\"field.type === 'number'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-number>\n\n  <!-- Date Field -->\n  <sia-dynamic-field-date\n    *ngIf=\"field.type === 'date'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-date>\n\n  <!-- Dropdown/Enum Field -->\n  <sia-dynamic-field-dropdown\n    *ngIf=\"field.type === 'dropdown' || field.type === 'enum'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-dropdown>\n\n  <!-- Lookup Field -->\n  <sia-dynamic-field-lookup\n    *ngIf=\"field.type === 'lookup'\"\n    [field]=\"field\"\n    [form]=\"form\"\n    [formGroup]=\"form\"\n    [mode]=\"fieldMode\">\n  </sia-dynamic-field-lookup>\n</ng-template>\n\n\n<!-- Template de Campo Inline Edit -->\n<ng-template #inlineEditFieldTemplate let-field=\"field\">\n  <div class=\"edit-inline-field\">\n    <label class=\"field-label\">{{ field.label | translate }}</label>\n    \n    <!-- Display Mode -->\n    <div class=\"field-display\" \n         *ngIf=\"!isFieldEditing(field.field)\" \n         (click)=\"startEditField(field.field)\"\n         [class.disabled]=\"field.disabled\">\n      <span class=\"display-value\" \n            [attr.data-placeholder]=\"field.placeholder || ('crmx.business.not_informed' | translate)\">\n        {{ getFieldDisplayValue(field.field) }}\n      </span>\n      <i class=\"pi pi-pencil edit-icon\" *ngIf=\"!field.disabled\"></i>\n    </div>\n    \n    <!-- Edit Mode -->\n    <div class=\"field-edit\" *ngIf=\"isFieldEditing(field.field)\">\n      <div class=\"input-container\">\n        <ng-container *ngTemplateOutlet=\"fieldTemplate; context: { field: field }\"></ng-container>\n      </div>\n      \n      <!-- Action Buttons -->\n      <div class=\"edit-actions\">\n        <p-button \n          type=\"button\" \n          icon=\"pi pi-check\" \n          [rounded]=\"true\"\n          severity=\"primary\"\n          (onClick)=\"saveInlineField(field.field)\" \n          [loading]=\"savingField\"\n          [disabled]=\"form.get(field.field)?.invalid\">\n        </p-button>\n        <p-button \n          type=\"button\" \n          icon=\"pi pi-times\" \n          [rounded]=\"true\"\n          severity=\"secondary\"\n          (onClick)=\"cancelInlineEdit()\"\n          [disabled]=\"savingField\">\n        </p-button>\n      </div>\n      \n      <!-- Error Message -->\n      <div class=\"field-error\" *ngIf=\"form.get(field.field)?.invalid && form.get(field.field)?.touched\">\n        <small class=\"p-error\">\n          {{ form.get(field.field)?.errors?.['required'] ? ('crmx.business.required_field' | translate) : '' }}\n        </small>\n      </div>\n    </div>\n  </div>\n</ng-template>\n"]}