@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
package/esm2022/lib/components/form-builder/sub-components/file-input/file-input.component.mjs
ADDED
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, forwardRef, ChangeDetectionStrategy, ViewEncapsulation, signal, } from '@angular/core';
|
|
2
|
+
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
3
|
+
import { LucideAngularModule } from 'lucide-angular';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "lucide-angular";
|
|
6
|
+
/**
|
|
7
|
+
* Componente file input standalone con drag & drop,
|
|
8
|
+
* anteprima, validazione e supporto multiplo.
|
|
9
|
+
*
|
|
10
|
+
* Implementa `ControlValueAccessor` per integrazione
|
|
11
|
+
* diretta con Angular Reactive Forms.
|
|
12
|
+
*
|
|
13
|
+
* @selector ui-file-input
|
|
14
|
+
*/
|
|
15
|
+
export class UiFileInputComponent {
|
|
16
|
+
constructor() {
|
|
17
|
+
/** File rimosso dalla lista. */
|
|
18
|
+
this.fileRemoved = new EventEmitter();
|
|
19
|
+
/** Errore di validazione file. */
|
|
20
|
+
this.validationError = new EventEmitter();
|
|
21
|
+
// ── Stato interno (signals) ──
|
|
22
|
+
this.files = signal([]);
|
|
23
|
+
this.disabled = signal(false);
|
|
24
|
+
this.isDragOver = signal(false);
|
|
25
|
+
this.onChange = () => { };
|
|
26
|
+
this.onTouched = () => { };
|
|
27
|
+
}
|
|
28
|
+
/** Stringa accept per l'input nativo. */
|
|
29
|
+
get acceptString() {
|
|
30
|
+
return this.config?.acceptedTypes?.join(',') || '';
|
|
31
|
+
}
|
|
32
|
+
// ── ControlValueAccessor ──
|
|
33
|
+
writeValue(value) {
|
|
34
|
+
this.files.set(Array.isArray(value) ? value : []);
|
|
35
|
+
}
|
|
36
|
+
registerOnChange(fn) {
|
|
37
|
+
this.onChange = fn;
|
|
38
|
+
}
|
|
39
|
+
registerOnTouched(fn) {
|
|
40
|
+
this.onTouched = fn;
|
|
41
|
+
}
|
|
42
|
+
setDisabledState(isDisabled) {
|
|
43
|
+
this.disabled.set(isDisabled);
|
|
44
|
+
}
|
|
45
|
+
// ── Drag & drop ──
|
|
46
|
+
onDragOver(event) {
|
|
47
|
+
event.preventDefault();
|
|
48
|
+
event.stopPropagation();
|
|
49
|
+
this.isDragOver.set(true);
|
|
50
|
+
}
|
|
51
|
+
onDragLeave() {
|
|
52
|
+
this.isDragOver.set(false);
|
|
53
|
+
}
|
|
54
|
+
onDrop(event) {
|
|
55
|
+
event.preventDefault();
|
|
56
|
+
event.stopPropagation();
|
|
57
|
+
this.isDragOver.set(false);
|
|
58
|
+
if (this.disabled())
|
|
59
|
+
return;
|
|
60
|
+
const droppedFiles = event.dataTransfer?.files;
|
|
61
|
+
if (droppedFiles?.length) {
|
|
62
|
+
this.processFiles(droppedFiles);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
onFileSelected(event) {
|
|
66
|
+
const input = event.target;
|
|
67
|
+
if (input.files?.length) {
|
|
68
|
+
this.processFiles(input.files);
|
|
69
|
+
input.value = ''; // Reset per permettere la ri-selezione dello stesso file
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
removeFile(index) {
|
|
73
|
+
const current = [...this.files()];
|
|
74
|
+
const removed = current.splice(index, 1);
|
|
75
|
+
this.files.set(current);
|
|
76
|
+
this.onChange(current);
|
|
77
|
+
if (removed[0])
|
|
78
|
+
this.fileRemoved.emit(removed[0]);
|
|
79
|
+
}
|
|
80
|
+
formatSize(bytes) {
|
|
81
|
+
if (bytes === 0)
|
|
82
|
+
return '0 B';
|
|
83
|
+
const k = 1024;
|
|
84
|
+
const sizes = ['B', 'KB', 'MB', 'GB'];
|
|
85
|
+
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
86
|
+
return parseFloat((bytes / Math.pow(k, i)).toFixed(1)) + ' ' + sizes[i];
|
|
87
|
+
}
|
|
88
|
+
// ── Processing ──
|
|
89
|
+
async processFiles(fileList) {
|
|
90
|
+
this.onTouched();
|
|
91
|
+
const maxFiles = this.config?.maxFiles || Infinity;
|
|
92
|
+
const current = this.config?.multiple ? [...this.files()] : [];
|
|
93
|
+
const remaining = maxFiles - current.length;
|
|
94
|
+
if (remaining <= 0) {
|
|
95
|
+
this.validationError.emit(`Numero massimo di file raggiunto (${maxFiles})`);
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
const filesToProcess = Array.from(fileList).slice(0, remaining);
|
|
99
|
+
for (const file of filesToProcess) {
|
|
100
|
+
// Validazione dimensione
|
|
101
|
+
if (this.config?.maxFileSize && file.size > this.config.maxFileSize) {
|
|
102
|
+
this.validationError.emit(`Il file "${file.name}" supera la dimensione massima consentita`);
|
|
103
|
+
continue;
|
|
104
|
+
}
|
|
105
|
+
// Validazione tipo
|
|
106
|
+
if (this.config?.acceptedTypes?.length) {
|
|
107
|
+
const isAccepted = this.config.acceptedTypes.some((type) => {
|
|
108
|
+
if (type.endsWith('/*')) {
|
|
109
|
+
return file.type.startsWith(type.split('/')[0] + '/');
|
|
110
|
+
}
|
|
111
|
+
return file.type === type;
|
|
112
|
+
});
|
|
113
|
+
if (!isAccepted) {
|
|
114
|
+
this.validationError.emit(`Il formato del file "${file.name}" non e accettato`);
|
|
115
|
+
continue;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
const content = new Uint8Array(await file.arrayBuffer());
|
|
119
|
+
const ext = file.name.split('.').pop() || '';
|
|
120
|
+
const fileData = {
|
|
121
|
+
filename: file.name,
|
|
122
|
+
content,
|
|
123
|
+
mimeType: file.type,
|
|
124
|
+
extension: ext,
|
|
125
|
+
size: file.size,
|
|
126
|
+
needsUpload: true,
|
|
127
|
+
id: `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`, // NOSONAR - Used for non-cryptographic file ID generation in UI
|
|
128
|
+
timestamp: new Date(),
|
|
129
|
+
};
|
|
130
|
+
// Genera preview per immagini
|
|
131
|
+
if (this.config?.showPreview && file.type.startsWith('image/')) {
|
|
132
|
+
fileData.previewUrl = URL.createObjectURL(file);
|
|
133
|
+
}
|
|
134
|
+
current.push(fileData);
|
|
135
|
+
}
|
|
136
|
+
this.files.set(current);
|
|
137
|
+
this.onChange(current);
|
|
138
|
+
}
|
|
139
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFileInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
140
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UiFileInputComponent, isStandalone: true, selector: "ui-file-input", inputs: { config: "config" }, outputs: { fileRemoved: "fileRemoved", validationError: "validationError" }, providers: [
|
|
141
|
+
{
|
|
142
|
+
provide: NG_VALUE_ACCESSOR,
|
|
143
|
+
useExisting: forwardRef(() => UiFileInputComponent),
|
|
144
|
+
multi: true,
|
|
145
|
+
},
|
|
146
|
+
], ngImport: i0, template: `
|
|
147
|
+
<div class="ui-file-input" [class.ui-file-input--disabled]="disabled()" [class.ui-file-input--drag-over]="isDragOver()">
|
|
148
|
+
<!-- Area drag & drop -->
|
|
149
|
+
@if (config?.enableDragDrop !== false) {
|
|
150
|
+
<div
|
|
151
|
+
class="ui-file-input__drop-zone"
|
|
152
|
+
(dragover)="onDragOver($event)"
|
|
153
|
+
(dragleave)="onDragLeave()"
|
|
154
|
+
(drop)="onDrop($event)"
|
|
155
|
+
(click)="fileInput.click()"
|
|
156
|
+
role="button"
|
|
157
|
+
tabindex="0"
|
|
158
|
+
[attr.aria-label]="config?.dragDropText || 'Trascina i file qui o clicca per selezionare'"
|
|
159
|
+
(keydown.enter)="fileInput.click()"
|
|
160
|
+
(keydown.space)="fileInput.click()"
|
|
161
|
+
>
|
|
162
|
+
<lucide-icon name="upload" [size]="24" aria-hidden="true" />
|
|
163
|
+
<span class="ui-file-input__drop-text">
|
|
164
|
+
{{ config?.dragDropText || 'Trascina i file qui o clicca per selezionare' }}
|
|
165
|
+
</span>
|
|
166
|
+
@if (config?.acceptedTypes?.length) {
|
|
167
|
+
<span class="ui-file-input__hint">
|
|
168
|
+
Formati: {{ config!.acceptedTypes!.join(', ') }}
|
|
169
|
+
</span>
|
|
170
|
+
}
|
|
171
|
+
</div>
|
|
172
|
+
} @else {
|
|
173
|
+
<button
|
|
174
|
+
class="ui-file-input__select-btn"
|
|
175
|
+
type="button"
|
|
176
|
+
[disabled]="disabled()"
|
|
177
|
+
(click)="fileInput.click()"
|
|
178
|
+
>
|
|
179
|
+
<lucide-icon name="paperclip" [size]="16" aria-hidden="true" />
|
|
180
|
+
{{ config?.placeholder || 'Seleziona file' }}
|
|
181
|
+
</button>
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
<input
|
|
185
|
+
#fileInput
|
|
186
|
+
type="file"
|
|
187
|
+
class="ui-file-input__native"
|
|
188
|
+
[accept]="acceptString"
|
|
189
|
+
[multiple]="config?.multiple ?? false"
|
|
190
|
+
[disabled]="disabled()"
|
|
191
|
+
(change)="onFileSelected($event)"
|
|
192
|
+
/>
|
|
193
|
+
|
|
194
|
+
<!-- Lista file selezionati -->
|
|
195
|
+
@if (files().length > 0) {
|
|
196
|
+
<div class="ui-file-input__file-list" role="list">
|
|
197
|
+
@for (file of files(); track file.id || $index) {
|
|
198
|
+
<div class="ui-file-input__file-item" role="listitem">
|
|
199
|
+
<div class="ui-file-input__file-info">
|
|
200
|
+
<lucide-icon name="file" [size]="16" aria-hidden="true" />
|
|
201
|
+
<span class="ui-file-input__file-name">{{ file.filename }}</span>
|
|
202
|
+
<span class="ui-file-input__file-size">{{ formatSize(file.size) }}</span>
|
|
203
|
+
</div>
|
|
204
|
+
@if (!disabled()) {
|
|
205
|
+
<button
|
|
206
|
+
class="ui-file-input__remove-btn"
|
|
207
|
+
type="button"
|
|
208
|
+
[attr.aria-label]="'Rimuovi ' + file.filename"
|
|
209
|
+
(click)="removeFile($index)"
|
|
210
|
+
>
|
|
211
|
+
<lucide-icon name="x" [size]="14" aria-hidden="true" />
|
|
212
|
+
</button>
|
|
213
|
+
}
|
|
214
|
+
</div>
|
|
215
|
+
}
|
|
216
|
+
</div>
|
|
217
|
+
}
|
|
218
|
+
</div>
|
|
219
|
+
`, isInline: true, styles: [".ui-file-input{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-file-input--disabled{opacity:.55;pointer-events:none}.ui-file-input__native{display:none}.ui-file-input__drop-zone{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-6) var(--ui-spacing-4);border:2px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);cursor:pointer;transition:border-color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-file-input__drop-zone:focus{outline:none}.ui-file-input__drop-zone:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__drop-zone:hover{border-color:var(--ui-color-primary);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .04))}.ui-file-input--drag-over .ui-file-input__drop-zone{border-color:var(--ui-color-primary);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .08))}.ui-file-input__drop-zone lucide-icon{color:var(--ui-color-text-muted)}.ui-file-input__drop-text{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);text-align:center}.ui-file-input__hint{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted)}.ui-file-input__select-btn{appearance:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-surface);padding:var(--ui-spacing-2) var(--ui-spacing-3);display:inline-flex;align-items:center;gap:var(--ui-spacing-2);cursor:pointer;font-family:var(--ui-font-family);font-size:var(--ui-font-size-sm);color:var(--ui-color-text);transition:background-color var(--ui-transition-fast)}.ui-file-input__select-btn:focus{outline:none}.ui-file-input__select-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__select-btn:hover:not(:disabled){background:var(--ui-color-surface-hover)}.ui-file-input__file-list{display:flex;flex-direction:column;gap:var(--ui-spacing-1)}.ui-file-input__file-item{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2) var(--ui-spacing-3);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-surface)}.ui-file-input__file-info{display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-file-input__file-info lucide-icon{flex-shrink:0;color:var(--ui-color-text-muted)}.ui-file-input__file-name{font-size:var(--ui-font-size-sm);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-file-input__file-size{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);flex-shrink:0}.ui-file-input__remove-btn{appearance:none;border:none;background:transparent;cursor:pointer;padding:var(--ui-spacing-1);border-radius:var(--ui-radius-sm);color:var(--ui-color-text-muted);transition:color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-file-input__remove-btn:focus{outline:none}.ui-file-input__remove-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__remove-btn:hover{color:var(--ui-color-error, #dc2626);background:var(--ui-color-error-subtle, #fef2f2)}\n"], dependencies: [{ 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 }); }
|
|
220
|
+
}
|
|
221
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UiFileInputComponent, decorators: [{
|
|
222
|
+
type: Component,
|
|
223
|
+
args: [{ selector: 'ui-file-input', standalone: true, imports: [LucideAngularModule], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, providers: [
|
|
224
|
+
{
|
|
225
|
+
provide: NG_VALUE_ACCESSOR,
|
|
226
|
+
useExisting: forwardRef(() => UiFileInputComponent),
|
|
227
|
+
multi: true,
|
|
228
|
+
},
|
|
229
|
+
], template: `
|
|
230
|
+
<div class="ui-file-input" [class.ui-file-input--disabled]="disabled()" [class.ui-file-input--drag-over]="isDragOver()">
|
|
231
|
+
<!-- Area drag & drop -->
|
|
232
|
+
@if (config?.enableDragDrop !== false) {
|
|
233
|
+
<div
|
|
234
|
+
class="ui-file-input__drop-zone"
|
|
235
|
+
(dragover)="onDragOver($event)"
|
|
236
|
+
(dragleave)="onDragLeave()"
|
|
237
|
+
(drop)="onDrop($event)"
|
|
238
|
+
(click)="fileInput.click()"
|
|
239
|
+
role="button"
|
|
240
|
+
tabindex="0"
|
|
241
|
+
[attr.aria-label]="config?.dragDropText || 'Trascina i file qui o clicca per selezionare'"
|
|
242
|
+
(keydown.enter)="fileInput.click()"
|
|
243
|
+
(keydown.space)="fileInput.click()"
|
|
244
|
+
>
|
|
245
|
+
<lucide-icon name="upload" [size]="24" aria-hidden="true" />
|
|
246
|
+
<span class="ui-file-input__drop-text">
|
|
247
|
+
{{ config?.dragDropText || 'Trascina i file qui o clicca per selezionare' }}
|
|
248
|
+
</span>
|
|
249
|
+
@if (config?.acceptedTypes?.length) {
|
|
250
|
+
<span class="ui-file-input__hint">
|
|
251
|
+
Formati: {{ config!.acceptedTypes!.join(', ') }}
|
|
252
|
+
</span>
|
|
253
|
+
}
|
|
254
|
+
</div>
|
|
255
|
+
} @else {
|
|
256
|
+
<button
|
|
257
|
+
class="ui-file-input__select-btn"
|
|
258
|
+
type="button"
|
|
259
|
+
[disabled]="disabled()"
|
|
260
|
+
(click)="fileInput.click()"
|
|
261
|
+
>
|
|
262
|
+
<lucide-icon name="paperclip" [size]="16" aria-hidden="true" />
|
|
263
|
+
{{ config?.placeholder || 'Seleziona file' }}
|
|
264
|
+
</button>
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
<input
|
|
268
|
+
#fileInput
|
|
269
|
+
type="file"
|
|
270
|
+
class="ui-file-input__native"
|
|
271
|
+
[accept]="acceptString"
|
|
272
|
+
[multiple]="config?.multiple ?? false"
|
|
273
|
+
[disabled]="disabled()"
|
|
274
|
+
(change)="onFileSelected($event)"
|
|
275
|
+
/>
|
|
276
|
+
|
|
277
|
+
<!-- Lista file selezionati -->
|
|
278
|
+
@if (files().length > 0) {
|
|
279
|
+
<div class="ui-file-input__file-list" role="list">
|
|
280
|
+
@for (file of files(); track file.id || $index) {
|
|
281
|
+
<div class="ui-file-input__file-item" role="listitem">
|
|
282
|
+
<div class="ui-file-input__file-info">
|
|
283
|
+
<lucide-icon name="file" [size]="16" aria-hidden="true" />
|
|
284
|
+
<span class="ui-file-input__file-name">{{ file.filename }}</span>
|
|
285
|
+
<span class="ui-file-input__file-size">{{ formatSize(file.size) }}</span>
|
|
286
|
+
</div>
|
|
287
|
+
@if (!disabled()) {
|
|
288
|
+
<button
|
|
289
|
+
class="ui-file-input__remove-btn"
|
|
290
|
+
type="button"
|
|
291
|
+
[attr.aria-label]="'Rimuovi ' + file.filename"
|
|
292
|
+
(click)="removeFile($index)"
|
|
293
|
+
>
|
|
294
|
+
<lucide-icon name="x" [size]="14" aria-hidden="true" />
|
|
295
|
+
</button>
|
|
296
|
+
}
|
|
297
|
+
</div>
|
|
298
|
+
}
|
|
299
|
+
</div>
|
|
300
|
+
}
|
|
301
|
+
</div>
|
|
302
|
+
`, styles: [".ui-file-input{display:flex;flex-direction:column;gap:var(--ui-spacing-2)}.ui-file-input--disabled{opacity:.55;pointer-events:none}.ui-file-input__native{display:none}.ui-file-input__drop-zone{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:var(--ui-spacing-2);padding:var(--ui-spacing-6) var(--ui-spacing-4);border:2px dashed var(--ui-color-border);border-radius:var(--ui-radius-lg);background:var(--ui-color-bg-subtle);cursor:pointer;transition:border-color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-file-input__drop-zone:focus{outline:none}.ui-file-input__drop-zone:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__drop-zone:hover{border-color:var(--ui-color-primary);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .04))}.ui-file-input--drag-over .ui-file-input__drop-zone{border-color:var(--ui-color-primary);background:var(--ui-color-primary-subtle, rgba(37, 99, 235, .08))}.ui-file-input__drop-zone lucide-icon{color:var(--ui-color-text-muted)}.ui-file-input__drop-text{font-size:var(--ui-font-size-sm);color:var(--ui-color-text-secondary);text-align:center}.ui-file-input__hint{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted)}.ui-file-input__select-btn{appearance:none;border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-surface);padding:var(--ui-spacing-2) var(--ui-spacing-3);display:inline-flex;align-items:center;gap:var(--ui-spacing-2);cursor:pointer;font-family:var(--ui-font-family);font-size:var(--ui-font-size-sm);color:var(--ui-color-text);transition:background-color var(--ui-transition-fast)}.ui-file-input__select-btn:focus{outline:none}.ui-file-input__select-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__select-btn:hover:not(:disabled){background:var(--ui-color-surface-hover)}.ui-file-input__file-list{display:flex;flex-direction:column;gap:var(--ui-spacing-1)}.ui-file-input__file-item{display:flex;align-items:center;justify-content:space-between;padding:var(--ui-spacing-2) var(--ui-spacing-3);border:1px solid var(--ui-color-border);border-radius:var(--ui-radius-md);background:var(--ui-color-surface)}.ui-file-input__file-info{display:flex;align-items:center;gap:var(--ui-spacing-2);min-width:0}.ui-file-input__file-info lucide-icon{flex-shrink:0;color:var(--ui-color-text-muted)}.ui-file-input__file-name{font-size:var(--ui-font-size-sm);color:var(--ui-color-text);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.ui-file-input__file-size{font-size:var(--ui-font-size-xs);color:var(--ui-color-text-muted);flex-shrink:0}.ui-file-input__remove-btn{appearance:none;border:none;background:transparent;cursor:pointer;padding:var(--ui-spacing-1);border-radius:var(--ui-radius-sm);color:var(--ui-color-text-muted);transition:color var(--ui-transition-fast),background-color var(--ui-transition-fast)}.ui-file-input__remove-btn:focus{outline:none}.ui-file-input__remove-btn:focus-visible{outline:var(--ui-focus-ring-width) solid var(--ui-focus-ring-color);outline-offset:var(--ui-focus-ring-offset)}.ui-file-input__remove-btn:hover{color:var(--ui-color-error, #dc2626);background:var(--ui-color-error-subtle, #fef2f2)}\n"] }]
|
|
303
|
+
}], propDecorators: { config: [{
|
|
304
|
+
type: Input
|
|
305
|
+
}], fileRemoved: [{
|
|
306
|
+
type: Output
|
|
307
|
+
}], validationError: [{
|
|
308
|
+
type: Output
|
|
309
|
+
}] } });
|
|
310
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-input.component.js","sourceRoot":"","sources":["../../../../../../../../packages/ng-ui-system/src/lib/components/form-builder/sub-components/file-input/file-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n  forwardRef,\r\n  ChangeDetectionStrategy,\r\n  ViewEncapsulation,\r\n  signal,\r\n  computed,\r\n} from '@angular/core';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\r\nimport { LucideAngularModule } from 'lucide-angular';\r\nimport { UiFileData, UiFileInputConfig } from '../../types/field.types';\r\n\r\n/**\r\n * Componente file input standalone con drag & drop,\r\n * anteprima, validazione e supporto multiplo.\r\n *\r\n * Implementa `ControlValueAccessor` per integrazione\r\n * diretta con Angular Reactive Forms.\r\n *\r\n * @selector ui-file-input\r\n */\r\n@Component({\r\n  selector: 'ui-file-input',\r\n  standalone: true,\r\n  imports: [LucideAngularModule],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  encapsulation: ViewEncapsulation.None,\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => UiFileInputComponent),\r\n      multi: true,\r\n    },\r\n  ],\r\n  template: `\r\n    <div class=\"ui-file-input\" [class.ui-file-input--disabled]=\"disabled()\" [class.ui-file-input--drag-over]=\"isDragOver()\">\r\n      <!-- Area drag & drop -->\r\n      @if (config?.enableDragDrop !== false) {\r\n        <div\r\n          class=\"ui-file-input__drop-zone\"\r\n          (dragover)=\"onDragOver($event)\"\r\n          (dragleave)=\"onDragLeave()\"\r\n          (drop)=\"onDrop($event)\"\r\n          (click)=\"fileInput.click()\"\r\n          role=\"button\"\r\n          tabindex=\"0\"\r\n          [attr.aria-label]=\"config?.dragDropText || 'Trascina i file qui o clicca per selezionare'\"\r\n          (keydown.enter)=\"fileInput.click()\"\r\n          (keydown.space)=\"fileInput.click()\"\r\n        >\r\n          <lucide-icon name=\"upload\" [size]=\"24\" aria-hidden=\"true\" />\r\n          <span class=\"ui-file-input__drop-text\">\r\n            {{ config?.dragDropText || 'Trascina i file qui o clicca per selezionare' }}\r\n          </span>\r\n          @if (config?.acceptedTypes?.length) {\r\n            <span class=\"ui-file-input__hint\">\r\n              Formati: {{ config!.acceptedTypes!.join(', ') }}\r\n            </span>\r\n          }\r\n        </div>\r\n      } @else {\r\n        <button\r\n          class=\"ui-file-input__select-btn\"\r\n          type=\"button\"\r\n          [disabled]=\"disabled()\"\r\n          (click)=\"fileInput.click()\"\r\n        >\r\n          <lucide-icon name=\"paperclip\" [size]=\"16\" aria-hidden=\"true\" />\r\n          {{ config?.placeholder || 'Seleziona file' }}\r\n        </button>\r\n      }\r\n\r\n      <input\r\n        #fileInput\r\n        type=\"file\"\r\n        class=\"ui-file-input__native\"\r\n        [accept]=\"acceptString\"\r\n        [multiple]=\"config?.multiple ?? false\"\r\n        [disabled]=\"disabled()\"\r\n        (change)=\"onFileSelected($event)\"\r\n      />\r\n\r\n      <!-- Lista file selezionati -->\r\n      @if (files().length > 0) {\r\n        <div class=\"ui-file-input__file-list\" role=\"list\">\r\n          @for (file of files(); track file.id || $index) {\r\n            <div class=\"ui-file-input__file-item\" role=\"listitem\">\r\n              <div class=\"ui-file-input__file-info\">\r\n                <lucide-icon name=\"file\" [size]=\"16\" aria-hidden=\"true\" />\r\n                <span class=\"ui-file-input__file-name\">{{ file.filename }}</span>\r\n                <span class=\"ui-file-input__file-size\">{{ formatSize(file.size) }}</span>\r\n              </div>\r\n              @if (!disabled()) {\r\n                <button\r\n                  class=\"ui-file-input__remove-btn\"\r\n                  type=\"button\"\r\n                  [attr.aria-label]=\"'Rimuovi ' + file.filename\"\r\n                  (click)=\"removeFile($index)\"\r\n                >\r\n                  <lucide-icon name=\"x\" [size]=\"14\" aria-hidden=\"true\" />\r\n                </button>\r\n              }\r\n            </div>\r\n          }\r\n        </div>\r\n      }\r\n    </div>\r\n  `,\r\n  styleUrl: './file-input.component.scss',\r\n})\r\nexport class UiFileInputComponent implements ControlValueAccessor {\r\n  /** Configurazione del file input. */\r\n  @Input() config: UiFileInputConfig | undefined;\r\n\r\n  /** File rimosso dalla lista. */\r\n  @Output() fileRemoved = new EventEmitter<UiFileData>();\r\n\r\n  /** Errore di validazione file. */\r\n  @Output() validationError = new EventEmitter<string>();\r\n\r\n  // ── Stato interno (signals) ──\r\n  readonly files = signal<UiFileData[]>([]);\r\n  readonly disabled = signal(false);\r\n  readonly isDragOver = signal(false);\r\n\r\n  private onChange: (value: UiFileData[]) => void = () => {};\r\n  private onTouched: () => void = () => {};\r\n\r\n  /** Stringa accept per l'input nativo. */\r\n  get acceptString(): string {\r\n    return this.config?.acceptedTypes?.join(',') || '';\r\n  }\r\n\r\n  // ── ControlValueAccessor ──\r\n  writeValue(value: UiFileData[]): void {\r\n    this.files.set(Array.isArray(value) ? value : []);\r\n  }\r\n  registerOnChange(fn: any): void {\r\n    this.onChange = fn;\r\n  }\r\n  registerOnTouched(fn: any): void {\r\n    this.onTouched = fn;\r\n  }\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled.set(isDisabled);\r\n  }\r\n\r\n  // ── Drag & drop ──\r\n  onDragOver(event: DragEvent): void {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n    this.isDragOver.set(true);\r\n  }\r\n\r\n  onDragLeave(): void {\r\n    this.isDragOver.set(false);\r\n  }\r\n\r\n  onDrop(event: DragEvent): void {\r\n    event.preventDefault();\r\n    event.stopPropagation();\r\n    this.isDragOver.set(false);\r\n    if (this.disabled()) return;\r\n\r\n    const droppedFiles = event.dataTransfer?.files;\r\n    if (droppedFiles?.length) {\r\n      this.processFiles(droppedFiles);\r\n    }\r\n  }\r\n\r\n  onFileSelected(event: Event): void {\r\n    const input = event.target as HTMLInputElement;\r\n    if (input.files?.length) {\r\n      this.processFiles(input.files);\r\n      input.value = ''; // Reset per permettere la ri-selezione dello stesso file\r\n    }\r\n  }\r\n\r\n  removeFile(index: number): void {\r\n    const current = [...this.files()];\r\n    const removed = current.splice(index, 1);\r\n    this.files.set(current);\r\n    this.onChange(current);\r\n    if (removed[0]) this.fileRemoved.emit(removed[0]);\r\n  }\r\n\r\n  formatSize(bytes: number): string {\r\n    if (bytes === 0) return '0 B';\r\n    const k = 1024;\r\n    const sizes = ['B', 'KB', 'MB', 'GB'];\r\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\r\n    return parseFloat((bytes / Math.pow(k, i)).toFixed(1)) + ' ' + sizes[i];\r\n  }\r\n\r\n  // ── Processing ──\r\n  private async processFiles(fileList: FileList): Promise<void> {\r\n    this.onTouched();\r\n    const maxFiles = this.config?.maxFiles || Infinity;\r\n    const current = this.config?.multiple ? [...this.files()] : [];\r\n    const remaining = maxFiles - current.length;\r\n\r\n    if (remaining <= 0) {\r\n      this.validationError.emit(`Numero massimo di file raggiunto (${maxFiles})`);\r\n      return;\r\n    }\r\n\r\n    const filesToProcess = Array.from(fileList).slice(0, remaining);\r\n\r\n    for (const file of filesToProcess) {\r\n      // Validazione dimensione\r\n      if (this.config?.maxFileSize && file.size > this.config.maxFileSize) {\r\n        this.validationError.emit(`Il file \"${file.name}\" supera la dimensione massima consentita`);\r\n        continue;\r\n      }\r\n\r\n      // Validazione tipo\r\n      if (this.config?.acceptedTypes?.length) {\r\n        const isAccepted = this.config.acceptedTypes.some((type) => {\r\n          if (type.endsWith('/*')) {\r\n            return file.type.startsWith(type.split('/')[0] + '/');\r\n          }\r\n          return file.type === type;\r\n        });\r\n        if (!isAccepted) {\r\n          this.validationError.emit(`Il formato del file \"${file.name}\" non e accettato`);\r\n          continue;\r\n        }\r\n      }\r\n\r\n      const content = new Uint8Array(await file.arrayBuffer());\r\n      const ext = file.name.split('.').pop() || '';\r\n\r\n      const fileData: UiFileData = {\r\n        filename: file.name,\r\n        content,\r\n        mimeType: file.type,\r\n        extension: ext,\r\n        size: file.size,\r\n        needsUpload: true,\r\n        id: `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`, // NOSONAR - Used for non-cryptographic file ID generation in UI\r\n        timestamp: new Date(),\r\n      };\r\n\r\n      // Genera preview per immagini\r\n      if (this.config?.showPreview && file.type.startsWith('image/')) {\r\n        fileData.previewUrl = URL.createObjectURL(file);\r\n      }\r\n\r\n      current.push(fileData);\r\n    }\r\n\r\n    this.files.set(current);\r\n    this.onChange(current);\r\n  }\r\n}\r\n"]}
|