alimetry-detail-edit 1.0.0 → 1.2.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.
- package/README.md +8 -4
- package/esm2020/lib/detail-edit.component.mjs +68 -55
- package/esm2020/lib/utils.mjs +6 -1
- package/fesm2015/alimetry-detail-edit.mjs +72 -56
- package/fesm2015/alimetry-detail-edit.mjs.map +1 -1
- package/fesm2020/alimetry-detail-edit.mjs +71 -53
- package/fesm2020/alimetry-detail-edit.mjs.map +1 -1
- package/lib/detail-edit.component.d.ts +15 -7
- package/lib/utils.d.ts +8 -0
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -6,10 +6,14 @@ Selector: `lib-detail-edit`
|
|
|
6
6
|
|
|
7
7
|
### Config Object
|
|
8
8
|
|
|
9
|
+
```js
|
|
9
10
|
{
|
|
10
|
-
owners:
|
|
11
|
-
deviceDataModelId,
|
|
12
|
-
|
|
11
|
+
owners: string[], // user ids
|
|
12
|
+
deviceDataModelId: string,
|
|
13
|
+
currentUserId: string
|
|
13
14
|
}
|
|
15
|
+
```
|
|
14
16
|
|
|
15
|
-
|
|
17
|
+
### Dependencies
|
|
18
|
+
|
|
19
|
+
`alimetry-shared` library as a peer dependency
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
import { Component, Input } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup, Validators } from '@angular/forms';
|
|
3
3
|
import { forkJoin, tap } from 'rxjs';
|
|
4
|
-
import { getAge, globaliseDateTime, localiseDateTime, shallowCompare } from './utils';
|
|
4
|
+
import { getAge, globaliseDateTime, localiseDateTime, shallowCompare, shallowDiff } from './utils';
|
|
5
5
|
import { dateComparison, withinNYears, withinNHours, translateError } from './validators';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
7
|
-
import * as i1 from "shared";
|
|
7
|
+
import * as i1 from "alimetry-shared";
|
|
8
8
|
import * as i2 from "@angular/forms";
|
|
9
9
|
import * as i3 from "@angular/common";
|
|
10
10
|
import * as i4 from "./entries.pipe";
|
|
@@ -12,8 +12,10 @@ const appDataPropertySetId = '34d46477-ad4b-47b4-b555-06358b92f60c';
|
|
|
12
12
|
const deviceDataPropertySetId = '7a32d851-5e89-46a6-a4b6-cca162dace29';
|
|
13
13
|
const propertyCodes = ['RecordingSession', 'Status', 'UserDetailChangeLog'];
|
|
14
14
|
export class DetailEditComponent {
|
|
15
|
-
constructor(shared) {
|
|
15
|
+
constructor(shared, kvDiffers, itDiffers) {
|
|
16
16
|
this.shared = shared;
|
|
17
|
+
this.kvDiffers = kvDiffers;
|
|
18
|
+
this.itDiffers = itDiffers;
|
|
17
19
|
this.status = 'loading';
|
|
18
20
|
this.formGroup = new FormGroup({
|
|
19
21
|
dob: new FormControl('', [Validators.required, withinNYears(120, 'before')]),
|
|
@@ -65,7 +67,7 @@ export class DetailEditComponent {
|
|
|
65
67
|
patient.age = getAge(mods.dob);
|
|
66
68
|
recordingSession.mealStart = globaliseDateTime(mods.mealStart);
|
|
67
69
|
recordingSession.mealEnd = globaliseDateTime(mods.mealEnd);
|
|
68
|
-
this.
|
|
70
|
+
this.logChange(mods);
|
|
69
71
|
return {
|
|
70
72
|
'Age': patient.age,
|
|
71
73
|
'RecordingSession': JSON.stringify(recordingSession),
|
|
@@ -73,55 +75,27 @@ export class DetailEditComponent {
|
|
|
73
75
|
};
|
|
74
76
|
}
|
|
75
77
|
postDeviceData(devicePropertySetId, data) {
|
|
76
|
-
return this.shared.saveDeviceData(this.config.deviceDataModelId, this.deviceDataId, devicePropertySetId, this.config.owners[0]
|
|
78
|
+
return this.shared.saveDeviceData(this.config.deviceDataModelId, this.deviceDataId, devicePropertySetId, this.config.owners[0], data);
|
|
77
79
|
}
|
|
78
80
|
// patch the device data record with the modifications
|
|
79
81
|
saveModifications() {
|
|
80
82
|
const mods = this.formGroup.getRawValue();
|
|
81
83
|
const patchedDeviceData = this.integrateFields(mods, this.recordingSession);
|
|
82
|
-
const patchedUserData = Object.assign(this.
|
|
84
|
+
const patchedUserData = Object.assign(this.owner, { dateOfBirth: mods.dob });
|
|
83
85
|
this.status = 'sending';
|
|
84
86
|
forkJoin([
|
|
85
87
|
this.postDeviceData(appDataPropertySetId, patchedDeviceData),
|
|
86
|
-
this.shared.updateUser(this.config.owners[0]
|
|
88
|
+
this.shared.updateUser(this.config.owners[0], patchedUserData)
|
|
87
89
|
]).pipe(tap(() => this.regenReport())).subscribe();
|
|
88
90
|
}
|
|
89
|
-
|
|
90
|
-
const
|
|
91
|
-
const newChangeLog = {};
|
|
92
|
-
// Only include fields that have changed
|
|
93
|
-
for (const key in mods) {
|
|
94
|
-
if (this.initialFormState[key] !== mods[key]) {
|
|
95
|
-
prevChangeLog[key] = this.initialFormState[key];
|
|
96
|
-
newChangeLog[key] = mods[key];
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
const changeLog = {
|
|
91
|
+
logChange(mods) {
|
|
92
|
+
const entry = {
|
|
100
93
|
author: this.currentUser.emailAddress,
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
timestamp: new Date().toISOString()
|
|
94
|
+
timestamp: new Date().toISOString(),
|
|
95
|
+
...shallowDiff(this.initialFormState, mods)
|
|
104
96
|
};
|
|
105
|
-
this.detailChangeLog.push(
|
|
106
|
-
|
|
107
|
-
// get the device data record and initialise the form
|
|
108
|
-
loadDeviceData() {
|
|
109
|
-
this.shared.getDeviceData(this.config.deviceDataModelId, this.config.owners[0].userId, propertyCodes).subscribe({
|
|
110
|
-
next: (res) => {
|
|
111
|
-
this.deviceDataId = res.deviceDataId;
|
|
112
|
-
this.recordingSession = JSON.parse(res.data['RecordingSession'].value);
|
|
113
|
-
this.initialiseForm(this.extractFields(this.recordingSession));
|
|
114
|
-
// checking for undefined in case no changes have been logged yet
|
|
115
|
-
this.detailChangeLog = res.data['UserDetailChangeLog'] === undefined ? [] : res.data['UserDetailChangeLog'].value;
|
|
116
|
-
this.status = res.data['Status']?.value === 'Generating Report' ?
|
|
117
|
-
'generating' :
|
|
118
|
-
'success';
|
|
119
|
-
},
|
|
120
|
-
error: (err) => {
|
|
121
|
-
console.error(err);
|
|
122
|
-
this.status = "error";
|
|
123
|
-
}
|
|
124
|
-
});
|
|
97
|
+
this.detailChangeLog.push(entry);
|
|
98
|
+
console.log(this.detailChangeLog);
|
|
125
99
|
}
|
|
126
100
|
regenReport() {
|
|
127
101
|
this.postDeviceData(deviceDataPropertySetId, { 'Algorithmhasrun': false, 'AlgorithmRun': true }).subscribe(() => {
|
|
@@ -131,26 +105,65 @@ export class DetailEditComponent {
|
|
|
131
105
|
resetModifications() {
|
|
132
106
|
this.formGroup.patchValue(this.initialFormState);
|
|
133
107
|
}
|
|
134
|
-
|
|
135
|
-
|
|
108
|
+
loadData(currentUserId, ownerId, deviceDataModelId) {
|
|
109
|
+
forkJoin([
|
|
110
|
+
this.shared.getUser(currentUserId),
|
|
111
|
+
this.shared.getUser(ownerId),
|
|
112
|
+
this.shared.getDeviceData(deviceDataModelId, ownerId, propertyCodes)
|
|
113
|
+
]).subscribe({
|
|
114
|
+
next: ([currentUser, owner, { deviceDataId, data: deviceData }]) => {
|
|
115
|
+
this.currentUser = currentUser;
|
|
116
|
+
this.owner = owner;
|
|
117
|
+
this.deviceDataId = deviceDataId;
|
|
118
|
+
this.recordingSession = JSON.parse(deviceData['RecordingSession'].value);
|
|
119
|
+
this.initialiseForm(this.extractFields(this.recordingSession));
|
|
120
|
+
// data field uninitialised for unmodified record
|
|
121
|
+
this.detailChangeLog = deviceData['UserDetailChangeLog']?.value ?? [];
|
|
122
|
+
this.status = deviceData['Status']?.value === 'Generating Report' ?
|
|
123
|
+
'generating' : 'success';
|
|
124
|
+
},
|
|
125
|
+
error: (err) => {
|
|
126
|
+
console.error(err);
|
|
127
|
+
this.status = "error";
|
|
128
|
+
}
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
init() {
|
|
132
|
+
if (!this.config?.currentUserId || !this.config.deviceDataModelId) {
|
|
133
|
+
this.status = 'cfgerror';
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
else if (!this.config?.owners?.[0]) {
|
|
136
137
|
this.status = 'nouser';
|
|
137
138
|
return;
|
|
138
139
|
}
|
|
139
|
-
this.
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
140
|
+
this.status = 'loading';
|
|
141
|
+
this.loadData(this.config.currentUserId, this.config.owners[0], this.config.deviceDataModelId);
|
|
142
|
+
}
|
|
143
|
+
// this should be called from the parent using the ViewChild attr
|
|
144
|
+
refresh() {
|
|
145
|
+
this.init();
|
|
146
|
+
}
|
|
147
|
+
// NOTE: there could be something wrong / unusual here but it seems to work ok
|
|
148
|
+
ngOnInit() {
|
|
149
|
+
if (this.config && !this.configDiffer)
|
|
150
|
+
this.configDiffer = this.kvDiffers.find(this.config).create();
|
|
151
|
+
if (this.config?.owners && !this.ownersDiffer)
|
|
152
|
+
this.ownersDiffer = this.itDiffers.find(this.config.owners).create();
|
|
153
|
+
}
|
|
154
|
+
ngDoCheck() {
|
|
155
|
+
const configChanges = this.configDiffer?.diff(this.config);
|
|
156
|
+
const ownersChanges = this.ownersDiffer?.diff(this.config?.owners);
|
|
157
|
+
if (configChanges || ownersChanges)
|
|
158
|
+
this.init();
|
|
146
159
|
}
|
|
147
160
|
}
|
|
148
|
-
DetailEditComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, deps: [{ token: i1.SharedService }], target: i0.ɵɵFactoryTarget.Component });
|
|
149
|
-
DetailEditComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: DetailEditComponent, selector: "lib-detail-edit", inputs: { config: "config" }, ngImport: i0, template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"status
|
|
161
|
+
DetailEditComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, deps: [{ token: i1.SharedService }, { token: i0.KeyValueDiffers }, { token: i0.IterableDiffers }], target: i0.ɵɵFactoryTarget.Component });
|
|
162
|
+
DetailEditComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: DetailEditComponent, selector: "lib-detail-edit", inputs: { config: "config" }, ngImport: i0, template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'cfgerror'\">\n The widget has not been configured correctly, or is missing configuration information.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"!status.includes('error') && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n <div class=\"inputs\">\n <div class=\"form-row\">\n <label for=\"dob\">Date of Birth:</label>\n <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"weight\">Patient Weight (kg):</label>\n <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"height\">Patient Height (cm):</label>\n <input id=\"height\" type=\"number\" formControlName=\"height\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealStart\">Meal Start Time:</label>\n <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealEnd\">Meal End Time:</label>\n <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n </div>\n </div>\n\n <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n <div *ngFor=\"let control of formGroup.controls | entries\">\n {{ parseInputErrors(control[0], control[1]) }}\n </div>\n </div>\n\n <div *ngIf=\"status === 'generating'\" class=\"info\">\n <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n </div>\n\n <div class=\"buttons\">\n <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n </div>\n </form>\n</div>\n", styles: [".wrapper{width:100%;height:100%}.status{font-size:2rem;margin:5rem auto;text-align:center}form{display:flex;flex-direction:column;padding:2rem;gap:2rem;font-size:.9rem;max-width:70rem;margin:0 auto}form .inputs{display:flex;flex-flow:row wrap;gap:1rem 3rem}form .inputs .form-row{display:flex;gap:1rem;align-items:center;flex:1 0 calc(50% - 3rem);min-width:400px}form .inputs .form-row label{min-width:9rem}form .inputs .form-row input{background:#f5f5f6;height:36px;border:none;padding-inline:10px;flex-grow:1}form.ng-dirty .form-row .ng-invalid{outline:1px auto red}form .buttons{display:flex;gap:1rem}form .buttons button{padding:.5rem 1rem;height:36px;border:none;font-size:inherit;background:#f5f5f6}form .buttons button:not([disabled]){cursor:pointer}form .buttons button[type=submit]{background:#242C69;color:#fff}form .buttons button[disabled]{background:#fafafa;cursor:not-allowed}form .buttons button[type=submit][disabled]{background:#444C89}form .buttons button:first-child{margin-right:auto}.errors{padding:1rem;background:#fee;border:2px solid red;border-radius:4px;color:red}.info{padding:1rem;background:#eef;border:2px solid #242C69;border-radius:4px;color:#242c69}\n"], dependencies: [{ kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i4.EntriesPipe, name: "entries" }] });
|
|
150
163
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, decorators: [{
|
|
151
164
|
type: Component,
|
|
152
|
-
args: [{ selector: 'lib-detail-edit', template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"status
|
|
153
|
-
}], ctorParameters: function () { return [{ type: i1.SharedService }]; }, propDecorators: { config: [{
|
|
165
|
+
args: [{ selector: 'lib-detail-edit', template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'cfgerror'\">\n The widget has not been configured correctly, or is missing configuration information.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"!status.includes('error') && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n <div class=\"inputs\">\n <div class=\"form-row\">\n <label for=\"dob\">Date of Birth:</label>\n <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"weight\">Patient Weight (kg):</label>\n <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"height\">Patient Height (cm):</label>\n <input id=\"height\" type=\"number\" formControlName=\"height\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealStart\">Meal Start Time:</label>\n <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealEnd\">Meal End Time:</label>\n <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n </div>\n </div>\n\n <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n <div *ngFor=\"let control of formGroup.controls | entries\">\n {{ parseInputErrors(control[0], control[1]) }}\n </div>\n </div>\n\n <div *ngIf=\"status === 'generating'\" class=\"info\">\n <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n </div>\n\n <div class=\"buttons\">\n <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n </div>\n </form>\n</div>\n", styles: [".wrapper{width:100%;height:100%}.status{font-size:2rem;margin:5rem auto;text-align:center}form{display:flex;flex-direction:column;padding:2rem;gap:2rem;font-size:.9rem;max-width:70rem;margin:0 auto}form .inputs{display:flex;flex-flow:row wrap;gap:1rem 3rem}form .inputs .form-row{display:flex;gap:1rem;align-items:center;flex:1 0 calc(50% - 3rem);min-width:400px}form .inputs .form-row label{min-width:9rem}form .inputs .form-row input{background:#f5f5f6;height:36px;border:none;padding-inline:10px;flex-grow:1}form.ng-dirty .form-row .ng-invalid{outline:1px auto red}form .buttons{display:flex;gap:1rem}form .buttons button{padding:.5rem 1rem;height:36px;border:none;font-size:inherit;background:#f5f5f6}form .buttons button:not([disabled]){cursor:pointer}form .buttons button[type=submit]{background:#242C69;color:#fff}form .buttons button[disabled]{background:#fafafa;cursor:not-allowed}form .buttons button[type=submit][disabled]{background:#444C89}form .buttons button:first-child{margin-right:auto}.errors{padding:1rem;background:#fee;border:2px solid red;border-radius:4px;color:red}.info{padding:1rem;background:#eef;border:2px solid #242C69;border-radius:4px;color:#242c69}\n"] }]
|
|
166
|
+
}], ctorParameters: function () { return [{ type: i1.SharedService }, { type: i0.KeyValueDiffers }, { type: i0.IterableDiffers }]; }, propDecorators: { config: [{
|
|
154
167
|
type: Input
|
|
155
168
|
}] } });
|
|
156
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detail-edit.component.js","sourceRoot":"","sources":["../../../../projects/detail-edit/src/lib/detail-edit.component.ts","../../../../projects/detail-edit/src/lib/detail-edit.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input } from '@angular/core';\nimport { AbstractControl, FormControl, FormGroup, Validators } from '@angular/forms';\nimport { forkJoin, tap } from 'rxjs';\nimport { Config, SharedService, User } from 'shared';\nimport { getAge, globaliseDateTime, localiseDateTime, shallowCompare } from './utils';\nimport { dateComparison, withinNYears, withinNHours, translateError } from './validators';\n\nconst appDataPropertySetId = '34d46477-ad4b-47b4-b555-06358b92f60c';\nconst deviceDataPropertySetId = '7a32d851-5e89-46a6-a4b6-cca162dace29';\nconst propertyCodes = ['RecordingSession', 'Status', 'UserDetailChangeLog'];\n\n@Component({\n  selector: 'lib-detail-edit', templateUrl: './detail-edit.component.html',\n  styleUrls: ['./detail-edit.component.css']\n})\nexport class DetailEditComponent {\n  @Input() config!: Config;\n\n  status: 'error' | 'loading' | 'success' | 'generating' | 'sending' | 'nouser' = 'loading';\n\n  // populated by load request\n  private deviceDataId: string;\n  private recordingSession: object;\n  private recordingStart: string;\n  private initialFormState: object;\n  private detailChangeLog: object[];\n  private currentUser: User;\n  private owner: User;\n\n  formGroup = new FormGroup({\n    dob: new FormControl<string>('', [Validators.required, withinNYears(120, 'before')]),\n    height: new FormControl<number>(0, [Validators.required, Validators.min(30), Validators.max(272)]),\n    weight: new FormControl<number>(0, [Validators.required, Validators.min(1), Validators.max(635)]),\n    mealStart: new FormControl<string>('', [Validators.required]),\n    mealEnd: new FormControl<string>('', [Validators.required])\n  }, [\n    dateComparison('mealEnd', 'mealStart', 'after'),\n    dateComparison('mealStart', 'dob', 'after'),\n  ]);\n\n  constructor(private shared: SharedService) { }\n\n  parseInputErrors(name: string, control: AbstractControl) {\n    return translateError(name, control);\n  }\n\n  // make sure than modifications that don't actually differ from the current are marked as such\n  private monitorFormChanges() {\n    this.formGroup.valueChanges.subscribe(current => {\n      if (shallowCompare(current, this.initialFormState)) {\n        this.formGroup.markAsPristine();\n      }\n    })\n  }\n\n  // handle the relevant data from the response body\n  private extractFields(recordingSession: Record<string, any>) {\n    const { patient, mealStart, mealEnd, recordingStart } = recordingSession;\n    this.recordingStart = recordingStart;\n\n    return {\n      dob: this.owner.dateOfBirth,\n      height: patient.height,\n      weight: patient.weight,\n      mealStart: localiseDateTime(mealStart),\n      mealEnd: localiseDateTime(mealEnd)\n    }\n  }\n\n  private initialiseForm(state: object) {\n    this.formGroup.patchValue(state);\n    this.formGroup.markAsPristine();\n\n    this.formGroup.get('mealStart').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));\n    this.formGroup.get('mealEnd').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));\n\n    this.initialFormState = state; // save this for use in resets\n    this.monitorFormChanges();\n  }\n\n  // construct a patch object for the save request\n  private integrateFields(mods: typeof this.formGroup.value, recordingSession: Record<string, any>) {\n    const { patient } = recordingSession;\n    patient.height = mods.height;\n    patient.weight = mods.weight;\n    patient.age = getAge(mods.dob);\n\n    recordingSession.mealStart = globaliseDateTime(mods.mealStart);\n    recordingSession.mealEnd = globaliseDateTime(mods.mealEnd);\n\n    this.appendChangeLog(mods);\n\n    return {\n      'Age': patient.age,\n      'RecordingSession': JSON.stringify(recordingSession),\n      'UserDetailChangeLog': this.detailChangeLog\n    }\n  }\n\n  private postDeviceData(devicePropertySetId: string, data: object) {\n    return this.shared.saveDeviceData(\n      this.config.deviceDataModelId,\n      this.deviceDataId,\n      devicePropertySetId,\n      this.config.owners[0].userId,\n      data\n    );\n  }\n\n  // patch the device data record with the modifications\n  saveModifications() {\n    const mods = this.formGroup.getRawValue();\n    const patchedDeviceData = this.integrateFields(mods, this.recordingSession);\n    const patchedUserData = Object.assign(this.config.owners[0], { dateOfBirth: mods.dob });\n\n    this.status = 'sending';\n    forkJoin([\n      this.postDeviceData(appDataPropertySetId, patchedDeviceData),\n      this.shared.updateUser(this.config.owners[0].userId, patchedUserData)\n    ]).pipe(\n      tap(() => this.regenReport())\n    ).subscribe();\n  }\n\n  private appendChangeLog(mods: typeof this.formGroup.value) {\n    const prevChangeLog: Partial<typeof this.formGroup.value> = {};\n    const newChangeLog: Partial<typeof this.formGroup.value> = {};\n\n    // Only include fields that have changed\n    for (const key in mods) {\n      if (this.initialFormState[key] !== mods[key]) {\n        prevChangeLog[key] = this.initialFormState[key];\n        newChangeLog[key] = mods[key];\n      }\n    }\n\n    const changeLog = {\n      author: this.currentUser.emailAddress,\n      prevChangeLog: prevChangeLog,\n      newChangeLog: newChangeLog,\n      timestamp: new Date().toISOString()\n    }\n\n    this.detailChangeLog.push(changeLog);\n  }\n\n  // get the device data record and initialise the form\n  private loadDeviceData() {\n    this.shared.getDeviceData(\n      this.config.deviceDataModelId,\n      this.config.owners[0].userId,\n      propertyCodes\n    ).subscribe({\n      next: (res) => {\n        this.deviceDataId = res.deviceDataId;\n        this.recordingSession = JSON.parse(res.data['RecordingSession'].value);\n        this.initialiseForm(this.extractFields(this.recordingSession));\n        // checking for undefined in case no changes have been logged yet\n        this.detailChangeLog = res.data['UserDetailChangeLog'] === undefined ? [] : res.data['UserDetailChangeLog'].value;\n\n        this.status = res.data['Status']?.value === 'Generating Report' ?\n          'generating' :\n          'success';\n      },\n      error: (err) => {\n        console.error(err);\n        this.status = \"error\";\n      }\n    });\n  }\n\n  regenReport() {\n    this.postDeviceData(\n      deviceDataPropertySetId,\n      { 'Algorithmhasrun': false, 'AlgorithmRun': true }\n    ).subscribe(() => {\n      this.status = 'generating'; // assume report starts generating immediately\n    })\n  }\n\n  resetModifications() {\n    this.formGroup.patchValue(this.initialFormState);\n  }\n\n  ngOnInit() {\n    if (!this.config?.owners?.[0]?.userId) {\n      this.status = 'nouser';\n      return;\n    }\n    this.shared.getUser(this.config.currentUser.userId).subscribe(user => {\n      this.currentUser = user;\n    });\n\n    this.shared.getUser(this.config.owners[0].userId).subscribe(owner => {\n      this.owner = owner;\n    });\n\n    this.loadDeviceData();\n  }\n}\n","<div class=\"wrapper\">\n  <div class=\"status\" *ngIf=\"status === 'error'\">\n    An error occured while trying to get the details.\n  </div>\n  <div class=\"status\" *ngIf=\"status === 'nouser'\">\n    There is no user specified as the data owner.\n  </div>\n  <form *ngIf=\"status !== 'error' && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n    <div class=\"inputs\">\n      <div class=\"form-row\">\n        <label for=\"dob\">Date of Birth:</label>\n        <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"weight\">Patient Weight (kg):</label>\n        <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"height\">Patient Height (cm):</label>\n        <input id=\"height\" type=\"number\" formControlName=\"height\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"mealStart\">Meal Start Time:</label>\n        <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"mealEnd\">Meal End Time:</label>\n        <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n      </div>\n    </div>\n\n    <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n      <div *ngFor=\"let control of formGroup.controls | entries\">\n        {{ parseInputErrors(control[0], control[1]) }}\n      </div>\n    </div>\n\n    <div *ngIf=\"status === 'generating'\" class=\"info\">\n      <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n    </div>\n\n    <div class=\"buttons\">\n      <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n      <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n      <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n    </div>\n  </form>\n</div>\n"]}
|
|
169
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detail-edit.component.js","sourceRoot":"","sources":["../../../../projects/detail-edit/src/lib/detail-edit.component.ts","../../../../projects/detail-edit/src/lib/detail-edit.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, DoCheck, Input, KeyValueDiffers, KeyValueDiffer, IterableDiffer, IterableDiffers } from '@angular/core';\nimport { AbstractControl, FormControl, FormGroup, Validators } from '@angular/forms';\nimport { forkJoin, tap } from 'rxjs';\nimport { Config, SharedService, User } from 'alimetry-shared';\nimport { getAge, globaliseDateTime, localiseDateTime, shallowCompare, shallowDiff } from './utils';\nimport { dateComparison, withinNYears, withinNHours, translateError } from './validators';\n\nconst appDataPropertySetId = '34d46477-ad4b-47b4-b555-06358b92f60c';\nconst deviceDataPropertySetId = '7a32d851-5e89-46a6-a4b6-cca162dace29';\nconst propertyCodes = ['RecordingSession', 'Status', 'UserDetailChangeLog'];\n\ninterface FormValue {\n  dob: string,\n  height: number,\n  weight: number,\n  mealStart: string,\n  mealEnd: string\n}\n\ninterface Change {\n  timestamp: string,\n  author: string,\n  from: Partial<FormValue>,\n  to: Partial<FormValue>\n}\n\n@Component({\n  selector: 'lib-detail-edit',\n  templateUrl: './detail-edit.component.html',\n  styleUrls: ['./detail-edit.component.css']\n})\nexport class DetailEditComponent implements DoCheck {\n  @Input() config!: Config;\n\n  private configDiffer?: KeyValueDiffer<keyof Config, any>;\n  private ownersDiffer?: IterableDiffer<string>;\n\n  status: 'error' | 'cfgerror' | 'loading' | 'success' | 'generating' | 'sending' | 'nouser' = 'loading';\n\n  // device data populated by load request\n  private deviceDataId: string;\n  private recordingSession: object;\n  private recordingStart: string;\n  private initialFormState: object;\n\n  // user data expanded from ids passed in config\n  private currentUser: User;\n  private owner: User;\n\n  private detailChangeLog: Change[];\n\n  formGroup = new FormGroup({\n    dob: new FormControl<string>('', [Validators.required, withinNYears(120, 'before')]),\n    height: new FormControl<number>(0, [Validators.required, Validators.min(30), Validators.max(272)]),\n    weight: new FormControl<number>(0, [Validators.required, Validators.min(1), Validators.max(635)]),\n    mealStart: new FormControl<string>('', [Validators.required]),\n    mealEnd: new FormControl<string>('', [Validators.required])\n  }, [\n    dateComparison('mealEnd', 'mealStart', 'after'),\n    dateComparison('mealStart', 'dob', 'after'),\n  ]);\n\n  constructor(private shared: SharedService, private kvDiffers: KeyValueDiffers, private itDiffers: IterableDiffers) { }\n\n  parseInputErrors(name: string, control: AbstractControl) {\n    return translateError(name, control);\n  }\n\n  // make sure than modifications that don't actually differ from the current are marked as such\n  private monitorFormChanges() {\n    this.formGroup.valueChanges.subscribe(current => {\n      if (shallowCompare(current, this.initialFormState)) {\n        this.formGroup.markAsPristine();\n      }\n    })\n  }\n\n  // handle the relevant data from the response body\n  private extractFields(recordingSession: Record<string, any>) {\n    const { patient, mealStart, mealEnd, recordingStart } = recordingSession;\n    this.recordingStart = recordingStart;\n\n    return {\n      dob: this.owner.dateOfBirth,\n      height: patient.height,\n      weight: patient.weight,\n      mealStart: localiseDateTime(mealStart),\n      mealEnd: localiseDateTime(mealEnd)\n    }\n  }\n\n  private initialiseForm(state: object) {\n    this.formGroup.patchValue(state);\n    this.formGroup.markAsPristine();\n\n    this.formGroup.get('mealStart').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));\n    this.formGroup.get('mealEnd').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));\n\n    this.initialFormState = state; // save this for use in resets\n    this.monitorFormChanges();\n  }\n\n  // construct a patch object for the save request\n  private integrateFields(mods: FormValue, recordingSession: Record<string, any>) {\n    const { patient } = recordingSession;\n    patient.height = mods.height;\n    patient.weight = mods.weight;\n    patient.age = getAge(mods.dob);\n\n    recordingSession.mealStart = globaliseDateTime(mods.mealStart);\n    recordingSession.mealEnd = globaliseDateTime(mods.mealEnd);\n\n    this.logChange(mods);\n\n    return {\n      'Age': patient.age,\n      'RecordingSession': JSON.stringify(recordingSession),\n      'UserDetailChangeLog': this.detailChangeLog\n    }\n  }\n\n  private postDeviceData(devicePropertySetId: string, data: object) {\n    return this.shared.saveDeviceData(\n      this.config.deviceDataModelId,\n      this.deviceDataId,\n      devicePropertySetId,\n      this.config.owners[0],\n      data\n    );\n  }\n\n  // patch the device data record with the modifications\n  saveModifications() {\n    const mods = this.formGroup.getRawValue();\n    const patchedDeviceData = this.integrateFields(mods, this.recordingSession);\n    const patchedUserData = Object.assign(this.owner, { dateOfBirth: mods.dob });\n\n    this.status = 'sending';\n    forkJoin([\n      this.postDeviceData(appDataPropertySetId, patchedDeviceData),\n      this.shared.updateUser(this.config.owners[0], patchedUserData)\n    ]).pipe(\n      tap(() => this.regenReport())\n    ).subscribe();\n  }\n\n  private logChange(mods: FormValue) {\n    const entry = {\n      author: this.currentUser.emailAddress,\n      timestamp: new Date().toISOString(),\n      ...shallowDiff(this.initialFormState, mods)\n    }\n\n    this.detailChangeLog.push(entry);\n\n    console.log(this.detailChangeLog);\n  }\n\n  regenReport() {\n    this.postDeviceData(\n      deviceDataPropertySetId,\n      { 'Algorithmhasrun': false, 'AlgorithmRun': true }\n    ).subscribe(() => {\n      this.status = 'generating'; // assume report starts generating immediately\n    })\n  }\n\n  resetModifications() {\n    this.formGroup.patchValue(this.initialFormState);\n  }\n\n  private loadData(currentUserId: string, ownerId: string, deviceDataModelId: string) {\n    forkJoin([\n      this.shared.getUser(currentUserId),\n      this.shared.getUser(ownerId),\n      this.shared.getDeviceData(deviceDataModelId, ownerId, propertyCodes)\n    ]).subscribe({\n      next: ([currentUser, owner, { deviceDataId, data: deviceData }]) => {\n        this.currentUser = currentUser;\n        this.owner = owner;\n        this.deviceDataId = deviceDataId;\n\n        this.recordingSession = JSON.parse(deviceData['RecordingSession'].value);\n        this.initialiseForm(this.extractFields(this.recordingSession));\n\n        // data field uninitialised for unmodified record\n        this.detailChangeLog = deviceData['UserDetailChangeLog']?.value ?? [];\n\n        this.status = deviceData['Status']?.value === 'Generating Report' ?\n          'generating' : 'success';\n      },\n      error: (err) => {\n        console.error(err);\n        this.status = \"error\";\n      }\n    });\n  }\n\n  private init() {\n    if (!this.config?.currentUserId || !this.config.deviceDataModelId) {\n      this.status = 'cfgerror'\n      return;\n    } else if (!this.config?.owners?.[0]) {\n      this.status = 'nouser';\n      return;\n    }\n\n    this.status = 'loading';\n    this.loadData(this.config.currentUserId, this.config.owners[0], this.config.deviceDataModelId);\n  }\n\n  // this should be called from the parent using the ViewChild attr\n  refresh() {\n    this.init();\n  }\n\n  // NOTE: there could be something wrong / unusual here but it seems to work ok\n  ngOnInit() {\n    if (this.config && !this.configDiffer) this.configDiffer = this.kvDiffers.find(this.config).create();\n    if (this.config?.owners && !this.ownersDiffer) this.ownersDiffer = this.itDiffers.find(this.config.owners).create();\n  }\n\n  ngDoCheck() {\n    const configChanges = this.configDiffer?.diff(this.config);\n    const ownersChanges = this.ownersDiffer?.diff(this.config?.owners);\n\n    if (configChanges || ownersChanges) this.init();\n  }\n}\n","<div class=\"wrapper\">\n  <div class=\"status\" *ngIf=\"status === 'error'\">\n    An error occured while trying to get the details.\n  </div>\n  <div class=\"status\" *ngIf=\"status === 'cfgerror'\">\n    The widget has not been configured correctly, or is missing configuration information.\n  </div>\n  <div class=\"status\" *ngIf=\"status === 'nouser'\">\n    There is no user specified as the data owner.\n  </div>\n  <form *ngIf=\"!status.includes('error') && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n    <div class=\"inputs\">\n      <div class=\"form-row\">\n        <label for=\"dob\">Date of Birth:</label>\n        <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"weight\">Patient Weight (kg):</label>\n        <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"height\">Patient Height (cm):</label>\n        <input id=\"height\" type=\"number\" formControlName=\"height\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"mealStart\">Meal Start Time:</label>\n        <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n      </div>\n\n      <div class=\"form-row\">\n        <label for=\"mealEnd\">Meal End Time:</label>\n        <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n      </div>\n    </div>\n\n    <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n      <div *ngFor=\"let control of formGroup.controls | entries\">\n        {{ parseInputErrors(control[0], control[1]) }}\n      </div>\n    </div>\n\n    <div *ngIf=\"status === 'generating'\" class=\"info\">\n      <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n    </div>\n\n    <div class=\"buttons\">\n      <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n      <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n      <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n    </div>\n  </form>\n</div>\n"]}
|
package/esm2020/lib/utils.mjs
CHANGED
|
@@ -45,4 +45,9 @@ export function titleise(camel) {
|
|
|
45
45
|
export function shallowCompare(a, b) {
|
|
46
46
|
return Object.entries(a).every(([k, v]) => b[k] === v);
|
|
47
47
|
}
|
|
48
|
-
|
|
48
|
+
export function shallowDiff(a, b) {
|
|
49
|
+
const from = Object.fromEntries(Object.entries(a).filter(([k, v]) => b[k] !== v));
|
|
50
|
+
const to = Object.fromEntries(Object.entries(b).filter(([k, _]) => from[k]));
|
|
51
|
+
return { from, to };
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9kZXRhaWwtZWRpdC9zcmMvbGliL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLCtEQUErRDtBQUMvRCxNQUFNLFVBQVUsZ0JBQWdCLENBQUMsUUFBZ0I7SUFDL0MsTUFBTSxJQUFJLEdBQUcsSUFBSSxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7SUFFaEMsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQUMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUNqRyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN2RCxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN6RCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUM3RCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUU3RCxPQUFPLEdBQUcsSUFBSSxJQUFJLEtBQUssSUFBSSxHQUFHLElBQUksSUFBSSxJQUFJLE1BQU0sSUFBSSxNQUFNLEVBQUUsQ0FBQztBQUMvRCxDQUFDO0FBRUQsa0NBQWtDO0FBQ2xDLE1BQU0sVUFBVSxpQkFB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LENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUM7QUFDdkUsQ0FBQztBQUVELE1BQU0sVUFBVSxjQUFjLENBQUMsQ0FBUyxFQUFFLENBQVM7SUFDakQsT0FBTyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFDekQsQ0FBQztBQUVELE1BQU0sVUFBVSxXQUFXLENBQUMsQ0FBUyxFQUFFLENBQVM7SUFDOUMsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsRixNQUFNLEVBQUUsR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDN0UsT0FBTyxFQUFFLElBQUksRUFBRSxFQUFFLEVBQUUsQ0FBQztBQUN0QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gc3RyaXAgdGhlIHRpbWV6b25lIG9mZnNldCB0byBjcmVhdGUgZm9ybWF0IHVzYWJsZSBieSA8aW5wdXQ+XG5leHBvcnQgZnVuY3Rpb24gbG9jYWxpc2VEYXRlVGltZShkYXRldGltZTogc3RyaW5nKSB7XG4gIGNvbnN0IGRhdGUgPSBuZXcgRGF0ZShkYXRldGltZSk7XG5cbiAgY29uc3QgeWVhciA9IGRhdGUuZ2V0RnVsbFllYXIoKTsgY29uc3QgbW9udGggPSAoZGF0ZS5nZXRNb250aCgpICsgMSkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBkYXkgPSBkYXRlLmdldERhdGUoKS50b1N0cmluZygpLnBhZFN0YXJ0KDIsICcwJyk7XG4gIGNvbnN0IGhvdXIgPSBkYXRlLmdldEhvdXJzKCkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBtaW51dGUgPSBkYXRlLmdldE1pbnV0ZXMoKS50b1N0cmluZygpLnBhZFN0YXJ0KDIsICcwJyk7XG4gIGNvbnN0IHNlY29uZCA9IGRhdGUuZ2V0U2Vjb25kcygpLnRvU3RyaW5nKCkucGFkU3RhcnQoMiwgJzAnKTtcblxuICByZXR1cm4gYCR7eWVhcn0tJHttb250aH0tJHtkYXl9VCR7aG91cn06JHttaW51dGV9OiR7c2Vjb25kfWA7XG59XG5cbi8vIHJlY29uc3RydWN0IG9yaWdpbmFsIElTTyBmb3JtYXRcbmV4cG9ydCBmdW5jdGlvbiBnbG9iYWxpc2VEYXRlVGltZShkYXRldGltZTogc3RyaW5nKSB7XG4gIGNvbnN0IGRhdGUgPSBuZXcgRGF0ZShkYXRldGltZSk7XG5cbiAgY29uc3Qgb2Zmc2V0ID0gLWRhdGUuZ2V0VGltZXpvbmVPZmZzZXQoKTtcbiAgY29uc3Qgb2Zmc2V0QWJzID0gTWF0aC5hYnMob2Zmc2V0KTtcblxuICBjb25zdCBzaWduID0gb2Zmc2V0ID49IDAgPyAnKycgOiAnLSc7XG4gIGNvbnN0IGhvdXJzID0gU3RyaW5nKE1hdGguZmxvb3Iob2Zmc2V0QWJzIC8gNjApKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBtaW51dGVzID0gU3RyaW5nKG9mZnNldEFicyAlIDYwKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBvZmZzZXRTdHJpbmcgPSBgJHtzaWdufSR7aG91cnN9OiR7bWludXRlc31gO1xuICBjb25zdCB5ZWFyID0gZGF0ZS5nZXRGdWxsWWVhcigpO1xuICBjb25zdCBtb250aCA9IFN0cmluZyhkYXRlLmdldE1vbnRoKCkgKyAxKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBkYXkgPSBTdHJpbmcoZGF0ZS5nZXREYXRlKCkpLnBhZFN0YXJ0KDIsICcwJyk7XG4gIGNvbnN0IGhvdXIgPSBTdHJpbmcoZGF0ZS5nZXRIb3VycygpKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBtaW51dGUgPSBTdHJpbmcoZGF0ZS5nZXRNaW51dGVzKCkpLnBhZFN0YXJ0KDIsICcwJyk7XG4gIGNvbnN0IHNlY29uZCA9IFN0cmluZyhkYXRlLmdldFNlY29uZHMoKSkucGFkU3RhcnQoMiwgJzAnKTtcblxuICBjb25zdCBkYXRlUGFydCA9IGAke3llYXJ9LSR7bW9udGh9LSR7ZGF5fWA7XG4gIGNvbnN0IHRpbWVQYXJ0ID0gYCR7aG91cn06JHttaW51dGV9OiR7c2Vjb25kfS4wMDBgO1xuXG4gIHJldHVybiBgJHtkYXRlUGFydH1UJHt0aW1lUGFydH0ke29mZnNldFN0cmluZ31gO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0QWdlKGRvYjogc3RyaW5nKSB7XG4gIGNvbnN0IHRoZW4gPSBuZXcgRGF0ZShkb2IpO1xuICBjb25zdCBub3cgPSBuZXcgRGF0ZSgpO1xuXG4gIGxldCBhZ2UgPSBub3cuZ2V0RnVsbFllYXIoKSAtIHRoZW4uZ2V0RnVsbFllYXIoKTtcblxuICBpZiAobm93LmdldE1vbnRoKCkgPCB0aGVuLmdldE1vbnRoKCkgfHxcbiAgICAobm93LmdldE1vbnRoKCkgPT09IHRoZW4uZ2V0TW9udGgoKSAmJiBub3cuZ2V0RGF0ZSgpIDwgdGhlbi5nZXREYXRlKCkpKSB7XG4gICAgYWdlLS07XG4gIH1cblxuICByZXR1cm4gYWdlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gdGl0bGVpc2UoY2FtZWw6IHN0cmluZykge1xuICBsZXQgc3BhY2VkID0gY2FtZWwucmVwbGFjZSgvKD89W0EtWl0pL2csICcgJyk7XG4gIHJldHVybiBzcGFjZWQudG9Mb3dlckNhc2UoKS5yZXBsYWNlKC9cXGJcXHcvZywgY2ggPT4gY2gudG9VcHBlckNhc2UoKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzaGFsbG93Q29tcGFyZShhOiBvYmplY3QsIGI6IG9iamVjdCkge1xuICByZXR1cm4gT2JqZWN0LmVudHJpZXMoYSkuZXZlcnkoKFtrLCB2XSkgPT4gYltrXSA9PT0gdik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzaGFsbG93RGlmZihhOiBvYmplY3QsIGI6IG9iamVjdCkge1xuICBjb25zdCBmcm9tID0gT2JqZWN0LmZyb21FbnRyaWVzKE9iamVjdC5lbnRyaWVzKGEpLmZpbHRlcigoW2ssIHZdKSA9PiBiW2tdICE9PSB2KSk7XG4gIGNvbnN0IHRvID0gT2JqZWN0LmZyb21FbnRyaWVzKE9iamVjdC5lbnRyaWVzKGIpLmZpbHRlcigoW2ssIF9dKSA9PiBmcm9tW2tdKSk7XG4gIHJldHVybiB7IGZyb20sIHRvIH07XG59XG4iXX0=
|
|
@@ -3,7 +3,7 @@ import { Injectable, Pipe, Component, Input, NgModule } from '@angular/core';
|
|
|
3
3
|
import * as i2 from '@angular/forms';
|
|
4
4
|
import { FormGroup, FormControl, Validators, ReactiveFormsModule } from '@angular/forms';
|
|
5
5
|
import { forkJoin, tap } from 'rxjs';
|
|
6
|
-
import * as i1 from 'shared';
|
|
6
|
+
import * as i1 from 'alimetry-shared';
|
|
7
7
|
import * as i3 from '@angular/common';
|
|
8
8
|
import { JsonPipe, CommonModule } from '@angular/common';
|
|
9
9
|
|
|
@@ -66,6 +66,11 @@ function titleise(camel) {
|
|
|
66
66
|
function shallowCompare(a, b) {
|
|
67
67
|
return Object.entries(a).every(([k, v]) => b[k] === v);
|
|
68
68
|
}
|
|
69
|
+
function shallowDiff(a, b) {
|
|
70
|
+
const from = Object.fromEntries(Object.entries(a).filter(([k, v]) => b[k] !== v));
|
|
71
|
+
const to = Object.fromEntries(Object.entries(b).filter(([k, _]) => from[k]));
|
|
72
|
+
return { from, to };
|
|
73
|
+
}
|
|
69
74
|
|
|
70
75
|
function setControlError(control, errorKey, errorValue) {
|
|
71
76
|
const currentErrors = control.errors || {};
|
|
@@ -182,8 +187,10 @@ const appDataPropertySetId = '34d46477-ad4b-47b4-b555-06358b92f60c';
|
|
|
182
187
|
const deviceDataPropertySetId = '7a32d851-5e89-46a6-a4b6-cca162dace29';
|
|
183
188
|
const propertyCodes = ['RecordingSession', 'Status', 'UserDetailChangeLog'];
|
|
184
189
|
class DetailEditComponent {
|
|
185
|
-
constructor(shared) {
|
|
190
|
+
constructor(shared, kvDiffers, itDiffers) {
|
|
186
191
|
this.shared = shared;
|
|
192
|
+
this.kvDiffers = kvDiffers;
|
|
193
|
+
this.itDiffers = itDiffers;
|
|
187
194
|
this.status = 'loading';
|
|
188
195
|
this.formGroup = new FormGroup({
|
|
189
196
|
dob: new FormControl('', [Validators.required, withinNYears(120, 'before')]),
|
|
@@ -235,7 +242,7 @@ class DetailEditComponent {
|
|
|
235
242
|
patient.age = getAge(mods.dob);
|
|
236
243
|
recordingSession.mealStart = globaliseDateTime(mods.mealStart);
|
|
237
244
|
recordingSession.mealEnd = globaliseDateTime(mods.mealEnd);
|
|
238
|
-
this.
|
|
245
|
+
this.logChange(mods);
|
|
239
246
|
return {
|
|
240
247
|
'Age': patient.age,
|
|
241
248
|
'RecordingSession': JSON.stringify(recordingSession),
|
|
@@ -243,56 +250,23 @@ class DetailEditComponent {
|
|
|
243
250
|
};
|
|
244
251
|
}
|
|
245
252
|
postDeviceData(devicePropertySetId, data) {
|
|
246
|
-
return this.shared.saveDeviceData(this.config.deviceDataModelId, this.deviceDataId, devicePropertySetId, this.config.owners[0]
|
|
253
|
+
return this.shared.saveDeviceData(this.config.deviceDataModelId, this.deviceDataId, devicePropertySetId, this.config.owners[0], data);
|
|
247
254
|
}
|
|
248
255
|
// patch the device data record with the modifications
|
|
249
256
|
saveModifications() {
|
|
250
257
|
const mods = this.formGroup.getRawValue();
|
|
251
258
|
const patchedDeviceData = this.integrateFields(mods, this.recordingSession);
|
|
252
|
-
const patchedUserData = Object.assign(this.
|
|
259
|
+
const patchedUserData = Object.assign(this.owner, { dateOfBirth: mods.dob });
|
|
253
260
|
this.status = 'sending';
|
|
254
261
|
forkJoin([
|
|
255
262
|
this.postDeviceData(appDataPropertySetId, patchedDeviceData),
|
|
256
|
-
this.shared.updateUser(this.config.owners[0]
|
|
263
|
+
this.shared.updateUser(this.config.owners[0], patchedUserData)
|
|
257
264
|
]).pipe(tap(() => this.regenReport())).subscribe();
|
|
258
265
|
}
|
|
259
|
-
|
|
260
|
-
const
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
for (const key in mods) {
|
|
264
|
-
if (this.initialFormState[key] !== mods[key]) {
|
|
265
|
-
prevChangeLog[key] = this.initialFormState[key];
|
|
266
|
-
newChangeLog[key] = mods[key];
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
const changeLog = {
|
|
270
|
-
author: this.currentUser.emailAddress,
|
|
271
|
-
prevChangeLog: prevChangeLog,
|
|
272
|
-
newChangeLog: newChangeLog,
|
|
273
|
-
timestamp: new Date().toISOString()
|
|
274
|
-
};
|
|
275
|
-
this.detailChangeLog.push(changeLog);
|
|
276
|
-
}
|
|
277
|
-
// get the device data record and initialise the form
|
|
278
|
-
loadDeviceData() {
|
|
279
|
-
this.shared.getDeviceData(this.config.deviceDataModelId, this.config.owners[0].userId, propertyCodes).subscribe({
|
|
280
|
-
next: (res) => {
|
|
281
|
-
var _a;
|
|
282
|
-
this.deviceDataId = res.deviceDataId;
|
|
283
|
-
this.recordingSession = JSON.parse(res.data['RecordingSession'].value);
|
|
284
|
-
this.initialiseForm(this.extractFields(this.recordingSession));
|
|
285
|
-
// checking for undefined in case no changes have been logged yet
|
|
286
|
-
this.detailChangeLog = res.data['UserDetailChangeLog'] === undefined ? [] : res.data['UserDetailChangeLog'].value;
|
|
287
|
-
this.status = ((_a = res.data['Status']) === null || _a === void 0 ? void 0 : _a.value) === 'Generating Report' ?
|
|
288
|
-
'generating' :
|
|
289
|
-
'success';
|
|
290
|
-
},
|
|
291
|
-
error: (err) => {
|
|
292
|
-
console.error(err);
|
|
293
|
-
this.status = "error";
|
|
294
|
-
}
|
|
295
|
-
});
|
|
266
|
+
logChange(mods) {
|
|
267
|
+
const entry = Object.assign({ author: this.currentUser.emailAddress, timestamp: new Date().toISOString() }, shallowDiff(this.initialFormState, mods));
|
|
268
|
+
this.detailChangeLog.push(entry);
|
|
269
|
+
console.log(this.detailChangeLog);
|
|
296
270
|
}
|
|
297
271
|
regenReport() {
|
|
298
272
|
this.postDeviceData(deviceDataPropertySetId, { 'Algorithmhasrun': false, 'AlgorithmRun': true }).subscribe(() => {
|
|
@@ -302,27 +276,69 @@ class DetailEditComponent {
|
|
|
302
276
|
resetModifications() {
|
|
303
277
|
this.formGroup.patchValue(this.initialFormState);
|
|
304
278
|
}
|
|
305
|
-
|
|
279
|
+
loadData(currentUserId, ownerId, deviceDataModelId) {
|
|
280
|
+
forkJoin([
|
|
281
|
+
this.shared.getUser(currentUserId),
|
|
282
|
+
this.shared.getUser(ownerId),
|
|
283
|
+
this.shared.getDeviceData(deviceDataModelId, ownerId, propertyCodes)
|
|
284
|
+
]).subscribe({
|
|
285
|
+
next: ([currentUser, owner, { deviceDataId, data: deviceData }]) => {
|
|
286
|
+
var _a, _b, _c;
|
|
287
|
+
this.currentUser = currentUser;
|
|
288
|
+
this.owner = owner;
|
|
289
|
+
this.deviceDataId = deviceDataId;
|
|
290
|
+
this.recordingSession = JSON.parse(deviceData['RecordingSession'].value);
|
|
291
|
+
this.initialiseForm(this.extractFields(this.recordingSession));
|
|
292
|
+
// data field uninitialised for unmodified record
|
|
293
|
+
this.detailChangeLog = (_b = (_a = deviceData['UserDetailChangeLog']) === null || _a === void 0 ? void 0 : _a.value) !== null && _b !== void 0 ? _b : [];
|
|
294
|
+
this.status = ((_c = deviceData['Status']) === null || _c === void 0 ? void 0 : _c.value) === 'Generating Report' ?
|
|
295
|
+
'generating' : 'success';
|
|
296
|
+
},
|
|
297
|
+
error: (err) => {
|
|
298
|
+
console.error(err);
|
|
299
|
+
this.status = "error";
|
|
300
|
+
}
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
init() {
|
|
306
304
|
var _a, _b, _c;
|
|
307
|
-
if (!((
|
|
305
|
+
if (!((_a = this.config) === null || _a === void 0 ? void 0 : _a.currentUserId) || !this.config.deviceDataModelId) {
|
|
306
|
+
this.status = 'cfgerror';
|
|
307
|
+
return;
|
|
308
|
+
}
|
|
309
|
+
else if (!((_c = (_b = this.config) === null || _b === void 0 ? void 0 : _b.owners) === null || _c === void 0 ? void 0 : _c[0])) {
|
|
308
310
|
this.status = 'nouser';
|
|
309
311
|
return;
|
|
310
312
|
}
|
|
311
|
-
this.
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
313
|
+
this.status = 'loading';
|
|
314
|
+
this.loadData(this.config.currentUserId, this.config.owners[0], this.config.deviceDataModelId);
|
|
315
|
+
}
|
|
316
|
+
// this should be called from the parent using the ViewChild attr
|
|
317
|
+
refresh() {
|
|
318
|
+
this.init();
|
|
319
|
+
}
|
|
320
|
+
// NOTE: there could be something wrong / unusual here but it seems to work ok
|
|
321
|
+
ngOnInit() {
|
|
322
|
+
var _a;
|
|
323
|
+
if (this.config && !this.configDiffer)
|
|
324
|
+
this.configDiffer = this.kvDiffers.find(this.config).create();
|
|
325
|
+
if (((_a = this.config) === null || _a === void 0 ? void 0 : _a.owners) && !this.ownersDiffer)
|
|
326
|
+
this.ownersDiffer = this.itDiffers.find(this.config.owners).create();
|
|
327
|
+
}
|
|
328
|
+
ngDoCheck() {
|
|
329
|
+
var _a, _b, _c;
|
|
330
|
+
const configChanges = (_a = this.configDiffer) === null || _a === void 0 ? void 0 : _a.diff(this.config);
|
|
331
|
+
const ownersChanges = (_b = this.ownersDiffer) === null || _b === void 0 ? void 0 : _b.diff((_c = this.config) === null || _c === void 0 ? void 0 : _c.owners);
|
|
332
|
+
if (configChanges || ownersChanges)
|
|
333
|
+
this.init();
|
|
318
334
|
}
|
|
319
335
|
}
|
|
320
|
-
DetailEditComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, deps: [{ token: i1.SharedService }], target: i0.ɵɵFactoryTarget.Component });
|
|
321
|
-
DetailEditComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: DetailEditComponent, selector: "lib-detail-edit", inputs: { config: "config" }, ngImport: i0, template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"status
|
|
336
|
+
DetailEditComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, deps: [{ token: i1.SharedService }, { token: i0.KeyValueDiffers }, { token: i0.IterableDiffers }], target: i0.ɵɵFactoryTarget.Component });
|
|
337
|
+
DetailEditComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: DetailEditComponent, selector: "lib-detail-edit", inputs: { config: "config" }, ngImport: i0, template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'cfgerror'\">\n The widget has not been configured correctly, or is missing configuration information.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"!status.includes('error') && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n <div class=\"inputs\">\n <div class=\"form-row\">\n <label for=\"dob\">Date of Birth:</label>\n <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"weight\">Patient Weight (kg):</label>\n <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"height\">Patient Height (cm):</label>\n <input id=\"height\" type=\"number\" formControlName=\"height\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealStart\">Meal Start Time:</label>\n <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealEnd\">Meal End Time:</label>\n <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n </div>\n </div>\n\n <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n <div *ngFor=\"let control of formGroup.controls | entries\">\n {{ parseInputErrors(control[0], control[1]) }}\n </div>\n </div>\n\n <div *ngIf=\"status === 'generating'\" class=\"info\">\n <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n </div>\n\n <div class=\"buttons\">\n <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n </div>\n </form>\n</div>\n", styles: [".wrapper{width:100%;height:100%}.status{font-size:2rem;margin:5rem auto;text-align:center}form{display:flex;flex-direction:column;padding:2rem;gap:2rem;font-size:.9rem;max-width:70rem;margin:0 auto}form .inputs{display:flex;flex-flow:row wrap;gap:1rem 3rem}form .inputs .form-row{display:flex;gap:1rem;align-items:center;flex:1 0 calc(50% - 3rem);min-width:400px}form .inputs .form-row label{min-width:9rem}form .inputs .form-row input{background:#f5f5f6;height:36px;border:none;padding-inline:10px;flex-grow:1}form.ng-dirty .form-row .ng-invalid{outline:1px auto red}form .buttons{display:flex;gap:1rem}form .buttons button{padding:.5rem 1rem;height:36px;border:none;font-size:inherit;background:#f5f5f6}form .buttons button:not([disabled]){cursor:pointer}form .buttons button[type=submit]{background:#242C69;color:#fff}form .buttons button[disabled]{background:#fafafa;cursor:not-allowed}form .buttons button[type=submit][disabled]{background:#444C89}form .buttons button:first-child{margin-right:auto}.errors{padding:1rem;background:#fee;border:2px solid red;border-radius:4px;color:red}.info{padding:1rem;background:#eef;border:2px solid #242C69;border-radius:4px;color:#242c69}\n"], dependencies: [{ kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: EntriesPipe, name: "entries" }] });
|
|
322
338
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, decorators: [{
|
|
323
339
|
type: Component,
|
|
324
|
-
args: [{ selector: 'lib-detail-edit', template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"status
|
|
325
|
-
}], ctorParameters: function () { return [{ type: i1.SharedService }]; }, propDecorators: { config: [{
|
|
340
|
+
args: [{ selector: 'lib-detail-edit', template: "<div class=\"wrapper\">\n <div class=\"status\" *ngIf=\"status === 'error'\">\n An error occured while trying to get the details.\n </div>\n <div class=\"status\" *ngIf=\"status === 'cfgerror'\">\n The widget has not been configured correctly, or is missing configuration information.\n </div>\n <div class=\"status\" *ngIf=\"status === 'nouser'\">\n There is no user specified as the data owner.\n </div>\n <form *ngIf=\"!status.includes('error') && status !== 'nouser'\" [formGroup]=\"formGroup\" (ngSubmit)=\"saveModifications()\">\n <div class=\"inputs\">\n <div class=\"form-row\">\n <label for=\"dob\">Date of Birth:</label>\n <input id=\"dob\" type=\"date\" formControlName=\"dob\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"weight\">Patient Weight (kg):</label>\n <input id=\"weight\" type=\"number\" formControlName=\"weight\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"height\">Patient Height (cm):</label>\n <input id=\"height\" type=\"number\" formControlName=\"height\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealStart\">Meal Start Time:</label>\n <input id=\"mealStart\" type=\"datetime-local\" formControlName=\"mealStart\">\n </div>\n\n <div class=\"form-row\">\n <label for=\"mealEnd\">Meal End Time:</label>\n <input id=\"mealEnd\" type=\"datetime-local\" formControlName=\"mealEnd\">\n </div>\n </div>\n\n <div *ngIf=\"!formGroup.valid && formGroup.dirty\" class=\"errors\">\n <div *ngFor=\"let control of formGroup.controls | entries\">\n {{ parseInputErrors(control[0], control[1]) }}\n </div>\n </div>\n\n <div *ngIf=\"status === 'generating'\" class=\"info\">\n <div>The report is generating at the moment, check back later if you need to modify this entry.</div>\n </div>\n\n <div class=\"buttons\">\n <button type=\"button\" (click)=\"regenReport()\" [disabled]=\"status === 'sending' || status === 'generating' || status === 'loading'\">Regenerate Report</button>\n <button type=\"button\" (click)=\"resetModifications()\" [disabled]=\"status === 'sending' || status === 'generating' || formGroup.pristine\">Cancel Modification</button>\n <button type=\"submit\" [disabled]=\"status === 'sending' || status === 'generating' || !formGroup.valid || formGroup.pristine\">Save Details</button>\n </div>\n </form>\n</div>\n", styles: [".wrapper{width:100%;height:100%}.status{font-size:2rem;margin:5rem auto;text-align:center}form{display:flex;flex-direction:column;padding:2rem;gap:2rem;font-size:.9rem;max-width:70rem;margin:0 auto}form .inputs{display:flex;flex-flow:row wrap;gap:1rem 3rem}form .inputs .form-row{display:flex;gap:1rem;align-items:center;flex:1 0 calc(50% - 3rem);min-width:400px}form .inputs .form-row label{min-width:9rem}form .inputs .form-row input{background:#f5f5f6;height:36px;border:none;padding-inline:10px;flex-grow:1}form.ng-dirty .form-row .ng-invalid{outline:1px auto red}form .buttons{display:flex;gap:1rem}form .buttons button{padding:.5rem 1rem;height:36px;border:none;font-size:inherit;background:#f5f5f6}form .buttons button:not([disabled]){cursor:pointer}form .buttons button[type=submit]{background:#242C69;color:#fff}form .buttons button[disabled]{background:#fafafa;cursor:not-allowed}form .buttons button[type=submit][disabled]{background:#444C89}form .buttons button:first-child{margin-right:auto}.errors{padding:1rem;background:#fee;border:2px solid red;border-radius:4px;color:red}.info{padding:1rem;background:#eef;border:2px solid #242C69;border-radius:4px;color:#242c69}\n"] }]
|
|
341
|
+
}], ctorParameters: function () { return [{ type: i1.SharedService }, { type: i0.KeyValueDiffers }, { type: i0.IterableDiffers }]; }, propDecorators: { config: [{
|
|
326
342
|
type: Input
|
|
327
343
|
}] } });
|
|
328
344
|
|