ngx-material-entity 0.1.1 → 0.1.4

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 (133) hide show
  1. package/CONTRIBUTING.md +7 -1
  2. package/README.md +536 -328
  3. package/classes/base.builder.d.ts +35 -0
  4. package/classes/date.utilities.d.ts +58 -0
  5. package/classes/entity.model.d.ts +13 -0
  6. package/classes/{entity-service.class.d.ts → entity.service.d.ts} +35 -20
  7. package/classes/entity.utilities.d.ts +153 -0
  8. package/components/confirm-dialog/confirm-dialog-data.builder.d.ts +23 -0
  9. package/components/confirm-dialog/confirm-dialog-data.d.ts +18 -8
  10. package/components/confirm-dialog/confirm-dialog.component.d.ts +15 -5
  11. package/components/get-validation-error-message.function.d.ts +3 -2
  12. package/components/input/add-array-item-dialog-data.builder.d.ts +20 -0
  13. package/components/input/add-array-item-dialog-data.d.ts +19 -0
  14. package/components/input/input.component.d.ts +151 -30
  15. package/components/input/input.module.d.ts +7 -4
  16. package/components/table/create-dialog/create-dialog-data.builder.d.ts +21 -0
  17. package/components/table/create-dialog/create-entity-dialog-data.builder.d.ts +21 -0
  18. package/components/table/create-dialog/create-entity-dialog-data.d.ts +4 -5
  19. package/components/table/create-dialog/create-entity-dialog.component.d.ts +21 -8
  20. package/components/table/edit-dialog/edit-dialog-data.builder.d.ts +24 -0
  21. package/components/table/edit-dialog/edit-entity-dialog-data.d.ts +7 -8
  22. package/components/table/edit-dialog/edit-entity-dialog.builder.d.ts +22 -0
  23. package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +25 -8
  24. package/components/table/table-data.builder.d.ts +51 -0
  25. package/components/table/table-data.d.ts +46 -30
  26. package/components/table/table.component.d.ts +53 -7
  27. package/components/table/table.module.d.ts +3 -1
  28. package/decorators/array/array-decorator-internal.data.d.ts +45 -0
  29. package/decorators/array/array-decorator.data.d.ts +129 -0
  30. package/decorators/array/array.decorator.d.ts +9 -0
  31. package/decorators/base/base-property.decorator.d.ts +7 -6
  32. package/decorators/base/decorator-types.enum.d.ts +12 -9
  33. package/decorators/base/dropdown-value.interface.d.ts +14 -0
  34. package/decorators/base/property-decorator-internal.data.d.ts +24 -0
  35. package/decorators/base/property-decorator.data.d.ts +70 -0
  36. package/decorators/boolean/boolean-decorator-internal.data.d.ts +25 -0
  37. package/decorators/boolean/boolean-decorator.data.d.ts +37 -0
  38. package/decorators/boolean/boolean.decorator.d.ts +8 -0
  39. package/decorators/date/date-decorator-internal.data.d.ts +44 -0
  40. package/decorators/date/date-decorator.data.d.ts +129 -0
  41. package/decorators/number/number-decorator-internal.data.d.ts +20 -0
  42. package/decorators/number/number-decorator.data.d.ts +36 -0
  43. package/decorators/number/number.decorator.d.ts +8 -0
  44. package/decorators/object/object-decorator-internal.data.d.ts +11 -0
  45. package/decorators/object/object-decorator.data.d.ts +25 -0
  46. package/decorators/object/object.decorator.d.ts +8 -0
  47. package/decorators/string/string-decorator-internal.data.d.ts +41 -0
  48. package/decorators/string/string-decorator.data.d.ts +77 -0
  49. package/decorators/string/string.decorator.d.ts +8 -0
  50. package/esm2020/classes/base.builder.mjs +43 -0
  51. package/esm2020/classes/date.utilities.mjs +138 -0
  52. package/esm2020/classes/entity.model.mjs +19 -0
  53. package/esm2020/classes/entity.service.mjs +83 -0
  54. package/esm2020/classes/entity.utilities.mjs +538 -0
  55. package/esm2020/components/confirm-dialog/confirm-dialog-data.builder.mjs +44 -0
  56. package/esm2020/components/confirm-dialog/confirm-dialog-data.mjs +1 -1
  57. package/esm2020/components/confirm-dialog/confirm-dialog.component.mjs +18 -22
  58. package/esm2020/components/get-validation-error-message.function.mjs +8 -3
  59. package/esm2020/components/input/add-array-item-dialog-data.builder.mjs +30 -0
  60. package/esm2020/components/input/add-array-item-dialog-data.mjs +2 -0
  61. package/esm2020/components/input/input.component.mjs +240 -36
  62. package/esm2020/components/input/input.module.mjs +23 -9
  63. package/esm2020/components/table/create-dialog/create-dialog-data.builder.mjs +32 -0
  64. package/esm2020/components/table/create-dialog/create-entity-dialog-data.builder.mjs +26 -0
  65. package/esm2020/components/table/create-dialog/create-entity-dialog-data.mjs +1 -1
  66. package/esm2020/components/table/create-dialog/create-entity-dialog.component.mjs +31 -31
  67. package/esm2020/components/table/create-dialog/create-entity-dialog.module.mjs +20 -4
  68. package/esm2020/components/table/edit-dialog/edit-dialog-data.builder.mjs +41 -0
  69. package/esm2020/components/table/edit-dialog/edit-entity-dialog-data.mjs +1 -1
  70. package/esm2020/components/table/edit-dialog/edit-entity-dialog.builder.mjs +27 -0
  71. package/esm2020/components/table/edit-dialog/edit-entity-dialog.component.mjs +45 -49
  72. package/esm2020/components/table/table-data.builder.mjs +105 -0
  73. package/esm2020/components/table/table-data.mjs +1 -1
  74. package/esm2020/components/table/table.component.mjs +91 -83
  75. package/esm2020/components/table/table.module.mjs +12 -4
  76. package/esm2020/decorators/array/array-decorator-internal.data.mjs +51 -0
  77. package/esm2020/decorators/array/array-decorator.data.mjs +7 -0
  78. package/esm2020/decorators/array/array.decorator.mjs +24 -0
  79. package/esm2020/decorators/base/base-property.decorator.mjs +6 -5
  80. package/esm2020/decorators/base/decorator-types.enum.mjs +4 -1
  81. package/esm2020/decorators/base/dropdown-value.interface.mjs +2 -0
  82. package/esm2020/decorators/base/property-decorator-internal.data.mjs +38 -0
  83. package/esm2020/decorators/base/property-decorator.data.mjs +6 -0
  84. package/esm2020/decorators/boolean/boolean-decorator-internal.data.mjs +33 -0
  85. package/esm2020/decorators/boolean/boolean-decorator.data.mjs +7 -0
  86. package/esm2020/decorators/boolean/boolean.decorator.mjs +21 -0
  87. package/esm2020/decorators/date/date-decorator-internal.data.mjs +48 -0
  88. package/esm2020/decorators/date/date-decorator.data.mjs +7 -0
  89. package/esm2020/decorators/number/number-decorator-internal.data.mjs +23 -0
  90. package/esm2020/decorators/number/number-decorator.data.mjs +7 -0
  91. package/esm2020/decorators/number/number.decorator.mjs +18 -0
  92. package/esm2020/decorators/object/object-decorator-internal.data.mjs +12 -0
  93. package/esm2020/decorators/object/object-decorator.data.mjs +7 -0
  94. package/esm2020/decorators/object/object.decorator.mjs +13 -0
  95. package/esm2020/decorators/string/string-decorator-internal.data.mjs +48 -0
  96. package/esm2020/decorators/string/string-decorator.data.mjs +7 -0
  97. package/esm2020/decorators/string/string.decorator.mjs +24 -0
  98. package/esm2020/public-api.mjs +20 -13
  99. package/fesm2015/ngx-material-entity.mjs +1664 -944
  100. package/fesm2015/ngx-material-entity.mjs.map +1 -1
  101. package/fesm2020/ngx-material-entity.mjs +1667 -941
  102. package/fesm2020/ngx-material-entity.mjs.map +1 -1
  103. package/package.json +7 -1
  104. package/public-api.d.ts +21 -10
  105. package/classes/entity-model.class.d.ts +0 -9
  106. package/classes/entity-utilities.class.d.ts +0 -95
  107. package/components/input/array-table/add-array-item-dialog/add-array-item-dialog.component.d.ts +0 -35
  108. package/components/input/array-table/add-array-item-dialog/add-array-item-dialog.module.d.ts +0 -12
  109. package/components/input/array-table/array-table.component.d.ts +0 -34
  110. package/components/input/array-table/array-table.module.d.ts +0 -19
  111. package/components/input/internal-input/internal-input.component.d.ts +0 -57
  112. package/components/input/internal-input/internal-input.module.d.ts +0 -16
  113. package/decorators/array.decorator.d.ts +0 -125
  114. package/decorators/base/property-decorator-config.interface.d.ts +0 -50
  115. package/decorators/boolean.decorator.d.ts +0 -42
  116. package/decorators/number.decorator.d.ts +0 -40
  117. package/decorators/object.decorator.d.ts +0 -27
  118. package/decorators/string.decorator.d.ts +0 -76
  119. package/esm2020/classes/entity-model.class.mjs +0 -19
  120. package/esm2020/classes/entity-service.class.mjs +0 -70
  121. package/esm2020/classes/entity-utilities.class.mjs +0 -296
  122. package/esm2020/components/input/array-table/add-array-item-dialog/add-array-item-dialog.component.mjs +0 -43
  123. package/esm2020/components/input/array-table/add-array-item-dialog/add-array-item-dialog.module.mjs +0 -22
  124. package/esm2020/components/input/array-table/array-table.component.mjs +0 -116
  125. package/esm2020/components/input/array-table/array-table.module.mjs +0 -66
  126. package/esm2020/components/input/internal-input/internal-input.component.mjs +0 -73
  127. package/esm2020/components/input/internal-input/internal-input.module.mjs +0 -54
  128. package/esm2020/decorators/array.decorator.mjs +0 -70
  129. package/esm2020/decorators/base/property-decorator-config.interface.mjs +0 -31
  130. package/esm2020/decorators/boolean.decorator.mjs +0 -44
  131. package/esm2020/decorators/number.decorator.mjs +0 -36
  132. package/esm2020/decorators/object.decorator.mjs +0 -23
  133. package/esm2020/decorators/string.decorator.mjs +0 -61
