ngx-material-entity 16.0.2 → 16.0.3

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 (97) hide show
  1. package/components/edit-page/edit-page.component.d.ts +13 -2
  2. package/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.d.ts +1 -0
  3. package/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.d.ts +2 -1
  4. package/components/input/input.component.d.ts +1 -1
  5. package/components/input/input.module.d.ts +4 -2
  6. package/components/input/number/number-slider-input/number-slider-input.component.d.ts +2 -1
  7. package/components/table/create-dialog/create-entity-dialog.component.d.ts +3 -0
  8. package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +4 -0
  9. package/components/tooltip/tooltip.component.d.ts +9 -0
  10. package/decorators/array/array-decorator-internal.data.d.ts +6 -6
  11. package/decorators/array/array-decorator.data.d.ts +7 -7
  12. package/decorators/base/property-decorator-internal.data.d.ts +11 -2
  13. package/decorators/base/property-decorator.data.d.ts +9 -1
  14. package/decorators/boolean/boolean-decorator-internal.data.d.ts +3 -3
  15. package/decorators/boolean/boolean-decorator.data.d.ts +1 -1
  16. package/decorators/custom/custom-decorator-internal.data.d.ts +2 -2
  17. package/decorators/custom/custom-decorator.data.d.ts +4 -4
  18. package/decorators/date/date-decorator-internal.data.d.ts +4 -4
  19. package/decorators/date/date-decorator.data.d.ts +5 -5
  20. package/decorators/file/file-decorator-internal.data.d.ts +3 -3
  21. package/decorators/file/file-decorator.data.d.ts +1 -1
  22. package/decorators/has-many/has-many-decorator-internal.data.d.ts +1 -1
  23. package/decorators/has-many/has-many-decorator.data.d.ts +1 -1
  24. package/decorators/number/number-decorator-internal.data.d.ts +3 -3
  25. package/decorators/number/number-decorator.data.d.ts +1 -1
  26. package/decorators/object/object-decorator-internal.data.d.ts +1 -1
  27. package/decorators/object/object-decorator.data.d.ts +1 -1
  28. package/decorators/references-many/references-many-decorator-internal.data.d.ts +1 -1
  29. package/decorators/references-many/references-many-decorator.data.d.ts +1 -1
  30. package/decorators/references-one/references-one-decorator-internal.data.d.ts +1 -1
  31. package/decorators/references-one/references-one-decorator.data.d.ts +1 -1
  32. package/decorators/string/string-decorator-internal.data.d.ts +5 -6
  33. package/decorators/string/string-decorator.data.d.ts +1 -7
  34. package/directives/number.directive.d.ts +17 -0
  35. package/directives/password-match.directive.d.ts +14 -0
  36. package/directives/tooltip.directive.d.ts +33 -0
  37. package/esm2022/components/edit-page/edit-page.component.mjs +57 -13
  38. package/esm2022/components/input/array/array-date-range-input/array-date-range-input.component.mjs +1 -1
  39. package/esm2022/components/input/base-input.component.mjs +2 -1
  40. package/esm2022/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.mjs +6 -3
  41. package/esm2022/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.mjs +6 -3
  42. package/esm2022/components/input/date/date-range-input/date-range-input.component.mjs +3 -1
  43. package/esm2022/components/input/file/file-image-input/file-image-input.component.mjs +3 -3
  44. package/esm2022/components/input/file/file-input/file-input.component.mjs +1 -1
  45. package/esm2022/components/input/input.component.mjs +7 -6
  46. package/esm2022/components/input/input.module.mjs +10 -4
  47. package/esm2022/components/input/number/number-input/number-input.component.mjs +4 -3
  48. package/esm2022/components/input/number/number-slider-input/number-slider-input.component.mjs +6 -3
  49. package/esm2022/components/input/string/string-password-input/string-password-input.component.mjs +4 -3
  50. package/esm2022/components/table/create-dialog/create-entity-dialog.component.mjs +18 -6
  51. package/esm2022/components/table/edit-dialog/edit-entity-dialog.component.mjs +26 -13
  52. package/esm2022/components/table/table.component.mjs +7 -4
  53. package/esm2022/components/tooltip/tooltip.component.mjs +20 -0
  54. package/esm2022/decorators/array/array-decorator-internal.data.mjs +1 -1
  55. package/esm2022/decorators/array/array-decorator.data.mjs +1 -1
  56. package/esm2022/decorators/base/property-decorator-internal.data.mjs +22 -1
  57. package/esm2022/decorators/base/property-decorator.data.mjs +10 -1
  58. package/esm2022/decorators/boolean/boolean-decorator-internal.data.mjs +1 -1
  59. package/esm2022/decorators/boolean/boolean-decorator.data.mjs +1 -1
  60. package/esm2022/decorators/custom/custom-decorator-internal.data.mjs +2 -2
  61. package/esm2022/decorators/custom/custom-decorator.data.mjs +1 -1
  62. package/esm2022/decorators/date/date-decorator-internal.data.mjs +2 -1
  63. package/esm2022/decorators/date/date-decorator.data.mjs +1 -1
  64. package/esm2022/decorators/file/file-decorator-internal.data.mjs +1 -1
  65. package/esm2022/decorators/file/file-decorator.data.mjs +1 -1
  66. package/esm2022/decorators/has-many/has-many-decorator-internal.data.mjs +1 -1
  67. package/esm2022/decorators/has-many/has-many-decorator.data.mjs +1 -1
  68. package/esm2022/decorators/number/number-decorator-internal.data.mjs +1 -1
  69. package/esm2022/decorators/number/number-decorator.data.mjs +1 -1
  70. package/esm2022/decorators/object/object-decorator-internal.data.mjs +1 -1
  71. package/esm2022/decorators/object/object-decorator.data.mjs +1 -1
  72. package/esm2022/decorators/references-many/references-many-decorator-internal.data.mjs +1 -1
  73. package/esm2022/decorators/references-many/references-many-decorator.data.mjs +1 -1
  74. package/esm2022/decorators/references-one/references-one-decorator-internal.data.mjs +1 -1
  75. package/esm2022/decorators/references-one/references-one-decorator.data.mjs +1 -1
  76. package/esm2022/decorators/string/string-decorator-internal.data.mjs +1 -4
  77. package/esm2022/decorators/string/string-decorator.data.mjs +1 -1
  78. package/esm2022/directives/drag-drop.directive.mjs +62 -0
  79. package/esm2022/directives/number.directive.mjs +38 -0
  80. package/esm2022/directives/password-match.directive.mjs +30 -0
  81. package/esm2022/directives/tooltip.directive.mjs +112 -0
  82. package/esm2022/functions/get-validation-error-message.function.mjs +49 -0
  83. package/esm2022/functions/get-validation-errors-tooltip-content.function.ts.mjs +25 -0
  84. package/esm2022/public-api.mjs +6 -2
  85. package/esm2022/utilities/entity.utilities.mjs +83 -386
  86. package/esm2022/utilities/validation.utilities.mjs +455 -0
  87. package/fesm2022/ngx-material-entity.mjs +1374 -905
  88. package/fesm2022/ngx-material-entity.mjs.map +1 -1
  89. package/functions/get-validation-errors-tooltip-content.function.ts.d.ts +10 -0
  90. package/package.json +1 -1
  91. package/public-api.d.ts +4 -1
  92. package/utilities/entity.utilities.d.ts +33 -30
  93. package/utilities/validation.utilities.d.ts +56 -0
  94. package/esm2022/components/get-validation-error-message.function.mjs +0 -42
  95. package/esm2022/components/input/file/file-input/dragDrop.directive.mjs +0 -62
  96. /package/{components/input/file/file-input/dragDrop.directive.d.ts → directives/drag-drop.directive.d.ts} +0 -0
  97. /package/{components → functions}/get-validation-error-message.function.d.ts +0 -0
