@bnsights/bbsf-controls 1.0.175 → 1.0.177
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -1
- package/esm2022/lib/Shared/Models/FileDTO.mjs +2 -1
- package/esm2022/lib/controls/FileUpload/FileUpload.component.mjs +1048 -0
- package/esm2022/lib/controls/Repeater/repeater-field-builder/repeater-field-builder.component.mjs +2 -2
- package/esm2022/lib/controls/bbsf-controls.module.mjs +5 -3
- package/esm2022/public-api.mjs +2 -2
- package/fesm2022/bnsights-bbsf-controls.mjs +909 -446
- package/fesm2022/bnsights-bbsf-controls.mjs.map +1 -1
- package/lib/Shared/Models/FileDTO.d.ts +1 -0
- package/lib/controls/FileUpload/FileUpload.component.d.ts +162 -0
- package/lib/controls/bbsf-controls.module.d.ts +1 -1
- package/package.json +2 -2
- package/public-api.d.ts +1 -1
- package/esm2022/lib/controls/FileUplaod/FileUplaod.component.mjs +0 -588
- package/lib/controls/FileUplaod/FileUplaod.component.d.ts +0 -62
|
@@ -0,0 +1,1048 @@
|
|
|
1
|
+
import { Component, Input, Optional, ViewChild, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { FormControl, Validators } from '@angular/forms';
|
|
3
|
+
import { FileUploader } from 'ng2-file-upload';
|
|
4
|
+
import { FileUploadModel } from '../../Shared/Models/FileUploadModel';
|
|
5
|
+
import { MultipleFileUploadModel } from '../../Shared/Models/MultipleFileUploadModel';
|
|
6
|
+
import { HttpEventType } from '@angular/common/http';
|
|
7
|
+
import { Subscription } from 'rxjs';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/forms";
|
|
10
|
+
import * as i2 from "../../Shared/services/ControlUtility";
|
|
11
|
+
import * as i3 from "@bnsights/bbsf-utilities";
|
|
12
|
+
import * as i4 from "../../Shared/services/GlobalSettings.service";
|
|
13
|
+
import * as i5 from "../../Shared/services/file-upload.service";
|
|
14
|
+
import * as i6 from "@angular/common";
|
|
15
|
+
import * as i7 from "ng2-file-upload";
|
|
16
|
+
export class FileUploadComponent {
|
|
17
|
+
static { this.controlContainerStatic = null; }
|
|
18
|
+
constructor(controlContainer, multipleFileUploadControlHost, controlUtility, utilityService, controlValidationService, globalSettings, fileUploadService) {
|
|
19
|
+
this.controlContainer = controlContainer;
|
|
20
|
+
this.multipleFileUploadControlHost = multipleFileUploadControlHost;
|
|
21
|
+
this.controlUtility = controlUtility;
|
|
22
|
+
this.utilityService = utilityService;
|
|
23
|
+
this.controlValidationService = controlValidationService;
|
|
24
|
+
this.globalSettings = globalSettings;
|
|
25
|
+
this.fileUploadService = fileUploadService;
|
|
26
|
+
this.BYTES_TO_MB = 1024 * 1024;
|
|
27
|
+
this.PROGRESS_COMPLETE = 100;
|
|
28
|
+
this.PROGRESS_NEAR_COMPLETE = 95;
|
|
29
|
+
this.ERROR_DISPLAY_DURATION = 5000;
|
|
30
|
+
this.MAX_MEMORY_USAGE = 100 * 1024 * 1024; // 100MB limit
|
|
31
|
+
this.currentMemoryUsage = 0;
|
|
32
|
+
this.isSubmitted = false;
|
|
33
|
+
this.OnChange = new EventEmitter();
|
|
34
|
+
this.isUploadComplete = new EventEmitter();
|
|
35
|
+
this.validationMessage = '';
|
|
36
|
+
this.validationCountMessage = '';
|
|
37
|
+
this.hasAnotherDropZoneOver = false;
|
|
38
|
+
this.acceptedType = '';
|
|
39
|
+
this.acceptedTypeArray = [];
|
|
40
|
+
this.toolTipTypeArray = [];
|
|
41
|
+
this.markAllAsTouched = false;
|
|
42
|
+
this.validationRules = [];
|
|
43
|
+
this.validationRulesAsync = [];
|
|
44
|
+
this.file = null;
|
|
45
|
+
this.deletedFiles = [];
|
|
46
|
+
this.subscriptions = new Subscription();
|
|
47
|
+
this.resetError = () => {
|
|
48
|
+
this.controlValidationService.removeGlobalError();
|
|
49
|
+
};
|
|
50
|
+
this.removeRequiredValidation = () => {
|
|
51
|
+
this.controlUtility.removeRequiredValidation(this.fileUploadFormControl, this.validationRules, this.options);
|
|
52
|
+
};
|
|
53
|
+
this.addRequiredValidation = () => {
|
|
54
|
+
this.controlUtility.addRequiredValidation(this.fileUploadFormControl, this.validationRules, this.options);
|
|
55
|
+
};
|
|
56
|
+
this.removeCustomValidation = (customValidation) => {
|
|
57
|
+
this.controlUtility.removeCustomValidation(this.fileUploadFormControl, this.validationRules, customValidation);
|
|
58
|
+
};
|
|
59
|
+
this.addCustomValidation = (customValidation) => {
|
|
60
|
+
this.controlUtility.addCustomValidation(this.fileUploadFormControl, this.validationRules, customValidation);
|
|
61
|
+
};
|
|
62
|
+
this.isValid = () => {
|
|
63
|
+
this.controlUtility.isValid(this.fileUploadFormControl);
|
|
64
|
+
return this.fileUploadFormControl.valid;
|
|
65
|
+
};
|
|
66
|
+
FileUploadComponent.controlContainerStatic = this.controlContainer;
|
|
67
|
+
this.initializeUploader();
|
|
68
|
+
}
|
|
69
|
+
initializeUploader() {
|
|
70
|
+
this.uploader = new FileUploader({
|
|
71
|
+
disableMultipart: false
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
ngOnInit() {
|
|
75
|
+
this.initializeModels();
|
|
76
|
+
this.setViewType();
|
|
77
|
+
this.processInitialValue();
|
|
78
|
+
this.setupLabels();
|
|
79
|
+
this.setupFileTypeValidation();
|
|
80
|
+
this.setupFormControl();
|
|
81
|
+
this.setupSubscriptions();
|
|
82
|
+
}
|
|
83
|
+
initializeModels() {
|
|
84
|
+
this.fileUploadModel = new FileUploadModel();
|
|
85
|
+
this.multipleFileUploadModel = new MultipleFileUploadModel();
|
|
86
|
+
}
|
|
87
|
+
setViewType() {
|
|
88
|
+
if (!this.options.viewType) {
|
|
89
|
+
this.options.viewType = this.globalSettings.viewType;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
processInitialValue() {
|
|
93
|
+
if (this.options.value == null) {
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
if (this.options.isMultipleFile) {
|
|
97
|
+
this.processMultipleFileValue();
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
this.processSingleFileValue();
|
|
101
|
+
}
|
|
102
|
+
this.uploader.queue.forEach((element) => {
|
|
103
|
+
element.progress = this.PROGRESS_COMPLETE;
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
processMultipleFileValue() {
|
|
107
|
+
const files = [];
|
|
108
|
+
this.multipleFileUploadModel.existingFiles = this.options.value.existingFiles;
|
|
109
|
+
this.multipleFileUploadModel.uploadedFiles = [];
|
|
110
|
+
for (const element of this.options.value.existingFiles) {
|
|
111
|
+
const fileLikeObject = this.createFileLikeObject(element);
|
|
112
|
+
files.push(fileLikeObject);
|
|
113
|
+
}
|
|
114
|
+
this.uploader.addToQueue(files);
|
|
115
|
+
}
|
|
116
|
+
processSingleFileValue() {
|
|
117
|
+
const element = this.options.value.file ?? this.options.value;
|
|
118
|
+
const fileLikeObject = this.createFileLikeObject(element);
|
|
119
|
+
this.file = element;
|
|
120
|
+
this.uploader.addToQueue([fileLikeObject]);
|
|
121
|
+
if (!this.options.value.file) {
|
|
122
|
+
this.fileUploadModel = new FileUploadModel();
|
|
123
|
+
this.fileUploadModel.file = this.options.value;
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
this.fileUploadModel = this.options.value;
|
|
127
|
+
}
|
|
128
|
+
this.options.value = this.fileUploadModel;
|
|
129
|
+
}
|
|
130
|
+
createFileLikeObject(element) {
|
|
131
|
+
const bytes = new Uint8Array(element.bytes);
|
|
132
|
+
const base64 = btoa(String.fromCharCode(...bytes));
|
|
133
|
+
return {
|
|
134
|
+
name: element.nameWithExtension || element.fileName,
|
|
135
|
+
type: element.mimeType || element.fileType,
|
|
136
|
+
rawFile: base64,
|
|
137
|
+
size: element.fileSizeInMB ? element.fileSizeInMB * this.BYTES_TO_MB : 0,
|
|
138
|
+
lastModifiedDate: new Date(),
|
|
139
|
+
url: element.fullFileURL
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
setupLabels() {
|
|
143
|
+
if (this.options.labelKey) {
|
|
144
|
+
this.options.labelValue = this.utilityService.getResourceValue(this.options.labelKey);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
setupFileTypeValidation() {
|
|
148
|
+
if (this.options.fileUploadAcceptsTypes?.length) {
|
|
149
|
+
this.processAcceptedTypes();
|
|
150
|
+
this.buildValidationMessage();
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
processAcceptedTypes() {
|
|
154
|
+
this.acceptedType = this.options.fileUploadAcceptsTypes.join(',');
|
|
155
|
+
this.acceptedTypeArray = this.options.fileUploadAcceptsTypes.filter(type => type.trim());
|
|
156
|
+
const mimeTypeMap = this.getMimeTypeMap();
|
|
157
|
+
for (const type of this.acceptedTypeArray) {
|
|
158
|
+
const displayType = mimeTypeMap[type];
|
|
159
|
+
if (displayType && !this.toolTipTypeArray.includes(displayType)) {
|
|
160
|
+
this.toolTipTypeArray.push(displayType);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
getMimeTypeMap() {
|
|
165
|
+
return {
|
|
166
|
+
'application/pdf': 'PDF',
|
|
167
|
+
'application/msword': 'Word',
|
|
168
|
+
'application/vnd.openxmlformats-officedocument.wordprocessingml.document': 'Word',
|
|
169
|
+
'application/vnd.ms-excel': 'Excel',
|
|
170
|
+
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet': 'Excel',
|
|
171
|
+
'application/vnd.ms-powerpoint': 'PowerPoint',
|
|
172
|
+
'application/vnd.openxmlformats-officedocument.presentationml.presentation': 'PowerPoint',
|
|
173
|
+
'image/png': 'PNG',
|
|
174
|
+
'image/bmp': 'BMP',
|
|
175
|
+
'image/jpeg': 'JPEG',
|
|
176
|
+
'application/zip': 'ZIP',
|
|
177
|
+
'application/x-rar-compressed': 'RAR',
|
|
178
|
+
'video/mp4': 'MP4',
|
|
179
|
+
'video/avi': 'AVI',
|
|
180
|
+
'video/quicktime': 'MOV',
|
|
181
|
+
'video/mpeg': 'MPEG',
|
|
182
|
+
'audio/mpeg': 'MP3',
|
|
183
|
+
'video/x-flv': 'FLV',
|
|
184
|
+
'video/x-ms-wmv': 'WMV',
|
|
185
|
+
'image/svg+xml': 'SVG',
|
|
186
|
+
'text/plain': 'Txt',
|
|
187
|
+
'application/BN': 'License'
|
|
188
|
+
};
|
|
189
|
+
}
|
|
190
|
+
buildValidationMessage() {
|
|
191
|
+
const messages = [];
|
|
192
|
+
if (this.toolTipTypeArray.length) {
|
|
193
|
+
messages.push(`${this.utilityService.getResourceValue('Extensions')} (${this.toolTipTypeArray.join(', ')})`);
|
|
194
|
+
}
|
|
195
|
+
if (this.options.fileMaxSizeInMB > 0) {
|
|
196
|
+
messages.push(`${this.utilityService.getResourceValue('FileMaxSizeInMB')}${this.options.fileMaxSizeInMB}`);
|
|
197
|
+
}
|
|
198
|
+
if (this.options.minNoOfFiles > 0) {
|
|
199
|
+
messages.push(`${this.utilityService.getResourceValue('MinFileCountValidationKey')}${this.options.minNoOfFiles}`);
|
|
200
|
+
}
|
|
201
|
+
if (this.options.maxNoOfFiles > 0) {
|
|
202
|
+
messages.push(`${this.utilityService.getResourceValue('MaxFileCountValidationKey')}${this.options.maxNoOfFiles}`);
|
|
203
|
+
}
|
|
204
|
+
this.validationMessage = messages.join('<br/>');
|
|
205
|
+
}
|
|
206
|
+
setupFormControl() {
|
|
207
|
+
this.group.addControl(this.options.name, new FormControl(''));
|
|
208
|
+
this.fileUploadFormControl = this.group.controls[this.options.name];
|
|
209
|
+
this.setupValidators();
|
|
210
|
+
this.setupCountMessage();
|
|
211
|
+
this.applyValidatorsAndState();
|
|
212
|
+
this.fileUploadFormControl.setValue(this.options.value, { emitEvent: false });
|
|
213
|
+
}
|
|
214
|
+
setupValidators() {
|
|
215
|
+
if (this.options.customValidation?.length) {
|
|
216
|
+
for (const validation of this.options.customValidation) {
|
|
217
|
+
this.validationRules.push(validation.functionBody);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
if (this.options.isRequired) {
|
|
221
|
+
this.validationRules.push(Validators.required);
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
setupCountMessage() {
|
|
225
|
+
if (this.options.isMultipleFile && this.options.maxNoOfFiles > 0) {
|
|
226
|
+
this.validationCountMessage = `${this.utilityService.getResourceValue('MaxFilesCount')} : ${this.options.maxNoOfFiles}`;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
applyValidatorsAndState() {
|
|
230
|
+
this.fileUploadFormControl.setValidators(this.validationRules);
|
|
231
|
+
if (this.validationRulesAsync.length > 0) {
|
|
232
|
+
this.fileUploadFormControl.setAsyncValidators(this.validationRulesAsync);
|
|
233
|
+
}
|
|
234
|
+
if (this.options.isDisabled) {
|
|
235
|
+
this.fileUploadFormControl.disable();
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
setupSubscriptions() {
|
|
239
|
+
this.multipleFileUploadControlHost.ngSubmit.subscribe(() => {
|
|
240
|
+
this.group.markAllAsTouched();
|
|
241
|
+
this.markAllAsTouched = true;
|
|
242
|
+
});
|
|
243
|
+
}
|
|
244
|
+
ngAfterViewInit() {
|
|
245
|
+
this.applyAttributes();
|
|
246
|
+
}
|
|
247
|
+
applyAttributes() {
|
|
248
|
+
if (!this.options.attributeList?.length) {
|
|
249
|
+
return;
|
|
250
|
+
}
|
|
251
|
+
const element = document.getElementById(this.options.name);
|
|
252
|
+
if (element) {
|
|
253
|
+
for (const attribute of this.options.attributeList) {
|
|
254
|
+
element.setAttribute(attribute.key, attribute.value);
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
showGlobalError() {
|
|
259
|
+
this.controlUtility.showGlobalError();
|
|
260
|
+
}
|
|
261
|
+
getErrorValidation(errorList) {
|
|
262
|
+
if (this.markAllAsTouched && this.group.invalid) {
|
|
263
|
+
this.showGlobalError();
|
|
264
|
+
this.markAllAsTouched = false;
|
|
265
|
+
}
|
|
266
|
+
if (errorList && errorList.length > 0) {
|
|
267
|
+
for (const error of errorList) {
|
|
268
|
+
if (error.key === 'InvalidFiles') {
|
|
269
|
+
return error.value;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
return this.controlUtility.getErrorValidationMassage(errorList, this.group, this.options);
|
|
274
|
+
}
|
|
275
|
+
fileOverAnother(event) {
|
|
276
|
+
this.hasAnotherDropZoneOver = !!event;
|
|
277
|
+
}
|
|
278
|
+
isHideInput() {
|
|
279
|
+
if (this.options.isMultipleFile) {
|
|
280
|
+
return this.options.maxNoOfFiles > 0 &&
|
|
281
|
+
this.options.maxNoOfFiles === this.uploader.queue.length;
|
|
282
|
+
}
|
|
283
|
+
return this.uploader.queue.length > 0;
|
|
284
|
+
}
|
|
285
|
+
onFileChange() {
|
|
286
|
+
this.validateFileConstraints();
|
|
287
|
+
const fileProcessingResult = this.processNewlyAddedFiles();
|
|
288
|
+
this.handleFileValidationResults(fileProcessingResult);
|
|
289
|
+
this.processValidFilesForUpload();
|
|
290
|
+
}
|
|
291
|
+
processNewlyAddedFiles() {
|
|
292
|
+
const addedQueue = this.getNewlyAddedFiles();
|
|
293
|
+
const validationResult = this.validateFilesInQueue(addedQueue);
|
|
294
|
+
return {
|
|
295
|
+
validFiles: addedQueue.filter(file => !validationResult.invalidFiles.includes(file)),
|
|
296
|
+
invalidFiles: validationResult.invalidFiles,
|
|
297
|
+
errors: validationResult.errors
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
getNewlyAddedFiles() {
|
|
301
|
+
return this.uploader.queue.filter((obj) => obj['some']?.lastModified != null);
|
|
302
|
+
}
|
|
303
|
+
validateFilesInQueue(fileQueue) {
|
|
304
|
+
const validationErrors = [];
|
|
305
|
+
const invalidFiles = [];
|
|
306
|
+
const processedDuplicateNames = new Set();
|
|
307
|
+
for (const element of fileQueue) {
|
|
308
|
+
const file = element.file;
|
|
309
|
+
if (!file)
|
|
310
|
+
continue;
|
|
311
|
+
const fileValidation = this.validateSingleFile(file, element, processedDuplicateNames);
|
|
312
|
+
if (!fileValidation.isValid) {
|
|
313
|
+
invalidFiles.push(element);
|
|
314
|
+
validationErrors.push(...fileValidation.errors);
|
|
315
|
+
}
|
|
316
|
+
}
|
|
317
|
+
return {
|
|
318
|
+
isValid: invalidFiles.length === 0,
|
|
319
|
+
errors: validationErrors,
|
|
320
|
+
invalidFiles
|
|
321
|
+
};
|
|
322
|
+
}
|
|
323
|
+
validateSingleFile(file, element, processedDuplicateNames) {
|
|
324
|
+
const errors = [];
|
|
325
|
+
const sizeValid = this.validateIndividualFileSize(file);
|
|
326
|
+
const typeValid = this.validateIndividualFileType(file);
|
|
327
|
+
const nameValid = this.validateDuplicateFileName(file, element);
|
|
328
|
+
if (!sizeValid) {
|
|
329
|
+
errors.push(this.createFileSizeErrorMessage(file.name));
|
|
330
|
+
}
|
|
331
|
+
if (!typeValid) {
|
|
332
|
+
errors.push(this.createFileTypeErrorMessage(file.name));
|
|
333
|
+
}
|
|
334
|
+
if (!nameValid) {
|
|
335
|
+
const duplicateError = this.createDuplicateFileErrorMessage(file.name, processedDuplicateNames);
|
|
336
|
+
if (duplicateError) {
|
|
337
|
+
errors.push(duplicateError);
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
return {
|
|
341
|
+
isValid: sizeValid && typeValid && nameValid,
|
|
342
|
+
errors,
|
|
343
|
+
invalidFiles: []
|
|
344
|
+
};
|
|
345
|
+
}
|
|
346
|
+
createFileSizeErrorMessage(fileName) {
|
|
347
|
+
return this.utilityService.getResourceValue('FileExceedsMaxSize')
|
|
348
|
+
.replace('{fileName}', fileName)
|
|
349
|
+
.replace('{maxSize}', this.options.fileMaxSizeInMB.toString());
|
|
350
|
+
}
|
|
351
|
+
createFileTypeErrorMessage(fileName) {
|
|
352
|
+
return this.utilityService.getResourceValue('FileTypeNotAccepted')
|
|
353
|
+
.replace('{fileName}', fileName);
|
|
354
|
+
}
|
|
355
|
+
createDuplicateFileErrorMessage(fileName, processedDuplicateNames) {
|
|
356
|
+
const fileNameLower = fileName.toLowerCase();
|
|
357
|
+
if (processedDuplicateNames.has(fileNameLower)) {
|
|
358
|
+
return null;
|
|
359
|
+
}
|
|
360
|
+
processedDuplicateNames.add(fileNameLower);
|
|
361
|
+
let duplicateErrorMsg = this.utilityService.getResourceValue('DuplicateFileName');
|
|
362
|
+
if (!duplicateErrorMsg || duplicateErrorMsg === 'DuplicateFileName') {
|
|
363
|
+
duplicateErrorMsg = `File '{fileName}' already exists. Please choose a different file or rename it.`;
|
|
364
|
+
}
|
|
365
|
+
return duplicateErrorMsg.replace('{fileName}', fileName);
|
|
366
|
+
}
|
|
367
|
+
handleFileValidationResults(result) {
|
|
368
|
+
if (result.invalidFiles.length > 0) {
|
|
369
|
+
this.removeInvalidFiles(result.invalidFiles);
|
|
370
|
+
}
|
|
371
|
+
if (result.errors.length > 0) {
|
|
372
|
+
this.showValidationErrors(result.errors);
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
processValidFilesForUpload() {
|
|
376
|
+
const filesArray = [];
|
|
377
|
+
const validQueue = this.getNewlyAddedFiles();
|
|
378
|
+
for (const element of validQueue) {
|
|
379
|
+
const file = element.file;
|
|
380
|
+
if (!file)
|
|
381
|
+
continue;
|
|
382
|
+
if (this.shouldUseAsyncUpload(element)) {
|
|
383
|
+
this.handleAsyncFileUpload(element, filesArray);
|
|
384
|
+
}
|
|
385
|
+
else {
|
|
386
|
+
this.handleSyncFileUpload(file, filesArray);
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
shouldUseAsyncUpload(element) {
|
|
391
|
+
return this.options.isUploadFileAsync && !element._file['iD_GUID'];
|
|
392
|
+
}
|
|
393
|
+
validateFileConstraints() {
|
|
394
|
+
if (this.options.isMultipleFile) {
|
|
395
|
+
if (!this.validateMinFileCount() || !this.validateMaxFileCount() || !this.validateTotalFileSize()) {
|
|
396
|
+
return false;
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
return true;
|
|
400
|
+
}
|
|
401
|
+
validateMinFileCount() {
|
|
402
|
+
if (this.options.minNoOfFiles > 0 && this.options.minNoOfFiles > this.uploader.queue.length) {
|
|
403
|
+
const minFileMsg = this.utilityService.getResourceValue('MinimumFilesRequired')
|
|
404
|
+
.replace('{count}', this.options.minNoOfFiles.toString());
|
|
405
|
+
this.showFileCountError('MinFileCountValidationKey', minFileMsg);
|
|
406
|
+
return false;
|
|
407
|
+
}
|
|
408
|
+
return true;
|
|
409
|
+
}
|
|
410
|
+
validateMaxFileCount() {
|
|
411
|
+
if (this.options.maxNoOfFiles > 0 && this.options.maxNoOfFiles < this.uploader.queue.length) {
|
|
412
|
+
const maxFileMsg = this.utilityService.getResourceValue('MaximumFilesExceeded') ||
|
|
413
|
+
`Maximum {maxCount} files allowed. You have selected {currentCount} files.`;
|
|
414
|
+
const finalMsg = maxFileMsg
|
|
415
|
+
.replace('{maxCount}', this.options.maxNoOfFiles.toString())
|
|
416
|
+
.replace('{currentCount}', this.uploader.queue.length.toString());
|
|
417
|
+
this.showFileCountError('MaxFileCountValidationKey', finalMsg);
|
|
418
|
+
return false;
|
|
419
|
+
}
|
|
420
|
+
return true;
|
|
421
|
+
}
|
|
422
|
+
showFileCountError(errorKey, message) {
|
|
423
|
+
const currentErrors = this.fileUploadFormControl.errors || {};
|
|
424
|
+
currentErrors[errorKey] = message;
|
|
425
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
426
|
+
this.fileUploadFormControl.markAsTouched();
|
|
427
|
+
}
|
|
428
|
+
clearFileCountError(errorKey) {
|
|
429
|
+
const currentErrors = this.fileUploadFormControl.errors;
|
|
430
|
+
if (currentErrors && currentErrors[errorKey]) {
|
|
431
|
+
delete currentErrors[errorKey];
|
|
432
|
+
if (Object.keys(currentErrors).length === 0) {
|
|
433
|
+
this.fileUploadFormControl.setErrors(null);
|
|
434
|
+
}
|
|
435
|
+
else {
|
|
436
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
437
|
+
}
|
|
438
|
+
}
|
|
439
|
+
}
|
|
440
|
+
validateTotalFileSize() {
|
|
441
|
+
if (this.options.maxSizeForAllFilesInMB > 0) {
|
|
442
|
+
const totalSize = this.uploader.queue.reduce((sum, element) => sum + element.file.size, 0);
|
|
443
|
+
const maxSizeBytes = this.options.maxSizeForAllFilesInMB * this.BYTES_TO_MB;
|
|
444
|
+
if (totalSize > maxSizeBytes) {
|
|
445
|
+
this.showTotalSizeError();
|
|
446
|
+
return false;
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
return true;
|
|
450
|
+
}
|
|
451
|
+
showTotalSizeError() {
|
|
452
|
+
const totalSizeMsg = this.utilityService.getResourceValue('TotalFileSizeExceeded')
|
|
453
|
+
.replace('{maxSize}', this.options.maxSizeForAllFilesInMB.toString());
|
|
454
|
+
const currentErrors = this.fileUploadFormControl.errors || {};
|
|
455
|
+
currentErrors['MaxSizeForAllFilesInMB'] = totalSizeMsg;
|
|
456
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
457
|
+
this.fileUploadFormControl.markAsTouched();
|
|
458
|
+
}
|
|
459
|
+
validateFileSize(file) {
|
|
460
|
+
const maxFileSize = this.options.fileMaxSizeInMB * this.BYTES_TO_MB;
|
|
461
|
+
if (file.size > maxFileSize) {
|
|
462
|
+
this.setFormControlError('FileMaxSizeInMB', `${this.options.fileMaxSizeInMB}MB`);
|
|
463
|
+
return false;
|
|
464
|
+
}
|
|
465
|
+
return true;
|
|
466
|
+
}
|
|
467
|
+
validateFileType(file) {
|
|
468
|
+
if (this.options.fileUploadAcceptsTypes?.length) {
|
|
469
|
+
const fileType = file.type;
|
|
470
|
+
const isAccepted = this.acceptedTypeArray.some(type => type.toLowerCase() === fileType.toLowerCase());
|
|
471
|
+
if (!isAccepted) {
|
|
472
|
+
this.setFormControlError('ToolTipTypeError', this.toolTipTypeArray);
|
|
473
|
+
return false;
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
return true;
|
|
477
|
+
}
|
|
478
|
+
validateIndividualFileSize(file) {
|
|
479
|
+
const maxFileSize = this.options.fileMaxSizeInMB * this.BYTES_TO_MB;
|
|
480
|
+
return file.size <= maxFileSize;
|
|
481
|
+
}
|
|
482
|
+
validateIndividualFileType(file) {
|
|
483
|
+
if (!this.options.fileUploadAcceptsTypes?.length) {
|
|
484
|
+
return true;
|
|
485
|
+
}
|
|
486
|
+
const fileType = file.type;
|
|
487
|
+
return this.acceptedTypeArray.some(type => type.toLowerCase() === fileType.toLowerCase());
|
|
488
|
+
}
|
|
489
|
+
validateDuplicateFileName(file, currentElement) {
|
|
490
|
+
if (!file?.name) {
|
|
491
|
+
return true;
|
|
492
|
+
}
|
|
493
|
+
const currentFileName = file.name.toLowerCase();
|
|
494
|
+
const existingFiles = this.uploader.queue.filter(item => item !== currentElement);
|
|
495
|
+
const duplicateExists = existingFiles.some(item => {
|
|
496
|
+
const existingFileName = item.file?.name || item._file?.name;
|
|
497
|
+
return existingFileName && existingFileName.toLowerCase() === currentFileName;
|
|
498
|
+
});
|
|
499
|
+
if (duplicateExists) {
|
|
500
|
+
return false;
|
|
501
|
+
}
|
|
502
|
+
if (this.options.value) {
|
|
503
|
+
const uploadedFiles = Array.isArray(this.options.value) ? this.options.value : [this.options.value];
|
|
504
|
+
const duplicateInUploaded = uploadedFiles.some(uploadedFile => {
|
|
505
|
+
const uploadedFileName = uploadedFile?.fileName || uploadedFile?.name;
|
|
506
|
+
return uploadedFileName && uploadedFileName.toLowerCase() === currentFileName;
|
|
507
|
+
});
|
|
508
|
+
return !duplicateInUploaded;
|
|
509
|
+
}
|
|
510
|
+
return true;
|
|
511
|
+
}
|
|
512
|
+
removeInvalidFiles(invalidFiles) {
|
|
513
|
+
invalidFiles.forEach(invalidFile => {
|
|
514
|
+
const index = this.uploader.queue.indexOf(invalidFile);
|
|
515
|
+
if (index > -1) {
|
|
516
|
+
this.uploader.queue.splice(index, 1);
|
|
517
|
+
}
|
|
518
|
+
});
|
|
519
|
+
}
|
|
520
|
+
showValidationErrors(errors) {
|
|
521
|
+
if (errors.length === 0)
|
|
522
|
+
return;
|
|
523
|
+
const errorMessage = errors.join('<br/>');
|
|
524
|
+
this.setValidationError('InvalidFiles', errorMessage);
|
|
525
|
+
setTimeout(() => {
|
|
526
|
+
this.clearInvalidFilesError();
|
|
527
|
+
}, this.ERROR_DISPLAY_DURATION);
|
|
528
|
+
}
|
|
529
|
+
setValidationError(errorKey, errorMessage) {
|
|
530
|
+
const currentErrors = this.fileUploadFormControl.errors || {};
|
|
531
|
+
currentErrors[errorKey] = errorMessage;
|
|
532
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
533
|
+
this.fileUploadFormControl.markAsTouched();
|
|
534
|
+
}
|
|
535
|
+
clearInvalidFilesError() {
|
|
536
|
+
const currentErrors = this.fileUploadFormControl.errors;
|
|
537
|
+
if (currentErrors && currentErrors['InvalidFiles']) {
|
|
538
|
+
delete currentErrors['InvalidFiles'];
|
|
539
|
+
if (Object.keys(currentErrors).length === 0) {
|
|
540
|
+
this.fileUploadFormControl.setErrors(null);
|
|
541
|
+
}
|
|
542
|
+
else {
|
|
543
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
544
|
+
}
|
|
545
|
+
}
|
|
546
|
+
}
|
|
547
|
+
setFormControlError(errorKey, errorValue) {
|
|
548
|
+
this.fileUploadFormControl.setErrors({ [errorKey]: errorValue });
|
|
549
|
+
this.fileUploadFormControl.markAsTouched();
|
|
550
|
+
this.uploader.queue = [];
|
|
551
|
+
}
|
|
552
|
+
handleAsyncFileUpload(element, filesArray) {
|
|
553
|
+
const uploadSubscription = this.fileUploadService.uploadFile(element._file).subscribe({
|
|
554
|
+
next: (event) => {
|
|
555
|
+
if (event.type === HttpEventType.UploadProgress) {
|
|
556
|
+
this.handleUploadProgress(element, event);
|
|
557
|
+
}
|
|
558
|
+
else if (event.type === HttpEventType.Response) {
|
|
559
|
+
this.handleUploadComplete(element, event, filesArray);
|
|
560
|
+
}
|
|
561
|
+
},
|
|
562
|
+
error: (error) => {
|
|
563
|
+
console.error('Upload failed:', error);
|
|
564
|
+
// Handle upload error - you can add custom error handling here
|
|
565
|
+
}
|
|
566
|
+
});
|
|
567
|
+
// Store subscription for cleanup
|
|
568
|
+
this.subscriptions.add(uploadSubscription);
|
|
569
|
+
}
|
|
570
|
+
handleUploadProgress(element, event) {
|
|
571
|
+
const queueIndex = this.uploader.queue.findIndex((file) => file === element);
|
|
572
|
+
if (queueIndex === -1)
|
|
573
|
+
return;
|
|
574
|
+
const progress = Math.round((100 * event.loaded) / event.total);
|
|
575
|
+
this.uploader.queue[queueIndex].progress =
|
|
576
|
+
progress >= this.PROGRESS_NEAR_COMPLETE ? this.PROGRESS_NEAR_COMPLETE : progress;
|
|
577
|
+
}
|
|
578
|
+
handleUploadComplete(element, event, filesArray) {
|
|
579
|
+
const queueIndex = this.uploader.queue.findIndex((file) => file === element);
|
|
580
|
+
if (queueIndex === -1)
|
|
581
|
+
return;
|
|
582
|
+
this.uploader.queue[queueIndex].progress = this.PROGRESS_COMPLETE;
|
|
583
|
+
const fileID = event.body.val;
|
|
584
|
+
this.updateElementWithFileInfo(element, fileID, event.body.downloadUrl);
|
|
585
|
+
const addedFile = this.createFileDTO(element, fileID, event.body.downloadUrl);
|
|
586
|
+
this.updateFormValue(addedFile, filesArray);
|
|
587
|
+
}
|
|
588
|
+
updateElementWithFileInfo(element, fileID, downloadUrl) {
|
|
589
|
+
element._file['iD_GUID'] = fileID;
|
|
590
|
+
element._file['isNew'] = true;
|
|
591
|
+
if (downloadUrl) {
|
|
592
|
+
element._file['url'] = downloadUrl;
|
|
593
|
+
element.file.url = downloadUrl;
|
|
594
|
+
}
|
|
595
|
+
}
|
|
596
|
+
createFileDTO(element, fileID, downloadUrl) {
|
|
597
|
+
return {
|
|
598
|
+
iD_GUID: fileID,
|
|
599
|
+
fileName: element._file['name'],
|
|
600
|
+
fileType: element._file['type'],
|
|
601
|
+
isNew: true,
|
|
602
|
+
fileBase64: '',
|
|
603
|
+
fileSizeInMB: element._file.size / this.BYTES_TO_MB,
|
|
604
|
+
nameWithExtension: element._file['name'],
|
|
605
|
+
fullFileURL: downloadUrl || null
|
|
606
|
+
};
|
|
607
|
+
}
|
|
608
|
+
handleSyncFileUpload(file, filesArray) {
|
|
609
|
+
this.trackMemoryUsage(file.size);
|
|
610
|
+
const reader = new FileReader();
|
|
611
|
+
// Store reader reference for cleanup
|
|
612
|
+
const readerRef = reader;
|
|
613
|
+
reader.onload = () => {
|
|
614
|
+
try {
|
|
615
|
+
const existingGUID = this.getExistingFileGUID(file);
|
|
616
|
+
this.updateQueueItemForSync(file);
|
|
617
|
+
const addedFile = this.createSyncFileDTO(file, reader.result, existingGUID);
|
|
618
|
+
this.updateFormValue(addedFile, filesArray);
|
|
619
|
+
this.handlePatchAndEmit();
|
|
620
|
+
}
|
|
621
|
+
finally {
|
|
622
|
+
// Clean up reader
|
|
623
|
+
readerRef.onload = null;
|
|
624
|
+
readerRef.onerror = null;
|
|
625
|
+
readerRef.onabort = null;
|
|
626
|
+
}
|
|
627
|
+
};
|
|
628
|
+
reader.onerror = () => {
|
|
629
|
+
console.error('File reading failed');
|
|
630
|
+
readerRef.onload = null;
|
|
631
|
+
readerRef.onerror = null;
|
|
632
|
+
readerRef.onabort = null;
|
|
633
|
+
};
|
|
634
|
+
reader.readAsDataURL(file.rawFile);
|
|
635
|
+
}
|
|
636
|
+
updateQueueItemForSync(file) {
|
|
637
|
+
const queueItem = this.findQueueItemByFile(file);
|
|
638
|
+
if (queueItem) {
|
|
639
|
+
this.preserveFileReference(queueItem, file);
|
|
640
|
+
queueItem.progress = this.PROGRESS_COMPLETE;
|
|
641
|
+
}
|
|
642
|
+
}
|
|
643
|
+
findQueueItemByFile(file) {
|
|
644
|
+
return this.uploader.queue.find(item => item.file.name === file.name && item.file.size === file.size);
|
|
645
|
+
}
|
|
646
|
+
preserveFileReference(queueItem, file) {
|
|
647
|
+
queueItem._file.rawFile = file.rawFile;
|
|
648
|
+
queueItem.file.rawFile = file.rawFile;
|
|
649
|
+
}
|
|
650
|
+
createSyncFileDTO(file, readerResult, existingGUID) {
|
|
651
|
+
return {
|
|
652
|
+
fileName: file.name,
|
|
653
|
+
fileType: file.type,
|
|
654
|
+
fileBase64: readerResult.split(',')[1],
|
|
655
|
+
fileSizeInMB: file.size / this.BYTES_TO_MB,
|
|
656
|
+
nameWithExtension: file.name,
|
|
657
|
+
iD_GUID: existingGUID,
|
|
658
|
+
isNew: true,
|
|
659
|
+
fullFileURL: null
|
|
660
|
+
};
|
|
661
|
+
}
|
|
662
|
+
getExistingFileGUID(file) {
|
|
663
|
+
if (!this.options.isMultipleFile && this.file) {
|
|
664
|
+
return this.file.nameWithExtension === file.name ? this.file.iD_GUID : null;
|
|
665
|
+
}
|
|
666
|
+
return null;
|
|
667
|
+
}
|
|
668
|
+
updateFormValue(addedFile, filesArray) {
|
|
669
|
+
if (!this.options.isMultipleFile) {
|
|
670
|
+
this.fileUploadModel = new FileUploadModel();
|
|
671
|
+
this.fileUploadModel.file = addedFile;
|
|
672
|
+
this.updateFormControl(this.fileUploadModel);
|
|
673
|
+
}
|
|
674
|
+
else {
|
|
675
|
+
filesArray.push(addedFile);
|
|
676
|
+
this.multipleFileUploadModel.uploadedFiles = filesArray;
|
|
677
|
+
this.setupMultipleFileModel();
|
|
678
|
+
this.updateFormControl(this.multipleFileUploadModel);
|
|
679
|
+
this.isUploadComplete.emit(true);
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
setupMultipleFileModel() {
|
|
683
|
+
if (this.options.value?.correlationID_GUID == null) {
|
|
684
|
+
this.multipleFileUploadModel.removedFiles = [];
|
|
685
|
+
}
|
|
686
|
+
this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;
|
|
687
|
+
}
|
|
688
|
+
updateFormControl(value) {
|
|
689
|
+
this.preserveErrorsAndUpdateValue(value);
|
|
690
|
+
this.options.value = value;
|
|
691
|
+
}
|
|
692
|
+
preserveErrorsAndUpdateValue(value) {
|
|
693
|
+
const currentErrors = this.fileUploadFormControl.errors;
|
|
694
|
+
this.fileUploadFormControl.setValue(value, { emitEvent: false });
|
|
695
|
+
this.group.get(this.options.name)?.setValue(value, { emitEvent: false });
|
|
696
|
+
if (currentErrors) {
|
|
697
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
698
|
+
}
|
|
699
|
+
}
|
|
700
|
+
handlePatchAndEmit() {
|
|
701
|
+
const originalValue = this.group.get(this.options.name)?.value;
|
|
702
|
+
if (this.options.patchFunction && this.options.patchPath && this.group.get(this.options.name)?.valid) {
|
|
703
|
+
this.controlUtility.patchControlValue(originalValue, this.options.patchFunction, this.options.patchPath);
|
|
704
|
+
}
|
|
705
|
+
this.OnChange.emit(originalValue);
|
|
706
|
+
}
|
|
707
|
+
removeFromControlValue(item) {
|
|
708
|
+
// Clean up blob URL before removing
|
|
709
|
+
const downloadUrl = this.getFileDownloadUrl(item);
|
|
710
|
+
if (downloadUrl && downloadUrl.startsWith('blob:')) {
|
|
711
|
+
this.cleanupBlobUrl(downloadUrl);
|
|
712
|
+
}
|
|
713
|
+
this.handleAsyncFileDeletion(item);
|
|
714
|
+
if (!this.options.isMultipleFile) {
|
|
715
|
+
this.handleSingleFileRemoval();
|
|
716
|
+
}
|
|
717
|
+
else {
|
|
718
|
+
this.handleMultipleFileRemoval(item);
|
|
719
|
+
}
|
|
720
|
+
this.checkAndClearMaxFileCountValidation();
|
|
721
|
+
}
|
|
722
|
+
handleAsyncFileDeletion(item) {
|
|
723
|
+
if (this.options.isUploadFileAsync &&
|
|
724
|
+
item.progress === this.PROGRESS_COMPLETE &&
|
|
725
|
+
item._file['isNew']) {
|
|
726
|
+
const deleteSubscription = this.fileUploadService.deleteFile(item._file['iD_GUID']).subscribe({
|
|
727
|
+
error: (error) => console.error('Delete failed:', error)
|
|
728
|
+
});
|
|
729
|
+
this.subscriptions.add(deleteSubscription);
|
|
730
|
+
}
|
|
731
|
+
}
|
|
732
|
+
handleSingleFileRemoval() {
|
|
733
|
+
this.uploader.queue = [];
|
|
734
|
+
this.fileUploadModel = null;
|
|
735
|
+
if (this.options.isRequired) {
|
|
736
|
+
this.fileUploadFormControl.markAsTouched();
|
|
737
|
+
}
|
|
738
|
+
const currentErrors = this.fileUploadFormControl.errors;
|
|
739
|
+
this.group.get(this.options.name)?.setValue(this.fileUploadModel, { emitEvent: false });
|
|
740
|
+
if (currentErrors) {
|
|
741
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
742
|
+
}
|
|
743
|
+
this.options.value = this.fileUploadModel;
|
|
744
|
+
}
|
|
745
|
+
handleMultipleFileRemoval(item) {
|
|
746
|
+
// Clean up blob URL
|
|
747
|
+
const downloadUrl = this.getFileDownloadUrl(item);
|
|
748
|
+
if (downloadUrl && downloadUrl.startsWith('blob:')) {
|
|
749
|
+
this.cleanupBlobUrl(downloadUrl);
|
|
750
|
+
}
|
|
751
|
+
const queueIndex = this.uploader.queue.indexOf(item);
|
|
752
|
+
if (queueIndex > -1) {
|
|
753
|
+
this.uploader.queue.splice(queueIndex, 1);
|
|
754
|
+
}
|
|
755
|
+
this.processFileRemovalFromExisting(item);
|
|
756
|
+
this.removeFromUploadedFiles(item);
|
|
757
|
+
this.validateRemainingFiles();
|
|
758
|
+
this.updateMultipleFileModel();
|
|
759
|
+
}
|
|
760
|
+
processFileRemovalFromExisting(item) {
|
|
761
|
+
if (!this.options.value) {
|
|
762
|
+
this.resetDeletedFiles();
|
|
763
|
+
return;
|
|
764
|
+
}
|
|
765
|
+
if (!this.options.value.correlationID_GUID) {
|
|
766
|
+
this.resetDeletedFiles();
|
|
767
|
+
}
|
|
768
|
+
else {
|
|
769
|
+
this.handleExistingFileRemoval(item);
|
|
770
|
+
}
|
|
771
|
+
}
|
|
772
|
+
resetDeletedFiles() {
|
|
773
|
+
this.deletedFiles = [];
|
|
774
|
+
this.multipleFileUploadModel.removedFiles = [];
|
|
775
|
+
}
|
|
776
|
+
handleExistingFileRemoval(item) {
|
|
777
|
+
const fileName = item.file.rawFile.name;
|
|
778
|
+
const existingFile = this.multipleFileUploadModel.existingFiles
|
|
779
|
+
.find(obj => obj.nameWithExtension === fileName);
|
|
780
|
+
if (existingFile && !this.deletedFiles.some(obj => obj.nameWithExtension === fileName)) {
|
|
781
|
+
this.multipleFileUploadModel.existingFiles =
|
|
782
|
+
this.multipleFileUploadModel.existingFiles.filter(obj => obj.nameWithExtension !== fileName);
|
|
783
|
+
this.deletedFiles.push(existingFile);
|
|
784
|
+
this.multipleFileUploadModel.removedFiles.push(existingFile.iD_GUID);
|
|
785
|
+
}
|
|
786
|
+
}
|
|
787
|
+
removeFromUploadedFiles(item) {
|
|
788
|
+
const itemFileName = item._file.name || item.file.name;
|
|
789
|
+
const itemFileSize = item._file.size || item.file.size;
|
|
790
|
+
const itemGUID = item._file['iD_GUID'];
|
|
791
|
+
this.multipleFileUploadModel.uploadedFiles =
|
|
792
|
+
this.multipleFileUploadModel.uploadedFiles.filter(obj => {
|
|
793
|
+
if (itemGUID && obj.iD_GUID) {
|
|
794
|
+
return obj.iD_GUID !== itemGUID;
|
|
795
|
+
}
|
|
796
|
+
const objFileName = obj.nameWithExtension || obj.fileName;
|
|
797
|
+
const objFileSize = obj.fileSizeInMB ? obj.fileSizeInMB * this.BYTES_TO_MB : 0;
|
|
798
|
+
return !(objFileName === itemFileName && Math.abs(objFileSize - itemFileSize) < 1000);
|
|
799
|
+
});
|
|
800
|
+
}
|
|
801
|
+
validateRemainingFiles() {
|
|
802
|
+
if ((!this.multipleFileUploadModel.uploadedFiles ||
|
|
803
|
+
this.multipleFileUploadModel.uploadedFiles.length === 0) &&
|
|
804
|
+
this.options.isRequired) {
|
|
805
|
+
this.fileUploadFormControl.setErrors({
|
|
806
|
+
MinFileCountValidationKey: this.options.minNoOfFiles
|
|
807
|
+
});
|
|
808
|
+
this.fileUploadFormControl.markAsTouched();
|
|
809
|
+
}
|
|
810
|
+
}
|
|
811
|
+
updateMultipleFileModel() {
|
|
812
|
+
this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;
|
|
813
|
+
const currentErrors = this.fileUploadFormControl.errors;
|
|
814
|
+
this.fileUploadFormControl.setValue(this.multipleFileUploadModel, { emitEvent: false });
|
|
815
|
+
this.group.get(this.options.name)?.setValue(this.multipleFileUploadModel, { emitEvent: false });
|
|
816
|
+
this.options.value = this.multipleFileUploadModel;
|
|
817
|
+
if (currentErrors) {
|
|
818
|
+
this.fileUploadFormControl.setErrors(currentErrors);
|
|
819
|
+
}
|
|
820
|
+
}
|
|
821
|
+
convertSizeToMB(size) {
|
|
822
|
+
if (size === 0) {
|
|
823
|
+
return 0;
|
|
824
|
+
}
|
|
825
|
+
const BYTES_TO_MB_ACCURATE = 1024 * 1024;
|
|
826
|
+
const megabytes = size / BYTES_TO_MB_ACCURATE;
|
|
827
|
+
return Math.round(megabytes * 100) / 100;
|
|
828
|
+
}
|
|
829
|
+
trackByFunction(index, item) {
|
|
830
|
+
return item._file ? item._file.name + item._file.size : index;
|
|
831
|
+
}
|
|
832
|
+
shouldShowFileList() {
|
|
833
|
+
return this.uploader?.queue && this.uploader.queue.length > 0;
|
|
834
|
+
}
|
|
835
|
+
isDownloadEnabled() {
|
|
836
|
+
return true;
|
|
837
|
+
}
|
|
838
|
+
isRemoveEnabled() {
|
|
839
|
+
return !this.options.isReadonly && !this.options.isDisabled;
|
|
840
|
+
}
|
|
841
|
+
getFileDownloadUrl(item) {
|
|
842
|
+
const existingUrl = this.getExistingFileUrl(item);
|
|
843
|
+
if (existingUrl) {
|
|
844
|
+
return existingUrl;
|
|
845
|
+
}
|
|
846
|
+
return this.createFileUrl(item);
|
|
847
|
+
}
|
|
848
|
+
getExistingFileUrl(item) {
|
|
849
|
+
return item?.file?.url ||
|
|
850
|
+
item?._file?.url ||
|
|
851
|
+
item?.url ||
|
|
852
|
+
item?.file?.rawFile?.url ||
|
|
853
|
+
item?._file?.rawFile?.url ||
|
|
854
|
+
null;
|
|
855
|
+
}
|
|
856
|
+
createFileUrl(item) {
|
|
857
|
+
const fileName = this.getFileName(item);
|
|
858
|
+
const fileType = item?.file?.type || item?._file?.type;
|
|
859
|
+
const originalFile = item?._file?.rawFile || item?.file?.rawFile;
|
|
860
|
+
if (originalFile && originalFile instanceof File) {
|
|
861
|
+
return URL.createObjectURL(originalFile);
|
|
862
|
+
}
|
|
863
|
+
const base64Data = typeof originalFile === 'string' ? originalFile : null;
|
|
864
|
+
if (base64Data && fileName) {
|
|
865
|
+
return this.createBlobUrlWithFilename(base64Data, fileType, fileName);
|
|
866
|
+
}
|
|
867
|
+
const fileId = item?._file?.['iD_GUID'];
|
|
868
|
+
if (fileId && this.options.isUploadFileAsync) {
|
|
869
|
+
return this.constructDownloadUrl(fileId, fileName);
|
|
870
|
+
}
|
|
871
|
+
return null;
|
|
872
|
+
}
|
|
873
|
+
createBlobUrlWithFilename(base64Data, fileType, fileName) {
|
|
874
|
+
try {
|
|
875
|
+
const byteCharacters = atob(base64Data);
|
|
876
|
+
const byteNumbers = new Array(byteCharacters.length);
|
|
877
|
+
for (let i = 0; i < byteCharacters.length; i++) {
|
|
878
|
+
byteNumbers[i] = byteCharacters.charCodeAt(i);
|
|
879
|
+
}
|
|
880
|
+
const byteArray = new Uint8Array(byteNumbers);
|
|
881
|
+
const blob = new Blob([byteArray], { type: fileType || 'application/octet-stream' });
|
|
882
|
+
return URL.createObjectURL(blob);
|
|
883
|
+
}
|
|
884
|
+
catch (error) {
|
|
885
|
+
const errorMsg = this.utilityService.getResourceValue('ErrorCreatingBlobUrl');
|
|
886
|
+
console.error(errorMsg, error);
|
|
887
|
+
return `data:${fileType || 'application/octet-stream'};base64,${base64Data}`;
|
|
888
|
+
}
|
|
889
|
+
}
|
|
890
|
+
constructDownloadUrl(fileId, fileName) {
|
|
891
|
+
const downloadBaseUrl = this.options.downloadBaseUrl;
|
|
892
|
+
let url;
|
|
893
|
+
if (downloadBaseUrl) {
|
|
894
|
+
url = `${downloadBaseUrl}/${fileId}`;
|
|
895
|
+
}
|
|
896
|
+
else {
|
|
897
|
+
url = `/api/files/download/${fileId}`;
|
|
898
|
+
}
|
|
899
|
+
if (fileName) {
|
|
900
|
+
const separator = url.includes('?') ? '&' : '?';
|
|
901
|
+
url += `${separator}filename=${encodeURIComponent(fileName)}`;
|
|
902
|
+
}
|
|
903
|
+
return url;
|
|
904
|
+
}
|
|
905
|
+
getFileName(item) {
|
|
906
|
+
return item?.file?.name || item?._file?.name || 'file';
|
|
907
|
+
}
|
|
908
|
+
downloadFile(item) {
|
|
909
|
+
const downloadInfo = this.prepareFileDownload(item);
|
|
910
|
+
if (!downloadInfo.url) {
|
|
911
|
+
this.handleDownloadError(downloadInfo.fileName);
|
|
912
|
+
return;
|
|
913
|
+
}
|
|
914
|
+
this.executeFileDownload(downloadInfo.url, downloadInfo.fileName);
|
|
915
|
+
}
|
|
916
|
+
prepareFileDownload(item) {
|
|
917
|
+
return {
|
|
918
|
+
url: this.getFileDownloadUrl(item),
|
|
919
|
+
fileName: this.getFileName(item)
|
|
920
|
+
};
|
|
921
|
+
}
|
|
922
|
+
handleDownloadError(fileName) {
|
|
923
|
+
const errorMsg = this.utilityService.getResourceValue('NoDownloadUrlAvailable')
|
|
924
|
+
.replace('{fileName}', fileName);
|
|
925
|
+
console.error(errorMsg);
|
|
926
|
+
}
|
|
927
|
+
executeFileDownload(url, fileName) {
|
|
928
|
+
const link = this.createDownloadLink(url, fileName);
|
|
929
|
+
this.triggerDownload(link);
|
|
930
|
+
this.cleanupBlobUrl(url);
|
|
931
|
+
}
|
|
932
|
+
createDownloadLink(url, fileName) {
|
|
933
|
+
const link = document.createElement('a');
|
|
934
|
+
link.href = url;
|
|
935
|
+
link.download = fileName;
|
|
936
|
+
link.style.display = 'none';
|
|
937
|
+
return link;
|
|
938
|
+
}
|
|
939
|
+
triggerDownload(link) {
|
|
940
|
+
document.body.appendChild(link);
|
|
941
|
+
link.click();
|
|
942
|
+
document.body.removeChild(link);
|
|
943
|
+
}
|
|
944
|
+
cleanupBlobUrl(url) {
|
|
945
|
+
if (url.startsWith('blob:')) {
|
|
946
|
+
setTimeout(() => {
|
|
947
|
+
URL.revokeObjectURL(url);
|
|
948
|
+
}, 100);
|
|
949
|
+
}
|
|
950
|
+
}
|
|
951
|
+
trackMemoryUsage(fileSize) {
|
|
952
|
+
this.currentMemoryUsage += fileSize;
|
|
953
|
+
// If memory usage exceeds limit, clean up old files
|
|
954
|
+
if (this.currentMemoryUsage > this.MAX_MEMORY_USAGE) {
|
|
955
|
+
this.cleanupOldFiles();
|
|
956
|
+
}
|
|
957
|
+
}
|
|
958
|
+
cleanupOldFiles() {
|
|
959
|
+
if (this.uploader?.queue && this.uploader.queue.length > 0) {
|
|
960
|
+
// Remove oldest files to free memory
|
|
961
|
+
const oldestFile = this.uploader.queue.shift();
|
|
962
|
+
if (oldestFile) {
|
|
963
|
+
const url = this.getFileDownloadUrl(oldestFile);
|
|
964
|
+
if (url && url.startsWith('blob:')) {
|
|
965
|
+
this.cleanupBlobUrl(url);
|
|
966
|
+
}
|
|
967
|
+
this.currentMemoryUsage -= (oldestFile.file?.size || 0);
|
|
968
|
+
}
|
|
969
|
+
}
|
|
970
|
+
}
|
|
971
|
+
cleanupEventListeners() {
|
|
972
|
+
// Clear file input references
|
|
973
|
+
if (this.fileInput?.nativeElement) {
|
|
974
|
+
this.fileInput.nativeElement.value = '';
|
|
975
|
+
}
|
|
976
|
+
// Clear uploader queue to prevent memory leaks
|
|
977
|
+
if (this.uploader?.queue) {
|
|
978
|
+
this.uploader.queue = [];
|
|
979
|
+
}
|
|
980
|
+
}
|
|
981
|
+
cleanupUploaderQueue() {
|
|
982
|
+
if (this.uploader?.queue) {
|
|
983
|
+
// Clear all items and their associated resources
|
|
984
|
+
this.uploader.queue.forEach(item => {
|
|
985
|
+
// Clean up blob URLs
|
|
986
|
+
const url = this.getFileDownloadUrl(item);
|
|
987
|
+
if (url && url.startsWith('blob:')) {
|
|
988
|
+
this.cleanupBlobUrl(url);
|
|
989
|
+
}
|
|
990
|
+
// Clear file references
|
|
991
|
+
if (item._file) {
|
|
992
|
+
item._file = null;
|
|
993
|
+
}
|
|
994
|
+
if (item.file) {
|
|
995
|
+
item.file = null;
|
|
996
|
+
}
|
|
997
|
+
});
|
|
998
|
+
// Clear the queue
|
|
999
|
+
this.uploader.queue = [];
|
|
1000
|
+
}
|
|
1001
|
+
}
|
|
1002
|
+
ngOnDestroy() {
|
|
1003
|
+
// Clean up subscriptions
|
|
1004
|
+
this.subscriptions.unsubscribe();
|
|
1005
|
+
// Clean up uploader queue
|
|
1006
|
+
this.cleanupUploaderQueue();
|
|
1007
|
+
// Clean up event listeners
|
|
1008
|
+
this.cleanupEventListeners();
|
|
1009
|
+
// Clean up blob URLs
|
|
1010
|
+
if (this.uploader?.queue) {
|
|
1011
|
+
this.uploader.queue.forEach(item => {
|
|
1012
|
+
const url = this.getFileDownloadUrl(item);
|
|
1013
|
+
if (url && url.startsWith('blob:')) {
|
|
1014
|
+
URL.revokeObjectURL(url);
|
|
1015
|
+
}
|
|
1016
|
+
});
|
|
1017
|
+
}
|
|
1018
|
+
}
|
|
1019
|
+
checkAndClearMaxFileCountValidation() {
|
|
1020
|
+
if (!this.options.maxNoOfFiles || this.options.maxNoOfFiles <= 0) {
|
|
1021
|
+
return;
|
|
1022
|
+
}
|
|
1023
|
+
const currentQueueLength = this.uploader.queue.length;
|
|
1024
|
+
if (currentQueueLength <= this.options.maxNoOfFiles) {
|
|
1025
|
+
this.clearFileCountError('MaxFileCountValidationKey');
|
|
1026
|
+
}
|
|
1027
|
+
}
|
|
1028
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FileUploadComponent, deps: [{ token: i1.ControlContainer, optional: true }, { token: i1.FormGroupDirective }, { token: i2.ControlUtility }, { token: i3.UtilityService }, { token: i3.ControlValidationService }, { token: i4.GlobalSettings }, { token: i5.FileUploadService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
1029
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: FileUploadComponent, selector: "BBSF-FileUpload", inputs: { group: "group", options: "options" }, outputs: { OnChange: "OnChange", isUploadComplete: "isUploadComplete" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], ngImport: i0, template: "<div class=\"form-group bbsf-control bbsf-file-upload\" [formGroup]=\"group\">\r\n <div [ngClass]=\"options.viewType === 1 ? 'bbsf-vertical' : 'bbsf-horizontal'\">\r\n <!-- Label -->\r\n <label [hidden]=\"options.hideLabel\" class=\"bbsf-label {{ options.labelExtraClasses }}\">\r\n {{ options.labelValue }}\r\n <!-- Required asterisk -->\r\n <span *ngIf=\"options.isRequired && !options.isReadonly && (options.showAsterisk || true)\"\r\n class=\"text-danger\">*</span>\r\n </label>\r\n <!-- Drop zone enabled -->\r\n <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n *ngIf=\"options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n [ngClass]=\"{ 'another-file-over-class': hasAnotherDropZoneOver }\" (onFileDrop)=\"onFileChange()\"\r\n (fileOver)=\"fileOverAnother($event)\" [uploader]=\"uploader\" [accept]=\"acceptedType\" [id]=\"options.name\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\"\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n (click)=\"fileInputControl.click()\">\r\n\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\r\n <!-- Upload icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n\r\n <!-- Instruction text -->\r\n <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n {{ utilityService.getResourceValue('DragAndDropHere') }}\r\n </div>\r\n\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n </div>\r\n\r\n <div class=\"bbsf-validation-msg text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\">\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Hidden file input -->\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\"\r\n #fileInputControl [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- Click to upload (no drop zone) -->\r\n <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n (click)=\"fileInput.click()\">\r\n\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\r\n <!-- Upload icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n\r\n <!-- Upload text -->\r\n <div class=\"bbsf-validation-msg text-center\">\r\n {{ utilityService.getResourceValue('Upload') }}\r\n </div>\r\n\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n </div>\r\n\r\n <div class=\"bbsf-validation-msg text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\">\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Hidden file input -->\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\" #fileInput\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- Read-only state with no files -->\r\n <div *ngIf=\"options.isReadonly && (!options.value || (uploader.queue && uploader.queue.length === 0))\">\r\n <span class=\"readonly-view\">{{ utilityService.getResourceValue('NA') }}</span>\r\n </div>\r\n </div>\r\n <!-- Uploaded files list -->\r\n <div class=\"uploaded-items\" *ngIf=\"shouldShowFileList()\">\r\n <div class=\"btn-group\" *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n\r\n <!-- Async upload completed files -->\r\n <ng-container *ngIf=\"item?.progress === 100 && options.isUploadFileAsync\">\r\n <!-- Download link - always visible -->\r\n <button *ngIf=\"getFileDownloadUrl(item); else noUrlTemplate\" type=\"button\"\r\n class=\"btn-download-file btn-sm btn-progress-upload\" (click)=\"downloadFile(item)\"\r\n [title]=\"'Download ' + getFileName(item)\">\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </button>\r\n\r\n <!-- File name display when no URL available -->\r\n <ng-template #noUrlTemplate>\r\n <span class=\"btn-download-file btn-sm btn-progress-upload\" [title]=\"getFileName(item)\">\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </span>\r\n </ng-template>\r\n\r\n <!-- Remove button - only show when not readonly and not disabled -->\r\n <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm\" type=\"button\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n [title]=\"'Remove ' + getFileName(item)\">\r\n <!-- Delete icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M9.33301 3.70584V3.26663C9.33301 2.65166 9.33301 2.34419 9.20587 2.1093C9.09405 1.9027 8.91555 1.73471 8.69604 1.62944C8.44647 1.50977 8.11977 1.50977 7.46638 1.50977H6.53305C5.87965 1.50977 5.55296 1.50977 5.30339 1.62944C5.08387 1.73471 4.90539 1.9027 4.79354 2.1093C4.66638 2.34419 4.66638 2.65166 4.66638 3.26663V3.70584\"\r\n stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path\r\n d=\"M1.75 3.70605H12.25M11.0834 3.70605V9.8551C11.0834 10.7775 11.0834 11.2387 10.8926 11.591C10.7248 11.901 10.4571 12.1529 10.1278 12.3109C9.75345 12.4904 9.26345 12.4904 8.28334 12.4904H5.71666C4.73658 12.4904 4.24653 12.4904 3.87218 12.3109C3.5429 12.1529 3.27519 11.901 3.10741 11.591C2.91666 11.2387 2.91666 10.7775 2.91666 9.8551V3.70605\"\r\n stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n </button>\r\n </ng-container>\r\n\r\n <!-- Sync upload files -->\r\n <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n <!-- Download link - always visible -->\r\n <button type=\"button\" class=\"btn btn-download-file btn-sm\" (click)=\"downloadFile(item)\"\r\n [title]=\"'Download ' + getFileName(item)\">\r\n <!-- Download icon -->\r\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path\r\n d=\"M21 22H3C2.4 22 2 21.6 2 21C2 20.4 2.4 20 3 20H21C21.6 20 22 20.4 22 21C22 21.6 21.6 22 21 22ZM13 13.4V3C13 2.4 12.6 2 12 2C11.4 2 11 2.4 11 3V13.4H13Z\"\r\n fill=\"currentColor\"></path>\r\n <path opacity=\"0.3\" d=\"M7 13.4H17L12.7 17.7C12.3 18.1 11.7 18.1 11.3 17.7L7 13.4Z\" fill=\"currentColor\">\r\n </path>\r\n </svg>\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </button>\r\n\r\n <!-- Remove button - only show when not readonly and not disabled -->\r\n <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm btn-danger\" type=\"button\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n [title]=\"'Remove ' + getFileName(item)\">\r\n <i class=\"fa fa-times px-0\" aria-hidden=\"true\"></i>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </div>\r\n <!-- File upload progress indicators -->\r\n <div *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile }\"\r\n *ngIf=\"item?.progress < 100 && options.isUploadFileAsync && !options.isReadonly\">\r\n\r\n <div class=\"upload-items-toolbar\">\r\n <h4>{{ getFileName(item) }}</h4>\r\n <button *ngIf=\"isRemoveEnabled()\" type=\"button\" class=\"btn-cancel-upload\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\"\r\n [attr.aria-label]=\"'Cancel upload for ' + getFileName(item)\">\r\n <!-- Cancel icon -->\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_1324_13216)\">\r\n <path opacity=\"0.4\"\r\n d=\"M9 16.5C13.1421 16.5 16.5 13.1421 16.5 9C16.5 4.85786 13.1421 1.5 9 1.5C4.85786 1.5 1.5 4.85786 1.5 9C1.5 13.1421 4.85786 16.5 9 16.5Z\"\r\n stroke=\"#DBE1F0\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M11.25 6.75L6.75 11.25M6.75 6.75L11.25 11.25\" stroke=\"#DBE1F0\" stroke-width=\"2\"\r\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_1324_13216\">\r\n <rect width=\"18\" height=\"18\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </button>\r\n </div>\r\n\r\n <div class=\"progress\">\r\n <div class=\"progress-bar\" role=\"progressbar\" [attr.aria-valuenow]=\"item?.progress\" aria-valuemin=\"0\"\r\n aria-valuemax=\"100\" [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\"\r\n *ngIf=\"item?.progress > 0\">\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Validation and description section -->\r\n <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n </div>\r\n\r\n <!-- Control description -->\r\n <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription\">\r\n {{ options.labelDescription }}\r\n </div>\r\n\r\n <!-- Reset error state -->\r\n <div *ngIf=\"(group.valid && group.dirty && group.touched) || (group.untouched && group.invalid && group.dirty)\">\r\n {{ resetError() }}\r\n </div>\r\n </div>\r\n</div>", dependencies: [{ kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i7.FileDropDirective, selector: "[ng2FileDrop]", inputs: ["uploader"], outputs: ["fileOver", "onFileDrop"] }, { kind: "directive", type: i7.FileSelectDirective, selector: "[ng2FileSelect]", inputs: ["uploader"], outputs: ["onFileSelected"] }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "pipe", type: i6.KeyValuePipe, name: "keyvalue" }] }); }
|
|
1030
|
+
}
|
|
1031
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FileUploadComponent, decorators: [{
|
|
1032
|
+
type: Component,
|
|
1033
|
+
args: [{ selector: 'BBSF-FileUpload', template: "<div class=\"form-group bbsf-control bbsf-file-upload\" [formGroup]=\"group\">\r\n <div [ngClass]=\"options.viewType === 1 ? 'bbsf-vertical' : 'bbsf-horizontal'\">\r\n <!-- Label -->\r\n <label [hidden]=\"options.hideLabel\" class=\"bbsf-label {{ options.labelExtraClasses }}\">\r\n {{ options.labelValue }}\r\n <!-- Required asterisk -->\r\n <span *ngIf=\"options.isRequired && !options.isReadonly && (options.showAsterisk || true)\"\r\n class=\"text-danger\">*</span>\r\n </label>\r\n <!-- Drop zone enabled -->\r\n <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n *ngIf=\"options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n [ngClass]=\"{ 'another-file-over-class': hasAnotherDropZoneOver }\" (onFileDrop)=\"onFileChange()\"\r\n (fileOver)=\"fileOverAnother($event)\" [uploader]=\"uploader\" [accept]=\"acceptedType\" [id]=\"options.name\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\"\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n (click)=\"fileInputControl.click()\">\r\n\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\r\n <!-- Upload icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n\r\n <!-- Instruction text -->\r\n <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n {{ utilityService.getResourceValue('DragAndDropHere') }}\r\n </div>\r\n\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n </div>\r\n\r\n <div class=\"bbsf-validation-msg text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\">\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Hidden file input -->\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\"\r\n #fileInputControl [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- Click to upload (no drop zone) -->\r\n <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n (click)=\"fileInput.click()\">\r\n\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\r\n <!-- Upload icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n\r\n <!-- Upload text -->\r\n <div class=\"bbsf-validation-msg text-center\">\r\n {{ utilityService.getResourceValue('Upload') }}\r\n </div>\r\n\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n </div>\r\n\r\n <div class=\"bbsf-validation-msg text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\">\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Hidden file input -->\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\" #fileInput\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- Read-only state with no files -->\r\n <div *ngIf=\"options.isReadonly && (!options.value || (uploader.queue && uploader.queue.length === 0))\">\r\n <span class=\"readonly-view\">{{ utilityService.getResourceValue('NA') }}</span>\r\n </div>\r\n </div>\r\n <!-- Uploaded files list -->\r\n <div class=\"uploaded-items\" *ngIf=\"shouldShowFileList()\">\r\n <div class=\"btn-group\" *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n\r\n <!-- Async upload completed files -->\r\n <ng-container *ngIf=\"item?.progress === 100 && options.isUploadFileAsync\">\r\n <!-- Download link - always visible -->\r\n <button *ngIf=\"getFileDownloadUrl(item); else noUrlTemplate\" type=\"button\"\r\n class=\"btn-download-file btn-sm btn-progress-upload\" (click)=\"downloadFile(item)\"\r\n [title]=\"'Download ' + getFileName(item)\">\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </button>\r\n\r\n <!-- File name display when no URL available -->\r\n <ng-template #noUrlTemplate>\r\n <span class=\"btn-download-file btn-sm btn-progress-upload\" [title]=\"getFileName(item)\">\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </span>\r\n </ng-template>\r\n\r\n <!-- Remove button - only show when not readonly and not disabled -->\r\n <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm\" type=\"button\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n [title]=\"'Remove ' + getFileName(item)\">\r\n <!-- Delete icon -->\r\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\">\r\n <path opacity=\"0.4\"\r\n d=\"M9.33301 3.70584V3.26663C9.33301 2.65166 9.33301 2.34419 9.20587 2.1093C9.09405 1.9027 8.91555 1.73471 8.69604 1.62944C8.44647 1.50977 8.11977 1.50977 7.46638 1.50977H6.53305C5.87965 1.50977 5.55296 1.50977 5.30339 1.62944C5.08387 1.73471 4.90539 1.9027 4.79354 2.1093C4.66638 2.34419 4.66638 2.65166 4.66638 3.26663V3.70584\"\r\n stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path\r\n d=\"M1.75 3.70605H12.25M11.0834 3.70605V9.8551C11.0834 10.7775 11.0834 11.2387 10.8926 11.591C10.7248 11.901 10.4571 12.1529 10.1278 12.3109C9.75345 12.4904 9.26345 12.4904 8.28334 12.4904H5.71666C4.73658 12.4904 4.24653 12.4904 3.87218 12.3109C3.5429 12.1529 3.27519 11.901 3.10741 11.591C2.91666 11.2387 2.91666 10.7775 2.91666 9.8551V3.70605\"\r\n stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </svg>\r\n </button>\r\n </ng-container>\r\n\r\n <!-- Sync upload files -->\r\n <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n <!-- Download link - always visible -->\r\n <button type=\"button\" class=\"btn btn-download-file btn-sm\" (click)=\"downloadFile(item)\"\r\n [title]=\"'Download ' + getFileName(item)\">\r\n <!-- Download icon -->\r\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path\r\n d=\"M21 22H3C2.4 22 2 21.6 2 21C2 20.4 2.4 20 3 20H21C21.6 20 22 20.4 22 21C22 21.6 21.6 22 21 22ZM13 13.4V3C13 2.4 12.6 2 12 2C11.4 2 11 2.4 11 3V13.4H13Z\"\r\n fill=\"currentColor\"></path>\r\n <path opacity=\"0.3\" d=\"M7 13.4H17L12.7 17.7C12.3 18.1 11.7 18.1 11.3 17.7L7 13.4Z\" fill=\"currentColor\">\r\n </path>\r\n </svg>\r\n <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n </button>\r\n\r\n <!-- Remove button - only show when not readonly and not disabled -->\r\n <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm btn-danger\" type=\"button\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n [title]=\"'Remove ' + getFileName(item)\">\r\n <i class=\"fa fa-times px-0\" aria-hidden=\"true\"></i>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </div>\r\n <!-- File upload progress indicators -->\r\n <div *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile }\"\r\n *ngIf=\"item?.progress < 100 && options.isUploadFileAsync && !options.isReadonly\">\r\n\r\n <div class=\"upload-items-toolbar\">\r\n <h4>{{ getFileName(item) }}</h4>\r\n <button *ngIf=\"isRemoveEnabled()\" type=\"button\" class=\"btn-cancel-upload\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\"\r\n [attr.aria-label]=\"'Cancel upload for ' + getFileName(item)\">\r\n <!-- Cancel icon -->\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_1324_13216)\">\r\n <path opacity=\"0.4\"\r\n d=\"M9 16.5C13.1421 16.5 16.5 13.1421 16.5 9C16.5 4.85786 13.1421 1.5 9 1.5C4.85786 1.5 1.5 4.85786 1.5 9C1.5 13.1421 4.85786 16.5 9 16.5Z\"\r\n stroke=\"#DBE1F0\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n <path d=\"M11.25 6.75L6.75 11.25M6.75 6.75L11.25 11.25\" stroke=\"#DBE1F0\" stroke-width=\"2\"\r\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n </g>\r\n <defs>\r\n <clipPath id=\"clip0_1324_13216\">\r\n <rect width=\"18\" height=\"18\" fill=\"white\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n </button>\r\n </div>\r\n\r\n <div class=\"progress\">\r\n <div class=\"progress-bar\" role=\"progressbar\" [attr.aria-valuenow]=\"item?.progress\" aria-valuemin=\"0\"\r\n aria-valuemax=\"100\" [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\"\r\n *ngIf=\"item?.progress > 0\">\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Validation and description section -->\r\n <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n <!-- Validation messages -->\r\n <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n </div>\r\n\r\n <!-- Control description -->\r\n <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription\">\r\n {{ options.labelDescription }}\r\n </div>\r\n\r\n <!-- Reset error state -->\r\n <div *ngIf=\"(group.valid && group.dirty && group.touched) || (group.untouched && group.invalid && group.dirty)\">\r\n {{ resetError() }}\r\n </div>\r\n </div>\r\n</div>" }]
|
|
1034
|
+
}], ctorParameters: () => [{ type: i1.ControlContainer, decorators: [{
|
|
1035
|
+
type: Optional
|
|
1036
|
+
}] }, { type: i1.FormGroupDirective }, { type: i2.ControlUtility }, { type: i3.UtilityService }, { type: i3.ControlValidationService }, { type: i4.GlobalSettings }, { type: i5.FileUploadService }], propDecorators: { fileInput: [{
|
|
1037
|
+
type: ViewChild,
|
|
1038
|
+
args: ['fileInput', { static: false }]
|
|
1039
|
+
}], group: [{
|
|
1040
|
+
type: Input
|
|
1041
|
+
}], options: [{
|
|
1042
|
+
type: Input
|
|
1043
|
+
}], OnChange: [{
|
|
1044
|
+
type: Output
|
|
1045
|
+
}], isUploadComplete: [{
|
|
1046
|
+
type: Output
|
|
1047
|
+
}] } });
|
|
1048
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FileUpload.component.js","sourceRoot":"","sources":["../../../../../../projects/bbsf-controls/src/lib/controls/FileUpload/FileUpload.component.ts","../../../../../../projects/bbsf-controls/src/lib/controls/FileUpload/FileUpload.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  OnInit,\r\n  AfterViewInit,\r\n  OnDestroy,\r\n  Input,\r\n  Optional,\r\n  ViewChild,\r\n  ElementRef,\r\n  Output,\r\n  EventEmitter\r\n} from '@angular/core';\r\nimport {\r\n  FormControl,\r\n  Validators,\r\n  FormGroup,\r\n  AbstractControl,\r\n  ControlContainer,\r\n  FormGroupDirective,\r\n  ValidatorFn\r\n} from '@angular/forms';\r\nimport { FileUploadOptions } from '../../Shared/Models/FileUploadOptions';\r\nimport { ControlUtility } from '../../Shared/services/ControlUtility';\r\nimport { FileUploader, FileLikeObject, FileUploaderOptions, FileItem } from 'ng2-file-upload';\r\nimport { FileUploadModel } from '../../Shared/Models/FileUploadModel';\r\nimport { MultipleFileUploadModel } from '../../Shared/Models/MultipleFileUploadModel';\r\nimport { ControlValidationService, UtilityService } from '@bnsights/bbsf-utilities';\r\nimport { GlobalSettings } from '../../Shared/services/GlobalSettings.service';\r\nimport { FileDTO } from '../../Shared/Models/FileDTO';\r\nimport { FileUploadService } from '../../Shared/services/file-upload.service';\r\nimport { HttpEventType } from '@angular/common/http';\r\nimport { Subscription } from 'rxjs';\r\ninterface FileItemExtended extends FileItem {\r\n  _file: FileExtended;\r\n}\r\n\r\ninterface FileExtended extends File {\r\n  iD_GUID?: string;\r\n  isNew?: boolean;\r\n  url?: string;\r\n  rawFile?: File;\r\n}\r\n\r\ninterface FileValidationResult {\r\n  isValid: boolean;\r\n  errors: string[];\r\n  invalidFiles: FileItem[];\r\n}\r\n\r\ninterface FileProcessingResult {\r\n  validFiles: FileItem[];\r\n  invalidFiles: FileItem[];\r\n  errors: string[];\r\n}\r\n\r\ninterface DownloadInfo {\r\n  url: string | null;\r\n  fileName: string;\r\n}\r\n\r\ninterface UploadProgressEvent {\r\n  loaded: number;\r\n  total: number;\r\n  type: number;\r\n}\r\n\r\ninterface UploadCompleteEvent {\r\n  body: {\r\n    val: string;\r\n    downloadUrl?: string;\r\n  };\r\n  type: number;\r\n}\r\n\r\n@Component({\r\n  selector: 'BBSF-FileUpload',\r\n  templateUrl: './FileUpload.component.html'\r\n})\r\nexport class FileUploadComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  private static controlContainerStatic: ControlContainer | null = null;\r\n\r\n  private readonly BYTES_TO_MB = 1024 * 1024;\r\n  private readonly PROGRESS_COMPLETE = 100;\r\n  private readonly PROGRESS_NEAR_COMPLETE = 95;\r\n  private readonly ERROR_DISPLAY_DURATION = 5000;\r\n  private readonly MAX_MEMORY_USAGE = 100 * 1024 * 1024; // 100MB limit\r\n  private currentMemoryUsage = 0;\r\n\r\n  @ViewChild('fileInput', { static: false }) fileInput!: ElementRef<HTMLInputElement>;\r\n\r\n  isSubmitted = false;\r\n  @Input() group!: FormGroup;\r\n  @Input() options!: FileUploadOptions;\r\n  @Output() OnChange = new EventEmitter<any>();\r\n  @Output() isUploadComplete = new EventEmitter<boolean>();\r\n\r\n  fileUploadFormControl!: AbstractControl;\r\n  validationMessage = '';\r\n  validationCountMessage = '';\r\n  uploader!: FileUploader;\r\n  hasAnotherDropZoneOver = false;\r\n  acceptedType = '';\r\n  acceptedTypeArray: string[] = [];\r\n  toolTipTypeArray: string[] = [];\r\n  fileLikeObject!: FileLikeObject;\r\n  markAllAsTouched = false;\r\n  validationRules: ValidatorFn[] = [];\r\n  validationRulesAsync: ValidatorFn[] = [];\r\n  fileUploadModel!: FileUploadModel;\r\n  multipleFileUploadModel!: MultipleFileUploadModel;\r\n  file: FileDTO | null = null;\r\n  deletedFiles: FileDTO[] = [];\r\n  private subscriptions = new Subscription();\r\n\r\n  constructor(\r\n    @Optional() private controlContainer: ControlContainer,\r\n    public multipleFileUploadControlHost: FormGroupDirective,\r\n    private controlUtility: ControlUtility,\r\n    public utilityService: UtilityService,\r\n    private controlValidationService: ControlValidationService,\r\n    private globalSettings: GlobalSettings,\r\n    private fileUploadService: FileUploadService\r\n  ) {\r\n    FileUploadComponent.controlContainerStatic = this.controlContainer;\r\n    this.initializeUploader();\r\n  }\r\n\r\n  private initializeUploader(): void {\r\n    this.uploader = new FileUploader({\r\n      disableMultipart: false\r\n    } as FileUploaderOptions);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.initializeModels();\r\n    this.setViewType();\r\n    this.processInitialValue();\r\n    this.setupLabels();\r\n    this.setupFileTypeValidation();\r\n    this.setupFormControl();\r\n    this.setupSubscriptions();\r\n  }\r\n\r\n  private initializeModels(): void {\r\n    this.fileUploadModel = new FileUploadModel();\r\n    this.multipleFileUploadModel = new MultipleFileUploadModel();\r\n  }\r\n\r\n  private setViewType(): void {\r\n    if (!this.options.viewType) {\r\n      this.options.viewType = this.globalSettings.viewType;\r\n    }\r\n  }\r\n\r\n  private processInitialValue(): void {\r\n\r\n    if (this.options.value == null) {\r\n      return;\r\n    }\r\n\r\n    if (this.options.isMultipleFile) {\r\n      this.processMultipleFileValue();\r\n    } else {\r\n      this.processSingleFileValue();\r\n    }\r\n\r\n    this.uploader.queue.forEach((element) => {\r\n      element.progress = this.PROGRESS_COMPLETE;\r\n    });\r\n  }\r\n\r\n  private processMultipleFileValue(): void {\r\n    const files: FileLikeObject[] = [];\r\n    this.multipleFileUploadModel.existingFiles = this.options.value.existingFiles;\r\n    this.multipleFileUploadModel.uploadedFiles = [];\r\n\r\n    for (const element of this.options.value.existingFiles) {\r\n      const fileLikeObject = this.createFileLikeObject(element);\r\n      files.push(fileLikeObject);\r\n    }\r\n\r\n    this.uploader.addToQueue(files as any);\r\n  }\r\n\r\n  private processSingleFileValue(): void {\r\n    const element = this.options.value.file ?? this.options.value;\r\n    const fileLikeObject = this.createFileLikeObject(element);\r\n    this.file = element;\r\n    this.uploader.addToQueue([fileLikeObject] as any);\r\n\r\n    if (!this.options.value.file) {\r\n      this.fileUploadModel = new FileUploadModel();\r\n      this.fileUploadModel.file = this.options.value;\r\n    } else {\r\n      this.fileUploadModel = this.options.value;\r\n    }\r\n    this.options.value = this.fileUploadModel;\r\n  }\r\n\r\n  private createFileLikeObject(element: FileDTO): any {\r\n    const bytes = new Uint8Array(element.bytes);\r\n    const base64 = btoa(String.fromCharCode(...bytes));\r\n    return {\r\n      name: element.nameWithExtension || element.fileName,\r\n      type: element.mimeType || element.fileType,\r\n      rawFile: base64,\r\n      size: element.fileSizeInMB ? element.fileSizeInMB * this.BYTES_TO_MB : 0,\r\n      lastModifiedDate: new Date(),\r\n      url: element.fullFileURL\r\n    };\r\n  }\r\n\r\n  private setupLabels(): void {\r\n\r\n    if (this.options.labelKey) {\r\n      this.options.labelValue = this.utilityService.getResourceValue(this.options.labelKey);\r\n    }\r\n  }\r\n\r\n  private setupFileTypeValidation(): void {\r\n\r\n    if (this.options.fileUploadAcceptsTypes?.length) {\r\n      this.processAcceptedTypes();\r\n      this.buildValidationMessage();\r\n    }\r\n  }\r\n\r\n  private processAcceptedTypes(): void {\r\n    this.acceptedType = this.options.fileUploadAcceptsTypes.join(',');\r\n    this.acceptedTypeArray = this.options.fileUploadAcceptsTypes.filter(type => type.trim());\r\n\r\n    const mimeTypeMap = this.getMimeTypeMap();\r\n\r\n    for (const type of this.acceptedTypeArray) {\r\n      const displayType = mimeTypeMap[type];\r\n      if (displayType && !this.toolTipTypeArray.includes(displayType)) {\r\n        this.toolTipTypeArray.push(displayType);\r\n      }\r\n    }\r\n  }\r\n\r\n  private getMimeTypeMap(): Record<string, string> {\r\n    return {\r\n      'application/pdf': 'PDF',\r\n      'application/msword': 'Word',\r\n      'application/vnd.openxmlformats-officedocument.wordprocessingml.document': 'Word',\r\n      'application/vnd.ms-excel': 'Excel',\r\n      'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet': 'Excel',\r\n      'application/vnd.ms-powerpoint': 'PowerPoint',\r\n      'application/vnd.openxmlformats-officedocument.presentationml.presentation': 'PowerPoint',\r\n      'image/png': 'PNG',\r\n      'image/bmp': 'BMP',\r\n      'image/jpeg': 'JPEG',\r\n      'application/zip': 'ZIP',\r\n      'application/x-rar-compressed': 'RAR',\r\n      'video/mp4': 'MP4',\r\n      'video/avi': 'AVI',\r\n      'video/quicktime': 'MOV',\r\n      'video/mpeg': 'MPEG',\r\n      'audio/mpeg': 'MP3',\r\n      'video/x-flv': 'FLV',\r\n      'video/x-ms-wmv': 'WMV',\r\n      'image/svg+xml': 'SVG',\r\n      'text/plain': 'Txt',\r\n      'application/BN': 'License'\r\n    };\r\n  }\r\n\r\n  private buildValidationMessage(): void {\r\n    const messages: string[] = [];\r\n\r\n    if (this.toolTipTypeArray.length) {\r\n      messages.push(`${this.utilityService.getResourceValue('Extensions')} (${this.toolTipTypeArray.join(', ')})`);\r\n    }\r\n    if (this.options.fileMaxSizeInMB > 0) {\r\n      messages.push(`${this.utilityService.getResourceValue('FileMaxSizeInMB')}${this.options.fileMaxSizeInMB}`);\r\n    }\r\n\r\n    if (this.options.minNoOfFiles > 0) {\r\n      messages.push(`${this.utilityService.getResourceValue('MinFileCountValidationKey')}${this.options.minNoOfFiles}`);\r\n    }\r\n\r\n    if (this.options.maxNoOfFiles > 0) {\r\n      messages.push(`${this.utilityService.getResourceValue('MaxFileCountValidationKey')}${this.options.maxNoOfFiles}`);\r\n    }\r\n\r\n    this.validationMessage = messages.join('<br/>');\r\n  }\r\n\r\n  private setupFormControl(): void {\r\n\r\n\r\n    this.group.addControl(this.options.name, new FormControl(''));\r\n    this.fileUploadFormControl = this.group.controls[this.options.name];\r\n\r\n    this.setupValidators();\r\n    this.setupCountMessage();\r\n    this.applyValidatorsAndState();\r\n    this.fileUploadFormControl.setValue(this.options.value, { emitEvent: false });\r\n  }\r\n\r\n  private setupValidators(): void {\r\n    if (this.options.customValidation?.length) {\r\n      for (const validation of this.options.customValidation) {\r\n        this.validationRules.push(validation.functionBody);\r\n      }\r\n    }\r\n\r\n    if (this.options.isRequired) {\r\n      this.validationRules.push(Validators.required);\r\n    }\r\n  }\r\n\r\n  private setupCountMessage(): void {\r\n    if (this.options.isMultipleFile && this.options.maxNoOfFiles > 0) {\r\n      this.validationCountMessage = `${this.utilityService.getResourceValue('MaxFilesCount')} : ${this.options.maxNoOfFiles}`;\r\n    }\r\n  }\r\n\r\n  private applyValidatorsAndState(): void {\r\n    this.fileUploadFormControl.setValidators(this.validationRules);\r\n    if (this.validationRulesAsync.length > 0) {\r\n      this.fileUploadFormControl.setAsyncValidators(this.validationRulesAsync as any);\r\n    }\r\n\r\n    if (this.options.isDisabled) {\r\n      this.fileUploadFormControl.disable();\r\n    }\r\n  }\r\n\r\n  private setupSubscriptions(): void {\r\n    this.multipleFileUploadControlHost.ngSubmit.subscribe(() => {\r\n      this.group.markAllAsTouched();\r\n      this.markAllAsTouched = true;\r\n    });\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.applyAttributes();\r\n  }\r\n\r\n  private applyAttributes(): void {\r\n    if (!this.options.attributeList?.length) {\r\n      return;\r\n    }\r\n\r\n    const element = document.getElementById(this.options.name);\r\n    if (element) {\r\n      for (const attribute of this.options.attributeList) {\r\n        element.setAttribute(attribute.key, attribute.value);\r\n      }\r\n    }\r\n  }\r\n\r\n  resetError = (): void => {\r\n    this.controlValidationService.removeGlobalError();\r\n  };\r\n\r\n  showGlobalError(): void {\r\n    this.controlUtility.showGlobalError();\r\n  }\r\n\r\n  getErrorValidation(errorList: any): string {\r\n    if (this.markAllAsTouched && this.group.invalid) {\r\n      this.showGlobalError();\r\n      this.markAllAsTouched = false;\r\n    }\r\n\r\n    if (errorList && errorList.length > 0) {\r\n      for (const error of errorList) {\r\n        if (error.key === 'InvalidFiles') {\r\n          return error.value;\r\n        }\r\n      }\r\n    }\r\n\r\n    return this.controlUtility.getErrorValidationMassage(errorList, this.group, this.options);\r\n  }\r\n\r\n  fileOverAnother(event: any): void {\r\n    this.hasAnotherDropZoneOver = !!event;\r\n  }\r\n\r\n  isHideInput(): boolean {\r\n    if (this.options.isMultipleFile) {\r\n      return this.options.maxNoOfFiles > 0 &&\r\n        this.options.maxNoOfFiles === this.uploader.queue.length;\r\n    }\r\n    return this.uploader.queue.length > 0;\r\n  }\r\n  onFileChange(): void {\r\n    this.validateFileConstraints();\r\n\r\n    const fileProcessingResult = this.processNewlyAddedFiles();\r\n\r\n    this.handleFileValidationResults(fileProcessingResult);\r\n\r\n    this.processValidFilesForUpload();\r\n  }\r\n\r\n  private processNewlyAddedFiles(): FileProcessingResult {\r\n    const addedQueue = this.getNewlyAddedFiles();\r\n    const validationResult = this.validateFilesInQueue(addedQueue);\r\n\r\n    return {\r\n      validFiles: addedQueue.filter(file => !validationResult.invalidFiles.includes(file)),\r\n      invalidFiles: validationResult.invalidFiles,\r\n      errors: validationResult.errors\r\n    };\r\n  }\r\n\r\n  private getNewlyAddedFiles(): FileItem[] {\r\n    return this.uploader.queue.filter((obj) => obj['some']?.lastModified != null);\r\n  }\r\n\r\n  private validateFilesInQueue(fileQueue: FileItem[]): FileValidationResult {\r\n    const validationErrors: string[] = [];\r\n    const invalidFiles: FileItem[] = [];\r\n    const processedDuplicateNames = new Set<string>();\r\n\r\n    for (const element of fileQueue) {\r\n      const file = element.file;\r\n      if (!file) continue;\r\n\r\n      const fileValidation = this.validateSingleFile(file, element, processedDuplicateNames);\r\n\r\n      if (!fileValidation.isValid) {\r\n        invalidFiles.push(element);\r\n        validationErrors.push(...fileValidation.errors);\r\n      }\r\n    }\r\n\r\n    return {\r\n      isValid: invalidFiles.length === 0,\r\n      errors: validationErrors,\r\n      invalidFiles\r\n    };\r\n  }\r\n\r\n  private validateSingleFile(file: FileLikeObject, element: FileItem, processedDuplicateNames: Set<string>): FileValidationResult {\r\n    const errors: string[] = [];\r\n\r\n    const sizeValid = this.validateIndividualFileSize(file);\r\n    const typeValid = this.validateIndividualFileType(file);\r\n    const nameValid = this.validateDuplicateFileName(file, element);\r\n\r\n    if (!sizeValid) {\r\n      errors.push(this.createFileSizeErrorMessage(file.name));\r\n    }\r\n\r\n    if (!typeValid) {\r\n      errors.push(this.createFileTypeErrorMessage(file.name));\r\n    }\r\n\r\n    if (!nameValid) {\r\n      const duplicateError = this.createDuplicateFileErrorMessage(file.name, processedDuplicateNames);\r\n      if (duplicateError) {\r\n        errors.push(duplicateError);\r\n      }\r\n    }\r\n\r\n    return {\r\n      isValid: sizeValid && typeValid && nameValid,\r\n      errors,\r\n      invalidFiles: []\r\n    };\r\n  }\r\n\r\n  private createFileSizeErrorMessage(fileName: string): string {\r\n    return this.utilityService.getResourceValue('FileExceedsMaxSize')\r\n      .replace('{fileName}', fileName)\r\n      .replace('{maxSize}', this.options.fileMaxSizeInMB.toString());\r\n  }\r\n\r\n  private createFileTypeErrorMessage(fileName: string): string {\r\n    return this.utilityService.getResourceValue('FileTypeNotAccepted')\r\n      .replace('{fileName}', fileName);\r\n  }\r\n\r\n  private createDuplicateFileErrorMessage(fileName: string, processedDuplicateNames: Set<string>): string | null {\r\n    const fileNameLower = fileName.toLowerCase();\r\n\r\n    if (processedDuplicateNames.has(fileNameLower)) {\r\n      return null;\r\n    }\r\n\r\n    processedDuplicateNames.add(fileNameLower);\r\n\r\n    let duplicateErrorMsg = this.utilityService.getResourceValue('DuplicateFileName');\r\n\r\n    if (!duplicateErrorMsg || duplicateErrorMsg === 'DuplicateFileName') {\r\n      duplicateErrorMsg = `File '{fileName}' already exists. Please choose a different file or rename it.`;\r\n    }\r\n\r\n    return duplicateErrorMsg.replace('{fileName}', fileName);\r\n  }\r\n\r\n  private handleFileValidationResults(result: FileProcessingResult): void {\r\n    if (result.invalidFiles.length > 0) {\r\n      this.removeInvalidFiles(result.invalidFiles);\r\n    }\r\n\r\n    if (result.errors.length > 0) {\r\n      this.showValidationErrors(result.errors);\r\n    }\r\n  }\r\n\r\n  private processValidFilesForUpload(): void {\r\n    const filesArray: FileDTO[] = [];\r\n    const validQueue = this.getNewlyAddedFiles();\r\n\r\n    for (const element of validQueue) {\r\n      const file = element.file;\r\n      if (!file) continue;\r\n\r\n      if (this.shouldUseAsyncUpload(element)) {\r\n        this.handleAsyncFileUpload(element, filesArray);\r\n      } else {\r\n        this.handleSyncFileUpload(file, filesArray);\r\n      }\r\n    }\r\n  }\r\n\r\n  private shouldUseAsyncUpload(element: FileItem): boolean {\r\n    return this.options.isUploadFileAsync && !element._file['iD_GUID'];\r\n  }\r\n\r\n  private validateFileConstraints(): boolean {\r\n    if (this.options.isMultipleFile) {\r\n      if (!this.validateMinFileCount() || !this.validateMaxFileCount() || !this.validateTotalFileSize()) {\r\n        return false;\r\n      }\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private validateMinFileCount(): boolean {\r\n    if (this.options.minNoOfFiles > 0 && this.options.minNoOfFiles > this.uploader.queue.length) {\r\n      const minFileMsg = this.utilityService.getResourceValue('MinimumFilesRequired')\r\n        .replace('{count}', this.options.minNoOfFiles.toString());\r\n      this.showFileCountError('MinFileCountValidationKey', minFileMsg);\r\n      return false;\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private validateMaxFileCount(): boolean {\r\n    if (this.options.maxNoOfFiles > 0 && this.options.maxNoOfFiles < this.uploader.queue.length) {\r\n      const maxFileMsg = this.utilityService.getResourceValue('MaximumFilesExceeded') ||\r\n        `Maximum {maxCount} files allowed. You have selected {currentCount} files.`;\r\n      const finalMsg = maxFileMsg\r\n        .replace('{maxCount}', this.options.maxNoOfFiles.toString())\r\n        .replace('{currentCount}', this.uploader.queue.length.toString());\r\n      this.showFileCountError('MaxFileCountValidationKey', finalMsg);\r\n      return false;\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private showFileCountError(errorKey: string, message: string): void {\r\n    const currentErrors = this.fileUploadFormControl.errors || {};\r\n    currentErrors[errorKey] = message;\r\n\r\n    this.fileUploadFormControl.setErrors(currentErrors);\r\n    this.fileUploadFormControl.markAsTouched();\r\n  }\r\n\r\n  private clearFileCountError(errorKey: string): void {\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n    if (currentErrors && currentErrors[errorKey]) {\r\n      delete currentErrors[errorKey];\r\n\r\n      if (Object.keys(currentErrors).length === 0) {\r\n        this.fileUploadFormControl.setErrors(null);\r\n      } else {\r\n        this.fileUploadFormControl.setErrors(currentErrors);\r\n      }\r\n    }\r\n  }\r\n\r\n  private validateTotalFileSize(): boolean {\r\n    if (this.options.maxSizeForAllFilesInMB > 0) {\r\n      const totalSize = this.uploader.queue.reduce((sum, element) => sum + element.file.size, 0);\r\n      const maxSizeBytes = this.options.maxSizeForAllFilesInMB * this.BYTES_TO_MB;\r\n\r\n      if (totalSize > maxSizeBytes) {\r\n        this.showTotalSizeError();\r\n        return false;\r\n      }\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private showTotalSizeError(): void {\r\n    const totalSizeMsg = this.utilityService.getResourceValue('TotalFileSizeExceeded')\r\n      .replace('{maxSize}', this.options.maxSizeForAllFilesInMB.toString());\r\n\r\n    const currentErrors = this.fileUploadFormControl.errors || {};\r\n    currentErrors['MaxSizeForAllFilesInMB'] = totalSizeMsg;\r\n\r\n    this.fileUploadFormControl.setErrors(currentErrors);\r\n    this.fileUploadFormControl.markAsTouched();\r\n  }\r\n\r\n  private validateFileSize(file: FileLikeObject): boolean {\r\n    const maxFileSize = this.options.fileMaxSizeInMB * this.BYTES_TO_MB;\r\n    if (file.size > maxFileSize) {\r\n      this.setFormControlError('FileMaxSizeInMB', `${this.options.fileMaxSizeInMB}MB`);\r\n      return false;\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private validateFileType(file: FileLikeObject): boolean {\r\n    if (this.options.fileUploadAcceptsTypes?.length) {\r\n      const fileType = file.type;\r\n      const isAccepted = this.acceptedTypeArray.some(type =>\r\n        type.toLowerCase() === fileType.toLowerCase()\r\n      );\r\n\r\n      if (!isAccepted) {\r\n        this.setFormControlError('ToolTipTypeError', this.toolTipTypeArray);\r\n        return false;\r\n      }\r\n    }\r\n    return true;\r\n  }\r\n\r\n  private validateIndividualFileSize(file: FileLikeObject): boolean {\r\n    const maxFileSize = this.options.fileMaxSizeInMB * this.BYTES_TO_MB;\r\n    return file.size <= maxFileSize;\r\n  }\r\n\r\n  private validateIndividualFileType(file: FileLikeObject): boolean {\r\n    if (!this.options.fileUploadAcceptsTypes?.length) {\r\n      return true;\r\n    }\r\n\r\n    const fileType = file.type;\r\n    return this.acceptedTypeArray.some(type =>\r\n      type.toLowerCase() === fileType.toLowerCase()\r\n    );\r\n  }\r\n\r\n\r\n  private validateDuplicateFileName(file: FileLikeObject, currentElement: FileItem): boolean {\r\n    if (!file?.name) {\r\n      return true;\r\n    }\r\n\r\n    const currentFileName = file.name.toLowerCase();\r\n\r\n    const existingFiles = this.uploader.queue.filter(item => item !== currentElement);\r\n    const duplicateExists = existingFiles.some(item => {\r\n      const existingFileName = item.file?.name || item._file?.name;\r\n      return existingFileName && existingFileName.toLowerCase() === currentFileName;\r\n    });\r\n\r\n    if (duplicateExists) {\r\n      return false;\r\n    }\r\n\r\n    if (this.options.value) {\r\n      const uploadedFiles = Array.isArray(this.options.value) ? this.options.value : [this.options.value];\r\n      const duplicateInUploaded = uploadedFiles.some(uploadedFile => {\r\n        const uploadedFileName = uploadedFile?.fileName || uploadedFile?.name;\r\n        return uploadedFileName && uploadedFileName.toLowerCase() === currentFileName;\r\n      });\r\n\r\n      return !duplicateInUploaded;\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n\r\n\r\n  private removeInvalidFiles(invalidFiles: FileItem[]): void {\r\n    invalidFiles.forEach(invalidFile => {\r\n      const index = this.uploader.queue.indexOf(invalidFile);\r\n      if (index > -1) {\r\n        this.uploader.queue.splice(index, 1);\r\n      }\r\n    });\r\n  }\r\n\r\n  private showValidationErrors(errors: string[]): void {\r\n    if (errors.length === 0) return;\r\n\r\n    const errorMessage = errors.join('<br/>');\r\n\r\n    this.setValidationError('InvalidFiles', errorMessage);\r\n    setTimeout(() => {\r\n      this.clearInvalidFilesError();\r\n    }, this.ERROR_DISPLAY_DURATION);\r\n  }\r\n\r\n  private setValidationError(errorKey: string, errorMessage: string): void {\r\n    const currentErrors = this.fileUploadFormControl.errors || {};\r\n    currentErrors[errorKey] = errorMessage;\r\n\r\n    this.fileUploadFormControl.setErrors(currentErrors);\r\n    this.fileUploadFormControl.markAsTouched();\r\n  }\r\n\r\n  private clearInvalidFilesError(): void {\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n    if (currentErrors && currentErrors['InvalidFiles']) {\r\n      delete currentErrors['InvalidFiles'];\r\n\r\n      if (Object.keys(currentErrors).length === 0) {\r\n        this.fileUploadFormControl.setErrors(null);\r\n      } else {\r\n        this.fileUploadFormControl.setErrors(currentErrors);\r\n      }\r\n    }\r\n  }\r\n\r\n  private setFormControlError(errorKey: string, errorValue: any): void {\r\n    this.fileUploadFormControl.setErrors({ [errorKey]: errorValue });\r\n    this.fileUploadFormControl.markAsTouched();\r\n    this.uploader.queue = [];\r\n  }\r\n\r\n  private handleAsyncFileUpload(element: FileItem, filesArray: FileDTO[]): void {\r\n    const uploadSubscription = this.fileUploadService.uploadFile(element._file).subscribe({\r\n      next: (event: any) => {\r\n        if (event.type === HttpEventType.UploadProgress) {\r\n          this.handleUploadProgress(element, event);\r\n        } else if (event.type === HttpEventType.Response) {\r\n          this.handleUploadComplete(element, event, filesArray);\r\n        }\r\n      },\r\n      error: (error) => {\r\n        console.error('Upload failed:', error);\r\n        // Handle upload error - you can add custom error handling here\r\n      }\r\n    });\r\n\r\n    // Store subscription for cleanup\r\n    this.subscriptions.add(uploadSubscription);\r\n  }\r\n\r\n  private handleUploadProgress(element: FileItem, event: UploadProgressEvent): void {\r\n    const queueIndex = this.uploader.queue.findIndex((file) => file === element);\r\n    if (queueIndex === -1) return;\r\n\r\n    const progress = Math.round((100 * event.loaded) / event.total);\r\n    this.uploader.queue[queueIndex].progress =\r\n      progress >= this.PROGRESS_NEAR_COMPLETE ? this.PROGRESS_NEAR_COMPLETE : progress;\r\n  }\r\n\r\n  private handleUploadComplete(element: FileItem, event: UploadCompleteEvent, filesArray: FileDTO[]): void {\r\n    const queueIndex = this.uploader.queue.findIndex((file) => file === element);\r\n    if (queueIndex === -1) return;\r\n\r\n    this.uploader.queue[queueIndex].progress = this.PROGRESS_COMPLETE;\r\n\r\n    const fileID = event.body.val;\r\n    this.updateElementWithFileInfo(element, fileID, event.body.downloadUrl);\r\n\r\n    const addedFile = this.createFileDTO(element, fileID, event.body.downloadUrl);\r\n    this.updateFormValue(addedFile, filesArray);\r\n  }\r\n\r\n  private updateElementWithFileInfo(element: FileItem, fileID: string, downloadUrl?: string): void {\r\n    element._file['iD_GUID'] = fileID;\r\n    element._file['isNew'] = true;\r\n\r\n    if (downloadUrl) {\r\n      element._file['url'] = downloadUrl;\r\n      (element.file as any).url = downloadUrl;\r\n    }\r\n  }\r\n\r\n  private createFileDTO(element: FileItem, fileID: string, downloadUrl?: string): FileDTO {\r\n    return {\r\n      iD_GUID: fileID,\r\n      fileName: element._file['name'],\r\n      fileType: element._file['type'],\r\n      isNew: true,\r\n      fileBase64: '',\r\n      fileSizeInMB: element._file.size / this.BYTES_TO_MB,\r\n      nameWithExtension: element._file['name'],\r\n      fullFileURL: downloadUrl || null\r\n    };\r\n  }\r\n\r\n  private handleSyncFileUpload(file: FileLikeObject, filesArray: FileDTO[]): void {\r\n    this.trackMemoryUsage(file.size);\r\n    const reader = new FileReader();\r\n\r\n    // Store reader reference for cleanup\r\n    const readerRef = reader;\r\n\r\n    reader.onload = () => {\r\n      try {\r\n        const existingGUID = this.getExistingFileGUID(file);\r\n        this.updateQueueItemForSync(file);\r\n\r\n        const addedFile = this.createSyncFileDTO(file, reader.result as string, existingGUID);\r\n        this.updateFormValue(addedFile, filesArray);\r\n        this.handlePatchAndEmit();\r\n      } finally {\r\n        // Clean up reader\r\n        readerRef.onload = null;\r\n        readerRef.onerror = null;\r\n        readerRef.onabort = null;\r\n      }\r\n    };\r\n\r\n    reader.onerror = () => {\r\n      console.error('File reading failed');\r\n      readerRef.onload = null;\r\n      readerRef.onerror = null;\r\n      readerRef.onabort = null;\r\n    };\r\n\r\n    reader.readAsDataURL(file.rawFile as File);\r\n  }\r\n\r\n  private updateQueueItemForSync(file: FileLikeObject): void {\r\n    const queueItem = this.findQueueItemByFile(file);\r\n\r\n    if (queueItem) {\r\n      this.preserveFileReference(queueItem, file);\r\n      queueItem.progress = this.PROGRESS_COMPLETE;\r\n    }\r\n  }\r\n\r\n  private findQueueItemByFile(file: FileLikeObject): FileItem | undefined {\r\n    return this.uploader.queue.find(item =>\r\n      item.file.name === file.name && item.file.size === file.size\r\n    );\r\n  }\r\n\r\n  private preserveFileReference(queueItem: FileItem, file: FileLikeObject): void {\r\n    (queueItem._file as any).rawFile = file.rawFile;\r\n    (queueItem.file as any).rawFile = file.rawFile;\r\n  }\r\n\r\n  private createSyncFileDTO(file: FileLikeObject, readerResult: string, existingGUID: string | null): FileDTO {\r\n    return {\r\n      fileName: file.name,\r\n      fileType: file.type,\r\n      fileBase64: readerResult.split(',')[1],\r\n      fileSizeInMB: file.size / this.BYTES_TO_MB,\r\n      nameWithExtension: file.name,\r\n      iD_GUID: existingGUID,\r\n      isNew: true,\r\n      fullFileURL: null\r\n    };\r\n  }\r\n\r\n  private getExistingFileGUID(file: FileLikeObject): string | null {\r\n    if (!this.options.isMultipleFile && this.file) {\r\n      return this.file.nameWithExtension === file.name ? this.file.iD_GUID : null;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  private updateFormValue(addedFile: FileDTO, filesArray: FileDTO[]): void {\r\n    if (!this.options.isMultipleFile) {\r\n      this.fileUploadModel = new FileUploadModel();\r\n      this.fileUploadModel.file = addedFile;\r\n      this.updateFormControl(this.fileUploadModel);\r\n    } else {\r\n      filesArray.push(addedFile);\r\n      this.multipleFileUploadModel.uploadedFiles = filesArray;\r\n      this.setupMultipleFileModel();\r\n      this.updateFormControl(this.multipleFileUploadModel);\r\n      this.isUploadComplete.emit(true);\r\n    }\r\n  }\r\n\r\n  private setupMultipleFileModel(): void {\r\n    if (this.options.value?.correlationID_GUID == null) {\r\n      this.multipleFileUploadModel.removedFiles = [];\r\n    }\r\n    this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;\r\n  }\r\n\r\n  private updateFormControl(value: FileUploadModel | MultipleFileUploadModel): void {\r\n    this.preserveErrorsAndUpdateValue(value);\r\n    this.options.value = value;\r\n  }\r\n\r\n  private preserveErrorsAndUpdateValue(value: FileUploadModel | MultipleFileUploadModel): void {\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n\r\n    this.fileUploadFormControl.setValue(value, { emitEvent: false });\r\n    this.group.get(this.options.name)?.setValue(value, { emitEvent: false });\r\n    if (currentErrors) {\r\n      this.fileUploadFormControl.setErrors(currentErrors);\r\n    }\r\n  }\r\n\r\n  private handlePatchAndEmit(): void {\r\n    const originalValue = this.group.get(this.options.name)?.value;\r\n\r\n    if (this.options.patchFunction && this.options.patchPath && this.group.get(this.options.name)?.valid) {\r\n      this.controlUtility.patchControlValue(\r\n        originalValue,\r\n        this.options.patchFunction,\r\n        this.options.patchPath\r\n      );\r\n    }\r\n\r\n    this.OnChange.emit(originalValue);\r\n  }\r\n  removeFromControlValue(item: FileItem): void {\r\n    // Clean up blob URL before removing\r\n    const downloadUrl = this.getFileDownloadUrl(item);\r\n    if (downloadUrl && downloadUrl.startsWith('blob:')) {\r\n      this.cleanupBlobUrl(downloadUrl);\r\n    }\r\n\r\n    this.handleAsyncFileDeletion(item);\r\n\r\n    if (!this.options.isMultipleFile) {\r\n      this.handleSingleFileRemoval();\r\n    } else {\r\n      this.handleMultipleFileRemoval(item);\r\n    }\r\n\r\n    this.checkAndClearMaxFileCountValidation();\r\n  }\r\n\r\n  private handleAsyncFileDeletion(item: FileItem): void {\r\n    if (this.options.isUploadFileAsync &&\r\n      item.progress === this.PROGRESS_COMPLETE &&\r\n      item._file['isNew']) {\r\n\r\n      const deleteSubscription = this.fileUploadService.deleteFile(item._file['iD_GUID']).subscribe({\r\n        error: (error) => console.error('Delete failed:', error)\r\n      });\r\n\r\n      this.subscriptions.add(deleteSubscription);\r\n    }\r\n  }\r\n\r\n  private handleSingleFileRemoval(): void {\r\n    this.uploader.queue = [];\r\n\r\n    this.fileUploadModel = null as any;\r\n\r\n    if (this.options.isRequired) {\r\n      this.fileUploadFormControl.markAsTouched();\r\n    }\r\n\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n\r\n    this.group.get(this.options.name)?.setValue(this.fileUploadModel, { emitEvent: false });\r\n    if (currentErrors) {\r\n      this.fileUploadFormControl.setErrors(currentErrors);\r\n    }\r\n    this.options.value = this.fileUploadModel;\r\n  }\r\n\r\n  private handleMultipleFileRemoval(item: FileItem): void {\r\n    // Clean up blob URL\r\n    const downloadUrl = this.getFileDownloadUrl(item);\r\n    if (downloadUrl && downloadUrl.startsWith('blob:')) {\r\n      this.cleanupBlobUrl(downloadUrl);\r\n    }\r\n\r\n    const queueIndex = this.uploader.queue.indexOf(item);\r\n    if (queueIndex > -1) {\r\n      this.uploader.queue.splice(queueIndex, 1);\r\n    }\r\n\r\n    this.processFileRemovalFromExisting(item);\r\n    this.removeFromUploadedFiles(item);\r\n    this.validateRemainingFiles();\r\n    this.updateMultipleFileModel();\r\n  }\r\n\r\n  private processFileRemovalFromExisting(item: FileItem): void {\r\n    if (!this.options.value) {\r\n      this.resetDeletedFiles();\r\n      return;\r\n    }\r\n\r\n    if (!this.options.value.correlationID_GUID) {\r\n      this.resetDeletedFiles();\r\n    } else {\r\n      this.handleExistingFileRemoval(item);\r\n    }\r\n  }\r\n\r\n  private resetDeletedFiles(): void {\r\n    this.deletedFiles = [];\r\n    this.multipleFileUploadModel.removedFiles = [];\r\n  }\r\n\r\n  private handleExistingFileRemoval(item: FileItem): void {\r\n    const fileName = item.file.rawFile.name;\r\n    const existingFile = this.multipleFileUploadModel.existingFiles\r\n      .find(obj => obj.nameWithExtension === fileName);\r\n\r\n    if (existingFile && !this.deletedFiles.some(obj => obj.nameWithExtension === fileName)) {\r\n      this.multipleFileUploadModel.existingFiles =\r\n        this.multipleFileUploadModel.existingFiles.filter(\r\n          obj => obj.nameWithExtension !== fileName\r\n        );\r\n      this.deletedFiles.push(existingFile);\r\n      this.multipleFileUploadModel.removedFiles.push(existingFile.iD_GUID);\r\n    }\r\n  }\r\n\r\n  private removeFromUploadedFiles(item: FileItem): void {\r\n    const itemFileName = item._file.name || item.file.name;\r\n    const itemFileSize = item._file.size || item.file.size;\r\n    const itemGUID = item._file['iD_GUID'];\r\n\r\n    this.multipleFileUploadModel.uploadedFiles =\r\n      this.multipleFileUploadModel.uploadedFiles.filter(obj => {\r\n        if (itemGUID && obj.iD_GUID) {\r\n          return obj.iD_GUID !== itemGUID;\r\n        }\r\n        const objFileName = obj.nameWithExtension || obj.fileName;\r\n        const objFileSize = obj.fileSizeInMB ? obj.fileSizeInMB * this.BYTES_TO_MB : 0;\r\n\r\n        return !(objFileName === itemFileName && Math.abs(objFileSize - itemFileSize) < 1000);\r\n      });\r\n  }\r\n\r\n  private validateRemainingFiles(): void {\r\n    if ((!this.multipleFileUploadModel.uploadedFiles ||\r\n      this.multipleFileUploadModel.uploadedFiles.length === 0) &&\r\n      this.options.isRequired) {\r\n      this.fileUploadFormControl.setErrors({\r\n        MinFileCountValidationKey: this.options.minNoOfFiles\r\n      });\r\n      this.fileUploadFormControl.markAsTouched();\r\n    }\r\n  }\r\n\r\n  private updateMultipleFileModel(): void {\r\n    this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;\r\n\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n\r\n    this.fileUploadFormControl.setValue(this.multipleFileUploadModel, { emitEvent: false });\r\n    this.group.get(this.options.name)?.setValue(this.multipleFileUploadModel, { emitEvent: false });\r\n    this.options.value = this.multipleFileUploadModel;\r\n    if (currentErrors) {\r\n      this.fileUploadFormControl.setErrors(currentErrors);\r\n    }\r\n  }\r\n  removeRequiredValidation = (): void => {\r\n    this.controlUtility.removeRequiredValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      this.options\r\n    );\r\n  };\r\n\r\n  addRequiredValidation = (): void => {\r\n    this.controlUtility.addRequiredValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      this.options\r\n    );\r\n  };\r\n\r\n  removeCustomValidation = (customValidation: any): void => {\r\n    this.controlUtility.removeCustomValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      customValidation\r\n    );\r\n  };\r\n\r\n  addCustomValidation = (customValidation: any): void => {\r\n    this.controlUtility.addCustomValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      customValidation\r\n    );\r\n  };\r\n\r\n  isValid = (): boolean => {\r\n    this.controlUtility.isValid(this.fileUploadFormControl);\r\n    return this.fileUploadFormControl.valid;\r\n  };\r\n  convertSizeToMB(size: number): number {\r\n    if (size === 0) {\r\n      return 0;\r\n    }\r\n\r\n    const BYTES_TO_MB_ACCURATE = 1024 * 1024;\r\n    const megabytes = size / BYTES_TO_MB_ACCURATE;\r\n    return Math.round(megabytes * 100) / 100;\r\n  }\r\n\r\n  trackByFunction(index: number, item: FileItem): any {\r\n    return item._file ? item._file.name + item._file.size : index;\r\n  }\r\n\r\n  shouldShowFileList(): boolean {\r\n    return this.uploader?.queue && this.uploader.queue.length > 0;\r\n  }\r\n\r\n  isDownloadEnabled(): boolean {\r\n    return true;\r\n  }\r\n\r\n  isRemoveEnabled(): boolean {\r\n    return !this.options.isReadonly && !this.options.isDisabled;\r\n  }\r\n\r\n  getFileDownloadUrl(item: FileItem): string | null {\r\n    const existingUrl = this.getExistingFileUrl(item);\r\n    if (existingUrl) {\r\n      return existingUrl;\r\n    }\r\n\r\n    return this.createFileUrl(item);\r\n  }\r\n\r\n  private getExistingFileUrl(item: FileItem): string | null {\r\n    return (item?.file as any)?.url ||\r\n      (item?._file as any)?.url ||\r\n      (item as any)?.url ||\r\n      (item?.file as any)?.rawFile?.url ||\r\n      (item?._file as any)?.rawFile?.url ||\r\n      null;\r\n  }\r\n\r\n  private createFileUrl(item: FileItem): string | null {\r\n    const fileName = this.getFileName(item);\r\n    const fileType = item?.file?.type || (item?._file as any)?.type;\r\n\r\n    const originalFile = (item?._file as any)?.rawFile || (item?.file as any)?.rawFile;\r\n    if (originalFile && originalFile instanceof File) {\r\n      return URL.createObjectURL(originalFile);\r\n    }\r\n\r\n    const base64Data = typeof originalFile === 'string' ? originalFile : null;\r\n    if (base64Data && fileName) {\r\n      return this.createBlobUrlWithFilename(base64Data, fileType, fileName);\r\n    }\r\n\r\n    const fileId = (item?._file as any)?.['iD_GUID'];\r\n    if (fileId && this.options.isUploadFileAsync) {\r\n      return this.constructDownloadUrl(fileId, fileName);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  private createBlobUrlWithFilename(base64Data: string, fileType: string, fileName: string): string {\r\n    try {\r\n      const byteCharacters = atob(base64Data);\r\n      const byteNumbers = new Array(byteCharacters.length);\r\n      for (let i = 0; i < byteCharacters.length; i++) {\r\n        byteNumbers[i] = byteCharacters.charCodeAt(i);\r\n      }\r\n      const byteArray = new Uint8Array(byteNumbers);\r\n      const blob = new Blob([byteArray], { type: fileType || 'application/octet-stream' });\r\n\r\n      return URL.createObjectURL(blob);\r\n    } catch (error) {\r\n      const errorMsg = this.utilityService.getResourceValue('ErrorCreatingBlobUrl');\r\n      console.error(errorMsg, error);\r\n      return `data:${fileType || 'application/octet-stream'};base64,${base64Data}`;\r\n    }\r\n  }\r\n\r\n  private constructDownloadUrl(fileId: string, fileName?: string): string {\r\n    const downloadBaseUrl = (this.options as any).downloadBaseUrl;\r\n    let url: string;\r\n\r\n    if (downloadBaseUrl) {\r\n      url = `${downloadBaseUrl}/${fileId}`;\r\n    } else {\r\n      url = `/api/files/download/${fileId}`;\r\n    }\r\n\r\n    if (fileName) {\r\n      const separator = url.includes('?') ? '&' : '?';\r\n      url += `${separator}filename=${encodeURIComponent(fileName)}`;\r\n    }\r\n\r\n    return url;\r\n  }\r\n\r\n  getFileName(item: FileItem): string {\r\n    return item?.file?.name || (item?._file as any)?.name || 'file';\r\n  }\r\n\r\n  downloadFile(item: FileItem): void {\r\n    const downloadInfo = this.prepareFileDownload(item);\r\n\r\n    if (!downloadInfo.url) {\r\n      this.handleDownloadError(downloadInfo.fileName);\r\n      return;\r\n    }\r\n\r\n    this.executeFileDownload(downloadInfo.url, downloadInfo.fileName);\r\n  }\r\n\r\n  private prepareFileDownload(item: FileItem): DownloadInfo {\r\n    return {\r\n      url: this.getFileDownloadUrl(item),\r\n      fileName: this.getFileName(item)\r\n    };\r\n  }\r\n\r\n  private handleDownloadError(fileName: string): void {\r\n    const errorMsg = this.utilityService.getResourceValue('NoDownloadUrlAvailable')\r\n      .replace('{fileName}', fileName);\r\n    console.error(errorMsg);\r\n\r\n  }\r\n\r\n  private executeFileDownload(url: string, fileName: string): void {\r\n    const link = this.createDownloadLink(url, fileName);\r\n    this.triggerDownload(link);\r\n    this.cleanupBlobUrl(url);\r\n  }\r\n\r\n  private createDownloadLink(url: string, fileName: string): HTMLAnchorElement {\r\n    const link = document.createElement('a');\r\n    link.href = url;\r\n    link.download = fileName;\r\n    link.style.display = 'none';\r\n    return link;\r\n  }\r\n\r\n  private triggerDownload(link: HTMLAnchorElement): void {\r\n    document.body.appendChild(link);\r\n    link.click();\r\n    document.body.removeChild(link);\r\n  }\r\n\r\n  private cleanupBlobUrl(url: string): void {\r\n    if (url.startsWith('blob:')) {\r\n      setTimeout(() => {\r\n        URL.revokeObjectURL(url);\r\n      }, 100);\r\n    }\r\n  }\r\n\r\n  private trackMemoryUsage(fileSize: number): void {\r\n    this.currentMemoryUsage += fileSize;\r\n\r\n    // If memory usage exceeds limit, clean up old files\r\n    if (this.currentMemoryUsage > this.MAX_MEMORY_USAGE) {\r\n      this.cleanupOldFiles();\r\n    }\r\n  }\r\n\r\n  private cleanupOldFiles(): void {\r\n    if (this.uploader?.queue && this.uploader.queue.length > 0) {\r\n      // Remove oldest files to free memory\r\n      const oldestFile = this.uploader.queue.shift();\r\n      if (oldestFile) {\r\n        const url = this.getFileDownloadUrl(oldestFile);\r\n        if (url && url.startsWith('blob:')) {\r\n          this.cleanupBlobUrl(url);\r\n        }\r\n        this.currentMemoryUsage -= (oldestFile.file?.size || 0);\r\n      }\r\n    }\r\n  }\r\n\r\n  private cleanupEventListeners(): void {\r\n    // Clear file input references\r\n    if (this.fileInput?.nativeElement) {\r\n      this.fileInput.nativeElement.value = '';\r\n    }\r\n\r\n    // Clear uploader queue to prevent memory leaks\r\n    if (this.uploader?.queue) {\r\n      this.uploader.queue = [];\r\n    }\r\n  }\r\n\r\n  private cleanupUploaderQueue(): void {\r\n    if (this.uploader?.queue) {\r\n      // Clear all items and their associated resources\r\n      this.uploader.queue.forEach(item => {\r\n        // Clean up blob URLs\r\n        const url = this.getFileDownloadUrl(item);\r\n        if (url && url.startsWith('blob:')) {\r\n          this.cleanupBlobUrl(url);\r\n        }\r\n\r\n        // Clear file references\r\n        if (item._file) {\r\n          item._file = null as any;\r\n        }\r\n        if (item.file) {\r\n          item.file = null as any;\r\n        }\r\n      });\r\n\r\n      // Clear the queue\r\n      this.uploader.queue = [];\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    // Clean up subscriptions\r\n    this.subscriptions.unsubscribe();\r\n\r\n    // Clean up uploader queue\r\n    this.cleanupUploaderQueue();\r\n\r\n    // Clean up event listeners\r\n    this.cleanupEventListeners();\r\n\r\n    // Clean up blob URLs\r\n    if (this.uploader?.queue) {\r\n      this.uploader.queue.forEach(item => {\r\n        const url = this.getFileDownloadUrl(item);\r\n        if (url && url.startsWith('blob:')) {\r\n          URL.revokeObjectURL(url);\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  private checkAndClearMaxFileCountValidation(): void {\r\n    if (!this.options.maxNoOfFiles || this.options.maxNoOfFiles <= 0) {\r\n      return;\r\n    }\r\n\r\n    const currentQueueLength = this.uploader.queue.length;\r\n    if (currentQueueLength <= this.options.maxNoOfFiles) {\r\n      this.clearFileCountError('MaxFileCountValidationKey');\r\n    }\r\n  }\r\n}\r\n","<div class=\"form-group bbsf-control bbsf-file-upload\" [formGroup]=\"group\">\r\n  <div [ngClass]=\"options.viewType === 1 ? 'bbsf-vertical' : 'bbsf-horizontal'\">\r\n    <!-- Label -->\r\n    <label [hidden]=\"options.hideLabel\" class=\"bbsf-label {{ options.labelExtraClasses }}\">\r\n      {{ options.labelValue }}\r\n      <!-- Required asterisk -->\r\n      <span *ngIf=\"options.isRequired && !options.isReadonly && (options.showAsterisk || true)\"\r\n        class=\"text-danger\">*</span>\r\n    </label>\r\n    <!-- Drop zone enabled -->\r\n    <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n      *ngIf=\"options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n      [ngClass]=\"{ 'another-file-over-class': hasAnotherDropZoneOver }\" (onFileDrop)=\"onFileChange()\"\r\n      (fileOver)=\"fileOverAnother($event)\" [uploader]=\"uploader\" [accept]=\"acceptedType\" [id]=\"options.name\"\r\n      [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\"\r\n      [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n      (click)=\"fileInputControl.click()\">\r\n\r\n      <div class=\"dropzone-label\">\r\n        <div class=\"svg-and-validation\">\r\n          <!-- Upload icon -->\r\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n            <path opacity=\"0.4\"\r\n              d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n              stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n            <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n              stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n          </svg>\r\n\r\n          <!-- Instruction text -->\r\n          <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n            {{ utilityService.getResourceValue('DragAndDropHere') }}\r\n          </div>\r\n\r\n          <!-- Validation messages -->\r\n          <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n          </div>\r\n\r\n          <div class=\"bbsf-validation-msg text-center text-danger\"\r\n            *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n            [innerHTML]=\"validationCountMessage\">\r\n          </div>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Hidden file input -->\r\n      <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n        class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n        [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n        (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\"\r\n        #fileInputControl [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n    </div>\r\n    <!-- Click to upload (no drop zone) -->\r\n    <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n      (click)=\"fileInput.click()\">\r\n\r\n      <div class=\"dropzone-label\">\r\n        <div class=\"svg-and-validation\">\r\n          <!-- Upload icon -->\r\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"70\" height=\"70\" viewBox=\"0 0 70 70\" fill=\"none\">\r\n            <path opacity=\"0.4\"\r\n              d=\"M58.333 48.8332C61.8957 45.8908 64.1663 41.4397 64.1663 36.4583C64.1663 27.5988 56.9843 20.4167 48.1247 20.4167C47.4874 20.4167 46.8912 20.0842 46.5675 19.5351C42.7641 13.0808 35.7417 8.75 27.708 8.75C15.6268 8.75 5.83301 18.5438 5.83301 30.625C5.83301 36.6511 8.26974 42.1082 12.2116 46.0644\"\r\n              stroke=\"#4B5489\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n            <path d=\"M23.333 46.6667L34.9997 35M34.9997 35L46.6663 46.6667M34.9997 35V61.25\" stroke=\"#4B5489\"\r\n              stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n          </svg>\r\n\r\n          <!-- Upload text -->\r\n          <div class=\"bbsf-validation-msg text-center\">\r\n            {{ utilityService.getResourceValue('Upload') }}\r\n          </div>\r\n\r\n          <!-- Validation messages -->\r\n          <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\">\r\n          </div>\r\n\r\n          <div class=\"bbsf-validation-msg text-center text-danger\"\r\n            *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n            [innerHTML]=\"validationCountMessage\">\r\n          </div>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Hidden file input -->\r\n      <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n        class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n        [attr.multiple]=\"options.isMultipleFile ? 'multiple' : null\" name=\"file\" type=\"file\" autocomplete=\"off\"\r\n        (change)=\"onFileChange()\" [ngClass]=\"options.viewType === 1 ? '' : 'col-md-9'\" [id]=\"options.name\" #fileInput\r\n        [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n    </div>\r\n    <!-- Read-only state with no files -->\r\n    <div *ngIf=\"options.isReadonly && (!options.value || (uploader.queue && uploader.queue.length === 0))\">\r\n      <span class=\"readonly-view\">{{ utilityService.getResourceValue('NA') }}</span>\r\n    </div>\r\n  </div>\r\n  <!-- Uploaded files list -->\r\n  <div class=\"uploaded-items\" *ngIf=\"shouldShowFileList()\">\r\n    <div class=\"btn-group\" *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n\r\n      <!-- Async upload completed files -->\r\n      <ng-container *ngIf=\"item?.progress === 100 && options.isUploadFileAsync\">\r\n        <!-- Download link - always visible -->\r\n        <button *ngIf=\"getFileDownloadUrl(item); else noUrlTemplate\" type=\"button\"\r\n          class=\"btn-download-file btn-sm btn-progress-upload\" (click)=\"downloadFile(item)\"\r\n          [title]=\"'Download ' + getFileName(item)\">\r\n          <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n        </button>\r\n\r\n        <!-- File name display when no URL available -->\r\n        <ng-template #noUrlTemplate>\r\n          <span class=\"btn-download-file btn-sm btn-progress-upload\" [title]=\"getFileName(item)\">\r\n            <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n          </span>\r\n        </ng-template>\r\n\r\n        <!-- Remove button - only show when not readonly and not disabled -->\r\n        <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm\" type=\"button\"\r\n          (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n          [title]=\"'Remove ' + getFileName(item)\">\r\n          <!-- Delete icon -->\r\n          <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\">\r\n            <path opacity=\"0.4\"\r\n              d=\"M9.33301 3.70584V3.26663C9.33301 2.65166 9.33301 2.34419 9.20587 2.1093C9.09405 1.9027 8.91555 1.73471 8.69604 1.62944C8.44647 1.50977 8.11977 1.50977 7.46638 1.50977H6.53305C5.87965 1.50977 5.55296 1.50977 5.30339 1.62944C5.08387 1.73471 4.90539 1.9027 4.79354 2.1093C4.66638 2.34419 4.66638 2.65166 4.66638 3.26663V3.70584\"\r\n              stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n            <path\r\n              d=\"M1.75 3.70605H12.25M11.0834 3.70605V9.8551C11.0834 10.7775 11.0834 11.2387 10.8926 11.591C10.7248 11.901 10.4571 12.1529 10.1278 12.3109C9.75345 12.4904 9.26345 12.4904 8.28334 12.4904H5.71666C4.73658 12.4904 4.24653 12.4904 3.87218 12.3109C3.5429 12.1529 3.27519 11.901 3.10741 11.591C2.91666 11.2387 2.91666 10.7775 2.91666 9.8551V3.70605\"\r\n              stroke=\"#D83731\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n          </svg>\r\n        </button>\r\n      </ng-container>\r\n\r\n      <!-- Sync upload files -->\r\n      <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n        <!-- Download link - always visible -->\r\n        <button type=\"button\" class=\"btn btn-download-file btn-sm\" (click)=\"downloadFile(item)\"\r\n          [title]=\"'Download ' + getFileName(item)\">\r\n          <!-- Download icon -->\r\n          <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\r\n            <path\r\n              d=\"M21 22H3C2.4 22 2 21.6 2 21C2 20.4 2.4 20 3 20H21C21.6 20 22 20.4 22 21C22 21.6 21.6 22 21 22ZM13 13.4V3C13 2.4 12.6 2 12 2C11.4 2 11 2.4 11 3V13.4H13Z\"\r\n              fill=\"currentColor\"></path>\r\n            <path opacity=\"0.3\" d=\"M7 13.4H17L12.7 17.7C12.3 18.1 11.7 18.1 11.3 17.7L7 13.4Z\" fill=\"currentColor\">\r\n            </path>\r\n          </svg>\r\n          <span class=\"file-name\">{{ getFileName(item) }}</span>\r\n        </button>\r\n\r\n        <!-- Remove button - only show when not readonly and not disabled -->\r\n        <button *ngIf=\"isRemoveEnabled()\" class=\"btn btn-download-file btn-sm btn-danger\" type=\"button\"\r\n          (click)=\"item.remove(); removeFromControlValue(item)\" [attr.aria-label]=\"'Remove ' + getFileName(item)\"\r\n          [title]=\"'Remove ' + getFileName(item)\">\r\n          <i class=\"fa fa-times px-0\" aria-hidden=\"true\"></i>\r\n        </button>\r\n      </ng-container>\r\n    </div>\r\n  </div>\r\n  <!-- File upload progress indicators -->\r\n  <div *ngFor=\"let item of uploader.queue; trackBy: trackByFunction\">\r\n    <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile }\"\r\n      *ngIf=\"item?.progress < 100 && options.isUploadFileAsync && !options.isReadonly\">\r\n\r\n      <div class=\"upload-items-toolbar\">\r\n        <h4>{{ getFileName(item) }}</h4>\r\n        <button *ngIf=\"isRemoveEnabled()\" type=\"button\" class=\"btn-cancel-upload\"\r\n          (click)=\"item.remove(); removeFromControlValue(item)\"\r\n          [attr.aria-label]=\"'Cancel upload for ' + getFileName(item)\">\r\n          <!-- Cancel icon -->\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_1324_13216)\">\r\n              <path opacity=\"0.4\"\r\n                d=\"M9 16.5C13.1421 16.5 16.5 13.1421 16.5 9C16.5 4.85786 13.1421 1.5 9 1.5C4.85786 1.5 1.5 4.85786 1.5 9C1.5 13.1421 4.85786 16.5 9 16.5Z\"\r\n                stroke=\"#DBE1F0\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n              <path d=\"M11.25 6.75L6.75 11.25M6.75 6.75L11.25 11.25\" stroke=\"#DBE1F0\" stroke-width=\"2\"\r\n                stroke-linecap=\"round\" stroke-linejoin=\"round\" />\r\n            </g>\r\n            <defs>\r\n              <clipPath id=\"clip0_1324_13216\">\r\n                <rect width=\"18\" height=\"18\" fill=\"white\" />\r\n              </clipPath>\r\n            </defs>\r\n          </svg>\r\n        </button>\r\n      </div>\r\n\r\n      <div class=\"progress\">\r\n        <div class=\"progress-bar\" role=\"progressbar\" [attr.aria-valuenow]=\"item?.progress\" aria-valuemin=\"0\"\r\n          aria-valuemax=\"100\" [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\"\r\n          *ngIf=\"item?.progress > 0\">\r\n        </div>\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n  <!-- Validation and description section -->\r\n  <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n    <!-- Validation messages -->\r\n    <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n      {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n    </div>\r\n\r\n    <!-- Control description -->\r\n    <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription\">\r\n      {{ options.labelDescription }}\r\n    </div>\r\n\r\n    <!-- Reset error state -->\r\n    <div *ngIf=\"(group.valid && group.dirty && group.touched) || (group.untouched && group.invalid && group.dirty)\">\r\n      {{ resetError() }}\r\n    </div>\r\n  </div>\r\n</div>"]}
|