@muraai/mnl-form 0.0.1-alpha-db495b1 → 0.0.1-alpha-11e96eb

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.
Files changed (61) hide show
  1. package/fesm2022/muraai-mnl-form.mjs +150 -143
  2. package/fesm2022/muraai-mnl-form.mjs.map +1 -1
  3. package/lib/types/upload-file.d.ts +1 -1
  4. package/lib/workflow-status/ workflow-status.module.d.ts +8 -8
  5. package/lib/workflow-status/workflow-status.component.d.ts +1 -1
  6. package/package.json +4 -5
  7. package/esm2022/lib/confirmation-dialog/confirmation-dialog.component.mjs +0 -34
  8. package/esm2022/lib/mnl-form/mnl-form-standalone.module.mjs +0 -155
  9. package/esm2022/lib/mnl-form/mnl-form.component.mjs +0 -199
  10. package/esm2022/lib/models/custom-form-field.model.mjs +0 -2
  11. package/esm2022/lib/models/workflow-status.model.mjs +0 -2
  12. package/esm2022/lib/services/graphql.service.mjs +0 -30
  13. package/esm2022/lib/services/scoring.service.mjs +0 -103
  14. package/esm2022/lib/types/action-button-renderer.component.mjs +0 -59
  15. package/esm2022/lib/types/autocomplete-type.component.mjs +0 -68
  16. package/esm2022/lib/types/button.type.mjs +0 -72
  17. package/esm2022/lib/types/comments.type.mjs +0 -102
  18. package/esm2022/lib/types/country-code-input.type.mjs +0 -181
  19. package/esm2022/lib/types/custom-date.type.mjs +0 -20
  20. package/esm2022/lib/types/custom-input.component.mjs +0 -62
  21. package/esm2022/lib/types/date-format.type.mjs +0 -196
  22. package/esm2022/lib/types/expansion-panel.types.mjs +0 -85
  23. package/esm2022/lib/types/formly-field-panel.type.mjs +0 -70
  24. package/esm2022/lib/types/grid-actionable.type.mjs +0 -197
  25. package/esm2022/lib/types/grid-formly-cell.component.mjs +0 -42
  26. package/esm2022/lib/types/grid.type.mjs +0 -125
  27. package/esm2022/lib/types/input.type.mjs +0 -43
  28. package/esm2022/lib/types/mu-helper-text.type.mjs +0 -331
  29. package/esm2022/lib/types/multi-select-autocomplete.mjs +0 -139
  30. package/esm2022/lib/types/multi-select-checkbox.component.mjs +0 -247
  31. package/esm2022/lib/types/radio.type.mjs +0 -71
  32. package/esm2022/lib/types/scrollable-tabs.types.mjs +0 -220
  33. package/esm2022/lib/types/select-api-auto-complete.component.mjs +0 -111
  34. package/esm2022/lib/types/select-autocomplete-redefined.component.mjs +0 -142
  35. package/esm2022/lib/types/select-autocomplete.type.mjs +0 -254
  36. package/esm2022/lib/types/stepper.type.mjs +0 -268
  37. package/esm2022/lib/types/table.types.mjs +0 -79
  38. package/esm2022/lib/types/tabs.types.mjs +0 -293
  39. package/esm2022/lib/types/tabular-form-dialog.component.mjs +0 -331
  40. package/esm2022/lib/types/tabular-form.type.mjs +0 -641
  41. package/esm2022/lib/types/upload-file.mjs +0 -524
  42. package/esm2022/lib/types/view-file-dialog.component.mjs +0 -90
  43. package/esm2022/lib/utils/formly.utils.mjs +0 -105
  44. package/esm2022/lib/validators/custom-email-validator.mjs +0 -18
  45. package/esm2022/lib/validators/ein-validators.mjs +0 -11
  46. package/esm2022/lib/validators/gst-validator.mjs +0 -11
  47. package/esm2022/lib/validators/normal-email-validator.mjs +0 -17
  48. package/esm2022/lib/validators/npi-validator.mjs +0 -11
  49. package/esm2022/lib/validators/pan-validator.mjs +0 -11
  50. package/esm2022/lib/validators/phone-number-validator.mjs +0 -22
  51. package/esm2022/lib/validators/select-validator.mjs +0 -7
  52. package/esm2022/lib/validators/tinSsn-validator.mjs +0 -12
  53. package/esm2022/lib/workflow-status/ workflow-status.module.mjs +0 -52
  54. package/esm2022/lib/workflow-status/workflow-status.component.mjs +0 -70
  55. package/esm2022/lib/wrappers/form-field-description-wrapper.component.mjs +0 -44
  56. package/esm2022/lib/wrappers/form-field-wrapper.component.mjs +0 -34
  57. package/esm2022/lib/wrappers/formly-wrapper-with-old-value.component.mjs +0 -140
  58. package/esm2022/lib/wrappers/prefix-suffix-wrapper.component.mjs +0 -95
  59. package/esm2022/lib/wrappers/prefix-sufix-extension.component.mjs +0 -9
  60. package/esm2022/muraai-mnl-form.mjs +0 -5
  61. 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"]}