@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.
- package/esm2022/gnggln-ng-ui-system.mjs +5 -0
- package/esm2022/lib/components/accordion/accordion.component.mjs +353 -0
- package/esm2022/lib/components/accordion/accordion.types.mjs +6 -0
- package/esm2022/lib/components/accordion/index.mjs +2 -0
- package/esm2022/lib/components/base-layout/base-layout.component.mjs +218 -0
- package/esm2022/lib/components/base-layout/base-layout.types.mjs +6 -0
- package/esm2022/lib/components/base-layout/index.mjs +14 -0
- package/esm2022/lib/components/button/button-area.component.mjs +196 -0
- package/esm2022/lib/components/button/button.component.mjs +164 -0
- package/esm2022/lib/components/button/button.types.mjs +6 -0
- package/esm2022/lib/components/button/index.mjs +16 -0
- package/esm2022/lib/components/crud-table/crud-table.component.mjs +789 -0
- package/esm2022/lib/components/crud-table/crud-table.types.mjs +6 -0
- package/esm2022/lib/components/crud-table/index.mjs +16 -0
- package/esm2022/lib/components/form-builder/adapters/it-date-adapter.mjs +82 -0
- package/esm2022/lib/components/form-builder/directives/currency-input.directive.mjs +184 -0
- package/esm2022/lib/components/form-builder/form-builder.component.mjs +824 -0
- package/esm2022/lib/components/form-builder/form-wizard.component.mjs +510 -0
- package/esm2022/lib/components/form-builder/index.mjs +19 -0
- package/esm2022/lib/components/form-builder/services/form-condition.service.mjs +132 -0
- package/esm2022/lib/components/form-builder/services/form-validation.service.mjs +381 -0
- package/esm2022/lib/components/form-builder/services/location.service.mjs +140 -0
- package/esm2022/lib/components/form-builder/services/wizard-sync.service.mjs +84 -0
- package/esm2022/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.mjs +161 -0
- package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs +310 -0
- package/esm2022/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.mjs +648 -0
- package/esm2022/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.mjs +432 -0
- package/esm2022/lib/components/form-builder/types/condition.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/field.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/index.mjs +2 -0
- package/esm2022/lib/components/form-builder/types/schema.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/territoriale.types.mjs +6 -0
- package/esm2022/lib/components/form-builder/types/validation.types.mjs +6 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.component.mjs +730 -0
- package/esm2022/lib/components/form-builder-editor/form-builder-editor.service.mjs +56 -0
- package/esm2022/lib/components/form-builder-editor/index.mjs +21 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-persistence.service.mjs +190 -0
- package/esm2022/lib/components/form-builder-editor/services/editor-state.service.mjs +324 -0
- package/esm2022/lib/components/form-builder-editor/services/field-factory.service.mjs +188 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.mjs +667 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.mjs +317 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.mjs +611 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.mjs +267 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.mjs +276 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.mjs +323 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.mjs +238 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.mjs +472 -0
- package/esm2022/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.mjs +473 -0
- package/esm2022/lib/components/form-builder-editor/types/editor.types.mjs +6 -0
- package/esm2022/lib/components/layout-builder/index.mjs +18 -0
- package/esm2022/lib/components/layout-builder/layout-builder.component.mjs +1730 -0
- package/esm2022/lib/components/layout-builder/layout-builder.types.mjs +9 -0
- package/esm2022/lib/components/layout-builder/layout.service.mjs +239 -0
- package/esm2022/lib/components/modal/confirm-dialog.component.mjs +151 -0
- package/esm2022/lib/components/modal/index.mjs +4 -0
- package/esm2022/lib/components/modal/modal.component.mjs +139 -0
- package/esm2022/lib/components/modal/modal.service.mjs +194 -0
- package/esm2022/lib/components/modal/modal.types.mjs +6 -0
- package/esm2022/lib/components/page-header/breadcrumb.service.mjs +242 -0
- package/esm2022/lib/components/page-header/index.mjs +20 -0
- package/esm2022/lib/components/page-header/page-header.component.mjs +243 -0
- package/esm2022/lib/components/page-header/page-header.types.mjs +21 -0
- package/esm2022/lib/components/table/index.mjs +2 -0
- package/esm2022/lib/components/table/paginated-table.component.mjs +407 -0
- package/esm2022/lib/components/table/table.types.mjs +6 -0
- package/esm2022/lib/core/types/index.mjs +6 -0
- package/esm2022/lib/core/utils/index.mjs +53 -0
- package/esm2022/lib/sources/location-data.opt.json +8942 -0
- package/esm2022/lib/sources/nazioni.opt.json +215 -0
- package/esm2022/public-api.mjs +34 -0
- package/fesm2022/gnggln-ng-ui-system.mjs +55752 -0
- package/fesm2022/gnggln-ng-ui-system.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/accordion/accordion.component.d.ts +118 -0
- package/lib/components/accordion/accordion.types.d.ts +62 -0
- package/lib/components/accordion/index.d.ts +2 -0
- package/lib/components/base-layout/base-layout.component.d.ts +83 -0
- package/lib/components/base-layout/base-layout.types.d.ts +26 -0
- package/lib/components/base-layout/index.d.ts +13 -0
- package/lib/components/button/button-area.component.d.ts +88 -0
- package/lib/components/button/button.component.d.ts +55 -0
- package/lib/components/button/button.types.d.ts +70 -0
- package/lib/components/button/index.d.ts +15 -0
- package/lib/components/crud-table/crud-table.component.d.ts +143 -0
- package/lib/components/crud-table/crud-table.types.d.ts +207 -0
- package/lib/components/crud-table/index.d.ts +15 -0
- package/lib/components/form-builder/adapters/it-date-adapter.d.ts +32 -0
- package/lib/components/form-builder/directives/currency-input.directive.d.ts +48 -0
- package/lib/components/form-builder/form-builder.component.d.ts +183 -0
- package/lib/components/form-builder/form-wizard.component.d.ts +87 -0
- package/lib/components/form-builder/index.d.ts +13 -0
- package/lib/components/form-builder/services/form-condition.service.d.ts +46 -0
- package/lib/components/form-builder/services/form-validation.service.d.ts +63 -0
- package/lib/components/form-builder/services/location.service.d.ts +83 -0
- package/lib/components/form-builder/services/wizard-sync.service.d.ts +63 -0
- package/lib/components/form-builder/sub-components/error-summary/form-error-summary.component.d.ts +28 -0
- package/lib/components/form-builder/sub-components/file-input/file-input.component.d.ts +41 -0
- package/lib/components/form-builder/sub-components/specifica-territoriale/specifica-territoriale.component.d.ts +145 -0
- package/lib/components/form-builder/sub-components/table-territoriale/table-territoriale.component.d.ts +108 -0
- package/lib/components/form-builder/types/condition.types.d.ts +51 -0
- package/lib/components/form-builder/types/field.types.d.ts +288 -0
- package/lib/components/form-builder/types/index.d.ts +5 -0
- package/lib/components/form-builder/types/schema.types.d.ts +227 -0
- package/lib/components/form-builder/types/territoriale.types.d.ts +170 -0
- package/lib/components/form-builder/types/validation.types.d.ts +174 -0
- package/lib/components/form-builder-editor/form-builder-editor.component.d.ts +117 -0
- package/lib/components/form-builder-editor/form-builder-editor.service.d.ts +38 -0
- package/lib/components/form-builder-editor/index.d.ts +15 -0
- package/lib/components/form-builder-editor/services/editor-persistence.service.d.ts +42 -0
- package/lib/components/form-builder-editor/services/editor-state.service.d.ts +66 -0
- package/lib/components/form-builder-editor/services/field-factory.service.d.ts +28 -0
- package/lib/components/form-builder-editor/sub-components/condition-editor/condition-editor.component.d.ts +139 -0
- package/lib/components/form-builder-editor/sub-components/editor-toolbar/editor-toolbar.component.d.ts +43 -0
- package/lib/components/form-builder-editor/sub-components/field-config-panel/field-config-panel.component.d.ts +83 -0
- package/lib/components/form-builder-editor/sub-components/field-palette/field-palette.component.d.ts +40 -0
- package/lib/components/form-builder-editor/sub-components/form-values-panel/form-values-panel.component.d.ts +51 -0
- package/lib/components/form-builder-editor/sub-components/options-editor/options-editor.component.d.ts +63 -0
- package/lib/components/form-builder-editor/sub-components/preview-container/preview-container.component.d.ts +68 -0
- package/lib/components/form-builder-editor/sub-components/section-editor/section-editor.component.d.ts +82 -0
- package/lib/components/form-builder-editor/sub-components/validation-editor/validation-editor.component.d.ts +112 -0
- package/lib/components/form-builder-editor/types/editor.types.d.ts +124 -0
- package/lib/components/layout-builder/index.d.ts +16 -0
- package/lib/components/layout-builder/layout-builder.component.d.ts +85 -0
- package/lib/components/layout-builder/layout-builder.types.d.ts +436 -0
- package/lib/components/layout-builder/layout.service.d.ts +100 -0
- package/lib/components/modal/confirm-dialog.component.d.ts +46 -0
- package/lib/components/modal/index.d.ts +4 -0
- package/lib/components/modal/modal.component.d.ts +44 -0
- package/lib/components/modal/modal.service.d.ts +93 -0
- package/lib/components/modal/modal.types.d.ts +110 -0
- package/lib/components/page-header/breadcrumb.service.d.ts +96 -0
- package/lib/components/page-header/index.d.ts +16 -0
- package/lib/components/page-header/page-header.component.d.ts +59 -0
- package/lib/components/page-header/page-header.types.d.ts +96 -0
- package/lib/components/table/index.d.ts +2 -0
- package/lib/components/table/paginated-table.component.d.ts +85 -0
- package/lib/components/table/table.types.d.ts +81 -0
- package/lib/core/types/index.d.ts +57 -0
- package/lib/core/utils/index.d.ts +29 -0
- package/package.json +44 -0
- 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=
|