@muraai/mnl-form 0.0.1-alpha-db495b1 → 0.0.1-alpha-6116399
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/fesm2022/muraai-mnl-form.mjs +150 -143
- package/fesm2022/muraai-mnl-form.mjs.map +1 -1
- package/lib/types/upload-file.d.ts +1 -1
- package/lib/workflow-status/ workflow-status.module.d.ts +8 -8
- package/lib/workflow-status/workflow-status.component.d.ts +1 -1
- package/package.json +4 -5
- package/esm2022/lib/confirmation-dialog/confirmation-dialog.component.mjs +0 -34
- package/esm2022/lib/mnl-form/mnl-form-standalone.module.mjs +0 -155
- package/esm2022/lib/mnl-form/mnl-form.component.mjs +0 -199
- package/esm2022/lib/models/custom-form-field.model.mjs +0 -2
- package/esm2022/lib/models/workflow-status.model.mjs +0 -2
- package/esm2022/lib/services/graphql.service.mjs +0 -30
- package/esm2022/lib/services/scoring.service.mjs +0 -103
- package/esm2022/lib/types/action-button-renderer.component.mjs +0 -59
- package/esm2022/lib/types/autocomplete-type.component.mjs +0 -68
- package/esm2022/lib/types/button.type.mjs +0 -72
- package/esm2022/lib/types/comments.type.mjs +0 -102
- package/esm2022/lib/types/country-code-input.type.mjs +0 -181
- package/esm2022/lib/types/custom-date.type.mjs +0 -20
- package/esm2022/lib/types/custom-input.component.mjs +0 -62
- package/esm2022/lib/types/date-format.type.mjs +0 -196
- package/esm2022/lib/types/expansion-panel.types.mjs +0 -85
- package/esm2022/lib/types/formly-field-panel.type.mjs +0 -70
- package/esm2022/lib/types/grid-actionable.type.mjs +0 -197
- package/esm2022/lib/types/grid-formly-cell.component.mjs +0 -42
- package/esm2022/lib/types/grid.type.mjs +0 -125
- package/esm2022/lib/types/input.type.mjs +0 -43
- package/esm2022/lib/types/mu-helper-text.type.mjs +0 -331
- package/esm2022/lib/types/multi-select-autocomplete.mjs +0 -139
- package/esm2022/lib/types/multi-select-checkbox.component.mjs +0 -247
- package/esm2022/lib/types/radio.type.mjs +0 -71
- package/esm2022/lib/types/scrollable-tabs.types.mjs +0 -220
- package/esm2022/lib/types/select-api-auto-complete.component.mjs +0 -111
- package/esm2022/lib/types/select-autocomplete-redefined.component.mjs +0 -142
- package/esm2022/lib/types/select-autocomplete.type.mjs +0 -254
- package/esm2022/lib/types/stepper.type.mjs +0 -268
- package/esm2022/lib/types/table.types.mjs +0 -79
- package/esm2022/lib/types/tabs.types.mjs +0 -293
- package/esm2022/lib/types/tabular-form-dialog.component.mjs +0 -331
- package/esm2022/lib/types/tabular-form.type.mjs +0 -641
- package/esm2022/lib/types/upload-file.mjs +0 -524
- package/esm2022/lib/types/view-file-dialog.component.mjs +0 -90
- package/esm2022/lib/utils/formly.utils.mjs +0 -105
- package/esm2022/lib/validators/custom-email-validator.mjs +0 -18
- package/esm2022/lib/validators/ein-validators.mjs +0 -11
- package/esm2022/lib/validators/gst-validator.mjs +0 -11
- package/esm2022/lib/validators/normal-email-validator.mjs +0 -17
- package/esm2022/lib/validators/npi-validator.mjs +0 -11
- package/esm2022/lib/validators/pan-validator.mjs +0 -11
- package/esm2022/lib/validators/phone-number-validator.mjs +0 -22
- package/esm2022/lib/validators/select-validator.mjs +0 -7
- package/esm2022/lib/validators/tinSsn-validator.mjs +0 -12
- package/esm2022/lib/workflow-status/ workflow-status.module.mjs +0 -52
- package/esm2022/lib/workflow-status/workflow-status.component.mjs +0 -70
- package/esm2022/lib/wrappers/form-field-description-wrapper.component.mjs +0 -44
- package/esm2022/lib/wrappers/form-field-wrapper.component.mjs +0 -34
- package/esm2022/lib/wrappers/formly-wrapper-with-old-value.component.mjs +0 -140
- package/esm2022/lib/wrappers/prefix-suffix-wrapper.component.mjs +0 -95
- package/esm2022/lib/wrappers/prefix-sufix-extension.component.mjs +0 -9
- package/esm2022/muraai-mnl-form.mjs +0 -5
- package/esm2022/public-api.mjs +0 -12
|
@@ -1,524 +0,0 @@
|
|
|
1
|
-
import { Breakpoints } from '@angular/cdk/layout';
|
|
2
|
-
import { CommonModule, DatePipe } from '@angular/common';
|
|
3
|
-
import { ChangeDetectionStrategy, Component, DestroyRef, inject, } from '@angular/core';
|
|
4
|
-
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
5
|
-
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
6
|
-
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
7
|
-
import { MatIconModule } from '@angular/material/icon';
|
|
8
|
-
import { MatInputModule } from '@angular/material/input';
|
|
9
|
-
import { MatSnackBarModule } from '@angular/material/snack-bar';
|
|
10
|
-
import { MatTableDataSource, MatTableModule } from '@angular/material/table';
|
|
11
|
-
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
12
|
-
import { FieldType, FormlyModule } from '@ngx-formly/core';
|
|
13
|
-
import { TranslateModule } from '@ngx-translate/core';
|
|
14
|
-
import { Observable } from 'rxjs';
|
|
15
|
-
import * as i0 from "@angular/core";
|
|
16
|
-
import * as i1 from "@angular/material/snack-bar";
|
|
17
|
-
import * as i2 from "@ngx-translate/core";
|
|
18
|
-
import * as i3 from "@angular/cdk/layout";
|
|
19
|
-
import * as i4 from "@ngx-formly/core";
|
|
20
|
-
import * as i5 from "@angular/common";
|
|
21
|
-
import * as i6 from "@angular/material/table";
|
|
22
|
-
import * as i7 from "@angular/material/icon";
|
|
23
|
-
import * as i8 from "@angular/material/tooltip";
|
|
24
|
-
export class FormlyFileUploadComponent extends FieldType {
|
|
25
|
-
constructor(snackBar, translateService, breakpointObserver, cdr) {
|
|
26
|
-
super();
|
|
27
|
-
this.snackBar = snackBar;
|
|
28
|
-
this.translateService = translateService;
|
|
29
|
-
this.breakpointObserver = breakpointObserver;
|
|
30
|
-
this.cdr = cdr;
|
|
31
|
-
this.uploadedFiles = [];
|
|
32
|
-
this.displayedColumns = [
|
|
33
|
-
'fileName',
|
|
34
|
-
'convertedFileSize',
|
|
35
|
-
'uploadedBy',
|
|
36
|
-
'uploadedAt',
|
|
37
|
-
'actions',
|
|
38
|
-
];
|
|
39
|
-
this.dataSource = new MatTableDataSource();
|
|
40
|
-
this.showUploadbtn = true;
|
|
41
|
-
this.destroyRef = inject(DestroyRef);
|
|
42
|
-
this.isMobileDevice = false;
|
|
43
|
-
this.breakpointObserver
|
|
44
|
-
?.observe([Breakpoints.Handset])
|
|
45
|
-
.pipe(takeUntilDestroyed(this.destroyRef))
|
|
46
|
-
.subscribe((result) => {
|
|
47
|
-
this.isMobileDevice = result?.matches;
|
|
48
|
-
});
|
|
49
|
-
}
|
|
50
|
-
ngOnInit() {
|
|
51
|
-
const userDetails = localStorage.getItem('loggedInUserDetails');
|
|
52
|
-
this.loggedInUserDetails = userDetails
|
|
53
|
-
? JSON.parse(userDetails)
|
|
54
|
-
: { email: '' };
|
|
55
|
-
if (this.field.props?.['isMultiFileUpload'] && this.formControl.value) {
|
|
56
|
-
this.uploadedFiles = this.formControl.value ?? [];
|
|
57
|
-
}
|
|
58
|
-
else {
|
|
59
|
-
if (this.formControl.value &&
|
|
60
|
-
Object.keys(this.formControl.value || {}).length) {
|
|
61
|
-
this.uploadedFiles = [this.formControl.value];
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
this.dataSource.data = [...this.uploadedFiles];
|
|
65
|
-
this.showUploadbtn = this.props?.['isMultiFileUpload']
|
|
66
|
-
? true
|
|
67
|
-
: !this.uploadedFiles.length;
|
|
68
|
-
}
|
|
69
|
-
onFileSelect(event) {
|
|
70
|
-
const selectedFile = event.target.files;
|
|
71
|
-
if (!selectedFile || !selectedFile.length)
|
|
72
|
-
return;
|
|
73
|
-
const maxFileSizeMB = this.field.props?.['maxFileSizeMB'] || 1;
|
|
74
|
-
const maxFileSize = maxFileSizeMB * 1024 * 1024;
|
|
75
|
-
if (this.props?.['isMultiFileUpload']) {
|
|
76
|
-
if (this.isFileAlreadyUploaded(event.target.files)) {
|
|
77
|
-
this.snackBar.open(this.translateService.instant('MNL_UPLOAD_FILE.DUPLICATE_UPLOADED'), 'Ok', {
|
|
78
|
-
duration: 3000,
|
|
79
|
-
panelClass: ['warn-snackbar'],
|
|
80
|
-
});
|
|
81
|
-
this.resetFileInput(event.target);
|
|
82
|
-
return;
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
const selectedFiles = Array.from(selectedFile);
|
|
86
|
-
const invalidChars = /[&,"]/;
|
|
87
|
-
const filesToProcess = this.field.props?.['isMultiFileUpload']
|
|
88
|
-
? selectedFiles
|
|
89
|
-
: [selectedFiles[0]];
|
|
90
|
-
filesToProcess.forEach((file) => {
|
|
91
|
-
if (file.size > maxFileSize) {
|
|
92
|
-
this.snackBar.open(`The file "${file.name}" exceeds the ${maxFileSizeMB} MB size limit.`, 'OK', {
|
|
93
|
-
duration: 3000,
|
|
94
|
-
panelClass: ['warn-snackbar'],
|
|
95
|
-
});
|
|
96
|
-
}
|
|
97
|
-
else if (invalidChars.test(file.name)) {
|
|
98
|
-
this.snackBar.open('MNL_UPLOAD_FILE.INVALID_CHARS_FILE_NAME', 'Ok', {
|
|
99
|
-
duration: 7000,
|
|
100
|
-
panelClass: ['error-snackbar'],
|
|
101
|
-
});
|
|
102
|
-
}
|
|
103
|
-
else {
|
|
104
|
-
const newFile = {
|
|
105
|
-
fileName: file.name,
|
|
106
|
-
fileSize: file.size,
|
|
107
|
-
fileType: file.type,
|
|
108
|
-
file: file,
|
|
109
|
-
uploadedBy: this.loggedInUserDetails.email,
|
|
110
|
-
uploadedAt: new Date(),
|
|
111
|
-
nodeId: '',
|
|
112
|
-
convertedFileSize: this.formatFileSize(file.size),
|
|
113
|
-
};
|
|
114
|
-
if (!this.props?.['isMultiFileUpload']) {
|
|
115
|
-
this.uploadedFiles = [newFile];
|
|
116
|
-
}
|
|
117
|
-
else {
|
|
118
|
-
this.uploadedFiles.push(newFile);
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
});
|
|
122
|
-
this.dataSource.data = [...this.uploadedFiles];
|
|
123
|
-
this.resetFileInput(event.target);
|
|
124
|
-
this.showUploadbtn = this.props?.['isMultiFileUpload']
|
|
125
|
-
? true
|
|
126
|
-
: !this.uploadedFiles.length;
|
|
127
|
-
if (this.field.props?.['onFilesAdded']) {
|
|
128
|
-
this.field.props['onFilesAdded'](this.uploadedFiles);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
handleActionClick(action, index, element) {
|
|
132
|
-
const onUploadFileAction = this.field.props?.['onUploadFileAction'];
|
|
133
|
-
if (typeof onUploadFileAction === 'function') {
|
|
134
|
-
const result = onUploadFileAction(action.icon, element, index);
|
|
135
|
-
this.invokeAction(() => result, element, index, () => {
|
|
136
|
-
if (action.icon === 'delete')
|
|
137
|
-
this.removeFile(index);
|
|
138
|
-
if (action.icon === 'cloud_upload')
|
|
139
|
-
this.refreshUploadedTable(element, index);
|
|
140
|
-
if (action.icon === 'cloud_download' || action.icon === 'visibility')
|
|
141
|
-
() => { };
|
|
142
|
-
}, action.icon);
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
refreshUploadedTable(element, index) {
|
|
146
|
-
this.uploadedFiles[index] = element;
|
|
147
|
-
if (this.props?.['isMultiFileUpload']) {
|
|
148
|
-
const uploaded = this.uploadedFiles.filter((f) => !!f.nodeId);
|
|
149
|
-
this.formControl.setValue(uploaded);
|
|
150
|
-
}
|
|
151
|
-
else {
|
|
152
|
-
this.formControl.setValue(this.uploadedFiles[0]);
|
|
153
|
-
}
|
|
154
|
-
this.formControl.markAsTouched();
|
|
155
|
-
this.form.updateValueAndValidity();
|
|
156
|
-
this.dataSource.data = [...this.uploadedFiles];
|
|
157
|
-
this.cdr.detectChanges();
|
|
158
|
-
}
|
|
159
|
-
removeFile(index) {
|
|
160
|
-
this.uploadedFiles.splice(index, 1);
|
|
161
|
-
this.dataSource.data = [...this.uploadedFiles];
|
|
162
|
-
if (!this.field.props?.['isMultiFileUpload']) {
|
|
163
|
-
this.formControl.setValue(null);
|
|
164
|
-
}
|
|
165
|
-
else {
|
|
166
|
-
this.formControl.setValue(this.uploadedFiles);
|
|
167
|
-
const uploaded = this.uploadedFiles.filter((f) => !!f.nodeId);
|
|
168
|
-
this.formControl.setValue(uploaded);
|
|
169
|
-
}
|
|
170
|
-
this.showUploadbtn = this.props?.['isMultiFileUpload']
|
|
171
|
-
? true
|
|
172
|
-
: !this.uploadedFiles.length;
|
|
173
|
-
}
|
|
174
|
-
isFileAlreadyUploaded(selectedFile) {
|
|
175
|
-
return this.uploadedFiles.some((file) => file.fileName === selectedFile[0].name);
|
|
176
|
-
}
|
|
177
|
-
resetFileInput(fileInput) {
|
|
178
|
-
fileInput.value = '';
|
|
179
|
-
}
|
|
180
|
-
showUploadAction(action, file) {
|
|
181
|
-
const hasNode = !!file.nodeId;
|
|
182
|
-
const userRoles = this.options.formState?.userRoles ?? [];
|
|
183
|
-
const enabledRoles = this.field?.enabledForRoles ?? [];
|
|
184
|
-
switch (action.icon) {
|
|
185
|
-
case 'cloud_upload':
|
|
186
|
-
return !hasNode;
|
|
187
|
-
case 'cloud_download':
|
|
188
|
-
case 'visibility':
|
|
189
|
-
return hasNode;
|
|
190
|
-
case 'delete':
|
|
191
|
-
return (hasNode && userRoles.some((role) => enabledRoles.includes(role)));
|
|
192
|
-
default:
|
|
193
|
-
return false;
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
formatFileSize(bytes) {
|
|
197
|
-
if (bytes < 1024)
|
|
198
|
-
return bytes + ' B';
|
|
199
|
-
if (bytes < 1048576)
|
|
200
|
-
return (bytes / 1024).toFixed(2) + ' KB';
|
|
201
|
-
return (bytes / 1048576).toFixed(2) + ' MB';
|
|
202
|
-
}
|
|
203
|
-
invokeAction(actionCallback, element, index, defaultFn, icon) {
|
|
204
|
-
if (!actionCallback) {
|
|
205
|
-
defaultFn();
|
|
206
|
-
return;
|
|
207
|
-
}
|
|
208
|
-
const result = actionCallback(element, index);
|
|
209
|
-
if (result instanceof Observable) {
|
|
210
|
-
result.subscribe((shouldProceed) => {
|
|
211
|
-
if (shouldProceed) {
|
|
212
|
-
this.successMessage(icon);
|
|
213
|
-
defaultFn();
|
|
214
|
-
}
|
|
215
|
-
});
|
|
216
|
-
}
|
|
217
|
-
else {
|
|
218
|
-
Promise.resolve(result).then((shouldProceed) => {
|
|
219
|
-
if (shouldProceed) {
|
|
220
|
-
this.successMessage(icon);
|
|
221
|
-
defaultFn();
|
|
222
|
-
}
|
|
223
|
-
});
|
|
224
|
-
}
|
|
225
|
-
}
|
|
226
|
-
successMessage(icon) {
|
|
227
|
-
if (icon === 'cloud_upload') {
|
|
228
|
-
this.snackBar.open(this.translateService.instant('MNL_UPLOAD_FILE.FILE_SUCCESSFUL_UPLOADED'), 'Ok', {
|
|
229
|
-
duration: 5000,
|
|
230
|
-
panelClass: ['success-snackbar'],
|
|
231
|
-
});
|
|
232
|
-
}
|
|
233
|
-
else if (icon === 'delete') {
|
|
234
|
-
this.snackBar.open(this.translateService.instant('MNL_UPLOAD_FILE.FILE_SUCCESSFUL_DELETED'), 'Ok', {
|
|
235
|
-
duration: 5000,
|
|
236
|
-
panelClass: ['success-snackbar'],
|
|
237
|
-
});
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyFileUploadComponent, deps: [{ token: i1.MatSnackBar }, { token: i2.TranslateService }, { token: i3.BreakpointObserver }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
241
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: FormlyFileUploadComponent, isStandalone: true, selector: "mu-formly-field-file", usesInheritance: true, ngImport: i0, template: `
|
|
242
|
-
@if(showUploadbtn) {
|
|
243
|
-
<div
|
|
244
|
-
class="input-btn"
|
|
245
|
-
[ngClass]="{ 'red-upload': !field.formControl?.valid }"
|
|
246
|
-
>
|
|
247
|
-
<button
|
|
248
|
-
type="button"
|
|
249
|
-
[disabled]="field.props?.['disabled']"
|
|
250
|
-
[ngClass]="[field.props?.['color']]"
|
|
251
|
-
(click)="fileInput.click()"
|
|
252
|
-
>
|
|
253
|
-
<mat-icon
|
|
254
|
-
*ngIf="field.props?.['icon']"
|
|
255
|
-
class="text-lg mb-1"
|
|
256
|
-
>{{ field.props?.['icon'] }}</mat-icon
|
|
257
|
-
>
|
|
258
|
-
{{ field.props?.label }}<span *ngIf="to.required">*</span>
|
|
259
|
-
</button>
|
|
260
|
-
<input
|
|
261
|
-
type="file"
|
|
262
|
-
#fileInput
|
|
263
|
-
(change)="onFileSelect($event)"
|
|
264
|
-
[formlyAttributes]="field"
|
|
265
|
-
[multiple]="field.props?.['isMultiFileUpload']"
|
|
266
|
-
[accept]="field.props?.['acceptedFileTypes'] || '.pdf,.xlsx,.jpg,.jpeg,.docx'"
|
|
267
|
-
class="hidden"
|
|
268
|
-
/>
|
|
269
|
-
<div class="text-[0.83rem] text-gray-500 mt-1">
|
|
270
|
-
Allowed file types:
|
|
271
|
-
{{ field.props?.['acceptedFileTypes'] || '.pdf, .xlsx, .jpg, jpeg, .docx' }}
|
|
272
|
-
<br />
|
|
273
|
-
Maximum file size: {{ field.props?.['maxFileSizeMB'] || 1 }}MB
|
|
274
|
-
</div>
|
|
275
|
-
</div>
|
|
276
|
-
}
|
|
277
|
-
<table
|
|
278
|
-
mat-table
|
|
279
|
-
*ngIf="this.props?.['isMultiFileUpload'] && uploadedFiles?.length"
|
|
280
|
-
[dataSource]="dataSource"
|
|
281
|
-
class="mat-elevation-z8"
|
|
282
|
-
>
|
|
283
|
-
<ng-container matColumnDef="fileName">
|
|
284
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">File Name</th>
|
|
285
|
-
<td mat-cell *matCellDef="let element">{{ element.fileName }}</td>
|
|
286
|
-
</ng-container>
|
|
287
|
-
|
|
288
|
-
<ng-container matColumnDef="convertedFileSize">
|
|
289
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">Size</th>
|
|
290
|
-
<td mat-cell *matCellDef="let element">
|
|
291
|
-
{{ element.convertedFileSize }}
|
|
292
|
-
</td>
|
|
293
|
-
</ng-container>
|
|
294
|
-
|
|
295
|
-
<ng-container matColumnDef="uploadedBy">
|
|
296
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">
|
|
297
|
-
Uploaded By
|
|
298
|
-
</th>
|
|
299
|
-
<td mat-cell *matCellDef="let element">{{ element.uploadedBy }}</td>
|
|
300
|
-
</ng-container>
|
|
301
|
-
|
|
302
|
-
<ng-container matColumnDef="uploadedAt">
|
|
303
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">
|
|
304
|
-
Uploaded Date
|
|
305
|
-
</th>
|
|
306
|
-
<td mat-cell *matCellDef="let element">
|
|
307
|
-
{{ element.uploadedAt | date : 'dd-MMM-yyyy, hh:mm:ss a' }}
|
|
308
|
-
</td>
|
|
309
|
-
</ng-container>
|
|
310
|
-
|
|
311
|
-
<ng-container matColumnDef="actions">
|
|
312
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">Actions</th>
|
|
313
|
-
<td mat-cell *matCellDef="let element; let i = index">
|
|
314
|
-
<div class="action-buttons">
|
|
315
|
-
<div
|
|
316
|
-
*ngFor="let action of field?.props?.['actions']"
|
|
317
|
-
class="actions-container"
|
|
318
|
-
>
|
|
319
|
-
<mat-icon
|
|
320
|
-
class="action-icons"
|
|
321
|
-
*ngIf="showUploadAction(action, element)"
|
|
322
|
-
matTooltip="{{ action.label | translate }}"
|
|
323
|
-
(click)="handleActionClick(action, i, element)"
|
|
324
|
-
>
|
|
325
|
-
{{ action.icon }}
|
|
326
|
-
</mat-icon>
|
|
327
|
-
</div>
|
|
328
|
-
</div>
|
|
329
|
-
</td>
|
|
330
|
-
</ng-container>
|
|
331
|
-
|
|
332
|
-
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
|
333
|
-
<tr mat-row *matRowDef="let row; columns: displayedColumns"></tr>
|
|
334
|
-
</table>
|
|
335
|
-
|
|
336
|
-
<div class="w-full">
|
|
337
|
-
<table
|
|
338
|
-
*ngIf="!this.props?.['isMultiFileUpload'] && uploadedFiles?.length"
|
|
339
|
-
class="w-full flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4"
|
|
340
|
-
>
|
|
341
|
-
<tr
|
|
342
|
-
class="flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4 w-full"
|
|
343
|
-
>
|
|
344
|
-
<td
|
|
345
|
-
class="text-sm truncate overflow-hidden whitespace-nowrap max-w-full flex-1 md:border-none"
|
|
346
|
-
[matTooltip]="uploadedFiles[0].fileName"
|
|
347
|
-
>
|
|
348
|
-
{{ uploadedFiles[0].fileName }}
|
|
349
|
-
</td>
|
|
350
|
-
<td class="flex items-center justify-end md:flex-shrink-0">
|
|
351
|
-
<div
|
|
352
|
-
*ngFor="let action of field?.props?.['actions']"
|
|
353
|
-
class="flex items-center"
|
|
354
|
-
>
|
|
355
|
-
<mat-icon
|
|
356
|
-
matTooltip="{{ action.label | translate }}"
|
|
357
|
-
*ngIf="showUploadAction(action, uploadedFiles[0])"
|
|
358
|
-
(click)="handleActionClick(action, 0, uploadedFiles[0])"
|
|
359
|
-
class="action-icons"
|
|
360
|
-
>
|
|
361
|
-
{{ action.icon }}
|
|
362
|
-
</mat-icon>
|
|
363
|
-
</div>
|
|
364
|
-
</td>
|
|
365
|
-
</tr>
|
|
366
|
-
</table>
|
|
367
|
-
</div>
|
|
368
|
-
<div
|
|
369
|
-
*ngIf="to['note']"
|
|
370
|
-
class="text-gray-500 mt-1 w-[900px]"
|
|
371
|
-
[innerHTML]="to['note']"
|
|
372
|
-
></div>
|
|
373
|
-
`, isInline: true, styles: [".mat-mdc-table{margin-top:5px;box-shadow:none}.header-cell{background-color:#7876761f!important}.action-buttons{display:flex;flex-direction:row;justify-content:start;align-items:center}.action-icons{font-size:20px}.actions-container{display:flex;justify-content:center;align-items:center;height:43px;padding-top:5px}.input-btn button,.red-upload button{color:#fff;padding:3px 10px;border:none;cursor:pointer;border-radius:50px;font-family:Poppins,Helvetica Neue,sans-serif;font-weight:500;font-size:14px;display:flex;align-items:center;justify-content:center}.input-btn button:disabled,.red-upload button:disabled{background-color:#d3d3d3;color:#b0b0b0}th,td,.upload-btn{font-family:Poppins,Helvetica Neue,sans-serif}.red-upload button{background-color:red}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatInputModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FormlyModule }, { kind: "directive", type: i4.ɵFormlyAttributes, selector: "[formlyAttributes]", inputs: ["formlyAttributes", "id"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i5.DatePipe, name: "date" }, { kind: "ngmodule", type: MatTableModule }, { kind: "component", type: i6.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i6.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i6.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i6.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i6.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i6.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i6.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i6.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "component", type: i6.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i6.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i8.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: MatSnackBarModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
374
|
-
}
|
|
375
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyFileUploadComponent, decorators: [{
|
|
376
|
-
type: Component,
|
|
377
|
-
args: [{ selector: 'mu-formly-field-file', standalone: true, imports: [
|
|
378
|
-
FormsModule,
|
|
379
|
-
MatFormFieldModule,
|
|
380
|
-
MatInputModule,
|
|
381
|
-
ReactiveFormsModule,
|
|
382
|
-
FormlyModule,
|
|
383
|
-
CommonModule,
|
|
384
|
-
MatTableModule,
|
|
385
|
-
MatIconModule,
|
|
386
|
-
MatTooltipModule,
|
|
387
|
-
TranslateModule,
|
|
388
|
-
MatSnackBarModule,
|
|
389
|
-
DatePipe,
|
|
390
|
-
], template: `
|
|
391
|
-
@if(showUploadbtn) {
|
|
392
|
-
<div
|
|
393
|
-
class="input-btn"
|
|
394
|
-
[ngClass]="{ 'red-upload': !field.formControl?.valid }"
|
|
395
|
-
>
|
|
396
|
-
<button
|
|
397
|
-
type="button"
|
|
398
|
-
[disabled]="field.props?.['disabled']"
|
|
399
|
-
[ngClass]="[field.props?.['color']]"
|
|
400
|
-
(click)="fileInput.click()"
|
|
401
|
-
>
|
|
402
|
-
<mat-icon
|
|
403
|
-
*ngIf="field.props?.['icon']"
|
|
404
|
-
class="text-lg mb-1"
|
|
405
|
-
>{{ field.props?.['icon'] }}</mat-icon
|
|
406
|
-
>
|
|
407
|
-
{{ field.props?.label }}<span *ngIf="to.required">*</span>
|
|
408
|
-
</button>
|
|
409
|
-
<input
|
|
410
|
-
type="file"
|
|
411
|
-
#fileInput
|
|
412
|
-
(change)="onFileSelect($event)"
|
|
413
|
-
[formlyAttributes]="field"
|
|
414
|
-
[multiple]="field.props?.['isMultiFileUpload']"
|
|
415
|
-
[accept]="field.props?.['acceptedFileTypes'] || '.pdf,.xlsx,.jpg,.jpeg,.docx'"
|
|
416
|
-
class="hidden"
|
|
417
|
-
/>
|
|
418
|
-
<div class="text-[0.83rem] text-gray-500 mt-1">
|
|
419
|
-
Allowed file types:
|
|
420
|
-
{{ field.props?.['acceptedFileTypes'] || '.pdf, .xlsx, .jpg, jpeg, .docx' }}
|
|
421
|
-
<br />
|
|
422
|
-
Maximum file size: {{ field.props?.['maxFileSizeMB'] || 1 }}MB
|
|
423
|
-
</div>
|
|
424
|
-
</div>
|
|
425
|
-
}
|
|
426
|
-
<table
|
|
427
|
-
mat-table
|
|
428
|
-
*ngIf="this.props?.['isMultiFileUpload'] && uploadedFiles?.length"
|
|
429
|
-
[dataSource]="dataSource"
|
|
430
|
-
class="mat-elevation-z8"
|
|
431
|
-
>
|
|
432
|
-
<ng-container matColumnDef="fileName">
|
|
433
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">File Name</th>
|
|
434
|
-
<td mat-cell *matCellDef="let element">{{ element.fileName }}</td>
|
|
435
|
-
</ng-container>
|
|
436
|
-
|
|
437
|
-
<ng-container matColumnDef="convertedFileSize">
|
|
438
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">Size</th>
|
|
439
|
-
<td mat-cell *matCellDef="let element">
|
|
440
|
-
{{ element.convertedFileSize }}
|
|
441
|
-
</td>
|
|
442
|
-
</ng-container>
|
|
443
|
-
|
|
444
|
-
<ng-container matColumnDef="uploadedBy">
|
|
445
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">
|
|
446
|
-
Uploaded By
|
|
447
|
-
</th>
|
|
448
|
-
<td mat-cell *matCellDef="let element">{{ element.uploadedBy }}</td>
|
|
449
|
-
</ng-container>
|
|
450
|
-
|
|
451
|
-
<ng-container matColumnDef="uploadedAt">
|
|
452
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">
|
|
453
|
-
Uploaded Date
|
|
454
|
-
</th>
|
|
455
|
-
<td mat-cell *matCellDef="let element">
|
|
456
|
-
{{ element.uploadedAt | date : 'dd-MMM-yyyy, hh:mm:ss a' }}
|
|
457
|
-
</td>
|
|
458
|
-
</ng-container>
|
|
459
|
-
|
|
460
|
-
<ng-container matColumnDef="actions">
|
|
461
|
-
<th mat-header-cell *matHeaderCellDef class="header-cell">Actions</th>
|
|
462
|
-
<td mat-cell *matCellDef="let element; let i = index">
|
|
463
|
-
<div class="action-buttons">
|
|
464
|
-
<div
|
|
465
|
-
*ngFor="let action of field?.props?.['actions']"
|
|
466
|
-
class="actions-container"
|
|
467
|
-
>
|
|
468
|
-
<mat-icon
|
|
469
|
-
class="action-icons"
|
|
470
|
-
*ngIf="showUploadAction(action, element)"
|
|
471
|
-
matTooltip="{{ action.label | translate }}"
|
|
472
|
-
(click)="handleActionClick(action, i, element)"
|
|
473
|
-
>
|
|
474
|
-
{{ action.icon }}
|
|
475
|
-
</mat-icon>
|
|
476
|
-
</div>
|
|
477
|
-
</div>
|
|
478
|
-
</td>
|
|
479
|
-
</ng-container>
|
|
480
|
-
|
|
481
|
-
<tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
|
|
482
|
-
<tr mat-row *matRowDef="let row; columns: displayedColumns"></tr>
|
|
483
|
-
</table>
|
|
484
|
-
|
|
485
|
-
<div class="w-full">
|
|
486
|
-
<table
|
|
487
|
-
*ngIf="!this.props?.['isMultiFileUpload'] && uploadedFiles?.length"
|
|
488
|
-
class="w-full flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4"
|
|
489
|
-
>
|
|
490
|
-
<tr
|
|
491
|
-
class="flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4 w-full"
|
|
492
|
-
>
|
|
493
|
-
<td
|
|
494
|
-
class="text-sm truncate overflow-hidden whitespace-nowrap max-w-full flex-1 md:border-none"
|
|
495
|
-
[matTooltip]="uploadedFiles[0].fileName"
|
|
496
|
-
>
|
|
497
|
-
{{ uploadedFiles[0].fileName }}
|
|
498
|
-
</td>
|
|
499
|
-
<td class="flex items-center justify-end md:flex-shrink-0">
|
|
500
|
-
<div
|
|
501
|
-
*ngFor="let action of field?.props?.['actions']"
|
|
502
|
-
class="flex items-center"
|
|
503
|
-
>
|
|
504
|
-
<mat-icon
|
|
505
|
-
matTooltip="{{ action.label | translate }}"
|
|
506
|
-
*ngIf="showUploadAction(action, uploadedFiles[0])"
|
|
507
|
-
(click)="handleActionClick(action, 0, uploadedFiles[0])"
|
|
508
|
-
class="action-icons"
|
|
509
|
-
>
|
|
510
|
-
{{ action.icon }}
|
|
511
|
-
</mat-icon>
|
|
512
|
-
</div>
|
|
513
|
-
</td>
|
|
514
|
-
</tr>
|
|
515
|
-
</table>
|
|
516
|
-
</div>
|
|
517
|
-
<div
|
|
518
|
-
*ngIf="to['note']"
|
|
519
|
-
class="text-gray-500 mt-1 w-[900px]"
|
|
520
|
-
[innerHTML]="to['note']"
|
|
521
|
-
></div>
|
|
522
|
-
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [".mat-mdc-table{margin-top:5px;box-shadow:none}.header-cell{background-color:#7876761f!important}.action-buttons{display:flex;flex-direction:row;justify-content:start;align-items:center}.action-icons{font-size:20px}.actions-container{display:flex;justify-content:center;align-items:center;height:43px;padding-top:5px}.input-btn button,.red-upload button{color:#fff;padding:3px 10px;border:none;cursor:pointer;border-radius:50px;font-family:Poppins,Helvetica Neue,sans-serif;font-weight:500;font-size:14px;display:flex;align-items:center;justify-content:center}.input-btn button:disabled,.red-upload button:disabled{background-color:#d3d3d3;color:#b0b0b0}th,td,.upload-btn{font-family:Poppins,Helvetica Neue,sans-serif}.red-upload button{background-color:red}\n"] }]
|
|
523
|
-
}], ctorParameters: () => [{ type: i1.MatSnackBar }, { type: i2.TranslateService }, { type: i3.BreakpointObserver }, { type: i0.ChangeDetectorRef }] });
|
|
524
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"upload-file.js","sourceRoot":"","sources":["../../../../../../libs/form/src/lib/types/upload-file.ts"],"names":[],"mappings":"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","sourcesContent":["import { BreakpointObserver, Breakpoints } from '@angular/cdk/layout';\nimport { CommonModule, DatePipe } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  DestroyRef,\n  inject,\n  OnInit,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\nimport { MatTableDataSource, MatTableModule } from '@angular/material/table';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { FieldType, FormlyModule } from '@ngx-formly/core';\nimport { TranslateModule, TranslateService } from '@ngx-translate/core';\nimport { Observable } from 'rxjs';\nimport {\n  CustomFormlyFieldConfig,\n  FileAction,\n  MaybeAsyncBool,\n} from '../models/custom-form-field.model';\nimport { Action } from 'rxjs/internal/scheduler/Action';\n\n@Component({\n  selector: 'mu-formly-field-file',\n  standalone: true,\n  imports: [\n    FormsModule,\n    MatFormFieldModule,\n    MatInputModule,\n    ReactiveFormsModule,\n    FormlyModule,\n    CommonModule,\n    MatTableModule,\n    MatIconModule,\n    MatTooltipModule,\n    TranslateModule,\n    MatSnackBarModule,\n    DatePipe,\n  ],\n  template: `\n    @if(showUploadbtn) {\n    <div\n      class=\"input-btn\"\n      [ngClass]=\"{ 'red-upload': !field.formControl?.valid }\"\n    >\n      <button\n        type=\"button\"\n        [disabled]=\"field.props?.['disabled']\"\n        [ngClass]=\"[field.props?.['color']]\"\n        (click)=\"fileInput.click()\"\n      >\n        <mat-icon\n          *ngIf=\"field.props?.['icon']\"\n          class=\"text-lg mb-1\"\n          >{{ field.props?.['icon'] }}</mat-icon\n        >\n        {{ field.props?.label }}<span *ngIf=\"to.required\">*</span>\n      </button>\n      <input\n        type=\"file\"\n        #fileInput\n        (change)=\"onFileSelect($event)\"\n        [formlyAttributes]=\"field\"\n        [multiple]=\"field.props?.['isMultiFileUpload']\"\n        [accept]=\"field.props?.['acceptedFileTypes'] || '.pdf,.xlsx,.jpg,.jpeg,.docx'\"\n        class=\"hidden\"\n      />\n      <div class=\"text-[0.83rem] text-gray-500 mt-1\">\n        Allowed file types:\n        {{ field.props?.['acceptedFileTypes'] || '.pdf, .xlsx, .jpg, jpeg, .docx' }}\n        <br />\n        Maximum file size: {{ field.props?.['maxFileSizeMB'] || 1 }}MB\n      </div>\n    </div>\n    }\n    <table\n      mat-table\n      *ngIf=\"this.props?.['isMultiFileUpload'] && uploadedFiles?.length\"\n      [dataSource]=\"dataSource\"\n      class=\"mat-elevation-z8\"\n    >\n      <ng-container matColumnDef=\"fileName\">\n        <th mat-header-cell *matHeaderCellDef class=\"header-cell\">File Name</th>\n        <td mat-cell *matCellDef=\"let element\">{{ element.fileName }}</td>\n      </ng-container>\n\n      <ng-container matColumnDef=\"convertedFileSize\">\n        <th mat-header-cell *matHeaderCellDef class=\"header-cell\">Size</th>\n        <td mat-cell *matCellDef=\"let element\">\n          {{ element.convertedFileSize }}\n        </td>\n      </ng-container>\n\n      <ng-container matColumnDef=\"uploadedBy\">\n        <th mat-header-cell *matHeaderCellDef class=\"header-cell\">\n          Uploaded By\n        </th>\n        <td mat-cell *matCellDef=\"let element\">{{ element.uploadedBy }}</td>\n      </ng-container>\n\n      <ng-container matColumnDef=\"uploadedAt\">\n        <th mat-header-cell *matHeaderCellDef class=\"header-cell\">\n          Uploaded Date\n        </th>\n        <td mat-cell *matCellDef=\"let element\">\n          {{ element.uploadedAt | date : 'dd-MMM-yyyy, hh:mm:ss a' }}\n        </td>\n      </ng-container>\n\n      <ng-container matColumnDef=\"actions\">\n        <th mat-header-cell *matHeaderCellDef class=\"header-cell\">Actions</th>\n        <td mat-cell *matCellDef=\"let element; let i = index\">\n          <div class=\"action-buttons\">\n            <div\n              *ngFor=\"let action of field?.props?.['actions']\"\n              class=\"actions-container\"\n            >\n              <mat-icon\n                class=\"action-icons\"\n                *ngIf=\"showUploadAction(action, element)\"\n                matTooltip=\"{{ action.label | translate }}\"\n                (click)=\"handleActionClick(action, i, element)\"\n              >\n                {{ action.icon }}\n              </mat-icon>\n            </div>\n          </div>\n        </td>\n      </ng-container>\n\n      <tr mat-header-row *matHeaderRowDef=\"displayedColumns\"></tr>\n      <tr mat-row *matRowDef=\"let row; columns: displayedColumns\"></tr>\n    </table>\n\n    <div class=\"w-full\">\n      <table\n        *ngIf=\"!this.props?.['isMultiFileUpload'] && uploadedFiles?.length\"\n        class=\"w-full flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4\"\n      >\n        <tr\n          class=\"flex flex-col md:flex-row items-start md:items-center gap-2 md:gap-4 w-full\"\n        >\n          <td\n            class=\"text-sm truncate overflow-hidden whitespace-nowrap max-w-full flex-1 md:border-none\"\n            [matTooltip]=\"uploadedFiles[0].fileName\"\n          >\n            {{ uploadedFiles[0].fileName }}\n          </td>\n          <td class=\"flex items-center justify-end md:flex-shrink-0\">\n            <div\n              *ngFor=\"let action of field?.props?.['actions']\"\n              class=\"flex items-center\"\n            >\n              <mat-icon\n                matTooltip=\"{{ action.label | translate }}\"\n                *ngIf=\"showUploadAction(action, uploadedFiles[0])\"\n                (click)=\"handleActionClick(action, 0, uploadedFiles[0])\"\n                class=\"action-icons\"\n              >\n                {{ action.icon }}\n              </mat-icon>\n            </div>\n          </td>\n        </tr>\n      </table>\n    </div>\n    <div\n      *ngIf=\"to['note']\"\n      class=\"text-gray-500 mt-1 w-[900px]\"\n      [innerHTML]=\"to['note']\"\n    ></div>\n  `,\n  styles: [\n    `\n      .mat-mdc-table {\n        margin-top: 5px;\n        box-shadow: none;\n      }\n      .header-cell {\n        background-color: rgb(120 118 118 / 12%) !important;\n      }\n      .action-buttons {\n        display: flex;\n        flex-direction: row;\n        justify-content: start;\n        align-items: center;\n      }\n      .action-icons {\n        font-size: 20px;\n      }\n      .actions-container {\n        display: flex;\n        justify-content: center;\n        align-items: center;\n        height: 43px;\n        padding-top: 5px;\n      }\n\n      .input-btn button,\n      .red-upload button {\n        color: white;\n        padding: 3px 10px;\n        border: none;\n        cursor: pointer;\n        border-radius: 50px;\n        font-family: Poppins, 'Helvetica Neue', sans-serif;\n        font-weight: 500;\n        font-size: 14px;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n      }\n      .input-btn button:disabled,\n      .red-upload button:disabled {\n        background-color: #d3d3d3;\n        color: #b0b0b0;\n      }\n      th,\n      td,\n      .upload-btn {\n        font-family: Poppins, 'Helvetica Neue', sans-serif;\n      }\n      .red-upload button {\n        background-color: red;\n      }\n    `,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FormlyFileUploadComponent\n  extends FieldType<CustomFormlyFieldConfig>\n  implements OnInit\n{\n  uploadedFiles: any[] = [];\n  displayedColumns: string[] = [\n    'fileName',\n    'convertedFileSize',\n    'uploadedBy',\n    'uploadedAt',\n    'actions',\n  ];\n  dataSource = new MatTableDataSource();\n  showUploadbtn = true;\n  loggedInUserDetails!: any;\n  destroyRef = inject(DestroyRef);\n  isMobileDevice: boolean = false;\n\n  constructor(\n    private snackBar: MatSnackBar,\n    private translateService: TranslateService,\n    private breakpointObserver: BreakpointObserver,\n    private cdr: ChangeDetectorRef\n  ) {\n    super();\n    this.breakpointObserver\n      ?.observe([Breakpoints.Handset])\n      .pipe(takeUntilDestroyed(this.destroyRef))\n      .subscribe((result) => {\n        this.isMobileDevice = result?.matches;\n      });\n  }\n\n  ngOnInit(): void {\n    const userDetails = localStorage.getItem('loggedInUserDetails');\n    this.loggedInUserDetails = userDetails\n      ? JSON.parse(userDetails)\n      : { email: '' };\n\n    if (this.field.props?.['isMultiFileUpload'] && this.formControl.value) {\n      this.uploadedFiles = this.formControl.value ?? [];\n    } else {\n      if (\n        this.formControl.value &&\n        Object.keys(this.formControl.value || {}).length\n      ) {\n        this.uploadedFiles = [this.formControl.value];\n      }\n    }\n    this.dataSource.data = [...this.uploadedFiles];\n    this.showUploadbtn = this.props?.['isMultiFileUpload']\n      ? true\n      : !this.uploadedFiles.length;\n  }\n\n  onFileSelect(event: any): void {\n    const selectedFile = event.target.files;\n    if (!selectedFile || !selectedFile.length) return;\n\n    const maxFileSizeMB = this.field.props?.['maxFileSizeMB'] || 1;\n    const maxFileSize = maxFileSizeMB * 1024 * 1024;\n\n    if (this.props?.['isMultiFileUpload']) {\n      if (this.isFileAlreadyUploaded(event.target.files)) {\n        this.snackBar.open(\n          this.translateService.instant('MNL_UPLOAD_FILE.DUPLICATE_UPLOADED'),\n          'Ok',\n          {\n            duration: 3000,\n            panelClass: ['warn-snackbar'],\n          }\n        );\n        this.resetFileInput(event.target);\n        return;\n      }\n    }\n\n    const selectedFiles = Array.from(selectedFile);\n    const invalidChars = /[&,\"]/;\n\n    const filesToProcess = this.field.props?.['isMultiFileUpload']\n      ? selectedFiles\n      : [selectedFiles[0]];\n\n    filesToProcess.forEach((file: any) => {\n      if (file.size > maxFileSize) {\n        this.snackBar.open(\n          `The file \"${file.name}\" exceeds the ${maxFileSizeMB} MB size limit.`,\n          'OK',\n          {\n            duration: 3000,\n            panelClass: ['warn-snackbar'],\n          }\n        );\n      } else if (invalidChars.test(file.name)) {\n        this.snackBar.open('MNL_UPLOAD_FILE.INVALID_CHARS_FILE_NAME', 'Ok', {\n          duration: 7000,\n          panelClass: ['error-snackbar'],\n        });\n      } else {\n        const newFile = {\n          fileName: file.name,\n          fileSize: file.size,\n          fileType: file.type,\n          file: file,\n          uploadedBy: this.loggedInUserDetails.email,\n          uploadedAt: new Date(),\n          nodeId: '',\n          convertedFileSize: this.formatFileSize(file.size),\n        };\n\n        if (!this.props?.['isMultiFileUpload']) {\n          this.uploadedFiles = [newFile];\n        } else {\n          this.uploadedFiles.push(newFile);\n        }\n      }\n    });\n\n    this.dataSource.data = [...this.uploadedFiles];\n\n    this.resetFileInput(event.target);\n    this.showUploadbtn = this.props?.['isMultiFileUpload']\n      ? true\n      : !this.uploadedFiles.length;\n\n    if (this.field.props?.['onFilesAdded']) {\n      this.field.props['onFilesAdded'](this.uploadedFiles);\n    }\n  }\n\n  handleActionClick(action: FileAction, index: number, element: any) {\n    const onUploadFileAction = this.field.props?.['onUploadFileAction'];\n    if (typeof onUploadFileAction === 'function') {\n      const result = onUploadFileAction(action.icon, element, index);\n      this.invokeAction(\n        () => result,\n        element,\n        index,\n        () => {\n          if (action.icon === 'delete') this.removeFile(index);\n          if (action.icon === 'cloud_upload')\n            this.refreshUploadedTable(element, index);\n          if (action.icon === 'cloud_download' || action.icon === 'visibility')\n            () => {};\n        },\n        action.icon\n      );\n    }\n  }\n\n  refreshUploadedTable(element: any, index: number) {\n    this.uploadedFiles[index] = element;\n    if (this.props?.['isMultiFileUpload']) {\n      const uploaded = this.uploadedFiles.filter((f) => !!f.nodeId);\n      this.formControl.setValue(uploaded);\n    } else {\n      this.formControl.setValue(this.uploadedFiles[0]);\n    }\n\n    this.formControl.markAsTouched();\n    this.form.updateValueAndValidity();\n    this.dataSource.data = [...this.uploadedFiles];\n    this.cdr.detectChanges();\n  }\n\n  removeFile(index: number) {\n    this.uploadedFiles.splice(index, 1);\n    this.dataSource.data = [...this.uploadedFiles];\n\n    if (!this.field.props?.['isMultiFileUpload']) {\n      this.formControl.setValue(null);\n    } else {\n      this.formControl.setValue(this.uploadedFiles);\n      const uploaded = this.uploadedFiles.filter((f) => !!f.nodeId);\n      this.formControl.setValue(uploaded);\n    }\n    this.showUploadbtn = this.props?.['isMultiFileUpload']\n      ? true\n      : !this.uploadedFiles.length;\n  }\n\n  isFileAlreadyUploaded(selectedFile: any): boolean {\n    return this.uploadedFiles.some(\n      (file) => file.fileName === selectedFile[0].name\n    );\n  }\n\n  resetFileInput(fileInput: any): void {\n    fileInput.value = '';\n  }\n\n  showUploadAction(action: any, file: any): boolean {\n    const hasNode = !!file.nodeId;\n    const userRoles = this.options.formState?.userRoles ?? [];\n    const enabledRoles = this.field?.enabledForRoles ?? [];\n\n    switch (action.icon) {\n      case 'cloud_upload':\n        return !hasNode;\n\n      case 'cloud_download':\n      case 'visibility':\n        return hasNode;\n\n      case 'delete':\n        return (\n          hasNode && userRoles.some((role: any) => enabledRoles.includes(role))\n        );\n\n      default:\n        return false;\n    }\n  }\n\n  formatFileSize(bytes: number): string {\n    if (bytes < 1024) return bytes + ' B';\n    if (bytes < 1048576) return (bytes / 1024).toFixed(2) + ' KB';\n    return (bytes / 1048576).toFixed(2) + ' MB';\n  }\n\n  private invokeAction(\n    actionCallback:\n      | undefined\n      | ((element: any, index: number) => MaybeAsyncBool),\n    element: any,\n    index: number,\n    defaultFn: () => void,\n    icon?: string\n  ): void {\n    if (!actionCallback) {\n      defaultFn();\n      return;\n    }\n\n    const result = actionCallback(element, index);\n\n    if (result instanceof Observable) {\n      (result as Observable<boolean>).subscribe((shouldProceed) => {\n        if (shouldProceed) {\n          this.successMessage(icon);\n          defaultFn();\n        }\n      });\n    } else {\n      Promise.resolve(result as boolean | Promise<boolean>).then(\n        (shouldProceed) => {\n          if (shouldProceed) {\n            this.successMessage(icon);\n            defaultFn();\n          }\n        }\n      );\n    }\n  }\n\n  successMessage(icon: string) {\n    if (icon === 'cloud_upload') {\n      this.snackBar.open(\n        this.translateService.instant(\n          'MNL_UPLOAD_FILE.FILE_SUCCESSFUL_UPLOADED'\n        ),\n        'Ok',\n        {\n          duration: 5000,\n          panelClass: ['success-snackbar'],\n        }\n      );\n    } else if (icon === 'delete') {\n      this.snackBar.open(\n        this.translateService.instant(\n          'MNL_UPLOAD_FILE.FILE_SUCCESSFUL_DELETED'\n        ),\n        'Ok',\n        {\n          duration: 5000,\n          panelClass: ['success-snackbar'],\n        }\n      );\n    }\n  }\n}\n"]}
|