@wertzui/ngx-restworld-client 1.0.2 → 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (31) hide show
  1. package/bundles/wertzui-ngx-restworld-client.umd.js +1101 -301
  2. package/bundles/wertzui-ngx-restworld-client.umd.js.map +1 -1
  3. package/esm2015/lib/models/problem-details.js +2 -2
  4. package/esm2015/lib/pipes/as.pipe.js +25 -0
  5. package/esm2015/lib/pipes/safe-url.pipe.js +23 -0
  6. package/esm2015/lib/restworld-client.module.js +68 -8
  7. package/esm2015/lib/services/avatar-generator.js +2 -2
  8. package/esm2015/lib/services/form.service.js +69 -0
  9. package/esm2015/lib/services/restworld-client-collection.js +1 -1
  10. package/esm2015/lib/services/restworld-client.js +46 -8
  11. package/esm2015/lib/views/restworld-edit-form/restworld-edit-form.component.js +238 -0
  12. package/esm2015/lib/views/restworld-edit-view/restworld-edit-view.component.js +209 -143
  13. package/esm2015/lib/views/restworld-file-view/restworld-file-view.component.js +63 -0
  14. package/esm2015/lib/views/restworld-image-view/restworld-image-view.component.js +107 -0
  15. package/esm2015/lib/views/restworld-list-view/restworld-list-view.component.js +64 -37
  16. package/esm2015/public-api.js +7 -1
  17. package/fesm2015/wertzui-ngx-restworld-client.js +880 -220
  18. package/fesm2015/wertzui-ngx-restworld-client.js.map +1 -1
  19. package/lib/models/problem-details.d.ts +4 -8
  20. package/lib/pipes/as.pipe.d.ts +18 -0
  21. package/lib/pipes/safe-url.pipe.d.ts +10 -0
  22. package/lib/restworld-client.module.d.ts +35 -21
  23. package/lib/services/form.service.d.ts +14 -0
  24. package/lib/services/restworld-client.d.ts +4 -1
  25. package/lib/views/restworld-edit-form/restworld-edit-form.component.d.ts +69 -0
  26. package/lib/views/restworld-edit-view/restworld-edit-view.component.d.ts +53 -32
  27. package/lib/views/restworld-file-view/restworld-file-view.component.d.ts +21 -0
  28. package/lib/views/restworld-image-view/restworld-image-view.component.d.ts +38 -0
  29. package/lib/views/restworld-list-view/restworld-list-view.component.d.ts +20 -8
  30. package/package.json +29 -8
  31. package/public-api.d.ts +6 -0
@@ -1,7 +1,7 @@
1
1
  import { __awaiter } from "tslib";
2
2
  import { Component, Input } from '@angular/core';
3
- import * as _ from 'lodash';
4
- import { FormControl, FormGroup } from '@angular/forms';
3
+ import { PropertyType } from '@wertzui/ngx-hal-client';
4
+ import { FormGroup } from '@angular/forms';
5
5
  import { ProblemDetails } from '../../models/problem-details';
6
6
  import { ContentChild } from '@angular/core';
7
7
  import * as i0 from "@angular/core";
@@ -9,34 +9,41 @@ import * as i1 from "../../services/restworld-client-collection";
9
9
  import * as i2 from "primeng/api";
10
10
  import * as i3 from "@angular/common";
11
11
  import * as i4 from "@angular/router";
12
- import * as i5 from "primeng/tabview";
13
- import * as i6 from "primeng/skeleton";
14
- import * as i7 from "primeng/calendar";
15
- import * as i8 from "primeng/inputnumber";
16
- import * as i9 from "primeng/checkbox";
12
+ import * as i5 from "../../services/form.service";
13
+ import * as i6 from "ngx-valdemort";
14
+ import * as i7 from "primeng/tabview";
15
+ import * as i8 from "primeng/skeleton";
16
+ import * as i9 from "../restworld-edit-form/restworld-edit-form.component";
17
17
  import * as i10 from "primeng/progressspinner";
18
18
  import * as i11 from "primeng/toast";
19
19
  import * as i12 from "primeng/confirmdialog";
20
20
  import * as i13 from "@angular/forms";