@@ -0,0 +1,538 @@
1
+ import { cloneDeep, isEqual } from 'lodash';
2
+ import { DecoratorTypes } from '../decorators/base/decorator-types.enum';
3
+ import { DateUtilities } from './date.utilities';
4
+ /**
5
+ * Contains HelperMethods around handling Entities and their property-metadata.
6
+ */
7
+ export class EntityUtilities {
8
+ /**
9
+ * Gets the properties to omit when updating the entity.
10
+ *
11
+ * @param entity - The entity to get the properties which should be left out for updating from.
12
+ * @returns The properties which should be left out for updating an Entity.
13
+ */
14
+ static getOmitForUpdate(entity) {
15
+ const res = [];
16
+ for (const key of EntityUtilities.keysOf(entity)) {
17
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
18
+ if (metadata.omitForUpdate) {
19
+ res.push(key);
20
+ }
21
+ }
22
+ return res;
23
+ }
24
+ /**
25
+ * Gets the properties to omit when creating new entities.
26
+ *
27
+ * @param entity - The entity to get the properties which should be left out for creating from.
28
+ * @returns The properties which should be left out for creating a new Entity.
29
+ */
30
+ static getOmitForCreate(entity) {
31
+ const res = [];
32
+ for (const key of EntityUtilities.keysOf(entity)) {
33
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
34
+ if (metadata.omitForCreate) {
35
+ res.push(key);
36
+ }
37
+ }
38
+ return res;
39
+ }
40
+ /**
41
+ * Gets the metadata included in an property.
42
+ *
43
+ * @param entity - The entity with the property to get the metadata from.
44
+ * @param propertyKey - The property on the given Entity to get the metadata from.
45
+ * @param type - For secure Typing, defines the returned PropertyConfig.
46
+ * @returns The metadata of the property.
47
+ * @throws When no metadata can be found for the given property.
48
+ */
49
+ static getPropertyMetadata(entity, propertyKey,
50
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
51
+ type) {
52
+ try {
53
+ const metadata = Reflect.getMetadata('metadata', entity, propertyKey);
54
+ if (!metadata) {
55
+ throw new Error(`Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
56
+ }
57
+ return metadata;
58
+ }
59
+ catch (error) {
60
+ throw new Error(`Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
61
+ }
62
+ }
63
+ /**
64
+ * Gets the type of the property-metadata.
65
+ *
66
+ * @param entity - The entity with the property to get the type from.
67
+ * @param propertyKey - The property on the given Entity to get the type from.
68
+ * @returns The type of the metadata.
69
+ * @throws Will throw an error if no metadata can be found for the given property.
70
+ */
71
+ static getPropertyType(entity, propertyKey) {
72
+ try {
73
+ const propertyType = Reflect.getMetadata('type', entity, propertyKey);
74
+ if (!propertyType) {
75
+ throw new Error(`Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
76
+ }
77
+ return propertyType;
78
+ }
79
+ catch (error) {
80
+ throw new Error(`Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`);
81
+ }
82
+ }
83
+ /**
84
+ * Sets all property values based on a given entity data-object.
85
+ *
86
+ * @param target - The target object that needs to be constructed (if called inside an Entity constructor its usually this).
87
+ * @param entity - The data object to get the property values from.
88
+ * @alias new
89
+ * @alias build
90
+ * @alias construct
91
+ */
92
+ static new(target, entity) {
93
+ for (const key in target) {
94
+ const type = EntityUtilities.getPropertyType(target, key);
95
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
96
+ let value = entity ? Reflect.get(entity, key) : undefined;
97
+ switch (type) {
98
+ case DecoratorTypes.OBJECT:
99
+ const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);
100
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-argument
101
+ value = new objectMetadata.EntityClass(value);
102
+ break;
103
+ case DecoratorTypes.ARRAY:
104
+ const inputArray = value;
105
+ const resArray = [];
106
+ if (inputArray) {
107
+ const arrayMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);
108
+ for (const item of inputArray) {
109
+ const itemWithMetadata = new arrayMetadata.EntityClass(item);
110
+ resArray.push(itemWithMetadata);
111
+ }
112
+ }
113
+ value = resArray;
114
+ break;
115
+ default:
116
+ break;
117
+ }
118
+ Reflect.set(target, key, value);
119
+ }
120
+ }
121
+ /**
122
+ * Checks if the values on an entity are valid.
123
+ * Also checks all the validators given by the metadata ("required", "maxLength" etc.).
124
+ *
125
+ * @param entity - The entity to validate.
126
+ * @param omit - Whether to check for creating or editing validity.
127
+ * @returns Whether or not the entity is valid.
128
+ */
129
+ static isEntityValid(entity, omit) {
130
+ for (const key in entity) {
131
+ if (!EntityUtilities.isPropertyValid(entity, key, omit)) {
132
+ return false;
133
+ }
134
+ }
135
+ return true;
136
+ }
137
+ /**
138
+ * Checks if a single property value is valid.
139
+ *
140
+ * @param entity - The entity where the property is from.
141
+ * @param key - The name of the property.
142
+ * @param omit - Whether to check if the given entity is valid for creation or updating.
143
+ * @returns Whether or not the property value is valid.
144
+ * @throws Throws when it extracts an unknown metadata type.
145
+ */
146
+ static isPropertyValid(entity, key, omit) {
147
+ const type = EntityUtilities.getPropertyType(entity, key);
148
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key, type);
149
+ if (metadata.omitForCreate && omit === 'create') {
150
+ return true;
151
+ }
152
+ if (metadata.omitForUpdate && omit === 'update') {
153
+ return true;
154
+ }
155
+ if (metadata.required && !entity[key]) {
156
+ return false;
157
+ }
158
+ switch (type) {
159
+ case DecoratorTypes.BOOLEAN_DROPDOWN:
160
+ case DecoratorTypes.BOOLEAN_CHECKBOX:
161
+ case DecoratorTypes.BOOLEAN_TOGGLE:
162
+ return true;
163
+ case DecoratorTypes.STRING_DROPDOWN:
164
+ return true;
165
+ case DecoratorTypes.STRING:
166
+ case DecoratorTypes.STRING_AUTOCOMPLETE:
167
+ const entityString = entity[key];
168
+ const stringMetadata = metadata;
169
+ if (!this.isStringValid(entityString, stringMetadata)) {
170
+ return false;
171
+ }
172
+ break;
173
+ case DecoratorTypes.STRING_TEXTBOX:
174
+ const entityTextbox = entity[key];
175
+ const textboxMetadata = metadata;
176
+ if (!this.isTextboxValid(entityTextbox, textboxMetadata)) {
177
+ return false;
178
+ }
179
+ break;
180
+ case DecoratorTypes.NUMBER_DROPDOWN:
181
+ return true;
182
+ case DecoratorTypes.NUMBER:
183
+ const entityNumber = entity[key];
184
+ const numberMetadata = metadata;
185
+ if (!this.isNumberValid(entityNumber, numberMetadata)) {
186
+ return false;
187
+ }
188
+ break;
189
+ case DecoratorTypes.OBJECT:
190
+ const entityObject = entity[key];
191
+ for (const parameterKey in entityObject) {
192
+ if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {
193
+ return false;
194
+ }
195
+ }
196
+ break;
197
+ case DecoratorTypes.ARRAY_STRING_CHIPS:
198
+ case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:
199
+ case DecoratorTypes.ARRAY:
200
+ const entityArray = entity[key];
201
+ const arrayMetadata = metadata;
202
+ if (arrayMetadata.required && !entityArray.length) {
203
+ return false;
204
+ }
205
+ break;
206
+ case DecoratorTypes.DATE:
207
+ const entityDate = new Date(entity[key]);
208
+ const dateMetadata = metadata;
209
+ if (!this.isDateValid(entityDate, dateMetadata)) {
210
+ return false;
211
+ }
212
+ break;
213
+ case DecoratorTypes.DATE_RANGE:
214
+ const entityDateRange = cloneDeep(entity[key]);
215
+ const dateRangeMetadata = metadata;
216
+ if (!this.isDateRangeValid(entityDateRange, dateRangeMetadata)) {
217
+ return false;
218
+ }
219
+ break;
220
+ case DecoratorTypes.DATE_TIME:
221
+ const entityDateTime = new Date(entity[key]);
222
+ const dateTimeMetadata = metadata;
223
+ if (!this.isDateTimeValid(entityDateTime, dateTimeMetadata)) {
224
+ return false;
225
+ }
226
+ break;
227
+ default:
228
+ throw new Error(`Could not validate the input because the DecoratorType ${type} is not known`);
229
+ }
230
+ return true;
231
+ }
232
+ static isStringValid(value, metadata) {
233
+ if (metadata.maxLength && value.length > metadata.maxLength) {
234
+ return false;
235
+ }
236
+ if (metadata.minLength && value.length < metadata.minLength) {
237
+ return false;
238
+ }
239
+ if (metadata.regex && !value.match(metadata.regex)) {
240
+ return false;
241
+ }
242
+ return true;
243
+ }
244
+ static isTextboxValid(value, metadata) {
245
+ if (metadata.maxLength && value.length > metadata.maxLength) {
246
+ return false;
247
+ }
248
+ if (metadata.minLength && value.length < metadata.minLength) {
249
+ return false;
250
+ }
251
+ return true;
252
+ }
253
+ static isNumberValid(value, metadata) {
254
+ if (metadata.max && value > metadata.max) {
255
+ return false;
256
+ }
257
+ if (metadata.min && value < metadata.min) {
258
+ return false;
259
+ }
260
+ return true;
261
+ }
262
+ static isDateValid(value, metadata) {
263
+ if (metadata.min && value.getTime() < metadata.min(value).getTime()) {
264
+ return false;
265
+ }
266
+ if (metadata.max && value.getTime() > metadata.max(value).getTime()) {
267
+ return false;
268
+ }
269
+ if (metadata.filter && !metadata.filter(value)) {
270
+ return false;
271
+ }
272
+ return true;
273
+ }
274
+ static isDateRangeValid(value, metadata) {
275
+ if (metadata.required && (!value.start || !value.end)) {
276
+ return false;
277
+ }
278
+ value.start = new Date(value.start);
279
+ value.end = new Date(value.end);
280
+ if (metadata.minStart && value.start.getTime() < metadata.minStart(value.start).getTime()) {
281
+ return false;
282
+ }
283
+ if (metadata.maxStart && value.start.getTime() > metadata.maxStart(value.start).getTime()) {
284
+ return false;
285
+ }
286
+ if (metadata.minEnd && value.end.getTime() < metadata.minEnd(value.end).getTime()) {
287
+ return false;
288
+ }
289
+ if (metadata.maxEnd && value.end.getTime() > metadata.maxEnd(value.end).getTime()) {
290
+ return false;
291
+ }
292
+ if (metadata.filter) {
293
+ if (!metadata.filter(value.start)) {
294
+ return false;
295
+ }
296
+ if (!metadata.filter(value.end)) {
297
+ return false;
298
+ }
299
+ if (value.values) {
300
+ for (const date of value.values) {
301
+ if (!metadata.filter(date)) {
302
+ return false;
303
+ }
304
+ }
305
+ }
306
+ }
307
+ return true;
308
+ }
309
+ static isDateTimeValid(value, metadata) {
310
+ if (metadata.minDate && value.getTime() < metadata.minDate(value).getTime()) {
311
+ return false;
312
+ }
313
+ if (metadata.maxDate && value.getTime() > metadata.maxDate(value).getTime()) {
314
+ return false;
315
+ }
316
+ if (metadata.filterDate && !metadata.filterDate(value)) {
317
+ return false;
318
+ }
319
+ const time = {
320
+ hours: value.getHours(),
321
+ minutes: value.getMinutes()
322
+ };
323
+ if (metadata.minTime) {
324
+ const minTime = metadata.minTime(value);
325
+ if (!(time.hours > minTime.hours
326
+ || (time.hours === minTime.hours
327
+ && time.minutes >= minTime.minutes))) {
328
+ return false;
329
+ }
330
+ }
331
+ if (metadata.maxTime) {
332
+ const maxTime = metadata.maxTime(value);
333
+ if (!(time.hours < maxTime.hours
334
+ || (time.hours === maxTime.hours
335
+ && time.minutes <= maxTime.minutes))) {
336
+ return false;
337
+ }
338
+ }
339
+ if (metadata.filterTime) {
340
+ if (!metadata.filterTime(time)) {
341
+ return false;
342
+ }
343
+ }
344
+ return true;
345
+ }
346
+ /**
347
+ * Checks if an entity is "dirty" (if its values have changed).
348
+ *
349
+ * @param entity - The entity after all changes.
350
+ * @param entityPriorChanges - The entity before the changes.
351
+ * @returns Whether or not the entity is dirty.
352
+ */
353
+ static dirty(entity, entityPriorChanges) {
354
+ if (!entityPriorChanges) {
355
+ return false;
356
+ }
357
+ else {
358
+ const differences = this.differencesForDirty(entity, entityPriorChanges);
359
+ return differences.length ? true : false;
360
+ }
361
+ }
362
+ static differencesForDirty(entity, entityPriorChanges) {
363
+ const res = [];
364
+ for (const key in entity) {
365
+ if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {
366
+ res.push({
367
+ key: key,
368
+ before: entityPriorChanges[key],
369
+ after: entity[key]
370
+ });
371
+ }
372
+ }
373
+ return res;
374
+ }
375
+ /**
376
+ * Compares two Entities and returns their difference in an object.
377
+ *
378
+ * @param entity - The first entity to compare.
379
+ * @param entityPriorChanges - The second entity to compare.
380
+ * @returns The difference between the two Entities in form of a Partial.
381
+ */
382
+ static difference(entity, entityPriorChanges) {
383
+ const res = {};
384
+ for (const key in entity) {
385
+ if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {
386
+ res[key] = entity[key];
387
+ }
388
+ }
389
+ return res;
390
+ }
391
+ static isEqual(value, valuePriorChanges, metadata) {
392
+ if (this.isDateRange(value) && this.isDateRange(valuePriorChanges)) {
393
+ const dateRange = cloneDeep(value);
394
+ dateRange.start = new Date(value.start);
395
+ dateRange.end = new Date(value.end);
396
+ dateRange.values = DateUtilities.getDatesBetween(dateRange.start, dateRange.end, metadata);
397
+ const dateRangePriorChanges = cloneDeep(valuePriorChanges);
398
+ dateRangePriorChanges.start = new Date(valuePriorChanges.start);
399
+ dateRangePriorChanges.end = new Date(valuePriorChanges.end);
400
+ dateRangePriorChanges.values = DateUtilities.getDatesBetween(dateRangePriorChanges.start, dateRangePriorChanges.end, metadata);
401
+ return isEqual(dateRange, dateRangePriorChanges);
402
+ }
403
+ if (metadata.displayStyle === 'date') {
404
+ const date = new Date(DateUtilities.asDate(value));
405
+ const datePriorChanges = new Date(DateUtilities.asDate(valuePriorChanges));
406
+ date.setHours(0, 0, 0, 0);
407
+ datePriorChanges.setHours(0, 0, 0, 0);
408
+ return isEqual(date, datePriorChanges);
409
+ }
410
+ return isEqual(value, valuePriorChanges);
411
+ }
412
+ static isDateRange(value) {
413
+ const dateRange = value;
414
+ if (dateRange.start && dateRange.end) {
415
+ try {
416
+ new Date(dateRange.start);
417
+ new Date(dateRange.end);
418
+ return true;
419
+ }
420
+ catch (error) { }
421
+ ;
422
+ }
423
+ return false;
424
+ }
425
+ /**
426
+ * Compare function for sorting entity keys by their order value.
427
+ *
428
+ * @param a - First key of entity.
429
+ * @param b - Second key of entity.
430
+ * @param entity - Current entity (used to get metadata of entity keys).
431
+ * @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'.
432
+ */
433
+ static compareOrder(a, b, entity) {
434
+ const metadataA = EntityUtilities.getPropertyMetadata(entity, a);
435
+ const metadataB = EntityUtilities.getPropertyMetadata(entity, b);
436
+ if (metadataA.position.order === -1) {
437
+ if (metadataB.position.order === -1) {
438
+ return 0;
439
+ }
440
+ return 1;
441
+ }
442
+ else if (metadataB.position.order === -1) {
443
+ return -1;
444
+ }
445
+ return ((metadataA.position.order) - (metadataB.position.order));
446
+ }
447
+ /**
448
+ * Gets the bootstrap column values for "lg", "md", "sm".
449
+ *
450
+ * @param entity - Entity to get the bootstrap column values of the key.
451
+ * @param key - Key of the property to get bootstrap column values from.
452
+ * @param type - Defines for which screen size the column values should be returned.
453
+ * @returns Bootstrap column value.
454
+ */
455
+ static getWidth(entity, key, type) {
456
+ const metadata = EntityUtilities.getPropertyMetadata(entity, key);
457
+ switch (type) {
458
+ case 'lg':
459
+ return metadata.defaultWidths[0];
460
+ case 'md':
461
+ return metadata.defaultWidths[1];
462
+ case 'sm':
463
+ return metadata.defaultWidths[2];
464
+ }
465
+ }
466
+ /**
467
+ * Resets all changes on an entity.
468
+ *
469
+ * @param entity - The entity to reset.
470
+ * @param entityPriorChanges - The entity before any changes.
471
+ */
472
+ static resetChangesOnEntity(entity, entityPriorChanges) {
473
+ for (const key in entityPriorChanges) {
474
+ Reflect.set(entity, key, Reflect.get(entityPriorChanges, key));
475
+ }
476
+ }
477
+ /**
478
+ * Gets the rows that are used to display the given entity.
479
+ *
480
+ * @param entity - The entity to get the rows from.
481
+ * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.
482
+ * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.
483
+ * @returns The sorted Rows containing the row number and the keys to display in that row.
484
+ */
485
+ static getEntityRows(entity, hideOmitForCreate = false, hideOmitForEdit = false) {
486
+ const res = [];
487
+ const keys = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit);
488
+ const numberOfRows = this.getNumberOfRows(keys, entity);
489
+ for (let i = 1; i <= numberOfRows; i++) {
490
+ const row = {
491
+ row: i,
492
+ keys: this.getKeysForRow(keys, entity, i)
493
+ };
494
+ res.push(row);
495
+ }
496
+ const lastRow = {
497
+ row: numberOfRows + 1,
498
+ keys: this.getKeysForRow(keys, entity, -1)
499
+ };
500
+ res.push(lastRow);
501
+ return res;
502
+ }
503
+ static getKeysForRow(keys, entity, i) {
504
+ return keys
505
+ .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === i)
506
+ .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));
507
+ }
508
+ static getNumberOfRows(keys, entity) {
509
+ return keys
510
+ .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)
511
+ .sort((a, b) => (a > b ? -1 : 1))[0];
512
+ }
513
+ /**
514
+ * Gets the keys of the provided entity correctly typed.
515
+ *
516
+ * @param entity - The entity to get the keys of.
517
+ * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.
518
+ * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.
519
+ * @returns An array of keys of the entity.
520
+ */
521
+ static keysOf(entity, hideOmitForCreate = false, hideOmitForEdit = false) {
522
+ let keys = Reflect.ownKeys(entity);
523
+ if (hideOmitForCreate) {
524
+ const omitForCreateKeys = EntityUtilities.getOmitForCreate(entity);
525
+ keys = keys.filter(k => !omitForCreateKeys.includes(k));
526
+ }
527
+ if (hideOmitForEdit) {
528
+ const omitForUpdateKeys = EntityUtilities.getOmitForUpdate(entity);
529
+ keys = keys.filter(k => !omitForUpdateKeys.includes(k));
530
+ }
531
+ return keys;
532
+ }
533
+ }
534
+ // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc
535
+ EntityUtilities.construct = EntityUtilities.new;
536
+ // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc
537
+ EntityUtilities.build = EntityUtilities.new;
538
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.utilities.js","sourceRoot":"","sources":["../../../../projects/ngx-material-entity/src/classes/entity.utilities.ts"],"names":[],"mappings":"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","sourcesContent":["import { cloneDeep, isEqual } from 'lodash';\nimport { DecoratorType, DecoratorTypes } from '../decorators/base/decorator-types.enum';\nimport { PropertyDecoratorConfigInternal } from '../decorators/base/property-decorator-internal.data';\nimport { EntityArrayDecoratorConfigInternal } from '../decorators/array/array-decorator-internal.data';\nimport { DefaultStringDecoratorConfigInternal, TextboxStringDecoratorConfigInternal } from '../decorators/string/string-decorator-internal.data';\nimport { DefaultNumberDecoratorConfigInternal } from '../decorators/number/number-decorator-internal.data';\nimport { DateRangeDateDecoratorConfigInternal, DateTimeDateDecoratorConfigInternal, DefaultDateDecoratorConfigInternal } from '../decorators/date/date-decorator-internal.data';\nimport { DateRange } from '../decorators/date/date-decorator.data';\nimport { Time } from '@angular/common';\nimport { DateUtilities } from './date.utilities';\n\n/**\n * Shows information about differences between two entities.\n */\ninterface Difference<EntityType extends object> {\n    /**\n     * The key where the two entities have different values.\n     */\n    key: keyof EntityType,\n    /**\n     * The value before any changes.\n     */\n    before: unknown,\n    /**\n     * The current value after changes.\n     */\n    after: unknown\n}\n\n/**\n * Contains HelperMethods around handling Entities and their property-metadata.\n */\nexport abstract class EntityUtilities {\n\n    /**\n     * Gets the properties to omit when updating the entity.\n     *\n     * @param entity - The entity to get the properties which should be left out for updating from.\n     * @returns The properties which should be left out for updating an Entity.\n     */\n    static getOmitForUpdate<EntityType extends object>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForUpdate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the properties to omit when creating new entities.\n     *\n     * @param entity - The entity to get the properties which should be left out for creating from.\n     * @returns The properties which should be left out for creating a new Entity.\n     */\n    static getOmitForCreate<EntityType extends object>(entity: EntityType): (keyof EntityType)[] {\n        const res: (keyof EntityType)[] = [];\n        for (const key of EntityUtilities.keysOf(entity)) {\n            const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n            if (metadata.omitForCreate) {\n                res.push(key);\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Gets the metadata included in an property.\n     *\n     * @param entity - The entity with the property to get the metadata from.\n     * @param propertyKey - The property on the given Entity to get the metadata from.\n     * @param type - For secure Typing, defines the returned PropertyConfig.\n     * @returns The metadata of the property.\n     * @throws When no metadata can be found for the given property.\n     */\n    static getPropertyMetadata<EntityType extends object, T extends DecoratorTypes>(\n        entity: EntityType,\n        propertyKey: keyof EntityType,\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n        type?: T\n    ): DecoratorType<T> {\n        try {\n            const metadata = Reflect.getMetadata('metadata', entity, propertyKey as string) as DecoratorType<T>;\n            if (!metadata) {\n                throw new Error(\n                    `Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n                );\n            }\n            return metadata;\n        }\n        catch (error) {\n            throw new Error(\n                `Could not find metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n    }\n\n    /**\n     * Gets the type of the property-metadata.\n     *\n     * @param entity - The entity with the property to get the type from.\n     * @param propertyKey - The property on the given Entity to get the type from.\n     * @returns The type of the metadata.\n     * @throws Will throw an error if no metadata can be found for the given property.\n     */\n    static getPropertyType<EntityType extends object>(\n        entity: EntityType, propertyKey: keyof EntityType\n    ): DecoratorTypes {\n        try {\n            const propertyType = Reflect.getMetadata('type', entity, propertyKey as string) as DecoratorTypes;\n            if (!propertyType) {\n                throw new Error(\n                    `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n                );\n            }\n            return propertyType;\n        }\n        catch (error) {\n            throw new Error(\n                `Could not find type metadata for property ${String(propertyKey)} on the entity ${JSON.stringify(entity)}`\n            );\n        }\n    }\n\n    /**\n     * Sets all property values based on a given entity data-object.\n     *\n     * @param target - The target object that needs to be constructed (if called inside an Entity constructor its usually this).\n     * @param entity - The data object to get the property values from.\n     * @alias new\n     * @alias build\n     * @alias construct\n     */\n    static new<EntityType extends object>(target: EntityType, entity?: EntityType): void {\n        for (const key in target) {\n            const type = EntityUtilities.getPropertyType(target, key);\n            // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n            let value = entity ? Reflect.get(entity, key) : undefined;\n            switch (type) {\n                case DecoratorTypes.OBJECT:\n                    const objectMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.OBJECT);\n                    // eslint-disable-next-line @typescript-eslint/no-unsafe-argument\n                    value = new objectMetadata.EntityClass(value);\n                    break;\n                case DecoratorTypes.ARRAY:\n                    const inputArray: EntityType[] = value as EntityType[];\n                    const resArray: EntityType[] = [];\n                    if (inputArray) {\n                        const arrayMetadata = EntityUtilities.getPropertyMetadata(target, key, DecoratorTypes.ARRAY);\n                        for (const item of inputArray) {\n                            const itemWithMetadata: EntityType = new arrayMetadata.EntityClass(item) as EntityType;\n                            resArray.push(itemWithMetadata);\n                        }\n                    }\n                    value = resArray;\n                    break;\n                default:\n                    break;\n            }\n            Reflect.set(target, key, value);\n        }\n    }\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static construct = EntityUtilities.new;\n    // eslint-disable-next-line @typescript-eslint/member-ordering, jsdoc/require-jsdoc\n    static build = EntityUtilities.new;\n\n    /**\n     * Checks if the values on an entity are valid.\n     * Also checks all the validators given by the metadata (\"required\", \"maxLength\" etc.).\n     *\n     * @param entity - The entity to validate.\n     * @param omit - Whether to check for creating or editing validity.\n     * @returns Whether or not the entity is valid.\n     */\n    static isEntityValid<EntityType extends object>(entity: EntityType, omit: 'create' | 'update'): boolean {\n        for (const key in entity) {\n            if (!EntityUtilities.isPropertyValid(entity, key, omit)) {\n                return false;\n            }\n        }\n        return true;\n    }\n    /**\n     * Checks if a single property value is valid.\n     *\n     * @param entity - The entity where the property is from.\n     * @param key - The name of the property.\n     * @param omit - Whether to check if the given entity is valid for creation or updating.\n     * @returns Whether or not the property value is valid.\n     * @throws Throws when it extracts an unknown metadata type.\n     */\n    private static isPropertyValid<EntityType extends object>(\n        entity: EntityType,\n        key: keyof EntityType,\n        omit: 'create' | 'update'\n    ): boolean {\n        const type = EntityUtilities.getPropertyType(entity, key);\n        const metadata: PropertyDecoratorConfigInternal = EntityUtilities.getPropertyMetadata(entity, key, type);\n\n        if (metadata.omitForCreate && omit === 'create') {\n            return true;\n        }\n        if (metadata.omitForUpdate && omit === 'update') {\n            return true;\n        }\n        if (metadata.required && !entity[key]) {\n            return false;\n        }\n        switch (type) {\n            case DecoratorTypes.BOOLEAN_DROPDOWN:\n            case DecoratorTypes.BOOLEAN_CHECKBOX:\n            case DecoratorTypes.BOOLEAN_TOGGLE:\n                return true;\n            case DecoratorTypes.STRING_DROPDOWN:\n                return true;\n            case DecoratorTypes.STRING:\n            case DecoratorTypes.STRING_AUTOCOMPLETE:\n                const entityString = entity[key] as unknown as string;\n                const stringMetadata = metadata as DefaultStringDecoratorConfigInternal;\n                if (!this.isStringValid(entityString, stringMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.STRING_TEXTBOX:\n                const entityTextbox = entity[key] as unknown as string;\n                const textboxMetadata = metadata as TextboxStringDecoratorConfigInternal;\n                if (!this.isTextboxValid(entityTextbox, textboxMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.NUMBER_DROPDOWN:\n                return true;\n            case DecoratorTypes.NUMBER:\n                const entityNumber = entity[key] as unknown as number;\n                const numberMetadata = metadata as DefaultNumberDecoratorConfigInternal;\n                if (!this.isNumberValid(entityNumber, numberMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.OBJECT:\n                const entityObject = entity[key] as unknown as EntityType;\n                for (const parameterKey in entityObject) {\n                    if (!EntityUtilities.isPropertyValid(entityObject, parameterKey, omit)) {\n                        return false;\n                    }\n                }\n                break;\n            case DecoratorTypes.ARRAY_STRING_CHIPS:\n            case DecoratorTypes.ARRAY_STRING_AUTOCOMPLETE_CHIPS:\n            case DecoratorTypes.ARRAY:\n                const entityArray = entity[key] as unknown as [];\n                const arrayMetadata = metadata as EntityArrayDecoratorConfigInternal<EntityType>;\n                if (arrayMetadata.required && !entityArray.length) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE:\n                const entityDate: Date = new Date(entity[key] as unknown as Date);\n                const dateMetadata = metadata as DefaultDateDecoratorConfigInternal;\n                if (!this.isDateValid(entityDate, dateMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_RANGE:\n                const entityDateRange: DateRange = cloneDeep(entity[key] as unknown as DateRange);\n                const dateRangeMetadata = metadata as DateRangeDateDecoratorConfigInternal;\n                if (!this.isDateRangeValid(entityDateRange, dateRangeMetadata)) {\n                    return false;\n                }\n                break;\n            case DecoratorTypes.DATE_TIME:\n                const entityDateTime: Date = new Date(entity[key] as unknown as Date);\n                const dateTimeMetadata = metadata as DateTimeDateDecoratorConfigInternal;\n                if (!this.isDateTimeValid(entityDateTime, dateTimeMetadata)) {\n                    return false;\n                }\n                break;\n            default:\n                throw new Error(`Could not validate the input because the DecoratorType ${type} is not known`);\n        }\n        return true;\n    }\n\n    private static isStringValid(value: string, metadata: DefaultStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        if (metadata.regex && !value.match(metadata.regex)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isTextboxValid(value: string, metadata: TextboxStringDecoratorConfigInternal): boolean {\n        if (metadata.maxLength && value.length > metadata.maxLength) {\n            return false;\n        }\n        if (metadata.minLength && value.length < metadata.minLength) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isNumberValid(value: number, metadata: DefaultNumberDecoratorConfigInternal): boolean {\n        if (metadata.max && value > metadata.max) {\n            return false;\n        }\n        if (metadata.min && value < metadata.min) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateValid(value: Date, metadata: DefaultDateDecoratorConfigInternal): boolean {\n        if (metadata.min && value.getTime() < metadata.min(value).getTime()) {\n            return false;\n        }\n        if (metadata.max && value.getTime() > metadata.max(value).getTime()) {\n            return false;\n        }\n        if (metadata.filter && !metadata.filter(value)) {\n            return false;\n        }\n        return true;\n    }\n\n    private static isDateRangeValid(value: DateRange, metadata: DateRangeDateDecoratorConfigInternal): boolean {\n        if (metadata.required && (!value.start || !value.end)) {\n            return false;\n        }\n        value.start = new Date(value.start);\n        value.end = new Date(value.end);\n        if (metadata.minStart && value.start.getTime() < metadata.minStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.maxStart && value.start.getTime() > metadata.maxStart(value.start).getTime()) {\n            return false;\n        }\n        if (metadata.minEnd && value.end.getTime() < metadata.minEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.maxEnd && value.end.getTime() > metadata.maxEnd(value.end).getTime()) {\n            return false;\n        }\n        if (metadata.filter) {\n            if (!metadata.filter(value.start)) {\n                return false;\n            }\n            if (!metadata.filter(value.end)) {\n                return false;\n            }\n            if (value.values) {\n                for (const date of value.values) {\n                    if (!metadata.filter(date)) {\n                        return false;\n                    }\n                }\n            }\n        }\n        return true;\n    }\n\n    private static isDateTimeValid(value: Date, metadata: DateTimeDateDecoratorConfigInternal): boolean {\n        if (metadata.minDate && value.getTime() < metadata.minDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.maxDate && value.getTime() > metadata.maxDate(value).getTime()) {\n            return false;\n        }\n        if (metadata.filterDate && !metadata.filterDate(value)) {\n            return false;\n        }\n        const time: Time = {\n            hours: value.getHours(),\n            minutes: value.getMinutes()\n        }\n        if (metadata.minTime) {\n            const minTime: Time = metadata.minTime(value);\n            if (\n                !(\n                    time.hours > minTime.hours\n                    || (\n                        time.hours === minTime.hours\n                        && time.minutes >= minTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.maxTime) {\n            const maxTime: Time = metadata.maxTime(value);\n            if (\n                !(\n                    time.hours < maxTime.hours\n                    || (\n                        time.hours === maxTime.hours\n                        && time.minutes <= maxTime.minutes\n                    )\n                )\n            ) {\n                return false;\n            }\n        }\n        if (metadata.filterTime) {\n            if (!metadata.filterTime(time)) {\n                return false;\n            }\n        }\n        return true;\n    }\n\n    /**\n     * Checks if an entity is \"dirty\" (if its values have changed).\n     *\n     * @param entity - The entity after all changes.\n     * @param entityPriorChanges - The entity before the changes.\n     * @returns Whether or not the entity is dirty.\n     */\n    static dirty<EntityType extends object>(entity: EntityType, entityPriorChanges: EntityType): boolean {\n        if (!entityPriorChanges) {\n            return false;\n        }\n        else {\n            const differences = this.differencesForDirty(entity, entityPriorChanges);\n            return differences.length ? true : false;\n        }\n    }\n\n    private static differencesForDirty<EntityType extends object>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Difference<EntityType>[] {\n        const res: Difference<EntityType>[] = [];\n        for (const key in entity) {\n            if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {\n                res.push({\n                    key: key,\n                    before: entityPriorChanges[key],\n                    after: entity[key]\n                });\n            }\n        }\n        return res;\n    }\n\n    /**\n     * Compares two Entities and returns their difference in an object.\n     *\n     * @param entity - The first entity to compare.\n     * @param entityPriorChanges - The second entity to compare.\n     * @returns The difference between the two Entities in form of a Partial.\n     */\n    static difference<EntityType extends object>(\n        entity: EntityType,\n        entityPriorChanges: EntityType\n    ): Partial<EntityType> {\n        const res: Partial<EntityType> = {};\n        for (const key in entity) {\n            if (!this.isEqual(entity[key], entityPriorChanges[key], EntityUtilities.getPropertyMetadata(entity, key))) {\n                res[key] = entity[key];\n            }\n        }\n        return res;\n    }\n\n    private static isEqual(value: unknown, valuePriorChanges: unknown, metadata: PropertyDecoratorConfigInternal): boolean {\n        if (this.isDateRange(value) && this.isDateRange(valuePriorChanges)) {\n            const dateRange = cloneDeep(value);\n            dateRange.start = new Date(value.start);\n            dateRange.end = new Date(value.end);\n            dateRange.values = DateUtilities.getDatesBetween(\n                dateRange.start,\n                dateRange.end,\n                metadata as DateRangeDateDecoratorConfigInternal\n            );\n\n            const dateRangePriorChanges = cloneDeep(valuePriorChanges);\n            dateRangePriorChanges.start = new Date(valuePriorChanges.start);\n            dateRangePriorChanges.end = new Date(valuePriorChanges.end);\n            dateRangePriorChanges.values = DateUtilities.getDatesBetween(\n                dateRangePriorChanges.start,\n                dateRangePriorChanges.end,\n                metadata as DateRangeDateDecoratorConfigInternal\n            );\n            return isEqual(dateRange, dateRangePriorChanges);\n        }\n        if ((metadata as DefaultDateDecoratorConfigInternal).displayStyle === 'date') {\n            const date = new Date(DateUtilities.asDate(value));\n            const datePriorChanges = new Date(DateUtilities.asDate(valuePriorChanges));\n            date.setHours(0, 0, 0, 0);\n            datePriorChanges.setHours(0, 0, 0, 0);\n            return isEqual(date, datePriorChanges);\n        }\n        return isEqual(value, valuePriorChanges);\n    }\n\n    private static isDateRange(value: unknown): value is DateRange {\n        const dateRange = value as DateRange;\n        if (dateRange.start && dateRange.end) {\n            try {\n                new Date(dateRange.start);\n                new Date(dateRange.end)\n                return true;\n            }\n            catch (error) {};\n        }\n        return false;\n    }\n\n    /**\n     * Compare function for sorting entity keys by their order value.\n     *\n     * @param a - First key of entity.\n     * @param b - Second key of entity.\n     * @param entity - Current entity (used to get metadata of entity keys).\n     * @returns 0 if both values have the same order, a negative value if 'a' comes before 'b', a positive value if 'a' comes behind 'b'.\n     */\n    static compareOrder<EntityType extends object>(a: keyof EntityType, b: keyof EntityType, entity: EntityType): number {\n        const metadataA = EntityUtilities.getPropertyMetadata(entity, a);\n        const metadataB = EntityUtilities.getPropertyMetadata(entity, b);\n\n        if (metadataA.position.order === -1) {\n            if (metadataB.position.order === -1) {\n                return 0;\n            }\n            return 1;\n        }\n        else if (metadataB.position.order === -1) {\n            return -1;\n        }\n\n        return ((metadataA.position.order ) - (metadataB.position.order ));\n    }\n\n    /**\n     * Gets the bootstrap column values for \"lg\", \"md\", \"sm\".\n     *\n     * @param entity - Entity to get the bootstrap column values of the key.\n     * @param key - Key of the property to get bootstrap column values from.\n     * @param type - Defines for which screen size the column values should be returned.\n     * @returns Bootstrap column value.\n     */\n    static getWidth<EntityType extends object>(entity: EntityType, key: keyof EntityType, type: 'lg' | 'md' | 'sm'): number {\n        const metadata = EntityUtilities.getPropertyMetadata(entity, key);\n        switch (type) {\n            case 'lg':\n                return metadata.defaultWidths[0];\n            case 'md':\n                return metadata.defaultWidths[1];\n            case 'sm':\n                return metadata.defaultWidths[2];\n        }\n    }\n\n    /**\n     * Resets all changes on an entity.\n     *\n     * @param entity - The entity to reset.\n     * @param entityPriorChanges - The entity before any changes.\n     */\n    static resetChangesOnEntity<EntityType extends object>(entity: EntityType, entityPriorChanges: EntityType): void {\n        for (const key in entityPriorChanges) {\n            Reflect.set(entity, key, Reflect.get(entityPriorChanges, key));\n        }\n    }\n\n    /**\n     * Gets the rows that are used to display the given entity.\n     *\n     * @param entity - The entity to get the rows from.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns The sorted Rows containing the row number and the keys to display in that row.\n     */\n    static getEntityRows<EntityType extends object>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): EntityRow<EntityType>[] {\n        const res: EntityRow<EntityType>[] = [];\n\n        const keys: (keyof EntityType)[] = EntityUtilities.keysOf(entity, hideOmitForCreate, hideOmitForEdit);\n        const numberOfRows: number = this.getNumberOfRows<EntityType>(keys, entity);\n        for (let i = 1; i <= numberOfRows; i++) {\n            const row: EntityRow<EntityType> = {\n                row: i,\n                keys: this.getKeysForRow<EntityType>(keys, entity, i)\n            };\n            res.push(row);\n        }\n        const lastRow: EntityRow<EntityType> = {\n            row: numberOfRows + 1,\n            keys: this.getKeysForRow<EntityType>(keys, entity, -1)\n        };\n        res.push(lastRow);\n        return res;\n    }\n\n    private static getKeysForRow<EntityType extends object>(\n        keys: (keyof EntityType)[],\n        entity: EntityType,\n        i: number\n    ): (keyof EntityType)[] {\n        return keys\n            .filter(k => EntityUtilities.getPropertyMetadata(entity, k).position.row === i)\n            .sort((a, b) => EntityUtilities.compareOrder(a, b, entity));\n    }\n\n    private static getNumberOfRows<EntityType extends object>(keys: (keyof EntityType)[], entity: EntityType): number {\n        return keys\n            .map(k => EntityUtilities.getPropertyMetadata(entity, k).position.row)\n            .sort((a, b) => (a > b ? -1 : 1))[0];\n    }\n\n    /**\n     * Gets the keys of the provided entity correctly typed.\n     *\n     * @param entity - The entity to get the keys of.\n     * @param hideOmitForCreate - Whether or not keys with the metadata omitForCreate should be filtered out.\n     * @param hideOmitForEdit - Whether or not keys with the metadata omitForUpdate should be filtered out.\n     * @returns An array of keys of the entity.\n     */\n    static keysOf<EntityType extends object>(\n        entity: EntityType,\n        hideOmitForCreate: boolean = false,\n        hideOmitForEdit: boolean = false\n    ): (keyof EntityType)[] {\n        let keys: (keyof EntityType)[] = Reflect.ownKeys(entity) as (keyof EntityType)[];\n        if (hideOmitForCreate) {\n            const omitForCreateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForCreate(entity);\n            keys = keys.filter(k => !omitForCreateKeys.includes(k));\n        }\n        if (hideOmitForEdit) {\n            const omitForUpdateKeys: (keyof EntityType)[] = EntityUtilities.getOmitForUpdate(entity);\n            keys = keys.filter(k => !omitForUpdateKeys.includes(k));\n        }\n        return keys;\n    }\n}\n\n/**\n * A row that contains all the information about how to display an entity.\n */\nexport interface EntityRow<EntityType extends object> {\n    /**\n     * The row in which this should be displayed.\n     */\n    row: number,\n    /**\n     * The keys of the values that should be displayed in that row.\n     */\n    keys: (keyof EntityType)[]\n}"]}