ngx-material-entity 0.1.4 → 1.0.5

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 (156) hide show
  1. package/README.md +416 -39
  2. package/capsulation/lodash.utilities.d.ts +62 -0
  3. package/capsulation/reflect.utilities.d.ts +56 -0
  4. package/classes/base.builder.d.ts +2 -1
  5. package/classes/date.utilities.d.ts +18 -6
  6. package/classes/entity.model.d.ts +9 -2
  7. package/classes/entity.service.d.ts +38 -1
  8. package/classes/entity.utilities.d.ts +45 -16
  9. package/classes/file.utilities.d.ts +52 -0
  10. package/components/input/add-array-item-dialog-data.builder.d.ts +3 -2
  11. package/components/input/add-array-item-dialog-data.d.ts +2 -1
  12. package/components/input/array/array-date-input/array-date-input.component.d.ts +11 -0
  13. package/components/input/array/array-date-range-input/array-date-range-input.component.d.ts +20 -0
  14. package/components/input/array/array-date-time-input/array-date-time-input.component.d.ts +22 -0
  15. package/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.d.ts +50 -0
  16. package/components/input/array/array-string-chips-input/array-string-chips-input.component.d.ts +42 -0
  17. package/components/input/array/array-table.class.d.ts +48 -0
  18. package/components/input/base-input.component.d.ts +57 -0
  19. package/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.d.ts +10 -0
  20. package/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.d.ts +9 -0
  21. package/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.d.ts +10 -0
  22. package/components/input/custom/custom.component.d.ts +13 -0
  23. package/components/input/date/date-input/date-input.component.d.ts +11 -0
  24. package/components/input/date/date-range-input/date-range-input.component.d.ts +19 -0
  25. package/components/input/date/date-time-input/date-time-input.component.d.ts +30 -0
  26. package/components/input/file/file-default-input/file-default-input.component.d.ts +13 -0
  27. package/components/input/file/file-image-input/file-image-input.component.d.ts +22 -0
  28. package/components/input/file/file-input/dragDrop.directive.d.ts +32 -0
  29. package/components/input/file/file-input/file-input.component.d.ts +33 -0
  30. package/components/input/input.component.d.ts +49 -92
  31. package/components/input/input.module.d.ts +41 -16
  32. package/components/input/number/number-dropdown-input/number-dropdown-input.component.d.ts +9 -0
  33. package/components/input/number/number-input/number-input.component.d.ts +9 -0
  34. package/components/input/number/number-slider-input/number-slider-input.component.d.ts +9 -0
  35. package/components/input/string/string-autocomplete-input/string-autocomplete-input.component.d.ts +18 -0
  36. package/components/input/string/string-dropdown-input/string-dropdown-input.component.d.ts +9 -0
  37. package/components/input/string/string-input/string-input.component.d.ts +9 -0
  38. package/components/input/string/string-password-input/string-password-input.component.d.ts +15 -0
  39. package/components/input/string/string-textbox-input/string-textbox-input.component.d.ts +9 -0
  40. package/components/table/create-dialog/create-entity-dialog-data.builder.d.ts +3 -2
  41. package/components/table/create-dialog/create-entity-dialog-data.d.ts +2 -1
  42. package/components/table/create-dialog/create-entity-dialog.component.d.ts +7 -2
  43. package/components/table/edit-dialog/edit-dialog-data.builder.d.ts +3 -2
  44. package/components/table/edit-dialog/edit-entity-dialog-data.d.ts +6 -1
  45. package/components/table/edit-dialog/edit-entity-dialog.builder.d.ts +5 -3
  46. package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +6 -2
  47. package/components/table/table-data.builder.d.ts +9 -8
  48. package/components/table/table-data.d.ts +18 -10
  49. package/components/table/table.component.d.ts +2 -1
  50. package/decorators/array/array-decorator-internal.data.d.ts +71 -6
  51. package/decorators/array/array-decorator.data.d.ts +178 -12
  52. package/decorators/array/array.decorator.d.ts +3 -2
  53. package/decorators/base/base-property.decorator.d.ts +2 -3
  54. package/decorators/base/decorator-types.enum.d.ts +15 -5
  55. package/decorators/custom/custom-decorator-internal.data.d.ts +17 -0
  56. package/decorators/custom/custom-decorator.data.d.ts +37 -0
  57. package/decorators/custom/custom.decorator.d.ts +11 -0
  58. package/decorators/date/date-decorator-internal.data.d.ts +2 -2
  59. package/decorators/date/date.decorator.d.ts +8 -0
  60. package/decorators/file/file-decorator-internal.data.d.ts +92 -0
  61. package/decorators/file/file-decorator.data.d.ts +92 -0
  62. package/decorators/file/file.decorator.d.ts +9 -0
  63. package/decorators/number/number-decorator-internal.data.d.ts +20 -1
  64. package/decorators/number/number-decorator.data.d.ts +27 -1
  65. package/decorators/number/number.decorator.d.ts +2 -2
  66. package/decorators/object/object-decorator-internal.data.d.ts +2 -2
  67. package/decorators/object/object-decorator.data.d.ts +3 -3
  68. package/decorators/object/object.decorator.d.ts +2 -1
  69. package/decorators/string/string-decorator-internal.data.d.ts +14 -1
  70. package/decorators/string/string-decorator.data.d.ts +37 -1
  71. package/decorators/string/string.decorator.d.ts +2 -2
  72. package/esm2020/capsulation/lodash.utilities.mjs +75 -0
  73. package/esm2020/capsulation/reflect.utilities.mjs +69 -0
  74. package/esm2020/classes/base.builder.mjs +2 -3
  75. package/esm2020/classes/date.utilities.mjs +35 -15
  76. package/esm2020/classes/entity.model.mjs +5 -1
  77. package/esm2020/classes/entity.service.mjs +103 -6
  78. package/esm2020/classes/entity.utilities.mjs +241 -71
  79. package/esm2020/classes/file.utilities.mjs +124 -0
  80. package/esm2020/components/confirm-dialog/confirm-dialog-data.builder.mjs +4 -4
  81. package/esm2020/components/confirm-dialog/confirm-dialog.component.mjs +3 -3
  82. package/esm2020/components/input/add-array-item-dialog-data.builder.mjs +2 -2
  83. package/esm2020/components/input/add-array-item-dialog-data.mjs +1 -1
  84. package/esm2020/components/input/array/array-date-input/array-date-input.component.mjs +26 -0
  85. package/esm2020/components/input/array/array-date-range-input/array-date-range-input.component.mjs +50 -0
  86. package/esm2020/components/input/array/array-date-time-input/array-date-time-input.component.mjs +50 -0
  87. package/esm2020/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.mjs +103 -0
  88. package/esm2020/components/input/array/array-string-chips-input/array-string-chips-input.component.mjs +85 -0
  89. package/esm2020/components/input/array/array-table.class.mjs +104 -0
  90. package/esm2020/components/input/base-input.component.mjs +65 -0
  91. package/esm2020/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.mjs +21 -0
  92. package/esm2020/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.mjs +17 -0
  93. package/esm2020/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.mjs +21 -0
  94. package/esm2020/components/input/custom/custom.component.mjs +26 -0
  95. package/esm2020/components/input/date/date-input/date-input.component.mjs +22 -0
  96. package/esm2020/components/input/date/date-range-input/date-range-input.component.mjs +51 -0
  97. package/esm2020/components/input/date/date-time-input/date-time-input.component.mjs +63 -0
  98. package/esm2020/components/input/file/file-default-input/file-default-input.component.mjs +23 -0
  99. package/esm2020/components/input/file/file-image-input/file-image-input.component.mjs +84 -0
  100. package/esm2020/components/input/file/file-input/dragDrop.directive.mjs +64 -0
  101. package/esm2020/components/input/file/file-input/file-input.component.mjs +154 -0
  102. package/esm2020/components/input/input.component.mjs +137 -236
  103. package/esm2020/components/input/input.module.mjs +82 -6
  104. package/esm2020/components/input/number/number-dropdown-input/number-dropdown-input.component.mjs +18 -0
  105. package/esm2020/components/input/number/number-input/number-input.component.mjs +16 -0
  106. package/esm2020/components/input/number/number-slider-input/number-slider-input.component.mjs +17 -0
  107. package/esm2020/components/input/string/string-autocomplete-input/string-autocomplete-input.component.mjs +35 -0
  108. package/esm2020/components/input/string/string-dropdown-input/string-dropdown-input.component.mjs +18 -0
  109. package/esm2020/components/input/string/string-input/string-input.component.mjs +16 -0
  110. package/esm2020/components/input/string/string-password-input/string-password-input.component.mjs +36 -0
  111. package/esm2020/components/input/string/string-textbox-input/string-textbox-input.component.mjs +17 -0
  112. package/esm2020/components/table/create-dialog/create-dialog-data.builder.mjs +2 -2
  113. package/esm2020/components/table/create-dialog/create-entity-dialog-data.builder.mjs +1 -1
  114. package/esm2020/components/table/create-dialog/create-entity-dialog-data.mjs +1 -1
  115. package/esm2020/components/table/create-dialog/create-entity-dialog.component.mjs +15 -8
  116. package/esm2020/components/table/edit-dialog/edit-dialog-data.builder.mjs +2 -2
  117. package/esm2020/components/table/edit-dialog/edit-entity-dialog-data.mjs +1 -1
  118. package/esm2020/components/table/edit-dialog/edit-entity-dialog.builder.mjs +4 -3
  119. package/esm2020/components/table/edit-dialog/edit-entity-dialog.component.mjs +19 -10
  120. package/esm2020/components/table/table-data.builder.mjs +13 -10
  121. package/esm2020/components/table/table-data.mjs +1 -1
  122. package/esm2020/components/table/table.component.mjs +35 -35
  123. package/esm2020/decorators/array/array-decorator-internal.data.mjs +102 -14
  124. package/esm2020/decorators/array/array-decorator.data.mjs +2 -2
  125. package/esm2020/decorators/array/array.decorator.mjs +8 -2
  126. package/esm2020/decorators/base/base-property.decorator.mjs +4 -3
  127. package/esm2020/decorators/base/decorator-types.enum.mjs +9 -1
  128. package/esm2020/decorators/base/property-decorator-internal.data.mjs +10 -10
  129. package/esm2020/decorators/base/property-decorator.data.mjs +1 -1
  130. package/esm2020/decorators/boolean/boolean-decorator-internal.data.mjs +3 -3
  131. package/esm2020/decorators/custom/custom-decorator-internal.data.mjs +26 -0
  132. package/esm2020/decorators/custom/custom-decorator.data.mjs +2 -0
  133. package/esm2020/decorators/custom/custom.decorator.mjs +13 -0
  134. package/esm2020/decorators/date/date-decorator-internal.data.mjs +5 -5
  135. package/esm2020/decorators/date/date.decorator.mjs +21 -0
  136. package/esm2020/decorators/file/file-decorator-internal.data.mjs +98 -0
  137. package/esm2020/decorators/file/file-decorator.data.mjs +7 -0
  138. package/esm2020/decorators/file/file.decorator.mjs +22 -0
  139. package/esm2020/decorators/number/number-decorator-internal.data.mjs +24 -1
  140. package/esm2020/decorators/number/number-decorator.data.mjs +1 -1
  141. package/esm2020/decorators/number/number.decorator.mjs +9 -7
  142. package/esm2020/decorators/object/object-decorator-internal.data.mjs +1 -1
  143. package/esm2020/decorators/object/object-decorator.data.mjs +1 -1
  144. package/esm2020/decorators/object/object.decorator.mjs +1 -1
  145. package/esm2020/decorators/string/string-decorator-internal.data.mjs +16 -1
  146. package/esm2020/decorators/string/string-decorator.data.mjs +1 -1
  147. package/esm2020/decorators/string/string.decorator.mjs +13 -13
  148. package/esm2020/mocks/placeholder-data.png.mjs +3 -0
  149. package/esm2020/public-api.mjs +9 -1
  150. package/fesm2015/ngx-material-entity.mjs +2488 -524
  151. package/fesm2015/ngx-material-entity.mjs.map +1 -1
  152. package/fesm2020/ngx-material-entity.mjs +2363 -493
  153. package/fesm2020/ngx-material-entity.mjs.map +1 -1
  154. package/mocks/placeholder-data.png.d.ts +1 -0
  155. package/package.json +1 -1
  156. package/public-api.d.ts +8 -0