21
- import * as i14 from "primeng/inputtext";
22
- import * as i15 from "primeng/button";
23
- import * as i16 from "primeng/ripple";
21
+ import * as i14 from "primeng/button";
22
+ import * as i15 from "primeng/ripple";
24
23
  export class RESTworldEditViewComponent {
25
- constructor(_clients, _confirmationService, _messageService, _location, _router) {
24
+ constructor(_clients, _confirmationService, _messageService, _location, _router, _formService, valdemortConfig) {
26
25
  this._clients = _clients;
27
26
  this._confirmationService = _confirmationService;
28
27
  this._messageService = _messageService;
29
28
  this._location = _location;
30
29
  this._router = _router;
31
- this._properties = [];
32
- this._formGroup = new FormGroup({});
30
+ this._formService = _formService;
31
+ this._templates = {};
32
+ this._formTabs = {};
33
33
  this.isLoading = false;
34
+ valdemortConfig.errorClasses = 'p-error text-sm';
34
35
  }
35
- get properties() {
36
- return this._properties;
36
+ get PropertyType() {
37
+ return PropertyType;
37
38
  }
38
- get formGroup() {
39
- return this._formGroup;
39
+ get templates() {
40
+ return this._templates;
41
+ }
42
+ get isLoadingForTheFirstTime() {
43
+ return Object.keys(this.templates).length === 0 && this.isLoading;
44
+ }
45
+ get formTabs() {
46
+ return this._formTabs;
40
47
  }
41
48
  set apiName(value) {
42
49
  this._apiName = value;
@@ -73,39 +80,68 @@ export class RESTworldEditViewComponent {
73
80
  const length = (_b = (_a = this.resource) === null || _a === void 0 ? void 0 : _a._links["delete"]) === null || _b === void 0 ? void 0 : _b.length;
74
81
  return length !== undefined && length > 0;
75
82
  }
76
- get dateFormat() {
77
- return new Date(3333, 10, 22)
78
- .toLocaleDateString()
79
- .replace("22", "dd")
80
- .replace("11", "mm")
81
- .replace("3333", "yy")
82
- .replace("33", "y");
83
+ canSubmit(templateName) {
84
+ const form = this.formTabs[templateName];
85
+ return form && form.valid;
86
+ }
87
+ getTooltip(resource, keysToExclude) {
88
+ const tooltip = Object.entries(resource)
89
+ .filter(([key]) => !(key.startsWith('_') || ['createdAt', 'createdBy', 'lastChangedAt', 'lastChangedBy', 'timestamp'].includes(key) || (keysToExclude === null || keysToExclude === void 0 ? void 0 : keysToExclude.includes(key))))
90
+ .reduce((prev, [key, value], index) => `${prev}${index === 0 ? '' : '\n'}${key}: ${RESTworldEditViewComponent.jsonStringifyWithElipsis(value)}`, '');
91
+ return tooltip;
92
+ }
93
+ static jsonStringifyWithElipsis(value) {
94
+ const maxLength = 200;
95
+ const end = 10;
96
+ const start = maxLength - end - 2;
97
+ const json = JSON.stringify(value);
98
+ const shortened = json.length > maxLength ? json.substring(0, start) + '…' + json.substring(json.length - end) : json;
99
+ return shortened;
83
100
  }
84
101
  getClient() {
85
102
  if (!this.apiName)
86
103
  throw new Error('Cannot get a client, because the apiName is not set.');
87
104
  return this._clients.getClient(this.apiName);
88
105
  }
89
- save() {
106
+ submit(templateName, template, formValue) {
90
107
  return __awaiter(this, void 0, void 0, function* () {
91
- if (!this.apiName || !this.uri || !this.resource)
92
- return;
93
- Object.assign(this.resource, this.formGroup.value);
94
- const selfHrefBeforeSave = this.resource._links.self[0].href;
95
108
  this.isLoading = true;
96
- const response = yield this.getClient().save(this.resource);
97
- this.isLoading = false;
98
- if (!response.ok || ProblemDetails.isProblemDetails(response.body)) {
99
- const message = response.status === 409 /* Conflict */ ? 'Someone else modified the resource. Try reloading it and apply your changes again.' : 'Error while saving the resource.';
100
- this._messageService.add({ severity: 'error', summary: 'Error', detail: message, data: response });
101
- }
102
- else {
103
- const selfHrefAfterSave = this.resource._links.self[0].href;
104
- setTimeout(() => this._messageService.add({ severity: 'success', summary: 'Saved', detail: 'The resource has been saved.' }), 100);
105
- if (selfHrefBeforeSave !== selfHrefAfterSave) {
106
- this._router.navigate(['/edit', this.apiName, selfHrefAfterSave]);
109
+ try {
110
+ const targetBeforeSave = template.target;
111
+ const response = yield this.getClient().submit(template, formValue);
112
+ if (!response.ok) {
113
+ let summary = 'Error';
114
+ let detail = 'Error while saving the resource.';
115
+ if (ProblemDetails.isProblemDetails(response.body)) {
116
+ const problemDetails = response.body;
117
+ summary = problemDetails.title || summary;
118
+ detail = problemDetails.detail || detail;
119
+ // display validation errors
120
+ if (problemDetails['errors']) {
121
+ const form = this.formTabs[templateName];
122
+ for (const [key, errorsForKey] of Object.entries(problemDetails['errors'])) {
123
+ const path = key.split(/\.|\[/).map(e => e.replace("]", ""));
124
+ const formControl = path.reduce((control, pathElement) => (control instanceof FormGroup ? control.controls[pathElement] : control) || control, form);
125
+ formControl.setErrors({ remote: errorsForKey });
126
+ }
127
+ }
128
+ }
129
+ this._messageService.add({ severity: 'error', summary: summary, detail: detail, data: response, life: 10000 });
107
130
  }
131
+ else {
132
+ const responseResource = response.body;
133
+ const targetAfterSave = responseResource._templates[templateName].target;
134
+ setTimeout(() => this._messageService.add({ severity: 'success', summary: 'Saved', detail: 'The resource has been saved.' }), 100);
135
+ if (targetBeforeSave !== targetAfterSave) {
136
+ this._router.navigate(['/edit', this.apiName, responseResource._links.self[0].href]);
137
+ }
138
+ }
139
+ }
140
+ catch (e) {
141
+ this._messageService.add({ severity: 'error', summary: 'Error', detail: `An unknown error occurred. ${JSON.stringify(e)}`, life: 10000 });
142
+ console.log(e);
108
143
  }
144
+ this.isLoading = false;
109
145
  });
110
146
  }
111
147
  showDeleteConfirmatioModal() {
@@ -120,7 +156,7 @@ export class RESTworldEditViewComponent {
120
156
  return __awaiter(this, void 0, void 0, function* () {
121
157
  if (!this.apiName || !this.uri || !this.resource)
122
158
  return;
123
- Object.assign(this.resource, this.formGroup.value);
159
+ Object.assign(this.resource, this.formTabs.value);
124
160
  yield this.getClient().delete(this.resource);
125
161
  setTimeout(() => this._messageService.add({ severity: 'success', summary: 'Deleted', detail: 'The resource has been deleted.' }), 100);
126
162
  this._location.back();
@@ -137,133 +173,163 @@ export class RESTworldEditViewComponent {
137
173
  }
138
174
  else {
139
175
  this._resource = response.body;
140
- this._properties = RESTworldEditViewComponent.createProperyInfos(this._resource);
141
- this._formGroup = RESTworldEditViewComponent.createFormGroup(this._properties);
176
+ this._templates = yield this.getAllTemplates(this._resource);
177
+ this._formTabs = this._formService.createFormGroupsFromTemplates(this._templates);
142
178
  }
143
179
  this.isLoading = false;
144
180
  });
145
181
  }
146
- static createFormGroup(properties) {
147
- const controls = Object.fromEntries(properties.map(p => [
148
- p.field,
149
- p.type === 'object' && p.children ? RESTworldEditViewComponent.createFormGroup(p.children) : new FormControl(p.value)
150
- ]));
151
- const formGroup = new FormGroup(controls);
152
- return formGroup;
182
+ setInitialSelectedOptionsElementsForTemplates(templates) {
183
+ return __awaiter(this, void 0, void 0, function* () {
184
+ return Promise.all(Object.values(templates)
185
+ .map(template => this.setInitialSelectedOptionsElementsForTemplate(template)));
186
+ });
153
187
  }
154
- static createProperyInfos(resource) {
155
- if (!resource)
156
- return [];
157
- // Get all distinct properties from all rows
158
- // We look at all rows to eliminate possible undefined values
159
- const properties = Object.entries(resource)
160
- .filter(p => p[0] !== '_links' &&
161
- p[0] !== '_embedded' &&
162
- p[0] !== 'id' &&
163
- p[0] !== 'timestamp');
164
- // Check if the rows are entities with change tracking
165
- const withoutChangeTrackingProperties = properties.filter(p => p[0] !== 'createdAt' &&
166
- p[0] !== 'createdBy' &&
167
- p[0] !== 'lastChangedAt' &&
168
- p[0] !== 'lastChangedBy');
169
- const hasChangeTrackingProperties = withoutChangeTrackingProperties.length < properties.length;
170
- // First the id, then all other properties
171
- const sortedProperties = [];
172
- if (Object.hasOwnProperty('id'))
173
- sortedProperties.push(['id', resource['id']]);
174
- sortedProperties.push(...withoutChangeTrackingProperties);
175
- // And change tracking properties at the end
176
- if (hasChangeTrackingProperties) {
177
- sortedProperties.push(['createdAt', resource['createdAt']]);
178
- sortedProperties.push(['createdBy', resource['createdBy']]);
179
- sortedProperties.push(['lastChangedAt', resource['lastChangedAt']]);
180
- sortedProperties.push(['lastChangedBy', resource['lastChangedBy']]);
181
- }
182
- const propertyDescriptions = sortedProperties
183
- .map(p => ({
184
- field: p[0],
185
- value: p[1],
186
- label: RESTworldEditViewComponent.toTitleCase(p[0]),
187
- type: RESTworldEditViewComponent.getColumnType(p[0], p[1]),
188
- isReadOnly: RESTworldEditViewComponent.getIsReadOnly(p[0])
189
- }));
190
- for (var description of propertyDescriptions) {
191
- if (description.type === 'object') {
192
- const children = RESTworldEditViewComponent.createProperyInfos(description.value);
193
- children.forEach(d => description.field + '.' + d.field);
194
- description.children = children;
195
- }
196
- }
197
- return propertyDescriptions;
188
+ imageChanged(formControl, event) {
189
+ const file = event.files[0];
190
+ console.log(file);
191
+ const reader = new FileReader();
192
+ reader.onload = () => {
193
+ const uri = reader.result;
194
+ console.log(uri);
195
+ formControl.setValue(uri);
196
+ };
197
+ reader.readAsDataURL(file);
198
198
  }
199
- static getIsReadOnly(field) {
200
- return field === 'id' || field === 'createdAt' || field === 'createdBy' || field === 'lastChangedAt' || field === 'lastChangedBy';
199
+ setInitialSelectedOptionsElementsForTemplate(template) {
200
+ return __awaiter(this, void 0, void 0, function* () {
201
+ return Promise.all(template.properties
202
+ .filter(property => { var _a, _b; return (_b = (_a = property === null || property === void 0 ? void 0 : property.options) === null || _a === void 0 ? void 0 : _a.link) === null || _b === void 0 ? void 0 : _b.href; })
203
+ .map(property => this.setInitialSelectedOptionsElementForProperty(property)));
204
+ });
201
205
  }
202
- static getColumnType(field, value) {
203
- if (value === null || value === undefined)
204
- return 'text';
205
- if (_.isNumber(value))
206
- return 'numeric';
207
- if (_.isDate(value))
208
- return 'date';
209
- if (_.isString(value)) {
210
- return 'text';
211
- }
212
- if (_.isBoolean(value))
213
- return 'boolean';
214
- if (_.isObject(value))
215
- return 'object';
216
- if (_.isArray(value))
217
- return 'array';
218
- return 'text';
206
+ setInitialSelectedOptionsElementForProperty(property) {
207
+ var _a;
208
+ return __awaiter(this, void 0, void 0, function* () {
209
+ const options = property === null || property === void 0 ? void 0 : property.options;
210
+ if (!((_a = options === null || options === void 0 ? void 0 : options.link) === null || _a === void 0 ? void 0 : _a.href))
211
+ return;
212
+ const templatedUri = options.link.href;
213
+ const filter = `${options.valueField} eq ${property.value}`;
214
+ const response = yield this.getClient().getListByUri(templatedUri, { $filter: filter, $top: 10 });
215
+ if (!response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body) {
216
+ const message = `An error occurred while getting the filtered items.`;
217
+ this._messageService.add({ severity: 'error', summary: 'Error', detail: message, data: response });
218
+ return;
219
+ }
220
+ const items = response.body._embedded.items;
221
+ options.inline = items;
222
+ });
219
223
  }
220
- static toTitleCase(anyCase) {
221
- return anyCase
222
- .replace(/(_)+/g, ' ') // underscore to whitespace
223
- .replace(/([a-z])([A-Z][a-z])/g, "$1 $2") // insert space before each new word if there is none
224
- .replace(/([A-Z][a-z])([A-Z])/g, "$1 $2") // insert space after each word if there is none
225
- .replace(/([a-z])([A-Z]+[a-z])/g, "$1 $2") // insert space after single letter word if there is none
226
- .replace(/([A-Z]+)([A-Z][a-z][a-z])/g, "$1 $2") // insert space before single letter word if there is none
227
- .replace(/([a-z]+)([A-Z0-9]+)/g, "$1 $2") // insert space after numbers
228
- .replace(/^./, (match) => match.toUpperCase()); // change first letter to be upper case
224
+ getAllTemplates(resource) {
225
+ return __awaiter(this, void 0, void 0, function* () {
226
+ const formResponses = yield this.getClient().getAllForms(resource);
227
+ const failedResponses = formResponses.filter(response => !response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body);
228
+ if (failedResponses.length !== 0) {
229
+ for (const response of failedResponses) {
230
+ this._messageService.add({ severity: 'error', summary: 'Error', detail: 'Error while loading the resource from the API.', data: response });
231
+ }
232
+ return Promise.resolve({});
233
+ }
234
+ const formTemplates = Object.assign({}, ...formResponses.map(response => response.body._templates));
235
+ yield this.setInitialSelectedOptionsElementsForTemplates(formTemplates);
236
+ return formTemplates;
237
+ });
229
238
  }
230
239
  }
231
- RESTworldEditViewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.11", ngImport: i0, type: RESTworldEditViewComponent, deps: [{ token: i1.RESTworldClientCollection }, { token: i2.ConfirmationService }, { token: i2.MessageService }, { token: i3.Location }, { token: i4.Router }], target: i0.ɵɵFactoryTarget.Component });
232
- RESTworldEditViewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.11", type: RESTworldEditViewComponent, selector: "restworld-edit-view", inputs: { apiName: "apiName", uri: "uri" }, queries: [{ propertyName: "visualTabRef", first: true, predicate: ["visualTab"], descendants: true }, { propertyName: "formRef", first: true, predicate: ["form"], descendants: true }, { propertyName: "formRowRef", first: true, predicate: ["formRow"], descendants: true }, { propertyName: "formLabelRef", first: true, predicate: ["formLabel"], descendants: true }, { propertyName: "formInputRef", first: true, predicate: ["formInput"], descendants: true }, { propertyName: "visualTabAdditionalRef", first: true, predicate: ["visualTabAdditional"], descendants: true }, { propertyName: "rawTabRef", first: true, predicate: ["rawTab"], descendants: true }, { propertyName: "buttonsRef", first: true, predicate: ["buttonsRef"], descendants: true }], ngImport: i0, template: "<div class=\"grid\">\r\n <div class=\"col\">\r\n <h1>Edit resource</h1>\r\n </div>\r\n</div>\r\n\r\n<form [formGroup]=\"formGroup\" (ngSubmit)=\"save()\">\r\n <p-tabView>\r\n <p-tabPanel header=\"Visual\" [disabled]=\"!resource || isLoading\">\r\n <ng-template #defaultVisualTab>\r\n <ng-container *ngIf=\"!resource && isLoading\">\r\n <div *ngFor=\"let i of [1, 2, 3, 4, 5]\" class=\"grid field\">\r\n <p-skeleton class=\"col-12 mb-2 md:col-2 md:mb-0\" height=\"39px\"></p-skeleton>\r\n <div class=\"col-12 md:col-10\">\r\n <p-skeleton class=\"w-full\" height=\"39px\"></p-skeleton>\r\n </div>\r\n </div>\r\n </ng-container>\r\n\r\n <div *ngIf=\"resource\" class=\"blockable-container\">\r\n <div class=\"blockable-element\">\r\n <ng-template #defaultForm let-properties=\"properties\" let-formGroup=\"formGroup\">\r\n <div [formGroup]=\"formGroup\">\r\n <div *ngFor=\"let property of properties\" class=\"grid field\">\r\n <ng-template #defaultFormRow>\r\n <ng-template #defaultFormLabel>\r\n <label [attr.for]=\"property.field\" class=\"col-12 mb-2 md:col-2 md:mb-0\" [class.p-disabled]=\"property.isReadOnly\" [class.hasChildren]=\"property.children?.length > 0\">{{property.label}}</label>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"formLabelRef || defaultFormLabel; context:{ $implicit: property }\"></ng-container>\r\n <ng-template #defaultFormInput let-property let-formGroup=\"formGroup\">\r\n <div [ngSwitch]=\"property.type\" class=\"col-12 md:col-10\">\r\n <input *ngSwitchCase=\"'text'\" [formControlName]=\"property.field\" [id]=\"property.field\" type=\"text\" pInputText class=\"w-full\" [class.p-disabled]=\"property.isReadOnly\">\r\n <p-calendar *ngSwitchCase=\"'date'\" [formControlName]=\"property.field\" [id]=\"property.field\" [dateFormat]=\"dateFormat\" [showTime]=\"true\" [showIcon]=\"true\" styleClass=\"w-full\" [class.p-disabled]=\"property.isReadOnly\"></p-calendar>\r\n <p-inputNumber *ngSwitchCase=\"'numeric'\" [formControlName]=\"property.field\" [id]=\"property.field\" mode=\"decimal\" [showButtons]=\"!property.isReadOnly\" class=\"w-full\" styleClass=\"w-full\" [class.p-disabled]=\"property.isReadOnly\"></p-inputNumber>\r\n <p-checkbox *ngSwitchCase=\"'boolean'\" [formControl]=\"formGroup.controls[property.field]\" [id]=\"property.field\" [readonly]=\"property.isReadOnly\"></p-checkbox>\r\n <div *ngSwitchCase=\"'object'\">\r\n <ng-container *ngTemplateOutlet=\"formRef || defaultForm; context:{ properties: property.children, formGroup: formGroup.controls[property.field] }\"></ng-container>\r\n </div>\r\n </div>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"formInputRef || defaultFormInput; context:{ $implicit: property, formGroup: formGroup }\"></ng-container>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"formRowRef || defaultFormRow; context:{ $implicit: property }\"></ng-container>\r\n </div>\r\n </div>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"formRef || defaultForm; context: { properties: properties, formGroup: formGroup }\"></ng-container>\r\n </div>\r\n <div class=\"blockable-overlay\" *ngIf=\"isLoading\">\r\n <p-progressSpinner></p-progressSpinner>\r\n </div>\r\n </div>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"visualTabRef || defaultVisualTab\"></ng-container>\r\n <ng-container *ngTemplateOutlet=\"visualTabAdditionalRef || null; context:{ $implicit: resource, properties: properties, isLoading: isLoading }\"></ng-container>\r\n </p-tabPanel>\r\n\r\n <p-tabPanel header=\"Raw\" [disabled]=\"!resource || isLoading\">\r\n <ng-template #defaultRawTab>\r\n <pre>{{resource | json}}</pre>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"rawTabRef || defaultRawTab\"></ng-container>\r\n </p-tabPanel>\r\n </p-tabView>\r\n\r\n <div class=\"grid\">\r\n <div class=\"col\">\r\n <div class=\"flex justify-content-end w-full\">\r\n <ng-template #defaultButtons>\r\n <button pButton pRipple type=\"submit\" label=\"Save\" icon=\"far fa-save\" class=\"mx-2 p-button-success\" [disabled]=\"!resource || isLoading || !canSave\"></button>\r\n <button pButton pRipple type=\"button\" label=\"Reload\" icon=\"fas fa-redo\" class=\"mx-2 p-button-info\" (click)=\"load()\" [disabled]=\"isLoading\"></button>\r\n <button pButton pRipple type=\"button\" label=\"Delete\" icon=\"far fa-trash-alt\" class=\"mx-2 p-button-danger\" (click)=\"showDeleteConfirmatioModal()\" [disabled]=\"!resource || isLoading || !canDelete\"></button>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"buttonsRef || defaultButtons\"></ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n</form>\r\n\r\n<p-toast></p-toast>\r\n<p-confirmDialog></p-confirmDialog>\r\n", styles: [".blockable-container{display:grid;place-items:center;grid-template-areas:\"inner\"}.blockable-element{grid-area:inner;width:100%}.blockable-overlay{grid-area:inner;height:100%;width:100%;background-color:#0006;display:flex;align-items:center;justify-content:center;z-index:1}.field.grid>label.hasChildren{border-right:1px solid rgba(0,0,0,.1)}\n"], components: [{ type: i5.TabView, selector: "p-tabView", inputs: ["orientation", "style", "styleClass", "controlClose", "scrollable", "activeIndex"], outputs: ["onChange", "onClose", "activeIndexChange"] }, { type: i5.TabPanel, selector: "p-tabPanel", inputs: ["closable", "headerStyle", "headerStyleClass", "cache", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "selected", "disabled", "header", "leftIcon", "rightIcon"] }, { type: i6.Skeleton, selector: "p-skeleton", inputs: ["styleClass", "style", "shape", "animation", "borderRadius", "size", "width", "height"] }, { type: i7.Calendar, selector: "p-calendar", inputs: ["style", "styleClass", "inputStyle", "inputId", "name", "inputStyleClass", "placeholder", "ariaLabelledBy", "iconAriaLabel", "disabled", "dateFormat", "multipleSeparator", "rangeSeparator", "inline", "showOtherMonths", "selectOtherMonths", "showIcon", "icon", "appendTo", "readonlyInput", "shortYearCutoff", "monthNavigator", "yearNavigator", "hourFormat", "timeOnly", "stepHour", "stepMinute", "stepSecond", "showSeconds", "required", "showOnFocus", "showWeek", "dataType", "selectionMode", "maxDateCount", "showButtonBar", "todayButtonStyleClass", "clearButtonStyleClass", "autoZIndex", "baseZIndex", "panelStyleClass", "panelStyle", "keepInvalid", "hideOnDateTimeSelect", "numberOfMonths", "view", "touchUI", "timeSeparator", "focusTrap", "firstDayOfWeek", "showTransitionOptions", "hideTransitionOptions", "tabindex", "defaultDate", "minDate", "maxDate", "disabledDates", "disabledDays", "yearRange", "showTime", "locale"], outputs: ["onFocus", "onBlur", "onClose", "onSelect", "onInput", "onTodayClick", "onClearClick", "onMonthChange", "onYearChange", "onClickOutside", "onShow"] }, { type: i8.InputNumber, selector: "p-inputNumber", inputs: ["showButtons", "format", "buttonLayout", "inputId", "styleClass", "style", "placeholder", "size", "maxlength", "tabindex", "title", "ariaLabel", "ariaRequired", "name", "required", "autocomplete", "min", "max", "incrementButtonClass", "decrementButtonClass", "incrementButtonIcon", "decrementButtonIcon", "readonly", "step", "allowEmpty", "locale", "localeMatcher", "mode", "currency", "currencyDisplay", "useGrouping", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "inputStyle", "inputStyleClass", "disabled"], outputs: ["onInput", "onFocus", "onBlur", "onKeyDown"] }, { type: i9.Checkbox, selector: "p-checkbox", inputs: ["value", "name", "disabled", "binary", "label", "ariaLabelledBy", "ariaLabel", "tabindex", "inputId", "style", "styleClass", "labelStyleClass", "formControl", "checkboxIcon", "readonly", "required", "trueValue", "falseValue"], outputs: ["onChange"] }, { type: i10.ProgressSpinner, selector: "p-progressSpinner", inputs: ["style", "styleClass", "strokeWidth", "fill", "animationDuration"] }, { type: i11.Toast, selector: "p-toast", inputs: ["key", "autoZIndex", "baseZIndex", "style", "styleClass", "position", "preventOpenDuplicates", "preventDuplicates", "showTransformOptions", "hideTransformOptions", "showTransitionOptions", "hideTransitionOptions", "breakpoints"], outputs: ["onClose"] }, { type: i12.ConfirmDialog, selector: "p-confirmDialog", inputs: ["header", "icon", "message", "style", "styleClass", "maskStyleClass", "acceptIcon", "acceptLabel", "acceptAriaLabel", "acceptVisible", "rejectIcon", "rejectLabel", "rejectAriaLabel", "rejectVisible", "acceptButtonStyleClass", "rejectButtonStyleClass", "closeOnEscape", "dismissableMask", "blockScroll", "rtl", "closable", "appendTo", "key", "autoZIndex", "baseZIndex", "transitionOptions", "focusTrap", "defaultFocus", "breakpoints", "visible", "position"], outputs: ["onHide"] }], directives: [{ type: i13.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i13.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i13.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i3.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i3.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i13.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i14.InputText, selector: "[pInputText]" }, { type: i13.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i13.FormControlName, selector: "[formControlName]", inputs: ["disabled", "formControlName", "ngModel"], outputs: ["ngModelChange"] }, { type: i13.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i15.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading"] }, { type: i16.Ripple, selector: "[pRipple]" }], pipes: { "json": i3.JsonPipe } });
240
+ RESTworldEditViewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.11", ngImport: i0, type: RESTworldEditViewComponent, deps: [{ token: i1.RESTworldClientCollection }, { token: i2.ConfirmationService }, { token: i2.MessageService }, { token: i3.Location }, { token: i4.Router }, { token: i5.FormService }, { token: i6.ValdemortConfig }], target: i0.ɵɵFactoryTarget.Component });
241
+ RESTworldEditViewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.11", type: RESTworldEditViewComponent, selector: "rw-edit", inputs: { apiName: "apiName", uri: "uri" }, queries: [{ propertyName: "extraTabsRef", first: true, predicate: ["extraTabs"], descendants: true }, { propertyName: "buttonsRef", first: true, predicate: ["buttons"], descendants: true }, { propertyName: "inputOptionsSingleRef", first: true, predicate: ["inputOptionsSingle"], descendants: true }, { propertyName: "inputOptionsMultipleRef", first: true, predicate: ["inputOptionsMultiple"], descendants: true }, { propertyName: "inputOptionsRef", first: true, predicate: ["inputOptions"], descendants: true }, { propertyName: "inputHiddenRef", first: true, predicate: ["inputHidden"], descendants: true }, { propertyName: "inputTextRef", first: true, predicate: ["inputText"], descendants: true }, { propertyName: "inputTextareaRef", first: true, predicate: ["inputTextarea"], descendants: true }, { propertyName: "inputSearchRef", first: true, predicate: ["inputSearch"], descendants: true }, { propertyName: "inputTelRef", first: true, predicate: ["inputTel"], descendants: true }, { propertyName: "inputUrlRef", first: true, predicate: ["inputUrl"], descendants: true }, { propertyName: "inputEmailRef", first: true, predicate: ["inputEmail"], descendants: true }, { propertyName: "inputPasswordRef", first: true, predicate: ["inputPassword"], descendants: true }, { propertyName: "inputDateRef", first: true, predicate: ["inputDate"], descendants: true }, { propertyName: "inputMonthRef", first: true, predicate: ["inputMonth"], descendants: true }, { propertyName: "inputWeekRef", first: true, predicate: ["inputWeek"], descendants: true }, { propertyName: "inputTimeRef", first: true, predicate: ["inputTime"], descendants: true }, { propertyName: "inputDatetimeLocalRef", first: true, predicate: ["inputDatetimeLocal"], descendants: true }, { propertyName: "inputNumberRef", first: true, predicate: ["inputNumber"], descendants: true }, { propertyName: "inputRangeRef", first: true, predicate: ["inputRange"], descendants: true }, { propertyName: "inputColorRef", first: true, predicate: ["inputColor"], descendants: true }, { propertyName: "inputBoolRef", first: true, predicate: ["inputBool"], descendants: true }, { propertyName: "inputDatetimeOffsetRef", first: true, predicate: ["inputDatetimeOffset"], descendants: true }, { propertyName: "inputDurationRef", first: true, predicate: ["inputDuration"], descendants: true }, { propertyName: "inputImageRef", first: true, predicate: ["inputImage"], descendants: true }, { propertyName: "inputFileRef", first: true, predicate: ["inputFile"], descendants: true }, { propertyName: "inputDefaultRef", first: true, predicate: ["inputDefault"], descendants: true }], ngImport: i0, template: "<div class=\"grid\">\r\n <div class=\"col\">\r\n <h1>Edit resource</h1>\r\n </div>\r\n</div>\r\n\r\n<p-tabView>\r\n\r\n <p-tabPanel *ngIf=\"isLoadingForTheFirstTime\" header=\"Loading\">\r\n <div *ngFor=\"let i of [1, 2, 3, 4, 5]\" class=\"grid field\">\r\n <p-skeleton class=\"col-12 mb-2 md:col-2 md:mb-0\" height=\"39px\"></p-skeleton>\r\n <div class=\"col-12 md:col-10\">\r\n <p-skeleton class=\"w-full\" height=\"39px\"></p-skeleton>\r\n </div>\r\n </div>\r\n <div class=\"grid\">\r\n <div class=\"col\">\r\n <div class=\"flex justify-content-end w-full\">\r\n <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n </div>\r\n </div>\r\n </div>\r\n </p-tabPanel>\r\n\r\n <p-tabPanel *ngFor=\"let item of templates | keyvalue\" [header]=\"item.value.title || item.key\" [disabled]=\"isLoading\">\r\n <form [formGroup]=\"formTabs[item.key]\" (ngSubmit)=\"submit(item.key, item.value, formTabs[item.key].value)\">\r\n <div class=\"blockable-container\">\r\n <div class=\"blockable-element\">\r\n <rw-form [formGroup]=\"formTabs[item.key]\" [template]=\"item.value\" [apiName]=\"apiName\"></rw-form>\r\n </div>\r\n <div class=\"blockable-overlay\" *ngIf=\"isLoading\">\r\n <p-progressSpinner></p-progressSpinner>\r\n </div>\r\n </div>\r\n\r\n <div class=\"grid\">\r\n <div class=\"col\">\r\n <div class=\"flex justify-content-end w-full\">\r\n <ng-template #defaultButtons>\r\n <button pButton pRipple type=\"submit\" label=\"Save\" icon=\"far fa-save\" class=\"mx-2 p-button-success\" [disabled]=\"isLoading || !canSubmit(item.key)\"></button>\r\n <button pButton pRipple type=\"button\" label=\"Reload\" icon=\"fas fa-redo\" class=\"mx-2 p-button-info\" (click)=\"load()\" [disabled]=\"isLoading\"></button>\r\n <button pButton pRipple type=\"button\" label=\"Delete\" icon=\"far fa-trash-alt\" class=\"ml-2 p-button-danger\" (click)=\"showDeleteConfirmatioModal()\" [disabled]=\"!resource || isLoading || !canDelete\"></button>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"buttonsRef || defaultButtons\"></ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n </form>\r\n </p-tabPanel>\r\n\r\n <ng-template #defaultExtraTabs>\r\n </ng-template>\r\n <ng-container *ngTemplateOutlet=\"extraTabsRef || defaultExtraTabs\"></ng-container>\r\n\r\n</p-tabView>\r\n\r\n<p-toast></p-toast>\r\n<p-confirmDialog></p-confirmDialog>\r\n", styles: ["::ng-deep .p-tooltip{max-width:-webkit-fit-content!important;max-width:-moz-fit-content!important;max-width:fit-content!important}.blockable-container{display:grid;place-items:center;grid-template-areas:\"inner\"}.blockable-element{grid-area:inner;width:100%}.blockable-overlay{grid-area:inner;height:100%;width:100%;background-color:#0006;display:flex;align-items:center;justify-content:center;z-index:1}.field.grid>label.hasChildren{border-right:1px solid rgba(0,0,0,.1)}\n"], components: [{ type: i7.TabView, selector: "p-tabView", inputs: ["orientation", "style", "styleClass", "controlClose", "scrollable", "activeIndex"], outputs: ["onChange", "onClose", "activeIndexChange"] }, { type: i7.TabPanel, selector: "p-tabPanel", inputs: ["closable", "headerStyle", "headerStyleClass", "cache", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "selected", "disabled", "header", "leftIcon", "rightIcon"] }, { type: i8.Skeleton, selector: "p-skeleton", inputs: ["styleClass", "style", "shape", "animation", "borderRadius", "size", "width", "height"] }, { type: i9.RestworldEditFormComponent, selector: "rw-form", inputs: ["formGroup", "template", "apiName"] }, { type: i10.ProgressSpinner, selector: "p-progressSpinner", inputs: ["style", "styleClass", "strokeWidth", "fill", "animationDuration"] }, { type: i11.Toast, selector: "p-toast", inputs: ["key", "autoZIndex", "baseZIndex", "style", "styleClass", "position", "preventOpenDuplicates", "preventDuplicates", "showTransformOptions", "hideTransformOptions", "showTransitionOptions", "hideTransitionOptions", "breakpoints"], outputs: ["onClose"] }, { type: i12.ConfirmDialog, selector: "p-confirmDialog", inputs: ["header", "icon", "message", "style", "styleClass", "maskStyleClass", "acceptIcon", "acceptLabel", "acceptAriaLabel", "acceptVisible", "rejectIcon", "rejectLabel", "rejectAriaLabel", "rejectVisible", "acceptButtonStyleClass", "rejectButtonStyleClass", "closeOnEscape", "dismissableMask", "blockScroll", "rtl", "closable", "appendTo", "key", "autoZIndex", "baseZIndex", "transitionOptions", "focusTrap", "defaultFocus", "breakpoints", "visible", "position"], outputs: ["onHide"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i13.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i13.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i13.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i14.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading"] }, { type: i15.Ripple, selector: "[pRipple]" }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "keyvalue": i3.KeyValuePipe } });
233
242
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.11", ngImport: i0, type: RESTworldEditViewComponent, decorators: [{
234
243
  type: Component,
235
244
  args: [{
236
- selector: 'restworld-edit-view',
245
+ selector: 'rw-edit',
237
246
  templateUrl: './restworld-edit-view.component.html',
238
247
  styleUrls: ['./restworld-edit-view.component.css']
239
248
  }]
240
- }], ctorParameters: function () { return [{ type: i1.RESTworldClientCollection }, { type: i2.ConfirmationService }, { type: i2.MessageService }, { type: i3.Location }, { type: i4.Router }]; }, propDecorators: { apiName: [{
249
+ }], ctorParameters: function () { return [{ type: i1.RESTworldClientCollection }, { type: i2.ConfirmationService }, { type: i2.MessageService }, { type: i3.Location }, { type: i4.Router }, { type: i5.FormService }, { type: i6.ValdemortConfig }]; }, propDecorators: { apiName: [{
241
250
  type: Input
242
251
  }], uri: [{
243
252
  type: Input
244
- }], visualTabRef: [{
253
+ }], extraTabsRef: [{
245
254
  type: ContentChild,
246
- args: ['visualTab', { static: false }]
247
- }], formRef: [{
255
+ args: ['extraTabs', { static: false }]
256
+ }], buttonsRef: [{
248
257
  type: ContentChild,
249
- args: ['form', { static: false }]
250
- }], formRowRef: [{
258
+ args: ['buttons', { static: false }]
259
+ }], inputOptionsSingleRef: [{
251
260
  type: ContentChild,
252
- args: ['formRow', { static: false }]
253
- }], formLabelRef: [{
261
+ args: ['inputOptionsSingle', { static: false }]
262
+ }], inputOptionsMultipleRef: [{
254
263
  type: ContentChild,
255
- args: ['formLabel', { static: false }]
256
- }], formInputRef: [{
264
+ args: ['inputOptionsMultiple', { static: false }]
265
+ }], inputOptionsRef: [{
257
266
  type: ContentChild,
258
- args: ['formInput', { static: false }]
259
- }], visualTabAdditionalRef: [{
267
+ args: ['inputOptions', { static: false }]
268
+ }], inputHiddenRef: [{
260
269
  type: ContentChild,
261
- args: ['visualTabAdditional', { static: false }]
262
- }], rawTabRef: [{
270
+ args: ['inputHidden', { static: false }]
271
+ }], inputTextRef: [{
263
272
  type: ContentChild,
264
- args: ['rawTab', { static: false }]
265
- }], buttonsRef: [{
273
+ args: ['inputText', { static: false }]
274
+ }], inputTextareaRef: [{
275
+ type: ContentChild,
276
+ args: ['inputTextarea', { static: false }]
277
+ }], inputSearchRef: [{
278
+ type: ContentChild,
279
+ args: ['inputSearch', { static: false }]
280
+ }], inputTelRef: [{
281
+ type: ContentChild,
282
+ args: ['inputTel', { static: false }]
283
+ }], inputUrlRef: [{
284
+ type: ContentChild,
285
+ args: ['inputUrl', { static: false }]
286
+ }], inputEmailRef: [{
287
+ type: ContentChild,
288
+ args: ['inputEmail', { static: false }]
289
+ }], inputPasswordRef: [{
290
+ type: ContentChild,
291
+ args: ['inputPassword', { static: false }]
292
+ }], inputDateRef: [{
293
+ type: ContentChild,
294
+ args: ['inputDate', { static: false }]
295
+ }], inputMonthRef: [{
296
+ type: ContentChild,
297
+ args: ['inputMonth', { static: false }]
298
+ }], inputWeekRef: [{
299
+ type: ContentChild,
300
+ args: ['inputWeek', { static: false }]
301
+ }], inputTimeRef: [{
302
+ type: ContentChild,
303
+ args: ['inputTime', { static: false }]
304
+ }], inputDatetimeLocalRef: [{
305
+ type: ContentChild,
306
+ args: ['inputDatetimeLocal', { static: false }]
307
+ }], inputNumberRef: [{
308
+ type: ContentChild,
309
+ args: ['inputNumber', { static: false }]
310
+ }], inputRangeRef: [{
311
+ type: ContentChild,
312
+ args: ['inputRange', { static: false }]
313
+ }], inputColorRef: [{
314
+ type: ContentChild,
315
+ args: ['inputColor', { static: false }]
316
+ }], inputBoolRef: [{
317
+ type: ContentChild,
318
+ args: ['inputBool', { static: false }]
319
+ }], inputDatetimeOffsetRef: [{
320
+ type: ContentChild,
321
+ args: ['inputDatetimeOffset', { static: false }]
322
+ }], inputDurationRef: [{
323
+ type: ContentChild,
324
+ args: ['inputDuration', { static: false }]
325
+ }], inputImageRef: [{
326
+ type: ContentChild,
327
+ args: ['inputImage', { static: false }]
328
+ }], inputFileRef: [{
329
+ type: ContentChild,
330
+ args: ['inputFile', { static: false }]
331
+ }], inputDefaultRef: [{
266
332
  type: ContentChild,
267
- args: ['buttonsRef', { static: false }]
333
+ args: ['inputDefault', { static: false }]
268
334
  }] } });
269
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"restworld-edit-view.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-restworld-client/src/lib/views/restworld-edit-view/restworld-edit-view.component.ts","../../../../../../projects/ngx-restworld-client/src/lib/views/restworld-edit-view/restworld-edit-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit } from '@angular/core';\r\nimport { Resource } from '@wertzui/ngx-hal-client';\r\nimport { RESTworldClient } from '../../services/restworld-client';\r\nimport * as _ from 'lodash'\r\nimport { RESTworldClientCollection } from '../../services/restworld-client-collection';\r\nimport { FormControl, FormGroup } from '@angular/forms';\r\nimport { ConfirmationService, MessageService } from 'primeng/api';\r\nimport { Location } from '@angular/common';\r\nimport { Router } from '@angular/router';\r\nimport { ProblemDetails } from '../../models/problem-details';\r\nimport { HttpStatusCode } from '@angular/common/http';\r\nimport { ContentChild } from '@angular/core';\r\nimport { TemplateRef } from '@angular/core';\r\n\r\ninterface PropertyDescription {\r\n  field: string,\r\n  value: any,\r\n  label: string,\r\n  type: 'text' | 'numeric' | 'date' | 'boolean' | 'object' | 'array',\r\n  isReadOnly: boolean,\r\n  children?: PropertyDescription[]\r\n}\r\n\r\n@Component({\r\n  selector: 'restworld-edit-view',\r\n  templateUrl: './restworld-edit-view.component.html',\r\n  styleUrls: ['./restworld-edit-view.component.css']\r\n})\r\nexport class RESTworldEditViewComponent {\r\n  public get properties() {\r\n    return this._properties;\r\n  }\r\n  private _properties: PropertyDescription[] = [];\r\n\r\n  public get formGroup() {\r\n    return this._formGroup;\r\n  }\r\n  private _formGroup: FormGroup = new FormGroup({});\r\n\r\n  @Input()\r\n  public set apiName(value: string | undefined) {\r\n    this._apiName = value;\r\n    this.load();\r\n  }\r\n  public get apiName(): string | undefined {\r\n    return this._apiName;\r\n  }\r\n  private _apiName?: string;\r\n  //@Input()\r\n  //public set rel(value: string | undefined) {\r\n  //  this._rel = value;\r\n  //}\r\n  //public get rel(): string | undefined {\r\n  //  return this._rel;\r\n  //}\r\n  //private _rel?: string;\r\n  @Input()\r\n  public set uri(value: string | undefined) {\r\n    this._uri = value;\r\n    this.load();\r\n  }\r\n  public get uri(): string | undefined {\r\n    return this._uri;\r\n  }\r\n  private _uri?: string;\r\n\r\n  public get resource() {\r\n    return this._resource;\r\n  }\r\n  private _resource?: Resource;\r\n  public isLoading = false;\r\n  public get canSave() {\r\n    const length = this.resource?._links[\"save\"]?.length;\r\n    return length !== undefined && length > 0;\r\n  }\r\n  public get canDelete() {\r\n    const length = this.resource?._links[\"delete\"]?.length;\r\n    return length !== undefined && length > 0;\r\n  }\r\n\r\n  public get dateFormat(): string {\r\n    return new Date(3333, 10, 22)\r\n      .toLocaleDateString()\r\n      .replace(\"22\", \"dd\")\r\n      .replace(\"11\", \"mm\")\r\n      .replace(\"3333\", \"yy\")\r\n      .replace(\"33\", \"y\");\r\n  }\r\n\r\n  @ContentChild('visualTab', { static: false })\r\n  visualTabRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('form', { static: false })\r\n  formRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('formRow', { static: false })\r\n  formRowRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('formLabel', { static: false })\r\n  formLabelRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('formInput', { static: false })\r\n  formInputRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('visualTabAdditional', { static: false })\r\n  visualTabAdditionalRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('rawTab', { static: false })\r\n  rawTabRef?: TemplateRef<any>;\r\n\r\n  @ContentChild('buttonsRef', { static: false })\r\n  buttonsRef?: TemplateRef<any>;\r\n\r\n  constructor(\r\n    private _clients: RESTworldClientCollection,\r\n    private _confirmationService: ConfirmationService,\r\n    private _messageService: MessageService,\r\n    private _location: Location,\r\n    private _router: Router) {\r\n\r\n  }\r\n\r\n  private getClient(): RESTworldClient {\r\n    if (!this.apiName)\r\n      throw new Error('Cannot get a client, because the apiName is not set.');\r\n\r\n    return this._clients.getClient(this.apiName);\r\n  }\r\n\r\n  public async save(): Promise<void> {\r\n    if (!this.apiName || !this.uri || !this.resource)\r\n      return;\r\n\r\n    Object.assign(this.resource, this.formGroup.value);\r\n    const selfHrefBeforeSave = this.resource._links.self[0].href;\r\n\r\n    this.isLoading = true;\r\n    const response = await this.getClient().save(this.resource);\r\n    this.isLoading = false;\r\n\r\n    if (!response.ok || ProblemDetails.isProblemDetails(response.body)) {\r\n      const message = response.status === HttpStatusCode.Conflict ? 'Someone else modified the resource. Try reloading it and apply your changes again.' : 'Error while saving the resource.';\r\n      this._messageService.add({ severity: 'error', summary: 'Error', detail: message, data: response });\r\n    }\r\n    else {\r\n      const selfHrefAfterSave = this.resource._links.self[0].href;\r\n\r\n      setTimeout(() =>\r\n        this._messageService.add({ severity: 'success', summary: 'Saved', detail: 'The resource has been saved.' }),\r\n        100);\r\n\r\n      if (selfHrefBeforeSave !== selfHrefAfterSave) {\r\n        this._router.navigate(['/edit', this.apiName, selfHrefAfterSave]);\r\n      }\r\n    }\r\n\r\n  }\r\n\r\n  public showDeleteConfirmatioModal() {\r\n    this._confirmationService.confirm({\r\n      message: 'Do you really want to delete this resource?',\r\n      header: 'Confirm delete',\r\n      icon: 'far fa-trash-alt',\r\n      accept: () => this.delete()\r\n    });\r\n  }\r\n\r\n  public async delete(): Promise<void> {\r\n    if (!this.apiName || !this.uri || !this.resource)\r\n      return;\r\n\r\n    Object.assign(this.resource, this.formGroup.value);\r\n\r\n    await this.getClient().delete(this.resource);\r\n    setTimeout(() =>\r\n      this._messageService.add({ severity: 'success', summary: 'Deleted', detail: 'The resource has been deleted.' }),\r\n      100);\r\n\r\n    this._location.back();\r\n  }\r\n\r\n  public async load(): Promise<void> {\r\n    if (!this.apiName || !this.uri)\r\n      return;\r\n\r\n    this.isLoading = true;\r\n\r\n    const response = await this.getClient().getSingle(this.uri);\r\n    if (!response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body) {\r\n      this._messageService.add({ severity: 'error', summary: 'Error', detail: 'Error while loading the resource from the API.', data: response })\r\n    }\r\n    else {\r\n      this._resource = response.body;\r\n      this._properties = RESTworldEditViewComponent.createProperyInfos(this._resource);\r\n      this._formGroup = RESTworldEditViewComponent.createFormGroup(this._properties);\r\n    }\r\n\r\n    this.isLoading = false;\r\n  }\r\n\r\n  static createFormGroup(properties: PropertyDescription[]): FormGroup {\r\n    const controls = Object.fromEntries(properties.map(p => [\r\n      p.field,\r\n      p.type === 'object' && p.children ? RESTworldEditViewComponent.createFormGroup(p.children) : new FormControl(p.value)\r\n    ]));\r\n    const formGroup = new FormGroup(controls);\r\n    return formGroup;\r\n  }\r\n\r\n  private static createProperyInfos(resource: any): PropertyDescription[] {\r\n    if (!resource)\r\n      return [];\r\n\r\n    // Get all distinct properties from all rows\r\n    // We look at all rows to eliminate possible undefined values\r\n    const properties = Object.entries(resource)\r\n      .filter(p =>\r\n        p[0] !== '_links' &&\r\n        p[0] !== '_embedded' &&\r\n        p[0] !== 'id' &&\r\n        p[0] !== 'timestamp');\r\n\r\n    // Check if the rows are entities with change tracking\r\n    const withoutChangeTrackingProperties = properties.filter(p =>\r\n      p[0] !== 'createdAt' &&\r\n      p[0] !== 'createdBy' &&\r\n      p[0] !== 'lastChangedAt' &&\r\n      p[0] !== 'lastChangedBy');\r\n    const hasChangeTrackingProperties = withoutChangeTrackingProperties.length < properties.length;\r\n\r\n    // First the id, then all other properties\r\n    const sortedProperties: [string, any][] = [];\r\n    if (Object.hasOwnProperty('id'))\r\n      sortedProperties.push(['id', resource['id']]);\r\n\r\n    sortedProperties.push(...withoutChangeTrackingProperties);\r\n\r\n    // And change tracking properties at the end\r\n    if (hasChangeTrackingProperties) {\r\n      sortedProperties.push(['createdAt', resource['createdAt']]);\r\n      sortedProperties.push(['createdBy', resource['createdBy']]);\r\n      sortedProperties.push(['lastChangedAt', resource['lastChangedAt']]);\r\n      sortedProperties.push(['lastChangedBy', resource['lastChangedBy']]);\r\n    }\r\n\r\n    const propertyDescriptions: PropertyDescription[] = sortedProperties\r\n      .map(p => ({\r\n        field: p[0],\r\n        value: p[1],\r\n        label: RESTworldEditViewComponent.toTitleCase(p[0]),\r\n        type: RESTworldEditViewComponent.getColumnType(p[0], p[1]),\r\n        isReadOnly: RESTworldEditViewComponent.getIsReadOnly(p[0])\r\n      }));\r\n\r\n    for (var description of propertyDescriptions) {\r\n      if (description.type === 'object') {\r\n        const children = RESTworldEditViewComponent.createProperyInfos(description.value);\r\n        children.forEach(d => description.field + '.' + d.field);\r\n        description.children = children;\r\n      }\r\n    }\r\n\r\n    return propertyDescriptions;\r\n  }\r\n\r\n  private static getIsReadOnly(field: string) {\r\n    return field === 'id' || field === 'createdAt' || field === 'createdBy' || field === 'lastChangedAt' || field === 'lastChangedBy';\r\n  }\r\n\r\n  private static getColumnType(field: string, value: any) {\r\n    if (value === null || value === undefined)\r\n      return 'text';\r\n\r\n    if (_.isNumber(value))\r\n      return 'numeric';\r\n\r\n    if (_.isDate(value))\r\n      return 'date'\r\n\r\n    if (_.isString(value)) {\r\n      return 'text';\r\n    }\r\n\r\n    if (_.isBoolean(value))\r\n      return 'boolean';\r\n\r\n    if (_.isObject(value))\r\n      return 'object'\r\n\r\n    if (_.isArray(value))\r\n      return 'array'\r\n\r\n    return 'text'\r\n  }\r\n\r\n  private static toTitleCase(anyCase: string) {\r\n    return anyCase\r\n      .replace(/(_)+/g, ' ')                              // underscore to whitespace\r\n      .replace(/([a-z])([A-Z][a-z])/g, \"$1 $2\")           // insert space before each new word if there is none\r\n      .replace(/([A-Z][a-z])([A-Z])/g, \"$1 $2\")           // insert space after each word if there is none\r\n      .replace(/([a-z])([A-Z]+[a-z])/g, \"$1 $2\")          // insert space after single letter word if there is none\r\n      .replace(/([A-Z]+)([A-Z][a-z][a-z])/g, \"$1 $2\")     // insert space before single letter word if there is none\r\n      .replace(/([a-z]+)([A-Z0-9]+)/g, \"$1 $2\")           // insert space after numbers\r\n      .replace(/^./, (match) => match.toUpperCase());     // change first letter to be upper case\r\n  }\r\n}\r\n","<div class=\"grid\">\r\n  <div class=\"col\">\r\n    <h1>Edit resource</h1>\r\n  </div>\r\n</div>\r\n\r\n<form [formGroup]=\"formGroup\" (ngSubmit)=\"save()\">\r\n  <p-tabView>\r\n    <p-tabPanel header=\"Visual\" [disabled]=\"!resource || isLoading\">\r\n      <ng-template #defaultVisualTab>\r\n        <ng-container *ngIf=\"!resource && isLoading\">\r\n          <div *ngFor=\"let i of [1, 2, 3, 4, 5]\" class=\"grid field\">\r\n            <p-skeleton class=\"col-12 mb-2 md:col-2 md:mb-0\" height=\"39px\"></p-skeleton>\r\n            <div class=\"col-12 md:col-10\">\r\n              <p-skeleton class=\"w-full\" height=\"39px\"></p-skeleton>\r\n            </div>\r\n          </div>\r\n        </ng-container>\r\n\r\n        <div *ngIf=\"resource\" class=\"blockable-container\">\r\n          <div class=\"blockable-element\">\r\n            <ng-template #defaultForm let-properties=\"properties\" let-formGroup=\"formGroup\">\r\n              <div [formGroup]=\"formGroup\">\r\n                <div *ngFor=\"let property of properties\" class=\"grid field\">\r\n                  <ng-template #defaultFormRow>\r\n                    <ng-template #defaultFormLabel>\r\n                      <label [attr.for]=\"property.field\" class=\"col-12 mb-2 md:col-2 md:mb-0\" [class.p-disabled]=\"property.isReadOnly\" [class.hasChildren]=\"property.children?.length > 0\">{{property.label}}</label>\r\n                    </ng-template>\r\n                    <ng-container *ngTemplateOutlet=\"formLabelRef || defaultFormLabel; context:{ $implicit: property }\"></ng-container>\r\n                    <ng-template #defaultFormInput let-property let-formGroup=\"formGroup\">\r\n                        <div [ngSwitch]=\"property.type\" class=\"col-12 md:col-10\">\r\n                            <input *ngSwitchCase=\"'text'\" [formControlName]=\"property.field\" [id]=\"property.field\" type=\"text\" pInputText class=\"w-full\" [class.p-disabled]=\"property.isReadOnly\">\r\n                            <p-calendar *ngSwitchCase=\"'date'\" [formControlName]=\"property.field\" [id]=\"property.field\" [dateFormat]=\"dateFormat\" [showTime]=\"true\" [showIcon]=\"true\" styleClass=\"w-full\" [class.p-disabled]=\"property.isReadOnly\"></p-calendar>\r\n                            <p-inputNumber *ngSwitchCase=\"'numeric'\" [formControlName]=\"property.field\" [id]=\"property.field\" mode=\"decimal\" [showButtons]=\"!property.isReadOnly\" class=\"w-full\" styleClass=\"w-full\" [class.p-disabled]=\"property.isReadOnly\"></p-inputNumber>\r\n                            <p-checkbox *ngSwitchCase=\"'boolean'\" [formControl]=\"formGroup.controls[property.field]\" [id]=\"property.field\" [readonly]=\"property.isReadOnly\"></p-checkbox>\r\n                            <div *ngSwitchCase=\"'object'\">\r\n                                <ng-container *ngTemplateOutlet=\"formRef || defaultForm; context:{ properties: property.children, formGroup: formGroup.controls[property.field] }\"></ng-container>\r\n                            </div>\r\n                        </div>\r\n                    </ng-template>\r\n                    <ng-container *ngTemplateOutlet=\"formInputRef || defaultFormInput; context:{ $implicit: property, formGroup: formGroup }\"></ng-container>\r\n                  </ng-template>\r\n                  <ng-container *ngTemplateOutlet=\"formRowRef || defaultFormRow; context:{ $implicit: property }\"></ng-container>\r\n                </div>\r\n              </div>\r\n            </ng-template>\r\n            <ng-container *ngTemplateOutlet=\"formRef || defaultForm; context: { properties: properties, formGroup: formGroup }\"></ng-container>\r\n          </div>\r\n          <div class=\"blockable-overlay\" *ngIf=\"isLoading\">\r\n            <p-progressSpinner></p-progressSpinner>\r\n          </div>\r\n        </div>\r\n      </ng-template>\r\n      <ng-container *ngTemplateOutlet=\"visualTabRef || defaultVisualTab\"></ng-container>\r\n      <ng-container *ngTemplateOutlet=\"visualTabAdditionalRef || null; context:{ $implicit: resource, properties: properties, isLoading: isLoading }\"></ng-container>\r\n    </p-tabPanel>\r\n\r\n    <p-tabPanel header=\"Raw\" [disabled]=\"!resource || isLoading\">\r\n      <ng-template #defaultRawTab>\r\n        <pre>{{resource | json}}</pre>\r\n      </ng-template>\r\n      <ng-container *ngTemplateOutlet=\"rawTabRef || defaultRawTab\"></ng-container>\r\n    </p-tabPanel>\r\n  </p-tabView>\r\n\r\n  <div class=\"grid\">\r\n    <div class=\"col\">\r\n      <div class=\"flex justify-content-end w-full\">\r\n        <ng-template #defaultButtons>\r\n          <button pButton pRipple type=\"submit\" label=\"Save\" icon=\"far fa-save\" class=\"mx-2 p-button-success\" [disabled]=\"!resource || isLoading || !canSave\"></button>\r\n          <button pButton pRipple type=\"button\" label=\"Reload\" icon=\"fas fa-redo\" class=\"mx-2 p-button-info\" (click)=\"load()\" [disabled]=\"isLoading\"></button>\r\n          <button pButton pRipple type=\"button\" label=\"Delete\" icon=\"far fa-trash-alt\" class=\"mx-2 p-button-danger\" (click)=\"showDeleteConfirmatioModal()\" [disabled]=\"!resource || isLoading || !canDelete\"></button>\r\n        </ng-template>\r\n        <ng-container *ngTemplateOutlet=\"buttonsRef || defaultButtons\"></ng-container>\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n</form>\r\n\r\n<p-toast></p-toast>\r\n<p-confirmDialog></p-confirmDialog>\r\n"]}
335
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"restworld-edit-view.component.js","sourceRoot":"","sources":["../../../../../../projects/ngx-restworld-client/src/lib/views/restworld-edit-view/restworld-edit-view.component.ts","../../../../../../projects/ngx-restworld-client/src/lib/views/restworld-edit-view/restworld-edit-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\r\nimport { PropertyType, Resource, Template, Templates, FormsResource, Property } from '@wertzui/ngx-hal-client';\r\nimport { RESTworldClient } from '../../services/restworld-client';\r\nimport { RESTworldClientCollection } from '../../services/restworld-client-collection';\r\nimport { AbstractControl, FormControl, FormGroup } from '@angular/forms';\r\nimport { ConfirmationService, MessageService } from 'primeng/api';\r\nimport { Location } from '@angular/common';\r\nimport { Router } from '@angular/router';\r\nimport { ProblemDetails } from '../../models/problem-details';\r\nimport { ContentChild } from '@angular/core';\r\nimport { TemplateRef } from '@angular/core';\r\nimport { ValdemortConfig } from 'ngx-valdemort';\r\nimport { FormService } from '../../services/form.service';\r\n\r\n@Component({\r\n  selector: 'rw-edit',\r\n  templateUrl: './restworld-edit-view.component.html',\r\n  styleUrls: ['./restworld-edit-view.component.css']\r\n})\r\nexport class RESTworldEditViewComponent {\r\n  public get PropertyType() {\r\n    return PropertyType;\r\n  }\r\n  public get templates() {\r\n    return this._templates;\r\n  }\r\n  private _templates: Templates = {};\r\n\r\n  public get isLoadingForTheFirstTime() {\r\n    return Object.keys(this.templates).length === 0 && this.isLoading;\r\n  }\r\n\r\n  public get formTabs() {\r\n    return this._formTabs;\r\n  }\r\n  private _formTabs: { [name: string]: FormGroup } = {};\r\n\r\n  @Input()\r\n  public set apiName(value: string | undefined) {\r\n    this._apiName = value;\r\n    this.load();\r\n  }\r\n  public get apiName(): string | undefined {\r\n    return this._apiName;\r\n  }\r\n  private _apiName?: string;\r\n  //@Input()\r\n  //public set rel(value: string | undefined) {\r\n  //  this._rel = value;\r\n  //}\r\n  //public get rel(): string | undefined {\r\n  //  return this._rel;\r\n  //}\r\n  //private _rel?: string;\r\n  @Input()\r\n  public set uri(value: string | undefined) {\r\n    this._uri = value;\r\n    this.load();\r\n  }\r\n  public get uri(): string | undefined {\r\n    return this._uri;\r\n  }\r\n  private _uri?: string;\r\n\r\n  public get resource() {\r\n    return this._resource;\r\n  }\r\n  private _resource?: Resource;\r\n  public isLoading = false;\r\n  public get canSave() {\r\n    const length = this.resource?._links[\"save\"]?.length;\r\n    return length !== undefined && length > 0;\r\n  }\r\n  public get canDelete() {\r\n    const length = this.resource?._links[\"delete\"]?.length;\r\n    return length !== undefined && length > 0;\r\n  }\r\n  public canSubmit(templateName: string) {\r\n    const form = this.formTabs[templateName];\r\n    return form && form.valid;\r\n  }\r\n\r\n  @ContentChild('extraTabs', { static: false })\r\n  extraTabsRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('buttons', { static: false })\r\n  buttonsRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputOptionsSingle', { static: false })\r\n  inputOptionsSingleRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputOptionsMultiple', { static: false })\r\n  inputOptionsMultipleRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputOptions', { static: false })\r\n  inputOptionsRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputHidden', { static: false })\r\n  inputHiddenRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputText', { static: false })\r\n  inputTextRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputTextarea', { static: false })\r\n  inputTextareaRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputSearch', { static: false })\r\n  inputSearchRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputTel', { static: false })\r\n  inputTelRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputUrl', { static: false })\r\n  inputUrlRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputEmail', { static: false })\r\n  inputEmailRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputPassword', { static: false })\r\n  inputPasswordRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputDate', { static: false })\r\n  inputDateRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputMonth', { static: false })\r\n  inputMonthRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputWeek', { static: false })\r\n  inputWeekRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputTime', { static: false })\r\n  inputTimeRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputDatetimeLocal', { static: false })\r\n  inputDatetimeLocalRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputNumber', { static: false })\r\n  inputNumberRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputRange', { static: false })\r\n  inputRangeRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputColor', { static: false })\r\n  inputColorRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputBool', { static: false })\r\n  inputBoolRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputDatetimeOffset', { static: false })\r\n  inputDatetimeOffsetRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputDuration', { static: false })\r\n  inputDurationRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputImage', { static: false })\r\n  inputImageRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputFile', { static: false })\r\n  inputFileRef?: TemplateRef<unknown>;\r\n\r\n  @ContentChild('inputDefault', { static: false })\r\n  inputDefaultRef?: TemplateRef<unknown>;\r\n\r\n  constructor(\r\n    private _clients: RESTworldClientCollection,\r\n    private _confirmationService: ConfirmationService,\r\n    private _messageService: MessageService,\r\n    private _location: Location,\r\n    private _router: Router,\r\n    private _formService: FormService,\r\n    valdemortConfig: ValdemortConfig) {\r\n    valdemortConfig.errorClasses = 'p-error text-sm';\r\n  }\r\n\r\n  public getTooltip(resource: Resource, keysToExclude?: string[]): string {\r\n    const tooltip = Object.entries(resource)\r\n      .filter(([key]) => !(key.startsWith('_') || ['createdAt', 'createdBy', 'lastChangedAt', 'lastChangedBy', 'timestamp'].includes(key) || keysToExclude?.includes(key)))\r\n      .reduce((prev, [key, value], index) => `${prev}${index === 0 ? '' : '\\n'}${key}: ${RESTworldEditViewComponent.jsonStringifyWithElipsis(value)}`, '');\r\n\r\n    return tooltip;\r\n  }\r\n\r\n  private static jsonStringifyWithElipsis(value: unknown) {\r\n    const maxLength = 200;\r\n    const end = 10;\r\n    const start = maxLength - end - 2;\r\n    const json = JSON.stringify(value);\r\n    const shortened = json.length > maxLength ? json.substring(0, start) + '…' + json.substring(json.length - end) : json;\r\n\r\n    return shortened;\r\n  }\r\n\r\n  private getClient(): RESTworldClient {\r\n    if (!this.apiName)\r\n      throw new Error('Cannot get a client, because the apiName is not set.');\r\n\r\n    return this._clients.getClient(this.apiName);\r\n  }\r\n\r\n  public async submit(templateName: string, template: Template, formValue: {}) {\r\n    this.isLoading = true;\r\n\r\n    try {\r\n      const targetBeforeSave = template.target;\r\n      const response = await this.getClient().submit(template, formValue);\r\n\r\n      if (!response.ok) {\r\n        let summary = 'Error';\r\n        let detail = 'Error while saving the resource.';\r\n        if (ProblemDetails.isProblemDetails(response.body)) {\r\n          const problemDetails = response.body as ProblemDetails;\r\n          summary = problemDetails.title || summary;\r\n          detail = problemDetails.detail || detail;\r\n          // display validation errors\r\n          if (problemDetails['errors'] as {}) {\r\n            const form = this.formTabs[templateName];\r\n            for (const [key, errorsForKey] of Object.entries(problemDetails['errors'] as {})) {\r\n              const path = key.split(/\\.|\\[/).map(e => e.replace(\"]\", \"\"));\r\n              const formControl = path.reduce<AbstractControl>((control, pathElement) => (control instanceof FormGroup ? control.controls[pathElement] : control) || control, form);\r\n              formControl.setErrors({ remote: errorsForKey });\r\n            }\r\n          }\r\n        }\r\n\r\n        this._messageService.add({ severity: 'error', summary: summary, detail: detail, data: response, life: 10000 });\r\n      }\r\n      else {\r\n        const responseResource = (response.body as FormsResource);\r\n        const targetAfterSave = responseResource._templates[templateName].target;\r\n\r\n        setTimeout(() =>\r\n          this._messageService.add({ severity: 'success', summary: 'Saved', detail: 'The resource has been saved.' }),\r\n          100);\r\n\r\n        if (targetBeforeSave !== targetAfterSave) {\r\n          this._router.navigate(['/edit', this.apiName, responseResource._links.self[0].href]);\r\n        }\r\n      }\r\n\r\n    }\r\n    catch (e: unknown) {\r\n      this._messageService.add({ severity: 'error', summary: 'Error', detail: `An unknown error occurred. ${JSON.stringify(e)}`, life: 10000 });\r\n      console.log(e);\r\n    }\r\n\r\n    this.isLoading = false;\r\n  }\r\n\r\n  public showDeleteConfirmatioModal() {\r\n    this._confirmationService.confirm({\r\n      message: 'Do you really want to delete this resource?',\r\n      header: 'Confirm delete',\r\n      icon: 'far fa-trash-alt',\r\n      accept: () => this.delete()\r\n    });\r\n  }\r\n\r\n  public async delete(): Promise<void> {\r\n    if (!this.apiName || !this.uri || !this.resource)\r\n      return;\r\n\r\n    Object.assign(this.resource, this.formTabs.value);\r\n\r\n    await this.getClient().delete(this.resource);\r\n    setTimeout(() =>\r\n      this._messageService.add({ severity: 'success', summary: 'Deleted', detail: 'The resource has been deleted.' }),\r\n      100);\r\n\r\n    this._location.back();\r\n  }\r\n\r\n  public async load(): Promise<void> {\r\n    if (!this.apiName || !this.uri)\r\n      return;\r\n\r\n    this.isLoading = true;\r\n\r\n    const response = await this.getClient().getSingle(this.uri);\r\n    if (!response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body) {\r\n      this._messageService.add({ severity: 'error', summary: 'Error', detail: 'Error while loading the resource from the API.', data: response });\r\n    }\r\n    else {\r\n      this._resource = response.body;\r\n      this._templates = await this.getAllTemplates(this._resource);\r\n      this._formTabs = this._formService.createFormGroupsFromTemplates(this._templates);\r\n    }\r\n\r\n    this.isLoading = false;\r\n  }\r\n\r\n  private async setInitialSelectedOptionsElementsForTemplates(templates: Templates) {\r\n    return Promise.all(Object.values(templates)\r\n      .map(template => this.setInitialSelectedOptionsElementsForTemplate(template)));\r\n  }\r\n\r\n  public imageChanged(formControl: FormControl, event: { files: File[] }): void {\r\n    const file = event.files[0];\r\n    console.log(file);\r\n    const reader = new FileReader();\r\n    reader.onload = () => {\r\n      const uri = reader.result;\r\n      console.log(uri);\r\n      formControl.setValue(uri);\r\n    };\r\n    reader.readAsDataURL(file);\r\n  }\r\n\r\n  private async setInitialSelectedOptionsElementsForTemplate(template: Template) {\r\n    return Promise.all(template.properties\r\n      .filter(property => property?.options?.link?.href)\r\n      .map(property => this.setInitialSelectedOptionsElementForProperty(property)));\r\n  }\r\n\r\n  private async setInitialSelectedOptionsElementForProperty(property: Property) {\r\n    const options = property?.options;\r\n\r\n    if (!options?.link?.href)\r\n      return;\r\n\r\n    const templatedUri = options.link.href;\r\n    const filter = `${options.valueField} eq ${property.value}`;\r\n    const response = await this.getClient().getListByUri(templatedUri, { $filter: filter, $top: 10 });\r\n    if (!response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body) {\r\n      const message = `An error occurred while getting the filtered items.`;\r\n      this._messageService.add({ severity: 'error', summary: 'Error', detail: message, data: response });\r\n      return;\r\n    }\r\n\r\n    const items = response.body._embedded.items;\r\n    options.inline = items;\r\n  }\r\n\r\n  private async getAllTemplates(resource: Resource): Promise<Templates> {\r\n    const formResponses = await this.getClient().getAllForms(resource);\r\n\r\n    const failedResponses = formResponses.filter(response => !response.ok || ProblemDetails.isProblemDetails(response.body) || !response.body);\r\n    if (failedResponses.length !== 0) {\r\n      for (const response of failedResponses) {\r\n        this._messageService.add({ severity: 'error', summary: 'Error', detail: 'Error while loading the resource from the API.', data: response });\r\n      }\r\n      return Promise.resolve({});\r\n    }\r\n\r\n    const formTemplates = Object.assign({}, ...formResponses.map(response => (response.body as FormsResource)._templates)) as Templates;\r\n\r\n    await this.setInitialSelectedOptionsElementsForTemplates(formTemplates);\r\n\r\n    return formTemplates;\r\n  }\r\n}\r\n","<div class=\"grid\">\r\n  <div class=\"col\">\r\n    <h1>Edit resource</h1>\r\n  </div>\r\n</div>\r\n\r\n<p-tabView>\r\n\r\n  <p-tabPanel *ngIf=\"isLoadingForTheFirstTime\" header=\"Loading\">\r\n    <div *ngFor=\"let i of [1, 2, 3, 4, 5]\" class=\"grid field\">\r\n      <p-skeleton class=\"col-12 mb-2 md:col-2 md:mb-0\" height=\"39px\"></p-skeleton>\r\n      <div class=\"col-12 md:col-10\">\r\n        <p-skeleton class=\"w-full\" height=\"39px\"></p-skeleton>\r\n      </div>\r\n    </div>\r\n    <div class=\"grid\">\r\n      <div class=\"col\">\r\n        <div class=\"flex justify-content-end w-full\">\r\n          <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n          <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n          <p-skeleton width=\"120px\" height=\"39px\" class=\"mx-2\"></p-skeleton>\r\n        </div>\r\n      </div>\r\n    </div>\r\n  </p-tabPanel>\r\n\r\n  <p-tabPanel *ngFor=\"let item of templates | keyvalue\" [header]=\"item.value.title || item.key\" [disabled]=\"isLoading\">\r\n    <form [formGroup]=\"formTabs[item.key]\" (ngSubmit)=\"submit(item.key, item.value, formTabs[item.key].value)\">\r\n      <div class=\"blockable-container\">\r\n        <div class=\"blockable-element\">\r\n          <rw-form [formGroup]=\"formTabs[item.key]\" [template]=\"item.value\" [apiName]=\"apiName\"></rw-form>\r\n        </div>\r\n        <div class=\"blockable-overlay\" *ngIf=\"isLoading\">\r\n          <p-progressSpinner></p-progressSpinner>\r\n        </div>\r\n      </div>\r\n\r\n      <div class=\"grid\">\r\n        <div class=\"col\">\r\n          <div class=\"flex justify-content-end w-full\">\r\n            <ng-template #defaultButtons>\r\n              <button pButton pRipple type=\"submit\" label=\"Save\" icon=\"far fa-save\" class=\"mx-2 p-button-success\" [disabled]=\"isLoading || !canSubmit(item.key)\"></button>\r\n              <button pButton pRipple type=\"button\" label=\"Reload\" icon=\"fas fa-redo\" class=\"mx-2 p-button-info\" (click)=\"load()\" [disabled]=\"isLoading\"></button>\r\n              <button pButton pRipple type=\"button\" label=\"Delete\" icon=\"far fa-trash-alt\" class=\"ml-2 p-button-danger\" (click)=\"showDeleteConfirmatioModal()\" [disabled]=\"!resource || isLoading || !canDelete\"></button>\r\n            </ng-template>\r\n            <ng-container *ngTemplateOutlet=\"buttonsRef || defaultButtons\"></ng-container>\r\n          </div>\r\n        </div>\r\n      </div>\r\n    </form>\r\n  </p-tabPanel>\r\n\r\n  <ng-template #defaultExtraTabs>\r\n  </ng-template>\r\n  <ng-container *ngTemplateOutlet=\"extraTabsRef || defaultExtraTabs\"></ng-container>\r\n\r\n</p-tabView>\r\n\r\n<p-toast></p-toast>\r\n<p-confirmDialog></p-confirmDialog>\r\n"]}