@gnggln/ng-ui-system 1.0.0-alpha.0

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 (141) hide show
  1. package/esm2022/gnggln-ng-ui-system.mjs +5 -0
  2. package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
  3. package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
  4. package/esm2022/lib/components/accordion/index.mjs +2 -0
  5. package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
  6. package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
  7. package/esm2022/lib/components/base-layout/index.mjs +14 -0
  8. package/esm2022/lib/components/button/button-area.component.mjs +196 -0
  9. package/esm2022/lib/components/button/button.component.mjs +164 -0
  10. package/esm2022/lib/components/button/button.types.mjs +6 -0
  11. package/esm2022/lib/components/button/index.mjs +16 -0
  12. package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
  13. package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
  14. package/esm2022/lib/components/crud-table/index.mjs +16 -0
  15. package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
  16. package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
  17. package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
  18. package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
  19. package/esm2022/lib/components/form-builder/index.mjs +19 -0
  20. package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
  21. package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
  22. package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
  23. package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
  24. package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
  25. package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
  26. package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
  27. package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
  28. package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
  29. package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
  30. package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
  31. package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
  32. package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
  33. package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
  34. package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
  35. package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
  36. package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
  37. package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
  38. package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
  39. package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
  40. package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
  41. package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
  42. package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
  43. package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
  44. package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
  45. package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
  46. package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
  47. package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
  48. package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
  49. package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
  50. package/esm2022/lib/components/layout-builder/index.mjs +18 -0
  51. package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
  52. package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
  53. package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
  54. package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
  55. package/esm2022/lib/components/modal/index.mjs +4 -0
  56. package/esm2022/lib/components/modal/modal.component.mjs +139 -0
  57. package/esm2022/lib/components/modal/modal.service.mjs +194 -0
  58. package/esm2022/lib/components/modal/modal.types.mjs +6 -0
  59. package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
  60. package/esm2022/lib/components/page-header/index.mjs +20 -0
  61. package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
  62. package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
  63. package/esm2022/lib/components/table/index.mjs +2 -0
  64. package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
  65. package/esm2022/lib/components/table/table.types.mjs +6 -0
  66. package/esm2022/lib/core/types/index.mjs +6 -0
  67. package/esm2022/lib/core/utils/index.mjs +53 -0
  68. package/esm2022/lib/sources/location-data.opt.json +8942 -0
  69. package/esm2022/lib/sources/nazioni.opt.json +215 -0
  70. package/esm2022/public-api.mjs +34 -0
  71. package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
  72. package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
  73. package/index.d.ts +5 -0
  74. package/lib/components/accordion/accordion.component.d.ts +118 -0
  75. package/lib/components/accordion/accordion.types.d.ts +62 -0
  76. package/lib/components/accordion/index.d.ts +2 -0
  77. package/lib/components/base-layout/base-layout.component.d.ts +83 -0
  78. package/lib/components/base-layout/base-layout.types.d.ts +26 -0
  79. package/lib/components/base-layout/index.d.ts +13 -0
  80. package/lib/components/button/button-area.component.d.ts +88 -0
  81. package/lib/components/button/button.component.d.ts +55 -0
  82. package/lib/components/button/button.types.d.ts +70 -0
  83. package/lib/components/button/index.d.ts +15 -0
  84. package/lib/components/crud-table/crud-table.component.d.ts +143 -0
  85. package/lib/components/crud-table/crud-table.types.d.ts +207 -0
  86. package/lib/components/crud-table/index.d.ts +15 -0
  87. package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
  88. package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
  89. package/lib/components/form-builder/form-builder.component.d.ts +183 -0
  90. package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
  91. package/lib/components/form-builder/index.d.ts +13 -0
  92. package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
  93. package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
  94. package/lib/components/form-builder/services/location.service.d.ts +83 -0
  95. package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
  96. package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
  97. package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
  98. package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
  99. package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
  100. package/lib/components/form-builder/types/condition.types.d.ts +51 -0
  101. package/lib/components/form-builder/types/field.types.d.ts +288 -0
  102. package/lib/components/form-builder/types/index.d.ts +5 -0
  103. package/lib/components/form-builder/types/schema.types.d.ts +227 -0
  104. package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
  105. package/lib/components/form-builder/types/validation.types.d.ts +174 -0
  106. package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
  107. package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
  108. package/lib/components/form-builder-editor/index.d.ts +15 -0
  109. package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
  110. package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
  111. package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
  112. package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
  113. package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
  114. package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
  115. package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
  116. package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
  117. package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
  118. package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
  119. package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
  120. package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
  121. package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
  122. package/lib/components/layout-builder/index.d.ts +16 -0
  123. package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
  124. package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
  125. package/lib/components/layout-builder/layout.service.d.ts +100 -0
  126. package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
  127. package/lib/components/modal/index.d.ts +4 -0
  128. package/lib/components/modal/modal.component.d.ts +44 -0
  129. package/lib/components/modal/modal.service.d.ts +93 -0
  130. package/lib/components/modal/modal.types.d.ts +110 -0
  131. package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
  132. package/lib/components/page-header/index.d.ts +16 -0
  133. package/lib/components/page-header/page-header.component.d.ts +59 -0
  134. package/lib/components/page-header/page-header.types.d.ts +96 -0
  135. package/lib/components/table/index.d.ts +2 -0
  136. package/lib/components/table/paginated-table.component.d.ts +85 -0
  137. package/lib/components/table/table.types.d.ts +81 -0
  138. package/lib/core/types/index.d.ts +57 -0
  139. package/lib/core/utils/index.d.ts +29 -0
  140. package/package.json +44 -0
  141. package/public-api.d.ts +22 -0
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ25nZ2xuLW5nLXVpLXN5c3RlbS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3BhY2thZ2VzL25nLXVpLXN5c3RlbS9zcmMvZ25nZ2xuLW5nLXVpLXN5c3RlbS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,353 @@
1
+ import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, ChangeDetectorRef, ViewEncapsulation, inject, } from '@angular/core';
2
+ import { NgTemplateOutlet } from '@angular/common';
3
+ import { LucideAngularModule } from 'lucide-angular';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "lucide-angular";
6
+ /**
7
+ * Accordion configurabile, data-driven, con supporto multi/single expand,
8
+ * animazioni CSS, icone Lucide e template personalizzabili.
9
+ *
10
+ * Il contenuto dei pannelli viene fornito tramite un `itemTemplate`
11
+ * che riceve il descriptor del pannello come contesto.
12
+ *
13
+ * @selector ui-accordion
14
+ *
15
+ * @example
16
+ * ```html
17
+ * <ui-accordion
18
+ * [items]="sections"
19
+ * [itemTemplate]="panelTpl"
20
+ * mode="single"
21
+ * >
22
+ * <ng-template #panelTpl let-item>
23
+ * @switch (item.id) {
24
+ * @case ('info') { <app-info-panel /> }
25
+ * @case ('address') { <app-address-form /> }
26
+ * }
27
+ * </ng-template>
28
+ * </ui-accordion>
29
+ * ```
30
+ *
31
+ * @example
32
+ * ```typescript
33
+ * sections: UiAccordionDescriptor[] = [
34
+ * { id: 'info', title: 'Informazioni', icon: 'info', expanded: true },
35
+ * { id: 'address', title: 'Indirizzo', icon: 'map-pin' },
36
+ * ];
37
+ * ```
38
+ */
39
+ export class UiAccordionComponent {
40
+ constructor() {
41
+ this.cdr = inject(ChangeDetectorRef);
42
+ /**
43
+ * Fingerprint dell'ultimo stato dei descriptor.
44
+ * Usato da DoCheck per rilevare mutazioni in-place sulle proprieta
45
+ * degli item senza richiedere un nuovo riferimento dell'array.
46
+ */
47
+ this._lastFingerprint = '';
48
+ /**
49
+ * Array di descriptor dei pannelli.
50
+ * Pannelli con `hidden: true` vengono filtrati automaticamente.
51
+ *
52
+ * Supporta sia aggiornamenti immutabili (nuovo array) sia
53
+ * mutazioni in-place delle proprieta dei descriptor.
54
+ */
55
+ this.items = [];
56
+ /**
57
+ * Template per il contenuto di ogni pannello.
58
+ * Il contesto fornisce il descriptor come `$implicit`.
59
+ *
60
+ * @example
61
+ * ```html
62
+ * <ng-template #tpl let-item>
63
+ * <p>Contenuto per: {{ item.title }}</p>
64
+ * </ng-template>
65
+ * ```
66
+ */
67
+ this.itemTemplate = null;
68
+ /**
69
+ * Template opzionale per contenuto aggiuntivo nell'header.
70
+ * Il contesto fornisce il descriptor come `$implicit`.
71
+ * Viene renderizzato tra il titolo e il chevron.
72
+ */
73
+ this.headerTemplate = null;
74
+ /**
75
+ * Modalita di espansione.
76
+ * - `multi`: piu pannelli aperti contemporaneamente
77
+ * - `single`: un solo pannello aperto alla volta
78
+ */
79
+ this.mode = 'multi';
80
+ /** Label accessibile per il gruppo accordion. */
81
+ this.ariaLabel = 'Accordion';
82
+ /** Emesso quando un pannello viene aperto o chiuso. */
83
+ this.itemToggled = new EventEmitter();
84
+ }
85
+ /**
86
+ * Rileva mutazioni in-place sulle proprieta dei descriptor.
87
+ * Confronta un fingerprint leggero dello stato corrente con quello
88
+ * precedente e, se diverso, forza un ciclo di change detection.
89
+ *
90
+ * Questo permette al consumatore di fare sia:
91
+ * - `items[2].hidden = true` (mutazione in-place)
92
+ * - `items = [...newItems]` (nuovo riferimento)
93
+ */
94
+ ngDoCheck() {
95
+ const fp = this.computeFingerprint();
96
+ if (fp !== this._lastFingerprint) {
97
+ this._lastFingerprint = fp;
98
+ this.cdr.markForCheck();
99
+ }
100
+ }
101
+ /** Pannelli visibili (non hidden). */
102
+ get visibleItems() {
103
+ return this.items.filter((i) => !i.hidden);
104
+ }
105
+ /** Alterna l'espansione di un pannello. */
106
+ toggle(item) {
107
+ if (item.disabled)
108
+ return;
109
+ const willExpand = !item.expanded;
110
+ // In modalita single, chiudi tutti gli altri
111
+ if (willExpand && this.mode === 'single') {
112
+ this.items.forEach((i) => {
113
+ if (i !== item && i.expanded) {
114
+ i.expanded = false;
115
+ this.itemToggled.emit({ item: i, expanded: false });
116
+ }
117
+ });
118
+ }
119
+ item.expanded = willExpand;
120
+ this.itemToggled.emit({ item, expanded: willExpand });
121
+ }
122
+ /** Apre un pannello specifico per id. */
123
+ open(id) {
124
+ const item = this.items.find((i) => i.id === id);
125
+ if (item && !item.expanded && !item.disabled) {
126
+ this.toggle(item);
127
+ }
128
+ }
129
+ /** Chiude un pannello specifico per id. */
130
+ close(id) {
131
+ const item = this.items.find((i) => i.id === id);
132
+ if (item && item.expanded) {
133
+ item.expanded = false;
134
+ this.itemToggled.emit({ item, expanded: false });
135
+ }
136
+ }
137
+ /** Apre tutti i pannelli (solo in modalita multi). */
138
+ openAll() {
139
+ if (this.mode === 'single')
140
+ return;
141
+ this.items.forEach((item) => {
142
+ if (!item.expanded && !item.disabled && !item.hidden) {
143
+ item.expanded = true;
144
+ this.itemToggled.emit({ item, expanded: true });
145
+ }
146
+ });
147
+ }
148
+ /** Chiude tutti i pannelli. */
149
+ closeAll() {
150
+ this.items.forEach((item) => {
151
+ if (item.expanded) {
152
+ item.expanded = false;
153
+ this.itemToggled.emit({ item, expanded: false });
154
+ }
155
+ });
156
+ }
157
+ /** @internal Classi CSS per il pannello. */
158
+ getPanelClasses(item) {
159
+ return [
160
+ 'ui-accordion__panel',
161
+ item.expanded ? 'ui-accordion__panel--expanded' : '',
162
+ item.disabled ? 'ui-accordion__panel--disabled' : '',
163
+ ]
164
+ .filter(Boolean)
165
+ .join(' ');
166
+ }
167
+ /** @internal Sposta il focus al primo header. */
168
+ focusFirst(event) {
169
+ event.preventDefault();
170
+ const headers = this.getHeaders();
171
+ headers[0]?.focus();
172
+ }
173
+ /** @internal Sposta il focus all'ultimo header. */
174
+ focusLast(event) {
175
+ event.preventDefault();
176
+ const headers = this.getHeaders();
177
+ headers[headers.length - 1]?.focus();
178
+ }
179
+ /** @internal Recupera tutti gli header button del DOM. */
180
+ getHeaders() {
181
+ return Array.from(document.querySelectorAll('.ui-accordion__header:not(:disabled)'));
182
+ }
183
+ /**
184
+ * @internal Calcola un fingerprint leggero basato sulle proprieta
185
+ * reattive dei descriptor (quelle che influenzano il rendering).
186
+ */
187
+ computeFingerprint() {
188
+ return this.items.map((i) => `${i.id}:${i.hidden ?? 0}:${i.disabled ?? 0}:${i.expanded ?? 0}:${i.title}`).join('|');
189
+ }
190
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiAccordionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
191
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiAccordionComponent, isStandalone: true, selector: "ui-accordion", inputs: { items: "items", itemTemplate: "itemTemplate", headerTemplate: "headerTemplate", mode: "mode", ariaLabel: "ariaLabel" }, outputs: { itemToggled: "itemToggled" }, host: { classAttribute: "ui-accordion-host" }, ngImport: i0, template: `
192
+ @for (item of visibleItems; track item.id) {
193
+ <div
194
+ [class]="getPanelClasses(item)"
195
+ [id]="'ui-accordion-' + item.id"
196
+ >
197
+ <!-- Header -->
198
+ <button
199
+ class="ui-accordion__header"
200
+ [class.ui-accordion__header--expanded]="item.expanded"
201
+ [class.ui-accordion__header--disabled]="item.disabled"
202
+ [disabled]="item.disabled"
203
+ [attr.aria-expanded]="item.expanded"
204
+ [attr.aria-controls]="'ui-accordion-body-' + item.id"
205
+ [attr.id]="'ui-accordion-header-' + item.id"
206
+ type="button"
207
+ (click)="toggle(item)"
208
+ (keydown.home)="focusFirst($event)"
209
+ (keydown.end)="focusLast($event)"
210
+ >
211
+ <div class="ui-accordion__header-content">
212
+ @if (item.icon) {
213
+ <lucide-icon
214
+ class="ui-accordion__icon"
215
+ [name]="item.icon"
216
+ [size]="18"
217
+ aria-hidden="true"
218
+ />
219
+ }
220
+ <span class="ui-accordion__title">{{ item.title }}</span>
221
+ @if (item.subtitle) {
222
+ <span class="ui-accordion__subtitle">{{ item.subtitle }}</span>
223
+ }
224
+ </div>
225
+
226
+ @if (headerTemplate) {
227
+ <div class="ui-accordion__header-extra">
228
+ <ng-container
229
+ [ngTemplateOutlet]="headerTemplate"
230
+ [ngTemplateOutletContext]="{ $implicit: item }"
231
+ />
232
+ </div>
233
+ }
234
+
235
+ <lucide-icon
236
+ class="ui-accordion__chevron"
237
+ [class.ui-accordion__chevron--rotated]="item.expanded"
238
+ name="chevron-down"
239
+ [size]="16"
240
+ aria-hidden="true"
241
+ />
242
+ </button>
243
+
244
+ <!-- Body (animazione CSS grid) -->
245
+ <div
246
+ class="ui-accordion__body-wrapper"
247
+ [class.ui-accordion__body-wrapper--open]="item.expanded"
248
+ role="region"
249
+ [attr.id]="'ui-accordion-body-' + item.id"
250
+ [attr.aria-labelledby]="'ui-accordion-header-' + item.id"
251
+ >
252
+ <div class="ui-accordion__body">
253
+ @if (item.expanded && itemTemplate) {
254
+ <ng-container
255
+ [ngTemplateOutlet]="itemTemplate"
256
+ [ngTemplateOutletContext]="{ $implicit: item }"
257
+ />
258
+ }
259
+ </div>
260
+ </div>
261
+ </div>
262
+ }
263
+ `, isInline: true, styles: [".ui-accordion-host{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-accordion__panel{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-surface);overflow:hidden;transition:box-shadow var(--ui-transition-fast)}.ui-accordion__panel--expanded{box-shadow:var(--ui-shadow-xs)}.ui-accordion__panel--disabled{opacity:.55}.ui-accordion__header{appearance:none;border:none;background:transparent;cursor:pointer;font-family:var(--ui-font-family);display:flex;align-items:center;width:100%;padding:var(--ui-spacing-4) var(--ui-spacing-5);gap:var(--ui-spacing-3);text-align:left;transition:background-color var(--ui-transition-fast),color var(--ui-transition-fast)}.ui-accordion__header:focus{outline:none}.ui-accordion__header:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-accordion__header:hover:not(:disabled){background:var(--ui-color-surface-hover)}.ui-accordion__header--expanded{background:var(--ui-color-bg-subtle);border-bottom:1px solid var(--ui-color-border)}.ui-accordion__header--disabled{cursor:not-allowed}.ui-accordion__header-content{flex:1;display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-accordion__icon{color:var(--ui-color-primary);flex-shrink:0}.ui-accordion__title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-accordion__subtitle{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-accordion__header-extra{flex-shrink:0}.ui-accordion__chevron{flex-shrink:0;color:var(--ui-color-text-muted);transition:transform var(--ui-transition-normal)}.ui-accordion__chevron--rotated{transform:rotate(180deg)}.ui-accordion__body-wrapper{display:grid;grid-template-rows:0fr;transition:grid-template-rows var(--ui-transition-normal)}.ui-accordion__body-wrapper--open{grid-template-rows:1fr}.ui-accordion__body{overflow:hidden;padding:0 var(--ui-spacing-5);transition:padding var(--ui-transition-normal)}.ui-accordion__body-wrapper--open .ui-accordion__body{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-accordion__header{padding:var(--ui-spacing-3) var(--ui-spacing-4)}.ui-accordion__body-wrapper--open .ui-accordion__body{padding:var(--ui-spacing-4)}}\n"], dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
264
+ }
265
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiAccordionComponent, decorators: [{
266
+ type: Component,
267
+ args: [{ selector: 'ui-accordion', standalone: true, imports: [NgTemplateOutlet, LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: { class: 'ui-accordion-host' }, template: `
268
+ @for (item of visibleItems; track item.id) {
269
+ <div
270
+ [class]="getPanelClasses(item)"
271
+ [id]="'ui-accordion-' + item.id"
272
+ >
273
+ <!-- Header -->
274
+ <button
275
+ class="ui-accordion__header"
276
+ [class.ui-accordion__header--expanded]="item.expanded"
277
+ [class.ui-accordion__header--disabled]="item.disabled"
278
+ [disabled]="item.disabled"
279
+ [attr.aria-expanded]="item.expanded"
280
+ [attr.aria-controls]="'ui-accordion-body-' + item.id"
281
+ [attr.id]="'ui-accordion-header-' + item.id"
282
+ type="button"
283
+ (click)="toggle(item)"
284
+ (keydown.home)="focusFirst($event)"
285
+ (keydown.end)="focusLast($event)"
286
+ >
287
+ <div class="ui-accordion__header-content">
288
+ @if (item.icon) {
289
+ <lucide-icon
290
+ class="ui-accordion__icon"
291
+ [name]="item.icon"
292
+ [size]="18"
293
+ aria-hidden="true"
294
+ />
295
+ }
296
+ <span class="ui-accordion__title">{{ item.title }}</span>
297
+ @if (item.subtitle) {
298
+ <span class="ui-accordion__subtitle">{{ item.subtitle }}</span>
299
+ }
300
+ </div>
301
+
302
+ @if (headerTemplate) {
303
+ <div class="ui-accordion__header-extra">
304
+ <ng-container
305
+ [ngTemplateOutlet]="headerTemplate"
306
+ [ngTemplateOutletContext]="{ $implicit: item }"
307
+ />
308
+ </div>
309
+ }
310
+
311
+ <lucide-icon
312
+ class="ui-accordion__chevron"
313
+ [class.ui-accordion__chevron--rotated]="item.expanded"
314
+ name="chevron-down"
315
+ [size]="16"
316
+ aria-hidden="true"
317
+ />
318
+ </button>
319
+
320
+ <!-- Body (animazione CSS grid) -->
321
+ <div
322
+ class="ui-accordion__body-wrapper"
323
+ [class.ui-accordion__body-wrapper--open]="item.expanded"
324
+ role="region"
325
+ [attr.id]="'ui-accordion-body-' + item.id"
326
+ [attr.aria-labelledby]="'ui-accordion-header-' + item.id"
327
+ >
328
+ <div class="ui-accordion__body">
329
+ @if (item.expanded && itemTemplate) {
330
+ <ng-container
331
+ [ngTemplateOutlet]="itemTemplate"
332
+ [ngTemplateOutletContext]="{ $implicit: item }"
333
+ />
334
+ }
335
+ </div>
336
+ </div>
337
+ </div>
338
+ }
339
+ `, styles: [".ui-accordion-host{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-accordion__panel{border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-surface);overflow:hidden;transition:box-shadow var(--ui-transition-fast)}.ui-accordion__panel--expanded{box-shadow:var(--ui-shadow-xs)}.ui-accordion__panel--disabled{opacity:.55}.ui-accordion__header{appearance:none;border:none;background:transparent;cursor:pointer;font-family:var(--ui-font-family);display:flex;align-items:center;width:100%;padding:var(--ui-spacing-4) var(--ui-spacing-5);gap:var(--ui-spacing-3);text-align:left;transition:background-color var(--ui-transition-fast),color var(--ui-transition-fast)}.ui-accordion__header:focus{outline:none}.ui-accordion__header:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-accordion__header:hover:not(:disabled){background:var(--ui-color-surface-hover)}.ui-accordion__header--expanded{background:var(--ui-color-bg-subtle);border-bottom:1px solid var(--ui-color-border)}.ui-accordion__header--disabled{cursor:not-allowed}.ui-accordion__header-content{flex:1;display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-accordion__icon{color:var(--ui-color-primary);flex-shrink:0}.ui-accordion__title{font-size:var(--ui-font-size-sm);font-weight:var(--ui-font-weight-semibold);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-accordion__subtitle{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-accordion__header-extra{flex-shrink:0}.ui-accordion__chevron{flex-shrink:0;color:var(--ui-color-text-muted);transition:transform var(--ui-transition-normal)}.ui-accordion__chevron--rotated{transform:rotate(180deg)}.ui-accordion__body-wrapper{display:grid;grid-template-rows:0fr;transition:grid-template-rows var(--ui-transition-normal)}.ui-accordion__body-wrapper--open{grid-template-rows:1fr}.ui-accordion__body{overflow:hidden;padding:0 var(--ui-spacing-5);transition:padding var(--ui-transition-normal)}.ui-accordion__body-wrapper--open .ui-accordion__body{padding:var(--ui-spacing-5)}@media (max-width: 768px){.ui-accordion__header{padding:var(--ui-spacing-3) var(--ui-spacing-4)}.ui-accordion__body-wrapper--open .ui-accordion__body{padding:var(--ui-spacing-4)}}\n"] }]
340
+ }], propDecorators: { items: [{
341
+ type: Input
342
+ }], itemTemplate: [{
343
+ type: Input
344
+ }], headerTemplate: [{
345
+ type: Input
346
+ }], mode: [{
347
+ type: Input
348
+ }], ariaLabel: [{
349
+ type: Input
350
+ }], itemToggled: [{
351
+ type: Output
352
+ }] } });
353
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"accordion.component.js","sourceRoot":"","sources":["../../../../../../packages/ng-ui-system/src/lib/components/accordion/accordion.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  ChangeDetectionStrategy,\r\n  ChangeDetectorRef,\r\n  ViewEncapsulation,\r\n  DoCheck,\r\n  TemplateRef,\r\n  inject,\r\n} from '@angular/core';\r\nimport { NgTemplateOutlet } from '@angular/common';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiAccordionDescriptor, UiAccordionToggleEvent, UiAccordionMode } from './accordion.types';\r\n\r\n/**\r\n * Accordion configurabile, data-driven, con supporto multi/single expand,\r\n * animazioni CSS, icone Lucide e template personalizzabili.\r\n *\r\n * Il contenuto dei pannelli viene fornito tramite un `itemTemplate`\r\n * che riceve il descriptor del pannello come contesto.\r\n *\r\n * @selector ui-accordion\r\n *\r\n * @example\r\n * ```html\r\n * <ui-accordion\r\n *   [items]=\"sections\"\r\n *   [itemTemplate]=\"panelTpl\"\r\n *   mode=\"single\"\r\n * >\r\n *   <ng-template #panelTpl let-item>\r\n *     @switch (item.id) {\r\n *       @case ('info')    { <app-info-panel /> }\r\n *       @case ('address') { <app-address-form /> }\r\n *     }\r\n *   </ng-template>\r\n * </ui-accordion>\r\n * ```\r\n *\r\n * @example\r\n * ```typescript\r\n * sections: UiAccordionDescriptor[] = [\r\n *   { id: 'info', title: 'Informazioni', icon: 'info', expanded: true },\r\n *   { id: 'address', title: 'Indirizzo', icon: 'map-pin' },\r\n * ];\r\n * ```\r\n */\r\n@Component({\r\n  selector: 'ui-accordion',\r\n  standalone: true,\r\n  imports: [NgTemplateOutlet, LucideAngularModule],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  host: { class: 'ui-accordion-host' },\r\n  template: `\r\n    @for (item of visibleItems; track item.id) {\r\n      <div\r\n        [class]=\"getPanelClasses(item)\"\r\n        [id]=\"'ui-accordion-' + item.id\"\r\n      >\r\n        <!-- Header -->\r\n        <button\r\n          class=\"ui-accordion__header\"\r\n          [class.ui-accordion__header--expanded]=\"item.expanded\"\r\n          [class.ui-accordion__header--disabled]=\"item.disabled\"\r\n          [disabled]=\"item.disabled\"\r\n          [attr.aria-expanded]=\"item.expanded\"\r\n          [attr.aria-controls]=\"'ui-accordion-body-' + item.id\"\r\n          [attr.id]=\"'ui-accordion-header-' + item.id\"\r\n          type=\"button\"\r\n          (click)=\"toggle(item)\"\r\n          (keydown.home)=\"focusFirst($event)\"\r\n          (keydown.end)=\"focusLast($event)\"\r\n        >\r\n          <div class=\"ui-accordion__header-content\">\r\n            @if (item.icon) {\r\n              <lucide-icon\r\n                class=\"ui-accordion__icon\"\r\n                [name]=\"item.icon\"\r\n                [size]=\"18\"\r\n                aria-hidden=\"true\"\r\n              />\r\n            }\r\n            <span class=\"ui-accordion__title\">{{ item.title }}</span>\r\n            @if (item.subtitle) {\r\n              <span class=\"ui-accordion__subtitle\">{{ item.subtitle }}</span>\r\n            }\r\n          </div>\r\n\r\n          @if (headerTemplate) {\r\n            <div class=\"ui-accordion__header-extra\">\r\n              <ng-container\r\n                [ngTemplateOutlet]=\"headerTemplate\"\r\n                [ngTemplateOutletContext]=\"{ $implicit: item }\"\r\n              />\r\n            </div>\r\n          }\r\n\r\n          <lucide-icon\r\n            class=\"ui-accordion__chevron\"\r\n            [class.ui-accordion__chevron--rotated]=\"item.expanded\"\r\n            name=\"chevron-down\"\r\n            [size]=\"16\"\r\n            aria-hidden=\"true\"\r\n          />\r\n        </button>\r\n\r\n        <!-- Body (animazione CSS grid) -->\r\n        <div\r\n          class=\"ui-accordion__body-wrapper\"\r\n          [class.ui-accordion__body-wrapper--open]=\"item.expanded\"\r\n          role=\"region\"\r\n          [attr.id]=\"'ui-accordion-body-' + item.id\"\r\n          [attr.aria-labelledby]=\"'ui-accordion-header-' + item.id\"\r\n        >\r\n          <div class=\"ui-accordion__body\">\r\n            @if (item.expanded && itemTemplate) {\r\n              <ng-container\r\n                [ngTemplateOutlet]=\"itemTemplate\"\r\n                [ngTemplateOutletContext]=\"{ $implicit: item }\"\r\n              />\r\n            }\r\n          </div>\r\n        </div>\r\n      </div>\r\n    }\r\n  `,\r\n  styleUrl: './accordion.component.scss',\r\n})\r\nexport class UiAccordionComponent implements DoCheck {\r\n  private readonly cdr = inject(ChangeDetectorRef);\r\n\r\n  /**\r\n   * Fingerprint dell'ultimo stato dei descriptor.\r\n   * Usato da DoCheck per rilevare mutazioni in-place sulle proprieta\r\n   * degli item senza richiedere un nuovo riferimento dell'array.\r\n   */\r\n  private _lastFingerprint = '';\r\n\r\n  /**\r\n   * Array di descriptor dei pannelli.\r\n   * Pannelli con `hidden: true` vengono filtrati automaticamente.\r\n   *\r\n   * Supporta sia aggiornamenti immutabili (nuovo array) sia\r\n   * mutazioni in-place delle proprieta dei descriptor.\r\n   */\r\n  @Input() items: UiAccordionDescriptor[] = [];\r\n\r\n  /**\r\n   * Template per il contenuto di ogni pannello.\r\n   * Il contesto fornisce il descriptor come `$implicit`.\r\n   *\r\n   * @example\r\n   * ```html\r\n   * <ng-template #tpl let-item>\r\n   *   <p>Contenuto per: {{ item.title }}</p>\r\n   * </ng-template>\r\n   * ```\r\n   */\r\n  @Input() itemTemplate: TemplateRef<unknown> | null = null;\r\n\r\n  /**\r\n   * Template opzionale per contenuto aggiuntivo nell'header.\r\n   * Il contesto fornisce il descriptor come `$implicit`.\r\n   * Viene renderizzato tra il titolo e il chevron.\r\n   */\r\n  @Input() headerTemplate: TemplateRef<unknown> | null = null;\r\n\r\n  /**\r\n   * Modalita di espansione.\r\n   * - `multi`: piu pannelli aperti contemporaneamente\r\n   * - `single`: un solo pannello aperto alla volta\r\n   */\r\n  @Input() mode: UiAccordionMode = 'multi';\r\n\r\n  /** Label accessibile per il gruppo accordion. */\r\n  @Input() ariaLabel = 'Accordion';\r\n\r\n  /** Emesso quando un pannello viene aperto o chiuso. */\r\n  @Output() itemToggled = new EventEmitter<UiAccordionToggleEvent>();\r\n\r\n  /**\r\n   * Rileva mutazioni in-place sulle proprieta dei descriptor.\r\n   * Confronta un fingerprint leggero dello stato corrente con quello\r\n   * precedente e, se diverso, forza un ciclo di change detection.\r\n   *\r\n   * Questo permette al consumatore di fare sia:\r\n   * - `items[2].hidden = true`  (mutazione in-place)\r\n   * - `items = [...newItems]`   (nuovo riferimento)\r\n   */\r\n  ngDoCheck(): void {\r\n    const fp = this.computeFingerprint();\r\n    if (fp !== this._lastFingerprint) {\r\n      this._lastFingerprint = fp;\r\n      this.cdr.markForCheck();\r\n    }\r\n  }\r\n\r\n  /** Pannelli visibili (non hidden). */\r\n  get visibleItems(): UiAccordionDescriptor[] {\r\n    return this.items.filter((i) => !i.hidden);\r\n  }\r\n\r\n  /** Alterna l'espansione di un pannello. */\r\n  toggle(item: UiAccordionDescriptor): void {\r\n    if (item.disabled) return;\r\n\r\n    const willExpand = !item.expanded;\r\n\r\n    // In modalita single, chiudi tutti gli altri\r\n    if (willExpand && this.mode === 'single') {\r\n      this.items.forEach((i) => {\r\n        if (i !== item && i.expanded) {\r\n          i.expanded = false;\r\n          this.itemToggled.emit({ item: i, expanded: false });\r\n        }\r\n      });\r\n    }\r\n\r\n    item.expanded = willExpand;\r\n    this.itemToggled.emit({ item, expanded: willExpand });\r\n  }\r\n\r\n  /** Apre un pannello specifico per id. */\r\n  open(id: string): void {\r\n    const item = this.items.find((i) => i.id === id);\r\n    if (item && !item.expanded && !item.disabled) {\r\n      this.toggle(item);\r\n    }\r\n  }\r\n\r\n  /** Chiude un pannello specifico per id. */\r\n  close(id: string): void {\r\n    const item = this.items.find((i) => i.id === id);\r\n    if (item && item.expanded) {\r\n      item.expanded = false;\r\n      this.itemToggled.emit({ item, expanded: false });\r\n    }\r\n  }\r\n\r\n  /** Apre tutti i pannelli (solo in modalita multi). */\r\n  openAll(): void {\r\n    if (this.mode === 'single') return;\r\n    this.items.forEach((item) => {\r\n      if (!item.expanded && !item.disabled && !item.hidden) {\r\n        item.expanded = true;\r\n        this.itemToggled.emit({ item, expanded: true });\r\n      }\r\n    });\r\n  }\r\n\r\n  /** Chiude tutti i pannelli. */\r\n  closeAll(): void {\r\n    this.items.forEach((item) => {\r\n      if (item.expanded) {\r\n        item.expanded = false;\r\n        this.itemToggled.emit({ item, expanded: false });\r\n      }\r\n    });\r\n  }\r\n\r\n  /** @internal Classi CSS per il pannello. */\r\n  getPanelClasses(item: UiAccordionDescriptor): string {\r\n    return [\r\n      'ui-accordion__panel',\r\n      item.expanded ? 'ui-accordion__panel--expanded' : '',\r\n      item.disabled ? 'ui-accordion__panel--disabled' : '',\r\n    ]\r\n      .filter(Boolean)\r\n      .join(' ');\r\n  }\r\n\r\n  /** @internal Sposta il focus al primo header. */\r\n  focusFirst(event: Event): void {\r\n    event.preventDefault();\r\n    const headers = this.getHeaders();\r\n    headers[0]?.focus();\r\n  }\r\n\r\n  /** @internal Sposta il focus all'ultimo header. */\r\n  focusLast(event: Event): void {\r\n    event.preventDefault();\r\n    const headers = this.getHeaders();\r\n    headers[headers.length - 1]?.focus();\r\n  }\r\n\r\n  /** @internal Recupera tutti gli header button del DOM. */\r\n  private getHeaders(): HTMLButtonElement[] {\r\n    return Array.from(document.querySelectorAll<HTMLButtonElement>('.ui-accordion__header:not(:disabled)'));\r\n  }\r\n\r\n  /**\r\n   * @internal Calcola un fingerprint leggero basato sulle proprieta\r\n   * reattive dei descriptor (quelle che influenzano il rendering).\r\n   */\r\n  private computeFingerprint(): string {\r\n    return this.items.map((i) => `${i.id}:${i.hidden ?? 0}:${i.disabled ?? 0}:${i.expanded ?? 0}:${i.title}`).join('|');\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,6 @@
1
+ /**
2
+ * @module ng-ui-system/accordion
3
+ * Tipi e interfacce per UiAccordion.
4
+ */
5
+ export {};
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNjb3JkaW9uLnR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvbmctdWktc3lzdGVtL3NyYy9saWIvY29tcG9uZW50cy9hY2NvcmRpb24vYWNjb3JkaW9uLnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQG1vZHVsZSBuZy11aS1zeXN0ZW0vYWNjb3JkaW9uXG4gKiBUaXBpIGUgaW50ZXJmYWNjZSBwZXIgVWlBY2NvcmRpb24uXG4gKi9cblxuaW1wb3J0IHsgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFVpSWNvbk5hbWUgfSBmcm9tICcuLi8uLi9jb3JlL3R5cGVzJztcblxuLy8gUmUtZXhwb3J0IHBlciBjb21vZGl0YSBkZWwgY29uc3VtYXRvcmVcbmV4cG9ydCB7IFVpSWNvbk5hbWUgfTtcblxuLyoqXG4gKiBEZXNjcmlwdG9yIHBlciB1biBzaW5nb2xvIHBhbm5lbGxvIGRlbGwnYWNjb3JkaW9uLlxuICpcbiAqIENvbnNlbnRlIGRpIGRlZmluaXJlIGFjY29yZGlvbiBjb21lIHN0cnV0dHVyZSBkYXRpIGRpY2hpYXJhdGl2ZSxcbiAqIGFnZ2lvcm5hYmlsaSBkaW5hbWljYW1lbnRlIHRyYW1pdGUgbWFuaXBvbGF6aW9uZSBkZWxsJ2FycmF5LlxuICpcbiAqIEB1c2FnZU5vdGVzXG4gKiAjIyMgVXRpbGl6em8gYmFzZVxuICogYGBgdHlwZXNjcmlwdFxuICogY29uc3QgaXRlbXM6IFVpQWNjb3JkaW9uRGVzY3JpcHRvcltdID0gW1xuICogICB7IGlkOiAnaW5mbycsIHRpdGxlOiAnSW5mb3JtYXppb25pIGdlbmVyYWxpJywgaWNvbjogJ2luZm8nIH0sXG4gKiAgIHsgaWQ6ICdhZGRyZXNzJywgdGl0bGU6ICdJbmRpcml6em8nLCBpY29uOiAnbWFwLXBpbicgfSxcbiAqICAgeyBpZDogJ25vdGVzJywgdGl0bGU6ICdOb3RlIGFnZ2l1bnRpdmUnLCBleHBhbmRlZDogdHJ1ZSB9LFxuICogXTtcbiAqIGBgYFxuICpcbiAqICMjIyBWaXNpYmlsaXRhIGNvbmRpemlvbmFsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaXRlbXNbMl0uaGlkZGVuID0gIXRoaXMuaGFzTm90ZXM7XG4gKiBpdGVtc1sxXS5kaXNhYmxlZCA9IHRoaXMuaXNSZWFkb25seTtcbiAqIGBgYFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFVpQWNjb3JkaW9uRGVzY3JpcHRvciB7XG4gIC8qKiBJZGVudGlmaWNhdG9yZSB1bml2b2NvIGRlbCBwYW5uZWxsby4gKi9cbiAgaWQ6IHN0cmluZztcblxuICAvKiogVGl0b2xvIHZpc3VhbGl6emF0byBuZWxsJ2hlYWRlciBkZWwgcGFubmVsbG8uICovXG4gIHRpdGxlOiBzdHJpbmc7XG5cbiAgLyoqIEljb25hIEx1Y2lkZSBvcHppb25hbGUgbW9zdHJhdGEgcHJpbWEgZGVsIHRpdG9sby4gQHNlZSBodHRwczovL2x1Y2lkZS5kZXYvaWNvbnMgKi9cbiAgaWNvbj86IFVpSWNvbk5hbWU7XG5cbiAgLyoqIFN0YXRvIGluaXppYWxlIGRpIGVzcGFuc2lvbmUuIEBkZWZhdWx0IGZhbHNlICovXG4gIGV4cGFuZGVkPzogYm9vbGVhbjtcblxuICAvKiogRGlzYWJpbGl0YSBsJ2ludGVyYXppb25lIGNvbiBpbCBwYW5uZWxsby4gQGRlZmF1bHQgZmFsc2UgKi9cbiAgZGlzYWJsZWQ/OiBib29sZWFuO1xuXG4gIC8qKiBOYXNjb25kZSBpbCBwYW5uZWxsbyBkYWxsJ2ludGVyZmFjY2lhLiBAZGVmYXVsdCBmYWxzZSAqL1xuICBoaWRkZW4/OiBib29sZWFuO1xuXG4gIC8qKiBEZXNjcml6aW9uZSBzZWNvbmRhcmlhIG1vc3RyYXRhIGFjY2FudG8gYWwgdGl0b2xvLiAqL1xuICBzdWJ0aXRsZT86IHN0cmluZztcbn1cblxuLyoqXG4gKiBFdmVudG8gZW1lc3NvIHF1YW5kbyB1biBwYW5uZWxsbyB2aWVuZSBhcGVydG8gbyBjaGl1c28uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVWlBY2NvcmRpb25Ub2dnbGVFdmVudCB7XG4gIC8qKiBEZXNjcmlwdG9yIGRlbCBwYW5uZWxsbyBjb2ludm9sdG8uICovXG4gIGl0ZW06IFVpQWNjb3JkaW9uRGVzY3JpcHRvcjtcblxuICAvKiogYHRydWVgIHNlIGlsIHBhbm5lbGxvIGUgc3RhdG8gYXBlcnRvLCBgZmFsc2VgIHNlIGNoaXVzby4gKi9cbiAgZXhwYW5kZWQ6IGJvb2xlYW47XG59XG5cbi8qKlxuICogTW9kYWxpdGEgZGkgZXNwYW5zaW9uZSBkZWxsJ2FjY29yZGlvbi5cbiAqXG4gKiB8IE1vZGFsaXRhIHwgQ29tcG9ydGFtZW50byAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfFxuICogfC0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18XG4gKiB8IGBtdWx0aWAgfCBQaXUgcGFubmVsbGkgcG9zc29ubyBlc3NlcmUgYXBlcnRpIGNvbnRlbXBvcmFuZWFtZW50ZSB8XG4gKiB8IGBzaW5nbGVgfCBVbiBzb2xvIHBhbm5lbGxvIGFwZXJ0byBhbGxhIHZvbHRhIChnbGkgYWx0cmkgc2kgY2hpdWRvbm8pIHxcbiAqL1xuZXhwb3J0IHR5cGUgVWlBY2NvcmRpb25Nb2RlID0gJ211bHRpJyB8ICdzaW5nbGUnO1xuIl19
@@ -0,0 +1,2 @@
1
+ export { UiAccordionComponent } from './accordion.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9uZy11aS1zeXN0ZW0vc3JjL2xpYi9jb21wb25lbnRzL2FjY29yZGlvbi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFVpQWNjb3JkaW9uQ29tcG9uZW50IH0gZnJvbSAnLi9hY2NvcmRpb24uY29tcG9uZW50JztcclxuZXhwb3J0IHtcclxuICBVaUFjY29yZGlvbkRlc2NyaXB0b3IsXHJcbiAgVWlBY2NvcmRpb25Ub2dnbGVFdmVudCxcclxuICBVaUFjY29yZGlvbk1vZGUsXHJcbn0gZnJvbSAnLi9hY2NvcmRpb24udHlwZXMnO1xyXG4iXX0=