@@ -1,6 +1,8 @@
1
- import { cloneDeep, isEqual } from 'lodash';
2
1
  import { DecoratorTypes } from '../decorators/base/decorator-types.enum';
3
2
  import { DateUtilities } from './date.utilities';
3
+ import { ReflectUtilities } from '../capsulation/reflect.utilities';
4
+ import { LodashUtilities } from '../capsulation/lodash.utilities';
5
+ import { FileUtilities } from './file.utilities';
4
6
  /**
5
7
  * Contains HelperMethods around handling Entities and their property-metadata.
6
8
  */
@@ -37,6 +39,26 @@ export class EntityUtilities {
37
39
  }
38
40
  return res;
39
41
  }
42
+ /**
43
+ * Gets all properties on the given entity which are files.
44
+ *
45
+ * @param entity - The entity to check for file properties.
46
+ * @param omit - Whether to leave out values that are omitted for create or delete.
47
+ * @returns The keys of all file properties on the given entity.
48
+ */
49
+ static getFileProperties(entity, omit) {
50
+ const res = [];
51
+ for (const key of EntityUtilities.keysOf(entity)) {
52
+ const type = EntityUtilities.getPropertyType(entity, key);
53
+ if (type === DecoratorTypes.FILE_DEFAULT || type === DecoratorTypes.FILE_IMAGE) {
54
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
55
+ if (!(metadata.omitForCreate && omit === 'create') && !(metadata.omitForUpdate && omit === 'update')) {
56
+ res.push(key);
57
+ }
58
+ }
59
+ }
60
+ return res;
61
+ }
40
62
  /**
41
63
  * Gets the metadata included in an property.
42
64
  *
@@ -49,16 +71,11 @@ export class EntityUtilities {
49
71
  static getPropertyMetadata(entity, propertyKey,
50
72
  // eslint-disable-next-line @typescript-eslint/no-unused-vars
51
73
  type) {
52
- try {
53
- const metadata = Reflect.getMetadata('metadata', entity, propertyKey);
54
- if (!metadata) {
55
- throw new Error(`Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
56
- }
57
- return metadata;
58
- }
59
- catch (error) {
74
+ const metadata = ReflectUtilities.getMetadata('metadata', entity, propertyKey);
75
+ if (metadata == null) {
60
76
  throw new Error(`Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
61
77
  }
78
+ return metadata;
62
79
  }
63
80
  /**
64
81
  * Gets the type of the property-metadata.
@@ -70,8 +87,8 @@ export class EntityUtilities {
70
87
  */
