ngx-material-entity 16.1.5 → 18.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/components/create-page/create-page.component.d.ts +2 -2
- package/components/edit-page/edit-page.component.d.ts +4 -4
- package/components/form/form.component.d.ts +7 -2
- package/components/input/array/array-date-time-input/array-date-time-input.component.d.ts +1 -2
- package/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.d.ts +1 -1
- package/components/input/base-input.component.d.ts +2 -2
- package/components/input/date/date-time-input/date-time-input.component.d.ts +1 -2
- package/components/input/file/file-input/file-input.component.d.ts +3 -3
- package/components/input/input.component.d.ts +64 -37
- package/components/input/number/number-dropdown-input/number-dropdown-input.component.d.ts +1 -1
- package/components/input/relations/references-many-input/references-many-input.component.d.ts +1 -1
- package/components/input/string/string-autocomplete-input/string-autocomplete-input.component.d.ts +1 -1
- package/components/input/string/string-dropdown-input/string-dropdown-input.component.d.ts +1 -1
- package/components/table/create-dialog/create-entity-dialog.component.d.ts +1 -1
- package/components/table/display-column-value/base-display-column-value.component.d.ts +2 -5
- package/components/table/display-column-value/display-column-value.component.d.ts +1 -1
- package/components/table/edit-dialog/edit-entity-dialog.component.d.ts +4 -4
- package/components/table/table.component.d.ts +3 -3
- package/components/tooltip/tooltip.component.d.ts +1 -1
- package/decorators/array/array-decorator-internal.data.d.ts +1 -1
- package/decorators/array/array-decorator.data.d.ts +1 -1
- package/decorators/date/date-decorator-internal.data.d.ts +1 -1
- package/decorators/date/date-decorator.data.d.ts +1 -1
- package/directives/drag-drop.directive.d.ts +1 -1
- package/directives/tooltip.directive.d.ts +1 -1
- package/esm2022/classes/base.builder.mjs +2 -2
- package/esm2022/classes/entity.model.mjs +1 -1
- package/esm2022/components/confirm-dialog/confirm-dialog-data.builder.mjs +2 -2
- package/esm2022/components/confirm-dialog/confirm-dialog.component.mjs +7 -7
- package/esm2022/components/create-page/create-data.route.mjs +2 -1
- package/esm2022/components/create-page/create-page.component.mjs +26 -23
- package/esm2022/components/create-page/page-create-data.builder.mjs +2 -1
- package/esm2022/components/edit-page/edit-data.route.mjs +2 -1
- package/esm2022/components/edit-page/edit-page.component.mjs +55 -54
- package/esm2022/components/edit-page/page-edit-data.builder.mjs +2 -1
- package/esm2022/components/form/form.component.mjs +14 -6
- package/esm2022/components/input/array/array-date-input/array-date-input.component.mjs +7 -7
- package/esm2022/components/input/array/array-date-range-input/array-date-range-input.component.mjs +7 -7
- package/esm2022/components/input/array/array-date-time-input/array-date-time-input.component.mjs +8 -8
- package/esm2022/components/input/array/array-string-autocomplete-chips/array-string-autocomplete-chips.component.mjs +8 -8
- package/esm2022/components/input/array/array-string-chips-input/array-string-chips-input.component.mjs +5 -5
- package/esm2022/components/input/array/array-table.class.mjs +21 -20
- package/esm2022/components/input/base-input.component.mjs +14 -9
- package/esm2022/components/input/boolean/boolean-checkbox-input/boolean-checkbox-input.component.mjs +5 -5
- package/esm2022/components/input/boolean/boolean-dropdown-input/boolean-dropdown-input.component.mjs +6 -6
- package/esm2022/components/input/boolean/boolean-toggle-input/boolean-toggle-input.component.mjs +5 -5
- package/esm2022/components/input/custom/custom.component.mjs +4 -4
- package/esm2022/components/input/date/date-input/date-input.component.mjs +3 -3
- package/esm2022/components/input/date/date-range-input/date-range-input.component.mjs +7 -7
- package/esm2022/components/input/date/date-time-input/date-time-input.component.mjs +9 -12
- package/esm2022/components/input/file/file-default-input/file-default-input.component.mjs +5 -7
- package/esm2022/components/input/file/file-image-input/file-image-input.component.mjs +6 -6
- package/esm2022/components/input/file/file-input/file-input.component.mjs +22 -24
- package/esm2022/components/input/input.component.mjs +194 -147
- package/esm2022/components/input/number/number-dropdown-input/number-dropdown-input.component.mjs +10 -10
- package/esm2022/components/input/number/number-input/number-input.component.mjs +3 -3
- package/esm2022/components/input/number/number-slider-input/number-slider-input.component.mjs +3 -3
- package/esm2022/components/input/relations/references-many-input/references-many-input.component.mjs +21 -21
- package/esm2022/components/input/string/string-autocomplete-input/string-autocomplete-input.component.mjs +10 -11
- package/esm2022/components/input/string/string-dropdown-input/string-dropdown-input.component.mjs +9 -9
- package/esm2022/components/input/string/string-input/string-input.component.mjs +3 -3
- package/esm2022/components/input/string/string-password-input/string-password-input.component.mjs +8 -11
- package/esm2022/components/input/string/string-textbox-input/string-textbox-input.component.mjs +3 -3
- package/esm2022/components/table/create-dialog/create-data.builder.mjs +3 -1
- package/esm2022/components/table/create-dialog/create-entity-dialog.component.mjs +11 -7
- package/esm2022/components/table/default.actions.mjs +4 -3
- package/esm2022/components/table/display-column-value/base-display-column-value.component.mjs +8 -13
- package/esm2022/components/table/display-column-value/display-column-value.component.mjs +9 -13
- package/esm2022/components/table/edit-dialog/edit-data.builder.mjs +9 -3
- package/esm2022/components/table/edit-dialog/edit-entity-dialog.component.mjs +40 -34
- package/esm2022/components/table/table-data.builder.mjs +12 -5
- package/esm2022/components/table/table-data.mjs +1 -1
- package/esm2022/components/table/table.component.mjs +25 -24
- package/esm2022/components/tooltip/tooltip.component.mjs +7 -8
- package/esm2022/decorators/array/array-decorator-internal.data.mjs +1 -1
- package/esm2022/decorators/array/array-decorator.data.mjs +1 -1
- package/esm2022/decorators/array/array.decorator.mjs +1 -1
- package/esm2022/decorators/base/base-property.decorator.mjs +1 -1
- package/esm2022/decorators/base/decorator-types.enum.mjs +1 -1
- package/esm2022/decorators/base/property-decorator-internal.data.mjs +6 -6
- package/esm2022/decorators/boolean/boolean-decorator-internal.data.mjs +2 -1
- package/esm2022/decorators/boolean/boolean.decorator.mjs +1 -1
- package/esm2022/decorators/custom/custom-decorator-internal.data.mjs +1 -1
- package/esm2022/decorators/custom/custom-decorator.data.mjs +1 -1
- package/esm2022/decorators/date/date-decorator-internal.data.mjs +1 -1
- package/esm2022/decorators/date/date-decorator.data.mjs +1 -1
- package/esm2022/decorators/date/date.decorator.mjs +1 -1
- package/esm2022/decorators/file/file-decorator-internal.data.mjs +8 -4
- package/esm2022/decorators/file/file.decorator.mjs +1 -1
- package/esm2022/decorators/has-many/has-many-decorator-internal.data.mjs +2 -1
- package/esm2022/decorators/number/number-decorator.data.mjs +1 -1
- package/esm2022/decorators/number/number.decorator.mjs +1 -1
- package/esm2022/decorators/object/object-decorator.data.mjs +1 -1
- package/esm2022/decorators/object/object.decorator.mjs +2 -1
- package/esm2022/decorators/string/string-decorator-internal.data.mjs +1 -1
- package/esm2022/decorators/string/string-decorator.data.mjs +1 -1
- package/esm2022/decorators/string/string.decorator.mjs +1 -1
- package/esm2022/directives/drag-drop.directive.mjs +5 -5
- package/esm2022/directives/dynamic-style-class.directive.mjs +5 -5
- package/esm2022/directives/included-in.directive.mjs +6 -5
- package/esm2022/directives/number.directive.mjs +6 -6
- package/esm2022/directives/password-match.directive.mjs +5 -5
- package/esm2022/directives/tooltip.directive.mjs +22 -20
- package/esm2022/encapsulation/reflect.utilities.mjs +2 -2
- package/esm2022/functions/dropdown-values-to-function.function.mjs +1 -1
- package/esm2022/functions/get-config-value.function.mjs +2 -2
- package/esm2022/functions/get-validation-error-message.function.mjs +1 -1
- package/esm2022/functions/get-validation-errors-tooltip-content.function.ts.mjs +1 -1
- package/esm2022/global-configuration-values.mjs +2 -1
- package/esm2022/mocks/placeholder-data.png.mjs +2 -2
- package/esm2022/services/entity.service.mjs +10 -11
- package/esm2022/services/unsaved-changes.guard.mjs +1 -1
- package/esm2022/utilities/date.utilities.mjs +10 -25
- package/esm2022/utilities/entity.utilities.mjs +15 -19
- package/esm2022/utilities/file.utilities.mjs +8 -10
- package/esm2022/utilities/selection.utilities.mjs +6 -4
- package/esm2022/utilities/validation.utilities.mjs +35 -38
- package/fesm2022/ngx-material-entity.mjs +728 -671
- package/fesm2022/ngx-material-entity.mjs.map +1 -1
- package/package.json +8 -8
- package/utilities/date.utilities.d.ts +29 -1
|
@@ -32,12 +32,12 @@ import * as i9 from "../../services/entity.service";
|
|
|
32
32
|
/**
|
|
33
33
|
* The entity service that needs to be provided in the providers array of the edit page route.
|
|
34
34
|
*/
|
|
35
|
-
// eslint-disable-next-line typescript/no-explicit-any
|
|
35
|
+
// eslint-disable-next-line typescript/no-explicit-any, stylistic/max-len
|
|
36
36
|
export const NGX_EDIT_DATA_ENTITY_SERVICE = new InjectionToken('NGX_EDIT_DATA_ENTITY_SERVICE');
|
|
37
37
|
/**
|
|
38
38
|
* The entity class that needs to be provided in the providers array of the edit page route.
|
|
39
39
|
*/
|
|
40
|
-
// eslint-disable-next-line typescript/no-explicit-any
|
|
40
|
+
// eslint-disable-next-line typescript/no-explicit-any, stylistic/max-len
|
|
41
41
|
export const NGX_EDIT_DATA_ENTITY = new InjectionToken('NGX_EDIT_DATA_ENTITY');
|
|
42
42
|
/**
|
|
43
43
|
* The configuration that needs to be provided in the providers array of the edit page route.
|
|
@@ -122,27 +122,29 @@ export class NgxMatEntityEditPageComponent {
|
|
|
122
122
|
this.renderer = renderer;
|
|
123
123
|
this.globalConfig = globalConfig;
|
|
124
124
|
}
|
|
125
|
-
|
|
125
|
+
ngOnInit() {
|
|
126
126
|
this.data = new PageEditDataBuilder(this.inputData, this.globalConfig).getResult();
|
|
127
|
-
if (this.data ==
|
|
127
|
+
if (this.data == undefined) {
|
|
128
128
|
this.confirmNavigateBack();
|
|
129
129
|
throw new Error('No edit data was provided for "NGX_EDIT_DATA". You need to provide a value in your routes providers array.');
|
|
130
130
|
}
|
|
131
131
|
const id = this.route.snapshot.paramMap.get('id');
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
this.
|
|
141
|
-
|
|
132
|
+
// eslint-disable-next-line promise/prefer-await-to-then
|
|
133
|
+
void this.entityService.findById(id).then(foundEntity => {
|
|
134
|
+
if (foundEntity == undefined) {
|
|
135
|
+
this.confirmNavigateBack();
|
|
136
|
+
throw new Error(`Could not find entity with id ${id}`);
|
|
137
|
+
}
|
|
138
|
+
this.entity = new this.EntityClass(foundEntity);
|
|
139
|
+
this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);
|
|
140
|
+
runInInjectionContext(this.injector, () => {
|
|
141
|
+
this.isEntityReadOnly = !this.data.allowUpdate(this.entityPriorChanges);
|
|
142
|
+
this.allowDelete = this.data.allowDelete(this.entityPriorChanges);
|
|
143
|
+
});
|
|
144
|
+
this.isLoaded = true;
|
|
145
|
+
setTimeout(() => this.checkOffset(), 1);
|
|
146
|
+
setTimeout(() => void this.checkIsEntityValid(), 1);
|
|
142
147
|
});
|
|
143
|
-
this.isLoaded = true;
|
|
144
|
-
setTimeout(() => this.checkOffset(), 1);
|
|
145
|
-
setTimeout(() => void this.checkIsEntityValid(), 1);
|
|
146
148
|
}
|
|
147
149
|
/**
|
|
148
150
|
* Checks if the bottom row should be displayed as fixed.
|
|
@@ -182,14 +184,15 @@ export class NgxMatEntityEditPageComponent {
|
|
|
182
184
|
* Tries to save the changes and close the dialog afterwards.
|
|
183
185
|
* Also handles the confirmation if required.
|
|
184
186
|
*/
|
|
185
|
-
edit() {
|
|
187
|
+
async edit() {
|
|
186
188
|
if (this.isEntityReadOnly || !this.isEntityValid || !this.isEntityDirty) {
|
|
187
189
|
return;
|
|
188
190
|
}
|
|
189
191
|
if (!this.data.editData.editRequiresConfirmDialog) {
|
|
190
|
-
this.confirmEdit();
|
|
192
|
+
await this.confirmEdit();
|
|
191
193
|
return;
|
|
192
194
|
}
|
|
195
|
+
// eslint-disable-next-line stylistic/max-len
|
|
193
196
|
const dialogData = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmEditDialogData)
|
|
194
197
|
.withDefault('text', this.globalConfig.confirmSaveText)
|
|
195
198
|
.withDefault('confirmButtonLabel', this.globalConfig.saveLabel)
|
|
@@ -200,24 +203,25 @@ export class NgxMatEntityEditPageComponent {
|
|
|
200
203
|
autoFocus: false,
|
|
201
204
|
restoreFocus: false
|
|
202
205
|
});
|
|
203
|
-
dialogRef.afterClosed()
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
});
|
|
206
|
+
const res = await firstValueFrom(dialogRef.afterClosed());
|
|
207
|
+
if (res == true) {
|
|
208
|
+
await this.confirmEdit();
|
|
209
|
+
}
|
|
208
210
|
}
|
|
209
|
-
confirmEdit() {
|
|
210
|
-
|
|
211
|
+
async confirmEdit() {
|
|
212
|
+
await this.entityService.update(this.entity, this.entityPriorChanges);
|
|
213
|
+
this.confirmNavigateBack();
|
|
211
214
|
}
|
|
212
215
|
/**
|
|
213
216
|
* Tries to delete the entity and close the dialog afterwards.
|
|
214
217
|
* Also handles the confirmation if required.
|
|
215
218
|
*/
|
|
216
|
-
delete() {
|
|
219
|
+
async delete() {
|
|
217
220
|
if (!this.data.editData.deleteRequiresConfirmDialog) {
|
|
218
|
-
this.confirmDelete();
|
|
221
|
+
await this.confirmDelete();
|
|
219
222
|
return;
|
|
220
223
|
}
|
|
224
|
+
// eslint-disable-next-line stylistic/max-len
|
|
221
225
|
const dialogData = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmDeleteDialogData)
|
|
222
226
|
.withDefault('text', this.globalConfig.confirmDeleteText)
|
|
223
227
|
.withDefault('type', 'delete')
|
|
@@ -229,16 +233,14 @@ export class NgxMatEntityEditPageComponent {
|
|
|
229
233
|
autoFocus: false,
|
|
230
234
|
restoreFocus: false
|
|
231
235
|
});
|
|
232
|
-
dialogRef.afterClosed()
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
});
|
|
236
|
+
const res = await firstValueFrom(dialogRef.afterClosed());
|
|
237
|
+
if (res == true) {
|
|
238
|
+
await this.confirmDelete();
|
|
239
|
+
}
|
|
237
240
|
}
|
|
238
|
-
confirmDelete() {
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
.then(() => this.confirmNavigateBack());
|
|
241
|
+
async confirmDelete() {
|
|
242
|
+
await this.entityService.delete(this.entityPriorChanges);
|
|
243
|
+
this.confirmNavigateBack();
|
|
242
244
|
}
|
|
243
245
|
/**
|
|
244
246
|
* Tries to navigate back.
|
|
@@ -260,7 +262,7 @@ export class NgxMatEntityEditPageComponent {
|
|
|
260
262
|
autoFocus: false,
|
|
261
263
|
restoreFocus: false
|
|
262
264
|
});
|
|
263
|
-
return
|
|
265
|
+
return await firstValueFrom(dialogRef.afterClosed()) ?? false;
|
|
264
266
|
}
|
|
265
267
|
confirmNavigateBack() {
|
|
266
268
|
this.inConfirmNavigation = true;
|
|
@@ -271,12 +273,12 @@ export class NgxMatEntityEditPageComponent {
|
|
|
271
273
|
* Runs the edit action on the entity.
|
|
272
274
|
* @param action - The action to run.
|
|
273
275
|
*/
|
|
274
|
-
runEditAction(action) {
|
|
276
|
+
async runEditAction(action) {
|
|
275
277
|
const requireConfirmDialog = runInInjectionContext(this.injector, () => {
|
|
276
278
|
return action.requireConfirmDialog(this.entityPriorChanges);
|
|
277
279
|
});
|
|
278
280
|
if (!requireConfirmDialog) {
|
|
279
|
-
this.confirmRunEditAction(action);
|
|
281
|
+
await this.confirmRunEditAction(action);
|
|
280
282
|
return;
|
|
281
283
|
}
|
|
282
284
|
const dialogRef = this.dialog.open(NgxMatEntityConfirmDialogComponent, {
|
|
@@ -284,14 +286,13 @@ export class NgxMatEntityEditPageComponent {
|
|
|
284
286
|
autoFocus: false,
|
|
285
287
|
restoreFocus: false
|
|
286
288
|
});
|
|
287
|
-
dialogRef.afterClosed()
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
});
|
|
289
|
+
const res = await firstValueFrom(dialogRef.afterClosed());
|
|
290
|
+
if (res == true) {
|
|
291
|
+
await this.confirmRunEditAction(action);
|
|
292
|
+
}
|
|
292
293
|
}
|
|
293
|
-
confirmRunEditAction(action) {
|
|
294
|
-
|
|
294
|
+
async confirmRunEditAction(action) {
|
|
295
|
+
await runInInjectionContext(this.injector, async () => {
|
|
295
296
|
await action.action(this.entity, this.entityPriorChanges);
|
|
296
297
|
await this.checkEntity();
|
|
297
298
|
});
|
|
@@ -304,10 +305,10 @@ export class NgxMatEntityEditPageComponent {
|
|
|
304
305
|
editActionDisabled(action) {
|
|
305
306
|
return runInInjectionContext(this.injector, () => !action.enabled(this.entityPriorChanges));
|
|
306
307
|
}
|
|
307
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
308
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
308
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.0", ngImport: i0, type: NgxMatEntityEditPageComponent, deps: [{ token: i1.MatDialog }, { token: i2.Location }, { token: i3.ActivatedRoute }, { token: i0.EnvironmentInjector }, { token: NGX_EDIT_DATA_ENTITY_SERVICE }, { token: NGX_EDIT_DATA_ENTITY }, { token: NGX_EDIT_DATA }, { token: i4.HttpClient }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: NGX_COMPLETE_GLOBAL_DEFAULT_VALUES }], target: i0.ɵɵFactoryTarget.Component });
|
|
309
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.0", type: NgxMatEntityEditPageComponent, isStandalone: true, selector: "ngx-mat-entity-edit-page", host: { listeners: { "window:scroll": "checkOffset()", "window:beforeunload": "canDeactivate()" } }, ngImport: i0, template: " \n<!-- eslint-disable angular/no-call-expression -->\n@if (!isLoaded && data.displayLoadingSpinner) {\n <div class=\"container\">\n <br>\n <mat-spinner></mat-spinner>\n <br>\n </div>\n}\n\n@if (isLoaded) {\n <div class=\"container\">\n <br>\n \n <!------------>\n <!-- Header -->\n <!------------>\n <div class=\"header\">\n <div class=\"cancel-container\">\n <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n <i class=\"fas fa-chevron-left\"></i>\n {{data.editData.cancelButtonLabel}}\n @if (hasUnsavedChanges) {\n <i class=\"fas fa-warning\"></i>\n }\n </button>\n </div>\n <div class=\"actions-container\">\n @if (data.editData.actions.length) {\n <button type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n {{data.editData.actionsLabel}}\n </button>\n }\n <mat-menu #menu=\"matMenu\">\n @for (action of data.editData.actions; track $index) {\n <button type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n {{action.displayName}}\n </button>\n }\n </mat-menu>\n \n @if (allowDelete) {\n <button type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n {{data.editData.deleteButtonLabel}}\n </button>\n }\n </div>\n </div>\n \n <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n \n <!----------->\n <!-- Input -->\n <!----------->\n <form>\n <ngx-mat-entity-form [hideOmitForEdit]=\"true\" [isEntityReadOnly]=\"isEntityReadOnly\" [entity]=\"entity\" (selectedTabChange)=\"checkOffset()\" (formChange)=\"checkEntity()\">\n </ngx-mat-entity-form>\n \n <div class=\"bottom-row-container\">\n <div class=\"bottom-row container\">\n <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n [matBadge]=\"validationErrors.length\"\n [matBadgeHidden]=\"!validationErrors.length\"\n (click)=\"edit()\"\n >\n {{data.editData.confirmButtonLabel}}\n </button>\n @if (validationErrors.length) {\n <ngx-mat-entity-tooltip [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n }\n </div>\n </div>\n </form>\n <br>\n </div>\n}", styles: ["h1{text-align:center}mat-spinner{margin:10px auto}.fa-warning{color:orange}.bottom-row{display:flex;align-items:center;column-gap:10px;margin-top:10px}.fixed{position:fixed;bottom:0;left:0;right:0;width:100%;z-index:1000;padding:8px 20px;background-color:#fff}.header{display:flex;margin-bottom:5px;gap:10px;flex-wrap:wrap}.header button{min-width:150px}.header .cancel-container{display:flex;justify-content:flex-start;align-items:center;column-gap:10px;width:calc(50% - 10px)}.header .actions-container{display:flex;justify-content:flex-end;gap:10px;width:calc(50% - 10px)}.unsavedChanges{background-color:#ffe48d}@media (max-width: 800px){.header{margin-bottom:10px;gap:15px}.header button{min-width:0px;width:50%}.header .cancel-container,.header .actions-container{width:100%;gap:15px}}\n"], dependencies: [{ kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i5.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i6.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i7.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i7.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i7.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "ngmodule", type: MatBadgeModule }, { kind: "directive", type: i8.MatBadge, selector: "[matBadge]", inputs: ["matBadgeColor", "matBadgeOverlap", "matBadgeDisabled", "matBadgePosition", "matBadge", "matBadgeDescription", "matBadgeSize", "matBadgeHidden"] }, { kind: "component", type: TooltipComponent, selector: "ngx-mat-entity-tooltip", inputs: ["tooltipContent"] }, { kind: "component", type: NgxMatEntityFormComponent, selector: "ngx-mat-entity-form", inputs: ["entity", "isEntityReadOnly", "hideOmitForCreate", "hideOmitForEdit", "additionalOmitKeys"], outputs: ["formChange", "selectedTabChange"] }] });
|
|
309
310
|
}
|
|
310
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
311
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.0", ngImport: i0, type: NgxMatEntityEditPageComponent, decorators: [{
|
|
311
312
|
type: Component,
|
|
312
313
|
args: [{ selector: 'ngx-mat-entity-edit-page', standalone: true, imports: [
|
|
313
314
|
NgIf,
|
|
@@ -318,8 +319,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImpor
|
|
|
318
319
|
MatBadgeModule,
|
|
319
320
|
TooltipComponent,
|
|
320
321
|
NgxMatEntityFormComponent
|
|
321
|
-
], template: " \n<!-- eslint-disable angular/no-call-expression -->\n
|
|
322
|
-
}], ctorParameters:
|
|
322
|
+
], template: " \n<!-- eslint-disable angular/no-call-expression -->\n@if (!isLoaded && data.displayLoadingSpinner) {\n <div class=\"container\">\n <br>\n <mat-spinner></mat-spinner>\n <br>\n </div>\n}\n\n@if (isLoaded) {\n <div class=\"container\">\n <br>\n \n <!------------>\n <!-- Header -->\n <!------------>\n <div class=\"header\">\n <div class=\"cancel-container\">\n <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n <i class=\"fas fa-chevron-left\"></i>\n {{data.editData.cancelButtonLabel}}\n @if (hasUnsavedChanges) {\n <i class=\"fas fa-warning\"></i>\n }\n </button>\n </div>\n <div class=\"actions-container\">\n @if (data.editData.actions.length) {\n <button type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n {{data.editData.actionsLabel}}\n </button>\n }\n <mat-menu #menu=\"matMenu\">\n @for (action of data.editData.actions; track $index) {\n <button type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n {{action.displayName}}\n </button>\n }\n </mat-menu>\n \n @if (allowDelete) {\n <button type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n {{data.editData.deleteButtonLabel}}\n </button>\n }\n </div>\n </div>\n \n <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n \n <!----------->\n <!-- Input -->\n <!----------->\n <form>\n <ngx-mat-entity-form [hideOmitForEdit]=\"true\" [isEntityReadOnly]=\"isEntityReadOnly\" [entity]=\"entity\" (selectedTabChange)=\"checkOffset()\" (formChange)=\"checkEntity()\">\n </ngx-mat-entity-form>\n \n <div class=\"bottom-row-container\">\n <div class=\"bottom-row container\">\n <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n [matBadge]=\"validationErrors.length\"\n [matBadgeHidden]=\"!validationErrors.length\"\n (click)=\"edit()\"\n >\n {{data.editData.confirmButtonLabel}}\n </button>\n @if (validationErrors.length) {\n <ngx-mat-entity-tooltip [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n }\n </div>\n </div>\n </form>\n <br>\n </div>\n}", styles: ["h1{text-align:center}mat-spinner{margin:10px auto}.fa-warning{color:orange}.bottom-row{display:flex;align-items:center;column-gap:10px;margin-top:10px}.fixed{position:fixed;bottom:0;left:0;right:0;width:100%;z-index:1000;padding:8px 20px;background-color:#fff}.header{display:flex;margin-bottom:5px;gap:10px;flex-wrap:wrap}.header button{min-width:150px}.header .cancel-container{display:flex;justify-content:flex-start;align-items:center;column-gap:10px;width:calc(50% - 10px)}.header .actions-container{display:flex;justify-content:flex-end;gap:10px;width:calc(50% - 10px)}.unsavedChanges{background-color:#ffe48d}@media (max-width: 800px){.header{margin-bottom:10px;gap:15px}.header button{min-width:0px;width:50%}.header .cancel-container,.header .actions-container{width:100%;gap:15px}}\n"] }]
|
|
323
|
+
}], ctorParameters: () => [{ type: i1.MatDialog }, { type: i2.Location }, { type: i3.ActivatedRoute }, { type: i0.EnvironmentInjector }, { type: i9.EntityService, decorators: [{
|
|
323
324
|
type: Inject,
|
|
324
325
|
args: [NGX_EDIT_DATA_ENTITY_SERVICE]
|
|
325
326
|
}] }, { type: undefined, decorators: [{
|
|
@@ -331,11 +332,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImpor
|
|
|
331
332
|
}] }, { type: i4.HttpClient }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: undefined, decorators: [{
|
|
332
333
|
type: Inject,
|
|
333
334
|
args: [NGX_COMPLETE_GLOBAL_DEFAULT_VALUES]
|
|
334
|
-
}] }]
|
|
335
|
+
}] }], propDecorators: { checkOffset: [{
|
|
335
336
|
type: HostListener,
|
|
336
337
|
args: ['window:scroll']
|
|
337
338
|
}], canDeactivate: [{
|
|
338
339
|
type: HostListener,
|
|
339
340
|
args: ['window:beforeunload']
|
|
340
341
|
}] } });
|
|
341
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edit-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.ts","../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AACxD,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,mBAAmB,EAAE,YAAY,EAAE,MAAM,EAAE,cAAc,EAAU,SAAS,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC3J,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAgB,MAAM,0BAA0B,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,wBAAwB,EAAE,MAAM,oCAAoC,CAAC;AAC9E,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,cAAc,EAAE,MAAM,MAAM,CAAC;AAEtC,OAAO,EAAE,mBAAmB,EAAwB,MAAM,0BAA0B,CAAC;AAErF,OAAO,EAAE,eAAe,EAAE,MAAM,sCAAsC,CAAC;AACvE,OAAO,EAAE,iCAAiC,EAAE,MAAM,mEAAmE,CAAC;AACtH,OAAO,EAAE,kCAAkC,EAA0B,MAAM,mCAAmC,CAAC;AAC/G,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAE9D,OAAO,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AACnE,OAAO,EAAmB,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAE5F,OAAO,EAAE,wBAAwB,EAA6B,MAAM,+CAA+C,CAAC;AACpH,OAAO,EAAE,kCAAkC,EAAE,MAAM,4CAA4C,CAAC;AAChG,OAAO,EAAE,yBAAyB,EAAE,MAAM,wBAAwB,CAAC;AAInE,OAAO,EAAE,gBAAgB,EAAE,MAAM,8BAA8B,CAAC;;;;;;;;;;;AA2BhE;;GAEG;AACH,sDAAsD;AACtD,MAAM,CAAC,MAAM,4BAA4B,GAAuC,IAAI,cAAc,CAAqB,8BAA8B,CAAC,CAAC;AACvJ;;GAEG;AACH,sDAAsD;AACtD,MAAM,CAAC,MAAM,oBAAoB,GAA4C,IAAI,cAAc,CAA0B,sBAAsB,CAAC,CAAC;AACjJ;;GAEG;AACH,sDAAsD;AACtD,MAAM,CAAC,MAAM,aAAa,GAAsC,IAAI,cAAc,CAAoB,eAAe,CAAC,CAAC;AAEvH;;;GAGG;AAiBH,MAAM,OAAO,6BAA6B;IA4DjB;IACA;IACA;IACA;IAER;IAEQ;IAEA;IACA;IACA;IACA;IAEE;IAxEvB;;OAEG;IACH,eAAe,GAA2B,eAAe,CAAC;IAE1D;;OAEG;IACH,MAAM,CAAc;IACpB;;OAEG;IACH,kBAAkB,CAAc;IAEhC;;OAEG;IACH,IAAI,CAAoC;IAExC;;OAEG;IACH,gBAAgB,GAAsB,EAAE,CAAC;IACzC;;OAEG;IACH,aAAa,GAAY,IAAI,CAAC;IAC9B;;OAEG;IACH,aAAa,GAAY,KAAK,CAAC;IAC/B;;OAEG;IACH,cAAc,GAAW,EAAE,CAAC;IAE5B;;OAEG;IACH,gBAAgB,CAAW;IAC3B;;OAEG;IACH,WAAW,CAAW;IAEtB;;OAEG;IACH,QAAQ,GAAY,KAAK,CAAC;IAElB,mBAAmB,GAAY,KAAK,CAAC;IAE7C,+CAA+C;IAC/C,IAAI,iBAAiB;QACjB,OAAO,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC;IACvF,CAAC;IAED,YACqB,MAAiB,EACjB,QAAkB,EAClB,KAAqB,EACrB,QAA6B,EAErC,aAAwC,EAEhC,WAA2C,EAE3C,SAAmC,EACnC,IAAgB,EAChB,EAAc,EACd,QAAmB,EAEjB,YAAoC;QAdtC,WAAM,GAAN,MAAM,CAAW;QACjB,aAAQ,GAAR,QAAQ,CAAU;QAClB,UAAK,GAAL,KAAK,CAAgB;QACrB,aAAQ,GAAR,QAAQ,CAAqB;QAErC,kBAAa,GAAb,aAAa,CAA2B;QAEhC,gBAAW,GAAX,WAAW,CAAgC;QAE3C,cAAS,GAAT,SAAS,CAA0B;QACnC,SAAI,GAAJ,IAAI,CAAY;QAChB,OAAE,GAAF,EAAE,CAAY;QACd,aAAQ,GAAR,QAAQ,CAAW;QAEjB,iBAAY,GAAZ,YAAY,CAAwB;IACvD,CAAC;IAEL,KAAK,CAAC,QAAQ;QACV,IAAI,CAAC,IAAI,GAAG,IAAI,mBAAmB,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC,SAAS,EAAE,CAAC;QACnF,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,EAAE;YACnB,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,4GAA4G,CAAC,CAAC;SACjI;QAED,MAAM,EAAE,GAAiC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAiC,CAAC;QAChH,MAAM,WAAW,GAA2B,MAAM,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QAElF,IAAI,WAAW,IAAI,IAAI,EAAE;YACrB,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,iCAAiC,EAAE,EAAE,CAAC,CAAC;SAC1D;QAED,IAAI,CAAC,MAAM,GAAG,IAAI,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;QAChD,IAAI,CAAC,kBAAkB,GAAG,eAAe,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEjE,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YACtC,IAAI,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;YACxE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QACtE,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,CAAC;QACxC,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,IAAI,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC,CAAC;IACxD,CAAC;IAED;;OAEG;IAEH,WAAW;QACP,MAAM,OAAO,GAAW,MAAM,CAAC,OAAO,CAAC;QACvC,MAAM,SAAS,GAAwB,IAAI,CAAC,EAAE,CAAC,aAA6B,CAAC,aAAa,CAAC,aAAa,CAAC,CAAC;QAC1G,MAAM,kBAAkB,GAAwB,IAAI,CAAC,EAAE,CAAC,aAA6B,CAAC,aAAa,CAAC,uBAAuB,CAAC,CAAC;QAE7H,IAAI,SAAS,IAAI,kBAAkB,EAAE;YACjC,MAAM,wBAAwB,GAAW,kBAAkB,CAAC,SAAS,CAAC;YACtE,MAAM,YAAY,GAAW,MAAM,CAAC,WAAW,CAAC;YAEhD,IAAI,OAAO,GAAG,YAAY,IAAI,wBAAwB,EAAE;gBACpD,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;aACjD;iBACI;gBACD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;aAC9C;SACJ;IACL,CAAC;IAED,+CAA+C;IAE/C,aAAa;QACT,OAAO,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,mBAAmB,CAAC;IAC/D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW;QACb,MAAM,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAChC,IAAI,CAAC,aAAa,GAAG,MAAM,eAAe,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;IACxG,CAAC;IAEO,KAAK,CAAC,kBAAkB;QAC5B,IAAI,CAAC,gBAAgB,GAAG,MAAM,mBAAmB,CAAC,yBAAyB,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;QAClH,IAAI,CAAC,cAAc,GAAG,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE,CAAC,iCAAiC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC;QAC3H,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,KAAK,CAAC,CAAC;IAC5D,CAAC;IAED;;;OAGG;IACH,IAAI;QACA,IAAI,IAAI,CAAC,gBAAgB,IAAI,CAAC,IAAI,CAAC,aAAa,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE;YACrE,OAAO;SACV;QACD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,yBAAyB,EAAE;YAC/C,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,OAAO;SACV;QAED,MAAM,UAAU,GAA8B,IAAI,wBAAwB,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,qBAAqB,CAAC;aAClI,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC;aACtD,WAAW,CAAC,oBAAoB,EAAE,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC;aAC9D,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC;aACjD,SAAS,EAAE,CAAC;QACjB,MAAM,SAAS,GAA8D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,kCAAkC,EAAE;YAC9H,IAAI,EAAE,UAAU;YAChB,SAAS,EAAE,KAAK;YAChB,YAAY,EAAE,KAAK;SACtB,CAAC,CAAC;QACH,SAAS,CAAC,WAAW,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACpC,IAAI,GAAG,IAAI,IAAI,EAAE;gBACb,IAAI,CAAC,WAAW,EAAE,CAAC;aACtB;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,WAAW;QACf,KAAK,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC;IAChH,CAAC;IAED;;;OAGG;IACH,MAAM;QACF,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,2BAA2B,EAAE;YACjD,IAAI,CAAC,aAAa,EAAE,CAAC;YACrB,OAAO;SACV;QAED,MAAM,UAAU,GAA8B,IAAI,wBAAwB,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,uBAAuB,CAAC;aACpI,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC;aACxD,WAAW,CAAC,MAAM,EAAE,QAAQ,CAAC;aAC7B,WAAW,CAAC,oBAAoB,EAAE,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC;aAChE,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC;aACnD,SAAS,EAAE,CAAC;QACjB,MAAM,SAAS,GAA8D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,kCAAkC,EAAE;YAC9H,IAAI,EAAE,UAAU;YAChB,SAAS,EAAE,KAAK;YAChB,YAAY,EAAE,KAAK;SACtB,CAAC,CAAC;QACH,SAAS,CAAC,WAAW,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACpC,IAAI,GAAG,IAAI,IAAI,EAAE;gBACb,IAAI,CAAC,aAAa,EAAE,CAAC;aACxB;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,aAAa;QACjB,KAAK,IAAI,CAAC,aAAa;aAClB,MAAM,CAAC,IAAI,CAAC,kBAAkB,CAAC;aAC/B,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY;QACd,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YACzB,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,OAAO;SACV;QAED,MAAM,GAAG,GAAY,MAAM,IAAI,CAAC,2BAA2B,EAAE,CAAC;QAC9D,IAAI,GAAG,EAAE;YACL,IAAI,CAAC,mBAAmB,EAAE,CAAC;SAC9B;IACL,CAAC;IAED,+CAA+C;IAC/C,KAAK,CAAC,2BAA2B;QAC7B,MAAM,SAAS,GAA8D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,kCAAkC,EAAE;YAC9H,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,+BAA+B;YACxD,SAAS,EAAE,KAAK;YAChB,YAAY,EAAE,KAAK;SACtB,CAAC,CAAC;QACH,OAAO,CAAC,MAAM,cAAc,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC;IACpE,CAAC;IAEO,mBAAmB;QACvB,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;QAChC,eAAe,CAAC,oBAAoB,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAC3E,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED;;;OAGG;IACH,aAAa,CAAC,MAAsC;QAChD,MAAM,oBAAoB,GAAY,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YAC5E,OAAO,MAAM,CAAC,oBAAoB,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAChE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,oBAAoB,EAAE;YACvB,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,CAAC;YAClC,OAAO;SACV;QACD,MAAM,SAAS,GAA8D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,kCAAkC,EAAE;YAC9H,IAAI,EAAE,MAAM,CAAC,iBAAiB;YAC9B,SAAS,EAAE,KAAK;YAChB,YAAY,EAAE,KAAK;SACtB,CAAC,CAAC;QACH,SAAS,CAAC,WAAW,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YACpC,IAAI,GAAG,IAAI,IAAI,EAAE;gBACb,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,CAAC;aACrC;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,oBAAoB,CAAC,MAAsC;QAC/D,KAAK,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;YACjD,MAAM,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;YAC1D,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC;QAC7B,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;OAIG;IACH,kBAAkB,CAAC,MAAsC;QACrD,OAAO,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC;IAChG,CAAC;uGA5RQ,6BAA6B,oIAgE1B,4BAA4B,aAE5B,oBAAoB,aAEpB,aAAa,0FAKb,kCAAkC;2FAzErC,6BAA6B,yLC1F1C,mqFAgEM,m1BDgBE,IAAI,6FACJ,KAAK,kHACL,eAAe,2QACf,wBAAwB,kOACxB,aAAa,+YACb,cAAc,4PACd,gBAAgB,+FAChB,yBAAyB;;2FAGpB,6BAA6B;kBAhBzC,SAAS;+BACI,0BAA0B,cAGxB,IAAI,WACP;wBACL,IAAI;wBACJ,KAAK;wBACL,eAAe;wBACf,wBAAwB;wBACxB,aAAa;wBACb,cAAc;wBACd,gBAAgB;wBAChB,yBAAyB;qBAC5B;;0BAkEI,MAAM;2BAAC,4BAA4B;;0BAEnC,MAAM;2BAAC,oBAAoB;;0BAE3B,MAAM;2BAAC,aAAa;;0BAKpB,MAAM;2BAAC,kCAAkC;4CAmC9C,WAAW;sBADV,YAAY;uBAAC,eAAe;gBAqB7B,aAAa;sBADZ,YAAY;uBAAC,qBAAqB","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { ActivatedRoute } from '@angular/router';\nimport { firstValueFrom } from 'rxjs';\n\nimport { PageEditDataBuilder, PageEditDataInternal } from './page-edit-data.builder';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NGX_COMPLETE_GLOBAL_DEFAULT_VALUES, NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { UnsavedChangesPage } from '../../services/unsaved-changes.guard';\nimport { EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityFormComponent } from '../form/form.component';\nimport { EditActionInternal } from '../table/edit-dialog/edit-data.builder';\nimport { EditEntityData } from '../table/edit-dialog/edit-entity-data';\nimport { EditData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\n\n/**\n * The data that needs to be provided for a route to be able to edit a entity.\n */\nexport type PageEditData<EntityType extends BaseEntityType<EntityType>> = Omit<EditEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the entity for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default edit page.\n     */\n    editData?: EditData<EntityType> & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any\nexport const NGX_EDIT_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_EDIT_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any\nexport const NGX_EDIT_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_EDIT_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any\nexport const NGX_EDIT_DATA: InjectionToken<PageEditData<any>> = new InjectionToken<PageEditData<any>>('NGX_EDIT_DATA');\n\n/**\n * A generic page that allows you to edit a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-edit-page',\n    templateUrl: './edit-page.component.html',\n    styleUrls: ['./edit-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatProgressSpinnerModule,\n        MatMenuModule,\n        MatBadgeModule,\n        TooltipComponent,\n        NgxMatEntityFormComponent\n    ]\n})\nexport class NgxMatEntityEditPageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit, UnsavedChangesPage {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The entity that is being edited.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for the component.\n     */\n    data!: PageEditDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors: ValidationError[] = [];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = true;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    /**\n     * Whether or not the entity is readonly.\n     */\n    isEntityReadOnly!: boolean;\n    /**\n     * Whether or not the current user is allowed to delete the entity.\n     */\n    allowDelete!: boolean;\n\n    /**\n     * Whether or not the page has been loaded.\n     */\n    isLoaded: boolean = false;\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.editData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly route: ActivatedRoute,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_EDIT_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_EDIT_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_EDIT_DATA)\n        private readonly inputData: PageEditData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_COMPLETE_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    async ngOnInit(): Promise<void> {\n        this.data = new PageEditDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == null) {\n            this.confirmNavigateBack();\n            throw new Error('No edit data was provided for \"NGX_EDIT_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        const id: EntityType[keyof EntityType] = this.route.snapshot.paramMap.get('id') as EntityType[keyof EntityType];\n        const foundEntity: EntityType | undefined = await this.entityService.findById(id);\n\n        if (foundEntity == null) {\n            this.confirmNavigateBack();\n            throw new Error(`Could not find entity with id ${id}`);\n        }\n\n        this.entity = new this.EntityClass(foundEntity);\n        this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n\n        runInInjectionContext(this.injector, () => {\n            this.isEntityReadOnly = !this.data.allowUpdate(this.entityPriorChanges);\n            this.allowDelete = this.data.allowDelete(this.entityPriorChanges);\n        });\n        this.isLoaded = true;\n        setTimeout(() => this.checkOffset(), 1);\n        setTimeout(() => void this.checkIsEntityValid(), 1);\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        await this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private async checkIsEntityValid(): Promise<void> {\n        this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'update');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries to save the changes and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    edit(): void {\n        if (this.isEntityReadOnly || !this.isEntityValid || !this.isEntityDirty) {\n            return;\n        }\n        if (!this.data.editData.editRequiresConfirmDialog) {\n            this.confirmEdit();\n            return;\n        }\n\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmEditDialogData)\n            .withDefault('text', this.globalConfig.confirmSaveText)\n            .withDefault('confirmButtonLabel', this.globalConfig.saveLabel)\n            .withDefault('title', this.globalConfig.editLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmEdit();\n            }\n        });\n    }\n\n    private confirmEdit(): void {\n        void this.entityService.update(this.entity, this.entityPriorChanges).then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to delete the entity and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    delete(): void {\n        if (!this.data.editData.deleteRequiresConfirmDialog) {\n            this.confirmDelete();\n            return;\n        }\n\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmDeleteDialogData)\n            .withDefault('text', this.globalConfig.confirmDeleteText)\n            .withDefault('type', 'delete')\n            .withDefault('confirmButtonLabel', this.globalConfig.deleteLabel)\n            .withDefault('title', this.globalConfig.deleteLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmDelete();\n            }\n        });\n    }\n\n    private confirmDelete(): void {\n        void this.entityService\n            .delete(this.entityPriorChanges)\n            .then(() => this.confirmNavigateBack());\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    async navigateBack(): Promise<void> {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        const res: boolean = await this.openConfirmNavigationDialog();\n        if (res) {\n            this.confirmNavigateBack();\n        }\n    }\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    async openConfirmNavigationDialog(): Promise<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.editData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return (await firstValueFrom(dialogRef.afterClosed())) ?? false;\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        EntityUtilities.resetChangesOnEntity(this.entity, this.entityPriorChanges);\n        this.location.back();\n    }\n\n    /**\n     * Runs the edit action on the entity.\n     * @param action - The action to run.\n     */\n    runEditAction(action: EditActionInternal<EntityType>): void {\n        const requireConfirmDialog: boolean = runInInjectionContext(this.injector, () => {\n            return action.requireConfirmDialog(this.entityPriorChanges);\n        });\n\n        if (!requireConfirmDialog) {\n            this.confirmRunEditAction(action);\n            return;\n        }\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: action.confirmDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        dialogRef.afterClosed().subscribe(res => {\n            if (res == true) {\n                this.confirmRunEditAction(action);\n            }\n        });\n    }\n\n    private confirmRunEditAction(action: EditActionInternal<EntityType>): void {\n        void runInInjectionContext(this.injector, async () => {\n            await action.action(this.entity, this.entityPriorChanges);\n            await this.checkEntity();\n        });\n    }\n\n    /**\n     * Checks if an EditAction is disabled (e.g. Because the current entry doesn't fullfil the requirements).\n     * @param action - The EditAction to check.\n     * @returns Whether or not the Action can be used.\n     */\n    editActionDisabled(action: EditActionInternal<EntityType>): boolean {\n        return runInInjectionContext(this.injector, () => !action.enabled(this.entityPriorChanges));\n    }\n}"," \n<!-- eslint-disable angular/no-call-expression -->\n<!-- eslint-disable angular/cyclomatic-complexity -->\n<div *ngIf=\"!isLoaded && data.displayLoadingSpinner\" class=\"container\">\n    <br>\n    <mat-spinner></mat-spinner>\n    <br>\n</div>\n\n<div *ngIf=\"isLoaded\" class=\"container\">\n    <br>\n\n    <!------------>\n    <!-- Header -->\n    <!------------>\n    <div class=\"header\">\n        <div class=\"cancel-container\">\n            <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                <i class=\"fas fa-chevron-left\"></i>\n                {{data.editData.cancelButtonLabel}}\n                <i *ngIf=\"hasUnsavedChanges\" class=\"fas fa-warning\"></i>\n            </button>\n        </div>\n        <div class=\"actions-container\">\n            <button *ngIf=\"data.editData.actions.length\" type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n                {{data.editData.actionsLabel}}\n            </button>\n            <mat-menu #menu=\"matMenu\">\n                <button *ngFor=\"let action of data.editData.actions\" type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n                    {{action.displayName}}\n                </button>\n            </mat-menu>\n        \n            <button *ngIf=\"allowDelete\" type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n                {{data.editData.deleteButtonLabel}}\n            </button>\n        </div>\n    </div>\n\n    <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n\n    <!----------->\n    <!-- Input -->\n    <!----------->\n    <form>\n        <ngx-mat-entity-form [hideOmitForEdit]=\"true\" [isEntityReadOnly]=\"isEntityReadOnly\" [entity]=\"entity\" (selectedTabChange)=\"checkOffset()\" (formChange)=\"checkEntity()\">\n        </ngx-mat-entity-form>\n\n        <div class=\"bottom-row-container\">\n            <div>\n                <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                    [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n                    [matBadge]=\"validationErrors.length\"\n                    [matBadgeHidden]=\"!validationErrors.length\"\n                    (click)=\"edit()\"\n                >\n                    {{data.editData.confirmButtonLabel}}\n                </button>\n                <ngx-mat-entity-tooltip *ngIf=\"validationErrors.length\" [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n            </div>\n        </div>\n    </form>\n\n    <br>\n</div>"]}
|
|
342
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edit-page.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.ts","../../../../../projects/ngx-material-entity/src/components/edit-page/edit-page.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Location, NgFor, NgIf } from '@angular/common';\nimport { HttpClient } from '@angular/common/http';\nimport { Component, ElementRef, EnvironmentInjector, HostListener, Inject, InjectionToken, OnInit, Renderer2, runInInjectionContext } from '@angular/core';\nimport { MatBadgeModule } from '@angular/material/badge';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\nimport { ActivatedRoute } from '@angular/router';\nimport { firstValueFrom } from 'rxjs';\n\nimport { PageEditDataBuilder, PageEditDataInternal } from './page-edit-data.builder';\nimport { BaseEntityType, EntityClassNewable } from '../../classes/entity.model';\nimport { LodashUtilities } from '../../encapsulation/lodash.utilities';\nimport { getValidationErrorsTooltipContent } from '../../functions/get-validation-errors-tooltip-content.function.ts';\nimport { NGX_COMPLETE_GLOBAL_DEFAULT_VALUES, NgxGlobalDefaultValues } from '../../global-configuration-values';\nimport { EntityService } from '../../services/entity.service';\nimport { UnsavedChangesPage } from '../../services/unsaved-changes.guard';\nimport { EntityUtilities } from '../../utilities/entity.utilities';\nimport { ValidationError, ValidationUtilities } from '../../utilities/validation.utilities';\nimport { ConfirmDialogData } from '../confirm-dialog/confirm-dialog-data';\nimport { ConfirmDialogDataBuilder, ConfirmDialogDataInternal } from '../confirm-dialog/confirm-dialog-data.builder';\nimport { NgxMatEntityConfirmDialogComponent } from '../confirm-dialog/confirm-dialog.component';\nimport { NgxMatEntityFormComponent } from '../form/form.component';\nimport { EditActionInternal } from '../table/edit-dialog/edit-data.builder';\nimport { EditEntityData } from '../table/edit-dialog/edit-entity-data';\nimport { EditData } from '../table/table-data';\nimport { TooltipComponent } from '../tooltip/tooltip.component';\n\n/**\n * The data that needs to be provided for a route to be able to edit a entity.\n */\n// eslint-disable-next-line stylistic/max-len\nexport type PageEditData<EntityType extends BaseEntityType<EntityType>> = Omit<EditEntityData<EntityType>, 'entity' | 'EntityServiceClass'> & {\n    /**\n     * Whether or not to display a loading spinner while the entity for the page is loaded.\n     * @default true\n     */\n    displayLoadingSpinner?: boolean,\n    /**\n     * The data of the default edit page.\n     */\n    editData?: EditData<EntityType> & {\n        /**\n         * The data for the dialog when the user tries to leave the site with unsaved changes.\n         */\n        confirmUnsavedChangesDialogData?: ConfirmDialogData,\n        /**\n         * Whether or not leaving with unsaved changes should require a confirm dialog.\n         * @default true\n         */\n        unsavedChangesRequireConfirmDialog?: boolean\n    }\n};\n\n/**\n * The entity service that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, stylistic/max-len\nexport const NGX_EDIT_DATA_ENTITY_SERVICE: InjectionToken<EntityService<any>> = new InjectionToken<EntityService<any>>('NGX_EDIT_DATA_ENTITY_SERVICE');\n/**\n * The entity class that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any, stylistic/max-len\nexport const NGX_EDIT_DATA_ENTITY: InjectionToken<EntityClassNewable<any>> = new InjectionToken<EntityClassNewable<any>>('NGX_EDIT_DATA_ENTITY');\n/**\n * The configuration that needs to be provided in the providers array of the edit page route.\n */\n// eslint-disable-next-line typescript/no-explicit-any\nexport const NGX_EDIT_DATA: InjectionToken<PageEditData<any>> = new InjectionToken<PageEditData<any>>('NGX_EDIT_DATA');\n\n/**\n * A generic page that allows you to edit a specific entity.\n * For this to work you need to provide some data for the route.\n */\n@Component({\n    selector: 'ngx-mat-entity-edit-page',\n    templateUrl: './edit-page.component.html',\n    styleUrls: ['./edit-page.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatButtonModule,\n        MatProgressSpinnerModule,\n        MatMenuModule,\n        MatBadgeModule,\n        TooltipComponent,\n        NgxMatEntityFormComponent\n    ]\n})\nexport class NgxMatEntityEditPageComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit, UnsavedChangesPage {\n\n    /**\n     * Contains HelperMethods around handling Entities and their property-metadata.\n     */\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The entity that is being edited.\n     */\n    entity!: EntityType;\n    /**\n     * The entity before any changes have been made.\n     */\n    entityPriorChanges!: EntityType;\n\n    /**\n     * Configuration data for the component.\n     */\n    data!: PageEditDataInternal<EntityType>;\n\n    /**\n     * All validation errors of the entity.\n     */\n    validationErrors: ValidationError[] = [];\n    /**\n     * Whether or not the entity is valid.\n     */\n    isEntityValid: boolean = true;\n    /**\n     * Whether or not the entity is dirty.\n     */\n    isEntityDirty: boolean = false;\n    /**\n     * What to display inside the tooltip.\n     */\n    tooltipContent: string = '';\n\n    /**\n     * Whether or not the entity is readonly.\n     */\n    isEntityReadOnly!: boolean;\n    /**\n     * Whether or not the current user is allowed to delete the entity.\n     */\n    allowDelete!: boolean;\n\n    /**\n     * Whether or not the page has been loaded.\n     */\n    isLoaded: boolean = false;\n\n    private inConfirmNavigation: boolean = false;\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    get hasUnsavedChanges(): boolean {\n        return this.isEntityDirty && this.data.editData.unsavedChangesRequireConfirmDialog;\n    }\n\n    constructor(\n        private readonly dialog: MatDialog,\n        private readonly location: Location,\n        private readonly route: ActivatedRoute,\n        private readonly injector: EnvironmentInjector,\n        @Inject(NGX_EDIT_DATA_ENTITY_SERVICE)\n        readonly entityService: EntityService<EntityType>,\n        @Inject(NGX_EDIT_DATA_ENTITY)\n        private readonly EntityClass: EntityClassNewable<EntityType>,\n        @Inject(NGX_EDIT_DATA)\n        private readonly inputData: PageEditData<EntityType>,\n        private readonly http: HttpClient,\n        private readonly el: ElementRef,\n        private readonly renderer: Renderer2,\n        @Inject(NGX_COMPLETE_GLOBAL_DEFAULT_VALUES)\n        protected readonly globalConfig: NgxGlobalDefaultValues\n    ) { }\n\n    ngOnInit(): void {\n        this.data = new PageEditDataBuilder(this.inputData, this.globalConfig).getResult();\n        if (this.data == undefined) {\n            this.confirmNavigateBack();\n            throw new Error('No edit data was provided for \"NGX_EDIT_DATA\". You need to provide a value in your routes providers array.');\n        }\n\n        const id: EntityType[keyof EntityType] = this.route.snapshot.paramMap.get('id') as EntityType[keyof EntityType];\n        // eslint-disable-next-line promise/prefer-await-to-then\n        void this.entityService.findById(id).then(foundEntity => {\n            if (foundEntity == undefined) {\n                this.confirmNavigateBack();\n                throw new Error(`Could not find entity with id ${id}`);\n            }\n\n            this.entity = new this.EntityClass(foundEntity);\n            this.entityPriorChanges = LodashUtilities.cloneDeep(this.entity);\n\n            runInInjectionContext(this.injector, () => {\n                this.isEntityReadOnly = !this.data.allowUpdate(this.entityPriorChanges);\n                this.allowDelete = this.data.allowDelete(this.entityPriorChanges);\n            });\n            this.isLoaded = true;\n            setTimeout(() => this.checkOffset(), 1);\n            setTimeout(() => void this.checkIsEntityValid(), 1);\n        });\n    }\n\n    /**\n     * Checks if the bottom row should be displayed as fixed.\n     */\n    @HostListener('window:scroll')\n    checkOffset(): void {\n        const scrollY: number = window.scrollY;\n        const bottomRow: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row');\n        const bottomRowContainer: HTMLElement | null = (this.el.nativeElement as HTMLElement).querySelector('.bottom-row-container');\n\n        if (bottomRow && bottomRowContainer) {\n            const bottomRowContainerOffset: number = bottomRowContainer.offsetTop;\n            const windowHeight: number = window.innerHeight;\n\n            if (scrollY + windowHeight >= bottomRowContainerOffset) {\n                this.renderer.removeClass(bottomRow, 'fixed');\n            }\n            else {\n                this.renderer.addClass(bottomRow, 'fixed');\n            }\n        }\n    }\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    @HostListener('window:beforeunload')\n    canDeactivate(): boolean {\n        return !this.hasUnsavedChanges || this.inConfirmNavigation;\n    }\n\n    /**\n     * Checks if the entity has become invalid or dirty.\n     */\n    async checkEntity(): Promise<void> {\n        await this.checkIsEntityValid();\n        this.isEntityDirty = await EntityUtilities.isDirty(this.entity, this.entityPriorChanges, this.http);\n    }\n\n    private async checkIsEntityValid(): Promise<void> {\n        this.validationErrors = await ValidationUtilities.getEntityValidationErrors(this.entity, this.injector, 'update');\n        this.tooltipContent = runInInjectionContext(this.injector, () => getValidationErrorsTooltipContent(this.validationErrors));\n        this.isEntityValid = this.validationErrors.length === 0;\n    }\n\n    /**\n     * Tries to save the changes and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    async edit(): Promise<void> {\n        if (this.isEntityReadOnly || !this.isEntityValid || !this.isEntityDirty) {\n            return;\n        }\n        if (!this.data.editData.editRequiresConfirmDialog) {\n            await this.confirmEdit();\n            return;\n        }\n\n        // eslint-disable-next-line stylistic/max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmEditDialogData)\n            .withDefault('text', this.globalConfig.confirmSaveText)\n            .withDefault('confirmButtonLabel', this.globalConfig.saveLabel)\n            .withDefault('title', this.globalConfig.editLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        const res: boolean | undefined = await firstValueFrom(dialogRef.afterClosed());\n        if (res == true) {\n            await this.confirmEdit();\n        }\n    }\n\n    private async confirmEdit(): Promise<void> {\n        await this.entityService.update(this.entity, this.entityPriorChanges);\n        this.confirmNavigateBack();\n    }\n\n    /**\n     * Tries to delete the entity and close the dialog afterwards.\n     * Also handles the confirmation if required.\n     */\n    async delete(): Promise<void> {\n        if (!this.data.editData.deleteRequiresConfirmDialog) {\n            await this.confirmDelete();\n            return;\n        }\n\n        // eslint-disable-next-line stylistic/max-len\n        const dialogData: ConfirmDialogDataInternal = new ConfirmDialogDataBuilder(this.globalConfig, this.data.editData.confirmDeleteDialogData)\n            .withDefault('text', this.globalConfig.confirmDeleteText)\n            .withDefault('type', 'delete')\n            .withDefault('confirmButtonLabel', this.globalConfig.deleteLabel)\n            .withDefault('title', this.globalConfig.deleteLabel)\n            .getResult();\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: dialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        const res: boolean | undefined = await firstValueFrom(dialogRef.afterClosed());\n        if (res == true) {\n            await this.confirmDelete();\n        }\n    }\n\n    private async confirmDelete(): Promise<void> {\n        await this.entityService.delete(this.entityPriorChanges);\n        this.confirmNavigateBack();\n    }\n\n    /**\n     * Tries to navigate back.\n     */\n    async navigateBack(): Promise<void> {\n        if (!this.hasUnsavedChanges) {\n            this.confirmNavigateBack();\n            return;\n        }\n\n        const res: boolean = await this.openConfirmNavigationDialog();\n        if (res) {\n            this.confirmNavigateBack();\n        }\n    }\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    async openConfirmNavigationDialog(): Promise<boolean> {\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: this.data.editData.confirmUnsavedChangesDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        return await firstValueFrom(dialogRef.afterClosed()) ?? false;\n    }\n\n    private confirmNavigateBack(): void {\n        this.inConfirmNavigation = true;\n        EntityUtilities.resetChangesOnEntity(this.entity, this.entityPriorChanges);\n        this.location.back();\n    }\n\n    /**\n     * Runs the edit action on the entity.\n     * @param action - The action to run.\n     */\n    async runEditAction(action: EditActionInternal<EntityType>): Promise<void> {\n        const requireConfirmDialog: boolean = runInInjectionContext(this.injector, () => {\n            return action.requireConfirmDialog(this.entityPriorChanges);\n        });\n\n        if (!requireConfirmDialog) {\n            await this.confirmRunEditAction(action);\n            return;\n        }\n        const dialogRef: MatDialogRef<NgxMatEntityConfirmDialogComponent, boolean> = this.dialog.open(NgxMatEntityConfirmDialogComponent, {\n            data: action.confirmDialogData,\n            autoFocus: false,\n            restoreFocus: false\n        });\n        const res: boolean | undefined = await firstValueFrom(dialogRef.afterClosed());\n        if (res == true) {\n            await this.confirmRunEditAction(action);\n        }\n    }\n\n    private async confirmRunEditAction(action: EditActionInternal<EntityType>): Promise<void> {\n        await runInInjectionContext(this.injector, async () => {\n            await action.action(this.entity, this.entityPriorChanges);\n            await this.checkEntity();\n        });\n    }\n\n    /**\n     * Checks if an EditAction is disabled (e.g. Because the current entry doesn't fullfil the requirements).\n     * @param action - The EditAction to check.\n     * @returns Whether or not the Action can be used.\n     */\n    editActionDisabled(action: EditActionInternal<EntityType>): boolean {\n        return runInInjectionContext(this.injector, () => !action.enabled(this.entityPriorChanges));\n    }\n}"," \n<!-- eslint-disable angular/no-call-expression -->\n@if (!isLoaded && data.displayLoadingSpinner) {\n    <div class=\"container\">\n        <br>\n        <mat-spinner></mat-spinner>\n        <br>\n    </div>\n}\n\n@if (isLoaded) {\n    <div class=\"container\">\n        <br>\n    \n        <!------------>\n        <!-- Header -->\n        <!------------>\n        <div class=\"header\">\n            <div class=\"cancel-container\">\n                <button type=\"button\" mat-raised-button class=\"back-button\" tabindex=\"-1\" [class.unsavedChanges]=\"hasUnsavedChanges\" (click)=\"navigateBack()\">\n                    <i class=\"fas fa-chevron-left\"></i>\n                    {{data.editData.cancelButtonLabel}}\n                    @if (hasUnsavedChanges) {\n                        <i class=\"fas fa-warning\"></i>\n                    }\n                </button>\n            </div>\n            <div class=\"actions-container\">\n                @if (data.editData.actions.length) {\n                    <button type=\"button\" mat-raised-button [matMenuTriggerFor]=\"menu\">\n                        {{data.editData.actionsLabel}}\n                    </button>\n                }\n                <mat-menu #menu=\"matMenu\">\n                    @for (action of data.editData.actions; track $index) {\n                        <button type=\"button\" mat-menu-item [disabled]=\"editActionDisabled(action)\" (click)=\"runEditAction(action)\">\n                            {{action.displayName}}\n                        </button>\n                    }\n                </mat-menu>\n            \n                @if (allowDelete) {\n                    <button type=\"button\" mat-raised-button color=\"warn\" class=\"delete-button\" tabindex=\"-1\" (click)=\"delete()\">\n                        {{data.editData.deleteButtonLabel}}\n                    </button>\n                }\n            </div>\n        </div>\n    \n        <h1>{{data.editData.title(entityPriorChanges)}}</h1>\n    \n        <!----------->\n        <!-- Input -->\n        <!----------->\n        <form>\n            <ngx-mat-entity-form [hideOmitForEdit]=\"true\" [isEntityReadOnly]=\"isEntityReadOnly\" [entity]=\"entity\" (selectedTabChange)=\"checkOffset()\" (formChange)=\"checkEntity()\">\n            </ngx-mat-entity-form>\n    \n            <div class=\"bottom-row-container\">\n                <div class=\"bottom-row container\">\n                    <button type=\"submit\" mat-raised-button matBadgeColor=\"warn\"\n                        [disabled]=\"isEntityReadOnly || !isEntityValid || !isEntityDirty\"\n                        [matBadge]=\"validationErrors.length\"\n                        [matBadgeHidden]=\"!validationErrors.length\"\n                        (click)=\"edit()\"\n                    >\n                        {{data.editData.confirmButtonLabel}}\n                    </button>\n                    @if (validationErrors.length) {\n                        <ngx-mat-entity-tooltip [tooltipContent]=\"tooltipContent\"></ngx-mat-entity-tooltip>\n                    }\n                </div>\n            </div>\n        </form>\n        <br>\n    </div>\n}"]}
|
|
@@ -12,6 +12,7 @@ export class PageEditDataBuilder extends BaseBuilder {
|
|
|
12
12
|
const editData = new EditDataBuilder(this.globalConfig, data.editData)
|
|
13
13
|
.withDefault('cancelButtonLabel', this.globalConfig.backLabel)
|
|
14
14
|
.getResult();
|
|
15
|
+
// eslint-disable-next-line stylistic/max-len
|
|
15
16
|
const confirmUnsavedChangesDialogData = new ConfirmDialogDataBuilder(this.globalConfig, data.editData?.confirmUnsavedChangesDialogData)
|
|
16
17
|
.withDefault('title', this.globalConfig.confirmUnsavedChangesTitle)
|
|
17
18
|
.withDefault('text', this.globalConfig.confirmUnsavedChangesText)
|
|
@@ -29,4 +30,4 @@ export class PageEditDataBuilder extends BaseBuilder {
|
|
|
29
30
|
};
|
|
30
31
|
}
|
|
31
32
|
}
|
|
32
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
33
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZS1lZGl0LWRhdGEuYnVpbGRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC1tYXRlcmlhbC1lbnRpdHkvc3JjL2NvbXBvbmVudHMvZWRpdC1wYWdlL3BhZ2UtZWRpdC1kYXRhLmJ1aWxkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBRXpELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUVwRSxPQUFPLEVBQUUsd0JBQXdCLEVBQTZCLE1BQU0sK0NBQStDLENBQUM7QUFDcEgsT0FBTyxFQUFFLGVBQWUsRUFBb0IsTUFBTSx3Q0FBd0MsQ0FBQztBQWdCM0YsK0NBQStDO0FBQy9DLE1BQU0sT0FBTyxtQkFDVCxTQUFRLFdBQXVFO0lBRS9FLFlBQVksSUFBOEIsRUFBRSxZQUFvQztRQUM1RSxLQUFLLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFFRCwrQ0FBK0M7SUFDckMsZ0JBQWdCLENBQUMsSUFBOEI7UUFDckQsTUFBTSxRQUFRLEdBQWlDLElBQUksZUFBZSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQzthQUMvRixXQUFXLENBQUMsbUJBQW1CLEVBQUUsSUFBSSxDQUFDLFlBQVksQ0FBQyxTQUFTLENBQUM7YUFDN0QsU0FBUyxFQUFFLENBQUM7UUFFakIsNkNBQTZDO1FBQzdDLE1BQU0sK0JBQStCLEdBQThCLElBQUksd0JBQXdCLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRSxJQUFJLENBQUMsUUFBUSxFQUFFLCtCQUErQixDQUFDO2FBQzdKLFdBQVcsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLFlBQVksQ0FBQywwQkFBMEIsQ0FBQzthQUNsRSxXQUFXLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxZQUFZLENBQUMseUJBQXlCLENBQUM7YUFDaEUsV0FBVyxDQUFDLG9CQUFvQixFQUFFLElBQUksQ0FBQyxZQUFZLENBQUMsMEJBQTBCLENBQUM7YUFDL0UsU0FBUyxFQUFFLENBQUM7UUFFakIsT0FBTztZQUNILFFBQVEsRUFBRTtnQkFDTixHQUFHLFFBQVE7Z0JBQ1gsK0JBQStCLEVBQUUsK0JBQStCO2dCQUNoRSxrQ0FBa0MsRUFBRSxJQUFJLENBQUMsUUFBUSxFQUFFLGtDQUFrQyxJQUFJLElBQUk7YUFDaEc7WUFDRCxXQUFXLEVBQUUsSUFBSSxDQUFDLFdBQVcsSUFBSSxXQUFXO1lBQzVDLFdBQVcsRUFBRSxJQUFJLENBQUMsV0FBVyxJQUFJLFdBQVc7WUFDNUMscUJBQXFCLEVBQUUsSUFBSSxDQUFDLHFCQUFxQixJQUFJLElBQUk7U0FDNUQsQ0FBQztJQUNOLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBhZ2VFZGl0RGF0YSB9IGZyb20gJy4vZWRpdC1wYWdlLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBCYXNlQnVpbGRlciB9IGZyb20gJy4uLy4uL2NsYXNzZXMvYmFzZS5idWlsZGVyJztcbmltcG9ydCB7IEJhc2VFbnRpdHlUeXBlIH0gZnJvbSAnLi4vLi4vY2xhc3Nlcy9lbnRpdHkubW9kZWwnO1xuaW1wb3J0IHsgZGVmYXVsdFRydWUgfSBmcm9tICcuLi8uLi9mdW5jdGlvbnMvZGVmYXVsdC10cnVlLmZ1bmN0aW9uJztcbmltcG9ydCB7IE5neEdsb2JhbERlZmF1bHRWYWx1ZXMgfSBmcm9tICcuLi8uLi9nbG9iYWwtY29uZmlndXJhdGlvbi12YWx1ZXMnO1xuaW1wb3J0IHsgQ29uZmlybURpYWxvZ0RhdGFCdWlsZGVyLCBDb25maXJtRGlhbG9nRGF0YUludGVybmFsIH0gZnJvbSAnLi4vY29uZmlybS1kaWFsb2cvY29uZmlybS1kaWFsb2ctZGF0YS5idWlsZGVyJztcbmltcG9ydCB7IEVkaXREYXRhQnVpbGRlciwgRWRpdERhdGFJbnRlcm5hbCB9IGZyb20gJy4uL3RhYmxlL2VkaXQtZGlhbG9nL2VkaXQtZGF0YS5idWlsZGVyJztcbmltcG9ydCB7IEVkaXRFbnRpdHlEYXRhSW50ZXJuYWwgfSBmcm9tICcuLi90YWJsZS9lZGl0LWRpYWxvZy9lZGl0LWVudGl0eS5idWlsZGVyJztcblxuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGpzZG9jL3JlcXVpcmUtanNkb2MsIHN0eWxpc3RpYy9tYXgtbGVuXG5leHBvcnQgdHlwZSBQYWdlRWRpdERhdGFJbnRlcm5hbDxFbnRpdHlUeXBlIGV4dGVuZHMgQmFzZUVudGl0eVR5cGU8RW50aXR5VHlwZT4+ID0gT21pdDxFZGl0RW50aXR5RGF0YUludGVybmFsPEVudGl0eVR5cGU+LCAnZW50aXR5JyB8ICdFbnRpdHlTZXJ2aWNlQ2xhc3MnPiAmIHtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUganNkb2MvcmVxdWlyZS1qc2RvY1xuICAgIGRpc3BsYXlMb2FkaW5nU3Bpbm5lcjogYm9vbGVhbixcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUganNkb2MvcmVxdWlyZS1qc2RvY1xuICAgIGVkaXREYXRhOiBFZGl0RGF0YUludGVybmFsPEVudGl0eVR5cGU+ICYge1xuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUganNkb2MvcmVxdWlyZS1qc2RvY1xuICAgICAgICBjb25maXJtVW5zYXZlZENoYW5nZXNEaWFsb2dEYXRhOiBDb25maXJtRGlhbG9nRGF0YUludGVybmFsLFxuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUganNkb2MvcmVxdWlyZS1qc2RvY1xuICAgICAgICB1bnNhdmVkQ2hhbmdlc1JlcXVpcmVDb25maXJtRGlhbG9nOiBib29sZWFuXG4gICAgfVxufTtcblxuLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGpzZG9jL3JlcXVpcmUtanNkb2NcbmV4cG9ydCBjbGFzcyBQYWdlRWRpdERhdGFCdWlsZGVyPEVudGl0eVR5cGUgZXh0ZW5kcyBCYXNlRW50aXR5VHlwZTxFbnRpdHlUeXBlPj5cbiAgICBleHRlbmRzIEJhc2VCdWlsZGVyPFBhZ2VFZGl0RGF0YUludGVybmFsPEVudGl0eVR5cGU+LCBQYWdlRWRpdERhdGE8RW50aXR5VHlwZT4+IHtcblxuICAgIGNvbnN0cnVjdG9yKGRhdGE6IFBhZ2VFZGl0RGF0YTxFbnRpdHlUeXBlPiwgZ2xvYmFsQ29uZmlnOiBOZ3hHbG9iYWxEZWZhdWx0VmFsdWVzKSB7XG4gICAgICAgIHN1cGVyKGdsb2JhbENvbmZpZywgZGF0YSk7XG4gICAgfVxuXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGpzZG9jL3JlcXVpcmUtanNkb2NcbiAgICBwcm90ZWN0ZWQgZ2VuZXJhdGVCYXNlRGF0YShkYXRhOiBQYWdlRWRpdERhdGE8RW50aXR5VHlwZT4pOiBQYWdlRWRpdERhdGFJbnRlcm5hbDxFbnRpdHlUeXBlPiB7XG4gICAgICAgIGNvbnN0IGVkaXREYXRhOiBFZGl0RGF0YUludGVybmFsPEVudGl0eVR5cGU+ID0gbmV3IEVkaXREYXRhQnVpbGRlcih0aGlzLmdsb2JhbENvbmZpZywgZGF0YS5lZGl0RGF0YSlcbiAgICAgICAgICAgIC53aXRoRGVmYXVsdCgnY2FuY2VsQnV0dG9uTGFiZWwnLCB0aGlzLmdsb2JhbENvbmZpZy5iYWNrTGFiZWwpXG4gICAgICAgICAgICAuZ2V0UmVzdWx0KCk7XG5cbiAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIHN0eWxpc3RpYy9tYXgtbGVuXG4gICAgICAgIGNvbnN0IGNvbmZpcm1VbnNhdmVkQ2hhbmdlc0RpYWxvZ0RhdGE6IENvbmZpcm1EaWFsb2dEYXRhSW50ZXJuYWwgPSBuZXcgQ29uZmlybURpYWxvZ0RhdGFCdWlsZGVyKHRoaXMuZ2xvYmFsQ29uZmlnLCBkYXRhLmVkaXREYXRhPy5jb25maXJtVW5zYXZlZENoYW5nZXNEaWFsb2dEYXRhKVxuICAgICAgICAgICAgLndpdGhEZWZhdWx0KCd0aXRsZScsIHRoaXMuZ2xvYmFsQ29uZmlnLmNvbmZpcm1VbnNhdmVkQ2hhbmdlc1RpdGxlKVxuICAgICAgICAgICAgLndpdGhEZWZhdWx0KCd0ZXh0JywgdGhpcy5nbG9iYWxDb25maWcuY29uZmlybVVuc2F2ZWRDaGFuZ2VzVGV4dClcbiAgICAgICAgICAgIC53aXRoRGVmYXVsdCgnY29uZmlybUJ1dHRvbkxhYmVsJywgdGhpcy5nbG9iYWxDb25maWcuY29uZmlybVVuc2F2ZWRDaGFuZ2VzTGFiZWwpXG4gICAgICAgICAgICAuZ2V0UmVzdWx0KCk7XG5cbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIGVkaXREYXRhOiB7XG4gICAgICAgICAgICAgICAgLi4uZWRpdERhdGEsXG4gICAgICAgICAgICAgICAgY29uZmlybVVuc2F2ZWRDaGFuZ2VzRGlhbG9nRGF0YTogY29uZmlybVVuc2F2ZWRDaGFuZ2VzRGlhbG9nRGF0YSxcbiAgICAgICAgICAgICAgICB1bnNhdmVkQ2hhbmdlc1JlcXVpcmVDb25maXJtRGlhbG9nOiBkYXRhLmVkaXREYXRhPy51bnNhdmVkQ2hhbmdlc1JlcXVpcmVDb25maXJtRGlhbG9nID8/IHRydWVcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBhbGxvd1VwZGF0ZTogZGF0YS5hbGxvd1VwZGF0ZSA/PyBkZWZhdWx0VHJ1ZSxcbiAgICAgICAgICAgIGFsbG93RGVsZXRlOiBkYXRhLmFsbG93RGVsZXRlID8/IGRlZmF1bHRUcnVlLFxuICAgICAgICAgICAgZGlzcGxheUxvYWRpbmdTcGlubmVyOiBkYXRhLmRpc3BsYXlMb2FkaW5nU3Bpbm5lciA/PyB0cnVlXG4gICAgICAgIH07XG4gICAgfVxufSJdfQ==
|
|
@@ -69,18 +69,26 @@ export class NgxMatEntityFormComponent {
|
|
|
69
69
|
return this.isEntityReadOnly || metadata.isReadOnly(this.entity);
|
|
70
70
|
});
|
|
71
71
|
}
|
|
72
|
-
|
|
73
|
-
|
|
72
|
+
/**
|
|
73
|
+
* What happens when an input changes its value.
|
|
74
|
+
* This refreshes the entity tabs to check if there are now different inputs to display and emits the form change.
|
|
75
|
+
*/
|
|
76
|
+
inputChange() {
|
|
77
|
+
this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, this.hideOmitForCreate, this.hideOmitForEdit, this.additionalOmitKeys);
|
|
78
|
+
this.formChange.emit();
|
|
79
|
+
}
|
|
80
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.0", ngImport: i0, type: NgxMatEntityFormComponent, deps: [{ token: i0.EnvironmentInjector }], target: i0.ɵɵFactoryTarget.Component });
|
|
81
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.0", type: NgxMatEntityFormComponent, isStandalone: true, selector: "ngx-mat-entity-form", inputs: { entity: "entity", isEntityReadOnly: "isEntityReadOnly", hideOmitForCreate: "hideOmitForCreate", hideOmitForEdit: "hideOmitForEdit", additionalOmitKeys: "additionalOmitKeys" }, outputs: { formChange: "formChange", selectedTabChange: "selectedTabChange" }, ngImport: i0, template: " <!-- eslint-disable angular/no-call-expression -->\n@if (entityTabs.length > 1) {\n <mat-tab-group preserveContent (selectedTabChange)=\"selectedTabChange.emit()\">\n @for (tab of entityTabs; track $index) {\n <mat-tab [label]=\"tab.tabName\">\n @for (row of tab.rows; track $index) {\n <div class=\"row\">\n @for (key of row.keys; track key) {\n <ngx-mat-entity-input\n [entity]=\"entity\"\n [propertyKey]=\"key\"\n [hideOmitForEdit]=\"hideOmitForEdit\"\n [hideOmitForCreate]=\"hideOmitForCreate\"\n [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n [isReadOnly]=\"isReadOnly(key)\"\n (inputChangeEvent)=\"inputChange()\"\n >\n </ngx-mat-entity-input>\n }\n </div>\n }\n </mat-tab>\n }\n </mat-tab-group>\n}\n@else {\n @if (!entityTabs.length) {\n <span class=\"no-entity-tabs\">\n ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n </span>\n }\n @else {\n @for (row of entityTabs[0].rows; track $index) {\n <div class=\"row\">\n @for (key of row.keys; track key) {\n <ngx-mat-entity-input\n [entity]=\"entity\"\n [propertyKey]=\"key\"\n [hideOmitForEdit]=\"hideOmitForEdit\"\n [hideOmitForCreate]=\"hideOmitForCreate\"\n [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n [isReadOnly]=\"isReadOnly(key)\"\n (inputChangeEvent)=\"formChange.emit()\"\n >\n </ngx-mat-entity-input>\n }\n </div>\n }\n }\n}", styles: [".no-entity-tabs{padding:10px;background-color:red;color:#f5f5f5}\n"], dependencies: [{ kind: "ngmodule", type: MatTabsModule }, { kind: "component", type: i1.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i1.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "component", type: NgxMatEntityInputComponent, selector: "ngx-mat-entity-input", inputs: ["entity", "propertyKey", "getValidationErrorMessage", "hideOmitForCreate", "hideOmitForEdit", "validEmpty", "isReadOnly"], outputs: ["inputChangeEvent"] }] });
|
|
74
82
|
}
|
|
75
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
83
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.0", ngImport: i0, type: NgxMatEntityFormComponent, decorators: [{
|
|
76
84
|
type: Component,
|
|
77
85
|
args: [{ selector: 'ngx-mat-entity-form', standalone: true, imports: [
|
|
78
86
|
NgIf,
|
|
79
87
|
NgFor,
|
|
80
88
|
MatTabsModule,
|
|
81
89
|
NgxMatEntityInputComponent
|
|
82
|
-
], template: "
|
|
83
|
-
}], ctorParameters:
|
|
90
|
+
], template: " <!-- eslint-disable angular/no-call-expression -->\n@if (entityTabs.length > 1) {\n <mat-tab-group preserveContent (selectedTabChange)=\"selectedTabChange.emit()\">\n @for (tab of entityTabs; track $index) {\n <mat-tab [label]=\"tab.tabName\">\n @for (row of tab.rows; track $index) {\n <div class=\"row\">\n @for (key of row.keys; track key) {\n <ngx-mat-entity-input\n [entity]=\"entity\"\n [propertyKey]=\"key\"\n [hideOmitForEdit]=\"hideOmitForEdit\"\n [hideOmitForCreate]=\"hideOmitForCreate\"\n [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n [isReadOnly]=\"isReadOnly(key)\"\n (inputChangeEvent)=\"inputChange()\"\n >\n </ngx-mat-entity-input>\n }\n </div>\n }\n </mat-tab>\n }\n </mat-tab-group>\n}\n@else {\n @if (!entityTabs.length) {\n <span class=\"no-entity-tabs\">\n ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n </span>\n }\n @else {\n @for (row of entityTabs[0].rows; track $index) {\n <div class=\"row\">\n @for (key of row.keys; track key) {\n <ngx-mat-entity-input\n [entity]=\"entity\"\n [propertyKey]=\"key\"\n [hideOmitForEdit]=\"hideOmitForEdit\"\n [hideOmitForCreate]=\"hideOmitForCreate\"\n [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n [isReadOnly]=\"isReadOnly(key)\"\n (inputChangeEvent)=\"formChange.emit()\"\n >\n </ngx-mat-entity-input>\n }\n </div>\n }\n }\n}", styles: [".no-entity-tabs{padding:10px;background-color:red;color:#f5f5f5}\n"] }]
|
|
91
|
+
}], ctorParameters: () => [{ type: i0.EnvironmentInjector }], propDecorators: { entity: [{
|
|
84
92
|
type: Input,
|
|
85
93
|
args: [{ required: true }]
|
|
86
94
|
}], isEntityReadOnly: [{
|
|
@@ -96,4 +104,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.0", ngImpor
|
|
|
96
104
|
}], selectedTabChange: [{
|
|
97
105
|
type: Output
|
|
98
106
|
}] } });
|
|
99
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/form/form.component.ts","../../../../../projects/ngx-material-entity/src/components/form/form.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC3H,OAAO,EAAqB,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAI1E,OAAO,EAAa,eAAe,EAAE,MAAM,kCAAkC,CAAC;AAC9E,OAAO,EAAE,0BAA0B,EAAE,MAAM,0BAA0B,CAAC;;;AAEtE;;GAEG;AAaH,MAAM,OAAO,yBAAyB;IAyDL;IAvD7B,+CAA+C;IAC/C,eAAe,GAA2B,eAAe,CAAC;IAE1D;;OAEG;IAEH,MAAM,CAAc;IAEpB;;OAEG;IACH,UAAU,CAA2B;IAErC;;;OAGG;IAEH,gBAAgB,GAAY,KAAK,CAAC;IAElC;;;;OAIG;IAEH,iBAAiB,CAAW;IAE5B;;;;OAIG;IAEH,eAAe,CAAW;IAE1B;;OAEG;IAEH,kBAAkB,CAAwB;IAE1C;;OAEG;IAEH,UAAU,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAE1D;;OAEG;IAEH,iBAAiB,GAAoC,IAAI,YAAY,EAAqB,CAAC;IAE3F,YAA6B,QAA6B;QAA7B,aAAQ,GAAR,QAAQ,CAAqB;IAAI,CAAC;IAE/D,QAAQ;QACJ,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;IACvJ,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,GAAqB;QAC5B,OAAO,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YAC7C,MAAM,QAAQ,GAAyD,eAAe,CAAC,mBAAmB,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;YAC7H,IAAI,CAAC,QAAQ,EAAE;gBACX,MAAM,IAAI,KAAK,CAAC,sCAAsC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;aACzE;YACD,OAAO,IAAI,CAAC,gBAAgB,IAAI,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;IACP,CAAC;uGA5EQ,yBAAyB;2FAAzB,yBAAyB,wVCxBtC,soDAqCM,4HDnBE,IAAI,6FACJ,KAAK,kHACL,aAAa,yTACb,0BAA0B;;2FAGrB,yBAAyB;kBAZrC,SAAS;+BACI,qBAAqB,cAGnB,IAAI,WACP;wBACL,IAAI;wBACJ,KAAK;wBACL,aAAa;wBACb,0BAA0B;qBAC7B;0GAWD,MAAM;sBADL,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAazB,gBAAgB;sBADf,KAAK;gBASN,iBAAiB;sBADhB,KAAK;gBASN,eAAe;sBADd,KAAK;gBAON,kBAAkB;sBADjB,KAAK;gBAON,UAAU;sBADT,MAAM;gBAOP,iBAAiB;sBADhB,MAAM","sourcesContent":["import { NgFor, NgIf } from '@angular/common';\nimport { Component, EnvironmentInjector, EventEmitter, Input, OnInit, Output, runInInjectionContext } from '@angular/core';\nimport { MatTabChangeEvent, MatTabsModule } from '@angular/material/tabs';\n\nimport { BaseEntityType } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { NgxMatEntityInputComponent } from '../input/input.component';\n\n/**\n * A form component based on the ngx-material-entity framework.\n */\n@Component({\n    selector: 'ngx-mat-entity-form',\n    templateUrl: './form.component.html',\n    styleUrls: ['./form.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatTabsModule,\n        NgxMatEntityInputComponent\n    ]\n})\nexport class NgxMatEntityFormComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The entity that should be represented by the form.\n     */\n    @Input({ required: true })\n    entity!: EntityType;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * Whether or not the entity is readonly.\n     * @default false\n     */\n    @Input()\n    isEntityReadOnly: boolean = false;\n\n    /**\n     * Whether to hide a value if it is omitted for creation.\n     * Is used internally for the object property.\n     * @default false\n     */\n    @Input()\n    hideOmitForCreate?: boolean;\n\n    /**\n     * Whether to hide a value if it is omitted for editing.\n     * Is used internally for the object property.\n     * @default false\n     */\n    @Input()\n    hideOmitForEdit?: boolean;\n\n    /**\n     * Additional keys that should be omitted.\n     */\n    @Input()\n    additionalOmitKeys?: (keyof EntityType)[];\n\n    /**\n     * Fires whenever an input of the form changes.\n     */\n    @Output()\n    formChange: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Fires when the selected tab has been changed.\n     */\n    @Output()\n    selectedTabChange: EventEmitter<MatTabChangeEvent> = new EventEmitter<MatTabChangeEvent>();\n\n    constructor(private readonly injector: EnvironmentInjector) { }\n\n    ngOnInit(): void {\n        this.entityTabs = EntityUtilities.getEntityTabs(this.entity, this.injector, this.hideOmitForCreate, this.hideOmitForEdit, this.additionalOmitKeys);\n    }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> | undefined = EntityUtilities.getPropertyMetadata(this.entity, key);\n            if (!metadata) {\n                throw new Error(`No metadata was found for the key \"${String(key)}\"`);\n            }\n            return this.isEntityReadOnly || metadata.isReadOnly(this.entity);\n        });\n    }\n}","<!-- eslint-disable angular/cyclomatic-complexity -->\n<!-- eslint-disable angular/no-call-expression -->\n<mat-tab-group *ngIf=\"entityTabs.length > 1\" preserveContent (selectedTabChange)=\"selectedTabChange.emit()\">\n    <mat-tab *ngFor=\"let tab of entityTabs\" [label]=\"tab.tabName\">\n        <div *ngFor=\"let row of tab.rows\" class=\"row\">\n            <ngx-mat-entity-input\n                *ngFor=\"let key of row.keys\"\n                [entity]=\"entity\"\n                [propertyKey]=\"key\"\n                [hideOmitForEdit]=\"hideOmitForEdit\"\n                [hideOmitForCreate]=\"hideOmitForCreate\"\n                [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                [isReadOnly]=\"isReadOnly(key)\"\n                (inputChangeEvent)=\"formChange.emit()\"\n            >\n            </ngx-mat-entity-input>\n        </div>\n    </mat-tab>\n</mat-tab-group>\n\n<div *ngIf=\"entityTabs.length <= 1\">\n    <span *ngIf=\"!entityTabs.length\" class=\"no-entity-tabs\">\n        ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n    </span>\n    <div *ngFor=\"let row of entityTabs[0]?.rows\" class=\"row\">\n        <ngx-mat-entity-input\n            *ngFor=\"let key of row.keys\"\n            [entity]=\"entity\"\n            [propertyKey]=\"key\"\n            [hideOmitForEdit]=\"hideOmitForEdit\"\n            [hideOmitForCreate]=\"hideOmitForCreate\"\n            [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n            [isReadOnly]=\"isReadOnly(key)\"\n            (inputChangeEvent)=\"formChange.emit()\"\n        >\n        </ngx-mat-entity-input>\n    </div>\n</div>"]}
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-material-entity/src/components/form/form.component.ts","../../../../../projects/ngx-material-entity/src/components/form/form.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC3H,OAAO,EAAqB,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAI1E,OAAO,EAAa,eAAe,EAAE,MAAM,kCAAkC,CAAC;AAC9E,OAAO,EAAE,0BAA0B,EAAE,MAAM,0BAA0B,CAAC;;;AAEtE;;GAEG;AAaH,MAAM,OAAO,yBAAyB;IAyDL;IAvD7B,+CAA+C;IAC/C,eAAe,GAA2B,eAAe,CAAC;IAE1D;;OAEG;IAEH,MAAM,CAAc;IAEpB;;OAEG;IACH,UAAU,CAA2B;IAErC;;;OAGG;IAEH,gBAAgB,GAAY,KAAK,CAAC;IAElC;;;;OAIG;IAEH,iBAAiB,CAAW;IAE5B;;;;OAIG;IAEH,eAAe,CAAW;IAE1B;;OAEG;IAEH,kBAAkB,CAAwB;IAE1C;;OAEG;IAEM,UAAU,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAEnE;;OAEG;IAEM,iBAAiB,GAAoC,IAAI,YAAY,EAAqB,CAAC;IAEpG,YAA6B,QAA6B;QAA7B,aAAQ,GAAR,QAAQ,CAAqB;IAAI,CAAC;IAE/D,QAAQ;QACJ,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,aAAa,CAC3C,IAAI,CAAC,MAAM,EACX,IAAI,CAAC,QAAQ,EACb,IAAI,CAAC,iBAAiB,EACtB,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,kBAAkB,CAC1B,CAAC;IACN,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,GAAqB;QAC5B,OAAO,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,EAAE;YAC7C,MAAM,QAAQ,GAAyD,eAAe,CAAC,mBAAmB,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;YAC7H,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACZ,MAAM,IAAI,KAAK,CAAC,sCAAsC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YAC1E,CAAC;YACD,OAAO,IAAI,CAAC,gBAAgB,IAAI,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;OAGG;IACH,WAAW;QACP,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,aAAa,CAC3C,IAAI,CAAC,MAAM,EACX,IAAI,CAAC,QAAQ,EACb,IAAI,CAAC,iBAAiB,EACtB,IAAI,CAAC,eAAe,EACpB,IAAI,CAAC,kBAAkB,CAC1B,CAAC;QACF,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;IAC3B,CAAC;uGAjGQ,yBAAyB;2FAAzB,yBAAyB,wVCxBtC,ilEAiDC,2HD7BO,aAAa,4mBACb,0BAA0B;;2FAGrB,yBAAyB;kBAZrC,SAAS;+BACI,qBAAqB,cAGnB,IAAI,WACP;wBACL,IAAI;wBACJ,KAAK;wBACL,aAAa;wBACb,0BAA0B;qBAC7B;wFAWD,MAAM;sBADL,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAazB,gBAAgB;sBADf,KAAK;gBASN,iBAAiB;sBADhB,KAAK;gBASN,eAAe;sBADd,KAAK;gBAON,kBAAkB;sBADjB,KAAK;gBAOG,UAAU;sBADlB,MAAM;gBAOE,iBAAiB;sBADzB,MAAM","sourcesContent":["import { NgFor, NgIf } from '@angular/common';\nimport { Component, EnvironmentInjector, EventEmitter, Input, OnInit, Output, runInInjectionContext } from '@angular/core';\nimport { MatTabChangeEvent, MatTabsModule } from '@angular/material/tabs';\n\nimport { BaseEntityType } from '../../classes/entity.model';\nimport { PropertyDecoratorConfigInternal } from '../../decorators/base/property-decorator-internal.data';\nimport { EntityTab, EntityUtilities } from '../../utilities/entity.utilities';\nimport { NgxMatEntityInputComponent } from '../input/input.component';\n\n/**\n * A form component based on the ngx-material-entity framework.\n */\n@Component({\n    selector: 'ngx-mat-entity-form',\n    templateUrl: './form.component.html',\n    styleUrls: ['./form.component.scss'],\n    standalone: true,\n    imports: [\n        NgIf,\n        NgFor,\n        MatTabsModule,\n        NgxMatEntityInputComponent\n    ]\n})\nexport class NgxMatEntityFormComponent<EntityType extends BaseEntityType<EntityType>> implements OnInit {\n\n    // eslint-disable-next-line jsdoc/require-jsdoc\n    EntityUtilities: typeof EntityUtilities = EntityUtilities;\n\n    /**\n     * The entity that should be represented by the form.\n     */\n    @Input({ required: true })\n    entity!: EntityType;\n\n    /**\n     * The tabs to display.\n     */\n    entityTabs!: EntityTab<EntityType>[];\n\n    /**\n     * Whether or not the entity is readonly.\n     * @default false\n     */\n    @Input()\n    isEntityReadOnly: boolean = false;\n\n    /**\n     * Whether to hide a value if it is omitted for creation.\n     * Is used internally for the object property.\n     * @default false\n     */\n    @Input()\n    hideOmitForCreate?: boolean;\n\n    /**\n     * Whether to hide a value if it is omitted for editing.\n     * Is used internally for the object property.\n     * @default false\n     */\n    @Input()\n    hideOmitForEdit?: boolean;\n\n    /**\n     * Additional keys that should be omitted.\n     */\n    @Input()\n    additionalOmitKeys?: (keyof EntityType)[];\n\n    /**\n     * Fires whenever an input of the form changes.\n     */\n    @Output()\n    readonly formChange: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Fires when the selected tab has been changed.\n     */\n    @Output()\n    readonly selectedTabChange: EventEmitter<MatTabChangeEvent> = new EventEmitter<MatTabChangeEvent>();\n\n    constructor(private readonly injector: EnvironmentInjector) { }\n\n    ngOnInit(): void {\n        this.entityTabs = EntityUtilities.getEntityTabs(\n            this.entity,\n            this.injector,\n            this.hideOmitForCreate,\n            this.hideOmitForEdit,\n            this.additionalOmitKeys\n        );\n    }\n\n    /**\n     * Checks if the input with the given key is readonly.\n     * @param key - The key for the input to check.\n     * @returns Whether or not the input for the key is read only.\n     */\n    isReadOnly(key: keyof EntityType): boolean {\n        return runInInjectionContext(this.injector, () => {\n            const metadata: PropertyDecoratorConfigInternal<unknown> | undefined = EntityUtilities.getPropertyMetadata(this.entity, key);\n            if (!metadata) {\n                throw new Error(`No metadata was found for the key \"${String(key)}\"`);\n            }\n            return this.isEntityReadOnly || metadata.isReadOnly(this.entity);\n        });\n    }\n\n    /**\n     * What happens when an input changes its value.\n     * This refreshes the entity tabs to check if there are now different inputs to display and emits the form change.\n     */\n    inputChange(): void {\n        this.entityTabs = EntityUtilities.getEntityTabs(\n            this.entity,\n            this.injector,\n            this.hideOmitForCreate,\n            this.hideOmitForEdit,\n            this.additionalOmitKeys\n        );\n        this.formChange.emit();\n    }\n}"," <!-- eslint-disable angular/no-call-expression -->\n@if (entityTabs.length > 1) {\n    <mat-tab-group preserveContent (selectedTabChange)=\"selectedTabChange.emit()\">\n        @for (tab of entityTabs; track $index) {\n            <mat-tab [label]=\"tab.tabName\">\n                @for (row of tab.rows; track $index) {\n                    <div class=\"row\">\n                        @for (key of row.keys; track key) {\n                            <ngx-mat-entity-input\n                                [entity]=\"entity\"\n                                [propertyKey]=\"key\"\n                                [hideOmitForEdit]=\"hideOmitForEdit\"\n                                [hideOmitForCreate]=\"hideOmitForCreate\"\n                                [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                                [isReadOnly]=\"isReadOnly(key)\"\n                                (inputChangeEvent)=\"inputChange()\"\n                            >\n                            </ngx-mat-entity-input>\n                        }\n                    </div>\n                }\n            </mat-tab>\n        }\n    </mat-tab-group>\n}\n@else {\n    @if (!entityTabs.length) {\n        <span class=\"no-entity-tabs\">\n            ERROR: No Inputs. Did you correctly assign all values in the model constructor?\n        </span>\n    }\n    @else {\n        @for (row of entityTabs[0].rows; track $index) {\n            <div class=\"row\">\n                @for (key of row.keys; track key) {\n                    <ngx-mat-entity-input\n                        [entity]=\"entity\"\n                        [propertyKey]=\"key\"\n                        [hideOmitForEdit]=\"hideOmitForEdit\"\n                        [hideOmitForCreate]=\"hideOmitForCreate\"\n                        [class]=\"EntityUtilities.getWidthClasses(entity, key)\"\n                        [isReadOnly]=\"isReadOnly(key)\"\n                        (inputChangeEvent)=\"formChange.emit()\"\n                    >\n                    </ngx-mat-entity-input>\n                }\n            </div>\n        }\n    }\n}"]}
|