ngx-material-entity 15.3.0 → 16.0.0
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/classes/entity.model.d.ts +2 -2
- package/components/confirm-dialog/confirm-dialog-data.d.ts +1 -1
- package/components/edit-page/edit-page.component.d.ts +4 -2
- package/components/edit-page/page-edit-data.builder.d.ts +1 -1
- package/components/input/array/array-string-chips-input/array-string-chips-input.component.d.ts +1 -0
- package/components/input/array/array-table.class.d.ts +4 -2
- package/components/input/base-input.component.d.ts +5 -1
- package/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.d.ts +1 -1
- package/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.d.ts +1 -1
- package/components/input/file/file-default-input/file-default-input.component.d.ts +0 -2
- package/components/input/file/file-image-input/file-image-input.component.d.ts +3 -2
- package/components/input/file/file-input/file-input.component.d.ts +4 -2
- package/components/input/input.component.d.ts +9 -4
- package/components/input/input.module.d.ts +2 -1
- package/components/table/display-column-value/base-display-column-value.component.d.ts +1 -1
- package/components/table/display-column-value/display-column-value.component.d.ts +1 -1
- package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +3 -1
- package/components/table/table-data.builder.d.ts +1 -1
- package/components/table/table-data.d.ts +1 -1
- package/components/table/table.component.d.ts +1 -1
- package/decorators/base/decorator-types.enum.d.ts +1 -1
- package/decorators/base/property-decorator.data.d.ts +1 -1
- package/decorators/file/file-decorator.data.d.ts +1 -1
- package/decorators/number/number-decorator.data.d.ts +1 -1
- package/decorators/string/string-decorator.data.d.ts +1 -1
- package/encapsulation/jszip.utilities.d.ts +1 -1
- package/encapsulation/uuid.utilities.d.ts +11 -0
- package/esm2022/classes/base.builder.mjs +44 -0
- package/esm2022/classes/entity.model.mjs +26 -0
- package/esm2022/components/confirm-dialog/confirm-dialog-data.builder.mjs +58 -0
- package/esm2022/components/confirm-dialog/confirm-dialog.component.mjs +60 -0
- package/esm2022/components/edit-page/edit-page.component.mjs +291 -0
- package/esm2022/components/input/array/array-date-input/array-date-input.component.mjs +23 -0
- package/esm2022/components/input/array/array-date-range-input/array-date-range-input.component.mjs +49 -0
- package/esm2022/components/input/array/array-date-time-input/array-date-time-input.component.mjs +49 -0
- package/esm2022/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.mjs +50 -0
- package/esm2022/components/input/array/array-string-chips-input/array-string-chips-input.component.mjs +69 -0
- package/esm2022/components/input/array/array-table.class.mjs +100 -0
- package/esm2022/components/input/base-input.component.mjs +88 -0
- package/esm2022/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.mjs +21 -0
- package/esm2022/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.mjs +17 -0
- package/esm2022/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.mjs +21 -0
- package/esm2022/components/input/custom/custom.component.mjs +28 -0
- package/esm2022/components/input/date/date-input/date-input.component.mjs +19 -0
- package/esm2022/components/input/date/date-range-input/date-range-input.component.mjs +68 -0
- package/esm2022/components/input/date/date-time-input/date-time-input.component.mjs +69 -0
- package/esm2022/components/input/file/file-default-input/file-default-input.component.mjs +18 -0
- package/esm2022/components/input/file/file-image-input/file-image-input.component.mjs +101 -0
- package/esm2022/components/input/file/file-input/dragDrop.directive.mjs +62 -0
- package/esm2022/components/input/file/file-input/file-input.component.mjs +206 -0
- package/esm2022/components/input/input.component.mjs +849 -0
- package/esm2022/components/input/input.module.mjs +172 -0
- package/esm2022/components/input/number/number-dropdown-input/number-dropdown-input.component.mjs +23 -0
- package/esm2022/components/input/number/number-input/number-input.component.mjs +16 -0
- package/esm2022/components/input/number/number-slider-input/number-slider-input.component.mjs +17 -0
- package/esm2022/components/input/relations/references-many-input/references-many-input.component.mjs +102 -0
- package/esm2022/components/input/string/string-autocomplete-input/string-autocomplete-input.component.mjs +36 -0
- package/esm2022/components/input/string/string-dropdown-input/string-dropdown-input.component.mjs +23 -0
- package/esm2022/components/input/string/string-input/string-input.component.mjs +16 -0
- package/esm2022/components/input/string/string-password-input/string-password-input.component.mjs +39 -0
- package/esm2022/components/input/string/string-textbox-input/string-textbox-input.component.mjs +17 -0
- package/esm2022/components/table/create-dialog/create-dialog-data.builder.mjs +42 -0
- package/esm2022/components/table/create-dialog/create-entity-dialog-data.builder.mjs +32 -0
- package/esm2022/components/table/create-dialog/create-entity-dialog.component.mjs +107 -0
- package/esm2022/components/table/display-column-value/base-display-column-value.component.mjs +28 -0
- package/esm2022/components/table/display-column-value/display-column-value.component.mjs +37 -0
- package/esm2022/components/table/edit-dialog/edit-data.builder.mjs +106 -0
- package/esm2022/components/table/edit-dialog/edit-entity-dialog.component.mjs +209 -0
- package/esm2022/components/table/edit-dialog/edit-entity.builder.mjs +39 -0
- package/esm2022/components/table/table-data.builder.mjs +269 -0
- package/esm2022/components/table/table.component.mjs +348 -0
- package/esm2022/decorators/array/array-decorator-internal.data.mjs +286 -0
- package/esm2022/decorators/array/array-decorator.data.mjs +21 -0
- package/esm2022/decorators/base/property-decorator-internal.data.mjs +82 -0
- package/esm2022/decorators/base/property-decorator.data.mjs +56 -0
- package/esm2022/decorators/boolean/boolean-decorator-internal.data.mjs +43 -0
- package/esm2022/decorators/boolean/boolean-decorator.data.mjs +11 -0
- package/esm2022/decorators/custom/custom-decorator-internal.data.mjs +35 -0
- package/{esm2020 → esm2022}/decorators/date/date-decorator-internal.data.mjs +43 -1
- package/esm2022/decorators/date/date-decorator.data.mjs +11 -0
- package/esm2022/decorators/file/file-decorator-internal.data.mjs +143 -0
- package/esm2022/decorators/file/file-decorator.data.mjs +64 -0
- package/esm2022/decorators/has-many/has-many-decorator-internal.data.mjs +42 -0
- package/esm2022/decorators/number/number-decorator-internal.data.mjs +76 -0
- package/esm2022/decorators/number/number-decorator.data.mjs +11 -0
- package/esm2022/decorators/object/object-decorator-internal.data.mjs +20 -0
- package/esm2022/decorators/object/object-decorator.data.mjs +22 -0
- package/esm2022/decorators/references-many/references-many-decorator-internal.data.mjs +48 -0
- package/esm2022/decorators/references-one/references-one-decorator-internal.data.mjs +24 -0
- package/esm2022/decorators/string/string-decorator-internal.data.mjs +114 -0
- package/esm2022/decorators/string/string-decorator.data.mjs +11 -0
- package/esm2022/encapsulation/uuid.utilities.mjs +15 -0
- package/esm2022/services/entity.service.mjs +276 -0
- package/esm2022/utilities/date.utilities.mjs +159 -0
- package/esm2022/utilities/entity.utilities.mjs +905 -0
- package/esm2022/utilities/file.utilities.mjs +180 -0
- package/{fesm2020 → fesm2022}/ngx-material-entity.mjs +1355 -484
- package/fesm2022/ngx-material-entity.mjs.map +1 -0
- package/package.json +11 -17
- package/utilities/date.utilities.d.ts +1 -1
- package/utilities/entity.utilities.d.ts +7 -3
- package/utilities/file.utilities.d.ts +6 -2
- package/esm2020/classes/base.builder.mjs +0 -42
- package/esm2020/classes/entity.model.mjs +0 -22
- package/esm2020/components/confirm-dialog/confirm-dialog-data.builder.mjs +0 -44
- package/esm2020/components/confirm-dialog/confirm-dialog.component.mjs +0 -57
- package/esm2020/components/edit-page/edit-page.component.mjs +0 -274
- package/esm2020/components/input/array/array-date-input/array-date-input.component.mjs +0 -26
- package/esm2020/components/input/array/array-date-range-input/array-date-range-input.component.mjs +0 -50
- package/esm2020/components/input/array/array-date-time-input/array-date-time-input.component.mjs +0 -49
- package/esm2020/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.mjs +0 -49
- package/esm2020/components/input/array/array-string-chips-input/array-string-chips-input.component.mjs +0 -70
- package/esm2020/components/input/array/array-table.class.mjs +0 -93
- package/esm2020/components/input/base-input.component.mjs +0 -64
- package/esm2020/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.mjs +0 -21
- package/esm2020/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.mjs +0 -17
- package/esm2020/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.mjs +0 -21
- package/esm2020/components/input/custom/custom.component.mjs +0 -26
- package/esm2020/components/input/date/date-input/date-input.component.mjs +0 -22
- package/esm2020/components/input/date/date-range-input/date-range-input.component.mjs +0 -71
- package/esm2020/components/input/date/date-time-input/date-time-input.component.mjs +0 -71
- package/esm2020/components/input/file/file-default-input/file-default-input.component.mjs +0 -23
- package/esm2020/components/input/file/file-image-input/file-image-input.component.mjs +0 -98
- package/esm2020/components/input/file/file-input/dragDrop.directive.mjs +0 -63
- package/esm2020/components/input/file/file-input/file-input.component.mjs +0 -195
- package/esm2020/components/input/input.component.mjs +0 -746
- package/esm2020/components/input/input.module.mjs +0 -169
- package/esm2020/components/input/number/number-dropdown-input/number-dropdown-input.component.mjs +0 -26
- package/esm2020/components/input/number/number-input/number-input.component.mjs +0 -16
- package/esm2020/components/input/number/number-slider-input/number-slider-input.component.mjs +0 -17
- package/esm2020/components/input/relations/references-many-input/references-many-input.component.mjs +0 -100
- package/esm2020/components/input/string/string-autocomplete-input/string-autocomplete-input.component.mjs +0 -34
- package/esm2020/components/input/string/string-dropdown-input/string-dropdown-input.component.mjs +0 -26
- package/esm2020/components/input/string/string-input/string-input.component.mjs +0 -16
- package/esm2020/components/input/string/string-password-input/string-password-input.component.mjs +0 -42
- package/esm2020/components/input/string/string-textbox-input/string-textbox-input.component.mjs +0 -17
- package/esm2020/components/table/create-dialog/create-dialog-data.builder.mjs +0 -32
- package/esm2020/components/table/create-dialog/create-entity-dialog-data.builder.mjs +0 -26
- package/esm2020/components/table/create-dialog/create-entity-dialog.component.mjs +0 -100
- package/esm2020/components/table/display-column-value/base-display-column-value.component.mjs +0 -27
- package/esm2020/components/table/display-column-value/display-column-value.component.mjs +0 -33
- package/esm2020/components/table/edit-dialog/edit-data.builder.mjs +0 -76
- package/esm2020/components/table/edit-dialog/edit-entity-dialog.component.mjs +0 -195
- package/esm2020/components/table/edit-dialog/edit-entity.builder.mjs +0 -29
- package/esm2020/components/table/table-data.builder.mjs +0 -205
- package/esm2020/components/table/table.component.mjs +0 -333
- package/esm2020/decorators/array/array-decorator-internal.data.mjs +0 -150
- package/esm2020/decorators/array/array-decorator.data.mjs +0 -7
- package/esm2020/decorators/base/property-decorator-internal.data.mjs +0 -58
- package/esm2020/decorators/base/property-decorator.data.mjs +0 -6
- package/esm2020/decorators/boolean/boolean-decorator-internal.data.mjs +0 -33
- package/esm2020/decorators/boolean/boolean-decorator.data.mjs +0 -7
- package/esm2020/decorators/custom/custom-decorator-internal.data.mjs +0 -27
- package/esm2020/decorators/date/date-decorator.data.mjs +0 -7
- package/esm2020/decorators/file/file-decorator-internal.data.mjs +0 -97
- package/esm2020/decorators/file/file-decorator.data.mjs +0 -7
- package/esm2020/decorators/has-many/has-many-decorator-internal.data.mjs +0 -32
- package/esm2020/decorators/number/number-decorator-internal.data.mjs +0 -54
- package/esm2020/decorators/number/number-decorator.data.mjs +0 -7
- package/esm2020/decorators/object/object-decorator-internal.data.mjs +0 -14
- package/esm2020/decorators/object/object-decorator.data.mjs +0 -7
- package/esm2020/decorators/references-many/references-many-decorator-internal.data.mjs +0 -30
- package/esm2020/decorators/references-one/references-one-decorator-internal.data.mjs +0 -16
- package/esm2020/decorators/string/string-decorator-internal.data.mjs +0 -72
- package/esm2020/decorators/string/string-decorator.data.mjs +0 -7
- package/esm2020/services/entity.service.mjs +0 -274
- package/esm2020/utilities/date.utilities.mjs +0 -159
- package/esm2020/utilities/entity.utilities.mjs +0 -896
- package/esm2020/utilities/file.utilities.mjs +0 -176
- package/fesm2015/ngx-material-entity.mjs +0 -6300
- package/fesm2015/ngx-material-entity.mjs.map +0 -1
- package/fesm2020/ngx-material-entity.mjs.map +0 -1
- /package/{esm2020 → esm2022}/components/confirm-dialog/confirm-dialog-data.mjs +0 -0
- /package/{esm2020 → esm2022}/components/edit-page/edit-data.route.mjs +0 -0
- /package/{esm2020 → esm2022}/components/edit-page/page-edit-data.builder.mjs +0 -0
- /package/{esm2020 → esm2022}/components/get-validation-error-message.function.mjs +0 -0
- /package/{esm2020 → esm2022}/components/table/create-dialog/create-entity-dialog-data.mjs +0 -0
- /package/{esm2020 → esm2022}/components/table/default.actions.mjs +0 -0
- /package/{esm2020 → esm2022}/components/table/edit-dialog/edit-entity-data.mjs +0 -0
- /package/{esm2020 → esm2022}/components/table/table-data.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/array/array.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/base/base-property.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/base/decorator-types.enum.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/base/dropdown-value.interface.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/boolean/boolean.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/custom/custom-decorator.data.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/custom/custom.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/date/date.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/file/file.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/has-many/has-many-decorator.data.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/has-many/has-many.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/number/number.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/object/object.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/references-many/references-many-decorator.data.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/references-many/references-many.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/references-one/references-one-decorator.data.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/references-one/references-one.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/decorators/string/string.decorator.mjs +0 -0
- /package/{esm2020 → esm2022}/encapsulation/js-2-xml.utilities.mjs +0 -0
- /package/{esm2020 → esm2022}/encapsulation/jszip.utilities.mjs +0 -0
- /package/{esm2020 → esm2022}/encapsulation/lodash.utilities.mjs +0 -0
- /package/{esm2020 → esm2022}/encapsulation/reflect.utilities.mjs +0 -0
- /package/{esm2020 → esm2022}/functions/default-false.function.mjs +0 -0
- /package/{esm2020 → esm2022}/functions/default-true.function.mjs +0 -0
- /package/{esm2020 → esm2022}/functions/is-async-function.function.mjs +0 -0
- /package/{esm2020 → esm2022}/mocks/placeholder-data.png.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-material-entity.mjs +0 -0
- /package/{esm2020 → esm2022}/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/services/unsaved-changes.guard.mjs +0 -0
- /package/{esm2020 → esm2022}/utilities/selection.utilities.mjs +0 -0
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
/* eslint-disable jsdoc/require-jsdoc */
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { Component } from '@angular/core';
|
|
4
|
+
import { ReflectUtilities } from '../../../../encapsulation/reflect.utilities';
|
|
5
|
+
import { placeholder } from '../../../../mocks/placeholder-data.png';
|
|
6
|
+
import { EntityUtilities } from '../../../../utilities/entity.utilities';
|
|
7
|
+
import { FileUtilities } from '../../../../utilities/file.utilities';
|
|
8
|
+
import { NgxMatEntityBaseInputComponent } from '../../base-input.component';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@angular/common/http";
|
|
11
|
+
import * as i2 from "@angular/common";
|
|
12
|
+
import * as i3 from "@angular/material/button";
|
|
13
|
+
import * as i4 from "../file-input/file-input.component";
|
|
14
|
+
export class FileImageInputComponent extends NgxMatEntityBaseInputComponent {
|
|
15
|
+
http;
|
|
16
|
+
get multiPreviewImages() {
|
|
17
|
+
return ReflectUtilities.getMetadata(EntityUtilities.MULTI_PREVIEW_IMAGES_KEY, this.entity, this.key);
|
|
18
|
+
}
|
|
19
|
+
set multiPreviewImages(value) {
|
|
20
|
+
ReflectUtilities.defineMetadata(EntityUtilities.MULTI_PREVIEW_IMAGES_KEY, value, this.entity, this.key);
|
|
21
|
+
}
|
|
22
|
+
get singlePreviewImage() {
|
|
23
|
+
return ReflectUtilities.getMetadata(EntityUtilities.SINGLE_PREVIEW_IMAGE_KEY, this.entity, this.key);
|
|
24
|
+
}
|
|
25
|
+
set singlePreviewImage(value) {
|
|
26
|
+
ReflectUtilities.defineMetadata(EntityUtilities.SINGLE_PREVIEW_IMAGE_KEY, value, this.entity, this.key);
|
|
27
|
+
}
|
|
28
|
+
imageIndex = 0;
|
|
29
|
+
placeHolder = placeholder;
|
|
30
|
+
constructor(http) {
|
|
31
|
+
super();
|
|
32
|
+
this.http = http;
|
|
33
|
+
}
|
|
34
|
+
async setSinglePreviewImage() {
|
|
35
|
+
if (this.propertyValue) {
|
|
36
|
+
this.propertyValue = await FileUtilities.getFileData(this.propertyValue, this.http);
|
|
37
|
+
this.singlePreviewImage = await FileUtilities.getDataURLFromFile(this.propertyValue.file);
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
this.singlePreviewImage = undefined;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
async setMultiPreviewImages(index) {
|
|
44
|
+
const multiFileData = this.propertyValue;
|
|
45
|
+
const previewImages = [];
|
|
46
|
+
if (multiFileData?.length) {
|
|
47
|
+
for (let i = 0; i < multiFileData.length; i++) {
|
|
48
|
+
if (i === index) {
|
|
49
|
+
multiFileData[index] = await FileUtilities.getFileData(multiFileData[index], this.http);
|
|
50
|
+
previewImages.push(await FileUtilities.getDataURLFromFile(multiFileData[index].file));
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
previewImages.push('empty');
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
this.multiPreviewImages = previewImages;
|
|
58
|
+
}
|
|
59
|
+
async refreshFileData(fileData) {
|
|
60
|
+
this.propertyValue = fileData;
|
|
61
|
+
this.emitChange();
|
|
62
|
+
if (this.metadata.multiple) {
|
|
63
|
+
fileData = fileData;
|
|
64
|
+
if (!fileData?.[this.imageIndex]) {
|
|
65
|
+
this.imageIndex = 0;
|
|
66
|
+
}
|
|
67
|
+
await this.setMultiPreviewImages(this.imageIndex);
|
|
68
|
+
}
|
|
69
|
+
else {
|
|
70
|
+
await this.setSinglePreviewImage();
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
async prev() {
|
|
74
|
+
if (this.imageIndex <= 0) {
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
await this.setMultiPreviewImages(this.imageIndex - 1);
|
|
78
|
+
this.imageIndex--;
|
|
79
|
+
}
|
|
80
|
+
async next() {
|
|
81
|
+
if (!this.multiPreviewImages?.length) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
if (this.imageIndex === (this.multiPreviewImages.length - 1)) {
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
await this.setMultiPreviewImages(this.imageIndex + 1);
|
|
88
|
+
this.imageIndex++;
|
|
89
|
+
}
|
|
90
|
+
async setIndex(index) {
|
|
91
|
+
await this.setMultiPreviewImages(index);
|
|
92
|
+
this.imageIndex = index;
|
|
93
|
+
}
|
|
94
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: FileImageInputComponent, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Component });
|
|
95
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.0", type: FileImageInputComponent, selector: "file-image-input", usesInheritance: true, ngImport: i0, template: "<div *ngIf=\"!metadata.dragAndDrop && !metadata.preview\">\n <file-input\n (fileDataChangeEvent)=\"refreshFileData($event)\"\n [propertyValue]=\"propertyValue\"\n [metadata]=\"metadata\"\n [getValidationErrorMessage]=\"getValidationErrorMessage\"\n [isReadOnly]=\"isReadOnly\"\n [entity]=\"entity\"\n [key]=\"key\"\n >\n </file-input>\n</div>\n\n<div *ngIf=\"metadata.dragAndDrop || metadata.preview\" class=\"file-input mat-elevation-z8\">\n <file-input\n (fileDataChangeEvent)=\"refreshFileData($event)\"\n [propertyValue]=\"propertyValue\"\n [metadata]=\"metadata\"\n [getValidationErrorMessage]=\"getValidationErrorMessage\"\n [isReadOnly]=\"isReadOnly\"\n [entity]=\"entity\"\n [key]=\"key\"\n >\n </file-input>\n\n <div class=\"image-preview\" *ngIf=\"metadata.preview && metadata.multiple\">\n <i (click)=\"prev()\" [class.disabled]=\"imageIndex === 0\" class=\"prev-button fa-solid fa-angle-left\"></i>\n <img *ngIf=\"multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"multiPreviewImages?.[imageIndex]\">\n <img *ngIf=\"!multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"metadata.previewPlaceholderUrl ?? placeHolder\">\n <i (click)=\"next()\"\n [class.disabled]=\"!multiPreviewImages || !multiPreviewImages.length || imageIndex === (multiPreviewImages.length - 1)\"\n class=\"next-button fa-solid fa-angle-right\"\n >\n </i>\n </div>\n <div class=\"preview-nav\" *ngIf=\"metadata.preview && metadata.multiple\">\n <button type=\"button\" (click)=\"setIndex(imageIndex-4)\" mat-icon-button *ngIf=\"\n this.multiPreviewImages\n && multiPreviewImages[imageIndex-4]\n && imageIndex === (this.multiPreviewImages.length - 1)\"\n >\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 3}}</span>\n </button>\n <!-- eslint-disable-next-line @angular-eslint/template/conditional-complexity -->\n <button type=\"button\" (click)=\"setIndex(imageIndex-3)\" mat-icon-button *ngIf=\"this.multiPreviewImages\n && multiPreviewImages[imageIndex-3]\n && (\n imageIndex === (this.multiPreviewImages.length - 2)\n || imageIndex === (this.multiPreviewImages.length - 1)\n )\"\n >\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 2}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex-2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-2]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 1}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex-1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-1]\">\n <i class=\"dot\"></i>\n <span class=\"image-index\">{{imageIndex}}</span>\n </button>\n <button type=\"button\" mat-icon-button disabled>\n <i class=\"dot selected\"></i>\n <span class=\"image-index\">{{imageIndex + 1}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+1]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 2}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+2]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 3}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+3)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+3] && imageIndex <= 1\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 4}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+4)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+4] && imageIndex === 0\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 5}}</span>\n </button>\n </div>\n\n <div class=\"image-preview\" *ngIf=\"metadata.preview && !metadata.multiple\">\n <i class=\"prev-button disabled fa-solid fa-angle-left\"></i>\n <img class=\"mat-elevation-z2\" [src]=\"singlePreviewImage ?? metadata.previewPlaceholderUrl ?? placeHolder\">\n <i class=\"next-button disabled fa-solid fa-angle-right\"></i>\n </div>\n <div class=\"preview-nav\" *ngIf=\"metadata.preview && !metadata.multiple\">\n <button type=\"button\" disabled mat-icon-button>\n <span class=\"dot selected\"></span>\n <span class=\"image-index\">1</span>\n </button>\n </div>\n</div>", styles: [".file-input{margin-top:15px;margin-bottom:15px;padding:15px;border-radius:5px}.image-preview{height:250px;display:flex;align-items:center;padding-top:15px;padding-bottom:15px}.image-preview .prev-button{font-size:100px;margin-left:5px;color:#0000008a}.image-preview .next-button{font-size:100px;margin-right:5px;color:#0000008a}.image-preview .prev-button:hover,.image-preview .next-button:hover{cursor:pointer;color:#000000b3;transition:all .5s ease}.image-preview .prev-button.disabled,.image-preview .next-button.disabled{color:#00000042}.image-preview .prev-button.disabled:hover,.image-preview .next-button.disabled:hover{color:#00000042;transition:none;cursor:default}.image-preview img{max-width:calc(100% - 100px);max-height:100%;margin-left:auto;margin-right:auto;border-radius:3px}.preview-nav{text-align:center}.preview-nav button{display:inline-block;width:18px;height:18px;margin-left:5px;margin-right:5px}.preview-nav button .dot{height:100%;width:100%;background-color:#00000061;border-radius:50%;display:block}.preview-nav button .dot.selected{background-color:#0000008a}.preview-nav button .image-index{position:absolute;height:100%;width:100%;display:block;top:-11.5px;color:#fff}.preview-nav button:hover .dot{background-color:#0000008a;transition:all .3s ease}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i4.FileInputComponent, selector: "file-input", inputs: ["propertyValue", "entity", "key", "metadata", "getValidationErrorMessage", "isReadOnly"], outputs: ["fileDataChangeEvent"] }] });
|
|
96
|
+
}
|
|
97
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: FileImageInputComponent, decorators: [{
|
|
98
|
+
type: Component,
|
|
99
|
+
args: [{ selector: 'file-image-input', template: "<div *ngIf=\"!metadata.dragAndDrop && !metadata.preview\">\n <file-input\n (fileDataChangeEvent)=\"refreshFileData($event)\"\n [propertyValue]=\"propertyValue\"\n [metadata]=\"metadata\"\n [getValidationErrorMessage]=\"getValidationErrorMessage\"\n [isReadOnly]=\"isReadOnly\"\n [entity]=\"entity\"\n [key]=\"key\"\n >\n </file-input>\n</div>\n\n<div *ngIf=\"metadata.dragAndDrop || metadata.preview\" class=\"file-input mat-elevation-z8\">\n <file-input\n (fileDataChangeEvent)=\"refreshFileData($event)\"\n [propertyValue]=\"propertyValue\"\n [metadata]=\"metadata\"\n [getValidationErrorMessage]=\"getValidationErrorMessage\"\n [isReadOnly]=\"isReadOnly\"\n [entity]=\"entity\"\n [key]=\"key\"\n >\n </file-input>\n\n <div class=\"image-preview\" *ngIf=\"metadata.preview && metadata.multiple\">\n <i (click)=\"prev()\" [class.disabled]=\"imageIndex === 0\" class=\"prev-button fa-solid fa-angle-left\"></i>\n <img *ngIf=\"multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"multiPreviewImages?.[imageIndex]\">\n <img *ngIf=\"!multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"metadata.previewPlaceholderUrl ?? placeHolder\">\n <i (click)=\"next()\"\n [class.disabled]=\"!multiPreviewImages || !multiPreviewImages.length || imageIndex === (multiPreviewImages.length - 1)\"\n class=\"next-button fa-solid fa-angle-right\"\n >\n </i>\n </div>\n <div class=\"preview-nav\" *ngIf=\"metadata.preview && metadata.multiple\">\n <button type=\"button\" (click)=\"setIndex(imageIndex-4)\" mat-icon-button *ngIf=\"\n this.multiPreviewImages\n && multiPreviewImages[imageIndex-4]\n && imageIndex === (this.multiPreviewImages.length - 1)\"\n >\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 3}}</span>\n </button>\n <!-- eslint-disable-next-line @angular-eslint/template/conditional-complexity -->\n <button type=\"button\" (click)=\"setIndex(imageIndex-3)\" mat-icon-button *ngIf=\"this.multiPreviewImages\n && multiPreviewImages[imageIndex-3]\n && (\n imageIndex === (this.multiPreviewImages.length - 2)\n || imageIndex === (this.multiPreviewImages.length - 1)\n )\"\n >\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 2}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex-2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-2]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex - 1}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex-1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-1]\">\n <i class=\"dot\"></i>\n <span class=\"image-index\">{{imageIndex}}</span>\n </button>\n <button type=\"button\" mat-icon-button disabled>\n <i class=\"dot selected\"></i>\n <span class=\"image-index\">{{imageIndex + 1}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+1]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 2}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+2]\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 3}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+3)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+3] && imageIndex <= 1\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 4}}</span>\n </button>\n <button type=\"button\" (click)=\"setIndex(imageIndex+4)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+4] && imageIndex === 0\">\n <span class=\"dot\"></span>\n <span class=\"image-index\">{{imageIndex + 5}}</span>\n </button>\n </div>\n\n <div class=\"image-preview\" *ngIf=\"metadata.preview && !metadata.multiple\">\n <i class=\"prev-button disabled fa-solid fa-angle-left\"></i>\n <img class=\"mat-elevation-z2\" [src]=\"singlePreviewImage ?? metadata.previewPlaceholderUrl ?? placeHolder\">\n <i class=\"next-button disabled fa-solid fa-angle-right\"></i>\n </div>\n <div class=\"preview-nav\" *ngIf=\"metadata.preview && !metadata.multiple\">\n <button type=\"button\" disabled mat-icon-button>\n <span class=\"dot selected\"></span>\n <span class=\"image-index\">1</span>\n </button>\n </div>\n</div>", styles: [".file-input{margin-top:15px;margin-bottom:15px;padding:15px;border-radius:5px}.image-preview{height:250px;display:flex;align-items:center;padding-top:15px;padding-bottom:15px}.image-preview .prev-button{font-size:100px;margin-left:5px;color:#0000008a}.image-preview .next-button{font-size:100px;margin-right:5px;color:#0000008a}.image-preview .prev-button:hover,.image-preview .next-button:hover{cursor:pointer;color:#000000b3;transition:all .5s ease}.image-preview .prev-button.disabled,.image-preview .next-button.disabled{color:#00000042}.image-preview .prev-button.disabled:hover,.image-preview .next-button.disabled:hover{color:#00000042;transition:none;cursor:default}.image-preview img{max-width:calc(100% - 100px);max-height:100%;margin-left:auto;margin-right:auto;border-radius:3px}.preview-nav{text-align:center}.preview-nav button{display:inline-block;width:18px;height:18px;margin-left:5px;margin-right:5px}.preview-nav button .dot{height:100%;width:100%;background-color:#00000061;border-radius:50%;display:block}.preview-nav button .dot.selected{background-color:#0000008a}.preview-nav button .image-index{position:absolute;height:100%;width:100%;display:block;top:-11.5px;color:#fff}.preview-nav button:hover .dot{background-color:#0000008a;transition:all .3s ease}\n"] }]
|
|
100
|
+
}], ctorParameters: function () { return [{ type: i1.HttpClient }]; } });
|
|
101
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-image-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-material-entity/src/components/input/file/file-image-input/file-image-input.component.ts","../../../../../../../projects/ngx-material-entity/src/components/input/file/file-image-input/file-image-input.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable jsdoc/require-jsdoc */\nimport { HttpClient } from '@angular/common/http';\nimport { Component, OnInit } from '@angular/core';\nimport { BaseEntityType } from '../../../../classes/entity.model';\nimport { DecoratorTypes } from '../../../../decorators/base/decorator-types.enum';\nimport { FileData } from '../../../../decorators/file/file-decorator.data';\nimport { ReflectUtilities } from '../../../../encapsulation/reflect.utilities';\nimport { placeholder } from '../../../../mocks/placeholder-data.png';\nimport { EntityUtilities } from '../../../../utilities/entity.utilities';\nimport { FileUtilities } from '../../../../utilities/file.utilities';\nimport { NgxMatEntityBaseInputComponent } from '../../base-input.component';\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'file-image-input',\n    templateUrl: './file-image-input.component.html',\n    styleUrls: ['./file-image-input.component.scss']\n})\nexport class FileImageInputComponent<EntityType extends BaseEntityType<EntityType>>\n    extends NgxMatEntityBaseInputComponent<EntityType, DecoratorTypes.FILE_IMAGE, FileData | FileData[]> implements OnInit {\n\n    get multiPreviewImages(): string[] | undefined {\n        return ReflectUtilities.getMetadata(EntityUtilities.MULTI_PREVIEW_IMAGES_KEY, this.entity, this.key) as string[] | undefined;\n    }\n    set multiPreviewImages(value: string[] | undefined) {\n        ReflectUtilities.defineMetadata(EntityUtilities.MULTI_PREVIEW_IMAGES_KEY, value, this.entity, this.key);\n    }\n\n    get singlePreviewImage(): string | undefined {\n        return ReflectUtilities.getMetadata(EntityUtilities.SINGLE_PREVIEW_IMAGE_KEY, this.entity, this.key) as string | undefined;\n    }\n    set singlePreviewImage(value: string | undefined) {\n        ReflectUtilities.defineMetadata(EntityUtilities.SINGLE_PREVIEW_IMAGE_KEY, value, this.entity, this.key);\n    }\n\n    imageIndex: number = 0;\n    placeHolder: string = placeholder;\n\n    constructor(private readonly http: HttpClient) {\n        super();\n    }\n\n    private async setSinglePreviewImage(): Promise<void> {\n        if (this.propertyValue) {\n            this.propertyValue = await FileUtilities.getFileData(this.propertyValue as FileData, this.http);\n            this.singlePreviewImage = await FileUtilities.getDataURLFromFile(this.propertyValue.file);\n        }\n        else {\n            this.singlePreviewImage = undefined;\n        }\n    }\n\n    private async setMultiPreviewImages(index: number): Promise<void> {\n        const multiFileData: FileData[] | undefined = this.propertyValue as FileData[] | undefined;\n        const previewImages: string[] = [];\n        if (multiFileData?.length) {\n            for (let i: number = 0; i < multiFileData.length; i++) {\n                if (i === index) {\n                    multiFileData[index] = await FileUtilities.getFileData(multiFileData[index], this.http);\n                    previewImages.push(await FileUtilities.getDataURLFromFile(multiFileData[index].file) as string);\n                }\n                else {\n                    previewImages.push('empty');\n                }\n            }\n        }\n        this.multiPreviewImages = previewImages;\n    }\n\n    async refreshFileData(fileData?: FileData | FileData[]): Promise<void> {\n        this.propertyValue = fileData;\n        this.emitChange();\n        if (this.metadata.multiple) {\n            fileData = (fileData as FileData[] | undefined);\n            if (!fileData?.[this.imageIndex]) {\n                this.imageIndex = 0;\n            }\n            await this.setMultiPreviewImages(this.imageIndex);\n        }\n        else {\n            await this.setSinglePreviewImage();\n        }\n    }\n\n    async prev(): Promise<void> {\n        if (this.imageIndex <= 0) {\n            return;\n        }\n        await this.setMultiPreviewImages(this.imageIndex - 1);\n        this.imageIndex--;\n    }\n\n    async next(): Promise<void> {\n        if (!this.multiPreviewImages?.length) {\n            return;\n        }\n        if (this.imageIndex === (this.multiPreviewImages.length - 1)) {\n            return;\n        }\n        await this.setMultiPreviewImages(this.imageIndex + 1);\n        this.imageIndex++;\n    }\n\n    async setIndex(index: number): Promise<void> {\n        await this.setMultiPreviewImages(index);\n        this.imageIndex = index;\n    }\n}","<div *ngIf=\"!metadata.dragAndDrop && !metadata.preview\">\n    <file-input\n        (fileDataChangeEvent)=\"refreshFileData($event)\"\n        [propertyValue]=\"propertyValue\"\n        [metadata]=\"metadata\"\n        [getValidationErrorMessage]=\"getValidationErrorMessage\"\n        [isReadOnly]=\"isReadOnly\"\n        [entity]=\"entity\"\n        [key]=\"key\"\n    >\n    </file-input>\n</div>\n\n<div *ngIf=\"metadata.dragAndDrop || metadata.preview\" class=\"file-input mat-elevation-z8\">\n    <file-input\n        (fileDataChangeEvent)=\"refreshFileData($event)\"\n        [propertyValue]=\"propertyValue\"\n        [metadata]=\"metadata\"\n        [getValidationErrorMessage]=\"getValidationErrorMessage\"\n        [isReadOnly]=\"isReadOnly\"\n        [entity]=\"entity\"\n        [key]=\"key\"\n    >\n    </file-input>\n\n    <div class=\"image-preview\" *ngIf=\"metadata.preview && metadata.multiple\">\n        <i (click)=\"prev()\" [class.disabled]=\"imageIndex === 0\" class=\"prev-button fa-solid fa-angle-left\"></i>\n        <img *ngIf=\"multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"multiPreviewImages?.[imageIndex]\">\n        <img *ngIf=\"!multiPreviewImages?.[imageIndex]\" class=\"mat-elevation-z2\" [src]=\"metadata.previewPlaceholderUrl ?? placeHolder\">\n        <i (click)=\"next()\"\n            [class.disabled]=\"!multiPreviewImages || !multiPreviewImages.length || imageIndex === (multiPreviewImages.length - 1)\"\n            class=\"next-button fa-solid fa-angle-right\"\n        >\n        </i>\n    </div>\n    <div class=\"preview-nav\" *ngIf=\"metadata.preview && metadata.multiple\">\n        <button type=\"button\" (click)=\"setIndex(imageIndex-4)\" mat-icon-button *ngIf=\"\n            this.multiPreviewImages\n            && multiPreviewImages[imageIndex-4]\n            && imageIndex === (this.multiPreviewImages.length - 1)\"\n        >\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex - 3}}</span>\n        </button>\n        <!-- eslint-disable-next-line @angular-eslint/template/conditional-complexity -->\n        <button type=\"button\" (click)=\"setIndex(imageIndex-3)\" mat-icon-button *ngIf=\"this.multiPreviewImages\n            && multiPreviewImages[imageIndex-3]\n            && (\n                imageIndex === (this.multiPreviewImages.length - 2)\n                || imageIndex === (this.multiPreviewImages.length - 1)\n            )\"\n        >\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex - 2}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex-2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-2]\">\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex - 1}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex-1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex-1]\">\n            <i class=\"dot\"></i>\n            <span class=\"image-index\">{{imageIndex}}</span>\n        </button>\n        <button type=\"button\" mat-icon-button disabled>\n            <i class=\"dot selected\"></i>\n            <span class=\"image-index\">{{imageIndex + 1}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex+1)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+1]\">\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex + 2}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex+2)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+2]\">\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex + 3}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex+3)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+3] && imageIndex <= 1\">\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex + 4}}</span>\n        </button>\n        <button type=\"button\" (click)=\"setIndex(imageIndex+4)\" mat-icon-button *ngIf=\"multiPreviewImages?.[imageIndex+4] && imageIndex === 0\">\n            <span class=\"dot\"></span>\n            <span class=\"image-index\">{{imageIndex + 5}}</span>\n        </button>\n    </div>\n\n    <div class=\"image-preview\" *ngIf=\"metadata.preview && !metadata.multiple\">\n        <i class=\"prev-button disabled fa-solid fa-angle-left\"></i>\n        <img class=\"mat-elevation-z2\" [src]=\"singlePreviewImage ?? metadata.previewPlaceholderUrl ?? placeHolder\">\n        <i class=\"next-button disabled fa-solid fa-angle-right\"></i>\n    </div>\n    <div class=\"preview-nav\" *ngIf=\"metadata.preview && !metadata.multiple\">\n        <button type=\"button\" disabled mat-icon-button>\n            <span class=\"dot selected\"></span>\n            <span class=\"image-index\">1</span>\n        </button>\n    </div>\n</div>"]}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { Directive, EventEmitter, HostListener, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Adds drag and drop functionality to an element.
|
|
5
|
+
*/
|
|
6
|
+
export class DragDropDirective {
|
|
7
|
+
/**
|
|
8
|
+
* Emits the dropped files to the parent.
|
|
9
|
+
*/
|
|
10
|
+
files = new EventEmitter();
|
|
11
|
+
constructor() { }
|
|
12
|
+
/**
|
|
13
|
+
* Prevents the event default.
|
|
14
|
+
*
|
|
15
|
+
* @param evt - The Event when dragged files hover over the parent.
|
|
16
|
+
*/
|
|
17
|
+
onDragOver(evt) {
|
|
18
|
+
evt.preventDefault();
|
|
19
|
+
evt.stopPropagation();
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Prevents the event default.
|
|
23
|
+
*
|
|
24
|
+
* @param evt - The Event when dragged files leave the parent.
|
|
25
|
+
*/
|
|
26
|
+
onDragLeave(evt) {
|
|
27
|
+
evt.preventDefault();
|
|
28
|
+
evt.stopPropagation();
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Prevents the event default and emits the dropped files with the output.
|
|
32
|
+
*
|
|
33
|
+
* @param evt - The Event when files are dropped.
|
|
34
|
+
*/
|
|
35
|
+
onDrop(evt) {
|
|
36
|
+
evt.preventDefault();
|
|
37
|
+
evt.stopPropagation();
|
|
38
|
+
if (evt.dataTransfer && evt.dataTransfer.files.length > 0) {
|
|
39
|
+
this.files.emit(Array.from(evt.dataTransfer.files));
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: DragDropDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
43
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.0", type: DragDropDirective, selector: "[dragDrop]", outputs: { files: "files" }, host: { listeners: { "dragover": "onDragOver($event)", "dragleave": "onDragLeave($event)", "drop": "onDrop($event)" } }, ngImport: i0 });
|
|
44
|
+
}
|
|
45
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: DragDropDirective, decorators: [{
|
|
46
|
+
type: Directive,
|
|
47
|
+
args: [{
|
|
48
|
+
selector: '[dragDrop]'
|
|
49
|
+
}]
|
|
50
|
+
}], ctorParameters: function () { return []; }, propDecorators: { files: [{
|
|
51
|
+
type: Output
|
|
52
|
+
}], onDragOver: [{
|
|
53
|
+
type: HostListener,
|
|
54
|
+
args: ['dragover', ['$event']]
|
|
55
|
+
}], onDragLeave: [{
|
|
56
|
+
type: HostListener,
|
|
57
|
+
args: ['dragleave', ['$event']]
|
|
58
|
+
}], onDrop: [{
|
|
59
|
+
type: HostListener,
|
|
60
|
+
args: ['drop', ['$event']]
|
|
61
|
+
}] } });
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJhZ0Ryb3AuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LW1hdGVyaWFsLWVudGl0eS9zcmMvY29tcG9uZW50cy9pbnB1dC9maWxlL2ZpbGUtaW5wdXQvZHJhZ0Ryb3AuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTlFOztHQUVHO0FBSUgsTUFBTSxPQUFPLGlCQUFpQjtJQUMxQjs7T0FFRztJQUVILEtBQUssR0FBeUIsSUFBSSxZQUFZLEVBQVUsQ0FBQztJQUV6RCxnQkFBZ0IsQ0FBQztJQUVqQjs7OztPQUlHO0lBRUgsVUFBVSxDQUFDLEdBQWM7UUFDckIsR0FBRyxDQUFDLGNBQWMsRUFBRSxDQUFDO1FBQ3JCLEdBQUcsQ0FBQyxlQUFlLEVBQUUsQ0FBQztJQUMxQixDQUFDO0lBRUQ7Ozs7T0FJRztJQUVILFdBQVcsQ0FBQyxHQUFjO1FBQ3RCLEdBQUcsQ0FBQyxjQUFjLEVBQUUsQ0FBQztRQUNyQixHQUFHLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDMUIsQ0FBQztJQUVEOzs7O09BSUc7SUFFSCxNQUFNLENBQUMsR0FBYztRQUNqQixHQUFHLENBQUMsY0FBYyxFQUFFLENBQUM7UUFDckIsR0FBRyxDQUFDLGVBQWUsRUFBRSxDQUFDO1FBQ3RCLElBQUksR0FBRyxDQUFDLFlBQVksSUFBSSxHQUFHLENBQUMsWUFBWSxDQUFDLEtBQUssQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFO1lBQ3ZELElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1NBQ3ZEO0lBQ0wsQ0FBQzt1R0EzQ1EsaUJBQWlCOzJGQUFqQixpQkFBaUI7OzJGQUFqQixpQkFBaUI7a0JBSDdCLFNBQVM7bUJBQUM7b0JBQ1AsUUFBUSxFQUFFLFlBQVk7aUJBQ3pCOzBFQU1HLEtBQUs7c0JBREosTUFBTTtnQkFXUCxVQUFVO3NCQURULFlBQVk7dUJBQUMsVUFBVSxFQUFFLENBQUMsUUFBUSxDQUFDO2dCQVlwQyxXQUFXO3NCQURWLFlBQVk7dUJBQUMsV0FBVyxFQUFFLENBQUMsUUFBUSxDQUFDO2dCQVlyQyxNQUFNO3NCQURMLFlBQVk7dUJBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFdmVudEVtaXR0ZXIsIEhvc3RMaXN0ZW5lciwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogQWRkcyBkcmFnIGFuZCBkcm9wIGZ1bmN0aW9uYWxpdHkgdG8gYW4gZWxlbWVudC5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gICAgc2VsZWN0b3I6ICdbZHJhZ0Ryb3BdJ1xufSlcbmV4cG9ydCBjbGFzcyBEcmFnRHJvcERpcmVjdGl2ZSB7XG4gICAgLyoqXG4gICAgICogRW1pdHMgdGhlIGRyb3BwZWQgZmlsZXMgdG8gdGhlIHBhcmVudC5cbiAgICAgKi9cbiAgICBAT3V0cHV0KClcbiAgICBmaWxlczogRXZlbnRFbWl0dGVyPEZpbGVbXT4gPSBuZXcgRXZlbnRFbWl0dGVyPEZpbGVbXT4oKTtcblxuICAgIGNvbnN0cnVjdG9yKCkgeyB9XG5cbiAgICAvKipcbiAgICAgKiBQcmV2ZW50cyB0aGUgZXZlbnQgZGVmYXVsdC5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBldnQgLSBUaGUgRXZlbnQgd2hlbiBkcmFnZ2VkIGZpbGVzIGhvdmVyIG92ZXIgdGhlIHBhcmVudC5cbiAgICAgKi9cbiAgICBASG9zdExpc3RlbmVyKCdkcmFnb3ZlcicsIFsnJGV2ZW50J10pXG4gICAgb25EcmFnT3ZlcihldnQ6IERyYWdFdmVudCk6IHZvaWQge1xuICAgICAgICBldnQucHJldmVudERlZmF1bHQoKTtcbiAgICAgICAgZXZ0LnN0b3BQcm9wYWdhdGlvbigpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFByZXZlbnRzIHRoZSBldmVudCBkZWZhdWx0LlxuICAgICAqXG4gICAgICogQHBhcmFtIGV2dCAtIFRoZSBFdmVudCB3aGVuIGRyYWdnZWQgZmlsZXMgbGVhdmUgdGhlIHBhcmVudC5cbiAgICAgKi9cbiAgICBASG9zdExpc3RlbmVyKCdkcmFnbGVhdmUnLCBbJyRldmVudCddKVxuICAgIG9uRHJhZ0xlYXZlKGV2dDogRHJhZ0V2ZW50KTogdm9pZCB7XG4gICAgICAgIGV2dC5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgICBldnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogUHJldmVudHMgdGhlIGV2ZW50IGRlZmF1bHQgYW5kIGVtaXRzIHRoZSBkcm9wcGVkIGZpbGVzIHdpdGggdGhlIG91dHB1dC5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBldnQgLSBUaGUgRXZlbnQgd2hlbiBmaWxlcyBhcmUgZHJvcHBlZC5cbiAgICAgKi9cbiAgICBASG9zdExpc3RlbmVyKCdkcm9wJywgWyckZXZlbnQnXSlcbiAgICBvbkRyb3AoZXZ0OiBEcmFnRXZlbnQpOiB2b2lkIHtcbiAgICAgICAgZXZ0LnByZXZlbnREZWZhdWx0KCk7XG4gICAgICAgIGV2dC5zdG9wUHJvcGFnYXRpb24oKTtcbiAgICAgICAgaWYgKGV2dC5kYXRhVHJhbnNmZXIgJiYgZXZ0LmRhdGFUcmFuc2Zlci5maWxlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICB0aGlzLmZpbGVzLmVtaXQoQXJyYXkuZnJvbShldnQuZGF0YVRyYW5zZmVyLmZpbGVzKSk7XG4gICAgICAgIH1cbiAgICB9XG59Il19
|
|
@@ -0,0 +1,206 @@
|
|
|
1
|
+
/* eslint-disable jsdoc/require-jsdoc */
|
|
2
|
+
import { HttpClient } from '@angular/common/http';
|
|
3
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
4
|
+
import { MatDialog } from '@angular/material/dialog';
|
|
5
|
+
import { LodashUtilities } from '../../../../encapsulation/lodash.utilities';
|
|
6
|
+
import { ReflectUtilities } from '../../../../encapsulation/reflect.utilities';
|
|
7
|
+
import { EntityUtilities } from '../../../../utilities/entity.utilities';
|
|
8
|
+
import { FileUtilities } from '../../../../utilities/file.utilities';
|
|
9
|
+
import { NgxMatEntityConfirmDialogComponent } from '../../../confirm-dialog/confirm-dialog.component';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
import * as i1 from "@angular/material/dialog";
|
|
12
|
+
import * as i2 from "@angular/common/http";
|
|
13
|
+
import * as i3 from "@angular/common";
|
|
14
|
+
import * as i4 from "@angular/material/form-field";
|
|
15
|
+
import * as i5 from "@angular/forms";
|
|
16
|
+
import * as i6 from "@angular/material/chips";
|
|
17
|
+
import * as i7 from "@angular/material/button";
|
|
18
|
+
import * as i8 from "./dragDrop.directive";
|
|
19
|
+
export class FileInputComponent {
|
|
20
|
+
dialog;
|
|
21
|
+
http;
|
|
22
|
+
get filenames() {
|
|
23
|
+
return ReflectUtilities.getMetadata(EntityUtilities.FILENAMES_KEY, this.entity, this.key);
|
|
24
|
+
}
|
|
25
|
+
set filenames(value) {
|
|
26
|
+
ReflectUtilities.defineMetadata(EntityUtilities.FILENAMES_KEY, value, this.entity, this.key);
|
|
27
|
+
}
|
|
28
|
+
FileUtilities = FileUtilities;
|
|
29
|
+
propertyValue;
|
|
30
|
+
entity;
|
|
31
|
+
key;
|
|
32
|
+
metadata;
|
|
33
|
+
getValidationErrorMessage;
|
|
34
|
+
isReadOnly;
|
|
35
|
+
fileDataChangeEvent = new EventEmitter();
|
|
36
|
+
constructor(dialog, http) {
|
|
37
|
+
this.dialog = dialog;
|
|
38
|
+
this.http = http;
|
|
39
|
+
}
|
|
40
|
+
async ngOnInit() {
|
|
41
|
+
if (this.metadata.multiple) {
|
|
42
|
+
this.initMultiFile();
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
this.initSingleFile();
|
|
46
|
+
}
|
|
47
|
+
this.fileDataChangeEvent.emit(this.propertyValue);
|
|
48
|
+
}
|
|
49
|
+
initMultiFile() {
|
|
50
|
+
if (this.propertyValue) {
|
|
51
|
+
this.filenames = this.propertyValue.map(f => f.name);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
initSingleFile() {
|
|
55
|
+
if (this.propertyValue) {
|
|
56
|
+
this.filenames = [this.propertyValue.name];
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
async setFileFromInput(event) {
|
|
60
|
+
const files = event.target.files ?? [];
|
|
61
|
+
await this.setFile(Array.from(files));
|
|
62
|
+
}
|
|
63
|
+
async setFile(files) {
|
|
64
|
+
// validation done inline
|
|
65
|
+
if (files.find(f => !FileUtilities.isMimeTypeValid(f.type, this.metadata.allowedMimeTypes))) {
|
|
66
|
+
this.dialog.open(NgxMatEntityConfirmDialogComponent, {
|
|
67
|
+
data: this.metadata.mimeTypeErrorDialog,
|
|
68
|
+
autoFocus: false,
|
|
69
|
+
restoreFocus: false
|
|
70
|
+
});
|
|
71
|
+
this.resetFileInputs();
|
|
72
|
+
return;
|
|
73
|
+
}
|
|
74
|
+
if (files.find(f => FileUtilities.transformToMegaBytes(f.size, 'B') > this.metadata.maxSize)) {
|
|
75
|
+
this.dialog.open(NgxMatEntityConfirmDialogComponent, {
|
|
76
|
+
data: this.metadata.maxSizeErrorDialog,
|
|
77
|
+
autoFocus: false,
|
|
78
|
+
restoreFocus: false
|
|
79
|
+
});
|
|
80
|
+
this.resetFileInputs();
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
let fileSizeTotal = 0;
|
|
84
|
+
for (const file of files) {
|
|
85
|
+
fileSizeTotal += file.size;
|
|
86
|
+
}
|
|
87
|
+
if (FileUtilities.transformToMegaBytes(fileSizeTotal, 'B') > this.metadata.maxSizeTotal) {
|
|
88
|
+
this.dialog.open(NgxMatEntityConfirmDialogComponent, {
|
|
89
|
+
data: this.metadata.maxSizeTotalErrorDialog,
|
|
90
|
+
autoFocus: false,
|
|
91
|
+
restoreFocus: false
|
|
92
|
+
});
|
|
93
|
+
this.resetFileInputs();
|
|
94
|
+
return;
|
|
95
|
+
}
|
|
96
|
+
if (this.metadata.multiple) {
|
|
97
|
+
await this.setMultiFile(Array.from(files));
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
await this.setSingleFile(files[0]);
|
|
101
|
+
}
|
|
102
|
+
this.fileDataChangeEvent.emit(this.propertyValue);
|
|
103
|
+
}
|
|
104
|
+
resetFileInputs() {
|
|
105
|
+
this.filenames = undefined;
|
|
106
|
+
this.propertyValue = undefined;
|
|
107
|
+
this.fileDataChangeEvent.emit(this.propertyValue);
|
|
108
|
+
}
|
|
109
|
+
async setMultiFile(files) {
|
|
110
|
+
const data = [];
|
|
111
|
+
for (const file of files) {
|
|
112
|
+
const fileData = {
|
|
113
|
+
file: file,
|
|
114
|
+
name: file.name,
|
|
115
|
+
type: file.type,
|
|
116
|
+
size: file.size
|
|
117
|
+
};
|
|
118
|
+
data.push(fileData);
|
|
119
|
+
}
|
|
120
|
+
this.propertyValue = LodashUtilities.cloneDeep(data);
|
|
121
|
+
this.filenames = this.propertyValue.map(f => f.name);
|
|
122
|
+
}
|
|
123
|
+
async setSingleFile(file) {
|
|
124
|
+
this.propertyValue = {
|
|
125
|
+
file: file,
|
|
126
|
+
name: file.name,
|
|
127
|
+
type: file.type,
|
|
128
|
+
size: file.size
|
|
129
|
+
};
|
|
130
|
+
this.filenames = [this.propertyValue.name];
|
|
131
|
+
}
|
|
132
|
+
removeFile(name) {
|
|
133
|
+
if (this.isReadOnly) {
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
if (this.metadata.multiple) {
|
|
137
|
+
this.filenames?.splice(this.filenames.indexOf(name), 1);
|
|
138
|
+
if (!this.filenames?.length) {
|
|
139
|
+
this.filenames = undefined;
|
|
140
|
+
}
|
|
141
|
+
const fileDataToRemove = this.propertyValue.find(f => f.name === name);
|
|
142
|
+
this.propertyValue.splice(this.propertyValue.indexOf(fileDataToRemove), 1);
|
|
143
|
+
if (!this.propertyValue.length) {
|
|
144
|
+
this.propertyValue = undefined;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
else {
|
|
148
|
+
this.filenames = undefined;
|
|
149
|
+
this.propertyValue = undefined;
|
|
150
|
+
}
|
|
151
|
+
this.fileDataChangeEvent.emit(this.propertyValue);
|
|
152
|
+
}
|
|
153
|
+
async downloadFile(name) {
|
|
154
|
+
if (this.metadata.multiple && this.propertyValue.length) {
|
|
155
|
+
const foundFileData = this.propertyValue.find(f => f.name === name);
|
|
156
|
+
// the index need to be saved in a constant because we edit foundFileData
|
|
157
|
+
// => .indexOf() returns undefined.
|
|
158
|
+
const index = this.propertyValue.indexOf(foundFileData);
|
|
159
|
+
this.propertyValue[index] = await FileUtilities.getFileData(foundFileData, this.http);
|
|
160
|
+
FileUtilities.downloadSingleFile(this.propertyValue[index]);
|
|
161
|
+
}
|
|
162
|
+
else if (this.propertyValue) {
|
|
163
|
+
this.propertyValue = await FileUtilities.getFileData(this.propertyValue, this.http);
|
|
164
|
+
FileUtilities.downloadSingleFile(this.propertyValue);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
downloadAllEnabled() {
|
|
168
|
+
if (!this.metadata.multiple) {
|
|
169
|
+
return false;
|
|
170
|
+
}
|
|
171
|
+
if (!this.propertyValue) {
|
|
172
|
+
return false;
|
|
173
|
+
}
|
|
174
|
+
if (this.propertyValue.length < 2) {
|
|
175
|
+
return false;
|
|
176
|
+
}
|
|
177
|
+
return true;
|
|
178
|
+
}
|
|
179
|
+
async downloadAll() {
|
|
180
|
+
if (this.propertyValue.length) {
|
|
181
|
+
// eslint-disable-next-line max-len
|
|
182
|
+
void FileUtilities.downloadMultipleFiles(this.metadata.displayName, LodashUtilities.cloneDeep(this.propertyValue), this.http);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: FileInputComponent, deps: [{ token: i1.MatDialog }, { token: i2.HttpClient }], target: i0.ɵɵFactoryTarget.Component });
|
|
186
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.0", type: FileInputComponent, selector: "file-input", inputs: { propertyValue: "propertyValue", entity: "entity", key: "key", metadata: "metadata", getValidationErrorMessage: "getValidationErrorMessage", isReadOnly: "isReadOnly" }, outputs: { fileDataChangeEvent: "fileDataChangeEvent" }, ngImport: i0, template: "<input #fileInput\n type=\"file\" hidden\n [multiple]=\"metadata.multiple\"\n [accept]=\"FileUtilities.getAcceptString(metadata.allowedMimeTypes)\"\n (change)=\"setFileFromInput($event)\"\n>\n\n<mat-form-field floatLabel=\"always\">\n <mat-label>{{metadata.displayName}}</mat-label>\n <mat-chip-grid #chipGrid\n [(ngModel)]=\"filenames\" name=\"file\" #model=\"ngModel\"\n [required]=\"metadata.required(entity)\"\n >\n <mat-chip-row *ngFor=\"let name of filenames\" (removed)=\"removeFile(name)\">\n {{name}}\n <span class=\"mat-mdc-chip-remove mat-mdc-chip-trailing-icon ngx-mat-grey\" (click)=\"downloadFile(name)\">\n <i class=\"fas fa-download\"></i>\n </span>\n <button *ngIf=\"!isReadOnly\" type=\"button\" matChipRemove>\n <i class=\"{{metadata.deleteIcon}}\"></i>\n </button>\n </mat-chip-row>\n <input [matChipInputFor]=\"chipGrid\" [readonly]=\"true\" hidden>\n </mat-chip-grid>\n <button *ngIf=\"downloadAllEnabled()\" type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix (click)=\"downloadAll()\">\n <i class=\"fas fa-file-zipper\"></i>\n </button>\n <button type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n <i class=\"fas fa-upload\"></i>\n </button>\n <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>\n\n<div *ngIf=\"metadata.dragAndDrop && !isReadOnly\" class=\"drag-drop\" dragDrop (files)=\"setFile($event)\">\n <button type=\"button\" mat-icon-button [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n <i class=\"fas fa-file-arrow-up ngx-mat-grey\"></i>\n </button>\n</div>", styles: ["mat-form-field{width:100%}.ngx-mat-grey{opacity:1;color:#0000008a}.drag-drop{display:flex;align-items:center;justify-content:center;height:200px;border:2px dashed rgba(0,0,0,.54);border-radius:15px;margin-top:5px;margin-bottom:5px}.drag-drop i{font-size:30px}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i6.MatChipGrid, selector: "mat-chip-grid", inputs: ["tabIndex", "disabled", "placeholder", "required", "value", "errorStateMatcher"], outputs: ["change", "valueChange"] }, { kind: "directive", type: i6.MatChipInput, selector: "input[matChipInputFor]", inputs: ["matChipInputFor", "matChipInputAddOnBlur", "matChipInputSeparatorKeyCodes", "placeholder", "id", "disabled"], outputs: ["matChipInputTokenEnd"], exportAs: ["matChipInput", "matChipInputFor"] }, { kind: "directive", type: i6.MatChipRemove, selector: "[matChipRemove]" }, { kind: "component", type: i6.MatChipRow, selector: "mat-chip-row, [mat-chip-row], mat-basic-chip-row, [mat-basic-chip-row]", inputs: ["color", "disabled", "disableRipple", "tabIndex", "editable"], outputs: ["edited"] }, { kind: "component", type: i7.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "directive", type: i8.DragDropDirective, selector: "[dragDrop]", outputs: ["files"] }] });
|
|
187
|
+
}
|
|
188
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImport: i0, type: FileInputComponent, decorators: [{
|
|
189
|
+
type: Component,
|
|
190
|
+
args: [{ selector: 'file-input', template: "<input #fileInput\n type=\"file\" hidden\n [multiple]=\"metadata.multiple\"\n [accept]=\"FileUtilities.getAcceptString(metadata.allowedMimeTypes)\"\n (change)=\"setFileFromInput($event)\"\n>\n\n<mat-form-field floatLabel=\"always\">\n <mat-label>{{metadata.displayName}}</mat-label>\n <mat-chip-grid #chipGrid\n [(ngModel)]=\"filenames\" name=\"file\" #model=\"ngModel\"\n [required]=\"metadata.required(entity)\"\n >\n <mat-chip-row *ngFor=\"let name of filenames\" (removed)=\"removeFile(name)\">\n {{name}}\n <span class=\"mat-mdc-chip-remove mat-mdc-chip-trailing-icon ngx-mat-grey\" (click)=\"downloadFile(name)\">\n <i class=\"fas fa-download\"></i>\n </span>\n <button *ngIf=\"!isReadOnly\" type=\"button\" matChipRemove>\n <i class=\"{{metadata.deleteIcon}}\"></i>\n </button>\n </mat-chip-row>\n <input [matChipInputFor]=\"chipGrid\" [readonly]=\"true\" hidden>\n </mat-chip-grid>\n <button *ngIf=\"downloadAllEnabled()\" type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix (click)=\"downloadAll()\">\n <i class=\"fas fa-file-zipper\"></i>\n </button>\n <button type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n <i class=\"fas fa-upload\"></i>\n </button>\n <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>\n\n<div *ngIf=\"metadata.dragAndDrop && !isReadOnly\" class=\"drag-drop\" dragDrop (files)=\"setFile($event)\">\n <button type=\"button\" mat-icon-button [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n <i class=\"fas fa-file-arrow-up ngx-mat-grey\"></i>\n </button>\n</div>", styles: ["mat-form-field{width:100%}.ngx-mat-grey{opacity:1;color:#0000008a}.drag-drop{display:flex;align-items:center;justify-content:center;height:200px;border:2px dashed rgba(0,0,0,.54);border-radius:15px;margin-top:5px;margin-bottom:5px}.drag-drop i{font-size:30px}\n"] }]
|
|
191
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialog }, { type: i2.HttpClient }]; }, propDecorators: { propertyValue: [{
|
|
192
|
+
type: Input
|
|
193
|
+
}], entity: [{
|
|
194
|
+
type: Input
|
|
195
|
+
}], key: [{
|
|
196
|
+
type: Input
|
|
197
|
+
}], metadata: [{
|
|
198
|
+
type: Input
|
|
199
|
+
}], getValidationErrorMessage: [{
|
|
200
|
+
type: Input
|
|
201
|
+
}], isReadOnly: [{
|
|
202
|
+
type: Input
|
|
203
|
+
}], fileDataChangeEvent: [{
|
|
204
|
+
type: Output
|
|
205
|
+
}] } });
|
|
206
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-material-entity/src/components/input/file/file-input/file-input.component.ts","../../../../../../../projects/ngx-material-entity/src/components/input/file/file-input/file-input.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable jsdoc/require-jsdoc */\nimport { HttpClient } from '@angular/common/http';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { NgModel } from '@angular/forms';\nimport { MatDialog } from '@angular/material/dialog';\nimport { BaseEntityType } from '../../../../classes/entity.model';\nimport { DefaultFileDecoratorConfigInternal, FileDataWithFile, ImageFileDecoratorConfigInternal } from '../../../../decorators/file/file-decorator-internal.data';\nimport { FileData } from '../../../../decorators/file/file-decorator.data';\nimport { LodashUtilities } from '../../../../encapsulation/lodash.utilities';\nimport { ReflectUtilities } from '../../../../encapsulation/reflect.utilities';\nimport { EntityUtilities } from '../../../../utilities/entity.utilities';\nimport { FileUtilities } from '../../../../utilities/file.utilities';\nimport { NgxMatEntityConfirmDialogComponent } from '../../../confirm-dialog/confirm-dialog.component';\n\n@Component({\n    // eslint-disable-next-line @angular-eslint/component-selector\n    selector: 'file-input',\n    templateUrl: './file-input.component.html',\n    styleUrls: ['./file-input.component.scss']\n})\nexport class FileInputComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    get filenames(): string[] | undefined {\n        return ReflectUtilities.getMetadata(EntityUtilities.FILENAMES_KEY, this.entity, this.key) as string[] | undefined;\n    }\n    set filenames(value: string[] | undefined) {\n        ReflectUtilities.defineMetadata(EntityUtilities.FILENAMES_KEY, value, this.entity, this.key);\n    }\n\n    FileUtilities: typeof FileUtilities = FileUtilities;\n\n    @Input()\n    propertyValue!: FileData | FileData[] | undefined;\n\n    @Input()\n    entity!: EntityType;\n\n    @Input()\n    key!: keyof EntityType;\n\n    @Input()\n    metadata!: DefaultFileDecoratorConfigInternal | ImageFileDecoratorConfigInternal;\n\n    @Input()\n    getValidationErrorMessage!: (model: NgModel) => string;\n\n    @Input()\n    isReadOnly!: boolean;\n\n    @Output()\n    fileDataChangeEvent: EventEmitter<FileData | FileData[]> = new EventEmitter<FileData | FileData[]>();\n\n    constructor(private readonly dialog: MatDialog, private readonly http: HttpClient) { }\n\n    async ngOnInit(): Promise<void> {\n        if (this.metadata.multiple) {\n            this.initMultiFile();\n        }\n        else {\n            this.initSingleFile();\n        }\n        this.fileDataChangeEvent.emit(this.propertyValue);\n    }\n\n    private initMultiFile(): void {\n        if (this.propertyValue) {\n            this.filenames = (this.propertyValue as FileData[]).map(f => f.name);\n        }\n    }\n\n    private initSingleFile(): void {\n        if (this.propertyValue) {\n            this.filenames = [(this.propertyValue as FileData).name];\n        }\n    }\n\n    async setFileFromInput(event: Event): Promise<void> {\n        const files: FileList | [] = (event.target as HTMLInputElement).files ?? [];\n        await this.setFile(Array.from(files));\n    }\n\n    async setFile(files: File[]): Promise<void> {\n        // validation done inline\n        if (files.find(f => !FileUtilities.isMimeTypeValid(f.type, this.metadata.allowedMimeTypes))) {\n            this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n                data: this.metadata.mimeTypeErrorDialog,\n                autoFocus: false,\n                restoreFocus: false\n            });\n            this.resetFileInputs();\n            return;\n        }\n        if (files.find(f => FileUtilities.transformToMegaBytes(f.size, 'B') > this.metadata.maxSize)) {\n            this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n                data: this.metadata.maxSizeErrorDialog,\n                autoFocus: false,\n                restoreFocus: false\n            });\n            this.resetFileInputs();\n            return;\n        }\n        let fileSizeTotal: number = 0;\n        for (const file of files) {\n            fileSizeTotal += file.size;\n        }\n        if (FileUtilities.transformToMegaBytes(fileSizeTotal, 'B') > this.metadata.maxSizeTotal) {\n            this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n                data: this.metadata.maxSizeTotalErrorDialog,\n                autoFocus: false,\n                restoreFocus: false\n            });\n            this.resetFileInputs();\n            return;\n        }\n        if (this.metadata.multiple) {\n            await this.setMultiFile(Array.from(files));\n        }\n        else {\n            await this.setSingleFile(files[0]);\n        }\n        this.fileDataChangeEvent.emit(this.propertyValue);\n    }\n\n    private resetFileInputs(): void {\n        this.filenames = undefined;\n        this.propertyValue = undefined;\n        this.fileDataChangeEvent.emit(this.propertyValue);\n    }\n\n    private async setMultiFile(files: File[]): Promise<void> {\n        const data: FileData[] = [];\n        for (const file of files) {\n            const fileData: FileData = {\n                file: file,\n                name: file.name,\n                type: file.type,\n                size: file.size\n            };\n            data.push(fileData);\n        }\n        this.propertyValue = LodashUtilities.cloneDeep(data);\n        this.filenames = this.propertyValue.map(f => f.name);\n    }\n\n    private async setSingleFile(file: File): Promise<void> {\n        this.propertyValue = {\n            file: file,\n            name: file.name,\n            type: file.type,\n            size: file.size\n        };\n        this.filenames = [this.propertyValue.name];\n    }\n\n    removeFile(name: string): void {\n        if (this.isReadOnly) {\n            return;\n        }\n        if (this.metadata.multiple) {\n            this.filenames?.splice(this.filenames.indexOf(name), 1);\n            if (!this.filenames?.length) {\n                this.filenames = undefined;\n            }\n            const fileDataToRemove: FileData = (this.propertyValue as FileData[]).find(f => f.name === name) as FileData;\n            (this.propertyValue as FileData[]).splice((this.propertyValue as FileData[]).indexOf(fileDataToRemove), 1);\n            if (!(this.propertyValue as FileData[]).length) {\n                this.propertyValue = undefined;\n            }\n        }\n        else {\n            this.filenames = undefined;\n            this.propertyValue = undefined;\n        }\n        this.fileDataChangeEvent.emit(this.propertyValue);\n    }\n\n    async downloadFile(name: string): Promise<void> {\n        if (this.metadata.multiple && (this.propertyValue as FileData[]).length) {\n            const foundFileData: FileData = (this.propertyValue as FileData[]).find(f => f.name === name) as FileData;\n            // the index need to be saved in a constant because we edit foundFileData\n            // => .indexOf() returns undefined.\n            const index: number = (this.propertyValue as FileData[]).indexOf(foundFileData);\n            (this.propertyValue as FileData[])[index] = await FileUtilities.getFileData(foundFileData, this.http);\n            FileUtilities.downloadSingleFile((this.propertyValue as FileData[])[index] as FileDataWithFile);\n        }\n        else if (this.propertyValue) {\n            this.propertyValue = await FileUtilities.getFileData(this.propertyValue as FileData, this.http);\n            FileUtilities.downloadSingleFile(this.propertyValue);\n        }\n    }\n\n    downloadAllEnabled(): boolean {\n        if (!this.metadata.multiple) {\n            return false;\n        }\n        if (!this.propertyValue) {\n            return false;\n        }\n        if ((this.propertyValue as FileData[]).length < 2) {\n            return false;\n        }\n        return true;\n    }\n\n    async downloadAll(): Promise<void> {\n        if ((this.propertyValue as FileData[]).length) {\n            // eslint-disable-next-line max-len\n            void FileUtilities.downloadMultipleFiles(this.metadata.displayName, LodashUtilities.cloneDeep(this.propertyValue as FileData[]), this.http);\n        }\n    }\n}","<input #fileInput\n    type=\"file\" hidden\n    [multiple]=\"metadata.multiple\"\n    [accept]=\"FileUtilities.getAcceptString(metadata.allowedMimeTypes)\"\n    (change)=\"setFileFromInput($event)\"\n>\n\n<mat-form-field floatLabel=\"always\">\n    <mat-label>{{metadata.displayName}}</mat-label>\n    <mat-chip-grid #chipGrid\n        [(ngModel)]=\"filenames\" name=\"file\" #model=\"ngModel\"\n        [required]=\"metadata.required(entity)\"\n    >\n        <mat-chip-row *ngFor=\"let name of filenames\" (removed)=\"removeFile(name)\">\n            {{name}}\n            <span class=\"mat-mdc-chip-remove mat-mdc-chip-trailing-icon ngx-mat-grey\" (click)=\"downloadFile(name)\">\n                <i class=\"fas fa-download\"></i>\n            </span>\n            <button *ngIf=\"!isReadOnly\" type=\"button\" matChipRemove>\n                <i class=\"{{metadata.deleteIcon}}\"></i>\n            </button>\n        </mat-chip-row>\n        <input [matChipInputFor]=\"chipGrid\" [readonly]=\"true\" hidden>\n    </mat-chip-grid>\n    <button *ngIf=\"downloadAllEnabled()\" type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix (click)=\"downloadAll()\">\n        <i class=\"fas fa-file-zipper\"></i>\n    </button>\n    <button type=\"button\" class=\"ngx-mat-grey\" mat-icon-button matSuffix [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n        <i class=\"fas fa-upload\"></i>\n    </button>\n    <mat-error>{{getValidationErrorMessage(model)}}</mat-error>\n</mat-form-field>\n\n<div *ngIf=\"metadata.dragAndDrop && !isReadOnly\" class=\"drag-drop\" dragDrop (files)=\"setFile($event)\">\n    <button type=\"button\" mat-icon-button [disabled]=\"isReadOnly\" (click)=\"fileInput.click()\">\n        <i class=\"fas fa-file-arrow-up ngx-mat-grey\"></i>\n    </button>\n</div>"]}
|