fx-form-builder-wrapper 2.0.4 → 2.0.7

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.
@@ -3,7 +3,7 @@ import { HttpClient } from '@angular/common/http';
3
3
  import { Component, inject } from '@angular/core';
4
4
  import { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';
5
5
  import { FxBaseComponent, FxComponent, FxSelectSetting, FxStringSetting, FxValidatorService } from '@instantsys-labs/fx';
6
- import { forkJoin, map, Subject, takeUntil } from 'rxjs';
6
+ import { Subject, takeUntil } from 'rxjs';
7
7
  import { v4 as uuidv4 } from 'uuid';
8
8
  import * as i0 from "@angular/core";
9
9
  import * as i1 from "../../fx-builder-wrapper.service";
@@ -19,6 +19,12 @@ export class UploaderComponent extends FxBaseComponent {
19
19
  uploadedFiles: [],
20
20
  deletedFiles: []
21
21
  };
22
+ categories = [
23
+ 'Oral Images',
24
+ 'Past Docs',
25
+ 'X-Rays',
26
+ 'Profile',
27
+ ];
22
28
  destroy$ = new Subject();
23
29
  http = inject(HttpClient);
24
30
  constructor(cdr, fxBuilderWrapperService) {
@@ -114,67 +120,68 @@ export class UploaderComponent extends FxBaseComponent {
114
120
  // }
115
121
  // });
116
122
  // }
117
- ngOnInit() {
118
- this.fxBuilderWrapperService.variables$
119
- .pipe(takeUntil(this.destroy$))
120
- .subscribe((variables) => {
121
- if (!variables)
122
- return;
123
- const uploadedFilesMap = {};
124
- // Extract uploader keys and their corresponding URL arrays
125
- for (const [key, value] of Object.entries(variables)) {
126
- if (key.includes('uploader') && Array.isArray(value)) {
127
- uploadedFilesMap[key] = value;
128
- }
129
- }
130
- for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {
131
- // const imageFetches: Observable<{ result: string; originalUrl: any; title:string ; notes:string }>[] = [];
132
- // urls.forEach((originalUrl:any) => {
133
- // if (originalUrl) {
134
- // const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(
135
- // map((blob: Blob) => ({
136
- // result: URL.createObjectURL(blob), // for preview
137
- // originalUrl: originalUrl?.originalUrl // preserve original
138
- // }))
139
- // );
140
- // imageFetches.push(image$);
141
- // }
142
- // });
143
- const imageFetches = [];
144
- urls.forEach((originalUrl) => {
145
- if (originalUrl) {
146
- const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(map((blob) => ({
147
- result: URL.createObjectURL(blob), // for preview
148
- originalUrl: originalUrl?.originalUrl, // preserve original
149
- title: originalUrl?.title || '', // default empty if not present
150
- notes: originalUrl?.notes || ''
151
- })));
152
- imageFetches.push(image$);
153
- }
154
- });
155
- if (imageFetches.length > 0) {
156
- forkJoin(imageFetches).subscribe({
157
- next: (imageData) => {
158
- const formatted = imageData.map(item => ({
159
- id: uuidv4(),
160
- file: null,
161
- originalUrl: item.originalUrl,
162
- result: item.result,
163
- title: item.title,
164
- notes: ''
165
- }));
166
- this.uploadedFiles = [...this.uploadedFiles, ...formatted];
167
- this.formattedData.uploadedFiles = this.uploadedFiles;
168
- this.uploadFileControl.setValue(this.formattedData);
169
- },
170
- error: (err) => {
171
- console.error(`Failed to fetch images for ${uploaderKey}:`, err);
172
- }
173
- });
174
- }
175
- }
176
- });
177
- }
123
+ // ngOnInit(): void {
124
+ // this.fxBuilderWrapperService.variables$
125
+ // .pipe(takeUntil(this.destroy$))
126
+ // .subscribe((variables: any) => {
127
+ // if (!variables) return;
128
+ // const uploadedFilesMap: { [key: string]: string[] } = {};
129
+ // // Extract uploader keys and their corresponding URL arrays
130
+ // for (const [key, value] of Object.entries(variables)) {
131
+ // if (key.includes('uploader') && Array.isArray(value)) {
132
+ // uploadedFilesMap[key] = value;
133
+ // }
134
+ // }
135
+ // for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {
136
+ // // const imageFetches: Observable<{ result: string; originalUrl: any; title:string ; notes:string }>[] = [];
137
+ // // urls.forEach((originalUrl:any) => {
138
+ // // if (originalUrl) {
139
+ // // const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(
140
+ // // map((blob: Blob) => ({
141
+ // // result: URL.createObjectURL(blob), // for preview
142
+ // // originalUrl: originalUrl?.originalUrl // preserve original
143
+ // // }))
144
+ // // );
145
+ // // imageFetches.push(image$);
146
+ // // }
147
+ // // });
148
+ // const imageFetches: Observable<{ result: string; originalUrl: any; title: string; notes: string }>[] = [];
149
+ // urls.forEach((originalUrl: any) => {
150
+ // if (originalUrl) {
151
+ // const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(
152
+ // map((blob: Blob) => ({
153
+ // result: URL.createObjectURL(blob), // for preview
154
+ // originalUrl: originalUrl?.originalUrl, // preserve original
155
+ // title: originalUrl?.title || '', // default empty if not present
156
+ // notes: originalUrl?.notes || ''
157
+ // }))
158
+ // );
159
+ // imageFetches.push(image$);
160
+ // }
161
+ // });
162
+ // if (imageFetches.length > 0) {
163
+ // forkJoin(imageFetches).subscribe({
164
+ // next: (imageData) => {
165
+ // const formatted = imageData.map(item => ({
166
+ // id: uuidv4(),
167
+ // file: null,
168
+ // originalUrl: item.originalUrl,
169
+ // result: item.result,
170
+ // title: item.title,
171
+ // notes: ''
172
+ // }));
173
+ // this.uploadedFiles = [...this.uploadedFiles, ...formatted];
174
+ // this.formattedData.uploadedFiles = this.uploadedFiles;
175
+ // this.uploadFileControl.setValue(this.formattedData);
176
+ // },
177
+ // error: (err) => {
178
+ // console.error(`Failed to fetch images for ${uploaderKey}:`, err);
179
+ // }
180
+ // });
181
+ // }
182
+ // }
183
+ // });
184
+ // }
178
185
  // ngOnInit(): void {
179
186
  // this.fxBuilderWrapperService.variables$
180
187
  // .pipe(takeUntil(this.destroy$))
@@ -236,29 +243,147 @@ export class UploaderComponent extends FxBaseComponent {
236
243
  // }
237
244
  // }
238
245
  // }
246
+ // onFileSelected(event: Event) {
247
+ // const input = event.target as HTMLInputElement;
248
+ // if (input.files) {
249
+ // for (let i = 0; i < input.files.length; i++) {
250
+ // const file = input.files[i];
251
+ // const reader = new FileReader();
252
+ // reader.onload = e => {
253
+ // const newFile = {
254
+ // id: uuidv4(),
255
+ // file: file,
256
+ // originalUrl: null,
257
+ // result: e.target?.result,
258
+ // name: file.name,
259
+ // title: '',
260
+ // notes: '',
261
+ // category: '',
262
+ // type: this.detectFileType(file),
263
+ // };
264
+ // this.uploadedFiles.push(newFile);
265
+ // this.formattedData.uploadedFiles = this.uploadedFiles;
266
+ // this.uploadFileControl.setValue(this.formattedData);
267
+ // };
268
+ // reader.readAsDataURL(file);
269
+ // }
270
+ // }
271
+ // }
272
+ // ngOnInit(): void {
273
+ // this.fxBuilderWrapperService.variables$
274
+ // .pipe(takeUntil(this.destroy$))
275
+ // .subscribe((variables: any) => {
276
+ // if (!variables) return;
277
+ // const uploadedFilesMap: { [key: string]: any[] } = {};
278
+ // for (const [key, value] of Object.entries(variables)) {
279
+ // if (key.includes('uploader') && Array.isArray(value)) {
280
+ // uploadedFilesMap[key] = value;
281
+ // }
282
+ // }
283
+ // for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {
284
+ // const formatted = urls
285
+ // .filter((fileObj: any) => !!fileObj)
286
+ // .map((fileObj: any) => {
287
+ // // const fileType = this.detectFileTypeFromUrl(fileObj?.originalUrl || fileObj?.name);
288
+ // return {
289
+ // id: uuidv4(),
290
+ // file: null,
291
+ // originalUrl: fileObj.originalUrl,
292
+ // result: fileObj?.type === 'image' ? fileObj.originalUrl.previewUrl : null,
293
+ // name: fileObj?.name || '',
294
+ // title: fileObj?.title || '',
295
+ // notes: fileObj?.notes || '',
296
+ // category: fileObj?.category || '',
297
+ // type: fileObj?.type,
298
+ // };
299
+ // });
300
+ // if (formatted.length > 0) {
301
+ // this.uploadedFiles = [...this.uploadedFiles, ...formatted];
302
+ // this.formattedData.uploadedFiles = this.uploadedFiles;
303
+ // this.uploadFileControl.setValue(this.formattedData);
304
+ // }
305
+ // }
306
+ // });
307
+ // }
308
+ ngOnInit() {
309
+ this.fxBuilderWrapperService.variables$
310
+ .pipe(takeUntil(this.destroy$))
311
+ .subscribe((variables) => {
312
+ if (!variables)
313
+ return;
314
+ const uploadedFilesMap = {};
315
+ for (const [key, value] of Object.entries(variables)) {
316
+ if (key.includes('uploader') && value?.uploadedFiles) {
317
+ uploadedFilesMap[key] = value.uploadedFiles;
318
+ }
319
+ }
320
+ for (const [uploaderKey, files] of Object.entries(uploadedFilesMap)) {
321
+ const formatted = files
322
+ .filter((fileObj) => !!fileObj)
323
+ .map((fileObj) => {
324
+ const originalUrlObj = fileObj.originalUrl;
325
+ const fileName = originalUrlObj?.fileName || '';
326
+ const fileUrl = originalUrlObj?.fileUrl || '';
327
+ const previewUrl = originalUrlObj?.previewUrl;
328
+ const type = fileObj?.type;
329
+ // detect type based on file extension
330
+ // const fileType = this.detectFileTypeFromUrl(fileName || fileUrl);
331
+ return {
332
+ id: uuidv4(),
333
+ file: null, // nothing local
334
+ originalUrl: originalUrlObj, // keep whole object
335
+ result: type === 'image'
336
+ ? (previewUrl) // prefer previewUrl, fallback to fileUrl
337
+ : null,
338
+ name: fileName,
339
+ title: fileObj?.title || '',
340
+ notes: fileObj?.notes || '',
341
+ category: fileObj?.category || '',
342
+ type: type, // computed type
343
+ };
344
+ });
345
+ if (formatted.length > 0) {
346
+ this.uploadedFiles = [...this.uploadedFiles, ...formatted];
347
+ this.formattedData.uploadedFiles = this.uploadedFiles;
348
+ this.uploadFileControl.setValue(this.formattedData);
349
+ }
350
+ }
351
+ });
352
+ }
239
353
  onFileSelected(event) {
240
354
  const input = event.target;
241
- if (input.files) {
242
- for (let i = 0; i < input.files.length; i++) {
243
- const file = input.files[i];
355
+ if (!input.files)
356
+ return;
357
+ Array.from(input.files).forEach(file => {
358
+ const fileType = this.detectFileType(file);
359
+ const newFile = {
360
+ id: uuidv4(),
361
+ file: file,
362
+ originalUrl: null,
363
+ result: null, // will be filled for images only
364
+ name: file.name,
365
+ title: '',
366
+ notes: '',
367
+ category: '',
368
+ type: fileType, // <-- keep detected type
369
+ };
370
+ if (fileType === 'image') {
244
371
  const reader = new FileReader();
245
372
  reader.onload = e => {
246
- const newFile = {
247
- id: uuidv4(),
248
- file: file,
249
- originalUrl: null,
250
- result: e.target?.result,
251
- name: file.name,
252
- title: '',
253
- notes: ''
254
- };
373
+ newFile.result = e.target?.result;
255
374
  this.uploadedFiles.push(newFile);
256
375
  this.formattedData.uploadedFiles = this.uploadedFiles;
257
376
  this.uploadFileControl.setValue(this.formattedData);
258
377
  };
259
- reader.readAsDataURL(file);
378
+ reader.readAsDataURL(file); // only images need preview
379
+ }
380
+ else {
381
+ // non-image → push directly
382
+ this.uploadedFiles.push(newFile);
383
+ this.formattedData.uploadedFiles = this.uploadedFiles;
384
+ this.uploadFileControl.setValue(this.formattedData);
260
385
  }
261
- }
386
+ });
262
387
  }
263
388
  onMetaChange() {
264
389
  this.formattedData.uploadedFiles = this.uploadedFiles;
@@ -307,11 +432,22 @@ export class UploaderComponent extends FxBaseComponent {
307
432
  validations() {
308
433
  return [FxValidatorService.required];
309
434
  }
435
+ detectFileType(file) {
436
+ const mime = (file.type || '').toLowerCase();
437
+ const name = (file.name || '').toLowerCase();
438
+ if (mime.startsWith('image/') || /\.(png|jpe?g|gif|webp|bmp|svg)$/i.test(name))
439
+ return 'image';
440
+ if (mime === 'text/csv' || name.endsWith('.csv'))
441
+ return 'csv';
442
+ if (mime.startsWith('text/') || /\.(txt|md|rtf|log|json|xml|yaml|yml)$/i.test(name))
443
+ return 'text';
444
+ return 'other';
445
+ }
310
446
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UploaderComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.FxBuilderWrapperService }], target: i0.ɵɵFactoryTarget.Component });
311
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: UploaderComponent, isStandalone: true, selector: "fx-uploader", usesInheritance: true, ngImport: i0, template: "<fx-component [fxData]=\"fxData\">\r\n <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n <div>\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n fill=\"#F3A041\" />\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n fill=\"#F3A041\" />\r\n <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n <path\r\n d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n fill=\"white\" />\r\n </svg>\r\n </div>\r\n <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n {{ setting('upload-text') }}\r\n </a>\r\n </div>\r\n <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n <div class=\"separator\"></div>\r\n <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"flex flex-col items-center justify-center\">\r\n <div class=\"relative\">\r\n <div class=\"close_icon cursor-pointer\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <g clip-path=\"url(#clip0_6837_29683)\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n height=\"24\">\r\n <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n </mask>\r\n <g mask=\"url(#mask0_6837_29683)\">\r\n <path\r\n d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n fill=\"white\" />\r\n </g>\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_6837_29683\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </div>\r\n <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n \r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n </div>\r\n \r\n\r\n </div>\r\n\r\n <p>{{file?.name}}</p>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n</fx-component>", styles: [".custom-upload{border:1px solid #ccc}.custom-upload a{color:#000}.custom-upload .file-list .file-thumbnail{width:120px;height:auto;border:1px solid #ccc;background-color:#fff;padding:4px}.custom-upload .separator{height:56px;width:2px;background-color:#ccc;margin:10px}.close_icon{position:absolute;top:-8px;right:-8px}[type=checkbox]{padding:12px}[type=checkbox]:checked{background-color:#f3a041!important;padding:12px;background-size:1em 1em!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: FxComponent, selector: "fx-component", inputs: ["fxData"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: ReactiveFormsModule }] });
447
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: UploaderComponent, isStandalone: true, selector: "fx-uploader", usesInheritance: true, ngImport: i0, template: "<fx-component [fxData]=\"fxData\">\r\n <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n <div>\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n fill=\"#F3A041\" />\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n fill=\"#F3A041\" />\r\n <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n <path\r\n d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n fill=\"white\" />\r\n </svg>\r\n </div>\r\n <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n {{ setting('upload-text') }}\r\n </a>\r\n </div>\r\n <!-- <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n <div class=\"separator\"></div>\r\n <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\" \r\n class=\"flex flex-col items-center justify-center\" (click)=\"deleteFile(i)\">\r\n <div class=\"relative\">\r\n <div class=\"close_icon cursor-pointer\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <g clip-path=\"url(#clip0_6837_29683)\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n height=\"24\">\r\n <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n </mask>\r\n <g mask=\"url(#mask0_6837_29683)\">\r\n <path\r\n d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n fill=\"white\" />\r\n </g>\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_6837_29683\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </div>\r\n <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n \r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n </div>\r\n \r\n\r\n </div>\r\n\r\n <p>{{file?.name}}</p>\r\n </div>\r\n </ng-container>\r\n </div> -->\r\n\r\n <input #fileInput\r\n type=\"file\"\r\n [multiple]=\"setting('multiple-upload')\"\r\n (change)=\"onFileSelected($event)\"\r\n hidden />\r\n\r\n<div class=\"separator\"></div>\r\n\r\n<div class=\"file-list flex items-center gap-6 flex-wrap\" *ngIf=\"false\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"flex flex-col items-center justify-center\">\r\n\r\n <div class=\"relative p-3 border rounded-lg shadow-sm bg-white w-64\">\r\n <!-- Close icon -->\r\n <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n fill=\"white\"/>\r\n </svg>\r\n </div>\r\n\r\n <!-- Thumbnail -->\r\n <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n <img class=\"file-thumbnail w-full h-32 object-cover rounded mb-3\"\r\n [src]=\"file?.result\"\r\n alt=\"preview\" />\r\n </ng-container>\r\n\r\n <ng-template #nonImageThumb>\r\n <div class=\"file-thumbnail flex items-center justify-center border rounded mb-3 w-full h-32 bg-gray-50\">\r\n <ng-container [ngSwitch]=\"file.type\">\r\n <!-- CSV -->\r\n <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n </svg>\r\n\r\n <!-- TEXT -->\r\n <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n\r\n <!-- Generic -->\r\n <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n </ng-container>\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Metadata -->\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n\r\n <label class=\"font-semibold text-sm mt-3 mb-1\">Category</label>\r\n <select [(ngModel)]=\"file.category\"\r\n name=\"category{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Select Category\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n </select>\r\n </div>\r\n </div>\r\n\r\n <p class=\"mt-2 text-sm text-gray-600\">{{ file?.name }}</p>\r\n </div>\r\n </ng-container>\r\n</div>\r\n\r\n<div class=\"file-list flex items-center gap-8 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"relative p-3 border rounded-lg shadow-sm bg-white w-[500px] flex\">\r\n\r\n <!-- Close icon -->\r\n <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n fill=\"white\"/>\r\n </svg>\r\n </div>\r\n\r\n <!-- Thumbnail on left -->\r\n <div class=\"w-28 h-28 flex-shrink-0\">\r\n <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n <img class=\"file-thumbnail w-full h-full object-cover rounded\"\r\n [src]=\"file?.result\"\r\n alt=\"preview\" />\r\n </ng-container>\r\n\r\n <ng-template #nonImageThumb>\r\n <div class=\"file-thumbnail flex items-center justify-center border rounded w-full h-full bg-gray-50\">\r\n <ng-container [ngSwitch]=\"file.type\">\r\n <!-- CSV -->\r\n <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n </svg>\r\n <!-- TEXT -->\r\n <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n <!-- Generic -->\r\n <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n </ng-container>\r\n </div>\r\n </ng-template>\r\n </div>\r\n\r\n <!-- Metadata on right -->\r\n <div class=\"flex flex-col flex-1 ml-4\">\r\n <!-- Title -->\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n <!-- Notes -->\r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n\r\n <!-- Category -->\r\n <label class=\"font-semibold text-sm mb-1\">Category</label>\r\n <select [(ngModel)]=\"file.category\"\r\n name=\"category{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n <option value=\"\">Select Category</option>\r\n <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n </select>\r\n </div>\r\n </div>\r\n </ng-container>\r\n</div>\r\n\r\n\r\n\r\n </div>\r\n</fx-component>", styles: [".custom-upload{border:1px solid #ccc}.custom-upload a{color:#000}.custom-upload .file-list .file-thumbnail{width:234px;height:234px;border:1px solid #ccc;background-color:#fff;padding:4px}.custom-upload .separator{height:56px;width:2px;background-color:#ccc;margin:10px}.close_icon{position:absolute;top:-8px;right:-8px}[type=checkbox]{padding:12px}[type=checkbox]:checked{background-color:#f3a041!important;padding:12px;background-size:1em 1em!important}.gap-8{gap:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i2.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { kind: "component", type: FxComponent, selector: "fx-component", inputs: ["fxData"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: ReactiveFormsModule }] });
312
448
  }
313
449
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UploaderComponent, decorators: [{
314
450
  type: Component,
315
- args: [{ selector: 'fx-uploader', standalone: true, imports: [CommonModule, FxComponent, FormsModule, ReactiveFormsModule], template: "<fx-component [fxData]=\"fxData\">\r\n <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n <div>\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n fill=\"#F3A041\" />\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n fill=\"#F3A041\" />\r\n <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n <path\r\n d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n fill=\"white\" />\r\n </svg>\r\n </div>\r\n <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n {{ setting('upload-text') }}\r\n </a>\r\n </div>\r\n <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n <div class=\"separator\"></div>\r\n <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"flex flex-col items-center justify-center\">\r\n <div class=\"relative\">\r\n <div class=\"close_icon cursor-pointer\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <g clip-path=\"url(#clip0_6837_29683)\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n height=\"24\">\r\n <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n </mask>\r\n <g mask=\"url(#mask0_6837_29683)\">\r\n <path\r\n d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n fill=\"white\" />\r\n </g>\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_6837_29683\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </div>\r\n <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n \r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n </div>\r\n \r\n\r\n </div>\r\n\r\n <p>{{file?.name}}</p>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n</fx-component>", styles: [".custom-upload{border:1px solid #ccc}.custom-upload a{color:#000}.custom-upload .file-list .file-thumbnail{width:120px;height:auto;border:1px solid #ccc;background-color:#fff;padding:4px}.custom-upload .separator{height:56px;width:2px;background-color:#ccc;margin:10px}.close_icon{position:absolute;top:-8px;right:-8px}[type=checkbox]{padding:12px}[type=checkbox]:checked{background-color:#f3a041!important;padding:12px;background-size:1em 1em!important}\n"] }]
451
+ args: [{ selector: 'fx-uploader', standalone: true, imports: [CommonModule, FxComponent, FormsModule, ReactiveFormsModule], template: "<fx-component [fxData]=\"fxData\">\r\n <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n <div>\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n fill=\"#F3A041\" />\r\n <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n fill=\"#F3A041\" />\r\n <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n <path\r\n d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n fill=\"white\" />\r\n </svg>\r\n </div>\r\n <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n {{ setting('upload-text') }}\r\n </a>\r\n </div>\r\n <!-- <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n <div class=\"separator\"></div>\r\n <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\" \r\n class=\"flex flex-col items-center justify-center\" (click)=\"deleteFile(i)\">\r\n <div class=\"relative\">\r\n <div class=\"close_icon cursor-pointer\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <g clip-path=\"url(#clip0_6837_29683)\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n height=\"24\">\r\n <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n </mask>\r\n <g mask=\"url(#mask0_6837_29683)\">\r\n <path\r\n d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n fill=\"white\" />\r\n </g>\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_6837_29683\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </div>\r\n <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n \r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n </div>\r\n \r\n\r\n </div>\r\n\r\n <p>{{file?.name}}</p>\r\n </div>\r\n </ng-container>\r\n </div> -->\r\n\r\n <input #fileInput\r\n type=\"file\"\r\n [multiple]=\"setting('multiple-upload')\"\r\n (change)=\"onFileSelected($event)\"\r\n hidden />\r\n\r\n<div class=\"separator\"></div>\r\n\r\n<div class=\"file-list flex items-center gap-6 flex-wrap\" *ngIf=\"false\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"flex flex-col items-center justify-center\">\r\n\r\n <div class=\"relative p-3 border rounded-lg shadow-sm bg-white w-64\">\r\n <!-- Close icon -->\r\n <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n fill=\"white\"/>\r\n </svg>\r\n </div>\r\n\r\n <!-- Thumbnail -->\r\n <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n <img class=\"file-thumbnail w-full h-32 object-cover rounded mb-3\"\r\n [src]=\"file?.result\"\r\n alt=\"preview\" />\r\n </ng-container>\r\n\r\n <ng-template #nonImageThumb>\r\n <div class=\"file-thumbnail flex items-center justify-center border rounded mb-3 w-full h-32 bg-gray-50\">\r\n <ng-container [ngSwitch]=\"file.type\">\r\n <!-- CSV -->\r\n <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n </svg>\r\n\r\n <!-- TEXT -->\r\n <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n\r\n <!-- Generic -->\r\n <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n </ng-container>\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Metadata -->\r\n <div class=\"flex flex-col w-full\">\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n\r\n <label class=\"font-semibold text-sm mt-3 mb-1\">Category</label>\r\n <select [(ngModel)]=\"file.category\"\r\n name=\"category{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Select Category\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n </select>\r\n </div>\r\n </div>\r\n\r\n <p class=\"mt-2 text-sm text-gray-600\">{{ file?.name }}</p>\r\n </div>\r\n </ng-container>\r\n</div>\r\n\r\n<div class=\"file-list flex items-center gap-8 flex-wrap\">\r\n <ng-container *ngIf=\"uploadedFiles?.length\">\r\n <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n class=\"relative p-3 border rounded-lg shadow-sm bg-white w-[500px] flex\">\r\n\r\n <!-- Close icon -->\r\n <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n fill=\"white\"/>\r\n </svg>\r\n </div>\r\n\r\n <!-- Thumbnail on left -->\r\n <div class=\"w-28 h-28 flex-shrink-0\">\r\n <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n <img class=\"file-thumbnail w-full h-full object-cover rounded\"\r\n [src]=\"file?.result\"\r\n alt=\"preview\" />\r\n </ng-container>\r\n\r\n <ng-template #nonImageThumb>\r\n <div class=\"file-thumbnail flex items-center justify-center border rounded w-full h-full bg-gray-50\">\r\n <ng-container [ngSwitch]=\"file.type\">\r\n <!-- CSV -->\r\n <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n </svg>\r\n <!-- TEXT -->\r\n <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n <!-- Generic -->\r\n <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n </svg>\r\n </ng-container>\r\n </div>\r\n </ng-template>\r\n </div>\r\n\r\n <!-- Metadata on right -->\r\n <div class=\"flex flex-col flex-1 ml-4\">\r\n <!-- Title -->\r\n <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n <input [(ngModel)]=\"file.title\"\r\n name=\"title{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter title\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n <!-- Notes -->\r\n <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n <textarea [(ngModel)]=\"file.notes\"\r\n name=\"notes{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n placeholder=\"Enter notes\"\r\n class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\"\r\n rows=\"2\"></textarea>\r\n\r\n <!-- Category -->\r\n <label class=\"font-semibold text-sm mb-1\">Category</label>\r\n <select [(ngModel)]=\"file.category\"\r\n name=\"category{{i}}\"\r\n (ngModelChange)=\"onMetaChange()\"\r\n class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n <option value=\"\">Select Category</option>\r\n <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n </select>\r\n </div>\r\n </div>\r\n </ng-container>\r\n</div>\r\n\r\n\r\n\r\n </div>\r\n</fx-component>", styles: [".custom-upload{border:1px solid #ccc}.custom-upload a{color:#000}.custom-upload .file-list .file-thumbnail{width:234px;height:234px;border:1px solid #ccc;background-color:#fff;padding:4px}.custom-upload .separator{height:56px;width:2px;background-color:#ccc;margin:10px}.close_icon{position:absolute;top:-8px;right:-8px}[type=checkbox]{padding:12px}[type=checkbox]:checked{background-color:#f3a041!important;padding:12px;background-size:1em 1em!important}.gap-8{gap:8px}\n"] }]
316
452
  }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.FxBuilderWrapperService }] });
317
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploader.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/uploader/uploader.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/uploader/uploader.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { AfterViewInit, ChangeDetectorRef, Component, inject, OnDestroy, OnInit } from '@angular/core';\r\nimport { FormControl, FormsModule, ReactiveFormsModule, UntypedFormControl } from '@angular/forms';\r\nimport { FxBaseComponent, FxComponent, FxSelectSetting, FxSetting, FxStringSetting, FxValidation, FxValidatorService } from '@instantsys-labs/fx';\r\nimport { FxBuilderWrapperService } from '../../fx-builder-wrapper.service';\r\nimport { forkJoin, map, Observable, Subject, takeUntil } from 'rxjs';\r\nimport { v4 as uuidv4} from 'uuid';\r\n\r\n@Component({\r\n  selector: 'fx-uploader',\r\n  standalone: true,\r\n  imports: [CommonModule, FxComponent, FormsModule, ReactiveFormsModule],\r\n  templateUrl: './uploader.component.html',\r\n  styleUrl: './uploader.component.css'\r\n})\r\nexport class UploaderComponent extends FxBaseComponent implements OnInit{\r\n  // public uploadFileControl = new UntypedFormControl();\r\n  public uploadFileControl = new FormControl();\r\n  public uploadedFiles: Array<any> = [];\r\n  public formattedData: any = {\r\n    uploadedFiles: [],\r\n    deletedFiles: []\r\n  };\r\n  private destroy$ = new Subject<Boolean>();\r\n  private http = inject(HttpClient);\r\n  constructor(private cdr: ChangeDetectorRef,private fxBuilderWrapperService: FxBuilderWrapperService) {\r\n    super(cdr)\r\n    this.onInit.subscribe((fxData)=>{\r\n      this._register(this.uploadFileControl);\r\n    })\r\n  }\r\n\r\n\r\n\r\n\r\n\r\n  uploadedImages: { [key: string]: { result: string, file: File | null }[] } = {};\r\n  deletedFiles: any[] = [];\r\n  // public ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n  \r\n  //       const uploadedFiles: { [key: string]: string[] } = {};\r\n  \r\n  //       for (const [key, value] of Object.entries(variables)) {\r\n  //         if (key.includes('uploader') && Array.isArray(value)) {\r\n  //           uploadedFiles[key] = value;\r\n  //         }\r\n  //       }\r\n  \r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFiles)) {\r\n  //         const imageFetches: Observable<string>[] = [];\r\n  \r\n  //         urls.forEach((url: string) => {\r\n  //           if (url) {\r\n  //             const image$ = this.http.get(url, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => URL.createObjectURL(blob))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n  \r\n  //         if (imageFetches.length) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageUrls: string[]) => {\r\n  //               this.uploadedImages[uploaderKey] = imageUrls.map(result => ({\r\n  //                 result,\r\n  //                 file: null\r\n  //               }));\r\n  //               // this.uploadedFiles = [...this.uploadedImages]\r\n  //               this.uploadedFiles = Object.values(this.uploadedImages).flat();\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n  \r\n  // ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n\r\n  //       const uploadedFiles: { [key: string]: string[] } = {};\r\n\r\n  //       for (const [key, value] of Object.entries(variables)) {\r\n  //         if (key.includes('uploader') && Array.isArray(value)) {\r\n  //           uploadedFiles[key] = value;\r\n  //         }\r\n  //       }\r\n\r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFiles)) {\r\n  //         const imageFetches: Observable<{ result: string, originalUrl: string }>[] = [];\r\n\r\n  //         urls.forEach((url: string) => {\r\n  //           if (url) {\r\n  //             const image$ = this.http.get(url, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => ({\r\n  //                 result: URL.createObjectURL(blob), // just for preview\r\n  //                 originalUrl: url\r\n  //               }))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n\r\n  //         if (imageFetches.length) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageData) => {\r\n  //               const formatted = imageData.map(item => ({\r\n  //                 id: uuidv4(),\r\n  //                 file: null,\r\n  //                 originalUrl: item.originalUrl,\r\n  //                 result: item.result\r\n  //               }));\r\n\r\n  //               this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n  //               this.uploadFileControl.setValue(this.uploadedFiles);\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n\r\n  ngOnInit(): void {\r\n    this.fxBuilderWrapperService.variables$\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe((variables: any) => {\r\n        if (!variables) return;\r\n  \r\n        const uploadedFilesMap: { [key: string]: string[] } = {};\r\n  \r\n        // Extract uploader keys and their corresponding URL arrays\r\n        for (const [key, value] of Object.entries(variables)) {\r\n          if (key.includes('uploader') && Array.isArray(value)) {\r\n            uploadedFilesMap[key] = value;\r\n          }\r\n        }\r\n  \r\n        for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {\r\n          // const imageFetches: Observable<{ result: string; originalUrl: any; title:string ; notes:string }>[] = [];\r\n  \r\n          // urls.forEach((originalUrl:any) => {\r\n          //   if (originalUrl) {\r\n          //     const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(\r\n          //       map((blob: Blob) => ({\r\n          //         result: URL.createObjectURL(blob), // for preview\r\n          //         originalUrl: originalUrl?.originalUrl          // preserve original\r\n          //       }))\r\n          //     );\r\n          //     imageFetches.push(image$);\r\n          //   }\r\n          // });\r\n\r\n          const imageFetches: Observable<{ result: string; originalUrl: any; title: string; notes: string }>[] = [];\r\n\r\nurls.forEach((originalUrl: any) => {\r\n  if (originalUrl) {\r\n    const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(\r\n      map((blob: Blob) => ({\r\n        result: URL.createObjectURL(blob), // for preview\r\n        originalUrl: originalUrl?.originalUrl, // preserve original\r\n        title: originalUrl?.title || '', // default empty if not present\r\n        notes: originalUrl?.notes || ''\r\n      }))\r\n    );\r\n    imageFetches.push(image$);\r\n  }\r\n});\r\n\r\n  \r\n          if (imageFetches.length > 0) {\r\n            forkJoin(imageFetches).subscribe({\r\n              next: (imageData) => {\r\n                const formatted = imageData.map(item => ({\r\n                  id: uuidv4(),\r\n                  file: null,\r\n                  originalUrl: item.originalUrl,\r\n                  result: item.result,\r\n                  title: item.title,\r\n                  notes: ''\r\n                }));\r\n  \r\n                this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n                this.formattedData.uploadedFiles = this.uploadedFiles;\r\n                this.uploadFileControl.setValue(this.formattedData);\r\n              },\r\n              error: (err) => {\r\n                console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n              }\r\n            });\r\n          }\r\n        }\r\n      });\r\n  }\r\n  \r\n  // ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n  \r\n  //       const uploadedFilesMap: { [key: string]: { originalUrl: string }[] } = variables.uploadedFiles;\r\n  \r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {\r\n  //         const imageFetches: Observable<{ result: string; originalUrl: string }>[] = [];\r\n  \r\n  //         urls.forEach((fileObj: any) => {\r\n  //           const originalUrl = fileObj?.originalUrl;\r\n  //           if (originalUrl) {\r\n  //             const image$ = this.http.get(originalUrl, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => ({\r\n  //                 result: URL.createObjectURL(blob), // for preview\r\n  //                 originalUrl\r\n  //               }))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n  \r\n  //         if (imageFetches.length > 0) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageData) => {\r\n  //               const formatted = imageData.map(item => ({\r\n  //                 id: uuidv4(),\r\n  //                 file: null,\r\n  //                 originalUrl: item.originalUrl,\r\n  //                 result: item.result\r\n  //               }));\r\n  \r\n  //               this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n  //               this.formattedData.uploadedFiles = this.uploadedFiles;\r\n  //               this.uploadFileControl.setValue(this.formattedData);\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n  \r\n\r\n  // public onFileSelected(event: Event) {\r\n  //   const input = event.target as HTMLInputElement;\r\n  //   if (input.files) {\r\n  //     for(let i = 0; i < input?.files?.length; i++) {\r\n  //       const file = input.files[i];\r\n  //       const reader = new FileReader();\r\n  //       reader.onload = e => {\r\n  //         this.uploadedFiles.push({\r\n  //           file: file,\r\n  //           result: e.target?.result,\r\n  //           name: file?.name,\r\n  //           id: uuidv4()\r\n  //         })\r\n  //         this.uploadFileControl.setValue(this.uploadedFiles);\r\n  //         console.log(this.uploadFileControl);\r\n  //       }\r\n  //       reader.readAsDataURL(file);\r\n  //     }\r\n  //   }\r\n  // }\r\n\r\n  onFileSelected(event: Event) {\r\n    const input = event.target as HTMLInputElement;\r\n    if (input.files) {\r\n      for (let i = 0; i < input.files.length; i++) {\r\n        const file = input.files[i];\r\n        const reader = new FileReader();\r\n\r\n        reader.onload = e => {\r\n          const newFile = {\r\n            id: uuidv4(),\r\n            file: file,\r\n            originalUrl: null,\r\n            result: e.target?.result,\r\n            name: file.name,\r\n            title: '',\r\n            notes: ''\r\n          };\r\n\r\n          this.uploadedFiles.push(newFile);\r\n          this.formattedData.uploadedFiles = this.uploadedFiles;\r\n          this.uploadFileControl.setValue(this.formattedData);\r\n        };\r\n\r\n        reader.readAsDataURL(file);\r\n      }\r\n    }\r\n  }\r\n\r\n  onMetaChange() {\r\n    this.formattedData.uploadedFiles = this.uploadedFiles;\r\n    this.uploadFileControl.setValue(this.formattedData);\r\n  }\r\n\r\n\r\n  // public deleteFile(id: string): void {\r\n  //   this.uploadedFiles = this.uploadedFiles.filter(file => file?.id !== id);\r\n  // }\r\n\r\n  // deleteFile(index: number): void {\r\n  //   const [deletedFile] = this.uploadedFiles.splice(index, 1);\r\n  \r\n  //   if (deletedFile) {\r\n  //     this.deletedFiles.push(deletedFile);\r\n  //     this.formattedData = {\r\n  //       ...this.formattedData,\r\n  //       uploadedFiles: [...this.uploadedFiles],\r\n  //       deletedFiles: [...this.deletedFiles]\r\n  //     };\r\n  //     this.uploadFileControl.setValue(this.formattedData);\r\n  //   }\r\n  // }\r\n\r\n  deleteFile(index: number): void {\r\n    const deletedFile = this.uploadedFiles?.[index];\r\n    if (!deletedFile) return;\r\n  \r\n    // Ensure formattedData exists and initialize if undefined\r\n    this.formattedData = this.formattedData || { uploadedFiles: [], deletedFiles: [] };\r\n  \r\n    // Add the deleted file to the deletedFiles array\r\n    this.deletedFiles = [...this.deletedFiles, deletedFile];\r\n  \r\n    // Remove the file from uploadedFiles using filter for immutability\r\n    this.uploadedFiles = this.uploadedFiles.filter((_, i) => i !== index);\r\n  \r\n    // Update formattedData\r\n    this.formattedData = {\r\n      ...this.formattedData,\r\n      uploadedFiles: this.uploadedFiles,\r\n      deletedFiles: this.deletedFiles\r\n    };\r\n  \r\n    // Set the value of the uploadFileControl\r\n    this.uploadFileControl.setValue(this.formattedData);\r\n  }\r\n  \r\n  \r\n\r\n  protected settings(): FxSetting[] {\r\n    return [\r\n      new FxStringSetting({ key: 'upload-text', $title: 'Upload Text', value: 'Upload File'}),\r\n      new FxSelectSetting({key: 'multiple-upload', $title: 'Multiple Upload', value: false}, [{option: 'Enable', value: true}, {option: 'Disable', value: false}])\r\n    ];\r\n  }\r\n\r\n  protected validations(): FxValidation[] {\r\n    return [FxValidatorService.required];\r\n  }\r\n}\r\n","<fx-component [fxData]=\"fxData\">\r\n  <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n    <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n      <div>\r\n        <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n          <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n            d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n            fill=\"#F3A041\" />\r\n          <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n            d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n            fill=\"#F3A041\" />\r\n          <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n          <path\r\n            d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n            fill=\"white\" />\r\n        </svg>\r\n      </div>\r\n      <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n        {{ setting('upload-text') }}\r\n      </a>\r\n    </div>\r\n    <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n    <div class=\"separator\"></div>\r\n    <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n      <ng-container *ngIf=\"uploadedFiles?.length\">\r\n        <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n          class=\"flex flex-col items-center justify-center\">\r\n          <div class=\"relative\">\r\n            <div class=\"close_icon cursor-pointer\">\r\n              <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n                <g clip-path=\"url(#clip0_6837_29683)\">\r\n                  <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n                  <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n                    height=\"24\">\r\n                    <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n                  </mask>\r\n                  <g mask=\"url(#mask0_6837_29683)\">\r\n                    <path\r\n                      d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n                      fill=\"white\" />\r\n                  </g>\r\n                </g>\r\n                <defs>\r\n                  <clipPath id=\"clip0_6837_29683\">\r\n                    <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n                  </clipPath>\r\n                </defs>\r\n              </svg>\r\n            </div>\r\n            <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n            <div class=\"flex flex-col w-full\">\r\n              <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n              <input [(ngModel)]=\"file.title\"\r\n                     name=\"title{{i}}\"\r\n                     (ngModelChange)=\"onMetaChange()\"\r\n                     placeholder=\"Enter title\"\r\n                     class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n          \r\n              <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n              <textarea [(ngModel)]=\"file.notes\"\r\n                        name=\"notes{{i}}\"\r\n                        (ngModelChange)=\"onMetaChange()\"\r\n                        placeholder=\"Enter notes\"\r\n                        class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n                        rows=\"2\"></textarea>\r\n            </div>\r\n          \r\n\r\n          </div>\r\n\r\n          <p>{{file?.name}}</p>\r\n        </div>\r\n      </ng-container>\r\n    </div>\r\n  </div>\r\n</fx-component>"]}
453
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploader.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/uploader/uploader.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/uploader/uploader.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { AfterViewInit, ChangeDetectorRef, Component, inject, OnDestroy, OnInit } from '@angular/core';\r\nimport { FormControl, FormsModule, ReactiveFormsModule, UntypedFormControl } from '@angular/forms';\r\nimport { FxBaseComponent, FxComponent, FxSelectSetting, FxSetting, FxStringSetting, FxValidation, FxValidatorService } from '@instantsys-labs/fx';\r\nimport { FxBuilderWrapperService } from '../../fx-builder-wrapper.service';\r\nimport { forkJoin, map, Observable, Subject, takeUntil } from 'rxjs';\r\nimport { v4 as uuidv4} from 'uuid';\r\n\r\n@Component({\r\n  selector: 'fx-uploader',\r\n  standalone: true,\r\n  imports: [CommonModule, FxComponent, FormsModule, ReactiveFormsModule],\r\n  templateUrl: './uploader.component.html',\r\n  styleUrl: './uploader.component.css'\r\n})\r\nexport class UploaderComponent extends FxBaseComponent implements OnInit{\r\n  // public uploadFileControl = new UntypedFormControl();\r\n  public uploadFileControl = new FormControl();\r\n  public uploadedFiles: Array<any> = [];\r\n  public formattedData: any = {\r\n    uploadedFiles: [],\r\n    deletedFiles: []\r\n  };\r\n  categories = [\r\n    'Oral Images',\r\n    'Past Docs',\r\n    'X-Rays',\r\n    'Profile',\r\n  ];\r\n  private destroy$ = new Subject<Boolean>();\r\n  private http = inject(HttpClient);\r\n  constructor(private cdr: ChangeDetectorRef,private fxBuilderWrapperService: FxBuilderWrapperService) {\r\n    super(cdr)\r\n    this.onInit.subscribe((fxData)=>{\r\n      this._register(this.uploadFileControl);\r\n    })\r\n  }\r\n\r\n\r\n\r\n\r\n\r\n  uploadedImages: { [key: string]: { result: string, file: File | null }[] } = {};\r\n  deletedFiles: any[] = [];\r\n  // public ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n  \r\n  //       const uploadedFiles: { [key: string]: string[] } = {};\r\n  \r\n  //       for (const [key, value] of Object.entries(variables)) {\r\n  //         if (key.includes('uploader') && Array.isArray(value)) {\r\n  //           uploadedFiles[key] = value;\r\n  //         }\r\n  //       }\r\n  \r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFiles)) {\r\n  //         const imageFetches: Observable<string>[] = [];\r\n  \r\n  //         urls.forEach((url: string) => {\r\n  //           if (url) {\r\n  //             const image$ = this.http.get(url, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => URL.createObjectURL(blob))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n  \r\n  //         if (imageFetches.length) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageUrls: string[]) => {\r\n  //               this.uploadedImages[uploaderKey] = imageUrls.map(result => ({\r\n  //                 result,\r\n  //                 file: null\r\n  //               }));\r\n  //               // this.uploadedFiles = [...this.uploadedImages]\r\n  //               this.uploadedFiles = Object.values(this.uploadedImages).flat();\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n  \r\n  // ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n\r\n  //       const uploadedFiles: { [key: string]: string[] } = {};\r\n\r\n  //       for (const [key, value] of Object.entries(variables)) {\r\n  //         if (key.includes('uploader') && Array.isArray(value)) {\r\n  //           uploadedFiles[key] = value;\r\n  //         }\r\n  //       }\r\n\r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFiles)) {\r\n  //         const imageFetches: Observable<{ result: string, originalUrl: string }>[] = [];\r\n\r\n  //         urls.forEach((url: string) => {\r\n  //           if (url) {\r\n  //             const image$ = this.http.get(url, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => ({\r\n  //                 result: URL.createObjectURL(blob), // just for preview\r\n  //                 originalUrl: url\r\n  //               }))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n\r\n  //         if (imageFetches.length) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageData) => {\r\n  //               const formatted = imageData.map(item => ({\r\n  //                 id: uuidv4(),\r\n  //                 file: null,\r\n  //                 originalUrl: item.originalUrl,\r\n  //                 result: item.result\r\n  //               }));\r\n\r\n  //               this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n  //               this.uploadFileControl.setValue(this.uploadedFiles);\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n\r\n//   ngOnInit(): void {\r\n//     this.fxBuilderWrapperService.variables$\r\n//       .pipe(takeUntil(this.destroy$))\r\n//       .subscribe((variables: any) => {\r\n//         if (!variables) return;\r\n  \r\n//         const uploadedFilesMap: { [key: string]: string[] } = {};\r\n  \r\n//         // Extract uploader keys and their corresponding URL arrays\r\n//         for (const [key, value] of Object.entries(variables)) {\r\n//           if (key.includes('uploader') && Array.isArray(value)) {\r\n//             uploadedFilesMap[key] = value;\r\n//           }\r\n//         }\r\n  \r\n//         for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {\r\n//           // const imageFetches: Observable<{ result: string; originalUrl: any; title:string ; notes:string }>[] = [];\r\n  \r\n//           // urls.forEach((originalUrl:any) => {\r\n//           //   if (originalUrl) {\r\n//           //     const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(\r\n//           //       map((blob: Blob) => ({\r\n//           //         result: URL.createObjectURL(blob), // for preview\r\n//           //         originalUrl: originalUrl?.originalUrl          // preserve original\r\n//           //       }))\r\n//           //     );\r\n//           //     imageFetches.push(image$);\r\n//           //   }\r\n//           // });\r\n\r\n//           const imageFetches: Observable<{ result: string; originalUrl: any; title: string; notes: string }>[] = [];\r\n\r\n// urls.forEach((originalUrl: any) => {\r\n//   if (originalUrl) {\r\n//     const image$ = this.http.get(originalUrl?.originalUrl, { responseType: 'blob' }).pipe(\r\n//       map((blob: Blob) => ({\r\n//         result: URL.createObjectURL(blob), // for preview\r\n//         originalUrl: originalUrl?.originalUrl, // preserve original\r\n//         title: originalUrl?.title || '', // default empty if not present\r\n//         notes: originalUrl?.notes || ''\r\n//       }))\r\n//     );\r\n//     imageFetches.push(image$);\r\n//   }\r\n// });\r\n\r\n  \r\n//           if (imageFetches.length > 0) {\r\n//             forkJoin(imageFetches).subscribe({\r\n//               next: (imageData) => {\r\n//                 const formatted = imageData.map(item => ({\r\n//                   id: uuidv4(),\r\n//                   file: null,\r\n//                   originalUrl: item.originalUrl,\r\n//                   result: item.result,\r\n//                   title: item.title,\r\n//                   notes: ''\r\n//                 }));\r\n  \r\n//                 this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n//                 this.formattedData.uploadedFiles = this.uploadedFiles;\r\n//                 this.uploadFileControl.setValue(this.formattedData);\r\n//               },\r\n//               error: (err) => {\r\n//                 console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n//               }\r\n//             });\r\n//           }\r\n//         }\r\n//       });\r\n//   }\r\n  \r\n  // ngOnInit(): void {\r\n  //   this.fxBuilderWrapperService.variables$\r\n  //     .pipe(takeUntil(this.destroy$))\r\n  //     .subscribe((variables: any) => {\r\n  //       if (!variables) return;\r\n  \r\n  //       const uploadedFilesMap: { [key: string]: { originalUrl: string }[] } = variables.uploadedFiles;\r\n  \r\n  //       for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {\r\n  //         const imageFetches: Observable<{ result: string; originalUrl: string }>[] = [];\r\n  \r\n  //         urls.forEach((fileObj: any) => {\r\n  //           const originalUrl = fileObj?.originalUrl;\r\n  //           if (originalUrl) {\r\n  //             const image$ = this.http.get(originalUrl, { responseType: 'blob' }).pipe(\r\n  //               map((blob: Blob) => ({\r\n  //                 result: URL.createObjectURL(blob), // for preview\r\n  //                 originalUrl\r\n  //               }))\r\n  //             );\r\n  //             imageFetches.push(image$);\r\n  //           }\r\n  //         });\r\n  \r\n  //         if (imageFetches.length > 0) {\r\n  //           forkJoin(imageFetches).subscribe({\r\n  //             next: (imageData) => {\r\n  //               const formatted = imageData.map(item => ({\r\n  //                 id: uuidv4(),\r\n  //                 file: null,\r\n  //                 originalUrl: item.originalUrl,\r\n  //                 result: item.result\r\n  //               }));\r\n  \r\n  //               this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n  //               this.formattedData.uploadedFiles = this.uploadedFiles;\r\n  //               this.uploadFileControl.setValue(this.formattedData);\r\n  //             },\r\n  //             error: (err) => {\r\n  //               console.error(`Failed to fetch images for ${uploaderKey}:`, err);\r\n  //             }\r\n  //           });\r\n  //         }\r\n  //       }\r\n  //     });\r\n  // }\r\n  \r\n\r\n  // public onFileSelected(event: Event) {\r\n  //   const input = event.target as HTMLInputElement;\r\n  //   if (input.files) {\r\n  //     for(let i = 0; i < input?.files?.length; i++) {\r\n  //       const file = input.files[i];\r\n  //       const reader = new FileReader();\r\n  //       reader.onload = e => {\r\n  //         this.uploadedFiles.push({\r\n  //           file: file,\r\n  //           result: e.target?.result,\r\n  //           name: file?.name,\r\n  //           id: uuidv4()\r\n  //         })\r\n  //         this.uploadFileControl.setValue(this.uploadedFiles);\r\n  //         console.log(this.uploadFileControl);\r\n  //       }\r\n  //       reader.readAsDataURL(file);\r\n  //     }\r\n  //   }\r\n  // }\r\n\r\n  // onFileSelected(event: Event) {\r\n  //   const input = event.target as HTMLInputElement;\r\n  //   if (input.files) {\r\n  //     for (let i = 0; i < input.files.length; i++) {\r\n  //       const file = input.files[i];\r\n  //       const reader = new FileReader();\r\n\r\n  //       reader.onload = e => {\r\n  //         const newFile = {\r\n  //           id: uuidv4(),\r\n  //           file: file,\r\n  //           originalUrl: null,\r\n  //           result: e.target?.result,\r\n  //           name: file.name,\r\n  //           title: '',\r\n  //           notes: '',\r\n  //           category: '',\r\n  //           type: this.detectFileType(file),\r\n  //         };\r\n\r\n  //         this.uploadedFiles.push(newFile);\r\n  //         this.formattedData.uploadedFiles = this.uploadedFiles;\r\n  //         this.uploadFileControl.setValue(this.formattedData);\r\n  //       };\r\n\r\n  //       reader.readAsDataURL(file);\r\n  //     }\r\n  //   }\r\n  // }\r\n\r\n//   ngOnInit(): void {\r\n//   this.fxBuilderWrapperService.variables$\r\n//     .pipe(takeUntil(this.destroy$))\r\n//     .subscribe((variables: any) => {\r\n//       if (!variables) return;\r\n\r\n//       const uploadedFilesMap: { [key: string]: any[] } = {};\r\n\r\n     \r\n//       for (const [key, value] of Object.entries(variables)) {\r\n//         if (key.includes('uploader') && Array.isArray(value)) {\r\n//           uploadedFilesMap[key] = value;\r\n//         }\r\n//       }\r\n\r\n//       for (const [uploaderKey, urls] of Object.entries(uploadedFilesMap)) {\r\n//         const formatted = urls\r\n//           .filter((fileObj: any) => !!fileObj) \r\n//           .map((fileObj: any) => {\r\n//             // const fileType = this.detectFileTypeFromUrl(fileObj?.originalUrl || fileObj?.name);\r\n\r\n//             return {\r\n//               id: uuidv4(),\r\n//               file: null,                         \r\n//               originalUrl: fileObj.originalUrl,    \r\n//               result: fileObj?.type === 'image' ? fileObj.originalUrl.previewUrl : null,\r\n//               name: fileObj?.name || '',\r\n//               title: fileObj?.title || '',\r\n//               notes: fileObj?.notes || '',\r\n//               category: fileObj?.category || '',\r\n//               type: fileObj?.type,                \r\n//             };\r\n//           });\r\n\r\n//         if (formatted.length > 0) {\r\n//           this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n//           this.formattedData.uploadedFiles = this.uploadedFiles;\r\n//           this.uploadFileControl.setValue(this.formattedData);\r\n//         }\r\n//       }\r\n//     });\r\n// }\r\n\r\nngOnInit(): void {\r\n  this.fxBuilderWrapperService.variables$\r\n    .pipe(takeUntil(this.destroy$))\r\n    .subscribe((variables: any) => {\r\n      if (!variables) return;\r\n\r\n      const uploadedFilesMap: { [key: string]: any[] } = {};\r\n\r\n      \r\n   for (const [key, value] of Object.entries(variables)) {\r\n  if (key.includes('uploader') && (value as any)?.uploadedFiles) {\r\n    uploadedFilesMap[key] = (value as any).uploadedFiles;\r\n  }\r\n}\r\n\r\n      for (const [uploaderKey, files] of Object.entries(uploadedFilesMap)) {\r\n        const formatted = files\r\n          .filter((fileObj: any) => !!fileObj)\r\n          .map((fileObj: any) => {\r\n            const originalUrlObj = fileObj.originalUrl;\r\n            const fileName = originalUrlObj?.fileName || '';\r\n            const fileUrl = originalUrlObj?.fileUrl || '';\r\n            const previewUrl = originalUrlObj?.previewUrl;\r\n            const type = fileObj?.type;\r\n\r\n            // detect type based on file extension\r\n            // const fileType = this.detectFileTypeFromUrl(fileName || fileUrl);\r\n\r\n            return {\r\n              id: uuidv4(),\r\n              file: null,                        // nothing local\r\n              originalUrl: originalUrlObj,       // keep whole object\r\n              result: type === 'image'\r\n                ? (previewUrl)        // prefer previewUrl, fallback to fileUrl\r\n                : null,\r\n              name: fileName,\r\n              title: fileObj?.title || '',\r\n              notes: fileObj?.notes || '',\r\n              category: fileObj?.category || '',\r\n              type: type,                    // computed type\r\n            };\r\n          });\r\n\r\n        if (formatted.length > 0) {\r\n          this.uploadedFiles = [...this.uploadedFiles, ...formatted];\r\n          this.formattedData.uploadedFiles = this.uploadedFiles;\r\n          this.uploadFileControl.setValue(this.formattedData);\r\n        }\r\n      }\r\n    });\r\n}\r\n\r\n\r\n\r\n  onFileSelected(event: Event) {\r\n  const input = event.target as HTMLInputElement;\r\n  if (!input.files) return;\r\n\r\n  Array.from(input.files).forEach(file => {\r\n    const fileType = this.detectFileType(file);\r\n\r\n    const newFile: any = {\r\n      id: uuidv4(),\r\n      file: file,\r\n      originalUrl: null,\r\n      result: null,          // will be filled for images only\r\n      name: file.name,\r\n      title: '',\r\n      notes: '',\r\n      category: '',\r\n      type: fileType,        // <-- keep detected type\r\n    };\r\n\r\n    if (fileType === 'image') {\r\n      const reader = new FileReader();\r\n      reader.onload = e => {\r\n        newFile.result = e.target?.result;\r\n        this.uploadedFiles.push(newFile);\r\n        this.formattedData.uploadedFiles = this.uploadedFiles;\r\n        this.uploadFileControl.setValue(this.formattedData);\r\n      };\r\n      reader.readAsDataURL(file); // only images need preview\r\n    } else {\r\n      // non-image → push directly\r\n      this.uploadedFiles.push(newFile);\r\n      this.formattedData.uploadedFiles = this.uploadedFiles;\r\n      this.uploadFileControl.setValue(this.formattedData);\r\n    }\r\n  });\r\n}\r\n\r\n\r\n  onMetaChange() {\r\n    this.formattedData.uploadedFiles = this.uploadedFiles;\r\n    this.uploadFileControl.setValue(this.formattedData);\r\n  }\r\n\r\n\r\n  // public deleteFile(id: string): void {\r\n  //   this.uploadedFiles = this.uploadedFiles.filter(file => file?.id !== id);\r\n  // }\r\n\r\n  // deleteFile(index: number): void {\r\n  //   const [deletedFile] = this.uploadedFiles.splice(index, 1);\r\n  \r\n  //   if (deletedFile) {\r\n  //     this.deletedFiles.push(deletedFile);\r\n  //     this.formattedData = {\r\n  //       ...this.formattedData,\r\n  //       uploadedFiles: [...this.uploadedFiles],\r\n  //       deletedFiles: [...this.deletedFiles]\r\n  //     };\r\n  //     this.uploadFileControl.setValue(this.formattedData);\r\n  //   }\r\n  // }\r\n\r\n  deleteFile(index: number): void {\r\n    const deletedFile = this.uploadedFiles?.[index];\r\n    if (!deletedFile) return;\r\n  \r\n    // Ensure formattedData exists and initialize if undefined\r\n    this.formattedData = this.formattedData || { uploadedFiles: [], deletedFiles: [] };\r\n  \r\n    // Add the deleted file to the deletedFiles array\r\n    this.deletedFiles = [...this.deletedFiles, deletedFile];\r\n  \r\n    // Remove the file from uploadedFiles using filter for immutability\r\n    this.uploadedFiles = this.uploadedFiles.filter((_, i) => i !== index);\r\n  \r\n    // Update formattedData\r\n    this.formattedData = {\r\n      ...this.formattedData,\r\n      uploadedFiles: this.uploadedFiles,\r\n      deletedFiles: this.deletedFiles\r\n    };\r\n  \r\n    // Set the value of the uploadFileControl\r\n    this.uploadFileControl.setValue(this.formattedData);\r\n  }\r\n  \r\n  \r\n\r\n  protected settings(): FxSetting[] {\r\n    return [\r\n      new FxStringSetting({ key: 'upload-text', $title: 'Upload Text', value: 'Upload File'}),\r\n      new FxSelectSetting({key: 'multiple-upload', $title: 'Multiple Upload', value: false}, [{option: 'Enable', value: true}, {option: 'Disable', value: false}])\r\n    ];\r\n  }\r\n\r\n  protected validations(): FxValidation[] {\r\n    return [FxValidatorService.required];\r\n  }\r\n\r\n    detectFileType(file: File): 'image' | 'csv' | 'text' | 'other' {\r\n    const mime = (file.type || '').toLowerCase();\r\n    const name = (file.name || '').toLowerCase();\r\n\r\n    if (mime.startsWith('image/') || /\\.(png|jpe?g|gif|webp|bmp|svg)$/i.test(name)) return 'image';\r\n    if (mime === 'text/csv' || name.endsWith('.csv')) return 'csv';\r\n    if (mime.startsWith('text/') || /\\.(txt|md|rtf|log|json|xml|yaml|yml)$/i.test(name)) return 'text';\r\n    return 'other';\r\n  }\r\n\r\n}\r\n","<fx-component [fxData]=\"fxData\">\r\n  <div class=\"custom-upload flex flex-row items-center gap-2 p-2 pb-0 rounded-sm\">\r\n    <div class=\"flex flex-col justify-center items-center\" (click)=\"fileInput.click()\">\r\n      <div>\r\n        <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"29\" height=\"28\" viewBox=\"0 0 29 28\" fill=\"none\">\r\n          <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n            d=\"M14.001 3.00195C11.5322 3.00195 9.16722 3.86956 7.23641 5.38414L7.23598 4.59695C7.23598 4.52473 7.22175 4.45321 7.19411 4.38648C7.16647 4.31975 7.12596 4.25912 7.07489 4.20804C7.02381 4.15697 6.96318 4.11646 6.89645 4.08882C6.82972 4.06118 6.7582 4.04695 6.68598 4.04695C6.61375 4.04695 6.54223 4.06118 6.4755 4.08882C6.40877 4.11646 6.34814 4.15697 6.29707 4.20804C6.24599 4.25912 6.20548 4.31975 6.17784 4.38648C6.1502 4.45321 6.13598 4.52473 6.13598 4.59695L6.13642 6.68652C6.13643 6.83239 6.19438 6.97228 6.29752 7.07542C6.40067 7.17856 6.54055 7.23651 6.68642 7.23652H8.77598C8.8482 7.23652 8.91972 7.2223 8.98645 7.19466C9.05318 7.16702 9.11381 7.12651 9.16489 7.07543C9.21596 7.02436 9.25647 6.96373 9.28411 6.897C9.31175 6.83027 9.32598 6.75875 9.32598 6.68652C9.32598 6.6143 9.31175 6.54278 9.28411 6.47605C9.25647 6.40932 9.21596 6.34869 9.16489 6.29761C9.11381 6.24654 9.05318 6.20603 8.98645 6.17839C8.91972 6.15075 8.8482 6.13652 8.77598 6.13652H8.0584C9.7702 4.84387 11.8399 4.10195 14.001 4.10195C19.4751 4.10195 23.901 8.52782 23.901 14.002C23.901 19.4761 19.4751 23.902 14.001 23.902C8.52684 23.902 4.10098 19.4761 4.10098 14.002C4.10098 12.6381 4.38263 11.2888 4.92855 10.0389C4.98691 9.90532 4.98982 9.75399 4.93665 9.61822C4.88347 9.48246 4.77855 9.37336 4.64496 9.31492C4.57877 9.28596 4.50753 9.27032 4.4353 9.2689C4.36306 9.26747 4.29126 9.2803 4.22398 9.30663C4.15671 9.33297 4.09528 9.3723 4.04321 9.42238C3.99114 9.47247 3.94944 9.53232 3.92051 9.59852C3.31393 10.9872 3.00098 12.4866 3.00098 14.002C3.00098 20.0706 7.93236 25.002 14.001 25.002C20.0696 25.002 25.001 20.0706 25.001 14.002C25.001 7.93334 20.0696 3.00195 14.001 3.00195Z\"\r\n            fill=\"#F3A041\" />\r\n          <path fill-rule=\"evenodd\" clip-rule=\"evenodd\"\r\n            d=\"M13.9988 6.13965C9.6616 6.13965 6.13379 9.66746 6.13379 14.0046C6.13379 18.3418 9.6616 21.8696 13.9988 21.8696C18.336 21.8696 21.8638 18.3418 21.8638 14.0046C21.8638 9.66746 18.336 6.13965 13.9988 6.13965ZM13.9988 8.22965C14.1447 8.22965 14.2846 8.28759 14.3877 8.39074C14.4908 8.49388 14.5488 8.63378 14.5488 8.77965V13.7769L17.5227 16.7508C17.6258 16.8539 17.6837 16.9938 17.6837 17.1396C17.6837 17.2855 17.6258 17.4254 17.5227 17.5285C17.4195 17.6316 17.2796 17.6896 17.1338 17.6896C16.9879 17.6896 16.8481 17.6316 16.7449 17.5285L13.6099 14.3935C13.5068 14.2904 13.4488 14.1505 13.4488 14.0046V8.77965C13.4488 8.63378 13.5067 8.49388 13.6099 8.39074C13.713 8.28759 13.8529 8.22965 13.9988 8.22965Z\"\r\n            fill=\"#F3A041\" />\r\n          <rect width=\"28.56\" height=\"28\" rx=\"1.51479\" fill=\"#F3A041\" />\r\n          <path\r\n            d=\"M15.068 22.7391V17.6708H18.4968L14.2108 12.6025L9.9248 17.6708H13.3536V22.7391H9.06759V22.6969C8.92358 22.7053 8.78643 22.7391 8.63899 22.7391C6.93391 22.7391 5.29866 22.0717 4.09298 20.8835C2.8873 19.6954 2.20996 18.084 2.20996 16.4037C2.20996 13.1532 4.70443 10.5042 7.90523 10.141C8.18585 8.6953 8.96892 7.39118 10.1197 6.45292C11.2706 5.51465 12.7172 5.00089 14.2108 5C15.7047 5.00082 17.1515 5.51452 18.3026 6.45277C19.4537 7.39101 20.2371 8.69516 20.5181 10.141C23.7189 10.5042 26.21 13.1532 26.21 16.4037C26.21 18.084 25.5326 19.6954 24.3269 20.8835C23.1213 22.0717 21.486 22.7391 19.7809 22.7391C19.6369 22.7391 19.4981 22.7053 19.3523 22.6969V22.7391H15.068Z\"\r\n            fill=\"white\" />\r\n        </svg>\r\n      </div>\r\n      <a type=\"button\" (click)=\"fileInput.click()\" class=\"text-base cursor-pointer\">\r\n        {{ setting('upload-text') }}\r\n      </a>\r\n    </div>\r\n    <!-- <input #fileInput type=\"file\" [multiple]=\"setting('multiple-upload')\" (change)=\"onFileSelected($event)\" hidden />\r\n    <div class=\"separator\"></div>\r\n    <div class=\"file-list flex items-center gap-4 flex-wrap\">\r\n      <ng-container *ngIf=\"uploadedFiles?.length\">\r\n        <div *ngFor=\"let file of uploadedFiles; let i = index\" \r\n          class=\"flex flex-col items-center justify-center\" (click)=\"deleteFile(i)\">\r\n          <div class=\"relative\">\r\n            <div class=\"close_icon cursor-pointer\">\r\n              <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n                <g clip-path=\"url(#clip0_6837_29683)\">\r\n                  <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n                  <mask id=\"mask0_6837_29683\" style=\"mask-type:alpha\" maskUnits=\"userSpaceOnUse\" x=\"4\" y=\"4\" width=\"24\"\r\n                    height=\"24\">\r\n                    <rect x=\"4\" y=\"4\" width=\"24\" height=\"24\" fill=\"#D9D9D9\" />\r\n                  </mask>\r\n                  <g mask=\"url(#mask0_6837_29683)\">\r\n                    <path\r\n                      d=\"M8.2045 8.99815L5.81802 6.61167L6.61351 5.81617L9 8.20266L11.3865 5.81617L12.182 6.61167L9.7955 8.99815L12.182 11.3846L11.3865 12.1801L9 9.79365L6.61351 12.1801L5.81802 11.3846L8.2045 8.99815Z\"\r\n                      fill=\"white\" />\r\n                  </g>\r\n                </g>\r\n                <defs>\r\n                  <clipPath id=\"clip0_6837_29683\">\r\n                    <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"white\" />\r\n                  </clipPath>\r\n                </defs>\r\n              </svg>\r\n            </div>\r\n            <img class=\"file-thumbnail\" style=\"border-radius: 4px\" [src]=\"file?.result\" alt=\"\" />\r\n            <div class=\"flex flex-col w-full\">\r\n              <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n              <input [(ngModel)]=\"file.title\"\r\n                     name=\"title{{i}}\"\r\n                     (ngModelChange)=\"onMetaChange()\"\r\n                     placeholder=\"Enter title\"\r\n                     class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n          \r\n              <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n              <textarea [(ngModel)]=\"file.notes\"\r\n                        name=\"notes{{i}}\"\r\n                        (ngModelChange)=\"onMetaChange()\"\r\n                        placeholder=\"Enter notes\"\r\n                        class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n                        rows=\"2\"></textarea>\r\n            </div>\r\n          \r\n\r\n          </div>\r\n\r\n          <p>{{file?.name}}</p>\r\n        </div>\r\n      </ng-container>\r\n    </div> -->\r\n\r\n    <input #fileInput\r\n       type=\"file\"\r\n       [multiple]=\"setting('multiple-upload')\"\r\n       (change)=\"onFileSelected($event)\"\r\n       hidden />\r\n\r\n<div class=\"separator\"></div>\r\n\r\n<div class=\"file-list flex items-center gap-6 flex-wrap\" *ngIf=\"false\">\r\n  <ng-container *ngIf=\"uploadedFiles?.length\">\r\n    <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n         class=\"flex flex-col items-center justify-center\">\r\n\r\n      <div class=\"relative p-3 border rounded-lg shadow-sm bg-white w-64\">\r\n        <!-- Close icon -->\r\n        <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n            <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n            <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n                  fill=\"white\"/>\r\n          </svg>\r\n        </div>\r\n\r\n        <!-- Thumbnail -->\r\n        <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n          <img class=\"file-thumbnail w-full h-32 object-cover rounded mb-3\"\r\n               [src]=\"file?.result\"\r\n               alt=\"preview\" />\r\n        </ng-container>\r\n\r\n        <ng-template #nonImageThumb>\r\n          <div class=\"file-thumbnail flex items-center justify-center border rounded mb-3 w-full h-32 bg-gray-50\">\r\n            <ng-container [ngSwitch]=\"file.type\">\r\n              <!-- CSV -->\r\n              <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n                <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n              </svg>\r\n\r\n              <!-- TEXT -->\r\n              <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n                <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n              </svg>\r\n\r\n              <!-- Generic -->\r\n              <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n                <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n              </svg>\r\n            </ng-container>\r\n          </div>\r\n        </ng-template>\r\n\r\n        <!-- Metadata -->\r\n        <div class=\"flex flex-col w-full\">\r\n          <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n          <input [(ngModel)]=\"file.title\"\r\n                 name=\"title{{i}}\"\r\n                 (ngModelChange)=\"onMetaChange()\"\r\n                 placeholder=\"Enter title\"\r\n                 class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n          <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n          <textarea [(ngModel)]=\"file.notes\"\r\n                    name=\"notes{{i}}\"\r\n                    (ngModelChange)=\"onMetaChange()\"\r\n                    placeholder=\"Enter notes\"\r\n                    class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\"\r\n                    rows=\"2\"></textarea>\r\n\r\n          <label class=\"font-semibold text-sm mt-3 mb-1\">Category</label>\r\n          <select [(ngModel)]=\"file.category\"\r\n                  name=\"category{{i}}\"\r\n                  (ngModelChange)=\"onMetaChange()\"\r\n                   placeholder=\"Select Category\"\r\n                  class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n            <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n          </select>\r\n        </div>\r\n      </div>\r\n\r\n      <p class=\"mt-2 text-sm text-gray-600\">{{ file?.name }}</p>\r\n    </div>\r\n  </ng-container>\r\n</div>\r\n\r\n<div class=\"file-list flex items-center gap-8 flex-wrap\">\r\n  <ng-container *ngIf=\"uploadedFiles?.length\">\r\n    <div *ngFor=\"let file of uploadedFiles; let i = index\"\r\n         class=\"relative p-3 border rounded-lg shadow-sm bg-white w-[500px] flex\">\r\n\r\n      <!-- Close icon -->\r\n      <div class=\"close_icon cursor-pointer absolute top-2 right-2\" (click)=\"deleteFile(i)\">\r\n        <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"18\" height=\"18\" viewBox=\"0 0 18 18\" fill=\"none\">\r\n          <rect width=\"18\" height=\"18\" rx=\"9\" fill=\"#F3A041\" />\r\n          <path d=\"M8.2 9l-2.4-2.4.8-.8L9 8.2l2.4-2.4.8.8L9.8 9l2.4 2.4-.8.8L9 9.8l-2.4 2.4-.8-.8L8.2 9z\"\r\n                fill=\"white\"/>\r\n        </svg>\r\n      </div>\r\n\r\n      <!-- Thumbnail on left -->\r\n      <div class=\"w-28 h-28 flex-shrink-0\">\r\n        <ng-container *ngIf=\"file.type === 'image'; else nonImageThumb\">\r\n          <img class=\"file-thumbnail w-full h-full object-cover rounded\"\r\n               [src]=\"file?.result\"\r\n               alt=\"preview\" />\r\n        </ng-container>\r\n\r\n        <ng-template #nonImageThumb>\r\n          <div class=\"file-thumbnail flex items-center justify-center border rounded w-full h-full bg-gray-50\">\r\n            <ng-container [ngSwitch]=\"file.type\">\r\n              <!-- CSV -->\r\n              <svg *ngSwitchCase=\"'csv'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#22c55e\" stroke-width=\"2\"/>\r\n                <text x=\"14\" y=\"34\" font-size=\"10\" font-family=\"monospace\" fill=\"#22c55e\">CSV</text>\r\n              </svg>\r\n              <!-- TEXT -->\r\n              <svg *ngSwitchCase=\"'text'\" xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#3b82f6\" stroke-width=\"2\"/>\r\n                <path d=\"M14 24h20M14 29h20M14 34h12\" stroke=\"#3b82f6\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n              </svg>\r\n              <!-- Generic -->\r\n              <svg *ngSwitchDefault xmlns=\"http://www.w3.org/2000/svg\" width=\"48\" height=\"48\" viewBox=\"0 0 48 48\" fill=\"none\">\r\n                <rect x=\"8\" y=\"6\" width=\"28\" height=\"36\" rx=\"3\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n                <path d=\"M28 6v8a2 2 0 0 0 2 2h6\" stroke=\"#9ca3af\" stroke-width=\"2\"/>\r\n                <path d=\"M16 30h16\" stroke=\"#9ca3af\" stroke-width=\"2\" stroke-linecap=\"round\"/>\r\n              </svg>\r\n            </ng-container>\r\n          </div>\r\n        </ng-template>\r\n      </div>\r\n\r\n      <!-- Metadata on right -->\r\n      <div class=\"flex flex-col flex-1 ml-4\">\r\n        <!-- Title -->\r\n        <label class=\"font-semibold text-sm mb-1\">Title</label>\r\n        <input [(ngModel)]=\"file.title\"\r\n               name=\"title{{i}}\"\r\n               (ngModelChange)=\"onMetaChange()\"\r\n               placeholder=\"Enter title\"\r\n               class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\" />\r\n\r\n        <!-- Notes -->\r\n        <label class=\"font-semibold text-sm mb-1\">Notes</label>\r\n        <textarea [(ngModel)]=\"file.notes\"\r\n                  name=\"notes{{i}}\"\r\n                  (ngModelChange)=\"onMetaChange()\"\r\n                  placeholder=\"Enter notes\"\r\n                  class=\"border border-orange-300 rounded px-3 py-2 mb-3 w-full outline-none focus:ring focus:ring-orange-200\"\r\n                  rows=\"2\"></textarea>\r\n\r\n        <!-- Category -->\r\n        <label class=\"font-semibold text-sm mb-1\">Category</label>\r\n        <select [(ngModel)]=\"file.category\"\r\n                name=\"category{{i}}\"\r\n                (ngModelChange)=\"onMetaChange()\"\r\n                class=\"border border-orange-300 rounded px-3 py-2 w-full outline-none focus:ring focus:ring-orange-200\">\r\n                <option value=\"\">Select Category</option>\r\n          <option *ngFor=\"let c of categories\" [ngValue]=\"c\">{{ c }}</option>\r\n        </select>\r\n      </div>\r\n    </div>\r\n  </ng-container>\r\n</div>\r\n\r\n\r\n\r\n  </div>\r\n</fx-component>"]}