@@ -52,7 +52,7 @@ export class EntityUtilities {
52
52
  static getOmitForUpdate(entity) {
53
53
  const res = [];
54
54
  for (const key of ReflectUtilities.ownKeys(entity)) {
55
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
55
+ const metadata = this.getPropertyMetadata(entity, key);
56
56
  if (metadata.omitForUpdate) {
57
57
  res.push(key);
58
58
  }
@@ -68,7 +68,7 @@ export class EntityUtilities {
68
68
  static getOmitForCreate(entity) {
69
69
  const res = [];
70
70
  for (const key of ReflectUtilities.ownKeys(entity)) {
71
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
71
+ const metadata = this.getPropertyMetadata(entity, key);
72
72
  if (metadata.omitForCreate) {
73
73
  res.push(key);
74
74
  }
@@ -82,7 +82,7 @@ export class EntityUtilities {
82
82
  * @returns The reduced entity object.
83
83
  */
84
84
  static getWithoutOmitCreateValues(entity) {
85
- return LodashUtilities.omit(entity, EntityUtilities.getOmitForCreate(entity));
85
+ return LodashUtilities.omit(entity, this.getOmitForCreate(entity));
86
86
  }
87
87
  /**
88
88
  * Returns the given entity without the values that should be omitted for updating.
@@ -95,18 +95,18 @@ export class EntityUtilities {
95
95
  */
96
96
  static async getWithoutOmitUpdateValues(entity, entityPriorChanges, http) {
97
97
  const res = {};
98
- for (const key of EntityUtilities.keysOf(entity, false, true)) {
99
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
100
- const type = EntityUtilities.getPropertyType(entity, key);
101
- if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {
98
+ for (const key of this.keysOf(entity, false, true)) {
99
+ const metadata = this.getPropertyMetadata(entity, key);
100
+ const type = this.getPropertyType(entity, key);
101
+ if (!(await this.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {
102
102
  switch (type) {
103
103
  case DecoratorTypes.OBJECT:
104
104
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
105
- res[key] = LodashUtilities.omit(entity[key], EntityUtilities.getOmitForCreate(entity[key]));
105
+ res[key] = LodashUtilities.omit(entity[key], this.getOmitForCreate(entity[key]));
106
106
  break;
107
107
  case DecoratorTypes.ARRAY:
108
108
  res[key] = entity[key]
109
- .map(value => LodashUtilities.omit(value, EntityUtilities.getOmitForCreate(value)));
109
+ .map(value => LodashUtilities.omit(value, this.getOmitForCreate(value)));
110
110
  break;
111
111
  default:
112
112
  res[key] = entity[key];
@@ -116,6 +116,20 @@ export class EntityUtilities {
116
116
  }
117
117
  return res;
118
118
  }
119
+ /**
120
+ * Sets all default values on the given entity.
121
+ *
122
+ * @param entity - The entity to set the default values on.
123
+ */
124
+ static setDefaultValues(entity) {
125
+ for (const key in entity) {
126
+ const metadata = this.getPropertyMetadata(entity, key);
127
+ if (metadata.default) {
128
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-explicit-any
129
+ entity[key] = metadata.default();
130
+ }
131
+ }
132
+ }
119
133
  /**
120
134
  * Gets all properties on the given entity which are files.
121
135
  *
@@ -126,9 +140,9 @@ export class EntityUtilities {
126
140
  static getFileProperties(entity, omit) {
127
141
  const res = [];
128
142
  for (const key of ReflectUtilities.ownKeys(entity)) {
129
- const type = EntityUtilities.getPropertyType(entity, key);
143
+ const type = this.getPropertyType(entity, key);
130
144
  if (type === DecoratorTypes.FILE_DEFAULT || type === DecoratorTypes.FILE_IMAGE) {
131
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
145
+ const metadata = this.getPropertyMetadata(entity, key);
132
146
  if (!(metadata.omitForCreate && omit === 'create') && !(metadata.omitForUpdate && omit === 'update')) {
133
147
  res.push(key);
134
148
  }
@@ -185,12 +199,12 @@ export class EntityUtilities {
185
199
  */
186
200
  static new(target, entity) {
187
201
  for (const key in target) {
188
- const type = EntityUtilities.getPropertyType(target, key);
202
+ const type = this.getPropertyType(target, key);
189
203
  let value = entity ? ReflectUtilities.get(entity, key) : undefined;
190
204
  switch (type) {
191
205
  case DecoratorTypes.OBJECT:
192
206
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
193
- const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);
207
+ const objectMetadata = this.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);
194
208
  value = new objectMetadata.EntityClass(value);
195
209
  break;
196
210
  case DecoratorTypes.ARRAY:
@@ -198,7 +212,7 @@ export class EntityUtilities {
198
212
  const resArray = [];
199
213
  if (inputArray) {
200
214
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
201
- const arrayMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);
215
+ const arrayMetadata = this.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);
202
216
  for (const item of inputArray) {
203
217
  const itemWithMetadata = new arrayMetadata.EntityClass(item);
204
218
  resArray.push(itemWithMetadata);
@@ -213,333 +227,9 @@ export class EntityUtilities {
213
227
  }
214
228
  }
215
229
  // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc, @typescript-eslint/typedef
216
- static construct = EntityUtilities.new;
230
+ static construct = this.new;
217
231
  // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc, @typescript-eslint/typedef
218
- static build = EntityUtilities.new;
219
- /**
220
- * Checks if the values on an entity are valid.
221
- * Also checks all the validators given by the metadata ("required", "maxLength" etc.).
222
- *
223
- * @param entity - The entity to validate.
224
- * @param omit - Whether to check for creating or editing validity.
225
- * @returns Whether or not the entity is valid.
226
- */
227
- static isEntityValid(entity, omit) {
228
- for (const key in entity) {
229
- if (!EntityUtilities.isPropertyValid(entity, key, omit)) {
230
- return false;
231
- }
232
- }
233
- return true;
234
- }
235
- /**
236
- * Checks if a single property value is valid.
237
- *
238
- * @param entity - The entity where the property is from.
239
- * @param key - The name of the property.
240
- * @param omit - Whether to check if the given entity is valid for creation or updating.
241
- * @returns Whether or not the property value is valid.
242
- * @throws Throws when it extracts an unknown metadata type.
243
- */
244
- static isPropertyValid(entity, key, omit) {
245
- const type = EntityUtilities.getPropertyType(entity, key);
246
- const metadata = EntityUtilities.getPropertyMetadata(entity, key, type);
247
- if (metadata.omitForCreate && omit === 'create') {
248
- return true;
249
- }
250
- if (metadata.omitForUpdate && omit === 'update') {
251
- return true;
252
- }
253
- if (metadata.required(entity) && type !== DecoratorTypes.HAS_MANY) {
254
- if (entity[key] == null || entity[key] === '') {
255
- return false;
256
- }
257
- }
258
- if (!metadata.required(entity)) {
259
- if (entity[key] == null || entity[key] === '') {
260
- return true;
261
- }
262
- }
263
- switch (type) {
264
- case DecoratorTypes.BOOLEAN_DROPDOWN:
265
- break;
266
- case DecoratorTypes.BOOLEAN_CHECKBOX:
267
- case DecoratorTypes.BOOLEAN_TOGGLE:
268
- const entityBoolean = entity[key];
269
- const booleanMetadata = metadata;
270
- if (!EntityUtilities.isBooleanValid(entity, entityBoolean, booleanMetadata)) {
271
- return false;
272
- }
273
- break;
274
- case DecoratorTypes.STRING_DROPDOWN:
275
- break;
276
- case DecoratorTypes.STRING:
277
- case DecoratorTypes.STRING_AUTOCOMPLETE:
278
- const entityString = entity[key];
279
- const stringMetadata = metadata;
280
- if (!EntityUtilities.isStringValid(entityString, stringMetadata)) {
281
- return false;
282
- }
283
- break;
284
- case DecoratorTypes.STRING_TEXTBOX:
285
- const entityTextbox = entity[key];
286
- const textboxMetadata = metadata;
287
- if (!EntityUtilities.isTextboxValid(entityTextbox, textboxMetadata)) {
288
- return false;
289
- }
290
- break;
291
- case DecoratorTypes.STRING_PASSWORD:
292
- const entityPassword = entity[key];
293
- const passwordMetadata = metadata;
294
- const confirmPassword = ReflectUtilities.getMetadata(this.CONFIRM_PASSWORD_KEY, entity, key);
295
- if (!EntityUtilities.isPasswordValid(entityPassword, passwordMetadata, confirmPassword)) {
296
- return false;
297
- }
298
- break;
299
- case DecoratorTypes.NUMBER_DROPDOWN:
300
- return true;
301
- case DecoratorTypes.NUMBER:
302
- case DecoratorTypes.NUMBER_SLIDER:
303
- const entityNumber = entity[key];
304
- const numberMetadata = metadata;
305
- if (!EntityUtilities.isNumberValid(entityNumber, numberMetadata)) {
306
- return false;
307
- }
308
- break;
309
- case DecoratorTypes.OBJECT:
310
- const entityObject = entity[key];
311
- for (const parameterKey in entityObject) {
312
- const value = entityObject[parameterKey];
313
- if (!metadata.omit.includes(parameterKey)
314
- && !(!metadata.required(entity) && (value == null || value == ''))) {
315
- if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {
316
- return false;
317
- }
318
- }
319
- }
320
- break;
321
- case DecoratorTypes.ARRAY_STRING_CHIPS:
322
- case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:
323
- case DecoratorTypes.ARRAY_DATE:
324
- case DecoratorTypes.ARRAY_DATE_TIME:
325
- case DecoratorTypes.ARRAY_DATE_RANGE:
326
- case DecoratorTypes.ARRAY:
327
- const entityArray = entity[key];
328
- // eslint-disable-next-line max-len
329
- const arrayMetadata = metadata;
330
- if (arrayMetadata.required(entity) && !entityArray.length) {
331
- return false;
332
- }
333
- break;
334
- case DecoratorTypes.DATE:
335
- const entityDate = new Date(entity[key]);
336
- const dateMetadata = metadata;
337
- if (!EntityUtilities.isDateValid(entityDate, dateMetadata)) {
338
- return false;
339
- }
340
- break;
341
- case DecoratorTypes.DATE_RANGE:
342
- const entityDateRange = LodashUtilities.cloneDeep(entity[key]);
343
- const dateRangeMetadata = metadata;
344
- if (!EntityUtilities.isDateRangeValid(entity, entityDateRange, dateRangeMetadata)) {
345
- return false;
346
- }
347
- break;
348
- case DecoratorTypes.DATE_TIME:
349
- const entityDateTime = new Date(entity[key]);
350
- const dateTimeMetadata = metadata;
351
- const hasTime = ReflectUtilities.hasMetadata(this.TIME_KEY, entity, key);
352
- if (!EntityUtilities.isDateTimeValid(entityDateTime, dateTimeMetadata, hasTime)) {
353
- return false;
354
- }
355
- break;
356
- case DecoratorTypes.FILE_DEFAULT:
357
- case DecoratorTypes.FILE_IMAGE:
358
- const entityFile = entity[key];
359
- const entityFileMetadata = metadata;
360
- if (!EntityUtilities.isFileDataValid(entityFile, entityFileMetadata)) {
361
- return false;
362
- }
363
- break;
364
- case DecoratorTypes.REFERENCES_MANY:
365
- case DecoratorTypes.REFERENCES_ONE:
366
- case DecoratorTypes.HAS_MANY:
367
- break;
368
- case DecoratorTypes.CUSTOM:
369
- // eslint-disable-next-line @typescript-eslint/no-explicit-any, max-len
370
- const customMetadata = metadata;
371
- if (!customMetadata.isValid(entity[key], omit)) {
372
- return false;
373
- }
374
- break;
375
- default:
376
- throw new Error(`Could not validate the input because the DecoratorType ${type} is not known`);
377
- }
378
- return true;
379
- }
380
- static isBooleanValid(entity, value, metadata) {
381
- if (metadata.required(entity) && !value) {
382
- return false;
383
- }
384
- return true;
385
- }
386
- static isStringValid(value, metadata) {
387
- if (metadata.maxLength && value.length > metadata.maxLength) {
388
- return false;
389
- }
390
- if (metadata.minLength && value.length < metadata.minLength) {
391
- return false;
392
- }
393
- if (metadata.regex && !value.match(metadata.regex)) {
394
- return false;
395
- }
396
- return true;
397
- }
398
- static isTextboxValid(value, metadata) {
399
- if (metadata.maxLength && value.length > metadata.maxLength) {
400
- return false;
401
- }
402
- if (metadata.minLength && value.length < metadata.minLength) {
403
- return false;
404
- }
405
- return true;
406
- }
407
- static isPasswordValid(value, metadata, confirmPassword) {
408
- if (value !== confirmPassword) {
409
- return false;
410
- }
411
- if (metadata.maxLength && value.length > metadata.maxLength) {
412
- return false;
413
- }
414
- if (metadata.minLength && value.length < metadata.minLength) {
415
- return false;
416
- }
417
- if (metadata.regex && !value.match(metadata.regex)) {
418
- return false;
419
- }
420
- return true;
421
- }
422
- static isNumberValid(value, metadata) {
423
- if (metadata.max && value > metadata.max) {
424
- return false;
425
- }
426
- if (metadata.min && value < metadata.min) {
427
- return false;
428
- }
429
- return true;
430
- }
431
- static isDateValid(value, metadata) {
432
- if (metadata.min && value.getTime() < metadata.min(value).getTime()) {
433
- return false;
434
- }
435
- if (metadata.max && value.getTime() > metadata.max(value).getTime()) {
436
- return false;
437
- }
438
- if (metadata.filter && !metadata.filter(value)) {
439
- return false;
440
- }
441
- return true;
442
- }
443
- static isDateRangeValid(entity, value, metadata) {
444
- if (metadata.required(entity)) {
445
- if (!value.start) {
446
- return false;
447
- }
448
- if (!value.end) {
449
- return false;
450
- }
451
- }
452
- value.start = new Date(value.start);
453
- value.end = new Date(value.end);
454
- if (metadata.minStart && value.start.getTime() < metadata.minStart(value.start).getTime()) {
455
- return false;
456
- }
457
- if (metadata.maxStart && value.start.getTime() > metadata.maxStart(value.start).getTime()) {
458
- return false;
459
- }
460
- if (metadata.minEnd && value.end.getTime() < metadata.minEnd(value.end).getTime()) {
461
- return false;
462
- }
463
- if (metadata.maxEnd && value.end.getTime() > metadata.maxEnd(value.end).getTime()) {
464
- return false;
465
- }
466
- if (metadata.filter) {
467
- if (!metadata.filter(value.start)) {
468
- return false;
469
- }
470
- if (!metadata.filter(value.end)) {
471
- return false;
472
- }
473
- if (value.values) {
474
- for (const date of value.values) {
475
- if (!metadata.filter(date)) {
476
- return false;
477
- }
478
- }
479
- }
480
- }
481
- return true;
482
- }
483
- static isDateTimeValid(value, metadata, hasTime) {
484
- if (!hasTime) {
485
- return false;
486
- }
487
- if (metadata.minDate && value.getTime() < metadata.minDate(value).getTime()) {
488
- return false;
489
- }
490
- if (metadata.maxDate && value.getTime() > metadata.maxDate(value).getTime()) {
491
- return false;
492
- }
493
- if (metadata.filterDate && !metadata.filterDate(value)) {
494
- return false;
495
- }
496
- const time = {
497
- hours: value.getHours(),
498
- minutes: value.getMinutes()
499
- };
500
- if (metadata.minTime) {
501
- const minTime = metadata.minTime(value);
502
- if (!(time.hours > minTime.hours
503
- || (time.hours === minTime.hours
504
- && time.minutes >= minTime.minutes))) {
505
- return false;
506
- }
507
- }
508
- if (metadata.maxTime) {
509
- const maxTime = metadata.maxTime(value);
510
- if (!(time.hours < maxTime.hours
511
- || (time.hours === maxTime.hours
512
- && time.minutes <= maxTime.minutes))) {
513
- return false;
514
- }
515
- }
516
- if (metadata.filterTime) {
517
- if (!metadata.filterTime(time)) {
518
- return false;
519
- }
520
- }
521
- return true;
522
- }
523
- static isFileDataValid(value, metadata) {
524
- const files = metadata.multiple ? value : [value];
525
- let fileSizeTotal = 0;
526
- for (const file of files) {
527
- if (!file.name || !file.file && !file.url) {
528
- return false;
529
- }
530
- if (!FileUtilities.isMimeTypeValid(file.type, metadata.allowedMimeTypes)) {
531
- return false;
532
- }
533
- if (FileUtilities.transformToMegaBytes(file.size, 'B') > metadata.maxSize) {
534
- return false;
535
- }
536
- fileSizeTotal += file.size;
537
- if (FileUtilities.transformToMegaBytes(fileSizeTotal, 'B') > metadata.maxSizeTotal) {
538
- return false;
539
- }
540
- }
541
- return true;
542
- }
232
+ static build = this.new;
543
233
  /**
544
234
  * Checks if an entity is "dirty" (if its values have changed).
545
235
  *
@@ -552,15 +242,23 @@ export class EntityUtilities {
552
242
  if (!entityPriorChanges) {
553
243
  return false;
554
244
  }
555
- const differences = await EntityUtilities.differencesForDirty(entity, entityPriorChanges, http);
245
+ const differences = await this.getDifferencesBetweenEntities(entity, entityPriorChanges, http);
556
246
  return differences.length ? true : false;
557
247
  }
558
- static async differencesForDirty(entity, entityPriorChanges, http) {
248
+ /**
249
+ * Gets the differences between the two given entities.
250
+ *
251
+ * @param entity - The entity as is.
252
+ * @param entityPriorChanges - The entity before any changes have been made.
253
+ * @param http - The angular http client, is needed to check if files are equal.
254
+ * @returns The differences as an array consisting of key, before and after.
255
+ */
256
+ static async getDifferencesBetweenEntities(entity, entityPriorChanges, http) {
559
257
  const res = [];
560
258
  for (const key of ReflectUtilities.ownKeys(entity)) {
561
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
562
- const type = EntityUtilities.getPropertyType(entity, key);
563
- if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {
259
+ const metadata = this.getPropertyMetadata(entity, key);
260
+ const type = this.getPropertyType(entity, key);
261
+ if (!(await this.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {
564
262
  res.push({
565
263
  key: key,
566
264
  before: entityPriorChanges[key],
@@ -588,9 +286,9 @@ export class EntityUtilities {
588
286
  // ): Promise<Partial<EntityType>> {
589
287
  // const res: Partial<EntityType> = {};
590
288
  // for (const key in entity) {
591
- // const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);
592
- // const type: DecoratorTypes = EntityUtilities.getPropertyType(entity, key);
593
- // if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {
289
+ // const metadata: PropertyDecoratorConfigInternal = this.getPropertyMetadata(entity, key);
290
+ // const type: DecoratorTypes = this.getPropertyType(entity, key);
291
+ // if (!(await this.isEqual(entity[key], entityPriorChanges[key], metadata, type))) {
594
292
  // res[key] = entity[key];
595
293
  // }
596
294
  // }
@@ -610,26 +308,25 @@ export class EntityUtilities {
610
308
  static async isEqual(value, valuePriorChanges, metadata, type, http) {
611
309
  switch (type) {
612
310
  case DecoratorTypes.DATE_RANGE:
613
- return EntityUtilities.isEqualDateRange(value, valuePriorChanges, metadata.filter);
311
+ return this.isEqualDateRange(value, valuePriorChanges, metadata.filter);
614
312
  case DecoratorTypes.DATE:
615
- return EntityUtilities.isEqualDate(value, valuePriorChanges);
313
+ return this.isEqualDate(value, valuePriorChanges);
616
314
  case DecoratorTypes.DATE_TIME:
617
- return EntityUtilities.isEqualDateTime(value, valuePriorChanges);
315
+ return this.isEqualDateTime(value, valuePriorChanges);
618
316
  case DecoratorTypes.ARRAY_DATE:
619
317
  case DecoratorTypes.ARRAY_DATE_TIME:
620
- return EntityUtilities.isEqualArrayDate(value, valuePriorChanges);
318
+ return this.isEqualArrayDate(value, valuePriorChanges);
621
319
  case DecoratorTypes.ARRAY_DATE_RANGE:
622
- return EntityUtilities.isEqualArrayDateRange(value, valuePriorChanges, metadata.filter);
320
+ return this.isEqualArrayDateRange(value, valuePriorChanges, metadata.filter);
623
321
  case DecoratorTypes.ARRAY_STRING_CHIPS:
624
322
  case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:
625
- return EntityUtilities.isEqualArrayString(value, valuePriorChanges);
323
+ return this.isEqualArrayString(value, valuePriorChanges);
626
324
  case DecoratorTypes.FILE_IMAGE:
627
325
  case DecoratorTypes.FILE_DEFAULT:
628
- // eslint-disable-next-line max-len
629
- return EntityUtilities.isEqualFile(value, valuePriorChanges, metadata.multiple, http);
326
+ return this.isEqualFile(value, valuePriorChanges, metadata.multiple, http);
630
327
  case DecoratorTypes.CUSTOM:
631
- // eslint-disable-next-line max-len, @typescript-eslint/no-explicit-any
632
- return EntityUtilities.isEqualCustom(value, valuePriorChanges, metadata);
328
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
329
+ return this.isEqualCustom(value, valuePriorChanges, metadata);
633
330
  default:
634
331
  return LodashUtilities.isEqual(value, valuePriorChanges);
635
332
  }
@@ -651,7 +348,7 @@ export class EntityUtilities {
651
348
  return false;
652
349
  }
653
350
  for (let i = 0; i < dateRanges.length; i++) {
654
- if (!EntityUtilities.isEqualDateRange(dateRanges[i], dateRangesPriorChanges[i], filter)) {
351
+ if (!this.isEqualDateRange(dateRanges[i], dateRangesPriorChanges[i], filter)) {
655
352
  return false;
656
353
  }
657
354
  }
@@ -732,8 +429,8 @@ export class EntityUtilities {
732
429
  * @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'.
733
430
  */
734
431
  static compareOrder(a, b, entity) {
735
- const metadataA = EntityUtilities.getPropertyMetadata(entity, a);
736
- const metadataB = EntityUtilities.getPropertyMetadata(entity, b);
432
+ const metadataA = this.getPropertyMetadata(entity, a);
433
+ const metadataB = this.getPropertyMetadata(entity, b);
737
434
  if (metadataA.position.order === -1) {
738
435
  if (metadataB.position.order === -1) {
739
436
  return 0;
@@ -754,7 +451,7 @@ export class EntityUtilities {
754
451
  * @returns Bootstrap column value.
755
452
  */
756
453
  static getWidth(entity, key, type) {
757
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
454
+ const metadata = this.getPropertyMetadata(entity, key);
758
455
  switch (type) {
759
456
  case 'lg':
760
457
  return metadata.defaultWidths[0];
@@ -784,20 +481,20 @@ export class EntityUtilities {
784
481
  }
785
482
  static getEntityRows(entity, tab, hideOmitForCreate, hideOmitForEdit, additionalOmitValues) {
786
483
  const res = [];
787
- const keys = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit)
484
+ const keys = this.keysOf(entity, hideOmitForCreate, hideOmitForEdit)
788
485
  .filter(k => !additionalOmitValues.includes(k));
789
- const numberOfRows = EntityUtilities.getNumberOfRows(keys, entity, tab);
486
+ const numberOfRows = this.getNumberOfRows(keys, entity, tab);
790
487
  for (let i = 1; i <= numberOfRows; i++) {
791
488
  const row = {
792
489
  row: i,
793
- keys: EntityUtilities.getKeysForRow(keys, entity, i, tab)
490
+ keys: this.getKeysForRow(keys, entity, i, tab)
794
491
  };
795
492
  res.push(row);
796
493
  }
797
- if (EntityUtilities.getKeysForRow(keys, entity, -1, tab).length) {
494
+ if (this.getKeysForRow(keys, entity, -1, tab).length) {
798
495
  const lastRow = {
799
496
  row: numberOfRows + 1,
800
- keys: EntityUtilities.getKeysForRow(keys, entity, -1, tab)
497
+ keys: this.getKeysForRow(keys, entity, -1, tab)
801
498
  };
802
499
  res.push(lastRow);
803
500
  }
@@ -814,24 +511,24 @@ export class EntityUtilities {
814
511
  */
815
512
  static getEntityTabs(entity, hideOmitForCreate = false, hideOmitForEdit = false, additionalOmitValues = []) {
816
513
  const res = [];
817
- const keys = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit)
514
+ const keys = this.keysOf(entity, hideOmitForCreate, hideOmitForEdit)
818
515
  .filter(k => !additionalOmitValues.includes(k));
819
- const numberOfTabs = EntityUtilities.getNumberOfTabs(keys, entity);
516
+ const numberOfTabs = this.getNumberOfTabs(keys, entity);
820
517
  // eslint-disable-next-line max-len
821
- const firstTabRows = EntityUtilities.getEntityRows(entity, -1, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);
518
+ const firstTabRows = this.getEntityRows(entity, -1, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);
822
519
  if (firstTabRows.length) {
823
520
  const firstTab = {
824
- tabName: EntityUtilities.getFirstTabName(entity),
521
+ tabName: this.getFirstTabName(entity),
825
522
  tab: -1,
826
523
  rows: firstTabRows
827
524
  };
828
525
  res.push(firstTab);
829
526
  }
830
527
  for (let i = 2; i <= numberOfTabs; i++) {
831
- const rows = EntityUtilities.getEntityRows(entity, i, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);
528
+ const rows = this.getEntityRows(entity, i, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);
832
529
  if (rows.length) {
833
530
  const tab = {
834
- tabName: EntityUtilities.getTabName(entity, i),
531
+ tabName: this.getTabName(entity, i),
835
532
  tab: i,
836
533
  rows: rows
837
534
  };
@@ -842,30 +539,30 @@ export class EntityUtilities {
842
539
  }
843
540
  static getKeysForRow(keys, entity, row, tab) {
844
541
  return keys
845
- .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === row)
846
- .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab === tab)
847
- .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));
542
+ .filter(k => this.getPropertyMetadata(entity, k).position.row === row)
543
+ .filter(k => this.getPropertyMetadata(entity, k).position.tab === tab)
544
+ .sort((a, b) => this.compareOrder(a, b, entity));
848
545
  }
849
546
  static getNumberOfRows(keys, entity, tab) {
850
547
  return keys
851
- .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab === tab)
852
- .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)
548
+ .filter(k => this.getPropertyMetadata(entity, k).position.tab === tab)
549
+ .map(k => this.getPropertyMetadata(entity, k).position.row)
853
550
  .sort((a, b) => (a > b ? -1 : 1))[0];
854
551
  }
855
552
  static getNumberOfTabs(keys, entity) {
856
553
  return keys
857
- .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab)
554
+ .map(k => this.getPropertyMetadata(entity, k).position.tab)
858
555
  .sort((a, b) => (a > b ? -1 : 1))[0];
859
556
  }
860
557
  static getTabName(entity, tab) {
861
558
  const providedTabName = ReflectUtilities.ownKeys(entity)
862
- .map(k => EntityUtilities.getPropertyMetadata(entity, k))
559
+ .map(k => this.getPropertyMetadata(entity, k))
863
560
  .find(m => m.position.tab === tab && m.position.tabName)?.position.tabName;
864
561
  return providedTabName ?? `Tab ${tab}`;
865
562
  }
866
563
  static getFirstTabName(entity) {
867
564
  const providedTabName = ReflectUtilities.ownKeys(entity)
868
- .map(k => EntityUtilities.getPropertyMetadata(entity, k))
565
+ .map(k => this.getPropertyMetadata(entity, k))
869
566
  .find(m => m.position.tabName && m.position.tab === -1)?.position.tabName;
870
567
  return providedTabName ?? 'Tab 1';
871
568
  }
@@ -879,14 +576,14 @@ export class EntityUtilities {
879
576
  */
880
577
  static keysOf(entity, hideOmitForCreate = false, hideOmitForEdit = false) {
881
578
  let keys = ReflectUtilities.ownKeys(entity);
882
- const dontDisplayKeys = EntityUtilities.getDontDisplayKeys(entity);
579
+ const dontDisplayKeys = this.getDontDisplayKeys(entity);
883
580
  keys = keys.filter(k => !dontDisplayKeys.includes(k));
884
581
  if (hideOmitForCreate) {
885
- const omitForCreateKeys = EntityUtilities.getOmitForCreate(entity);
582
+ const omitForCreateKeys = this.getOmitForCreate(entity);
886
583
  keys = keys.filter(k => !omitForCreateKeys.includes(k));
887
584
  }
888
585
  if (hideOmitForEdit) {
889
- const omitForUpdateKeys = EntityUtilities.getOmitForUpdate(entity);
586
+ const omitForUpdateKeys = this.getOmitForUpdate(entity);
890
587
  keys = keys.filter(k => !omitForUpdateKeys.includes(k));
891
588
  }
892
589
  return keys;
@@ -894,7 +591,7 @@ export class EntityUtilities {
894
591
  static getDontDisplayKeys(entity) {
895
592
  const res = [];
896
593
  for (const key of ReflectUtilities.ownKeys(entity)) {
897
- const metadata = EntityUtilities.getPropertyMetadata(entity, key);
594
+ const metadata = this.getPropertyMetadata(entity, key);
898
595
  if (!metadata.display(entity)) {
899
596
  res.push(key);
900
597
  }
@@ -902,4 +599,4 @@ export class EntityUtilities {
902
599
  return res;
903
600
  }
904
601
  }
905
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.utilities.js","sourceRoot":"","sources":["../../../../projects/ngx-material-entity/src/utilities/entity.utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { Time } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { DateFilterFn } from '@angular/material/datepicker';\nimport { BaseEntityType } from '../classes/entity.model';\nimport { DateRangeArrayDecoratorConfigInternal, EntityArrayDecoratorConfigInternal } from '../decorators/array/array-decorator-internal.data';\nimport { DecoratorType, DecoratorTypes } from '../decorators/base/decorator-types.enum';\nimport { PropertyDecoratorConfigInternal } from '../decorators/base/property-decorator-internal.data';\nimport { ToggleBooleanDecoratorConfigInternal } from '../decorators/boolean/boolean-decorator-internal.data';\nimport { CustomDecoratorConfigInternal } from '../decorators/custom/custom-decorator-internal.data';\nimport { DateRangeDateDecoratorConfigInternal, DateTimeDateDecoratorConfigInternal, DefaultDateDecoratorConfigInternal } from '../decorators/date/date-decorator-internal.data';\nimport { DateRange } from '../decorators/date/date-decorator.data';\nimport { DefaultFileDecoratorConfigInternal } from '../decorators/file/file-decorator-internal.data';\nimport { FileData } from '../decorators/file/file-decorator.data';\nimport { DefaultNumberDecoratorConfigInternal } from '../decorators/number/number-decorator-internal.data';\nimport { DefaultObjectDecoratorConfigInternal } from '../decorators/object/object-decorator-internal.data';\nimport { DefaultStringDecoratorConfigInternal, PasswordStringDecoratorConfigInternal, TextboxStringDecoratorConfigInternal } from '../decorators/string/string-decorator-internal.data';\nimport { LodashUtilities } from '../encapsulation/lodash.utilities';\nimport { ReflectUtilities } from '../encapsulation/reflect.utilities';\nimport { DateUtilities } from './date.utilities';\nimport { FileUtilities } from './file.utilities';\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     * The key for all keys of metadata that should be set to undefined when the entity gets reset.\n     */\n    static readonly METADATA_KEYS_TO_RESET_KEY: string = 'metadataKeysToReset';\n\n    /**\n     * The key for the metadata that saves the single preview image value on image properties.\n     */\n    static readonly SINGLE_PREVIEW_IMAGE_KEY: string = 'singlePreviewImage';\n\n    /**\n     * The key for the metadata that saves the multi preview images value on image properties.\n     */\n    static readonly MULTI_PREVIEW_IMAGES_KEY: string = 'multiPreviewImages';\n\n    /**\n     * The key for the metadata that saves the filenames value on file properties.\n     */\n    static readonly FILENAMES_KEY: string = 'fileNames';\n\n    /**\n     * The key for the metadata that saves the confirm password value on password properties.\n     */\n    static readonly CONFIRM_PASSWORD_KEY: string = 'confirmPassword';\n\n    /**\n     * The key for the metadata that saves the time value on date time properties.\n     */\n    static readonly TIME_KEY: string = 'time';\n\n    /**\n     * The key for the metadata that saves the date range value on date range properties.\n     */\n    static readonly DATE_RANGE_KEY: string = 'dateRange';\n\n    /**\n     * The key for the metadata that saves the date range start value on date range properties.\n     */\n    static readonly DATE_RANGE_START_KEY: string = 'dateRangeStart';\n\n    /**\n     * The key for the metadata that saves the date range end value on date range properties.\n     */\n    static readonly DATE_RANGE_END_KEY: string = 'dateRangeEnd';\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 ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal = 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 ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForCreate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Returns the given entity without the values that should be omitted for creation.\n     *\n     * @param entity - The entity with all its values.\n     * @returns The reduced entity object.\n     */\n    static getWithoutOmitCreateValues<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): Partial<EntityType> {\n        return LodashUtilities.omit(entity, EntityUtilities.getOmitForCreate(entity)) as Partial<EntityType>;\n    }\n\n    /**\n     * Returns the given entity without the values that should be omitted for updating.\n     * This also handles omitting keys for @object or @array values and removes values that haven't been changed by default.\n     *\n     * @param entity - The entity with all its values.\n     * @param entityPriorChanges - The entity before any changes were applied.\n     * @param http - The angular HttpClient. Used to fetch files.\n     * @returns The reduced entity object.\n     */\n    static async getWithoutOmitUpdateValues<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType,\n        http: HttpClient\n    ): Promise<Partial<EntityType>> {\n        const res: Partial<EntityType> = {};\n        for (const key of EntityUtilities.keysOf(entity, false, true)) {\n            const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);\n            const type: DecoratorTypes = EntityUtilities.getPropertyType(entity, key);\n            if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {\n                switch (type) {\n                    case DecoratorTypes.OBJECT:\n                        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                        (res[key] as object) = LodashUtilities.omit(entity[key] as any, EntityUtilities.getOmitForCreate(entity[key]));\n                        break;\n                    case DecoratorTypes.ARRAY:\n                        (res[key] as object[]) = (entity[key] as object[])\n                            .map(value => LodashUtilities.omit(value, EntityUtilities.getOmitForCreate(value)));\n                        break;\n                    default:\n                        res[key] = entity[key];\n                        break;\n                }\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 ReflectUtilities.ownKeys(entity)) {\n            const type: DecoratorTypes = EntityUtilities.getPropertyType(entity, key);\n            if (type === DecoratorTypes.FILE_DEFAULT || type === DecoratorTypes.FILE_IMAGE) {\n                const metadata: PropertyDecoratorConfigInternal = 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: unknown = 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: unknown = 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: DecoratorTypes = EntityUtilities.getPropertyType(target, key);\n            let value: unknown = entity ? ReflectUtilities.get(entity, key) : undefined;\n            switch (type) {\n                case DecoratorTypes.OBJECT:\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    const objectMetadata: DefaultObjectDecoratorConfigInternal<any>\n                        = 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                        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                        const arrayMetadata: EntityArrayDecoratorConfigInternal<any>\n                            = 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, @typescript-eslint/typedef\n    static construct = EntityUtilities.new;\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc, @typescript-eslint/typedef\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: DecoratorTypes = 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) && type !== DecoratorTypes.HAS_MANY) {\n            if (entity[key] == null || entity[key] === '') {\n                return false;\n            }\n        }\n        if (!metadata.required(entity)) {\n            if (entity[key] == null || entity[key] === '') {\n                return true;\n            }\n        }\n        switch (type) {\n            case DecoratorTypes.BOOLEAN_DROPDOWN:\n                break;\n            case DecoratorTypes.BOOLEAN_CHECKBOX:\n            case DecoratorTypes.BOOLEAN_TOGGLE:\n                const entityBoolean: boolean = entity[key] as boolean;\n                const booleanMetadata: ToggleBooleanDecoratorConfigInternal = metadata as ToggleBooleanDecoratorConfigInternal;\n                if (!EntityUtilities.isBooleanValid(entity, 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: string = entity[key] as string;\n                const stringMetadata: DefaultStringDecoratorConfigInternal = metadata as DefaultStringDecoratorConfigInternal;\n                if (!EntityUtilities.isStringValid(entityString, stringMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_TEXTBOX:\n                const entityTextbox: string = entity[key] as string;\n                const textboxMetadata: TextboxStringDecoratorConfigInternal = metadata as TextboxStringDecoratorConfigInternal;\n                if (!EntityUtilities.isTextboxValid(entityTextbox, textboxMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_PASSWORD:\n                const entityPassword: string = entity[key] as string;\n                const passwordMetadata: PasswordStringDecoratorConfigInternal = metadata as PasswordStringDecoratorConfigInternal;\n                const confirmPassword: string = ReflectUtilities.getMetadata(this.CONFIRM_PASSWORD_KEY, 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: number = entity[key] as number;\n                const numberMetadata: DefaultNumberDecoratorConfigInternal = metadata as DefaultNumberDecoratorConfigInternal;\n                if (!EntityUtilities.isNumberValid(entityNumber, numberMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.OBJECT:\n                const entityObject: EntityType = entity[key] as EntityType;\n                for (const parameterKey in entityObject) {\n                    const value: unknown = entityObject[parameterKey];\n                    if (\n                        !(metadata as DefaultObjectDecoratorConfigInternal<EntityType>).omit.includes(parameterKey)\n                        && !(!metadata.required(entity) && (value == null || value == ''))\n                    ) {\n                        if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {\n                            return false;\n                        }\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: unknown[] = entity[key] as unknown[];\n                // eslint-disable-next-line max-len\n                const arrayMetadata: EntityArrayDecoratorConfigInternal<EntityType> = metadata as EntityArrayDecoratorConfigInternal<EntityType>;\n                if (arrayMetadata.required(entity) && !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: DefaultDateDecoratorConfigInternal = 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: DateRangeDateDecoratorConfigInternal = metadata as DateRangeDateDecoratorConfigInternal;\n                if (!EntityUtilities.isDateRangeValid(entity, 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: DateTimeDateDecoratorConfigInternal = metadata as DateTimeDateDecoratorConfigInternal;\n                const hasTime: boolean = ReflectUtilities.hasMetadata(this.TIME_KEY, entity, key);\n                if (!EntityUtilities.isDateTimeValid(entityDateTime, dateTimeMetadata, hasTime)) {\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: DefaultFileDecoratorConfigInternal = metadata as DefaultFileDecoratorConfigInternal;\n                if (!EntityUtilities.isFileDataValid(entityFile, entityFileMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.REFERENCES_MANY:\n            case DecoratorTypes.REFERENCES_ONE:\n            case DecoratorTypes.HAS_MANY:\n                break;\n            case DecoratorTypes.CUSTOM:\n                // eslint-disable-next-line @typescript-eslint/no-explicit-any, max-len\n                const customMetadata: CustomDecoratorConfigInternal<EntityType, any, any, any> = 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<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        value: boolean,\n        metadata: ToggleBooleanDecoratorConfigInternal\n    ): boolean {\n        if (metadata.required(entity) && !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<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        value: DateRange,\n        metadata: DateRangeDateDecoratorConfigInternal\n    ): boolean {\n        if (metadata.required(entity)) {\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, hasTime: boolean): boolean {\n        if (!hasTime) {\n            return false;\n        }\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: FileData[] = metadata.multiple ? value as FileData[] : [value as FileData];\n        let fileSizeTotal: number = 0;\n        for (const file of files) {\n            if (!file.name || !file.file && !file.url) {\n                return false;\n            }\n            if (!FileUtilities.isMimeTypeValid(file.type, metadata.allowedMimeTypes)) {\n                return false;\n            }\n            if (FileUtilities.transformToMegaBytes(file.size, 'B') > metadata.maxSize) {\n                return false;\n            }\n            fileSizeTotal += file.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     * @param http - The angular HttpClient. Used to fetch files.\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        http: HttpClient\n    ): Promise<boolean> {\n        if (!(entityPriorChanges as EntityType | undefined)) {\n            return false;\n        }\n        const differences: Difference<EntityType>[] = await EntityUtilities.differencesForDirty(entity, entityPriorChanges, http);\n        return differences.length ? true : false;\n    }\n\n    private static async differencesForDirty<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType,\n        http: HttpClient\n    ): Promise<Difference<EntityType>[]> {\n        const res: Difference<EntityType>[] = [];\n        for (const key of ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);\n            const type: DecoratorTypes = EntityUtilities.getPropertyType(entity, key);\n            if (!(await EntityUtilities.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {\n                res.push({\n                    key: key,\n                    before: entityPriorChanges[key],\n                    after: entity[key]\n                });\n            }\n            else {\n                // This is needed to set blob file data so that it is only requested once.\n                entityPriorChanges[key] = LodashUtilities.cloneDeep(entity[key]);\n            }\n        }\n        return res;\n    }\n\n    // TODO Remove\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: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);\n    //         const type: DecoratorTypes = 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     * @param http - The angular HttpClient. Used to fetch files.\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        http: HttpClient\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.ARRAY_STRING_CHIPS:\n            case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:\n                return EntityUtilities.isEqualArrayString(value, valuePriorChanges);\n            case DecoratorTypes.FILE_IMAGE:\n            case DecoratorTypes.FILE_DEFAULT:\n                // eslint-disable-next-line max-len\n                return EntityUtilities.isEqualFile(value, valuePriorChanges, (metadata as DefaultFileDecoratorConfigInternal).multiple, http);\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 isEqualArrayString(value: unknown, valuePriorChanges: unknown): boolean | PromiseLike<boolean> {\n        const stringArray: string[] = LodashUtilities.cloneDeep(value as string[]).sort();\n        const stringArrayPriorChanges: string[] = LodashUtilities.cloneDeep(valuePriorChanges as string[]).sort();\n        return LodashUtilities.isEqual(stringArray, stringArrayPriorChanges);\n    }\n\n    private static isEqualArrayDate(value: unknown, valuePriorChanges: unknown): boolean {\n        const newValue: Date[] = (value as Date[]).map(v => new Date(v)).sort();\n        const newValuePriorChanges: Date[] = (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: DateRange[] = (value as DateRange[]).sort();\n        const dateRangesPriorChanges: DateRange[] = (valuePriorChanges as DateRange[]).sort();\n        if (dateRanges.length !== dateRangesPriorChanges.length) {\n            return false;\n        }\n        for (let i: number = 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: Date = new Date(value as Date);\n        const datePriorChanges: Date = 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: Date = new Date(value as Date);\n        const datePriorChanges: Date = 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: 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: DateRange = 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, http: HttpClient): Promise<boolean> {\n        if (value == null) {\n            if (valuePriorChanges == null) {\n                return true;\n            }\n            else {\n                return false;\n            }\n        }\n        const files: FileData[] = multiple ? (value as FileData[]).sort() : [value as FileData].sort();\n        const filesPriorChanges: FileData[] = multiple ? (valuePriorChanges as FileData[]).sort() : [valuePriorChanges as FileData].sort();\n        if (files.length !== filesPriorChanges.length) {\n            return false;\n        }\n        for (let i: number = 0; i < files.length; i++) {\n            // checks this before actually getting any files due to performance reasons.\n            if (!LodashUtilities.isEqual(LodashUtilities.omit(files[i], 'file'), LodashUtilities.omit(filesPriorChanges[i], 'file'))) {\n                return false;\n            }\n            if (filesPriorChanges[i].file && !files[i].file) {\n                files[i] = await FileUtilities.getFileData(files[i], http);\n                value = files[i];\n            }\n            if (files[i].file && !filesPriorChanges[i].file) {\n                filesPriorChanges[i] = await FileUtilities.getFileData(filesPriorChanges[i], http);\n                valuePriorChanges = filesPriorChanges[i];\n            }\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: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, a);\n        const metadataB: PropertyDecoratorConfigInternal = 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: PropertyDecoratorConfigInternal = 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            if (ReflectUtilities.hasMetadata(this.METADATA_KEYS_TO_RESET_KEY, entity, key)) {\n                for (const k of (ReflectUtilities.getMetadata(this.METADATA_KEYS_TO_RESET_KEY, entity, key) as string[])) {\n                    if (ReflectUtilities.hasMetadata(k, entity, key)) {\n                        ReflectUtilities.defineMetadata(k, undefined, entity, key);\n                    }\n                }\n            }\n        }\n    }\n\n    private static getEntityRows<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        tab: number,\n        hideOmitForCreate: boolean,\n        hideOmitForEdit: boolean,\n        additionalOmitValues: (keyof EntityType)[]\n    ): EntityRow<EntityType>[] {\n        const res: EntityRow<EntityType>[] = [];\n\n        const keys: (keyof EntityType)[] = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit)\n            .filter(k => !additionalOmitValues.includes(k));\n        const numberOfRows: number = EntityUtilities.getNumberOfRows<EntityType>(keys, entity, tab);\n        for (let i: number = 1; i <= numberOfRows; i++) {\n            const row: EntityRow<EntityType> = {\n                row: i,\n                keys: EntityUtilities.getKeysForRow<EntityType>(keys, entity, i, tab)\n            };\n            res.push(row);\n        }\n\n        if (EntityUtilities.getKeysForRow<EntityType>(keys, entity, -1, tab).length) {\n            const lastRow: EntityRow<EntityType> = {\n                row: numberOfRows + 1,\n                keys: EntityUtilities.getKeysForRow<EntityType>(keys, entity, -1, tab)\n            };\n            res.push(lastRow);\n        }\n\n        return res;\n    }\n\n    /**\n     * Gets the tabs 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     * @param additionalOmitValues - Additional omit values.\n     * @returns The sorted Tabs containing the rows and the keys to display in that row.\n     */\n    static getEntityTabs<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false,\n        additionalOmitValues: (keyof EntityType)[] = []\n    ): EntityTab<EntityType>[] {\n        const res: EntityTab<EntityType>[] = [];\n        const keys: (keyof EntityType)[] = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit)\n            .filter(k => !additionalOmitValues.includes(k));\n        const numberOfTabs: number = EntityUtilities.getNumberOfTabs<EntityType>(keys, entity);\n\n        // eslint-disable-next-line max-len\n        const firstTabRows: EntityRow<EntityType>[] = EntityUtilities.getEntityRows<EntityType>(entity, -1, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);\n        if (firstTabRows.length) {\n            const firstTab: EntityTab<EntityType> = {\n                tabName: EntityUtilities.getFirstTabName(entity),\n                tab: -1,\n                rows: firstTabRows\n            };\n            res.push(firstTab);\n        }\n\n        for (let i: number = 2; i <= numberOfTabs; i++) {\n            const rows: EntityRow<EntityType>[] = EntityUtilities.getEntityRows<EntityType>(\n                entity, i, hideOmitForCreate, hideOmitForEdit, additionalOmitValues\n            );\n            if (rows.length) {\n                const tab: EntityTab<EntityType> = {\n                    tabName: EntityUtilities.getTabName(entity, i),\n                    tab: i,\n                    rows: rows\n                };\n                res.push(tab);\n            }\n        }\n\n        return res;\n    }\n\n    private static getKeysForRow<EntityType extends BaseEntityType<EntityType>>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        row: number,\n        tab: number\n    ): (keyof EntityType)[] {\n        return keys\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === row)\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab === tab)\n            .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));\n    }\n\n    private static getNumberOfRows<EntityType extends BaseEntityType<EntityType>>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        tab: number\n    ): number {\n        return keys\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab === tab)\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    private static getNumberOfTabs<EntityType extends BaseEntityType<EntityType>>(keys: (keyof EntityType)[], entity: EntityType): number {\n        return keys\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.tab)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    private static getTabName<EntityType extends BaseEntityType<EntityType>>(entity: EntityType, tab: number): string {\n        const providedTabName: string | undefined = ReflectUtilities.ownKeys(entity)\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k))\n            .find(m => m.position.tab === tab && m.position.tabName)?.position.tabName;\n        return providedTabName ?? `Tab ${tab}`;\n    }\n\n    private static getFirstTabName<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): string {\n        const providedTabName: string | undefined = ReflectUtilities.ownKeys(entity)\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k))\n            .find(m => m.position.tabName && m.position.tab === -1)?.position.tabName;\n        return providedTabName ?? 'Tab 1';\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        const dontDisplayKeys: (keyof EntityType)[] = EntityUtilities.getDontDisplayKeys(entity);\n        keys = keys.filter(k => !dontDisplayKeys.includes(k));\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    private static getDontDisplayKeys<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key);\n            if (!metadata.display(entity)) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n}\n\n/**\n * A row that contains 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}\n\n/**\n * A tab that contains all the information about how to display an entity.\n */\nexport interface EntityTab<EntityType extends BaseEntityType<EntityType>> {\n    /**\n     * The tab in which the rows should be displayed.\n     */\n    tab: number,\n    /**\n     * The name to display inside the tab.\n     */\n    tabName: string,\n    /**\n     * The rows that should be displayed inside this tab,.\n     */\n    rows: EntityRow<EntityType>[]\n}"]}
602
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.utilities.js","sourceRoot":"","sources":["../../../../projects/ngx-material-entity/src/utilities/entity.utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\nimport { DateFilterFn } from '@angular/material/datepicker';\nimport { BaseEntityType } from '../classes/entity.model';\nimport { DateRangeArrayDecoratorConfigInternal, EntityArrayDecoratorConfigInternal } from '../decorators/array/array-decorator-internal.data';\nimport { DecoratorType, DecoratorTypes } from '../decorators/base/decorator-types.enum';\nimport { PropertyDecoratorConfigInternal } from '../decorators/base/property-decorator-internal.data';\nimport { CustomDecoratorConfigInternal } from '../decorators/custom/custom-decorator-internal.data';\nimport { DateRangeDateDecoratorConfigInternal } from '../decorators/date/date-decorator-internal.data';\nimport { DateRange } from '../decorators/date/date-decorator.data';\nimport { DefaultFileDecoratorConfigInternal } from '../decorators/file/file-decorator-internal.data';\nimport { FileData } from '../decorators/file/file-decorator.data';\nimport { DefaultObjectDecoratorConfigInternal } from '../decorators/object/object-decorator-internal.data';\nimport { LodashUtilities } from '../encapsulation/lodash.utilities';\nimport { ReflectUtilities } from '../encapsulation/reflect.utilities';\nimport { DateUtilities } from './date.utilities';\nimport { FileUtilities } from './file.utilities';\n\n/**\n * Shows information about differences between two entities.\n */\nexport interface 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     * The key for all keys of metadata that should be set to undefined when the entity gets reset.\n     */\n    static readonly METADATA_KEYS_TO_RESET_KEY: string = 'metadataKeysToReset';\n\n    /**\n     * The key for the metadata that saves the single preview image value on image properties.\n     */\n    static readonly SINGLE_PREVIEW_IMAGE_KEY: string = 'singlePreviewImage';\n\n    /**\n     * The key for the metadata that saves the multi preview images value on image properties.\n     */\n    static readonly MULTI_PREVIEW_IMAGES_KEY: string = 'multiPreviewImages';\n\n    /**\n     * The key for the metadata that saves the filenames value on file properties.\n     */\n    static readonly FILENAMES_KEY: string = 'fileNames';\n\n    /**\n     * The key for the metadata that saves the confirm password value on password properties.\n     */\n    static readonly CONFIRM_PASSWORD_KEY: string = 'confirmPassword';\n\n    /**\n     * The key for the metadata that saves the time value on date time properties.\n     */\n    static readonly TIME_KEY: string = 'time';\n\n    /**\n     * The key for the metadata that saves the date range value on date range properties.\n     */\n    static readonly DATE_RANGE_KEY: string = 'dateRange';\n\n    /**\n     * The key for the metadata that saves the date range start value on date range properties.\n     */\n    static readonly DATE_RANGE_START_KEY: string = 'dateRangeStart';\n\n    /**\n     * The key for the metadata that saves the date range end value on date range properties.\n     */\n    static readonly DATE_RANGE_END_KEY: string = 'dateRangeEnd';\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 ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.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 ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, key);\n            if (metadata.omitForCreate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Returns the given entity without the values that should be omitted for creation.\n     *\n     * @param entity - The entity with all its values.\n     * @returns The reduced entity object.\n     */\n    static getWithoutOmitCreateValues<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): Partial<EntityType> {\n        return LodashUtilities.omit(entity, this.getOmitForCreate(entity)) as Partial<EntityType>;\n    }\n\n    /**\n     * Returns the given entity without the values that should be omitted for updating.\n     * This also handles omitting keys for @object or @array values and removes values that haven't been changed by default.\n     *\n     * @param entity - The entity with all its values.\n     * @param entityPriorChanges - The entity before any changes were applied.\n     * @param http - The angular HttpClient. Used to fetch files.\n     * @returns The reduced entity object.\n     */\n    static async getWithoutOmitUpdateValues<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType,\n        http: HttpClient\n    ): Promise<Partial<EntityType>> {\n        const res: Partial<EntityType> = {};\n        for (const key of this.keysOf(entity, false, true)) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, key);\n            const type: DecoratorTypes = this.getPropertyType(entity, key);\n            if (!(await this.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {\n                switch (type) {\n                    case DecoratorTypes.OBJECT:\n                        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                        (res[key] as object) = LodashUtilities.omit(entity[key] as any, this.getOmitForCreate(entity[key]));\n                        break;\n                    case DecoratorTypes.ARRAY:\n                        (res[key] as object[]) = (entity[key] as object[])\n                            .map(value => LodashUtilities.omit(value, this.getOmitForCreate(value)));\n                        break;\n                    default:\n                        res[key] = entity[key];\n                        break;\n                }\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Sets all default values on the given entity.\n     *\n     * @param entity - The entity to set the default values on.\n     */\n    static setDefaultValues<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): void {\n        for (const key in entity) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, key);\n            if (metadata.default) {\n                // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-explicit-any\n                entity[key] = metadata.default() as any;\n            }\n        }\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 ReflectUtilities.ownKeys(entity)) {\n            const type: DecoratorTypes = this.getPropertyType(entity, key);\n            if (type === DecoratorTypes.FILE_DEFAULT || type === DecoratorTypes.FILE_IMAGE) {\n                const metadata: PropertyDecoratorConfigInternal<unknown> = this.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: unknown = 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: unknown = 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: DecoratorTypes = this.getPropertyType(target, key);\n            let value: unknown = entity ? ReflectUtilities.get(entity, key) : undefined;\n            switch (type) {\n                case DecoratorTypes.OBJECT:\n                    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                    const objectMetadata: DefaultObjectDecoratorConfigInternal<any>\n                        = this.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                        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                        const arrayMetadata: EntityArrayDecoratorConfigInternal<any>\n                            = this.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, @typescript-eslint/typedef\n    static construct = this.new;\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc, @typescript-eslint/typedef\n    static build = this.new;\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     * @param http - The angular HttpClient. Used to fetch files.\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        http: HttpClient\n    ): Promise<boolean> {\n        if (!(entityPriorChanges as EntityType | undefined)) {\n            return false;\n        }\n        const differences: Difference<EntityType>[] = await this.getDifferencesBetweenEntities(entity, entityPriorChanges, http);\n        return differences.length ? true : false;\n    }\n\n    /**\n     * Gets the differences between the two given entities.\n     *\n     * @param entity - The entity as is.\n     * @param entityPriorChanges - The entity before any changes have been made.\n     * @param http - The angular http client, is needed to check if files are equal.\n     * @returns The differences as an array consisting of key, before and after.\n     */\n    static async getDifferencesBetweenEntities<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        entityPriorChanges: EntityType,\n        http: HttpClient\n    ): Promise<Difference<EntityType>[]> {\n        const res: Difference<EntityType>[] = [];\n        for (const key of ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, key);\n            const type: DecoratorTypes = this.getPropertyType(entity, key);\n            if (!(await this.isEqual(entity[key], entityPriorChanges[key], metadata, type, http))) {\n                res.push({\n                    key: key,\n                    before: entityPriorChanges[key],\n                    after: entity[key]\n                });\n            }\n            else {\n                // This is needed to set blob file data so that it is only requested once.\n                entityPriorChanges[key] = LodashUtilities.cloneDeep(entity[key]);\n            }\n        }\n        return res;\n    }\n\n    // TODO Remove\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: PropertyDecoratorConfigInternal = this.getPropertyMetadata(entity, key);\n    //         const type: DecoratorTypes = this.getPropertyType(entity, key);\n    //         if (!(await this.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     * @param http - The angular HttpClient. Used to fetch files.\n     * @returns Whether or not the given values are equal.\n     */\n    static async isEqual(\n        value: unknown,\n        valuePriorChanges: unknown,\n        metadata: PropertyDecoratorConfigInternal<unknown>,\n        type: DecoratorTypes,\n        http: HttpClient\n    ): Promise<boolean> {\n        switch (type) {\n            case DecoratorTypes.DATE_RANGE:\n                return this.isEqualDateRange(\n                    value,\n                    valuePriorChanges,\n                    (metadata as DateRangeDateDecoratorConfigInternal).filter\n                );\n            case DecoratorTypes.DATE:\n                return this.isEqualDate(value, valuePriorChanges);\n            case DecoratorTypes.DATE_TIME:\n                return this.isEqualDateTime(value, valuePriorChanges);\n            case DecoratorTypes.ARRAY_DATE:\n            case DecoratorTypes.ARRAY_DATE_TIME:\n                return this.isEqualArrayDate(value, valuePriorChanges);\n            case DecoratorTypes.ARRAY_DATE_RANGE:\n                return this.isEqualArrayDateRange(\n                    value,\n                    valuePriorChanges,\n                    (metadata as DateRangeArrayDecoratorConfigInternal).filter\n                );\n            case DecoratorTypes.ARRAY_STRING_CHIPS:\n            case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:\n                return this.isEqualArrayString(value, valuePriorChanges);\n            case DecoratorTypes.FILE_IMAGE:\n            case DecoratorTypes.FILE_DEFAULT:\n                return this.isEqualFile(value, valuePriorChanges, (metadata as DefaultFileDecoratorConfigInternal).multiple, http);\n            case DecoratorTypes.CUSTOM:\n                // eslint-disable-next-line @typescript-eslint/no-explicit-any\n                return this.isEqualCustom(value, valuePriorChanges, metadata as CustomDecoratorConfigInternal<any, any, any, any>);\n            default:\n                return LodashUtilities.isEqual(value, valuePriorChanges);\n        }\n    }\n\n    private static isEqualArrayString(value: unknown, valuePriorChanges: unknown): boolean | PromiseLike<boolean> {\n        const stringArray: string[] = LodashUtilities.cloneDeep(value as string[]).sort();\n        const stringArrayPriorChanges: string[] = LodashUtilities.cloneDeep(valuePriorChanges as string[]).sort();\n        return LodashUtilities.isEqual(stringArray, stringArrayPriorChanges);\n    }\n\n    private static isEqualArrayDate(value: unknown, valuePriorChanges: unknown): boolean {\n        const newValue: Date[] = (value as Date[]).map(v => new Date(v)).sort();\n        const newValuePriorChanges: Date[] = (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: DateRange[] = (value as DateRange[]).sort();\n        const dateRangesPriorChanges: DateRange[] = (valuePriorChanges as DateRange[]).sort();\n        if (dateRanges.length !== dateRangesPriorChanges.length) {\n            return false;\n        }\n        for (let i: number = 0; i < dateRanges.length; i++) {\n            if (!this.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: Date = new Date(value as Date);\n        const datePriorChanges: Date = 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: Date = new Date(value as Date);\n        const datePriorChanges: Date = 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: 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: DateRange = 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, http: HttpClient): Promise<boolean> {\n        if (value == null) {\n            if (valuePriorChanges == null) {\n                return true;\n            }\n            else {\n                return false;\n            }\n        }\n        const files: FileData[] = multiple ? (value as FileData[]).sort() : [value as FileData].sort();\n        const filesPriorChanges: FileData[] = multiple ? (valuePriorChanges as FileData[]).sort() : [valuePriorChanges as FileData].sort();\n        if (files.length !== filesPriorChanges.length) {\n            return false;\n        }\n        for (let i: number = 0; i < files.length; i++) {\n            // checks this before actually getting any files due to performance reasons.\n            if (!LodashUtilities.isEqual(LodashUtilities.omit(files[i], 'file'), LodashUtilities.omit(filesPriorChanges[i], 'file'))) {\n                return false;\n            }\n            if (filesPriorChanges[i].file && !files[i].file) {\n                files[i] = await FileUtilities.getFileData(files[i], http);\n                value = files[i];\n            }\n            if (files[i].file && !filesPriorChanges[i].file) {\n                filesPriorChanges[i] = await FileUtilities.getFileData(filesPriorChanges[i], http);\n                valuePriorChanges = filesPriorChanges[i];\n            }\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: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, a);\n        const metadataB: PropertyDecoratorConfigInternal<unknown> = this.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: PropertyDecoratorConfigInternal<unknown> = this.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            if (ReflectUtilities.hasMetadata(this.METADATA_KEYS_TO_RESET_KEY, entity, key)) {\n                for (const k of (ReflectUtilities.getMetadata(this.METADATA_KEYS_TO_RESET_KEY, entity, key) as string[])) {\n                    if (ReflectUtilities.hasMetadata(k, entity, key)) {\n                        ReflectUtilities.defineMetadata(k, undefined, entity, key);\n                    }\n                }\n            }\n        }\n    }\n\n    private static getEntityRows<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        tab: number,\n        hideOmitForCreate: boolean,\n        hideOmitForEdit: boolean,\n        additionalOmitValues: (keyof EntityType)[]\n    ): EntityRow<EntityType>[] {\n        const res: EntityRow<EntityType>[] = [];\n\n        const keys: (keyof EntityType)[] = this.keysOf(entity, hideOmitForCreate, hideOmitForEdit)\n            .filter(k => !additionalOmitValues.includes(k));\n        const numberOfRows: number = this.getNumberOfRows<EntityType>(keys, entity, tab);\n        for (let i: number = 1; i <= numberOfRows; i++) {\n            const row: EntityRow<EntityType> = {\n                row: i,\n                keys: this.getKeysForRow<EntityType>(keys, entity, i, tab)\n            };\n            res.push(row);\n        }\n\n        if (this.getKeysForRow<EntityType>(keys, entity, -1, tab).length) {\n            const lastRow: EntityRow<EntityType> = {\n                row: numberOfRows + 1,\n                keys: this.getKeysForRow<EntityType>(keys, entity, -1, tab)\n            };\n            res.push(lastRow);\n        }\n\n        return res;\n    }\n\n    /**\n     * Gets the tabs 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     * @param additionalOmitValues - Additional omit values.\n     * @returns The sorted Tabs containing the rows and the keys to display in that row.\n     */\n    static getEntityTabs<EntityType extends BaseEntityType<EntityType>>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false,\n        additionalOmitValues: (keyof EntityType)[] = []\n    ): EntityTab<EntityType>[] {\n        const res: EntityTab<EntityType>[] = [];\n        const keys: (keyof EntityType)[] = this.keysOf(entity, hideOmitForCreate, hideOmitForEdit)\n            .filter(k => !additionalOmitValues.includes(k));\n        const numberOfTabs: number = this.getNumberOfTabs<EntityType>(keys, entity);\n\n        // eslint-disable-next-line max-len\n        const firstTabRows: EntityRow<EntityType>[] = this.getEntityRows<EntityType>(entity, -1, hideOmitForCreate, hideOmitForEdit, additionalOmitValues);\n        if (firstTabRows.length) {\n            const firstTab: EntityTab<EntityType> = {\n                tabName: this.getFirstTabName(entity),\n                tab: -1,\n                rows: firstTabRows\n            };\n            res.push(firstTab);\n        }\n\n        for (let i: number = 2; i <= numberOfTabs; i++) {\n            const rows: EntityRow<EntityType>[] = this.getEntityRows<EntityType>(\n                entity, i, hideOmitForCreate, hideOmitForEdit, additionalOmitValues\n            );\n            if (rows.length) {\n                const tab: EntityTab<EntityType> = {\n                    tabName: this.getTabName(entity, i),\n                    tab: i,\n                    rows: rows\n                };\n                res.push(tab);\n            }\n        }\n\n        return res;\n    }\n\n    private static getKeysForRow<EntityType extends BaseEntityType<EntityType>>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        row: number,\n        tab: number\n    ): (keyof EntityType)[] {\n        return keys\n            .filter(k => this.getPropertyMetadata(entity, k).position.row === row)\n            .filter(k => this.getPropertyMetadata(entity, k).position.tab === tab)\n            .sort((a, b) => this.compareOrder(a, b, entity));\n    }\n\n    private static getNumberOfRows<EntityType extends BaseEntityType<EntityType>>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        tab: number\n    ): number {\n        return keys\n            .filter(k => this.getPropertyMetadata(entity, k).position.tab === tab)\n            .map(k => this.getPropertyMetadata(entity, k).position.row)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    private static getNumberOfTabs<EntityType extends BaseEntityType<EntityType>>(keys: (keyof EntityType)[], entity: EntityType): number {\n        return keys\n            .map(k => this.getPropertyMetadata(entity, k).position.tab)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    private static getTabName<EntityType extends BaseEntityType<EntityType>>(entity: EntityType, tab: number): string {\n        const providedTabName: string | undefined = ReflectUtilities.ownKeys(entity)\n            .map(k => this.getPropertyMetadata(entity, k))\n            .find(m => m.position.tab === tab && m.position.tabName)?.position.tabName;\n        return providedTabName ?? `Tab ${tab}`;\n    }\n\n    private static getFirstTabName<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): string {\n        const providedTabName: string | undefined = ReflectUtilities.ownKeys(entity)\n            .map(k => this.getPropertyMetadata(entity, k))\n            .find(m => m.position.tabName && m.position.tab === -1)?.position.tabName;\n        return providedTabName ?? 'Tab 1';\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        const dontDisplayKeys: (keyof EntityType)[] = this.getDontDisplayKeys(entity);\n        keys = keys.filter(k => !dontDisplayKeys.includes(k));\n        if (hideOmitForCreate) {\n            const omitForCreateKeys: (keyof EntityType)[] = this.getOmitForCreate(entity);\n            keys = keys.filter(k => !omitForCreateKeys.includes(k));\n        }\n        if (hideOmitForEdit) {\n            const omitForUpdateKeys: (keyof EntityType)[] = this.getOmitForUpdate(entity);\n            keys = keys.filter(k => !omitForUpdateKeys.includes(k));\n        }\n        return keys;\n    }\n\n    private static getDontDisplayKeys<EntityType extends BaseEntityType<EntityType>>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of ReflectUtilities.ownKeys(entity)) {\n            const metadata: PropertyDecoratorConfigInternal<unknown> = this.getPropertyMetadata(entity, key);\n            if (!metadata.display(entity)) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n}\n\n/**\n * A row that contains 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}\n\n/**\n * A tab that contains all the information about how to display an entity.\n */\nexport interface EntityTab<EntityType extends BaseEntityType<EntityType>> {\n    /**\n     * The tab in which the rows should be displayed.\n     */\n    tab: number,\n    /**\n     * The name to display inside the tab.\n     */\n    tabName: string,\n    /**\n     * The rows that should be displayed inside this tab,.\n     */\n    rows: EntityRow<EntityType>[]\n}"]}