71
88
  static getPropertyType(entity, propertyKey) {
72
89
  try {
73
- const propertyType = Reflect.getMetadata('type', entity, propertyKey);
74
- if (!propertyType) {
90
+ const propertyType = ReflectUtilities.getMetadata('type', entity, propertyKey);
91
+ if (propertyType == null) {
75
92
  throw new Error(`Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
76
93
  }
77
94
  return propertyType;
@@ -92,12 +109,10 @@ export class EntityUtilities {
92
109
  static new(target, entity) {
93
110
  for (const key in target) {
94
111
  const type = EntityUtilities.getPropertyType(target, key);
95
- // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
96
- let value = entity ? Reflect.get(entity, key) : undefined;
112
+ let value = entity ? ReflectUtilities.get(entity, key) : undefined;
97
113
  switch (type) {
98
114
  case DecoratorTypes.OBJECT:
99
115
  const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);
100
- // eslint-disable-next-line @typescript-eslint/no-unsafe-argument
101
116
  value = new objectMetadata.EntityClass(value);
102
117
  break;
103
118
  case DecoratorTypes.ARRAY:
@@ -115,7 +130,7 @@ export class EntityUtilities {
115
130
  default:
116
131
  break;
117
132
  }
118
- Reflect.set(target, key, value);
133
+ ReflectUtilities.set(target, key, value);
119
134
  }
120
135
  }
121
136
  /**
@@ -152,37 +167,52 @@ export class EntityUtilities {
152
167
  if (metadata.omitForUpdate && omit === 'update') {
153
168
  return true;
154
169
  }
155
- if (metadata.required && !entity[key]) {
170
+ if (metadata.required && entity[key] == null) {
156
171
  return false;
157
172
  }
158
173
  switch (type) {
159
174
  case DecoratorTypes.BOOLEAN_DROPDOWN:
175
+ break;
160
176
  case DecoratorTypes.BOOLEAN_CHECKBOX:
161
177
  case DecoratorTypes.BOOLEAN_TOGGLE:
162
- return true;
178
+ const entityBoolean = entity[key];
179
+ const booleanMetadata = metadata;
180
+ if (!EntityUtilities.isBooleanValid(entityBoolean, booleanMetadata)) {
181
+ return false;
182
+ }
183
+ break;
163
184
  case DecoratorTypes.STRING_DROPDOWN:
164
- return true;
185
+ break;
165
186
  case DecoratorTypes.STRING:
166
187
  case DecoratorTypes.STRING_AUTOCOMPLETE:
167
188
  const entityString = entity[key];
168
189
  const stringMetadata = metadata;
169
- if (!this.isStringValid(entityString, stringMetadata)) {
190
+ if (!EntityUtilities.isStringValid(entityString, stringMetadata)) {
170
191
  return false;
171
192
  }
172
193
  break;
173
194
  case DecoratorTypes.STRING_TEXTBOX:
174
195
  const entityTextbox = entity[key];
175
196
  const textboxMetadata = metadata;
176
- if (!this.isTextboxValid(entityTextbox, textboxMetadata)) {
197
+ if (!EntityUtilities.isTextboxValid(entityTextbox, textboxMetadata)) {
198
+ return false;
199
+ }
200
+ break;
201
+ case DecoratorTypes.STRING_PASSWORD:
202
+ const entityPassword = entity[key];
203
+ const passwordMetadata = metadata;
204
+ const confirmPassword = ReflectUtilities.getMetadata('confirmPassword', entity, key);
205
+ if (!EntityUtilities.isPasswordValid(entityPassword, passwordMetadata, confirmPassword)) {
177
206
  return false;
178
207
  }
179
208
  break;
180
209
  case DecoratorTypes.NUMBER_DROPDOWN:
181
210
  return true;
182
211
  case DecoratorTypes.NUMBER:
212
+ case DecoratorTypes.NUMBER_SLIDER:
183
213
  const entityNumber = entity[key];
184
214
  const numberMetadata = metadata;
185
- if (!this.isNumberValid(entityNumber, numberMetadata)) {
215
+ if (!EntityUtilities.isNumberValid(entityNumber, numberMetadata)) {
186
216
  return false;
187
217
  }
188
218
  break;
@@ -196,6 +226,9 @@ export class EntityUtilities {
196
226
  break;
197
227
  case DecoratorTypes.ARRAY_STRING_CHIPS:
198
228
  case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:
229
+ case DecoratorTypes.ARRAY_DATE:
230
+ case DecoratorTypes.ARRAY_DATE_TIME:
231
+ case DecoratorTypes.ARRAY_DATE_RANGE:
199
232
  case DecoratorTypes.ARRAY:
200
233
  const entityArray = entity[key];
201
234
  const arrayMetadata = metadata;
@@ -206,21 +239,36 @@ export class EntityUtilities {
206
239
  case DecoratorTypes.DATE:
207
240
  const entityDate = new Date(entity[key]);
208
241
  const dateMetadata = metadata;
209
- if (!this.isDateValid(entityDate, dateMetadata)) {
242
+ if (!EntityUtilities.isDateValid(entityDate, dateMetadata)) {
210
243
  return false;
211
244
  }
212
245
  break;
213
246
  case DecoratorTypes.DATE_RANGE:
214
- const entityDateRange = cloneDeep(entity[key]);
247
+ const entityDateRange = LodashUtilities.cloneDeep(entity[key]);
215
248
  const dateRangeMetadata = metadata;
216
- if (!this.isDateRangeValid(entityDateRange, dateRangeMetadata)) {
249
+ if (!EntityUtilities.isDateRangeValid(entityDateRange, dateRangeMetadata)) {
217
250
  return false;
218
251
  }
219
252
  break;
220
253
  case DecoratorTypes.DATE_TIME:
221
254
  const entityDateTime = new Date(entity[key]);
222
255
  const dateTimeMetadata = metadata;
223
- if (!this.isDateTimeValid(entityDateTime, dateTimeMetadata)) {
256
+ if (!EntityUtilities.isDateTimeValid(entityDateTime, dateTimeMetadata)) {
257
+ return false;
258
+ }
259
+ break;
260
+ case DecoratorTypes.FILE_DEFAULT:
261
+ case DecoratorTypes.FILE_IMAGE:
262
+ const entityFile = entity[key];
263
+ const entityFileMetadata = metadata;
264
+ if (!EntityUtilities.isFileDataValid(entityFile, entityFileMetadata)) {
265
+ return false;
266
+ }
267
+ break;
268
+ case DecoratorTypes.CUSTOM:
269
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
270
+ const customMetadata = metadata;
271
+ if (!customMetadata.isValid(entity[key], omit)) {
224
272
  return false;
225
273
  }
226
274
  break;
@@ -229,6 +277,12 @@ export class EntityUtilities {
229
277
  }
230
278
  return true;
231
279
  }
280
+ static isBooleanValid(value, metadata) {
281
+ if (metadata.required && !value) {
282
+ return false;
283
+ }
284
+ return true;
285
+ }
232
286
  static isStringValid(value, metadata) {
233
287
  if (metadata.maxLength && value.length > metadata.maxLength) {
234
288
  return false;
@@ -250,6 +304,21 @@ export class EntityUtilities {
250
304
  }
251
305
  return true;
252
306
  }
307
+ static isPasswordValid(value, metadata, confirmPassword) {
308
+ if (value !== confirmPassword) {
309
+ return false;
310
+ }
311
+ if (metadata.maxLength && value.length > metadata.maxLength) {
312
+ return false;
313
+ }
314
+ if (metadata.minLength && value.length < metadata.minLength) {
315
+ return false;
316
+ }
317
+ if (metadata.regex && !value.match(metadata.regex)) {
318
+ return false;
319
+ }
320
+ return true;
321
+ }
253
322
  static isNumberValid(value, metadata) {
254
323
  if (metadata.max && value > metadata.max) {
255
324
  return false;
@@ -272,8 +341,13 @@ export class EntityUtilities {
272
341
  return true;
273
342
  }
274
343
  static isDateRangeValid(value, metadata) {
275
- if (metadata.required && (!value.start || !value.end)) {
276
- return false;
344
+ if (metadata.required) {
345
+ if (!value.start) {
346
+ return false;
347
+ }
348
+ if (!value.end) {
349
+ return false;
350
+ }
277
351
  }
278
352
  value.start = new Date(value.start);
279
353
  value.end = new Date(value.end);
@@ -343,6 +417,27 @@ export class EntityUtilities {
343
417
  }
344
418
  return true;
345
419
  }
420
+ static isFileDataValid(value, metadata) {
421
+ const files = metadata.multiple ? value : [value];
422
+ let fileSizeTotal = 0;
423
+ // eslint-disable-next-line @typescript-eslint/prefer-for-of
424
+ for (let i = 0; i < files.length; i++) {
425
+ if (!files[i].name || !files[i].file && !files[i].url) {
426
+ return false;
427
+ }
428
+ if (!FileUtilities.isMimeTypeValid(files[i].type, metadata.allowedMimeTypes)) {
429
+ return false;
430
+ }
431
+ if (FileUtilities.transformToMegaBytes(files[i].size, 'B') > metadata.maxSize) {
432
+ return false;
433
+ }
434
+ fileSizeTotal += files[i].size;
435
+ if (FileUtilities.transformToMegaBytes(fileSizeTotal, 'B') > metadata.maxSizeTotal) {
436
+ return false;
437
+ }
438
+ }
439
+ return true;
440
+ }
346
441
  /**
347
442
  * Checks if an entity is "dirty" (if its values have changed).
348
443
  *
@@ -350,19 +445,21 @@ export class EntityUtilities {
350
445
  * @param entityPriorChanges - The entity before the changes.
351
446
  * @returns Whether or not the entity is dirty.
352
447
  */
353
- static dirty(entity, entityPriorChanges) {
448
+ static async isDirty(entity, entityPriorChanges) {
354
449
  if (!entityPriorChanges) {
355
450
  return false;
356
451
  }
357
452
  else {
358
- const differences = this.differencesForDirty(entity, entityPriorChanges);
453
+ const differences = await EntityUtilities.differencesForDirty(entity, entityPriorChanges);
359
454
  return differences.length ? true : false;
360
455
  }
361
456
  }
362
- static differencesForDirty(entity, entityPriorChanges) {
457
+ static async differencesForDirty(entity, entityPriorChanges) {
363
458
  const res = [];
364
459
  for (const key in entity) {
365
- if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {
460
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
461
+ const type = EntityUtilities.getPropertyType(entity, key);
462
+ if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {
366
463
  res.push({
367
464
  key: key,
368
465
  before: entityPriorChanges[key],
@@ -379,48 +476,121 @@ export class EntityUtilities {
379
476
  * @param entityPriorChanges - The second entity to compare.
380
477
  * @returns The difference between the two Entities in form of a Partial.
381
478
  */
382
- static difference(entity, entityPriorChanges) {
479
+ static async difference(entity, entityPriorChanges) {
383
480
  const res = {};
384
481
  for (const key in entity) {
385
- if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {
482
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
483
+ const type = EntityUtilities.getPropertyType(entity, key);
484
+ if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {
386
485
  res[key] = entity[key];
387
486
  }
388
487
  }
389
488
  return res;
390
489
  }
391
- static isEqual(value, valuePriorChanges, metadata) {
392
- if (this.isDateRange(value) && this.isDateRange(valuePriorChanges)) {
393
- const dateRange = cloneDeep(value);
394
- dateRange.start = new Date(value.start);
395
- dateRange.end = new Date(value.end);
396
- dateRange.values = DateUtilities.getDatesBetween(dateRange.start, dateRange.end, metadata);
397
- const dateRangePriorChanges = cloneDeep(valuePriorChanges);
398
- dateRangePriorChanges.start = new Date(valuePriorChanges.start);
399
- dateRangePriorChanges.end = new Date(valuePriorChanges.end);
400
- dateRangePriorChanges.values = DateUtilities.getDatesBetween(dateRangePriorChanges.start, dateRangePriorChanges.end, metadata);
401
- return isEqual(dateRange, dateRangePriorChanges);
402
- }
403
- if (metadata.displayStyle === 'date') {
404
- const date = new Date(DateUtilities.asDate(value));
405
- const datePriorChanges = new Date(DateUtilities.asDate(valuePriorChanges));
406
- date.setHours(0, 0, 0, 0);
407
- datePriorChanges.setHours(0, 0, 0, 0);
408
- return isEqual(date, datePriorChanges);
409
- }
410
- return isEqual(value, valuePriorChanges);
411
- }
412
- static isDateRange(value) {
413
- const dateRange = value;
414
- if (dateRange.start && dateRange.end) {
415
- try {
416
- new Date(dateRange.start);
417
- new Date(dateRange.end);
418
- return true;
490
+ /**
491
+ * Checks if two given values are equal.
492
+ * It uses the isEqual method from LodashUtilities and extends it with functionality regarding Dates.
493
+ *
494
+ * @param value - The updated value.
495
+ * @param valuePriorChanges - The value before any changes.
496
+ * @param metadata - The metadata of the property.
497
+ * @param type - The type of the property.
498
+ * @returns Whether or not the given values are equal.
499
+ */
500
+ static async isEqual(value, valuePriorChanges, metadata, type) {
501
+ switch (type) {
502
+ case DecoratorTypes.DATE_RANGE:
503
+ return EntityUtilities.isEqualDateRange(value, valuePriorChanges, metadata.filter);
504
+ case DecoratorTypes.DATE:
505
+ return EntityUtilities.isEqualDate(value, valuePriorChanges);
506
+ case DecoratorTypes.DATE_TIME:
507
+ return EntityUtilities.isEqualDateTime(value, valuePriorChanges);
508
+ case DecoratorTypes.ARRAY_DATE:
509
+ case DecoratorTypes.ARRAY_DATE_TIME:
510
+ return EntityUtilities.isEqualArrayDate(value, valuePriorChanges);
511
+ case DecoratorTypes.ARRAY_DATE_RANGE:
512
+ return EntityUtilities.isEqualArrayDateRange(value, valuePriorChanges, metadata.filter);
513
+ case DecoratorTypes.FILE_IMAGE:
514
+ case DecoratorTypes.FILE_DEFAULT:
515
+ return EntityUtilities.isEqualFile(value, valuePriorChanges, metadata.multiple);
516
+ case DecoratorTypes.CUSTOM:
517
+ // eslint-disable-next-line max-len, @typescript-eslint/no-explicit-any
518
+ return EntityUtilities.isEqualCustom(value, valuePriorChanges, metadata);
519
+ default:
520
+ return LodashUtilities.isEqual(value, valuePriorChanges);
521
+ }
522
+ }
523
+ static isEqualArrayDate(value, valuePriorChanges) {
524
+ const newValue = value.map(v => new Date(v)).sort();
525
+ const newValuePriorChanges = valuePriorChanges.map(v => new Date(v)).sort();
526
+ return LodashUtilities.isEqual(newValue, newValuePriorChanges);
527
+ }
528
+ static isEqualArrayDateRange(value, valuePriorChanges, filter) {
529
+ const dateRanges = value.sort();
530
+ const dateRangesPriorChanges = valuePriorChanges.sort();
531
+ if (dateRanges.length !== dateRangesPriorChanges.length) {
532
+ return false;
533
+ }
534
+ for (let i = 0; i < dateRanges.length; i++) {
535
+ if (!EntityUtilities.isEqualDateRange(dateRanges[i], dateRangesPriorChanges[i], filter)) {
536
+ return false;
537
+ }
538
+ }
539
+ return true;
540
+ }
541
+ static isEqualDateTime(value, valuePriorChanges) {
542
+ const date = new Date(value);
543
+ const datePriorChanges = new Date(valuePriorChanges);
544
+ return LodashUtilities.isEqual(date, datePriorChanges);
545
+ }
546
+ static isEqualDate(value, valuePriorChanges) {
547
+ const date = new Date(value);
548
+ const datePriorChanges = new Date(valuePriorChanges);
549
+ date.setHours(0, 0, 0, 0);
550
+ datePriorChanges.setHours(0, 0, 0, 0);
551
+ return LodashUtilities.isEqual(date, datePriorChanges);
552
+ }
553
+ static isEqualDateRange(value, valuePriorChanges, filter) {
554
+ const dateRange = LodashUtilities.cloneDeep(value);
555
+ dateRange.start = new Date(value.start);
556
+ dateRange.end = new Date(value.end);
557
+ dateRange.values = DateUtilities.getDatesBetween(dateRange.start, dateRange.end, filter);
558
+ const dateRangePriorChanges = LodashUtilities.cloneDeep(valuePriorChanges);
559
+ dateRangePriorChanges.start = new Date(valuePriorChanges.start);
560
+ dateRangePriorChanges.end = new Date(valuePriorChanges.end);
561
+ dateRangePriorChanges.values = DateUtilities.getDatesBetween(dateRangePriorChanges.start, dateRangePriorChanges.end, filter);
562
+ return LodashUtilities.isEqual(dateRange, dateRangePriorChanges);
563
+ }
564
+ // TODO: Find a way to use blobs with jest
565
+ /* istanbul ignore next */
566
+ static async isEqualFile(value, valuePriorChanges, multiple) {
567
+ const files = multiple ? value.sort() : [value].sort();
568
+ const filesPriorChanges = multiple ? valuePriorChanges.sort() : [valuePriorChanges].sort();
569
+ if (files.length !== filesPriorChanges.length) {
570
+ return false;
571
+ }
572
+ for (let i = 0; i < files.length; i++) {
573
+ // checks this before actually getting any files due to performance reasons.
574
+ if (!LodashUtilities.isEqual(files[i]?.name, filesPriorChanges[i]?.name)
575
+ || !LodashUtilities.isEqual(files[i]?.url, filesPriorChanges[i]?.url)) {
576
+ return false;
577
+ }
578
+ files[i] = filesPriorChanges[i].file && !files[i].file ? await FileUtilities.getFileData(files[i]) : files[i];
579
+ // eslint-disable-next-line max-len
580
+ filesPriorChanges[i] = files[i].file && !filesPriorChanges[i].file ? await FileUtilities.getFileData(filesPriorChanges[i]) : filesPriorChanges[i];
581
+ if (!LodashUtilities.isEqual(await files[i].file?.text(), await filesPriorChanges[i].file?.text())) {
582
+ return false;
419
583
  }
420
- catch (error) { }
421
- ;
422
584
  }
423
- return false;
585
+ return true;
586
+ }
587
+ static isEqualCustom(value, valuePriorChanges,
588
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
589
+ metadata) {
590
+ if (!metadata.isEqual(value, valuePriorChanges, metadata)) {
591
+ return false;
592
+ }
593
+ return true;
424
594
  }
425
595
  /**
426
596
  * Compare function for sorting entity keys by their order value.
@@ -442,7 +612,7 @@ export class EntityUtilities {
442
612
  else if (metadataB.position.order === -1) {
443
613
  return -1;
444
614
  }
445
- return ((metadataA.position.order) - (metadataB.position.order));
615
+ return metadataA.position.order - metadataB.position.order;
446
616
  }
447
617
  /**
448
618
  * Gets the bootstrap column values for "lg", "md", "sm".
@@ -471,7 +641,7 @@ export class EntityUtilities {
471
641
  */
472
642
  static resetChangesOnEntity(entity, entityPriorChanges) {
473
643
  for (const key in entityPriorChanges) {
474
- Reflect.set(entity, key, Reflect.get(entityPriorChanges, key));
644
+ ReflectUtilities.set(entity, key, ReflectUtilities.get(entityPriorChanges, key));
475
645
  }
476
646
  }
477
647
  /**
@@ -485,17 +655,17 @@ export class EntityUtilities {
485
655
  static getEntityRows(entity, hideOmitForCreate = false, hideOmitForEdit = false) {
486
656
  const res = [];
487
657
  const keys = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit);
488
- const numberOfRows = this.getNumberOfRows(keys, entity);
658
+ const numberOfRows = EntityUtilities.getNumberOfRows(keys, entity);
489
659
  for (let i = 1; i <= numberOfRows; i++) {
490
660
  const row = {
491
661
  row: i,
492
- keys: this.getKeysForRow(keys, entity, i)
662
+ keys: EntityUtilities.getKeysForRow(keys, entity, i)
493
663
  };
494
664
  res.push(row);
495
665
  }
496
666
  const lastRow = {
497
667
  row: numberOfRows + 1,
498
- keys: this.getKeysForRow(keys, entity, -1)
668
+ keys: EntityUtilities.getKeysForRow(keys, entity, -1)
499
669
  };
500
670
  res.push(lastRow);
501
671
  return res;
@@ -519,7 +689,7 @@ export class EntityUtilities {
519
689
  * @returns An array of keys of the entity.
520
690
  */
521
691
  static keysOf(entity, hideOmitForCreate = false, hideOmitForEdit = false) {
522
- let keys = Reflect.ownKeys(entity);
692
+ let keys = ReflectUtilities.ownKeys(entity);
523
693
  if (hideOmitForCreate) {
524
694
  const omitForCreateKeys = EntityUtilities.getOmitForCreate(entity);
525
695
  keys = keys.filter(k => !omitForCreateKeys.includes(k));
@@ -535,4 +705,4 @@ export class EntityUtilities {
535
705
  EntityUtilities.construct = EntityUtilities.new;
536
706
  // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc
537
707
  EntityUtilities.build = EntityUtilities.new;
538
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.utilities.js","sourceRoot":"","sources":["../../../../projects/ngx-material-entity/src/classes/entity.utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { cloneDeep, isEqual } from 'lodash';\nimport { DecoratorType, DecoratorTypes } from '../decorators/base/decorator-types.enum';\nimport { PropertyDecoratorConfigInternal } from '../decorators/base/property-decorator-internal.data';\nimport { EntityArrayDecoratorConfigInternal } from '../decorators/array/array-decorator-internal.data';\nimport { DefaultStringDecoratorConfigInternal, TextboxStringDecoratorConfigInternal } from '../decorators/string/string-decorator-internal.data';\nimport { DefaultNumberDecoratorConfigInternal } from '../decorators/number/number-decorator-internal.data';\nimport { DateRangeDateDecoratorConfigInternal, DateTimeDateDecoratorConfigInternal, DefaultDateDecoratorConfigInternal } from '../decorators/date/date-decorator-internal.data';\nimport { DateRange } from '../decorators/date/date-decorator.data';\nimport { Time } from '@angular/common';\nimport { DateUtilities } from './date.utilities';\n\n/**\n * Shows information about differences between two entities.\n */\ninterface Difference<EntityType extends object> {\n    /**\n     * The key where the two entities have different values.\n     */\n    key: keyof EntityType,\n    /**\n     * The value before any changes.\n     */\n    before: unknown,\n    /**\n     * The current value after changes.\n     */\n    after: unknown\n}\n\n/**\n * Contains HelperMethods around handling Entities and their property-metadata.\n */\nexport abstract class EntityUtilities {\n\n    /**\n     * Gets the properties to omit when updating the entity.\n     *\n     * @param entity - The entity to get the properties which should be left out for updating from.\n     * @returns The properties which should be left out for updating an Entity.\n     */\n    static getOmitForUpdate<EntityType extends object>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForUpdate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the properties to omit when creating new entities.\n     *\n     * @param entity - The entity to get the properties which should be left out for creating from.\n     * @returns The properties which should be left out for creating a new Entity.\n     */\n    static getOmitForCreate<EntityType extends object>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForCreate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the metadata included in an property.\n     *\n     * @param entity - The entity with the property to get the metadata from.\n     * @param propertyKey - The property on the given Entity to get the metadata from.\n     * @param type - For secure Typing, defines the returned PropertyConfig.\n     * @returns The metadata of the property.\n     * @throws When no metadata can be found for the given property.\n     */\n    static getPropertyMetadata<EntityType extends object, T extends DecoratorTypes>(\n        entity: EntityType,\n        propertyKey: keyof EntityType,\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n        type?: T\n    ): DecoratorType<T> {\n        try {\n            const metadata = Reflect.getMetadata('metadata', entity, propertyKey as string) as DecoratorType<T>;\n            if (!metadata) {\n                throw new Error(\n                    `Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n                );\n            }\n            return metadata;\n        }\n        catch (error) {\n            throw new Error(\n                `Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n    }\n\n    /**\n     * Gets the type of the property-metadata.\n     *\n     * @param entity - The entity with the property to get the type from.\n     * @param propertyKey - The property on the given Entity to get the type from.\n     * @returns The type of the metadata.\n     * @throws Will throw an error if no metadata can be found for the given property.\n     */\n    static getPropertyType<EntityType extends object>(\n        entity: EntityType, propertyKey: keyof EntityType\n    ): DecoratorTypes {\n        try {\n            const propertyType = Reflect.getMetadata('type', entity, propertyKey as string) as DecoratorTypes;\n            if (!propertyType) {\n                throw new Error(\n                    `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n                );\n            }\n            return propertyType;\n        }\n        catch (error) {\n            throw new Error(\n                `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n    }\n\n    /**\n     * Sets all property values based on a given entity data-object.\n     *\n     * @param target - The target object that needs to be constructed (if called inside an Entity constructor its usually this).\n     * @param entity - The data object to get the property values from.\n     * @alias new\n     * @alias build\n     * @alias construct\n     */\n    static new<EntityType extends object>(target: EntityType, entity?: EntityType): void {\n        for (const key in target) {\n            const type = EntityUtilities.getPropertyType(target, key);\n            // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n            let value = entity ? Reflect.get(entity, key) : undefined;\n            switch (type) {\n                case DecoratorTypes.OBJECT:\n                    const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);\n                    // eslint-disable-next-line @typescript-eslint/no-unsafe-argument\n                    value = new objectMetadata.EntityClass(value);\n                    break;\n                case DecoratorTypes.ARRAY:\n                    const inputArray: EntityType[] = value as EntityType[];\n                    const resArray: EntityType[] = [];\n                    if (inputArray) {\n                        const arrayMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);\n                        for (const item of inputArray) {\n                            const itemWithMetadata: EntityType = new arrayMetadata.EntityClass(item) as EntityType;\n                            resArray.push(itemWithMetadata);\n                        }\n                    }\n                    value = resArray;\n                    break;\n                default:\n                    break;\n            }\n            Reflect.set(target, key, value);\n        }\n    }\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static construct = EntityUtilities.new;\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static build = EntityUtilities.new;\n\n    /**\n     * Checks if the values on an entity are valid.\n     * Also checks all the validators given by the metadata (\"required\", \"maxLength\" etc.).\n     *\n     * @param entity - The entity to validate.\n     * @param omit - Whether to check for creating or editing validity.\n     * @returns Whether or not the entity is valid.\n     */\n    static isEntityValid<EntityType extends object>(entity: EntityType, omit: 'create' | 'update'): boolean {\n        for (const key in entity) {\n            if (!EntityUtilities.isPropertyValid(entity, key, omit)) {\n                return false;\n            }\n        }\n        return true;\n    }\n    /**\n     * Checks if a single property value is valid.\n     *\n     * @param entity - The entity where the property is from.\n     * @param key - The name of the property.\n     * @param omit - Whether to check if the given entity is valid for creation or updating.\n     * @returns Whether or not the property value is valid.\n     * @throws Throws when it extracts an unknown metadata type.\n     */\n    private static isPropertyValid<EntityType extends object>(\n        entity: EntityType,\n        key: keyof EntityType,\n        omit: 'create' | 'update'\n    ): boolean {\n        const type = EntityUtilities.getPropertyType(entity, key);\n        const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key, type);\n\n        if (metadata.omitForCreate && omit === 'create') {\n            return true;\n        }\n        if (metadata.omitForUpdate && omit === 'update') {\n            return true;\n        }\n        if (metadata.required && !entity[key]) {\n            return false;\n        }\n        switch (type) {\n            case DecoratorTypes.BOOLEAN_DROPDOWN:\n            case DecoratorTypes.BOOLEAN_CHECKBOX:\n            case DecoratorTypes.BOOLEAN_TOGGLE:\n                return true;\n            case DecoratorTypes.STRING_DROPDOWN:\n                return true;\n            case DecoratorTypes.STRING:\n            case DecoratorTypes.STRING_AUTOCOMPLETE:\n                const entityString = entity[key] as unknown as string;\n                const stringMetadata = metadata as DefaultStringDecoratorConfigInternal;\n                if (!this.isStringValid(entityString, stringMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_TEXTBOX:\n                const entityTextbox = entity[key] as unknown as string;\n                const textboxMetadata = metadata as TextboxStringDecoratorConfigInternal;\n                if (!this.isTextboxValid(entityTextbox, textboxMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.NUMBER_DROPDOWN:\n                return true;\n            case DecoratorTypes.NUMBER:\n                const entityNumber = entity[key] as unknown as number;\n                const numberMetadata = metadata as DefaultNumberDecoratorConfigInternal;\n                if (!this.isNumberValid(entityNumber, numberMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.OBJECT:\n                const entityObject = entity[key] as unknown as EntityType;\n                for (const parameterKey in entityObject) {\n                    if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {\n                        return false;\n                    }\n                }\n                break;\n            case DecoratorTypes.ARRAY_STRING_CHIPS:\n            case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:\n            case DecoratorTypes.ARRAY:\n                const entityArray = entity[key] as unknown as [];\n                const arrayMetadata = metadata as EntityArrayDecoratorConfigInternal<EntityType>;\n                if (arrayMetadata.required && !entityArray.length) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE:\n                const entityDate: Date = new Date(entity[key] as unknown as Date);\n                const dateMetadata = metadata as DefaultDateDecoratorConfigInternal;\n                if (!this.isDateValid(entityDate, dateMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_RANGE:\n                const entityDateRange: DateRange = cloneDeep(entity[key] as unknown as DateRange);\n                const dateRangeMetadata = metadata as DateRangeDateDecoratorConfigInternal;\n                if (!this.isDateRangeValid(entityDateRange, dateRangeMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_TIME:\n                const entityDateTime: Date = new Date(entity[key] as unknown as Date);\n                const dateTimeMetadata = metadata as DateTimeDateDecoratorConfigInternal;\n                if (!this.isDateTimeValid(entityDateTime, dateTimeMetadata)) {\n                    return false;\n                }\n                break;\n            default:\n                throw new Error(`Could not validate the input because the DecoratorType ${type} is not known`);\n        }\n        return true;\n    }\n\n    private static isStringValid(value: string, metadata: DefaultStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        if (metadata.regex && !value.match(metadata.regex)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isTextboxValid(value: string, metadata: TextboxStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isNumberValid(value: number, metadata: DefaultNumberDecoratorConfigInternal): boolean {\n        if (metadata.max && value > metadata.max) {\n            return false;\n        }\n        if (metadata.min && value < metadata.min) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateValid(value: Date, metadata: DefaultDateDecoratorConfigInternal): boolean {\n        if (metadata.min && value.getTime() < metadata.min(value).getTime()) {\n            return false;\n        }\n        if (metadata.max && value.getTime() > metadata.max(value).getTime()) {\n            return false;\n        }\n        if (metadata.filter && !metadata.filter(value)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateRangeValid(value: DateRange, metadata: DateRangeDateDecoratorConfigInternal): boolean {\n        if (metadata.required && (!value.start || !value.end)) {\n            return false;\n        }\n        value.start = new Date(value.start);\n        value.end = new Date(value.end);\n        if (metadata.minStart && value.start.getTime() < metadata.minStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.maxStart && value.start.getTime() > metadata.maxStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.minEnd && value.end.getTime() < metadata.minEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.maxEnd && value.end.getTime() > metadata.maxEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.filter) {\n            if (!metadata.filter(value.start)) {\n                return false;\n            }\n            if (!metadata.filter(value.end)) {\n                return false;\n            }\n            if (value.values) {\n                for (const date of value.values) {\n                    if (!metadata.filter(date)) {\n                        return false;\n                    }\n                }\n            }\n        }\n        return true;\n    }\n\n    private static isDateTimeValid(value: Date, metadata: DateTimeDateDecoratorConfigInternal): boolean {\n        if (metadata.minDate && value.getTime() < metadata.minDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.maxDate && value.getTime() > metadata.maxDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.filterDate && !metadata.filterDate(value)) {\n            return false;\n        }\n        const time: Time = {\n            hours: value.getHours(),\n            minutes: value.getMinutes()\n        }\n        if (metadata.minTime) {\n            const minTime: Time = metadata.minTime(value);\n            if (\n                !(\n                    time.hours > minTime.hours\n                    || (\n                        time.hours === minTime.hours\n                        && time.minutes >= minTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.maxTime) {\n            const maxTime: Time = metadata.maxTime(value);\n            if (\n                !(\n                    time.hours < maxTime.hours\n                    || (\n                        time.hours === maxTime.hours\n                        && time.minutes <= maxTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.filterTime) {\n            if (!metadata.filterTime(time)) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    /**\n     * Checks if an entity is \"dirty\" (if its values have changed).\n     *\n     * @param entity - The entity after all changes.\n     * @param entityPriorChanges - The entity before the changes.\n     * @returns Whether or not the entity is dirty.\n     */\n    static dirty<EntityType extends object>(entity: EntityType, entityPriorChanges: EntityType): boolean {\n        if (!entityPriorChanges) {\n            return false;\n        }\n        else {\n            const differences = this.differencesForDirty(entity, entityPriorChanges);\n            return differences.length ? true : false;\n        }\n    }\n\n    private static differencesForDirty<EntityType extends object>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Difference<EntityType>[] {\n        const res: Difference<EntityType>[] = [];\n        for (const key in entity) {\n            if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {\n                res.push({\n                    key: key,\n                    before: entityPriorChanges[key],\n                    after: entity[key]\n                });\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Compares two Entities and returns their difference in an object.\n     *\n     * @param entity - The first entity to compare.\n     * @param entityPriorChanges - The second entity to compare.\n     * @returns The difference between the two Entities in form of a Partial.\n     */\n    static difference<EntityType extends object>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Partial<EntityType> {\n        const res: Partial<EntityType> = {};\n        for (const key in entity) {\n            if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {\n                res[key] = entity[key];\n            }\n        }\n        return res;\n    }\n\n    private static isEqual(value: unknown, valuePriorChanges: unknown, metadata: PropertyDecoratorConfigInternal): boolean {\n        if (this.isDateRange(value) && this.isDateRange(valuePriorChanges)) {\n            const dateRange = cloneDeep(value);\n            dateRange.start = new Date(value.start);\n            dateRange.end = new Date(value.end);\n            dateRange.values = DateUtilities.getDatesBetween(\n                dateRange.start,\n                dateRange.end,\n                metadata as DateRangeDateDecoratorConfigInternal\n            );\n\n            const dateRangePriorChanges = cloneDeep(valuePriorChanges);\n            dateRangePriorChanges.start = new Date(valuePriorChanges.start);\n            dateRangePriorChanges.end = new Date(valuePriorChanges.end);\n            dateRangePriorChanges.values = DateUtilities.getDatesBetween(\n                dateRangePriorChanges.start,\n                dateRangePriorChanges.end,\n                metadata as DateRangeDateDecoratorConfigInternal\n            );\n            return isEqual(dateRange, dateRangePriorChanges);\n        }\n        if ((metadata as DefaultDateDecoratorConfigInternal).displayStyle === 'date') {\n            const date = new Date(DateUtilities.asDate(value));\n            const datePriorChanges = new Date(DateUtilities.asDate(valuePriorChanges));\n            date.setHours(0, 0, 0, 0);\n            datePriorChanges.setHours(0, 0, 0, 0);\n            return isEqual(date, datePriorChanges);\n        }\n        return isEqual(value, valuePriorChanges);\n    }\n\n    private static isDateRange(value: unknown): value is DateRange {\n        const dateRange = value as DateRange;\n        if (dateRange.start && dateRange.end) {\n            try {\n                new Date(dateRange.start);\n                new Date(dateRange.end)\n                return true;\n            }\n            catch (error) {};\n        }\n        return false;\n    }\n\n    /**\n     * Compare function for sorting entity keys by their order value.\n     *\n     * @param a - First key of entity.\n     * @param b - Second key of entity.\n     * @param entity - Current entity (used to get metadata of entity keys).\n     * @returns 0 if both values have the same order, a negative value if 'a' comes before 'b', a positive value if 'a' comes behind 'b'.\n     */\n    static compareOrder<EntityType extends object>(a: keyof EntityType, b: keyof EntityType, entity: EntityType): number {\n        const metadataA = EntityUtilities.getPropertyMetadata(entity, a);\n        const metadataB = EntityUtilities.getPropertyMetadata(entity, b);\n\n        if (metadataA.position.order === -1) {\n            if (metadataB.position.order === -1) {\n                return 0;\n            }\n            return 1;\n        }\n        else if (metadataB.position.order === -1) {\n            return -1;\n        }\n\n        return ((metadataA.position.order ) - (metadataB.position.order ));\n    }\n\n    /**\n     * Gets the bootstrap column values for \"lg\", \"md\", \"sm\".\n     *\n     * @param entity - Entity to get the bootstrap column values of the key.\n     * @param key - Key of the property to get bootstrap column values from.\n     * @param type - Defines for which screen size the column values should be returned.\n     * @returns Bootstrap column value.\n     */\n    static getWidth<EntityType extends object>(entity: EntityType, key: keyof EntityType, type: 'lg' | 'md' | 'sm'): number {\n        const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n        switch (type) {\n            case 'lg':\n                return metadata.defaultWidths[0];\n            case 'md':\n                return metadata.defaultWidths[1];\n            case 'sm':\n                return metadata.defaultWidths[2];\n        }\n    }\n\n    /**\n     * Resets all changes on an entity.\n     *\n     * @param entity - The entity to reset.\n     * @param entityPriorChanges - The entity before any changes.\n     */\n    static resetChangesOnEntity<EntityType extends object>(entity: EntityType, entityPriorChanges: EntityType): void {\n        for (const key in entityPriorChanges) {\n            Reflect.set(entity, key, Reflect.get(entityPriorChanges, key));\n        }\n    }\n\n    /**\n     * Gets the rows that are used to display the given entity.\n     *\n     * @param entity - The entity to get the rows from.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns The sorted Rows containing the row number and the keys to display in that row.\n     */\n    static getEntityRows<EntityType extends object>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): EntityRow<EntityType>[] {\n        const res: EntityRow<EntityType>[] = [];\n\n        const keys: (keyof EntityType)[] = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit);\n        const numberOfRows: number = this.getNumberOfRows<EntityType>(keys, entity);\n        for (let i = 1; i <= numberOfRows; i++) {\n            const row: EntityRow<EntityType> = {\n                row: i,\n                keys: this.getKeysForRow<EntityType>(keys, entity, i)\n            };\n            res.push(row);\n        }\n        const lastRow: EntityRow<EntityType> = {\n            row: numberOfRows + 1,\n            keys: this.getKeysForRow<EntityType>(keys, entity, -1)\n        };\n        res.push(lastRow);\n        return res;\n    }\n\n    private static getKeysForRow<EntityType extends object>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        i: number\n    ): (keyof EntityType)[] {\n        return keys\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === i)\n            .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));\n    }\n\n    private static getNumberOfRows<EntityType extends object>(keys: (keyof EntityType)[], entity: EntityType): number {\n        return keys\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    /**\n     * Gets the keys of the provided entity correctly typed.\n     *\n     * @param entity - The entity to get the keys of.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns An array of keys of the entity.\n     */\n    static keysOf<EntityType extends object>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): (keyof EntityType)[] {\n        let keys: (keyof EntityType)[] = Reflect.ownKeys(entity) as (keyof EntityType)[];\n        if (hideOmitForCreate) {\n            const omitForCreateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForCreate(entity);\n            keys = keys.filter(k => !omitForCreateKeys.includes(k));\n        }\n        if (hideOmitForEdit) {\n            const omitForUpdateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForUpdate(entity);\n            keys = keys.filter(k => !omitForUpdateKeys.includes(k));\n        }\n        return keys;\n    }\n}\n\n/**\n * A row that contains all the information about how to display an entity.\n */\nexport interface EntityRow<EntityType extends object> {\n    /**\n     * The row in which this should be displayed.\n     */\n    row: number,\n    /**\n     * The keys of the values that should be displayed in that row.\n     */\n    keys: (keyof EntityType)[]\n}"]}
708
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.utilities.js","sourceRoot":"","sources":["../../../../projects/ngx-material-entity/src/classes/entity.utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { DecoratorType, DecoratorTypes } from '../decorators/base/decorator-types.enum';\nimport { PropertyDecoratorConfigInternal } from '../decorators/base/property-decorator-internal.data';\nimport { DateRangeArrayDecoratorConfigInternal, EntityArrayDecoratorConfigInternal } from '../decorators/array/array-decorator-internal.data';\nimport { DefaultStringDecoratorConfigInternal, PasswordStringDecoratorConfigInternal, TextboxStringDecoratorConfigInternal } from '../decorators/string/string-decorator-internal.data';\nimport { DefaultNumberDecoratorConfigInternal } from '../decorators/number/number-decorator-internal.data';\nimport { DateRangeDateDecoratorConfigInternal, DateTimeDateDecoratorConfigInternal, DefaultDateDecoratorConfigInternal } from '../decorators/date/date-decorator-internal.data';\nimport { DateRange } from '../decorators/date/date-decorator.data';\nimport { Time } from '@angular/common';\nimport { DateUtilities } from './date.utilities';\nimport { ReflectUtilities } from '../capsulation/reflect.utilities';\nimport { LodashUtilities } from '../capsulation/lodash.utilities';\nimport { ToggleBooleanDecoratorConfigInternal } from '../decorators/boolean/boolean-decorator-internal.data';\nimport { DateFilterFn } from '@angular/material/datepicker';\nimport { FileData } from '../decorators/file/file-decorator.data';\nimport { DefaultFileDecoratorConfigInternal } from '../decorators/file/file-decorator-internal.data';\nimport { FileUtilities } from './file.utilities';\nimport { BaseEntityType } from './entity.model';\nimport { CustomDecoratorConfigInternal } from '../decorators/custom/custom-decorator-internal.data';\n\n/**\n * Shows information about differences between two entities.\n */\ninterface Difference<EntityType extends BaseEntityType<EntityType>> {\n    /**\n     * The key where the two entities have different values.\n     */\n    key: keyof EntityType,\n    /**\n     * The value before any changes.\n     */\n    before: unknown,\n    /**\n     * The current value after changes.\n     */\n    after: unknown\n}\n\n/**\n * Contains HelperMethods around handling Entities and their property-metadata.\n */\nexport abstract class EntityUtilities {\n\n    /**\n     * Gets the properties to omit when updating the entity.\n     *\n     * @param entity - The entity to get the properties which should be left out for updating from.\n     * @returns The properties which should be left out for updating an Entity.\n     */\n    static getOmitForUpdate<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForUpdate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the properties to omit when creating new entities.\n     *\n     * @param entity - The entity to get the properties which should be left out for creating from.\n     * @returns The properties which should be left out for creating a new Entity.\n     */\n    static getOmitForCreate<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForCreate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets all properties on the given entity which are files.\n     *\n     * @param entity - The entity to check for file properties.\n     * @param omit - Whether to leave out values that are omitted for create or delete.\n     * @returns The keys of all file properties on the given entity.\n     */\n    static getFileProperties<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        omit?: 'create' | 'update'\n    ): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const type = EntityUtilities.getPropertyType(entity, key);\n            if (type === DecoratorTypes.FILE_DEFAULT || type === DecoratorTypes.FILE_IMAGE) {\n                const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n                if (!(metadata.omitForCreate && omit === 'create') && !(metadata.omitForUpdate && omit === 'update')) {\n                    res.push(key);\n                }\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the metadata included in an property.\n     *\n     * @param entity - The entity with the property to get the metadata from.\n     * @param propertyKey - The property on the given Entity to get the metadata from.\n     * @param type - For secure Typing, defines the returned PropertyConfig.\n     * @returns The metadata of the property.\n     * @throws When no metadata can be found for the given property.\n     */\n    static getPropertyMetadata<\n        EntityType extends BaseEntityType<EntityType>,\n        T extends DecoratorTypes,\n        CustomMetadataType extends Record<string, unknown>\n    >(\n        entity: EntityType,\n        propertyKey: keyof EntityType,\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n        type?: T\n    ): DecoratorType<T, CustomMetadataType> {\n        const metadata = ReflectUtilities.getMetadata('metadata', entity, propertyKey);\n        if (metadata == null) {\n            throw new Error(\n                `Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n        return metadata as DecoratorType<T, CustomMetadataType>;\n    }\n\n    /**\n     * Gets the type of the property-metadata.\n     *\n     * @param entity - The entity with the property to get the type from.\n     * @param propertyKey - The property on the given Entity to get the type from.\n     * @returns The type of the metadata.\n     * @throws Will throw an error if no metadata can be found for the given property.\n     */\n    static getPropertyType<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType, propertyKey: keyof EntityType\n    ): DecoratorTypes {\n        try {\n            const propertyType = ReflectUtilities.getMetadata('type', entity, propertyKey);\n            if (propertyType == null) {\n                throw new Error(\n                    `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n                );\n            }\n            return propertyType as DecoratorTypes;\n        }\n        catch (error) {\n            throw new Error(\n                `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n    }\n\n    /**\n     * Sets all property values based on a given entity data-object.\n     *\n     * @param target - The target object that needs to be constructed (if called inside an Entity constructor its usually this).\n     * @param entity - The data object to get the property values from.\n     * @alias new\n     * @alias build\n     * @alias construct\n     */\n    static new<EntityType extends BaseEntityType<EntityType>>(target: EntityType, entity?: EntityType): void {\n        for (const key in target) {\n            const type = EntityUtilities.getPropertyType(target, key);\n            let value = entity ? ReflectUtilities.get(entity, key) : undefined;\n            switch (type) {\n                case DecoratorTypes.OBJECT:\n                    const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);\n                    value = new objectMetadata.EntityClass(value as object | undefined);\n                    break;\n                case DecoratorTypes.ARRAY:\n                    const inputArray: EntityType[] | undefined = value as EntityType[] | undefined;\n                    const resArray: EntityType[] = [];\n                    if (inputArray) {\n                        const arrayMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);\n                        for (const item of inputArray) {\n                            const itemWithMetadata: EntityType = new arrayMetadata.EntityClass(item) as EntityType;\n                            resArray.push(itemWithMetadata);\n                        }\n                    }\n                    value = resArray;\n                    break;\n                default:\n                    break;\n            }\n            ReflectUtilities.set(target, key, value);\n        }\n    }\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static construct = EntityUtilities.new;\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static build = EntityUtilities.new;\n\n    /**\n     * Checks if the values on an entity are valid.\n     * Also checks all the validators given by the metadata (\"required\", \"maxLength\" etc.).\n     *\n     * @param entity - The entity to validate.\n     * @param omit - Whether to check for creating or editing validity.\n     * @returns Whether or not the entity is valid.\n     */\n    static isEntityValid<EntityType extends BaseEntityType<EntityType>>(entity: EntityType, omit: 'create' | 'update'): boolean {\n        for (const key in entity) {\n            if (!EntityUtilities.isPropertyValid(entity, key, omit)) {\n                return false;\n            }\n        }\n        return true;\n    }\n    /**\n     * Checks if a single property value is valid.\n     *\n     * @param entity - The entity where the property is from.\n     * @param key - The name of the property.\n     * @param omit - Whether to check if the given entity is valid for creation or updating.\n     * @returns Whether or not the property value is valid.\n     * @throws Throws when it extracts an unknown metadata type.\n     */\n    private static isPropertyValid<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        key: keyof EntityType,\n        omit: 'create' | 'update'\n    ): boolean {\n        const type = EntityUtilities.getPropertyType(entity, key);\n        const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key, type);\n\n        if (metadata.omitForCreate && omit === 'create') {\n            return true;\n        }\n        if (metadata.omitForUpdate && omit === 'update') {\n            return true;\n        }\n        if (metadata.required && entity[key] == null) {\n            return false;\n        }\n        switch (type) {\n            case DecoratorTypes.BOOLEAN_DROPDOWN:\n                break;\n            case DecoratorTypes.BOOLEAN_CHECKBOX:\n            case DecoratorTypes.BOOLEAN_TOGGLE:\n                const entityBoolean = entity[key] as boolean;\n                const booleanMetadata = metadata as ToggleBooleanDecoratorConfigInternal;\n                if (!EntityUtilities.isBooleanValid(entityBoolean, booleanMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_DROPDOWN:\n                break;\n            case DecoratorTypes.STRING:\n            case DecoratorTypes.STRING_AUTOCOMPLETE:\n                const entityString = entity[key] as string;\n                const stringMetadata = metadata as DefaultStringDecoratorConfigInternal;\n                if (!EntityUtilities.isStringValid(entityString, stringMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_TEXTBOX:\n                const entityTextbox = entity[key] as string;\n                const textboxMetadata = metadata as TextboxStringDecoratorConfigInternal;\n                if (!EntityUtilities.isTextboxValid(entityTextbox, textboxMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_PASSWORD:\n                const entityPassword = entity[key] as string;\n                const passwordMetadata = metadata as PasswordStringDecoratorConfigInternal;\n                const confirmPassword = ReflectUtilities.getMetadata('confirmPassword', entity, key) as string;\n                if (!EntityUtilities.isPasswordValid(entityPassword, passwordMetadata, confirmPassword)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.NUMBER_DROPDOWN:\n                return true;\n            case DecoratorTypes.NUMBER:\n            case DecoratorTypes.NUMBER_SLIDER:\n                const entityNumber = entity[key] as number;\n                const numberMetadata = metadata as DefaultNumberDecoratorConfigInternal;\n                if (!EntityUtilities.isNumberValid(entityNumber, numberMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.OBJECT:\n                const entityObject = entity[key] as EntityType;\n                for (const parameterKey in entityObject) {\n                    if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {\n                        return false;\n                    }\n                }\n                break;\n            case DecoratorTypes.ARRAY_STRING_CHIPS:\n            case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:\n            case DecoratorTypes.ARRAY_DATE:\n            case DecoratorTypes.ARRAY_DATE_TIME:\n            case DecoratorTypes.ARRAY_DATE_RANGE:\n            case DecoratorTypes.ARRAY:\n                const entityArray = entity[key] as unknown[];\n                const arrayMetadata = metadata as EntityArrayDecoratorConfigInternal<EntityType>;\n                if (arrayMetadata.required && !entityArray.length) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE:\n                const entityDate: Date = new Date(entity[key] as Date);\n                const dateMetadata = metadata as DefaultDateDecoratorConfigInternal;\n                if (!EntityUtilities.isDateValid(entityDate, dateMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_RANGE:\n                const entityDateRange: DateRange = LodashUtilities.cloneDeep(entity[key] as DateRange);\n                const dateRangeMetadata = metadata as DateRangeDateDecoratorConfigInternal;\n                if (!EntityUtilities.isDateRangeValid(entityDateRange, dateRangeMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_TIME:\n                const entityDateTime: Date = new Date(entity[key] as Date);\n                const dateTimeMetadata = metadata as DateTimeDateDecoratorConfigInternal;\n                if (!EntityUtilities.isDateTimeValid(entityDateTime, dateTimeMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.FILE_DEFAULT:\n            case DecoratorTypes.FILE_IMAGE:\n                const entityFile: FileData | FileData[] = entity[key] as FileData | FileData[];\n                const entityFileMetadata = metadata as DefaultFileDecoratorConfigInternal;\n                if (!EntityUtilities.isFileDataValid(entityFile, entityFileMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.CUSTOM:\n                // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                const customMetadata = metadata as CustomDecoratorConfigInternal<EntityType, any, any, any>;\n                if (!customMetadata.isValid(entity[key], omit)) {\n                    return false;\n                }\n                break;\n            default:\n                throw new Error(`Could not validate the input because the DecoratorType ${type} is not known`);\n        }\n        return true;\n    }\n\n    private static isBooleanValid(value: boolean, metadata: ToggleBooleanDecoratorConfigInternal): boolean {\n        if (metadata.required && !value) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isStringValid(value: string, metadata: DefaultStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        if (metadata.regex && !value.match(metadata.regex)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isTextboxValid(value: string, metadata: TextboxStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isPasswordValid(value: string, metadata: PasswordStringDecoratorConfigInternal, confirmPassword: string): boolean {\n        if (value !== confirmPassword) {\n            return false;\n        }\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        if (metadata.regex && !value.match(metadata.regex)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isNumberValid(value: number, metadata: DefaultNumberDecoratorConfigInternal): boolean {\n        if (metadata.max && value > metadata.max) {\n            return false;\n        }\n        if (metadata.min && value < metadata.min) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateValid(value: Date, metadata: DefaultDateDecoratorConfigInternal): boolean {\n        if (metadata.min && value.getTime() < metadata.min(value).getTime()) {\n            return false;\n        }\n        if (metadata.max && value.getTime() > metadata.max(value).getTime()) {\n            return false;\n        }\n        if (metadata.filter && !metadata.filter(value)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateRangeValid(value: DateRange, metadata: DateRangeDateDecoratorConfigInternal): boolean {\n        if (metadata.required) {\n            if (!(value.start as Date | undefined)) {\n                return false;\n            }\n            if (!(value.end as Date | undefined)) {\n                return false;\n            }\n        }\n        value.start = new Date(value.start);\n        value.end = new Date(value.end);\n        if (metadata.minStart && value.start.getTime() < metadata.minStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.maxStart && value.start.getTime() > metadata.maxStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.minEnd && value.end.getTime() < metadata.minEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.maxEnd && value.end.getTime() > metadata.maxEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.filter) {\n            if (!metadata.filter(value.start)) {\n                return false;\n            }\n            if (!metadata.filter(value.end)) {\n                return false;\n            }\n            if (value.values) {\n                for (const date of value.values) {\n                    if (!metadata.filter(date)) {\n                        return false;\n                    }\n                }\n            }\n        }\n        return true;\n    }\n\n    private static isDateTimeValid(value: Date, metadata: DateTimeDateDecoratorConfigInternal): boolean {\n        if (metadata.minDate && value.getTime() < metadata.minDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.maxDate && value.getTime() > metadata.maxDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.filterDate && !metadata.filterDate(value)) {\n            return false;\n        }\n        const time: Time = {\n            hours: value.getHours(),\n            minutes: value.getMinutes()\n        };\n        if (metadata.minTime) {\n            const minTime: Time = metadata.minTime(value);\n            if (\n                !(\n                    time.hours > minTime.hours\n                    || (\n                        time.hours === minTime.hours\n                        && time.minutes >= minTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.maxTime) {\n            const maxTime: Time = metadata.maxTime(value);\n            if (\n                !(\n                    time.hours < maxTime.hours\n                    || (\n                        time.hours === maxTime.hours\n                        && time.minutes <= maxTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.filterTime) {\n            if (!metadata.filterTime(time)) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    private static isFileDataValid(value: FileData | FileData[], metadata: DefaultFileDecoratorConfigInternal): boolean {\n        const files = metadata.multiple ? value as FileData[] : [value as FileData];\n        let fileSizeTotal: number = 0;\n        // eslint-disable-next-line @typescript-eslint/prefer-for-of\n        for (let i = 0; i < files.length; i++) {\n            if (!files[i].name || !files[i].file && !files[i].url) {\n                return false;\n            }\n            if (!FileUtilities.isMimeTypeValid(files[i].type, metadata.allowedMimeTypes)) {\n                return false;\n            }\n            if (FileUtilities.transformToMegaBytes(files[i].size, 'B') > metadata.maxSize) {\n                return false;\n            }\n            fileSizeTotal += files[i].size;\n            if (FileUtilities.transformToMegaBytes(fileSizeTotal, 'B') > metadata.maxSizeTotal) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    /**\n     * Checks if an entity is \"dirty\" (if its values have changed).\n     *\n     * @param entity - The entity after all changes.\n     * @param entityPriorChanges - The entity before the changes.\n     * @returns Whether or not the entity is dirty.\n     */\n    static async isDirty<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Promise<boolean> {\n        if (!(entityPriorChanges as EntityType | undefined)) {\n            return false;\n        }\n        else {\n            const differences = await EntityUtilities.differencesForDirty(entity, entityPriorChanges);\n            return differences.length ? true : false;\n        }\n    }\n\n    private static async differencesForDirty<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Promise<Difference<EntityType>[]> {\n        const res: Difference<EntityType>[] = [];\n        for (const key in entity) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            const type = EntityUtilities.getPropertyType(entity, key);\n            if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {\n                res.push({\n                    key: key,\n                    before: entityPriorChanges[key],\n                    after: entity[key]\n                });\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Compares two Entities and returns their difference in an object.\n     *\n     * @param entity - The first entity to compare.\n     * @param entityPriorChanges - The second entity to compare.\n     * @returns The difference between the two Entities in form of a Partial.\n     */\n    static async difference<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Promise<Partial<EntityType>> {\n        const res: Partial<EntityType> = {};\n        for (const key in entity) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            const type = EntityUtilities.getPropertyType(entity, key);\n            if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {\n                res[key] = entity[key];\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Checks if two given values are equal.\n     * It uses the isEqual method from LodashUtilities and extends it with functionality regarding Dates.\n     *\n     * @param value - The updated value.\n     * @param valuePriorChanges - The value before any changes.\n     * @param metadata - The metadata of the property.\n     * @param type - The type of the property.\n     * @returns Whether or not the given values are equal.\n     */\n    static async isEqual(\n        value: unknown,\n        valuePriorChanges: unknown,\n        metadata: PropertyDecoratorConfigInternal,\n        type: DecoratorTypes\n    ): Promise<boolean> {\n        switch (type) {\n            case DecoratorTypes.DATE_RANGE:\n                return EntityUtilities.isEqualDateRange(\n                    value,\n                    valuePriorChanges,\n                    (metadata as DateRangeDateDecoratorConfigInternal).filter\n                );\n            case DecoratorTypes.DATE:\n                return EntityUtilities.isEqualDate(value, valuePriorChanges);\n            case DecoratorTypes.DATE_TIME:\n                return EntityUtilities.isEqualDateTime(value, valuePriorChanges);\n            case DecoratorTypes.ARRAY_DATE:\n            case DecoratorTypes.ARRAY_DATE_TIME:\n                return EntityUtilities.isEqualArrayDate(value, valuePriorChanges);\n            case DecoratorTypes.ARRAY_DATE_RANGE:\n                return EntityUtilities.isEqualArrayDateRange(\n                    value,\n                    valuePriorChanges,\n                    (metadata as DateRangeArrayDecoratorConfigInternal).filter\n                );\n            case DecoratorTypes.FILE_IMAGE:\n            case DecoratorTypes.FILE_DEFAULT:\n                return EntityUtilities.isEqualFile(value, valuePriorChanges, (metadata as DefaultFileDecoratorConfigInternal).multiple);\n            case DecoratorTypes.CUSTOM:\n                // eslint-disable-next-line max-len, @typescript-eslint/no-explicit-any\n                return EntityUtilities.isEqualCustom(value, valuePriorChanges, metadata as CustomDecoratorConfigInternal<any, any, any, any>);\n            default:\n                return LodashUtilities.isEqual(value, valuePriorChanges);\n        }\n    }\n\n    private static isEqualArrayDate(value: unknown, valuePriorChanges: unknown): boolean {\n        const newValue = (value as Date[]).map(v => new Date(v)).sort();\n        const newValuePriorChanges = (valuePriorChanges as Date[]).map(v => new Date(v)).sort();\n        return LodashUtilities.isEqual(newValue, newValuePriorChanges);\n    }\n\n    private static isEqualArrayDateRange(value: unknown, valuePriorChanges: unknown, filter?: DateFilterFn<Date>): boolean {\n        const dateRanges = (value as DateRange[]).sort();\n        const dateRangesPriorChanges = (valuePriorChanges as DateRange[]).sort();\n        if (dateRanges.length !== dateRangesPriorChanges.length) {\n            return false;\n        }\n        for (let i = 0; i < dateRanges.length; i++) {\n            if (!EntityUtilities.isEqualDateRange(dateRanges[i], dateRangesPriorChanges[i], filter)) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    private static isEqualDateTime(value: unknown, valuePriorChanges: unknown): boolean {\n        const date = new Date(value as Date);\n        const datePriorChanges = new Date(valuePriorChanges as Date);\n        return LodashUtilities.isEqual(date, datePriorChanges);\n    }\n\n    private static isEqualDate(value: unknown, valuePriorChanges: unknown): boolean {\n        const date = new Date(value as Date);\n        const datePriorChanges = new Date(valuePriorChanges as Date);\n        date.setHours(0, 0, 0, 0);\n        datePriorChanges.setHours(0, 0, 0, 0);\n        return LodashUtilities.isEqual(date, datePriorChanges);\n    }\n\n    private static isEqualDateRange(value: unknown, valuePriorChanges: unknown, filter?: DateFilterFn<Date>): boolean {\n        const dateRange = LodashUtilities.cloneDeep(value) as DateRange;\n        dateRange.start = new Date((value as DateRange).start);\n        dateRange.end = new Date((value as DateRange).end);\n        dateRange.values = DateUtilities.getDatesBetween(\n            dateRange.start,\n            dateRange.end,\n            filter\n        );\n        const dateRangePriorChanges = LodashUtilities.cloneDeep(valuePriorChanges) as DateRange;\n        dateRangePriorChanges.start = new Date((valuePriorChanges as DateRange).start);\n        dateRangePriorChanges.end = new Date((valuePriorChanges as DateRange).end);\n        dateRangePriorChanges.values = DateUtilities.getDatesBetween(\n            dateRangePriorChanges.start,\n            dateRangePriorChanges.end,\n            filter\n        );\n        return LodashUtilities.isEqual(dateRange, dateRangePriorChanges);\n    }\n\n    // TODO: Find a way to use blobs with jest\n    /* istanbul ignore next */\n    private static async isEqualFile(value: unknown, valuePriorChanges: unknown, multiple: boolean): Promise<boolean> {\n        const files = multiple ? (value as FileData[]).sort() : [value as FileData].sort();\n        const filesPriorChanges = multiple ? (valuePriorChanges as FileData[]).sort() : [valuePriorChanges as FileData].sort();\n        if (files.length !== filesPriorChanges.length) {\n            return false;\n        }\n        for (let i = 0; i < files.length; i++) {\n            // checks this before actually getting any files due to performance reasons.\n            if (\n                !LodashUtilities.isEqual(files[i]?.name, filesPriorChanges[i]?.name)\n                || !LodashUtilities.isEqual(files[i]?.url, filesPriorChanges[i]?.url)\n            ) {\n                return false;\n            }\n            files[i] = filesPriorChanges[i].file && !files[i].file ? await FileUtilities.getFileData(files[i]) : files[i];\n            // eslint-disable-next-line max-len\n            filesPriorChanges[i] = files[i].file && !filesPriorChanges[i].file ? await FileUtilities.getFileData(filesPriorChanges[i]) : filesPriorChanges[i];\n            if (!LodashUtilities.isEqual(await files[i].file?.text(), await filesPriorChanges[i].file?.text())) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    private static isEqualCustom(\n        value: unknown,\n        valuePriorChanges: unknown,\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        metadata: CustomDecoratorConfigInternal<any, any, any, any>\n    ): boolean {\n        if (!metadata.isEqual(value, valuePriorChanges, metadata)) {\n            return false;\n        }\n        return true;\n    }\n\n    /**\n     * Compare function for sorting entity keys by their order value.\n     *\n     * @param a - First key of entity.\n     * @param b - Second key of entity.\n     * @param entity - Current entity (used to get metadata of entity keys).\n     * @returns 0 if both values have the same order, a negative value if 'a' comes before 'b', a positive value if 'a' comes behind 'b'.\n     */\n    static compareOrder<EntityType extends BaseEntityType<EntityType>>(\n        a: keyof EntityType,\n        b: keyof EntityType,\n        entity: EntityType\n    ): number {\n        const metadataA = EntityUtilities.getPropertyMetadata(entity, a);\n        const metadataB = EntityUtilities.getPropertyMetadata(entity, b);\n\n        if (metadataA.position.order === -1) {\n            if (metadataB.position.order === -1) {\n                return 0;\n            }\n            return 1;\n        }\n        else if (metadataB.position.order === -1) {\n            return -1;\n        }\n        return metadataA.position.order - metadataB.position.order;\n    }\n\n    /**\n     * Gets the bootstrap column values for \"lg\", \"md\", \"sm\".\n     *\n     * @param entity - Entity to get the bootstrap column values of the key.\n     * @param key - Key of the property to get bootstrap column values from.\n     * @param type - Defines for which screen size the column values should be returned.\n     * @returns Bootstrap column value.\n     */\n    static getWidth<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        key: keyof EntityType, type: 'lg' | 'md' | 'sm'\n    ): number {\n        const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n        switch (type) {\n            case 'lg':\n                return metadata.defaultWidths[0];\n            case 'md':\n                return metadata.defaultWidths[1];\n            case 'sm':\n                return metadata.defaultWidths[2];\n        }\n    }\n\n    /**\n     * Resets all changes on an entity.\n     *\n     * @param entity - The entity to reset.\n     * @param entityPriorChanges - The entity before any changes.\n     */\n    static resetChangesOnEntity<EntityType extends BaseEntityType<EntityType>>(entity: EntityType, entityPriorChanges: EntityType): void {\n        for (const key in entityPriorChanges) {\n            ReflectUtilities.set(entity, key, ReflectUtilities.get(entityPriorChanges, key));\n        }\n    }\n\n    /**\n     * Gets the rows that are used to display the given entity.\n     *\n     * @param entity - The entity to get the rows from.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns The sorted Rows containing the row number and the keys to display in that row.\n     */\n    static getEntityRows<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): EntityRow<EntityType>[] {\n        const res: EntityRow<EntityType>[] = [];\n\n        const keys: (keyof EntityType)[] = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit);\n        const numberOfRows: number = EntityUtilities.getNumberOfRows<EntityType>(keys, entity);\n        for (let i = 1; i <= numberOfRows; i++) {\n            const row: EntityRow<EntityType> = {\n                row: i,\n                keys: EntityUtilities.getKeysForRow<EntityType>(keys, entity, i)\n            };\n            res.push(row);\n        }\n        const lastRow: EntityRow<EntityType> = {\n            row: numberOfRows + 1,\n            keys: EntityUtilities.getKeysForRow<EntityType>(keys, entity, -1)\n        };\n        res.push(lastRow);\n        return res;\n    }\n\n    private static getKeysForRow<EntityType extends BaseEntityType<EntityType>>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        i: number\n    ): (keyof EntityType)[] {\n        return keys\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === i)\n            .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));\n    }\n\n    private static getNumberOfRows<EntityType extends BaseEntityType<EntityType>>(keys: (keyof EntityType)[], entity: EntityType): number {\n        return keys\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    /**\n     * Gets the keys of the provided entity correctly typed.\n     *\n     * @param entity - The entity to get the keys of.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns An array of keys of the entity.\n     */\n    static keysOf<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): (keyof EntityType)[] {\n        let keys: (keyof EntityType)[] = ReflectUtilities.ownKeys(entity);\n        if (hideOmitForCreate) {\n            const omitForCreateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForCreate(entity);\n            keys = keys.filter(k => !omitForCreateKeys.includes(k));\n        }\n        if (hideOmitForEdit) {\n            const omitForUpdateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForUpdate(entity);\n            keys = keys.filter(k => !omitForUpdateKeys.includes(k));\n        }\n        return keys;\n    }\n}\n\n/**\n * A row that contains all the information about how to display an entity.\n */\nexport interface EntityRow<EntityType extends BaseEntityType<EntityType>> {\n    /**\n     * The row in which this should be displayed.\n     */\n    row: number,\n    /**\n     * The keys of the values that should be displayed in that row.\n     */\n    keys: (keyof EntityType)[]\n}"]}