@bnsights/bbsf-controls 1.0.194-beta.9 → 1.2.1
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 +1315 -1245
- package/fesm2022/bnsights-bbsf-controls.mjs +375 -345
- package/fesm2022/bnsights-bbsf-controls.mjs.map +1 -1
- package/lib/Shared/Models/DropdownOptions.d.ts +3 -0
- package/lib/Shared/Models/RepeaterOptions.d.ts +2 -0
- package/lib/controls/ConfirmationModal/ConfirmationModal.component.d.ts +0 -1
- package/lib/controls/MapAutoComplete/MapAutoComplete.component.d.ts +0 -1
- package/lib/controls/MultiLingualTextBox/MultiLingualTextBox.component.d.ts +1 -1
- package/lib/controls/Repeater/repeater/repeater.component.d.ts +7 -0
- package/lib/controls/Repeater/repeater-table/repeater-table.component.d.ts +7 -0
- package/lib/modules/bbsf-editors.module.d.ts +6 -7
- package/package.json +34 -37
- package/src/lib/assets/Style.scss +2 -0
- package/src/lib/assets/ace-builds/ace.js +1 -1
- package/src/lib/assets/ace-builds/mode-markdown.js +1 -1
- package/esm2022/bnsights-bbsf-controls.mjs +0 -5
- package/esm2022/lib/Shared/Components/app-base-component.mjs +0 -226
- package/esm2022/lib/Shared/Components/ng-tag-input.mjs +0 -336
- package/esm2022/lib/Shared/Directives/appPreventDoubleClick.directive.mjs +0 -39
- package/esm2022/lib/Shared/Directives/template-name.directive.mjs +0 -19
- package/esm2022/lib/Shared/Enums/CalendarView.mjs +0 -8
- package/esm2022/lib/Shared/Enums/ControlLayout.mjs +0 -6
- package/esm2022/lib/Shared/Enums/Countries.mjs +0 -247
- package/esm2022/lib/Shared/Enums/Enums.mjs +0 -36
- package/esm2022/lib/Shared/Enums/FileType.mjs +0 -29
- package/esm2022/lib/Shared/Enums/FilterType.mjs +0 -11
- package/esm2022/lib/Shared/Enums/FontSize.mjs +0 -7
- package/esm2022/lib/Shared/Enums/ForceDirection.mjs +0 -6
- package/esm2022/lib/Shared/Enums/IconPosition.mjs +0 -6
- package/esm2022/lib/Shared/Enums/ImageType.mjs +0 -10
- package/esm2022/lib/Shared/Enums/InputType.mjs +0 -10
- package/esm2022/lib/Shared/Enums/Insert.mjs +0 -9
- package/esm2022/lib/Shared/Enums/LanguageMode.mjs +0 -11
- package/esm2022/lib/Shared/Enums/LanguageValidation.mjs +0 -6
- package/esm2022/lib/Shared/Enums/Markdown.mjs +0 -20
- package/esm2022/lib/Shared/Enums/Misc.mjs +0 -7
- package/esm2022/lib/Shared/Enums/PagingActionMode.mjs +0 -6
- package/esm2022/lib/Shared/Enums/Para.mjs +0 -9
- package/esm2022/lib/Shared/Enums/PickerType.mjs +0 -7
- package/esm2022/lib/Shared/Enums/SelectMode.mjs +0 -6
- package/esm2022/lib/Shared/Enums/StartView.mjs +0 -7
- package/esm2022/lib/Shared/Enums/Style.mjs +0 -8
- package/esm2022/lib/Shared/Enums/StyleConfirmationMode.mjs +0 -7
- package/esm2022/lib/Shared/Enums/TagInputView.mjs +0 -7
- package/esm2022/lib/Shared/Enums/ToolbarButtons.mjs +0 -33
- package/esm2022/lib/Shared/Enums/map-enums.mjs +0 -15
- package/esm2022/lib/Shared/Enums/menu-list-enum.mjs +0 -6
- package/esm2022/lib/Shared/Models/Attribute.mjs +0 -3
- package/esm2022/lib/Shared/Models/AutocompleteDTO.mjs +0 -8
- package/esm2022/lib/Shared/Models/AutocompleteOptions.mjs +0 -39
- package/esm2022/lib/Shared/Models/CalendarDTO.mjs +0 -3
- package/esm2022/lib/Shared/Models/CalendarEventDTO.mjs +0 -3
- package/esm2022/lib/Shared/Models/CalendarOptions.mjs +0 -19
- package/esm2022/lib/Shared/Models/CancelDTO.mjs +0 -7
- package/esm2022/lib/Shared/Models/CaptchaStyle.mjs +0 -3
- package/esm2022/lib/Shared/Models/CheckBoxOptions.mjs +0 -11
- package/esm2022/lib/Shared/Models/ConfirmationModalOptions.mjs +0 -39
- package/esm2022/lib/Shared/Models/ControlOptionsBase.mjs +0 -17
- package/esm2022/lib/Shared/Models/CustomValidation.mjs +0 -14
- package/esm2022/lib/Shared/Models/DatePickerOptions.mjs +0 -40
- package/esm2022/lib/Shared/Models/DropdownListItem.mjs +0 -3
- package/esm2022/lib/Shared/Models/DropdownOptions.mjs +0 -38
- package/esm2022/lib/Shared/Models/EditPersonalImage.mjs +0 -7
- package/esm2022/lib/Shared/Models/EnglishArabicDTO.mjs +0 -10
- package/esm2022/lib/Shared/Models/FileDTO.mjs +0 -24
- package/esm2022/lib/Shared/Models/FileOptions.mjs +0 -15
- package/esm2022/lib/Shared/Models/FileUploadModel.mjs +0 -3
- package/esm2022/lib/Shared/Models/FileUploadOptions.mjs +0 -13
- package/esm2022/lib/Shared/Models/FilterItem.mjs +0 -8
- package/esm2022/lib/Shared/Models/FormOptions.mjs +0 -32
- package/esm2022/lib/Shared/Models/HtmlEditorOptions.mjs +0 -12
- package/esm2022/lib/Shared/Models/ImageUploadOptions.mjs +0 -14
- package/esm2022/lib/Shared/Models/MapAutoCompleteOptions.mjs +0 -12
- package/esm2022/lib/Shared/Models/MapAutocompleteDTO.mjs +0 -6
- package/esm2022/lib/Shared/Models/MarkdownEditorOptions.mjs +0 -24
- package/esm2022/lib/Shared/Models/MultiLingualHtmlEditorOptions.mjs +0 -9
- package/esm2022/lib/Shared/Models/MultiLingualTextAreaOptions.mjs +0 -9
- package/esm2022/lib/Shared/Models/MultiLingualTextBoxOptions.mjs +0 -17
- package/esm2022/lib/Shared/Models/MultilingualControlOptionsBase.mjs +0 -21
- package/esm2022/lib/Shared/Models/MultipleFileUploadModel.mjs +0 -9
- package/esm2022/lib/Shared/Models/PagingDTO.mjs +0 -13
- package/esm2022/lib/Shared/Models/PagingOptions.mjs +0 -49
- package/esm2022/lib/Shared/Models/PagingPayload.mjs +0 -3
- package/esm2022/lib/Shared/Models/PhoneOptions.mjs +0 -16
- package/esm2022/lib/Shared/Models/ProfileImageUploadOptions.mjs +0 -11
- package/esm2022/lib/Shared/Models/ProfilePictureDTO.mjs +0 -3
- package/esm2022/lib/Shared/Models/RadioButtonItem.mjs +0 -3
- package/esm2022/lib/Shared/Models/RadioButtonOptions.mjs +0 -7
- package/esm2022/lib/Shared/Models/RangeNumber.mjs +0 -3
- package/esm2022/lib/Shared/Models/Recaptcha.mjs +0 -6
- package/esm2022/lib/Shared/Models/RecaptchaOptions.mjs +0 -22
- package/esm2022/lib/Shared/Models/RepeaterField.mjs +0 -3
- package/esm2022/lib/Shared/Models/RepeaterOptions.mjs +0 -14
- package/esm2022/lib/Shared/Models/SaveDTO.mjs +0 -7
- package/esm2022/lib/Shared/Models/TagsInputDTO.mjs +0 -3
- package/esm2022/lib/Shared/Models/TagsInputOptions.mjs +0 -56
- package/esm2022/lib/Shared/Models/TextAreaOptions.mjs +0 -19
- package/esm2022/lib/Shared/Models/TextBoxOptions.mjs +0 -27
- package/esm2022/lib/Shared/Models/ToggleSlideOptions.mjs +0 -9
- package/esm2022/lib/Shared/Models/UploadPersonalImage.mjs +0 -7
- package/esm2022/lib/Shared/Models/bread-crumb.mjs +0 -8
- package/esm2022/lib/Shared/Models/control-filter-Item.mjs +0 -6
- package/esm2022/lib/Shared/Models/filter-options.mjs +0 -9
- package/esm2022/lib/Shared/Models/index.mjs +0 -58
- package/esm2022/lib/Shared/Models/page-header-options.mjs +0 -78
- package/esm2022/lib/Shared/Pipes/bbsf-date-time.pipe.mjs +0 -22
- package/esm2022/lib/Shared/Pipes/bbsf-date.pipe.mjs +0 -22
- package/esm2022/lib/Shared/config/environment.mjs +0 -12
- package/esm2022/lib/Shared/default_intl.mjs +0 -29
- package/esm2022/lib/Shared/services/ControlUtility.mjs +0 -162
- package/esm2022/lib/Shared/services/GlobalSettings.service.mjs +0 -30
- package/esm2022/lib/Shared/services/OnPagingFiltersChange.service.mjs +0 -24
- package/esm2022/lib/Shared/services/file-upload.service.mjs +0 -29
- package/esm2022/lib/Shared/services/index.mjs +0 -7
- package/esm2022/lib/Shared/services/render-component-service.service.mjs +0 -32
- package/esm2022/lib/Shared/services/validationErrorMassage.service.mjs +0 -125
- package/esm2022/lib/Shared/utils/date-formatter.mjs +0 -48
- package/esm2022/lib/controls/AutocompleteTextBox/AutocompleteTextBox.component.mjs +0 -299
- package/esm2022/lib/controls/Calendar/Calendar.component.mjs +0 -179
- package/esm2022/lib/controls/CheckBox/CheckBox.component.mjs +0 -121
- package/esm2022/lib/controls/ConfirmationModal/ConfirmationModal.component.mjs +0 -143
- package/esm2022/lib/controls/DateTimePicker/DateTimePicker.component.mjs +0 -240
- package/esm2022/lib/controls/DropdownList/DropdownList.component.mjs +0 -296
- package/esm2022/lib/controls/FileUpload/FileUpload.component.mjs +0 -741
- package/esm2022/lib/controls/Form/Form.component.mjs +0 -117
- package/esm2022/lib/controls/HtmlEditor/HtmlEditor.component.mjs +0 -185
- package/esm2022/lib/controls/ImageUpload/ImageUpload.component.mjs +0 -360
- package/esm2022/lib/controls/MapAutoComplete/MapAutoComplete.component.mjs +0 -502
- package/esm2022/lib/controls/MarkdownEditor/markdown-editor.component.mjs +0 -243
- package/esm2022/lib/controls/MultiLingualHtmlEditor/MultiLingualHtmlEditor.component.mjs +0 -456
- package/esm2022/lib/controls/MultiLingualTextArea/MultiLingualTextArea.component.mjs +0 -441
- package/esm2022/lib/controls/MultiLingualTextBox/MultiLingualTextBox.component.mjs +0 -500
- package/esm2022/lib/controls/Paging/JwPagination.component.mjs +0 -160
- package/esm2022/lib/controls/Paging/Paging.component.mjs +0 -626
- package/esm2022/lib/controls/Phone/Phone.component.mjs +0 -134
- package/esm2022/lib/controls/ProfileImageUploader/ProfileImageUploader.component.mjs +0 -390
- package/esm2022/lib/controls/RadioButton/RadioButton.component.mjs +0 -113
- package/esm2022/lib/controls/Recaptcha/Recaptcha.component.mjs +0 -134
- package/esm2022/lib/controls/Repeater/repeater/repeater.component.mjs +0 -135
- package/esm2022/lib/controls/Repeater/repeater-field-builder/repeater-field-builder.component.mjs +0 -1291
- package/esm2022/lib/controls/Repeater/repeater-item-field/repeater-item-field.component.mjs +0 -22
- package/esm2022/lib/controls/Repeater/repeater-table/repeater-table.component.mjs +0 -118
- package/esm2022/lib/controls/TagsInput/TagsInput.component.mjs +0 -308
- package/esm2022/lib/controls/TextArea/TextArea.component.mjs +0 -581
- package/esm2022/lib/controls/TextBox/TextBox.component.mjs +0 -357
- package/esm2022/lib/controls/Toggleslide/Toggleslide.component.mjs +0 -96
- package/esm2022/lib/controls/bbsf-controls.module.mjs +0 -92
- package/esm2022/lib/controls/page-header-component/page-header-component.component.mjs +0 -42
- package/esm2022/lib/modules/bbsf-core.module.mjs +0 -210
- package/esm2022/lib/modules/bbsf-datetime.module.mjs +0 -90
- package/esm2022/lib/modules/bbsf-dropdown.module.mjs +0 -57
- package/esm2022/lib/modules/bbsf-editors.module.mjs +0 -97
- package/esm2022/lib/modules/bbsf-forms-basic.module.mjs +0 -79
- package/esm2022/lib/modules/bbsf-multilingual.module.mjs +0 -60
- package/esm2022/lib/modules/bbsf-phone.module.mjs +0 -57
- package/esm2022/lib/modules/bbsf-specialized.module.mjs +0 -160
- package/esm2022/lib/modules/bbsf-uploads.module.mjs +0 -106
- package/esm2022/lib/modules/bbsf-utility.module.mjs +0 -103
- package/esm2022/public-api.mjs +0 -169
|
@@ -1,741 +0,0 @@
|
|
|
1
|
-
import { Component, Input, Optional, ViewChild, Output, EventEmitter, CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA } from '@angular/core';
|
|
2
|
-
import { CommonModule } from '@angular/common';
|
|
3
|
-
import { HttpEventType } from '@angular/common/http';
|
|
4
|
-
import { FormControl, Validators, FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
5
|
-
import { FileUploader, FileUploadModule } from 'ng2-file-upload';
|
|
6
|
-
import { NgxDropzoneModule } from 'ngx-dropzone';
|
|
7
|
-
import { Subject } from 'rxjs';
|
|
8
|
-
import { takeUntil } from 'rxjs/operators';
|
|
9
|
-
import { FileUploadModel, MultipleFileUploadModel } from '../../Shared/Models';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "@angular/forms";
|
|
12
|
-
import * as i2 from "../../Shared/services";
|
|
13
|
-
import * as i3 from "@bnsights/bbsf-utilities/ui";
|
|
14
|
-
import * as i4 from "@angular/common";
|
|
15
|
-
import * as i5 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.OnChange = new EventEmitter();
|
|
27
|
-
this.isUploadComplete = new EventEmitter();
|
|
28
|
-
this.validationMessage = '';
|
|
29
|
-
this.validationCountMessage = '';
|
|
30
|
-
this.acceptedType = '';
|
|
31
|
-
this.acceptedTypeArray = [];
|
|
32
|
-
this.toolTipTypeArray = [];
|
|
33
|
-
this.markAllAsTouched = false;
|
|
34
|
-
this.validationRules = [];
|
|
35
|
-
this.validationRulesAsync = [];
|
|
36
|
-
this.deletedFiles = [];
|
|
37
|
-
this.destroy$ = new Subject();
|
|
38
|
-
this.pendingValidationErrors = null;
|
|
39
|
-
this.resetError = () => {
|
|
40
|
-
this.controlValidationService.removeGlobalError();
|
|
41
|
-
};
|
|
42
|
-
//External Method
|
|
43
|
-
this.removeRequiredValidation = () => {
|
|
44
|
-
this.controlUtility.removeRequiredValidation(this.fileUploadFormControl, this.validationRules, this.options);
|
|
45
|
-
};
|
|
46
|
-
//External Method
|
|
47
|
-
this.addRequiredValidation = () => {
|
|
48
|
-
this.controlUtility.addRequiredValidation(this.fileUploadFormControl, this.validationRules, this.options);
|
|
49
|
-
};
|
|
50
|
-
//External Method
|
|
51
|
-
this.removeCustomValidation = (customValidation) => {
|
|
52
|
-
this.controlUtility.removeCustomValidation(this.fileUploadFormControl, this.validationRules, customValidation);
|
|
53
|
-
};
|
|
54
|
-
//External Method
|
|
55
|
-
this.addCustomValidation = (customValidation) => {
|
|
56
|
-
this.controlUtility.addCustomValidation(this.fileUploadFormControl, this.validationRules, customValidation);
|
|
57
|
-
};
|
|
58
|
-
//External Method
|
|
59
|
-
this.isValid = () => {
|
|
60
|
-
this.controlUtility.isValid(this.fileUploadFormControl);
|
|
61
|
-
};
|
|
62
|
-
FileUploadComponent.controlContainerStatic = this.controlContainer;
|
|
63
|
-
// Initialize uploader with minimal options, URL will be set in ngOnInit only if needed
|
|
64
|
-
this.uploader = new FileUploader({
|
|
65
|
-
url: '',
|
|
66
|
-
disableMultipart: false // 'DisableMultipart' must be 'true' for formatDataFunction to be called.
|
|
67
|
-
});
|
|
68
|
-
this.hasAnotherDropZoneOver = false;
|
|
69
|
-
}
|
|
70
|
-
ngOnInit() {
|
|
71
|
-
this.fileUploadModel = new FileUploadModel();
|
|
72
|
-
this.multipleFileUploadModel = new MultipleFileUploadModel();
|
|
73
|
-
if (!this.options.viewType)
|
|
74
|
-
this.options.viewType = this.globalSettings.viewType;
|
|
75
|
-
// Configure uploader options - no URL needed since we handle uploads via FileUploadService
|
|
76
|
-
// Note: allowedMimeType in ng2-file-upload doesn't work well with comma-separated enum values
|
|
77
|
-
// We handle MIME type validation manually in onFileChange
|
|
78
|
-
this.uploader.setOptions({
|
|
79
|
-
url: '',
|
|
80
|
-
disableMultipart: false,
|
|
81
|
-
autoUpload: false // We manually control uploads
|
|
82
|
-
// allowedMimeType removed - we handle validation manually for better control
|
|
83
|
-
});
|
|
84
|
-
if (this.options.value == null) {
|
|
85
|
-
if (this.options.isMultipleFile == true) {
|
|
86
|
-
//this.options.Value=this.multipleFileUploadModel
|
|
87
|
-
}
|
|
88
|
-
else {
|
|
89
|
-
// this.options.Value=this.fileUploadModel
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
else {
|
|
93
|
-
if (this.options.isMultipleFile == true) {
|
|
94
|
-
let files = [];
|
|
95
|
-
this.multipleFileUploadModel.existingFiles = this.options.value.existingFiles;
|
|
96
|
-
this.multipleFileUploadModel.uploadedFiles = [];
|
|
97
|
-
for (let index = 0; index < this.options.value.existingFiles.length; index++) {
|
|
98
|
-
const element = this.options.value.existingFiles[index];
|
|
99
|
-
var bytes = new Uint8Array(element.bytes);
|
|
100
|
-
var base64 = btoa(String.fromCharCode(...Array.from(bytes)));
|
|
101
|
-
this.fileLikeObject = {
|
|
102
|
-
name: element.nameWithExtension,
|
|
103
|
-
type: element.mimeType,
|
|
104
|
-
rawFile: base64
|
|
105
|
-
};
|
|
106
|
-
// let blob: any;
|
|
107
|
-
// blob = new Blob(['']) as any;
|
|
108
|
-
// blob.name = element.FileName;
|
|
109
|
-
// blob.lastModifiedDate = null;
|
|
110
|
-
// blob.webkitRelativePath = '';
|
|
111
|
-
let file = this.fileLikeObject;
|
|
112
|
-
file.url = element.fullFileURL;
|
|
113
|
-
files.push(file);
|
|
114
|
-
}
|
|
115
|
-
this.uploader.addToQueue(files);
|
|
116
|
-
console.log(this.uploader.queue);
|
|
117
|
-
}
|
|
118
|
-
else {
|
|
119
|
-
const element = this.options.value.file ?? this.options.value;
|
|
120
|
-
var bytes = new Uint8Array(element.bytes);
|
|
121
|
-
var base64 = btoa(String.fromCharCode(...Array.from(bytes)));
|
|
122
|
-
this.fileLikeObject = {
|
|
123
|
-
name: element.nameWithExtension,
|
|
124
|
-
type: element.mimeType,
|
|
125
|
-
rawFile: base64
|
|
126
|
-
};
|
|
127
|
-
this.file = element;
|
|
128
|
-
let file = this.fileLikeObject;
|
|
129
|
-
file.url = element.fullFileURL;
|
|
130
|
-
this.uploader.addToQueue([file]);
|
|
131
|
-
if (!this.options.value.file) {
|
|
132
|
-
this.fileUploadModel = new FileUploadModel();
|
|
133
|
-
this.fileUploadModel.file = this.options.value;
|
|
134
|
-
}
|
|
135
|
-
else
|
|
136
|
-
this.fileUploadModel = this.options.value;
|
|
137
|
-
this.options.value = this.fileUploadModel;
|
|
138
|
-
}
|
|
139
|
-
this.uploader.queue.forEach((element) => {
|
|
140
|
-
element.progress = 100;
|
|
141
|
-
});
|
|
142
|
-
}
|
|
143
|
-
if (this.options.labelKey != null && this.options.labelKey != '')
|
|
144
|
-
this.options.labelValue = this.UtilityService.getResourceValue(this.options.labelKey);
|
|
145
|
-
if (this.options.fileUploadAcceptsTypes != null &&
|
|
146
|
-
this.options.fileUploadAcceptsTypes.length > 0) {
|
|
147
|
-
// Process FileType enum values which may contain comma-separated MIME types
|
|
148
|
-
for (let index = 0; index < this.options.fileUploadAcceptsTypes.length; index++) {
|
|
149
|
-
const Type = this.options.fileUploadAcceptsTypes[index];
|
|
150
|
-
this.acceptedType = this.acceptedType + Type + ',';
|
|
151
|
-
}
|
|
152
|
-
this.acceptedTypeArray = this.acceptedType.split(',');
|
|
153
|
-
this.acceptedTypeArray = this.acceptedTypeArray.filter((value) => value.trim() != '');
|
|
154
|
-
// Rebuild acceptedType string for HTML accept attribute (properly formatted)
|
|
155
|
-
this.acceptedType = this.acceptedTypeArray.join(',');
|
|
156
|
-
// Process each accepted MIME type to build tooltip display names
|
|
157
|
-
for (let index = 0; index < this.acceptedTypeArray.length; index++) {
|
|
158
|
-
const element = this.acceptedTypeArray[index].trim();
|
|
159
|
-
switch (element) {
|
|
160
|
-
case 'application/pdf':
|
|
161
|
-
if (!this.toolTipTypeArray.includes('PDF'))
|
|
162
|
-
this.toolTipTypeArray.push('PDF');
|
|
163
|
-
break;
|
|
164
|
-
case 'application/msword':
|
|
165
|
-
if (!this.toolTipTypeArray.includes('Word'))
|
|
166
|
-
this.toolTipTypeArray.push('Word');
|
|
167
|
-
break;
|
|
168
|
-
case 'application/vnd.openxmlformats-officedocument.wordprocessingml.document':
|
|
169
|
-
if (!this.toolTipTypeArray.includes('Word'))
|
|
170
|
-
this.toolTipTypeArray.push('Word');
|
|
171
|
-
break;
|
|
172
|
-
case 'application/vnd.ms-excel':
|
|
173
|
-
if (!this.toolTipTypeArray.includes('Excel'))
|
|
174
|
-
this.toolTipTypeArray.push('Excel');
|
|
175
|
-
break;
|
|
176
|
-
case 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet':
|
|
177
|
-
if (!this.toolTipTypeArray.includes('Excel'))
|
|
178
|
-
this.toolTipTypeArray.push('Excel');
|
|
179
|
-
break;
|
|
180
|
-
case 'application/vnd.ms-powerpoint':
|
|
181
|
-
if (!this.toolTipTypeArray.includes('PowerPoint'))
|
|
182
|
-
this.toolTipTypeArray.push('PowerPoint');
|
|
183
|
-
break;
|
|
184
|
-
case 'application/vnd.openxmlformats-officedocument.presentationml.presentation':
|
|
185
|
-
if (!this.toolTipTypeArray.includes('PowerPoint'))
|
|
186
|
-
this.toolTipTypeArray.push('PowerPoint');
|
|
187
|
-
break;
|
|
188
|
-
case 'image/png':
|
|
189
|
-
if (!this.toolTipTypeArray.includes('PNG'))
|
|
190
|
-
this.toolTipTypeArray.push('PNG');
|
|
191
|
-
break;
|
|
192
|
-
case 'image/bmp':
|
|
193
|
-
if (!this.toolTipTypeArray.includes('BMP'))
|
|
194
|
-
this.toolTipTypeArray.push('BMP');
|
|
195
|
-
break;
|
|
196
|
-
case 'image/jpeg':
|
|
197
|
-
if (!this.toolTipTypeArray.includes('JPEG'))
|
|
198
|
-
this.toolTipTypeArray.push('JPEG');
|
|
199
|
-
break;
|
|
200
|
-
case 'application/zip':
|
|
201
|
-
if (!this.toolTipTypeArray.includes('ZIP'))
|
|
202
|
-
this.toolTipTypeArray.push('ZIP');
|
|
203
|
-
break;
|
|
204
|
-
case 'application/x-rar-compressed':
|
|
205
|
-
if (!this.toolTipTypeArray.includes('RAR'))
|
|
206
|
-
this.toolTipTypeArray.push('RAR');
|
|
207
|
-
break;
|
|
208
|
-
case 'video/mp4':
|
|
209
|
-
if (!this.toolTipTypeArray.includes('MP4'))
|
|
210
|
-
this.toolTipTypeArray.push('MP4');
|
|
211
|
-
break;
|
|
212
|
-
case 'video/avi':
|
|
213
|
-
if (!this.toolTipTypeArray.includes('AVI'))
|
|
214
|
-
this.toolTipTypeArray.push('AVI');
|
|
215
|
-
break;
|
|
216
|
-
case 'video/quicktime':
|
|
217
|
-
if (!this.toolTipTypeArray.includes('MOV'))
|
|
218
|
-
this.toolTipTypeArray.push('MOV');
|
|
219
|
-
break;
|
|
220
|
-
case 'video/mpeg':
|
|
221
|
-
if (!this.toolTipTypeArray.includes('MPEG'))
|
|
222
|
-
this.toolTipTypeArray.push('MPEG');
|
|
223
|
-
break;
|
|
224
|
-
case 'audio/mpeg':
|
|
225
|
-
if (!this.toolTipTypeArray.includes('MP3'))
|
|
226
|
-
this.toolTipTypeArray.push('MP3');
|
|
227
|
-
break;
|
|
228
|
-
case 'video/x-flv':
|
|
229
|
-
if (!this.toolTipTypeArray.includes('FLV'))
|
|
230
|
-
this.toolTipTypeArray.push('FLV');
|
|
231
|
-
break;
|
|
232
|
-
case 'video/x-ms-wmv':
|
|
233
|
-
if (!this.toolTipTypeArray.includes('WMV'))
|
|
234
|
-
this.toolTipTypeArray.push('WMV');
|
|
235
|
-
break;
|
|
236
|
-
case 'image/svg+xml':
|
|
237
|
-
if (!this.toolTipTypeArray.includes('SVG'))
|
|
238
|
-
this.toolTipTypeArray.push('SVG');
|
|
239
|
-
break;
|
|
240
|
-
case 'text/plain':
|
|
241
|
-
if (!this.toolTipTypeArray.includes('Txt'))
|
|
242
|
-
this.toolTipTypeArray.push('Txt');
|
|
243
|
-
break;
|
|
244
|
-
case 'application/BN':
|
|
245
|
-
if (!this.toolTipTypeArray.includes('License'))
|
|
246
|
-
this.toolTipTypeArray.push('License');
|
|
247
|
-
break;
|
|
248
|
-
default:
|
|
249
|
-
break;
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
this.validationMessage = this.validationMessage + `${this.UtilityService.getResourceValue('Extensions')} (${this.toolTipTypeArray}) `;
|
|
253
|
-
}
|
|
254
|
-
if (this.options.fileMaxSizeInMB > 0) {
|
|
255
|
-
this.validationMessage = this.validationMessage + `<br /> ${this.UtilityService.getResourceValue('FileMaxSizeInMB') + this.options.fileMaxSizeInMB + ' ' + this.UtilityService.getResourceValue('MB')}`;
|
|
256
|
-
}
|
|
257
|
-
if (this.options.minNoOfFiles > 0) {
|
|
258
|
-
this.validationMessage = this.validationMessage + `<br /> ${this.UtilityService.getResourceValue('MinFileCountValidationKey') + this.options.minNoOfFiles}`;
|
|
259
|
-
}
|
|
260
|
-
if (this.options.maxNoOfFiles > 0) {
|
|
261
|
-
this.validationMessage = this.validationMessage + `<br /> ${this.UtilityService.getResourceValue('MaxFileCountValidationKey') + this.options.maxNoOfFiles}`;
|
|
262
|
-
}
|
|
263
|
-
this.group.addControl(this.options.name, new FormControl(''));
|
|
264
|
-
this.fileUploadFormControl = this.group.controls[this.options.name];
|
|
265
|
-
if (this.options.customValidation.length > 0) {
|
|
266
|
-
let Validations = this.options.customValidation;
|
|
267
|
-
for (let index = 0; index < Validations.length; index++) {
|
|
268
|
-
const Validation = Validations[index];
|
|
269
|
-
this.validationRules.push(Validation.functionBody);
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
if (this.options.isMultipleFile && this.options.maxNoOfFiles > 0) {
|
|
273
|
-
this.validationCountMessage = `${this.UtilityService.getResourceValue('MaxFilesCount')} : ${this.options.maxNoOfFiles}`;
|
|
274
|
-
}
|
|
275
|
-
if (this.options.isRequired) {
|
|
276
|
-
this.validationRules.push(Validators.required);
|
|
277
|
-
}
|
|
278
|
-
this.fileUploadFormControl.setValidators(this.validationRules);
|
|
279
|
-
this.fileUploadFormControl.setAsyncValidators(this.validationRulesAsync);
|
|
280
|
-
if (this.options.isDisabled) {
|
|
281
|
-
this.fileUploadFormControl.disable();
|
|
282
|
-
}
|
|
283
|
-
this.MultipleFileUploadControlHost.ngSubmit
|
|
284
|
-
.pipe(takeUntil(this.destroy$))
|
|
285
|
-
.subscribe((value) => {
|
|
286
|
-
this.group.markAllAsTouched();
|
|
287
|
-
this.markAllAsTouched = true;
|
|
288
|
-
});
|
|
289
|
-
this.fileUploadFormControl.setValue(this.options.value);
|
|
290
|
-
}
|
|
291
|
-
ngAfterViewInit() {
|
|
292
|
-
if (this.options.attributeList != null) {
|
|
293
|
-
var element = document.getElementById(this.options.name);
|
|
294
|
-
for (let index = 0; index < this.options.attributeList.length; index++) {
|
|
295
|
-
element.setAttribute(this.options.attributeList[index].key, this.options.attributeList[index].value);
|
|
296
|
-
}
|
|
297
|
-
}
|
|
298
|
-
}
|
|
299
|
-
showGlobalError() {
|
|
300
|
-
this.controlUtility.showGlobalError();
|
|
301
|
-
}
|
|
302
|
-
getErrorValidation(ErrorList) {
|
|
303
|
-
if (this.markAllAsTouched && this.group.invalid) {
|
|
304
|
-
this.showGlobalError();
|
|
305
|
-
this.markAllAsTouched = false;
|
|
306
|
-
}
|
|
307
|
-
return this.controlUtility.getErrorValidationMassage(ErrorList, this.group, this.options);
|
|
308
|
-
}
|
|
309
|
-
fileOverAnother(event) {
|
|
310
|
-
this.hasAnotherDropZoneOver = event;
|
|
311
|
-
}
|
|
312
|
-
isHideInput() {
|
|
313
|
-
if (this.options.isMultipleFile) {
|
|
314
|
-
if (this.options.maxNoOfFiles != null &&
|
|
315
|
-
this.options.maxNoOfFiles > 0 &&
|
|
316
|
-
this.options.maxNoOfFiles == this.uploader.queue.length) {
|
|
317
|
-
return true;
|
|
318
|
-
}
|
|
319
|
-
}
|
|
320
|
-
else {
|
|
321
|
-
if (this.uploader.queue.length > 0) {
|
|
322
|
-
return true;
|
|
323
|
-
}
|
|
324
|
-
}
|
|
325
|
-
return false;
|
|
326
|
-
}
|
|
327
|
-
onFileChange() {
|
|
328
|
-
let FilesArray = [];
|
|
329
|
-
// Clear any previous validation errors at the start of file change
|
|
330
|
-
this.pendingValidationErrors = null;
|
|
331
|
-
// Check for duplicate file names in the queue
|
|
332
|
-
const duplicateFiles = this.findDuplicateFiles();
|
|
333
|
-
if (duplicateFiles.length > 0) {
|
|
334
|
-
// Remove only the newly added duplicate files from queue (keep existing files)
|
|
335
|
-
this.removeDuplicateFilesFromQueue(duplicateFiles);
|
|
336
|
-
// Set validation error for duplicate files
|
|
337
|
-
const duplicateFileNames = duplicateFiles.map(f => f.file?.name).join(', ');
|
|
338
|
-
const formControl = this.fileUploadFormControl;
|
|
339
|
-
const resourceValue = this.UtilityService.getResourceValue('DuplicateFileError');
|
|
340
|
-
// If resource value is the same as the key, it means the resource wasn't found
|
|
341
|
-
const errorMessage = (resourceValue && resourceValue !== 'DuplicateFileError') ? resourceValue : 'File(s) already exist';
|
|
342
|
-
const errorObj = {
|
|
343
|
-
DuplicateFileError: `${errorMessage}: ${duplicateFileNames}`
|
|
344
|
-
};
|
|
345
|
-
this.pendingValidationErrors = errorObj;
|
|
346
|
-
formControl.setErrors(errorObj);
|
|
347
|
-
formControl.markAsTouched();
|
|
348
|
-
// Auto-clear duplicate error after 5 seconds to allow form submission
|
|
349
|
-
setTimeout(() => {
|
|
350
|
-
// Only clear if the current error is still the duplicate error we set
|
|
351
|
-
const currentErrors = formControl.errors;
|
|
352
|
-
if (currentErrors && currentErrors.DuplicateFileError === errorObj.DuplicateFileError) {
|
|
353
|
-
formControl.setErrors(null);
|
|
354
|
-
this.pendingValidationErrors = null;
|
|
355
|
-
}
|
|
356
|
-
}, 5000);
|
|
357
|
-
// If all files were duplicates, return early
|
|
358
|
-
if (this.uploader.queue.filter(item => item['some'].lastModified != null).length === 0) {
|
|
359
|
-
return;
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
if (this.options.isMultipleFile && this.options.minNoOfFiles != null && this.options.minNoOfFiles > 0 && this.options.minNoOfFiles > this.uploader.queue.length) {
|
|
363
|
-
const formControl = this.fileUploadFormControl;
|
|
364
|
-
const existingErrors = formControl.errors || {};
|
|
365
|
-
const newErrors = {
|
|
366
|
-
...existingErrors,
|
|
367
|
-
MinFileCountValidationKey: this.options.minNoOfFiles
|
|
368
|
-
};
|
|
369
|
-
formControl.setErrors(newErrors);
|
|
370
|
-
formControl.markAsTouched();
|
|
371
|
-
this.uploader.queue = [];
|
|
372
|
-
return;
|
|
373
|
-
}
|
|
374
|
-
if (this.options.isMultipleFile && this.options.maxNoOfFiles != null && this.options.maxNoOfFiles > 0 && this.options.maxNoOfFiles < this.uploader.queue.length) {
|
|
375
|
-
const formControl = this.fileUploadFormControl;
|
|
376
|
-
const existingErrors = formControl.errors || {};
|
|
377
|
-
const newErrors = {
|
|
378
|
-
...existingErrors,
|
|
379
|
-
MaxFileCountValidationKey: this.options.maxNoOfFiles
|
|
380
|
-
};
|
|
381
|
-
formControl.setErrors(newErrors);
|
|
382
|
-
formControl.markAsTouched();
|
|
383
|
-
this.uploader.queue = [];
|
|
384
|
-
return;
|
|
385
|
-
}
|
|
386
|
-
if (this.options.isMultipleFile && this.options.maxSizeForAllFilesInMB != null && this.options.maxSizeForAllFilesInMB > 0) {
|
|
387
|
-
let AllSizeFile = 0;
|
|
388
|
-
for (let index = 0; index < this.uploader.queue.length; index++) {
|
|
389
|
-
const element = this.uploader.queue[index];
|
|
390
|
-
const file = element.file;
|
|
391
|
-
AllSizeFile = AllSizeFile + file.size;
|
|
392
|
-
}
|
|
393
|
-
const MaxSizeForAllFiles = this.options.maxSizeForAllFilesInMB * 1000 * 1000;
|
|
394
|
-
if (AllSizeFile > MaxSizeForAllFiles) {
|
|
395
|
-
const formControl = this.fileUploadFormControl;
|
|
396
|
-
const existingErrors = formControl.errors || {};
|
|
397
|
-
const newErrors = {
|
|
398
|
-
...existingErrors,
|
|
399
|
-
MaxSizeForAllFilesInMB: this.options.maxSizeForAllFilesInMB + ' ' + this.UtilityService.getResourceValue('MB')
|
|
400
|
-
};
|
|
401
|
-
formControl.setErrors(newErrors);
|
|
402
|
-
formControl.markAsTouched();
|
|
403
|
-
this.uploader.queue = [];
|
|
404
|
-
return;
|
|
405
|
-
}
|
|
406
|
-
}
|
|
407
|
-
if (!this.options.isMultipleFile && this.uploader.queue.length > 1) {
|
|
408
|
-
const formControl = this.fileUploadFormControl;
|
|
409
|
-
const newErrors = {
|
|
410
|
-
MaxFileCountValidationKey: 1
|
|
411
|
-
};
|
|
412
|
-
formControl.setErrors(newErrors);
|
|
413
|
-
formControl.markAsTouched();
|
|
414
|
-
this.uploader.queue = [];
|
|
415
|
-
return;
|
|
416
|
-
}
|
|
417
|
-
let AddedQueue = this.uploader.queue.filter((obj) => obj['some'].lastModified != null);
|
|
418
|
-
let hasValidationError = false;
|
|
419
|
-
let validationErrorType = '';
|
|
420
|
-
let validationErrorValue = '';
|
|
421
|
-
for (let index = 0; index < AddedQueue.length; index++) {
|
|
422
|
-
const element = AddedQueue[index];
|
|
423
|
-
const file = element.file;
|
|
424
|
-
const maxFileSize = this.options.fileMaxSizeInMB * 1000 * 1000;
|
|
425
|
-
if (file) {
|
|
426
|
-
const fileType = file.type;
|
|
427
|
-
if (file.size > maxFileSize) {
|
|
428
|
-
// Remove only the file that violates the size limit
|
|
429
|
-
this.uploader.queue = this.uploader.queue.filter(queueItem => queueItem.file.name !== file.name);
|
|
430
|
-
hasValidationError = true;
|
|
431
|
-
validationErrorType = 'FileMaxSizeInMB';
|
|
432
|
-
validationErrorValue = this.options.fileMaxSizeInMB + ' ' + this.UtilityService.getResourceValue('MB');
|
|
433
|
-
// Continue checking other files instead of returning immediately
|
|
434
|
-
}
|
|
435
|
-
if (this.options.fileUploadAcceptsTypes != null &&
|
|
436
|
-
this.options.fileUploadAcceptsTypes.length > 0 &&
|
|
437
|
-
!this.isFileTypeAccepted(fileType)) {
|
|
438
|
-
// Remove only the file that violates the file type restriction
|
|
439
|
-
this.uploader.queue = this.uploader.queue.filter(queueItem => queueItem.file.name !== file.name);
|
|
440
|
-
hasValidationError = true;
|
|
441
|
-
validationErrorType = 'ToolTipTypeError';
|
|
442
|
-
validationErrorValue = this.toolTipTypeArray;
|
|
443
|
-
// Continue checking other files instead of returning immediately
|
|
444
|
-
}
|
|
445
|
-
}
|
|
446
|
-
}
|
|
447
|
-
// Set validation error if any files were removed due to validation failures
|
|
448
|
-
this.pendingValidationErrors = null;
|
|
449
|
-
if (hasValidationError) {
|
|
450
|
-
const formControl = this.fileUploadFormControl;
|
|
451
|
-
const errorObj = {};
|
|
452
|
-
errorObj[validationErrorType] = validationErrorValue;
|
|
453
|
-
this.pendingValidationErrors = errorObj; // Store errors to reapply later
|
|
454
|
-
formControl.setErrors(errorObj);
|
|
455
|
-
formControl.markAsTouched();
|
|
456
|
-
// Auto-clear FileMaxSizeInMB error after 5 seconds (similar to DuplicateFileError)
|
|
457
|
-
if (validationErrorType === 'FileMaxSizeInMB') {
|
|
458
|
-
setTimeout(() => {
|
|
459
|
-
// Only clear if the current error is still the FileMaxSizeInMB error we set
|
|
460
|
-
const currentErrors = formControl.errors;
|
|
461
|
-
if (currentErrors && currentErrors['FileMaxSizeInMB'] === errorObj['FileMaxSizeInMB']) {
|
|
462
|
-
formControl.setErrors(null);
|
|
463
|
-
this.pendingValidationErrors = null;
|
|
464
|
-
}
|
|
465
|
-
}, 5000);
|
|
466
|
-
}
|
|
467
|
-
// Don't return here, continue processing valid files
|
|
468
|
-
}
|
|
469
|
-
// Re-filter the queue to get only valid files for processing
|
|
470
|
-
AddedQueue = this.uploader.queue.filter((obj) => obj['some'].lastModified != null);
|
|
471
|
-
for (let index = 0; index < AddedQueue.length; index++) {
|
|
472
|
-
const element = AddedQueue[index];
|
|
473
|
-
const file = element.file;
|
|
474
|
-
if (file) {
|
|
475
|
-
if (this.options.isUploadFileAsync && !element._file['iD_GUID']) {
|
|
476
|
-
this.fileUploadService.uploadFile(element._file)
|
|
477
|
-
.pipe(takeUntil(this.destroy$))
|
|
478
|
-
.subscribe({
|
|
479
|
-
next: (event) => {
|
|
480
|
-
let queueIndex = this.uploader.queue.findIndex((file) => file == element);
|
|
481
|
-
if (event.type === HttpEventType.UploadProgress) {
|
|
482
|
-
let value = Math.round((100 * event.loaded) / event.total);
|
|
483
|
-
this.uploader.queue[queueIndex].progress = value >= 95 ? 95 : value;
|
|
484
|
-
}
|
|
485
|
-
else if (event.type === HttpEventType.Response) {
|
|
486
|
-
this.uploader.queue[queueIndex].progress = 100;
|
|
487
|
-
let fileID = event.body.val;
|
|
488
|
-
this.uploader.queue[queueIndex]._file['iD_GUID'] = fileID;
|
|
489
|
-
this.uploader.queue[queueIndex]._file['isNew'] = true;
|
|
490
|
-
let AddedFile = {
|
|
491
|
-
iD_GUID: fileID,
|
|
492
|
-
fileName: this.uploader.queue[queueIndex]._file['name'],
|
|
493
|
-
fileType: this.uploader.queue[queueIndex]._file['type'],
|
|
494
|
-
isNew: true
|
|
495
|
-
};
|
|
496
|
-
if (this.options.isMultipleFile == false) {
|
|
497
|
-
this.fileUploadModel = new FileUploadModel();
|
|
498
|
-
this.fileUploadModel.file = AddedFile;
|
|
499
|
-
this.setValuePreservingErrors(this.fileUploadModel, this.pendingValidationErrors);
|
|
500
|
-
}
|
|
501
|
-
else {
|
|
502
|
-
FilesArray.push(AddedFile);
|
|
503
|
-
this.multipleFileUploadModel.uploadedFiles = FilesArray;
|
|
504
|
-
if (this.options.value != null && this.options.value != undefined) {
|
|
505
|
-
if (this.options.value.correlationID_GUID == null) {
|
|
506
|
-
this.multipleFileUploadModel.removedFiles = [];
|
|
507
|
-
}
|
|
508
|
-
}
|
|
509
|
-
this.multipleFileUploadModel.correlationID_GUID =
|
|
510
|
-
this.options.value?.correlationID_GUID;
|
|
511
|
-
this.setValuePreservingErrors(this.multipleFileUploadModel, this.pendingValidationErrors);
|
|
512
|
-
this.isUploadComplete.emit(true);
|
|
513
|
-
}
|
|
514
|
-
}
|
|
515
|
-
}
|
|
516
|
-
});
|
|
517
|
-
}
|
|
518
|
-
else {
|
|
519
|
-
let reader = new FileReader();
|
|
520
|
-
let fileObject = file.rawFile;
|
|
521
|
-
reader.readAsDataURL(fileObject);
|
|
522
|
-
reader.onload = () => {
|
|
523
|
-
let existingID_GUID = null;
|
|
524
|
-
if (!this.options.isMultipleFile && this.file)
|
|
525
|
-
existingID_GUID = this.file.NameWithExtension == file.name ? this.file.iD_GUID : null;
|
|
526
|
-
let AddedFile = {
|
|
527
|
-
fileName: file.name,
|
|
528
|
-
fileType: file.type,
|
|
529
|
-
fileBase64: reader.result.toString().split(',')[1],
|
|
530
|
-
fileSizeInMB: file.size / 1000 / 1000,
|
|
531
|
-
nameWithExtension: file.name,
|
|
532
|
-
iD_GUID: existingID_GUID,
|
|
533
|
-
isNew: true
|
|
534
|
-
};
|
|
535
|
-
if (this.options.isMultipleFile == false) {
|
|
536
|
-
this.fileUploadModel = new FileUploadModel();
|
|
537
|
-
this.fileUploadModel.file = AddedFile;
|
|
538
|
-
this.setValuePreservingErrors(this.fileUploadModel, this.pendingValidationErrors);
|
|
539
|
-
}
|
|
540
|
-
else {
|
|
541
|
-
FilesArray.push(AddedFile);
|
|
542
|
-
this.multipleFileUploadModel.uploadedFiles = FilesArray;
|
|
543
|
-
if (this.options.value != null && this.options.value != undefined) {
|
|
544
|
-
if (this.options.value.correlationID_GUID == null) {
|
|
545
|
-
this.multipleFileUploadModel.removedFiles = [];
|
|
546
|
-
}
|
|
547
|
-
}
|
|
548
|
-
this.multipleFileUploadModel.correlationID_GUID =
|
|
549
|
-
this.options.value?.correlationID_GUID;
|
|
550
|
-
this.setValuePreservingErrors(this.multipleFileUploadModel, this.pendingValidationErrors);
|
|
551
|
-
}
|
|
552
|
-
};
|
|
553
|
-
let originalValue = this.group.get(this.options.name).value;
|
|
554
|
-
if (this.options.patchFunction &&
|
|
555
|
-
this.options.patchPath &&
|
|
556
|
-
this.group.get(this.options.name).valid) {
|
|
557
|
-
this.controlUtility.patchControlValue(originalValue, this.options.patchFunction, this.options.patchPath);
|
|
558
|
-
}
|
|
559
|
-
this.OnChange.emit(originalValue);
|
|
560
|
-
}
|
|
561
|
-
}
|
|
562
|
-
}
|
|
563
|
-
}
|
|
564
|
-
removeFromControlValue(item) {
|
|
565
|
-
if (this.options.isUploadFileAsync && item.progress == 100 && item._file['isNew']) {
|
|
566
|
-
this.fileUploadService.deleteFile(item._file['iD_GUID'])
|
|
567
|
-
.pipe(takeUntil(this.destroy$))
|
|
568
|
-
.subscribe((res) => { });
|
|
569
|
-
}
|
|
570
|
-
if (this.options.isMultipleFile == false) {
|
|
571
|
-
this.fileUploadModel = null;
|
|
572
|
-
if (this.options.isRequired == true) {
|
|
573
|
-
this.fileUploadFormControl.markAsTouched();
|
|
574
|
-
this.fileUploadFormControl.updateValueAndValidity(); // Trigger validation check
|
|
575
|
-
}
|
|
576
|
-
this.group.get(this.options.name).setValue(this.fileUploadModel);
|
|
577
|
-
this.options.value = this.fileUploadModel;
|
|
578
|
-
}
|
|
579
|
-
else {
|
|
580
|
-
if (this.options.value != null && this.options.value != undefined) {
|
|
581
|
-
if (this.options.value.correlationID_GUID == null) {
|
|
582
|
-
this.deletedFiles = [];
|
|
583
|
-
this.multipleFileUploadModel.removedFiles = [];
|
|
584
|
-
}
|
|
585
|
-
else {
|
|
586
|
-
if (this.multipleFileUploadModel.removedFiles.length == 0) {
|
|
587
|
-
let FileObject = item.file.rawFile;
|
|
588
|
-
let DeletedItem = this.multipleFileUploadModel.existingFiles.filter((obj) => obj.nameWithExtension == FileObject.name)[0];
|
|
589
|
-
this.multipleFileUploadModel.existingFiles =
|
|
590
|
-
this.multipleFileUploadModel.existingFiles.filter((obj) => obj.nameWithExtension != FileObject.name);
|
|
591
|
-
this.deletedFiles.push(DeletedItem);
|
|
592
|
-
this.multipleFileUploadModel.removedFiles.push(DeletedItem.iD_GUID);
|
|
593
|
-
}
|
|
594
|
-
else {
|
|
595
|
-
let FileObject = item.file.rawFile;
|
|
596
|
-
let deletedList = this.deletedFiles.filter((obj) => obj.nameWithExtension == FileObject.name);
|
|
597
|
-
if (deletedList.length == 0 || deletedList == undefined) {
|
|
598
|
-
let DeletedItem = this.multipleFileUploadModel.existingFiles.filter((obj) => obj.nameWithExtension == FileObject.name)[0];
|
|
599
|
-
this.multipleFileUploadModel.existingFiles =
|
|
600
|
-
this.multipleFileUploadModel.existingFiles.filter((obj) => obj.nameWithExtension != FileObject.name);
|
|
601
|
-
this.deletedFiles.push(DeletedItem);
|
|
602
|
-
this.multipleFileUploadModel.removedFiles.push(DeletedItem.iD_GUID);
|
|
603
|
-
}
|
|
604
|
-
}
|
|
605
|
-
}
|
|
606
|
-
}
|
|
607
|
-
else {
|
|
608
|
-
this.deletedFiles = [];
|
|
609
|
-
this.multipleFileUploadModel.removedFiles = [];
|
|
610
|
-
}
|
|
611
|
-
this.multipleFileUploadModel.uploadedFiles =
|
|
612
|
-
this.multipleFileUploadModel.uploadedFiles.filter((obj) => (obj.nameWithExtension && obj.nameWithExtension != item._file.name) ||
|
|
613
|
-
obj.iD_GUID != item._file['iD_GUID']);
|
|
614
|
-
if ((this.multipleFileUploadModel.uploadedFiles == null ||
|
|
615
|
-
this.multipleFileUploadModel.uploadedFiles.length == 0) &&
|
|
616
|
-
this.options.isRequired) {
|
|
617
|
-
const formControl = this.fileUploadFormControl;
|
|
618
|
-
formControl.setErrors({
|
|
619
|
-
MinFileCountValidationKey: this.options.minNoOfFiles
|
|
620
|
-
});
|
|
621
|
-
this.fileUploadFormControl.markAsTouched();
|
|
622
|
-
this.fileUploadFormControl.updateValueAndValidity(); // Trigger validation check
|
|
623
|
-
}
|
|
624
|
-
this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;
|
|
625
|
-
this.fileUploadFormControl.setValue(this.multipleFileUploadModel);
|
|
626
|
-
this.group.get(this.options.name).setValue(this.multipleFileUploadModel);
|
|
627
|
-
this.fileUploadFormControl.updateValueAndValidity();
|
|
628
|
-
//Use this line to enable two way binding.
|
|
629
|
-
this.options.value = this.multipleFileUploadModel;
|
|
630
|
-
}
|
|
631
|
-
}
|
|
632
|
-
convertSizeToMB(size) {
|
|
633
|
-
if (size === 0) {
|
|
634
|
-
return 0;
|
|
635
|
-
}
|
|
636
|
-
// Convert size to megabytes
|
|
637
|
-
const megabytes = size / (1024 * 1024);
|
|
638
|
-
// Round to two decimal places
|
|
639
|
-
const roundedMegabytes = Math.round(megabytes * 100) / 100;
|
|
640
|
-
return roundedMegabytes;
|
|
641
|
-
}
|
|
642
|
-
ngOnDestroy() {
|
|
643
|
-
// Complete all subscriptions to prevent memory leaks
|
|
644
|
-
this.destroy$.next();
|
|
645
|
-
this.destroy$.complete();
|
|
646
|
-
// Clear uploader queue to free memory
|
|
647
|
-
if (this.uploader) {
|
|
648
|
-
this.uploader.clearQueue();
|
|
649
|
-
}
|
|
650
|
-
}
|
|
651
|
-
sanitizeFileName(fileName) {
|
|
652
|
-
// Remove potentially dangerous characters from file names
|
|
653
|
-
return fileName.replace(/[<>:"/\\|?*]/g, '_').trim();
|
|
654
|
-
}
|
|
655
|
-
findDuplicateFiles() {
|
|
656
|
-
const duplicateFiles = [];
|
|
657
|
-
const existingFileNames = new Set();
|
|
658
|
-
// First pass: collect names of existing files (already in queue before this upload)
|
|
659
|
-
const existingFiles = this.uploader.queue.filter(item => item['some'].lastModified == null);
|
|
660
|
-
for (const item of existingFiles) {
|
|
661
|
-
const fileName = item.file.name.toLowerCase().trim();
|
|
662
|
-
existingFileNames.add(fileName);
|
|
663
|
-
}
|
|
664
|
-
// Second pass: check newly added files against existing file names
|
|
665
|
-
const newlyAddedFiles = this.uploader.queue.filter(item => item['some'].lastModified != null);
|
|
666
|
-
for (const item of newlyAddedFiles) {
|
|
667
|
-
const fileName = item.file.name.toLowerCase().trim();
|
|
668
|
-
// If this new file name already exists in the queue, it's a duplicate
|
|
669
|
-
if (existingFileNames.has(fileName)) {
|
|
670
|
-
duplicateFiles.push(item);
|
|
671
|
-
}
|
|
672
|
-
else {
|
|
673
|
-
// Add this new file name to the set so subsequent files with same name are also flagged
|
|
674
|
-
existingFileNames.add(fileName);
|
|
675
|
-
}
|
|
676
|
-
}
|
|
677
|
-
return duplicateFiles;
|
|
678
|
-
}
|
|
679
|
-
removeDuplicateFilesFromQueue(duplicateFiles) {
|
|
680
|
-
for (const duplicateFile of duplicateFiles) {
|
|
681
|
-
const index = this.uploader.queue.indexOf(duplicateFile);
|
|
682
|
-
if (index > -1) {
|
|
683
|
-
this.uploader.queue.splice(index, 1);
|
|
684
|
-
}
|
|
685
|
-
}
|
|
686
|
-
}
|
|
687
|
-
isFileTypeAccepted(fileType) {
|
|
688
|
-
if (!this.acceptedTypeArray || this.acceptedTypeArray.length === 0) {
|
|
689
|
-
return true; // No restrictions
|
|
690
|
-
}
|
|
691
|
-
// Normalize the file type for comparison
|
|
692
|
-
const normalizedFileType = fileType.toLowerCase().trim();
|
|
693
|
-
// Check against all accepted types (case-insensitive and trimmed)
|
|
694
|
-
return this.acceptedTypeArray.some(acceptedType => {
|
|
695
|
-
const normalizedAcceptedType = acceptedType.toLowerCase().trim();
|
|
696
|
-
return normalizedAcceptedType === normalizedFileType;
|
|
697
|
-
});
|
|
698
|
-
}
|
|
699
|
-
setValuePreservingErrors(value, preserveErrors) {
|
|
700
|
-
// Capture current duplicate errors before setValue clears them
|
|
701
|
-
const currentErrors = this.fileUploadFormControl.errors;
|
|
702
|
-
const duplicateError = currentErrors?.DuplicateFileError ? { DuplicateFileError: currentErrors.DuplicateFileError } : null;
|
|
703
|
-
// Set the form control value
|
|
704
|
-
this.fileUploadFormControl.setValue(value);
|
|
705
|
-
this.group.get(this.options.name).setValue(value);
|
|
706
|
-
// Apply errors in priority order: preserveErrors > duplicateError > pendingValidationErrors
|
|
707
|
-
const errorsToApply = preserveErrors || duplicateError || this.pendingValidationErrors;
|
|
708
|
-
if (errorsToApply) {
|
|
709
|
-
this.fileUploadFormControl.setErrors(errorsToApply);
|
|
710
|
-
this.fileUploadFormControl.markAsTouched();
|
|
711
|
-
}
|
|
712
|
-
// Update the options value for two-way binding
|
|
713
|
-
this.options.value = value;
|
|
714
|
-
}
|
|
715
|
-
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: i2.GlobalSettings }, { token: i2.FileUploadService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
716
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: FileUploadComponent, isStandalone: true, 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 <!--Asterisk-->\r\n <span *ngIf=\"((options.showAsterisk && options.isRequired) || options.isRequired) && !options.isReadonly\"\r\n class=\"text-danger\">*</span>\r\n </label>\r\n <!--Allow dropZone-->\r\n <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n *ngIf=\"options.isDropZone && !(options.isMultipleFile == false && uploader.queue.length > 0) && !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 multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" aria-invalid=\"true\" type=\"file\"\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n (click)=\"fileInputDropZone.click()\">\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\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 <!--Validation text-->\r\n <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n {{ UtilityService.getResourceValue('DragAndDropHere') }}\r\n </div>\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\"></div>\r\n </div>\r\n </div>\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n aria-invalid=\"true\" #fileInputDropZone\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!--Not allowed dropZone-->\r\n <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n (click)=\"fileInput.click()\">\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\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 <!--Validation text-->\r\n <div class=\"bbsf-validation-msg text-center\">{{ UtilityService.getResourceValue('Upload') }}</div>\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\"></div>\r\n </div>\r\n </div>\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n aria-invalid=\"true\" #fileInput\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- readonly -->\r\n <div *ngIf=\"options.isReadonly && !options.value\">\r\n <span class=\"readonly-view\">{{ UtilityService.getResourceValue('NA') }}</span>\r\n </div>\r\n </div>\r\n <!--items uploaded-->\r\n <div class=\"uploaded-items\">\r\n <div class=\"btn-group\" *ngFor=\"let item of uploader.queue\">\r\n <ng-container *ngIf=\"item?.progress == 100 && options.isUploadFileAsync\">\r\n <a *ngIf=\"item?.file?.rawFile['url']\" href=\"{{ item?.file?.rawFile['url'] }}\"\r\n class=\"btn-download-file btn-sm btn-progress-upload\" download>\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\">\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 <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n <a href=\"{{ item?.file?.rawFile['url'] }}\" *ngIf=\"item?.file?.rawFile['url']\" class=\"btn btn-download-file btn-sm\" download>\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\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm btn-danger\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\">\r\n <i class=\"fa fa-times px-0\"></i>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </div>\r\n <!--progress bar file upload-->\r\n <div *ngFor=\"let item of uploader.queue\">\r\n <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile == true }\"\r\n *ngIf=\"item?.progress < 100 && options.isUploadFileAsync\">\r\n <div class=\"upload-items-toolbar\">\r\n <h4>{{ item?.file?.name }}</h4>\r\n <span (click)=\"item.remove(); removeFromControlValue(item)\">\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 </span>\r\n </div>\r\n <div class=\"progress\">\r\n <div class=\"progress-bar\" role=\"progressbar\" aria-valuenow=\"70\" aria-valuemin=\"0\" aria-valuemax=\"100\"\r\n [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\" *ngIf=\"item?.progress > 0\">\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n <!-- required text-->\r\n <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n </div>\r\n <!-- LabelDescription-->\r\n <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription != null\">{{ options.labelDescription }}</div>\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: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i4.KeyValuePipe, name: "keyvalue" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FileUploadModule }, { kind: "directive", type: i5.FileDropDirective, selector: "[ng2FileDrop]", inputs: ["uploader"], outputs: ["fileOver", "onFileDrop"] }, { kind: "directive", type: i5.FileSelectDirective, selector: "[ng2FileSelect]", inputs: ["uploader"], outputs: ["onFileSelected"] }, { kind: "ngmodule", type: NgxDropzoneModule }] }); }
|
|
717
|
-
}
|
|
718
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FileUploadComponent, decorators: [{
|
|
719
|
-
type: Component,
|
|
720
|
-
args: [{ selector: 'BBSF-FileUpload', standalone: true, imports: [
|
|
721
|
-
CommonModule,
|
|
722
|
-
FormsModule,
|
|
723
|
-
ReactiveFormsModule,
|
|
724
|
-
FileUploadModule,
|
|
725
|
-
NgxDropzoneModule
|
|
726
|
-
], schemas: [CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA], 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 <!--Asterisk-->\r\n <span *ngIf=\"((options.showAsterisk && options.isRequired) || options.isRequired) && !options.isReadonly\"\r\n class=\"text-danger\">*</span>\r\n </label>\r\n <!--Allow dropZone-->\r\n <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n *ngIf=\"options.isDropZone && !(options.isMultipleFile == false && uploader.queue.length > 0) && !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 multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" aria-invalid=\"true\" type=\"file\"\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n (click)=\"fileInputDropZone.click()\">\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\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 <!--Validation text-->\r\n <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n {{ UtilityService.getResourceValue('DragAndDropHere') }}\r\n </div>\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\"></div>\r\n </div>\r\n </div>\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n aria-invalid=\"true\" #fileInputDropZone\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!--Not allowed dropZone-->\r\n <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n (click)=\"fileInput.click()\">\r\n <div class=\"dropzone-label\">\r\n <div class=\"svg-and-validation\">\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 <!--Validation text-->\r\n <div class=\"bbsf-validation-msg text-center\">{{ UtilityService.getResourceValue('Upload') }}</div>\r\n <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n [innerHTML]=\"validationCountMessage\"></div>\r\n </div>\r\n </div>\r\n <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n aria-invalid=\"true\" #fileInput\r\n [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n </div>\r\n <!-- readonly -->\r\n <div *ngIf=\"options.isReadonly && !options.value\">\r\n <span class=\"readonly-view\">{{ UtilityService.getResourceValue('NA') }}</span>\r\n </div>\r\n </div>\r\n <!--items uploaded-->\r\n <div class=\"uploaded-items\">\r\n <div class=\"btn-group\" *ngFor=\"let item of uploader.queue\">\r\n <ng-container *ngIf=\"item?.progress == 100 && options.isUploadFileAsync\">\r\n <a *ngIf=\"item?.file?.rawFile['url']\" href=\"{{ item?.file?.rawFile['url'] }}\"\r\n class=\"btn-download-file btn-sm btn-progress-upload\" download>\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\">\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 <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n <a href=\"{{ item?.file?.rawFile['url'] }}\" *ngIf=\"item?.file?.rawFile['url']\" class=\"btn btn-download-file btn-sm\" download>\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\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n </a>\r\n <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm btn-danger\"\r\n (click)=\"item.remove(); removeFromControlValue(item)\">\r\n <i class=\"fa fa-times px-0\"></i>\r\n </button>\r\n </ng-container>\r\n </div>\r\n </div>\r\n <!--progress bar file upload-->\r\n <div *ngFor=\"let item of uploader.queue\">\r\n <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile == true }\"\r\n *ngIf=\"item?.progress < 100 && options.isUploadFileAsync\">\r\n <div class=\"upload-items-toolbar\">\r\n <h4>{{ item?.file?.name }}</h4>\r\n <span (click)=\"item.remove(); removeFromControlValue(item)\">\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 </span>\r\n </div>\r\n <div class=\"progress\">\r\n <div class=\"progress-bar\" role=\"progressbar\" aria-valuenow=\"70\" aria-valuemin=\"0\" aria-valuemax=\"100\"\r\n [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\" *ngIf=\"item?.progress > 0\">\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n <!-- required text-->\r\n <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n </div>\r\n <!-- LabelDescription-->\r\n <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription != null\">{{ options.labelDescription }}</div>\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>" }]
|
|
727
|
-
}], ctorParameters: () => [{ type: i1.ControlContainer, decorators: [{
|
|
728
|
-
type: Optional
|
|
729
|
-
}] }, { type: i1.FormGroupDirective }, { type: i2.ControlUtility }, { type: i3.UtilityService }, { type: i3.ControlValidationService }, { type: i2.GlobalSettings }, { type: i2.FileUploadService }], propDecorators: { fileInput: [{
|
|
730
|
-
type: ViewChild,
|
|
731
|
-
args: ['fileInput', { static: false }]
|
|
732
|
-
}], group: [{
|
|
733
|
-
type: Input
|
|
734
|
-
}], options: [{
|
|
735
|
-
type: Input
|
|
736
|
-
}], OnChange: [{
|
|
737
|
-
type: Output
|
|
738
|
-
}], isUploadComplete: [{
|
|
739
|
-
type: Output
|
|
740
|
-
}] } });
|
|
741
|
-
//# 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":"AAAA,OAAO,EAAE,SAAS,EAAyB,KAAK,EAAE,QAAQ,EAAE,SAAS,EAAc,MAAM,EAAE,YAAY,EAAE,sBAAsB,EAAE,gBAAgB,EAAa,MAAM,eAAe,CAAC;AACpL,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,WAAW,EAAE,UAAU,EAAoE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC7J,OAAO,EAAE,YAAY,EAAiD,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAChH,OAAO,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC;AAEjD,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAC3C,OAAO,EAAW,eAAe,EAAqB,uBAAuB,EAAE,MAAM,qBAAqB,CAAC;;;;;;;AAgB3G,MAAM,OAAO,mBAAmB;aACvB,2BAAsB,GAAG,IAAI,AAAP,CAAQ;IA6BrC,YACsB,gBAAkC,EAC/C,6BAAiD,EAChD,cAA8B,EAC/B,cAA8B,EAC7B,wBAAkD,EAClD,cAA8B,EAC9B,iBAAoC;QANxB,qBAAgB,GAAhB,gBAAgB,CAAkB;QAC/C,kCAA6B,GAA7B,6BAA6B,CAAoB;QAChD,mBAAc,GAAd,cAAc,CAAgB;QAC/B,mBAAc,GAAd,cAAc,CAAgB;QAC7B,6BAAwB,GAAxB,wBAAwB,CAA0B;QAClD,mBAAc,GAAd,cAAc,CAAgB;QAC9B,sBAAiB,GAAjB,iBAAiB,CAAmB;QA7BpC,aAAQ,GAAG,IAAI,YAAY,EAAE,CAAC;QAC9B,qBAAgB,GAAG,IAAI,YAAY,EAAE,CAAC;QAGhD,sBAAiB,GAAG,EAAE,CAAC;QACvB,2BAAsB,GAAG,EAAE,CAAC;QAG5B,iBAAY,GAAW,EAAE,CAAC;QAC1B,sBAAiB,GAAa,EAAE,CAAC;QACjC,qBAAgB,GAAa,EAAE,CAAC;QAEhC,qBAAgB,GAAY,KAAK,CAAC;QAClC,oBAAe,GAAG,EAAE,CAAC;QACrB,yBAAoB,GAAG,EAAE,CAAC;QAI1B,iBAAY,GAAc,EAAE,CAAC;QACrB,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAC/B,4BAAuB,GAAQ,IAAI,CAAC;QAoP5C,eAAU,GAAG,GAAG,EAAE;YAChB,IAAI,CAAC,wBAAwB,CAAC,iBAAiB,EAAE,CAAC;QACpD,CAAC,CAAC;QA2XF,iBAAiB;QACjB,6BAAwB,GAAG,GAAG,EAAE;YAC9B,IAAI,CAAC,cAAc,CAAC,wBAAwB,CAC1C,IAAI,CAAC,qBAAqB,EAC1B,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,OAAO,CACb,CAAC;QACJ,CAAC,CAAC;QAEF,iBAAiB;QACjB,0BAAqB,GAAG,GAAG,EAAE;YAC3B,IAAI,CAAC,cAAc,CAAC,qBAAqB,CACvC,IAAI,CAAC,qBAAqB,EAC1B,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,OAAO,CACb,CAAC;QACJ,CAAC,CAAC;QAEF,iBAAiB;QACjB,2BAAsB,GAAG,CAAC,gBAAgB,EAAE,EAAE;YAC5C,IAAI,CAAC,cAAc,CAAC,sBAAsB,CACxC,IAAI,CAAC,qBAAqB,EAC1B,IAAI,CAAC,eAAe,EACpB,gBAAgB,CACjB,CAAC;QACJ,CAAC,CAAC;QAEF,iBAAiB;QACjB,wBAAmB,GAAG,CAAC,gBAAgB,EAAE,EAAE;YACzC,IAAI,CAAC,cAAc,CAAC,mBAAmB,CACrC,IAAI,CAAC,qBAAqB,EAC1B,IAAI,CAAC,eAAe,EACpB,gBAAgB,CACjB,CAAC;QACJ,CAAC,CAAC;QAEF,iBAAiB;QACjB,YAAO,GAAG,GAAG,EAAE;YACb,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;QAC1D,CAAC,CAAC;QA7oBA,mBAAmB,CAAC,sBAAsB,GAAG,IAAI,CAAC,gBAAgB,CAAC;QACnE,uFAAuF;QACvF,IAAI,CAAC,QAAQ,GAAG,IAAI,YAAY,CAAC;YAC/B,GAAG,EAAE,EAAE;YACP,gBAAgB,EAAE,KAAK,CAAC,yEAAyE;SAC3E,CAAC,CAAC;QAC1B,IAAI,CAAC,sBAAsB,GAAG,KAAK,CAAC;IACtC,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,EAAE,CAAC;QAC7C,IAAI,CAAC,uBAAuB,GAAG,IAAI,uBAAuB,EAAE,CAAC;QAC7D,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ;YAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC;QAEjF,2FAA2F;QAC3F,8FAA8F;QAC9F,0DAA0D;QAC1D,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC;YACvB,GAAG,EAAE,EAAE;YACP,gBAAgB,EAAE,KAAK;YACvB,UAAU,EAAE,KAAK,CAAC,8BAA8B;YAChD,6EAA6E;SAC9E,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,IAAI,EAAE;YAC9B,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,EAAE;gBACvC,iDAAiD;aAClD;iBAAM;gBACL,0CAA0C;aAC3C;SACF;aAAM;YACL,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,EAAE;gBACvC,IAAI,KAAK,GAAU,EAAE,CAAC;gBACtB,IAAI,CAAC,uBAAuB,CAAC,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC;gBAC9E,IAAI,CAAC,uBAAuB,CAAC,aAAa,GAAG,EAAE,CAAC;gBAChD,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;oBAC5E,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;oBACxD,IAAI,KAAK,GAAG,IAAI,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;oBAC1C,IAAI,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;oBAC7D,IAAI,CAAC,cAAc,GAAmB;wBACpC,IAAI,EAAE,OAAO,CAAC,iBAAiB;wBAC/B,IAAI,EAAE,OAAO,CAAC,QAAQ;wBACtB,OAAO,EAAO,MAAM;qBACrB,CAAC;oBACF,iBAAiB;oBACjB,gCAAgC;oBAChC,gCAAgC;oBAChC,gCAAgC;oBAChC,gCAAgC;oBAChC,IAAI,IAAI,GAAG,IAAI,CAAC,cAAqB,CAAC;oBACtC,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,WAAW,CAAC;oBAC/B,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;iBAClB;gBACD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;gBAChC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;aAClC;iBAAM;gBACL,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;gBAC9D,IAAI,KAAK,GAAG,IAAI,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;gBAC1C,IAAI,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC7D,IAAI,CAAC,cAAc,GAAmB;oBACpC,IAAI,EAAE,OAAO,CAAC,iBAAiB;oBAC/B,IAAI,EAAE,OAAO,CAAC,QAAQ;oBACtB,OAAO,EAAO,MAAM;iBACrB,CAAC;gBACF,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC;gBACpB,IAAI,IAAI,GAAG,IAAI,CAAC,cAAqB,CAAC;gBACtC,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,WAAW,CAAC;gBAC/B,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;gBACjC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,EAAE;oBAC5B,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,EAAE,CAAC;oBAC7C,IAAI,CAAC,eAAe,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;iBAChD;;oBAAM,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;gBACjD,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,eAAe,CAAC;aAC3C;YACD,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;gBACtC,OAAO,CAAC,QAAQ,GAAG,GAAG,CAAC;YACzB,CAAC,CAAC,CAAC;SACJ;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,EAAE;YAC9D,IAAI,CAAC,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAExF,IACE,IAAI,CAAC,OAAO,CAAC,sBAAsB,IAAI,IAAI;YAC3C,IAAI,CAAC,OAAO,CAAC,sBAAsB,CAAC,MAAM,GAAG,CAAC,EAC9C;YACA,4EAA4E;YAC5E,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,sBAAsB,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBAC/E,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC;gBACxD,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,GAAG,IAAI,GAAG,GAAG,CAAC;aACpD;YACD,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACtD,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;YAEtF,6EAA6E;YAC7E,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACrD,iEAAiE;YACjE,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBAClE,MAAM,OAAO,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;gBACrD,QAAQ,OAAO,EAAE;oBACf,KAAK,iBAAiB;wBACpB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,oBAAoB;wBACvB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,MAAM,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;wBAChF,MAAM;oBACR,KAAK,yEAAyE;wBAC5E,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,MAAM,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;wBAChF,MAAM;oBACR,KAAK,0BAA0B;wBAC7B,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,OAAO,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;wBAClF,MAAM;oBACR,KAAK,mEAAmE;wBACtE,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,OAAO,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;wBAClF,MAAM;oBACR,KAAK,+BAA+B;wBAClC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,YAAY,CAAC;4BAC/C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;wBAC3C,MAAM;oBACR,KAAK,2EAA2E;wBAC9E,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,YAAY,CAAC;4BAC/C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;wBAC3C,MAAM;oBACR,KAAK,WAAW;wBACd,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,WAAW;wBACd,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,YAAY;wBACf,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,MAAM,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;wBAChF,MAAM;oBACR,KAAK,iBAAiB;wBACpB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,8BAA8B;wBACjC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,WAAW;wBACd,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,WAAW;wBACd,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,iBAAiB;wBACpB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,YAAY;wBACf,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,MAAM,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;wBAChF,MAAM;oBACR,KAAK,YAAY;wBACf,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,aAAa;wBAChB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,gBAAgB;wBACnB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,eAAe;wBAClB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,YAAY;wBACf,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;wBAC9E,MAAM;oBACR,KAAK,gBAAgB;wBACnB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,SAAS,CAAC;4BAAE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;wBACtF,MAAM;oBACR;wBACE,MAAM;iBACT;aACF;YACD,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,YAAY,CAAC,KAAK,IAAI,CAAC,gBAAgB,IAAI,CAAC;SACvI;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,eAAe,GAAG,CAAC,EAAE;YACpC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,GAAG,WAAW,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,iBAAiB,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,eAAe,GAAG,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,EAAE,CAAC;SAC1M;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC,EAAE;YACjC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,GAAG,UAAU,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,2BAA2B,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;SAC7J;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC,EAAE;YACjC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,GAAG,UAAU,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,2BAA2B,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;SAC7J;QAED,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,IAAI,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9D,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAEpE,IAAI,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE;YAC5C,IAAI,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC;YAChD,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,WAAW,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBACvD,MAAM,UAAU,GAAG,WAAW,CAAC,KAAK,CAAC,CAAC;gBACtC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;aACpD;SACF;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC,EAAE;YAChE,IAAI,CAAC,sBAAsB,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,eAAe,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;SACzH;QACD,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE;YAC3B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;SAChD;QAED,IAAI,CAAC,qBAAqB,CAAC,aAAa,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAC/D,IAAI,CAAC,qBAAqB,CAAC,kBAAkB,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;QACzE,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE;YAC3B,IAAI,CAAC,qBAAqB,CAAC,OAAO,EAAE,CAAC;SACtC;QAED,IAAI,CAAC,6BAA6B,CAAC,QAAQ;aACxC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,IAAI,CAAC,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAC9B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC/B,CAAC,CAAC,CAAC;QAEL,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IAC1D,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,IAAI,IAAI,EAAE;YACtC,IAAI,OAAO,GAAG,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YACzD,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBACtE,OAAO,CAAC,YAAY,CAClB,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,GAAG,EACrC,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,KAAK,CACxC,CAAC;aACH;SACF;IACH,CAAC;IAMD,eAAe;QACb,IAAI,CAAC,cAAc,CAAC,eAAe,EAAE,CAAC;IACxC,CAAC;IAED,kBAAkB,CAAC,SAAS;QAC1B,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE;YAC/C,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;SAC/B;QAED,OAAO,IAAI,CAAC,cAAc,CAAC,yBAAyB,CAAC,SAAS,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IAC5F,CAAC;IAED,eAAe,CAAC,KAAU;QACxB,IAAI,CAAC,sBAAsB,GAAG,KAAK,CAAC;IACtC,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE;YAC/B,IACE,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI;gBACjC,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC;gBAC7B,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,EACvD;gBACA,OAAO,IAAI,CAAC;aACb;SACF;aAAM;YACL,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;gBAClC,OAAO,IAAI,CAAC;aACb;SACF;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,YAAY;QACV,IAAI,UAAU,GAAc,EAAE,CAAC;QAC/B,mEAAmE;QACnE,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC;QAEpC,8CAA8C;QAC9C,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QACjD,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;YAC7B,+EAA+E;YAC/E,IAAI,CAAC,6BAA6B,CAAC,cAAc,CAAC,CAAC;YAEnD,2CAA2C;YAC3C,MAAM,kBAAkB,GAAG,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC5E,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;YAC/C,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,oBAAoB,CAAC,CAAC;YACjF,+EAA+E;YAC/E,MAAM,YAAY,GAAG,CAAC,aAAa,IAAI,aAAa,KAAK,oBAAoB,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,uBAAuB,CAAC;YACzH,MAAM,QAAQ,GAAG;gBACf,kBAAkB,EAAE,GAAG,YAAY,KAAK,kBAAkB,EAAE;aAC7D,CAAC;YACF,IAAI,CAAC,uBAAuB,GAAG,QAAQ,CAAC;YACxC,WAAW,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;YAChC,WAAW,CAAC,aAAa,EAAE,CAAC;YAE5B,sEAAsE;YACtE,UAAU,CAAC,GAAG,EAAE;gBACd,sEAAsE;gBACtE,MAAM,aAAa,GAAG,WAAW,CAAC,MAAM,CAAC;gBACzC,IAAI,aAAa,IAAI,aAAa,CAAC,kBAAkB,KAAK,QAAQ,CAAC,kBAAkB,EAAE;oBACrF,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;oBAC5B,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC;iBACrC;YACH,CAAC,EAAE,IAAI,CAAC,CAAC;YAET,6CAA6C;YAC7C,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE;gBACtF,OAAO;aACR;SACF;QACD,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,EAAE;YAC/J,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;YAC/C,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,IAAI,EAAE,CAAC;YAChD,MAAM,SAAS,GAAG;gBAChB,GAAG,cAAc;gBACjB,yBAAyB,EAAE,IAAI,CAAC,OAAO,CAAC,YAAY;aACrD,CAAC;YACF,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YACjC,WAAW,CAAC,aAAa,EAAE,CAAC;YAC5B,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,EAAE,CAAC;YACzB,OAAO;SACR;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,EAAE;YAC/J,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;YAC/C,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,IAAI,EAAE,CAAC;YAChD,MAAM,SAAS,GAAG;gBAChB,GAAG,cAAc;gBACjB,yBAAyB,EAAE,IAAI,CAAC,OAAO,CAAC,YAAY;aACrD,CAAC;YACF,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YACjC,WAAW,CAAC,aAAa,EAAE,CAAC;YAC5B,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,EAAE,CAAC;YACzB,OAAO;SACR;QAGD,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,OAAO,CAAC,sBAAsB,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,sBAAsB,GAAG,CAAC,EAAE;YACzH,IAAI,WAAW,GAAG,CAAC,CAAC;YACpB,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;gBAC/D,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;gBAC1B,WAAW,GAAG,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC;aACvC;YACD,MAAM,kBAAkB,GAAG,IAAI,CAAC,OAAO,CAAC,sBAAsB,GAAG,IAAI,GAAG,IAAI,CAAC;YAC7E,IAAI,WAAW,GAAG,kBAAkB,EAAE;gBACpC,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;gBAC/C,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,IAAI,EAAE,CAAC;gBAChD,MAAM,SAAS,GAAG;oBAChB,GAAG,cAAc;oBACjB,sBAAsB,EAAE,IAAI,CAAC,OAAO,CAAC,sBAAsB,GAAG,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC;iBAC/G,CAAC;gBACF,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;gBACjC,WAAW,CAAC,aAAa,EAAE,CAAC;gBAC5B,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,EAAE,CAAC;gBACzB,OAAO;aACR;SACF;QACD,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;YAClE,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;YAC/C,MAAM,SAAS,GAAG;gBAChB,yBAAyB,EAAE,CAAC;aAC7B,CAAC;YACF,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YACjC,WAAW,CAAC,aAAa,EAAE,CAAC;YAC5B,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,EAAE,CAAC;YACzB,OAAO;SACR;QACD,IAAI,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC;QACvF,IAAI,kBAAkB,GAAG,KAAK,CAAC;QAC/B,IAAI,mBAAmB,GAAG,EAAE,CAAC;QAC7B,IAAI,oBAAoB,GAAQ,EAAE,CAAC;QAEnC,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,UAAU,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;YACtD,MAAM,OAAO,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;YAClC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;YAC1B,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,eAAe,GAAG,IAAI,GAAG,IAAI,CAAC;YAC/D,IAAI,IAAI,EAAE;gBACR,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC;gBAC3B,IAAI,IAAI,CAAC,IAAI,GAAG,WAAW,EAAE;oBAC3B,oDAAoD;oBACpD,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAC3D,SAAS,CAAC,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAClC,CAAC;oBACF,kBAAkB,GAAG,IAAI,CAAC;oBAC1B,mBAAmB,GAAG,iBAAiB,CAAC;oBACxC,oBAAoB,GAAG,IAAI,CAAC,OAAO,CAAC,eAAe,GAAG,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;oBACvG,iEAAiE;iBAClE;gBACD,IACE,IAAI,CAAC,OAAO,CAAC,sBAAsB,IAAI,IAAI;oBAC3C,IAAI,CAAC,OAAO,CAAC,sBAAsB,CAAC,MAAM,GAAG,CAAC;oBAC9C,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,EAClC;oBACA,+DAA+D;oBAC/D,IAAI,CAAC,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAC3D,SAAS,CAAC,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAClC,CAAC;oBACF,kBAAkB,GAAG,IAAI,CAAC;oBAC1B,mBAAmB,GAAG,kBAAkB,CAAC;oBACzC,oBAAoB,GAAG,IAAI,CAAC,gBAAgB,CAAC;oBAC7C,iEAAiE;iBAClE;aACF;SACF;QAED,4EAA4E;QAC5E,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC;QACpC,IAAI,kBAAkB,EAAE;YACtB,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;YAC/C,MAAM,QAAQ,GAAG,EAAE,CAAC;YACpB,QAAQ,CAAC,mBAAmB,CAAC,GAAG,oBAAoB,CAAC;YACrD,IAAI,CAAC,uBAAuB,GAAG,QAAQ,CAAC,CAAC,gCAAgC;YACzE,WAAW,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;YAChC,WAAW,CAAC,aAAa,EAAE,CAAC;YAE5B,mFAAmF;YACnF,IAAI,mBAAmB,KAAK,iBAAiB,EAAE;gBAC7C,UAAU,CAAC,GAAG,EAAE;oBACd,4EAA4E;oBAC5E,MAAM,aAAa,GAAG,WAAW,CAAC,MAAM,CAAC;oBACzC,IAAI,aAAa,IAAI,aAAa,CAAC,iBAAiB,CAAC,KAAK,QAAQ,CAAC,iBAAiB,CAAC,EAAE;wBACrF,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;wBAC5B,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC;qBACrC;gBACH,CAAC,EAAE,IAAI,CAAC,CAAC;aACV;YAED,qDAAqD;SACtD;QAED,6DAA6D;QAC7D,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC;QAEnF,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,UAAU,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE;YACtD,MAAM,OAAO,GAAG,UAAU,CAAC,KAAK,CAAC,CAAC;YAClC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;YAC1B,IAAI,IAAI,EAAE;gBACR,IAAI,IAAI,CAAC,OAAO,CAAC,iBAAiB,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE;oBAC/D,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC;yBAC7C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;yBAC9B,SAAS,CAAC;wBACT,IAAI,EAAE,CAAC,KAAU,EAAE,EAAE;4BACnB,IAAI,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,OAAO,CAAC,CAAC;4BAC1E,IAAI,KAAK,CAAC,IAAI,KAAK,aAAa,CAAC,cAAc,EAAE;gCAC/C,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;gCAC3D,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,QAAQ,GAAG,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC;6BACrE;iCAAM,IAAI,KAAK,CAAC,IAAI,KAAK,aAAa,CAAC,QAAQ,EAAE;gCAChD,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,QAAQ,GAAG,GAAG,CAAC;gCAC/C,IAAI,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC,GAAU,CAAC;gCACnC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,MAAM,CAAC;gCAC1D,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC;gCACtD,IAAI,SAAS,GAAY;oCACvB,OAAO,EAAE,MAAM;oCACf,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC;oCACvD,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC;oCACvD,KAAK,EAAE,IAAI;iCACZ,CAAC;gCACF,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,KAAK,EAAE;oCACxC,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,EAAE,CAAC;oCAC7C,IAAI,CAAC,eAAe,CAAC,IAAI,GAAG,SAAS,CAAC;oCACtC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;iCACnF;qCAAM;oCACL,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oCAC3B,IAAI,CAAC,uBAAuB,CAAC,aAAa,GAAG,UAAU,CAAC;oCACxD,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,SAAS,EAAE;wCACjE,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,kBAAkB,IAAI,IAAI,EAAE;4CACjD,IAAI,CAAC,uBAAuB,CAAC,YAAY,GAAG,EAAE,CAAC;yCAChD;qCACF;oCACD,IAAI,CAAC,uBAAuB,CAAC,kBAAkB;wCAC7C,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,kBAAkB,CAAC;oCACzC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,uBAAuB,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;oCAC1F,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;iCAClC;6BACF;wBACH,CAAC;qBACF,CAAC,CAAC;iBACN;qBAAM;oBACL,IAAI,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;oBAC9B,IAAI,UAAU,GAAG,IAAI,CAAC,OAAc,CAAC;oBACrC,MAAM,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;oBACjC,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;wBACnB,IAAI,eAAe,GAAG,IAAI,CAAC;wBAE3B,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,CAAC,IAAI;4BAC3C,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC;wBAExF,IAAI,SAAS,GAAY;4BACvB,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;4BAClD,YAAY,EAAE,IAAI,CAAC,IAAI,GAAG,IAAI,GAAG,IAAI;4BACrC,iBAAiB,EAAE,IAAI,CAAC,IAAI;4BAC5B,OAAO,EAAE,eAAe;4BACxB,KAAK,EAAE,IAAI;yBACZ,CAAC;wBAEF,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,KAAK,EAAE;4BACxC,IAAI,CAAC,eAAe,GAAG,IAAI,eAAe,EAAE,CAAC;4BAC7C,IAAI,CAAC,eAAe,CAAC,IAAI,GAAG,SAAS,CAAC;4BACtC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;yBACnF;6BAAM;4BACL,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;4BAC3B,IAAI,CAAC,uBAAuB,CAAC,aAAa,GAAG,UAAU,CAAC;4BACxD,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,SAAS,EAAE;gCACjE,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,kBAAkB,IAAI,IAAI,EAAE;oCACjD,IAAI,CAAC,uBAAuB,CAAC,YAAY,GAAG,EAAE,CAAC;iCAChD;6BACF;4BACD,IAAI,CAAC,uBAAuB,CAAC,kBAAkB;gCAC7C,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,kBAAkB,CAAC;4BACzC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,uBAAuB,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC;yBAC3F;oBACH,CAAC,CAAC;oBACF,IAAI,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC;oBAC5D,IACE,IAAI,CAAC,OAAO,CAAC,aAAa;wBAC1B,IAAI,CAAC,OAAO,CAAC,SAAS;wBACtB,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,KAAK,EACvC;wBACA,IAAI,CAAC,cAAc,CAAC,iBAAiB,CACnC,aAAa,EACb,IAAI,CAAC,OAAO,CAAC,aAAa,EAC1B,IAAI,CAAC,OAAO,CAAC,SAAS,CACvB,CAAC;qBACH;oBACD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;iBACnC;aACF;SACF;IACH,CAAC;IAED,sBAAsB,CAAC,IAAc;QACnC,IAAI,IAAI,CAAC,OAAO,CAAC,iBAAiB,IAAI,IAAI,CAAC,QAAQ,IAAI,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE;YACjF,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;iBACrD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBAC9B,SAAS,CAAC,CAAC,GAAG,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;SAC5B;QACD,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,KAAK,EAAE;YACxC,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;YAC5B,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,IAAI,IAAI,EAAE;gBACnC,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,CAAC;gBAC3C,IAAI,CAAC,qBAAqB,CAAC,sBAAsB,EAAE,CAAC,CAAC,2BAA2B;aACjF;YACD,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YACjE,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,eAAe,CAAC;SAC3C;aAAM;YACL,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,SAAS,EAAE;gBACjE,IAAI,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,kBAAkB,IAAI,IAAI,EAAE;oBACjD,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;oBACvB,IAAI,CAAC,uBAAuB,CAAC,YAAY,GAAG,EAAE,CAAC;iBAChD;qBAAM;oBACL,IAAI,IAAI,CAAC,uBAAuB,CAAC,YAAY,CAAC,MAAM,IAAI,CAAC,EAAE;wBACzD,IAAI,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC;wBACnC,IAAI,WAAW,GAAG,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,CACjE,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,IAAI,UAAU,CAAC,IAAI,CAClD,CAAC,CAAC,CAAC,CAAC;wBACL,IAAI,CAAC,uBAAuB,CAAC,aAAa;4BACxC,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,CAC/C,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,IAAI,UAAU,CAAC,IAAI,CAClD,CAAC;wBACJ,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;wBACpC,IAAI,CAAC,uBAAuB,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;qBACrE;yBAAM;wBACL,IAAI,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC;wBACnC,IAAI,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CACxC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,IAAI,UAAU,CAAC,IAAI,CAClD,CAAC;wBACF,IAAI,WAAW,CAAC,MAAM,IAAI,CAAC,IAAI,WAAW,IAAI,SAAS,EAAE;4BACvD,IAAI,WAAW,GAAG,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,CACjE,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,IAAI,UAAU,CAAC,IAAI,CAClD,CAAC,CAAC,CAAC,CAAC;4BACL,IAAI,CAAC,uBAAuB,CAAC,aAAa;gCACxC,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,CAC/C,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,iBAAiB,IAAI,UAAU,CAAC,IAAI,CAClD,CAAC;4BACJ,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;4BACpC,IAAI,CAAC,uBAAuB,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;yBACrE;qBACF;iBACF;aACF;iBAAM;gBACL,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;gBACvB,IAAI,CAAC,uBAAuB,CAAC,YAAY,GAAG,EAAE,CAAC;aAChD;YACD,IAAI,CAAC,uBAAuB,CAAC,aAAa;gBACxC,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,CAC/C,CAAC,GAAG,EAAE,EAAE,CACN,CAAC,GAAG,CAAC,iBAAiB,IAAI,GAAG,CAAC,iBAAiB,IAAI,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;oBACnE,GAAG,CAAC,OAAO,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CACvC,CAAC;YAEJ,IACE,CAAC,IAAI,CAAC,uBAAuB,CAAC,aAAa,IAAI,IAAI;gBACjD,IAAI,CAAC,uBAAuB,CAAC,aAAa,CAAC,MAAM,IAAI,CAAC,CAAC;gBACzD,IAAI,CAAC,OAAO,CAAC,UAAU,EACvB;gBACA,MAAM,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC;gBAC/C,WAAW,CAAC,SAAS,CAAC;oBACpB,yBAAyB,EAAE,IAAI,CAAC,OAAO,CAAC,YAAY;iBACrD,CAAC,CAAC;gBACH,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,CAAC;gBAC3C,IAAI,CAAC,qBAAqB,CAAC,sBAAsB,EAAE,CAAC,CAAC,2BAA2B;aACjF;YACD,IAAI,CAAC,uBAAuB,CAAC,kBAAkB,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,kBAAkB,CAAC;YACzF,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;YAClE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;YACzE,IAAI,CAAC,qBAAqB,CAAC,sBAAsB,EAAE,CAAC;YACpD,0CAA0C;YAC1C,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,uBAAuB,CAAC;SACnD;IACH,CAAC;IA2CD,eAAe,CAAC,IAAY;QAC1B,IAAI,IAAI,KAAK,CAAC,EAAE;YACd,OAAO,CAAC,CAAC;SACV;QAED,4BAA4B;QAC5B,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;QAEvC,8BAA8B;QAC9B,MAAM,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC;QAE3D,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED,WAAW;QACT,qDAAqD;QACrD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QAEzB,sCAAsC;QACtC,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;SAC5B;IACH,CAAC;IAEM,gBAAgB,CAAC,QAAgB;QACtC,0DAA0D;QAC1D,OAAO,QAAQ,CAAC,OAAO,CAAC,eAAe,EAAE,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;IACvD,CAAC;IAGO,kBAAkB;QACxB,MAAM,cAAc,GAAU,EAAE,CAAC;QACjC,MAAM,iBAAiB,GAAG,IAAI,GAAG,EAAU,CAAC;QAE5C,oFAAoF;QACpF,MAAM,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC;QAC5F,KAAK,MAAM,IAAI,IAAI,aAAa,EAAE;YAChC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;YACrD,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;SACjC;QAED,mEAAmE;QACnE,MAAM,eAAe,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC;QAE9F,KAAK,MAAM,IAAI,IAAI,eAAe,EAAE;YAClC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;YAErD,sEAAsE;YACtE,IAAI,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;gBACnC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC3B;iBAAM;gBACL,wFAAwF;gBACxF,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;aACjC;SACF;QAED,OAAO,cAAc,CAAC;IACxB,CAAC;IAGO,6BAA6B,CAAC,cAAqB;QACzD,KAAK,MAAM,aAAa,IAAI,cAAc,EAAE;YAC1C,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;YACzD,IAAI,KAAK,GAAG,CAAC,CAAC,EAAE;gBACd,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;aACtC;SACF;IACH,CAAC;IAEO,kBAAkB,CAAC,QAAgB;QACzC,IAAI,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAAE;YAClE,OAAO,IAAI,CAAC,CAAC,kBAAkB;SAChC;QAED,yCAAyC;QACzC,MAAM,kBAAkB,GAAG,QAAQ,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;QAEzD,kEAAkE;QAClE,OAAO,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE;YAChD,MAAM,sBAAsB,GAAG,YAAY,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;YACjE,OAAO,sBAAsB,KAAK,kBAAkB,CAAC;QACvD,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,wBAAwB,CAAC,KAAU,EAAE,cAAoB;QAC/D,+DAA+D;QAC/D,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,MAAM,CAAC;QACxD,MAAM,cAAc,GAAG,aAAa,EAAE,kBAAkB,CAAC,CAAC,CAAC,EAAE,kBAAkB,EAAE,aAAa,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;QAE3H,6BAA6B;QAC7B,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC3C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAElD,4FAA4F;QAC5F,MAAM,aAAa,GAAG,cAAc,IAAI,cAAc,IAAI,IAAI,CAAC,uBAAuB,CAAC;QACvF,IAAI,aAAa,EAAE;YACjB,IAAI,CAAC,qBAAqB,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;YACpD,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,CAAC;SAC5C;QAED,+CAA+C;QAC/C,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,KAAK,CAAC;IAC7B,CAAC;+GA7xBU,mBAAmB;mGAAnB,mBAAmB,4SCzBhC,gxWAiKM,2CDhJF,YAAY,wZACZ,WAAW,0LACX,mBAAmB,+KACnB,gBAAgB,4SAChB,iBAAiB;;4FAIR,mBAAmB;kBAb/B,SAAS;+BACE,iBAAiB,cAEf,IAAI,WACP;wBACP,YAAY;wBACZ,WAAW;wBACX,mBAAmB;wBACnB,gBAAgB;wBAChB,iBAAiB;qBAClB,WACQ,CAAC,sBAAsB,EAAE,gBAAgB,CAAC;;0BAiChD,QAAQ;wOA5BgC,SAAS;sBAAnD,SAAS;uBAAC,WAAW,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE;gBAGhC,KAAK;sBAAb,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACI,QAAQ;sBAAjB,MAAM;gBACG,gBAAgB;sBAAzB,MAAM","sourcesContent":["import { Component, OnInit, AfterViewInit, Input, Optional, ViewChild, ElementRef, Output, EventEmitter, CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA, OnDestroy } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { HttpEventType } from '@angular/common/http';\r\nimport { FormControl, Validators, FormGroup, AbstractControl, ControlContainer, FormGroupDirective, FormsModule, ReactiveFormsModule } from '@angular/forms';\r\nimport { FileUploader, FileLikeObject, FileUploaderOptions, FileItem, FileUploadModule } from 'ng2-file-upload';\r\nimport { NgxDropzoneModule } from 'ngx-dropzone';\r\nimport { ControlValidationService, UtilityService } from '@bnsights/bbsf-utilities/ui';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { FileDTO, FileUploadModel, FileUploadOptions, MultipleFileUploadModel } from '../../Shared/Models';\r\nimport { ControlUtility, FileUploadService, GlobalSettings } from '../../Shared/services';\r\n\r\n@Component({\r\n  selector: 'BBSF-FileUpload',\r\n  templateUrl: './FileUpload.component.html',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    FileUploadModule,\r\n    NgxDropzoneModule\r\n  ],\r\n  schemas: [CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA]\r\n})\r\nexport class FileUploadComponent implements OnInit, AfterViewInit, OnDestroy {\r\n  static controlContainerStatic = null;\r\n\r\n  @ViewChild('fileInput', { static: false }) fileInput: ElementRef;\r\n\r\n  isSubmitted: boolean;\r\n  @Input() group: FormGroup;\r\n  @Input() options: FileUploadOptions;\r\n  @Output() OnChange = new EventEmitter();\r\n  @Output() isUploadComplete = new EventEmitter();\r\n\r\n  fileUploadFormControl: AbstractControl;\r\n  validationMessage = '';\r\n  validationCountMessage = '';\r\n  uploader: FileUploader;\r\n  hasAnotherDropZoneOver: boolean;\r\n  acceptedType: string = '';\r\n  acceptedTypeArray: string[] = [];\r\n  toolTipTypeArray: string[] = [];\r\n  fileLikeObject: FileLikeObject;\r\n  markAllAsTouched: boolean = false;\r\n  validationRules = [];\r\n  validationRulesAsync = [];\r\n  fileUploadModel: FileUploadModel;\r\n  multipleFileUploadModel: MultipleFileUploadModel;\r\n  file: any;\r\n  deletedFiles: FileDTO[] = [];\r\n  private destroy$ = new Subject<void>();\r\n  private pendingValidationErrors: any = null;\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    // Initialize uploader with minimal options, URL will be set in ngOnInit only if needed\r\n    this.uploader = new FileUploader({\r\n      url: '', // Empty URL initially, will be set in ngOnInit only if isUploadFileAsync is true\r\n      disableMultipart: false // 'DisableMultipart' must be 'true' for formatDataFunction to be called.\r\n    } as FileUploaderOptions);\r\n    this.hasAnotherDropZoneOver = false;\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.fileUploadModel = new FileUploadModel();\r\n    this.multipleFileUploadModel = new MultipleFileUploadModel();\r\n    if (!this.options.viewType) this.options.viewType = this.globalSettings.viewType;\r\n\r\n    // Configure uploader options - no URL needed since we handle uploads via FileUploadService\r\n    // Note: allowedMimeType in ng2-file-upload doesn't work well with comma-separated enum values\r\n    // We handle MIME type validation manually in onFileChange\r\n    this.uploader.setOptions({\r\n      url: '', // Not used - uploads handled by FileUploadService when isUploadFileAsync is true\r\n      disableMultipart: false,\r\n      autoUpload: false // We manually control uploads\r\n      // allowedMimeType removed - we handle validation manually for better control\r\n    });\r\n\r\n    if (this.options.value == null) {\r\n      if (this.options.isMultipleFile == true) {\r\n        //this.options.Value=this.multipleFileUploadModel\r\n      } else {\r\n        // this.options.Value=this.fileUploadModel\r\n      }\r\n    } else {\r\n      if (this.options.isMultipleFile == true) {\r\n        let files: any[] = [];\r\n        this.multipleFileUploadModel.existingFiles = this.options.value.existingFiles;\r\n        this.multipleFileUploadModel.uploadedFiles = [];\r\n        for (let index = 0; index < this.options.value.existingFiles.length; index++) {\r\n          const element = this.options.value.existingFiles[index];\r\n          var bytes = new Uint8Array(element.bytes);\r\n          var base64 = btoa(String.fromCharCode(...Array.from(bytes)));\r\n          this.fileLikeObject = <FileLikeObject>{\r\n            name: element.nameWithExtension,\r\n            type: element.mimeType,\r\n            rawFile: <any>base64\r\n          };\r\n          // let blob: any;\r\n          // blob = new Blob(['']) as any;\r\n          // blob.name = element.FileName;\r\n          // blob.lastModifiedDate = null;\r\n          // blob.webkitRelativePath = '';\r\n          let file = this.fileLikeObject as any;\r\n          file.url = element.fullFileURL;\r\n          files.push(file);\r\n        }\r\n        this.uploader.addToQueue(files);\r\n        console.log(this.uploader.queue);\r\n      } else {\r\n        const element = this.options.value.file ?? this.options.value;\r\n        var bytes = new Uint8Array(element.bytes);\r\n        var base64 = btoa(String.fromCharCode(...Array.from(bytes)));\r\n        this.fileLikeObject = <FileLikeObject>{\r\n          name: element.nameWithExtension,\r\n          type: element.mimeType,\r\n          rawFile: <any>base64\r\n        };\r\n        this.file = element;\r\n        let file = this.fileLikeObject as any;\r\n        file.url = element.fullFileURL;\r\n        this.uploader.addToQueue([file]);\r\n        if (!this.options.value.file) {\r\n          this.fileUploadModel = new FileUploadModel();\r\n          this.fileUploadModel.file = this.options.value;\r\n        } else this.fileUploadModel = this.options.value;\r\n        this.options.value = this.fileUploadModel;\r\n      }\r\n      this.uploader.queue.forEach((element) => {\r\n        element.progress = 100;\r\n      });\r\n    }\r\n\r\n    if (this.options.labelKey != null && this.options.labelKey != '')\r\n      this.options.labelValue = this.UtilityService.getResourceValue(this.options.labelKey);\r\n\r\n    if (\r\n      this.options.fileUploadAcceptsTypes != null &&\r\n      this.options.fileUploadAcceptsTypes.length > 0\r\n    ) {\r\n      // Process FileType enum values which may contain comma-separated MIME types\r\n      for (let index = 0; index < this.options.fileUploadAcceptsTypes.length; index++) {\r\n        const Type = this.options.fileUploadAcceptsTypes[index];\r\n        this.acceptedType = this.acceptedType + Type + ',';\r\n      }\r\n      this.acceptedTypeArray = this.acceptedType.split(',');\r\n      this.acceptedTypeArray = this.acceptedTypeArray.filter((value) => value.trim() != '');\r\n\r\n      // Rebuild acceptedType string for HTML accept attribute (properly formatted)\r\n      this.acceptedType = this.acceptedTypeArray.join(',');\r\n      // Process each accepted MIME type to build tooltip display names\r\n      for (let index = 0; index < this.acceptedTypeArray.length; index++) {\r\n        const element = this.acceptedTypeArray[index].trim();\r\n        switch (element) {\r\n          case 'application/pdf':\r\n            if (!this.toolTipTypeArray.includes('PDF')) this.toolTipTypeArray.push('PDF');\r\n            break;\r\n          case 'application/msword':\r\n            if (!this.toolTipTypeArray.includes('Word')) this.toolTipTypeArray.push('Word');\r\n            break;\r\n          case 'application/vnd.openxmlformats-officedocument.wordprocessingml.document':\r\n            if (!this.toolTipTypeArray.includes('Word')) this.toolTipTypeArray.push('Word');\r\n            break;\r\n          case 'application/vnd.ms-excel':\r\n            if (!this.toolTipTypeArray.includes('Excel')) this.toolTipTypeArray.push('Excel');\r\n            break;\r\n          case 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet':\r\n            if (!this.toolTipTypeArray.includes('Excel')) this.toolTipTypeArray.push('Excel');\r\n            break;\r\n          case 'application/vnd.ms-powerpoint':\r\n            if (!this.toolTipTypeArray.includes('PowerPoint'))\r\n              this.toolTipTypeArray.push('PowerPoint');\r\n            break;\r\n          case 'application/vnd.openxmlformats-officedocument.presentationml.presentation':\r\n            if (!this.toolTipTypeArray.includes('PowerPoint'))\r\n              this.toolTipTypeArray.push('PowerPoint');\r\n            break;\r\n          case 'image/png':\r\n            if (!this.toolTipTypeArray.includes('PNG')) this.toolTipTypeArray.push('PNG');\r\n            break;\r\n          case 'image/bmp':\r\n            if (!this.toolTipTypeArray.includes('BMP')) this.toolTipTypeArray.push('BMP');\r\n            break;\r\n          case 'image/jpeg':\r\n            if (!this.toolTipTypeArray.includes('JPEG')) this.toolTipTypeArray.push('JPEG');\r\n            break;\r\n          case 'application/zip':\r\n            if (!this.toolTipTypeArray.includes('ZIP')) this.toolTipTypeArray.push('ZIP');\r\n            break;\r\n          case 'application/x-rar-compressed':\r\n            if (!this.toolTipTypeArray.includes('RAR')) this.toolTipTypeArray.push('RAR');\r\n            break;\r\n          case 'video/mp4':\r\n            if (!this.toolTipTypeArray.includes('MP4')) this.toolTipTypeArray.push('MP4');\r\n            break;\r\n          case 'video/avi':\r\n            if (!this.toolTipTypeArray.includes('AVI')) this.toolTipTypeArray.push('AVI');\r\n            break;\r\n          case 'video/quicktime':\r\n            if (!this.toolTipTypeArray.includes('MOV')) this.toolTipTypeArray.push('MOV');\r\n            break;\r\n          case 'video/mpeg':\r\n            if (!this.toolTipTypeArray.includes('MPEG')) this.toolTipTypeArray.push('MPEG');\r\n            break;\r\n          case 'audio/mpeg':\r\n            if (!this.toolTipTypeArray.includes('MP3')) this.toolTipTypeArray.push('MP3');\r\n            break;\r\n          case 'video/x-flv':\r\n            if (!this.toolTipTypeArray.includes('FLV')) this.toolTipTypeArray.push('FLV');\r\n            break;\r\n          case 'video/x-ms-wmv':\r\n            if (!this.toolTipTypeArray.includes('WMV')) this.toolTipTypeArray.push('WMV');\r\n            break;\r\n          case 'image/svg+xml':\r\n            if (!this.toolTipTypeArray.includes('SVG')) this.toolTipTypeArray.push('SVG');\r\n            break;\r\n          case 'text/plain':\r\n            if (!this.toolTipTypeArray.includes('Txt')) this.toolTipTypeArray.push('Txt');\r\n            break;\r\n          case 'application/BN':\r\n            if (!this.toolTipTypeArray.includes('License')) this.toolTipTypeArray.push('License');\r\n            break;\r\n          default:\r\n            break;\r\n        }\r\n      }\r\n      this.validationMessage = this.validationMessage + `${this.UtilityService.getResourceValue('Extensions')} (${this.toolTipTypeArray}) `;\r\n    }\r\n\r\n    if (this.options.fileMaxSizeInMB > 0) {\r\n      this.validationMessage = this.validationMessage + `<br />  ${this.UtilityService.getResourceValue('FileMaxSizeInMB') + this.options.fileMaxSizeInMB + ' ' + this.UtilityService.getResourceValue('MB')}`;\r\n    }\r\n\r\n    if (this.options.minNoOfFiles > 0) {\r\n      this.validationMessage = this.validationMessage + `<br /> ${this.UtilityService.getResourceValue('MinFileCountValidationKey') + this.options.minNoOfFiles}`;\r\n    }\r\n\r\n    if (this.options.maxNoOfFiles > 0) {\r\n      this.validationMessage = this.validationMessage + `<br /> ${this.UtilityService.getResourceValue('MaxFileCountValidationKey') + this.options.maxNoOfFiles}`;\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    if (this.options.customValidation.length > 0) {\r\n      let Validations = this.options.customValidation;\r\n      for (let index = 0; index < Validations.length; index++) {\r\n        const Validation = Validations[index];\r\n        this.validationRules.push(Validation.functionBody);\r\n      }\r\n    }\r\n\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    if (this.options.isRequired) {\r\n      this.validationRules.push(Validators.required);\r\n    }\r\n\r\n    this.fileUploadFormControl.setValidators(this.validationRules);\r\n    this.fileUploadFormControl.setAsyncValidators(this.validationRulesAsync);\r\n    if (this.options.isDisabled) {\r\n      this.fileUploadFormControl.disable();\r\n    }\r\n\r\n    this.MultipleFileUploadControlHost.ngSubmit\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe((value) => {\r\n        this.group.markAllAsTouched();\r\n        this.markAllAsTouched = true;\r\n      });\r\n\r\n    this.fileUploadFormControl.setValue(this.options.value);\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    if (this.options.attributeList != null) {\r\n      var element = document.getElementById(this.options.name);\r\n      for (let index = 0; index < this.options.attributeList.length; index++) {\r\n        element.setAttribute(\r\n          this.options.attributeList[index].key,\r\n          this.options.attributeList[index].value\r\n        );\r\n      }\r\n    }\r\n  }\r\n\r\n  resetError = () => {\r\n    this.controlValidationService.removeGlobalError();\r\n  };\r\n\r\n  showGlobalError() {\r\n    this.controlUtility.showGlobalError();\r\n  }\r\n\r\n  getErrorValidation(ErrorList) {\r\n    if (this.markAllAsTouched && this.group.invalid) {\r\n      this.showGlobalError();\r\n      this.markAllAsTouched = false;\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() {\r\n    if (this.options.isMultipleFile) {\r\n      if (\r\n        this.options.maxNoOfFiles != null &&\r\n        this.options.maxNoOfFiles > 0 &&\r\n        this.options.maxNoOfFiles == this.uploader.queue.length\r\n      ) {\r\n        return true;\r\n      }\r\n    } else {\r\n      if (this.uploader.queue.length > 0) {\r\n        return true;\r\n      }\r\n    }\r\n    return false;\r\n  }\r\n\r\n  onFileChange() {\r\n    let FilesArray: FileDTO[] = [];\r\n    // Clear any previous validation errors at the start of file change\r\n    this.pendingValidationErrors = null;\r\n\r\n    // Check for duplicate file names in the queue\r\n    const duplicateFiles = this.findDuplicateFiles();\r\n    if (duplicateFiles.length > 0) {\r\n      // Remove only the newly added duplicate files from queue (keep existing files)\r\n      this.removeDuplicateFilesFromQueue(duplicateFiles);\r\n\r\n      // Set validation error for duplicate files\r\n      const duplicateFileNames = duplicateFiles.map(f => f.file?.name).join(', ');\r\n      const formControl = this.fileUploadFormControl;\r\n      const resourceValue = this.UtilityService.getResourceValue('DuplicateFileError');\r\n      // If resource value is the same as the key, it means the resource wasn't found\r\n      const errorMessage = (resourceValue && resourceValue !== 'DuplicateFileError') ? resourceValue : 'File(s) already exist';\r\n      const errorObj = {\r\n        DuplicateFileError: `${errorMessage}: ${duplicateFileNames}`\r\n      };\r\n      this.pendingValidationErrors = errorObj;\r\n      formControl.setErrors(errorObj);\r\n      formControl.markAsTouched();\r\n\r\n      // Auto-clear duplicate error after 5 seconds to allow form submission\r\n      setTimeout(() => {\r\n        // Only clear if the current error is still the duplicate error we set\r\n        const currentErrors = formControl.errors;\r\n        if (currentErrors && currentErrors.DuplicateFileError === errorObj.DuplicateFileError) {\r\n          formControl.setErrors(null);\r\n          this.pendingValidationErrors = null;\r\n        }\r\n      }, 5000);\r\n\r\n      // If all files were duplicates, return early\r\n      if (this.uploader.queue.filter(item => item['some'].lastModified != null).length === 0) {\r\n        return;\r\n      }\r\n    }\r\n    if (this.options.isMultipleFile && this.options.minNoOfFiles != null && this.options.minNoOfFiles > 0 && this.options.minNoOfFiles > this.uploader.queue.length) {\r\n      const formControl = this.fileUploadFormControl;\r\n      const existingErrors = formControl.errors || {};\r\n      const newErrors = {\r\n        ...existingErrors,\r\n        MinFileCountValidationKey: this.options.minNoOfFiles\r\n      };\r\n      formControl.setErrors(newErrors);\r\n      formControl.markAsTouched();\r\n      this.uploader.queue = [];\r\n      return;\r\n    }\r\n\r\n    if (this.options.isMultipleFile && this.options.maxNoOfFiles != null && this.options.maxNoOfFiles > 0 && this.options.maxNoOfFiles < this.uploader.queue.length) {\r\n      const formControl = this.fileUploadFormControl;\r\n      const existingErrors = formControl.errors || {};\r\n      const newErrors = {\r\n        ...existingErrors,\r\n        MaxFileCountValidationKey: this.options.maxNoOfFiles\r\n      };\r\n      formControl.setErrors(newErrors);\r\n      formControl.markAsTouched();\r\n      this.uploader.queue = [];\r\n      return;\r\n    }\r\n\r\n\r\n    if (this.options.isMultipleFile && this.options.maxSizeForAllFilesInMB != null && this.options.maxSizeForAllFilesInMB > 0) {\r\n      let AllSizeFile = 0;\r\n      for (let index = 0; index < this.uploader.queue.length; index++) {\r\n        const element = this.uploader.queue[index];\r\n        const file = element.file;\r\n        AllSizeFile = AllSizeFile + file.size;\r\n      }\r\n      const MaxSizeForAllFiles = this.options.maxSizeForAllFilesInMB * 1000 * 1000;\r\n      if (AllSizeFile > MaxSizeForAllFiles) {\r\n        const formControl = this.fileUploadFormControl;\r\n        const existingErrors = formControl.errors || {};\r\n        const newErrors = {\r\n          ...existingErrors,\r\n          MaxSizeForAllFilesInMB: this.options.maxSizeForAllFilesInMB + ' ' + this.UtilityService.getResourceValue('MB')\r\n        };\r\n        formControl.setErrors(newErrors);\r\n        formControl.markAsTouched();\r\n        this.uploader.queue = [];\r\n        return;\r\n      }\r\n    }\r\n    if (!this.options.isMultipleFile && this.uploader.queue.length > 1) {\r\n      const formControl = this.fileUploadFormControl;\r\n      const newErrors = {\r\n        MaxFileCountValidationKey: 1\r\n      };\r\n      formControl.setErrors(newErrors);\r\n      formControl.markAsTouched();\r\n      this.uploader.queue = [];\r\n      return;\r\n    }\r\n    let AddedQueue = this.uploader.queue.filter((obj) => obj['some'].lastModified != null);\r\n    let hasValidationError = false;\r\n    let validationErrorType = '';\r\n    let validationErrorValue: any = '';\r\n\r\n    for (let index = 0; index < AddedQueue.length; index++) {\r\n      const element = AddedQueue[index];\r\n      const file = element.file;\r\n      const maxFileSize = this.options.fileMaxSizeInMB * 1000 * 1000;\r\n      if (file) {\r\n        const fileType = file.type;\r\n        if (file.size > maxFileSize) {\r\n          // Remove only the file that violates the size limit\r\n          this.uploader.queue = this.uploader.queue.filter(queueItem =>\r\n            queueItem.file.name !== file.name\r\n          );\r\n          hasValidationError = true;\r\n          validationErrorType = 'FileMaxSizeInMB';\r\n          validationErrorValue = this.options.fileMaxSizeInMB + ' ' + this.UtilityService.getResourceValue('MB');\r\n          // Continue checking other files instead of returning immediately\r\n        }\r\n        if (\r\n          this.options.fileUploadAcceptsTypes != null &&\r\n          this.options.fileUploadAcceptsTypes.length > 0 &&\r\n          !this.isFileTypeAccepted(fileType)\r\n        ) {\r\n          // Remove only the file that violates the file type restriction\r\n          this.uploader.queue = this.uploader.queue.filter(queueItem =>\r\n            queueItem.file.name !== file.name\r\n          );\r\n          hasValidationError = true;\r\n          validationErrorType = 'ToolTipTypeError';\r\n          validationErrorValue = this.toolTipTypeArray;\r\n          // Continue checking other files instead of returning immediately\r\n        }\r\n      }\r\n    }\r\n\r\n    // Set validation error if any files were removed due to validation failures\r\n    this.pendingValidationErrors = null;\r\n    if (hasValidationError) {\r\n      const formControl = this.fileUploadFormControl;\r\n      const errorObj = {};\r\n      errorObj[validationErrorType] = validationErrorValue;\r\n      this.pendingValidationErrors = errorObj; // Store errors to reapply later\r\n      formControl.setErrors(errorObj);\r\n      formControl.markAsTouched();\r\n\r\n      // Auto-clear FileMaxSizeInMB error after 5 seconds (similar to DuplicateFileError)\r\n      if (validationErrorType === 'FileMaxSizeInMB') {\r\n        setTimeout(() => {\r\n          // Only clear if the current error is still the FileMaxSizeInMB error we set\r\n          const currentErrors = formControl.errors;\r\n          if (currentErrors && currentErrors['FileMaxSizeInMB'] === errorObj['FileMaxSizeInMB']) {\r\n            formControl.setErrors(null);\r\n            this.pendingValidationErrors = null;\r\n          }\r\n        }, 5000);\r\n      }\r\n\r\n      // Don't return here, continue processing valid files\r\n    }\r\n\r\n    // Re-filter the queue to get only valid files for processing\r\n    AddedQueue = this.uploader.queue.filter((obj) => obj['some'].lastModified != null);\r\n\r\n    for (let index = 0; index < AddedQueue.length; index++) {\r\n      const element = AddedQueue[index];\r\n      const file = element.file;\r\n      if (file) {\r\n        if (this.options.isUploadFileAsync && !element._file['iD_GUID']) {\r\n          this.fileUploadService.uploadFile(element._file)\r\n            .pipe(takeUntil(this.destroy$))\r\n            .subscribe({\r\n              next: (event: any) => {\r\n                let queueIndex = this.uploader.queue.findIndex((file) => file == element);\r\n                if (event.type === HttpEventType.UploadProgress) {\r\n                  let value = Math.round((100 * event.loaded) / event.total);\r\n                  this.uploader.queue[queueIndex].progress = value >= 95 ? 95 : value;\r\n                } else if (event.type === HttpEventType.Response) {\r\n                  this.uploader.queue[queueIndex].progress = 100;\r\n                  let fileID = event.body.val as any;\r\n                  this.uploader.queue[queueIndex]._file['iD_GUID'] = fileID;\r\n                  this.uploader.queue[queueIndex]._file['isNew'] = true;\r\n                  let AddedFile = <FileDTO>{\r\n                    iD_GUID: fileID,\r\n                    fileName: this.uploader.queue[queueIndex]._file['name'],\r\n                    fileType: this.uploader.queue[queueIndex]._file['type'],\r\n                    isNew: true\r\n                  };\r\n                  if (this.options.isMultipleFile == false) {\r\n                    this.fileUploadModel = new FileUploadModel();\r\n                    this.fileUploadModel.file = AddedFile;\r\n                    this.setValuePreservingErrors(this.fileUploadModel, this.pendingValidationErrors);\r\n                  } else {\r\n                    FilesArray.push(AddedFile);\r\n                    this.multipleFileUploadModel.uploadedFiles = FilesArray;\r\n                    if (this.options.value != null && this.options.value != undefined) {\r\n                      if (this.options.value.correlationID_GUID == null) {\r\n                        this.multipleFileUploadModel.removedFiles = [];\r\n                      }\r\n                    }\r\n                    this.multipleFileUploadModel.correlationID_GUID =\r\n                      this.options.value?.correlationID_GUID;\r\n                    this.setValuePreservingErrors(this.multipleFileUploadModel, this.pendingValidationErrors);\r\n                    this.isUploadComplete.emit(true);\r\n                  }\r\n                }\r\n              }\r\n            });\r\n        } else {\r\n          let reader = new FileReader();\r\n          let fileObject = file.rawFile as any;\r\n          reader.readAsDataURL(fileObject);\r\n          reader.onload = () => {\r\n            let existingID_GUID = null;\r\n\r\n            if (!this.options.isMultipleFile && this.file)\r\n              existingID_GUID = this.file.NameWithExtension == file.name ? this.file.iD_GUID : null;\r\n\r\n            let AddedFile: FileDTO = {\r\n              fileName: file.name,\r\n              fileType: file.type,\r\n              fileBase64: reader.result.toString().split(',')[1],\r\n              fileSizeInMB: file.size / 1000 / 1000,\r\n              nameWithExtension: file.name,\r\n              iD_GUID: existingID_GUID,\r\n              isNew: true\r\n            };\r\n\r\n            if (this.options.isMultipleFile == false) {\r\n              this.fileUploadModel = new FileUploadModel();\r\n              this.fileUploadModel.file = AddedFile;\r\n              this.setValuePreservingErrors(this.fileUploadModel, this.pendingValidationErrors);\r\n            } else {\r\n              FilesArray.push(AddedFile);\r\n              this.multipleFileUploadModel.uploadedFiles = FilesArray;\r\n              if (this.options.value != null && this.options.value != undefined) {\r\n                if (this.options.value.correlationID_GUID == null) {\r\n                  this.multipleFileUploadModel.removedFiles = [];\r\n                }\r\n              }\r\n              this.multipleFileUploadModel.correlationID_GUID =\r\n                this.options.value?.correlationID_GUID;\r\n              this.setValuePreservingErrors(this.multipleFileUploadModel, this.pendingValidationErrors);\r\n            }\r\n          };\r\n          let originalValue = this.group.get(this.options.name).value;\r\n          if (\r\n            this.options.patchFunction &&\r\n            this.options.patchPath &&\r\n            this.group.get(this.options.name).valid\r\n          ) {\r\n            this.controlUtility.patchControlValue(\r\n              originalValue,\r\n              this.options.patchFunction,\r\n              this.options.patchPath\r\n            );\r\n          }\r\n          this.OnChange.emit(originalValue);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  removeFromControlValue(item: FileItem) {\r\n    if (this.options.isUploadFileAsync && item.progress == 100 && item._file['isNew']) {\r\n      this.fileUploadService.deleteFile(item._file['iD_GUID'])\r\n        .pipe(takeUntil(this.destroy$))\r\n        .subscribe((res) => { });\r\n    }\r\n    if (this.options.isMultipleFile == false) {\r\n      this.fileUploadModel = null;\r\n      if (this.options.isRequired == true) {\r\n        this.fileUploadFormControl.markAsTouched();\r\n        this.fileUploadFormControl.updateValueAndValidity(); // Trigger validation check\r\n      }\r\n      this.group.get(this.options.name).setValue(this.fileUploadModel);\r\n      this.options.value = this.fileUploadModel;\r\n    } else {\r\n      if (this.options.value != null && this.options.value != undefined) {\r\n        if (this.options.value.correlationID_GUID == null) {\r\n          this.deletedFiles = [];\r\n          this.multipleFileUploadModel.removedFiles = [];\r\n        } else {\r\n          if (this.multipleFileUploadModel.removedFiles.length == 0) {\r\n            let FileObject = item.file.rawFile;\r\n            let DeletedItem = this.multipleFileUploadModel.existingFiles.filter(\r\n              (obj) => obj.nameWithExtension == FileObject.name\r\n            )[0];\r\n            this.multipleFileUploadModel.existingFiles =\r\n              this.multipleFileUploadModel.existingFiles.filter(\r\n                (obj) => obj.nameWithExtension != FileObject.name\r\n              );\r\n            this.deletedFiles.push(DeletedItem);\r\n            this.multipleFileUploadModel.removedFiles.push(DeletedItem.iD_GUID);\r\n          } else {\r\n            let FileObject = item.file.rawFile;\r\n            let deletedList = this.deletedFiles.filter(\r\n              (obj) => obj.nameWithExtension == FileObject.name\r\n            );\r\n            if (deletedList.length == 0 || deletedList == undefined) {\r\n              let DeletedItem = this.multipleFileUploadModel.existingFiles.filter(\r\n                (obj) => obj.nameWithExtension == FileObject.name\r\n              )[0];\r\n              this.multipleFileUploadModel.existingFiles =\r\n                this.multipleFileUploadModel.existingFiles.filter(\r\n                  (obj) => obj.nameWithExtension != FileObject.name\r\n                );\r\n              this.deletedFiles.push(DeletedItem);\r\n              this.multipleFileUploadModel.removedFiles.push(DeletedItem.iD_GUID);\r\n            }\r\n          }\r\n        }\r\n      } else {\r\n        this.deletedFiles = [];\r\n        this.multipleFileUploadModel.removedFiles = [];\r\n      }\r\n      this.multipleFileUploadModel.uploadedFiles =\r\n        this.multipleFileUploadModel.uploadedFiles.filter(\r\n          (obj) =>\r\n            (obj.nameWithExtension && obj.nameWithExtension != item._file.name) ||\r\n            obj.iD_GUID != item._file['iD_GUID']\r\n        );\r\n\r\n      if (\r\n        (this.multipleFileUploadModel.uploadedFiles == null ||\r\n          this.multipleFileUploadModel.uploadedFiles.length == 0) &&\r\n        this.options.isRequired\r\n      ) {\r\n        const formControl = this.fileUploadFormControl;\r\n        formControl.setErrors({\r\n          MinFileCountValidationKey: this.options.minNoOfFiles\r\n        });\r\n        this.fileUploadFormControl.markAsTouched();\r\n        this.fileUploadFormControl.updateValueAndValidity(); // Trigger validation check\r\n      }\r\n      this.multipleFileUploadModel.correlationID_GUID = this.options.value?.correlationID_GUID;\r\n      this.fileUploadFormControl.setValue(this.multipleFileUploadModel);\r\n      this.group.get(this.options.name).setValue(this.multipleFileUploadModel);\r\n      this.fileUploadFormControl.updateValueAndValidity();\r\n      //Use this line to enable two way binding.\r\n      this.options.value = this.multipleFileUploadModel;\r\n    }\r\n  }\r\n\r\n  //External Method\r\n  removeRequiredValidation = () => {\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  //External Method\r\n  addRequiredValidation = () => {\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  //External Method\r\n  removeCustomValidation = (customValidation) => {\r\n    this.controlUtility.removeCustomValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      customValidation\r\n    );\r\n  };\r\n\r\n  //External Method\r\n  addCustomValidation = (customValidation) => {\r\n    this.controlUtility.addCustomValidation(\r\n      this.fileUploadFormControl,\r\n      this.validationRules,\r\n      customValidation\r\n    );\r\n  };\r\n\r\n  //External Method\r\n  isValid = () => {\r\n    this.controlUtility.isValid(this.fileUploadFormControl);\r\n  };\r\n\r\n  convertSizeToMB(size: number): number {\r\n    if (size === 0) {\r\n      return 0;\r\n    }\r\n\r\n    // Convert size to megabytes\r\n    const megabytes = size / (1024 * 1024);\r\n\r\n    // Round to two decimal places\r\n    const roundedMegabytes = Math.round(megabytes * 100) / 100;\r\n\r\n    return roundedMegabytes;\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    // Complete all subscriptions to prevent memory leaks\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n\r\n    // Clear uploader queue to free memory\r\n    if (this.uploader) {\r\n      this.uploader.clearQueue();\r\n    }\r\n  }\r\n\r\n  public sanitizeFileName(fileName: string): string {\r\n    // Remove potentially dangerous characters from file names\r\n    return fileName.replace(/[<>:\"/\\\\|?*]/g, '_').trim();\r\n  }\r\n\r\n\r\n  private findDuplicateFiles(): any[] {\r\n    const duplicateFiles: any[] = [];\r\n    const existingFileNames = new Set<string>();\r\n\r\n    // First pass: collect names of existing files (already in queue before this upload)\r\n    const existingFiles = this.uploader.queue.filter(item => item['some'].lastModified == null);\r\n    for (const item of existingFiles) {\r\n      const fileName = item.file.name.toLowerCase().trim();\r\n      existingFileNames.add(fileName);\r\n    }\r\n\r\n    // Second pass: check newly added files against existing file names\r\n    const newlyAddedFiles = this.uploader.queue.filter(item => item['some'].lastModified != null);\r\n\r\n    for (const item of newlyAddedFiles) {\r\n      const fileName = item.file.name.toLowerCase().trim();\r\n\r\n      // If this new file name already exists in the queue, it's a duplicate\r\n      if (existingFileNames.has(fileName)) {\r\n        duplicateFiles.push(item);\r\n      } else {\r\n        // Add this new file name to the set so subsequent files with same name are also flagged\r\n        existingFileNames.add(fileName);\r\n      }\r\n    }\r\n\r\n    return duplicateFiles;\r\n  }\r\n\r\n\r\n  private removeDuplicateFilesFromQueue(duplicateFiles: any[]): void {\r\n    for (const duplicateFile of duplicateFiles) {\r\n      const index = this.uploader.queue.indexOf(duplicateFile);\r\n      if (index > -1) {\r\n        this.uploader.queue.splice(index, 1);\r\n      }\r\n    }\r\n  }\r\n\r\n  private isFileTypeAccepted(fileType: string): boolean {\r\n    if (!this.acceptedTypeArray || this.acceptedTypeArray.length === 0) {\r\n      return true; // No restrictions\r\n    }\r\n\r\n    // Normalize the file type for comparison\r\n    const normalizedFileType = fileType.toLowerCase().trim();\r\n\r\n    // Check against all accepted types (case-insensitive and trimmed)\r\n    return this.acceptedTypeArray.some(acceptedType => {\r\n      const normalizedAcceptedType = acceptedType.toLowerCase().trim();\r\n      return normalizedAcceptedType === normalizedFileType;\r\n    });\r\n  }\r\n\r\n  private setValuePreservingErrors(value: any, preserveErrors?: any): void {\r\n    // Capture current duplicate errors before setValue clears them\r\n    const currentErrors = this.fileUploadFormControl.errors;\r\n    const duplicateError = currentErrors?.DuplicateFileError ? { DuplicateFileError: currentErrors.DuplicateFileError } : null;\r\n\r\n    // Set the form control value\r\n    this.fileUploadFormControl.setValue(value);\r\n    this.group.get(this.options.name).setValue(value);\r\n\r\n    // Apply errors in priority order: preserveErrors > duplicateError > pendingValidationErrors\r\n    const errorsToApply = preserveErrors || duplicateError || this.pendingValidationErrors;\r\n    if (errorsToApply) {\r\n      this.fileUploadFormControl.setErrors(errorsToApply);\r\n      this.fileUploadFormControl.markAsTouched();\r\n    }\r\n\r\n    // Update the options value for two-way binding\r\n    this.options.value = value;\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      <!--Asterisk-->\r\n      <span *ngIf=\"((options.showAsterisk && options.isRequired) || options.isRequired) && !options.isReadonly\"\r\n        class=\"text-danger\">*</span>\r\n    </label>\r\n    <!--Allow dropZone-->\r\n    <div ng2FileDrop class=\"bbsf-input-container {{ options.extraClasses }}\"\r\n      *ngIf=\"options.isDropZone && !(options.isMultipleFile == false && uploader.queue.length > 0) && !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      multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" aria-invalid=\"true\" type=\"file\"\r\n      [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\"\r\n      (click)=\"fileInputDropZone.click()\">\r\n      <div class=\"dropzone-label\">\r\n        <div class=\"svg-and-validation\">\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          <!--Validation text-->\r\n          <div class=\"bbsf-validation-msg validation-msg-header text-center\">\r\n            {{ UtilityService.getResourceValue('DragAndDropHere') }}\r\n          </div>\r\n          <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n          <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n            *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n            [innerHTML]=\"validationCountMessage\"></div>\r\n        </div>\r\n      </div>\r\n      <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n        class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n        multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n        (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n        aria-invalid=\"true\" #fileInputDropZone\r\n        [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n    </div>\r\n    <!--Not allowed dropZone-->\r\n    <div class=\"bbsf-input-container\" *ngIf=\"!options.isDropZone && !isHideInput() && !options.isReadonly\"\r\n      (click)=\"fileInput.click()\">\r\n      <div class=\"dropzone-label\">\r\n        <div class=\"svg-and-validation\">\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          <!--Validation text-->\r\n          <div class=\"bbsf-validation-msg text-center\">{{ UtilityService.getResourceValue('Upload') }}</div>\r\n          <div class=\"bbsf-validation-msg text-center\" *ngIf=\"validationMessage\" [innerHTML]=\"validationMessage\"></div>\r\n          <div class=\"bbsf-validation-msg ng-star-inserted text-center text-danger\"\r\n            *ngIf=\"validationCountMessage && options.isMultipleFile && options.maxNoOfFiles > 0\"\r\n            [innerHTML]=\"validationCountMessage\"></div>\r\n        </div>\r\n      </div>\r\n      <input ng2FileSelect [uploader]=\"uploader\" [accept]=\"acceptedType\"\r\n        class=\"fileSelector customFileUploadPlacment hidden v-required-multiplefiles d-none\"\r\n        multiple=\"{{ options.isMultipleFile ? 'multiple' : '' }}\" name=\"file\" type=\"file\" value=\"\" autocomplete=\"off\"\r\n        (change)=\"onFileChange()\" [ngClass]=\"options.viewType == 1 ? '' : 'col-md-9'\" id=\"{{ options.name }}\"\r\n        aria-invalid=\"true\" #fileInput\r\n        [class.is-invalid]=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\" />\r\n    </div>\r\n    <!-- readonly -->\r\n    <div *ngIf=\"options.isReadonly && !options.value\">\r\n      <span class=\"readonly-view\">{{ UtilityService.getResourceValue('NA') }}</span>\r\n    </div>\r\n  </div>\r\n  <!--items uploaded-->\r\n  <div class=\"uploaded-items\">\r\n    <div class=\"btn-group\" *ngFor=\"let item of uploader.queue\">\r\n      <ng-container *ngIf=\"item?.progress == 100 && options.isUploadFileAsync\">\r\n        <a *ngIf=\"item?.file?.rawFile['url']\" href=\"{{ item?.file?.rawFile['url'] }}\"\r\n          class=\"btn-download-file btn-sm btn-progress-upload\" download>\r\n          <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n        </a>\r\n        <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n          <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n        </a>\r\n        <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm\"\r\n          (click)=\"item.remove(); removeFromControlValue(item)\">\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      <ng-container *ngIf=\"!options.isUploadFileAsync\">\r\n        <a href=\"{{ item?.file?.rawFile['url'] }}\" *ngIf=\"item?.file?.rawFile['url']\" class=\"btn btn-download-file btn-sm\" download>\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\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n        </a>\r\n         <a *ngIf=\"item?.file?.rawFile['url'] == null\" class=\"btn-download-file btn-sm btn-progress-upload\">\r\n          <span class=\"file-name\">{{ sanitizeFileName(item?.file?.name || '') }}</span>\r\n        </a>\r\n        <button *ngIf=\"!options.isReadonly\" class=\"btn btn-download-file btn-sm btn-danger\"\r\n          (click)=\"item.remove(); removeFromControlValue(item)\">\r\n          <i class=\"fa fa-times px-0\"></i>\r\n        </button>\r\n      </ng-container>\r\n    </div>\r\n  </div>\r\n  <!--progress bar file upload-->\r\n  <div *ngFor=\"let item of uploader.queue\">\r\n    <div class=\"upload-items\" [ngClass]=\"{ 'mt-4': options.isMultipleFile == true }\"\r\n      *ngIf=\"item?.progress < 100 && options.isUploadFileAsync\">\r\n      <div class=\"upload-items-toolbar\">\r\n        <h4>{{ item?.file?.name }}</h4>\r\n        <span (click)=\"item.remove(); removeFromControlValue(item)\">\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        </span>\r\n      </div>\r\n      <div class=\"progress\">\r\n        <div class=\"progress-bar\" role=\"progressbar\" aria-valuenow=\"70\" aria-valuemin=\"0\" aria-valuemax=\"100\"\r\n          [class.file-uploaded]=\"item?.progress < 100\" [style.width.%]=\"item?.progress\" *ngIf=\"item?.progress > 0\">\r\n        </div>\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n  <div class=\"subtext-container\" *ngIf=\"!options.isReadonly\">\r\n    <!-- required text-->\r\n    <div class=\"bbsf-validation\" *ngIf=\"fileUploadFormControl.invalid && fileUploadFormControl.touched\">\r\n      {{ getErrorValidation(fileUploadFormControl.errors | keyvalue) }}\r\n    </div>\r\n    <!-- LabelDescription-->\r\n    <div class=\"bbsf-control-desc\" *ngIf=\"options.labelDescription != null\">{{ options.labelDescription }}</div>\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>"]}
|