alimetry-detail-edit 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,15 @@
1
+ # DetailEdit
2
+
3
+ ## Use Details
4
+
5
+ Selector: `lib-detail-edit`
6
+
7
+ ### Config Object
8
+
9
+ {
10
+ owners: User[],
11
+ deviceDataModelId,
12
+ currentUser: User
13
+ }
14
+
15
+ [NOTE:] the widget relies on the owner's user object containing the `userId` and `dateOfBirth` fields, and the current user's object containing the `emailAddress` field.
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWxpbWV0cnktZGV0YWlsLWVkaXQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9kZXRhaWwtZWRpdC9zcmMvYWxpbWV0cnktZGV0YWlsLWVkaXQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
@@ -0,0 +1,150 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import { FormControl, FormGroup, Validators } from '@angular/forms';
3
+ import { forkJoin, tap } from 'rxjs';
4
+ import { getAge, globaliseDateTime, localiseDateTime, shallowCompare } from './utils';
5
+ import { dateComparison, withinNYears, withinNHours, translateError } from './validators';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "shared";
8
+ import * as i2 from "@angular/forms";
9
+ import * as i3 from "@angular/common";
10
+ import * as i4 from "./entries.pipe";
11
+ const appDataPropertySetId = '34d46477-ad4b-47b4-b555-06358b92f60c';
12
+ const deviceDataPropertySetId = '7a32d851-5e89-46a6-a4b6-cca162dace29';
13
+ const propertyCodes = ['RecordingSession', 'Status', 'UserDetailChangeLog'];
14
+ export class DetailEditComponent {
15
+ constructor(shared) {
16
+ this.shared = shared;
17
+ this.status = 'loading';
18
+ this.formGroup = new FormGroup({
19
+ dob: new FormControl('', [Validators.required, withinNYears(120, 'before')]),
20
+ height: new FormControl(0, [Validators.required, Validators.min(30), Validators.max(272)]),
21
+ weight: new FormControl(0, [Validators.required, Validators.min(1), Validators.max(635)]),
22
+ mealStart: new FormControl('', [Validators.required]),
23
+ mealEnd: new FormControl('', [Validators.required])
24
+ }, [
25
+ dateComparison('mealEnd', 'mealStart', 'after'),
26
+ dateComparison('mealStart', 'dob', 'after'),
27
+ ]);
28
+ }
29
+ parseInputErrors(name, control) {
30
+ return translateError(name, control);
31
+ }
32
+ // make sure than modifications that don't actually differ from the current are marked as such
33
+ monitorFormChanges() {
34
+ this.formGroup.valueChanges.subscribe(current => {
35
+ if (shallowCompare(current, this.initialFormState)) {
36
+ this.formGroup.markAsPristine();
37
+ }
38
+ });
39
+ }
40
+ // handle the relevant data from the response body
41
+ extractFields(recordingSession) {
42
+ const { patient, mealStart, mealEnd, recordingStart } = recordingSession;
43
+ this.recordingStart = recordingStart;
44
+ return {
45
+ dob: this.config.owners[0].dateOfBirth,
46
+ height: patient.height,
47
+ weight: patient.weight,
48
+ mealStart: localiseDateTime(mealStart),
49
+ mealEnd: localiseDateTime(mealEnd)
50
+ };
51
+ }
52
+ initialiseForm(state) {
53
+ this.formGroup.patchValue(state);
54
+ this.formGroup.markAsPristine();
55
+ this.formGroup.get('mealStart').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));
56
+ this.formGroup.get('mealEnd').addValidators(withinNHours(24, 'after', new Date(this.recordingStart)));
57
+ this.initialFormState = state; // save this for use in resets
58
+ this.monitorFormChanges();
59
+ }
60
+ // construct a patch object for the save request
61
+ integrateFields(mods, recordingSession) {
62
+ const { patient } = recordingSession;
63
+ patient.height = mods.height;
64
+ patient.weight = mods.weight;
65
+ patient.age = getAge(mods.dob);
66
+ recordingSession.mealStart = globaliseDateTime(mods.mealStart);
67
+ recordingSession.mealEnd = globaliseDateTime(mods.mealEnd);
68
+ this.appendChangeLog(mods);
69
+ return {
70
+ 'Age': patient.age,
71
+ 'RecordingSession': JSON.stringify(recordingSession),
72
+ 'UserDetailChangeLog': this.detailChangeLog
73
+ };
74
+ }
75
+ postDeviceData(devicePropertySetId, data) {
76
+ return this.shared.saveDeviceData(this.config.deviceDataModelId, this.deviceDataId, devicePropertySetId, this.config.owners[0].userId, data);
77
+ }
78
+ // patch the device data record with the modifications
79
+ saveModifications() {
80
+ const mods = this.formGroup.getRawValue();
81
+ const patchedDeviceData = this.integrateFields(mods, this.recordingSession);
82
+ const patchedUserData = Object.assign(this.config.owners[0], { dateOfBirth: mods.dob });
83
+ this.status = 'sending';
84
+ forkJoin([
85
+ this.postDeviceData(appDataPropertySetId, patchedDeviceData),
86
+ this.shared.updateUser(this.config.owners[0].userId, patchedUserData)
87
+ ]).pipe(tap(() => this.regenReport())).subscribe();
88
+ }
89
+ appendChangeLog(mods) {
90
+ const prevChangeLog = {};
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 = {
100
+ author: this.config.currentUser.emailAddress,
101
+ prevChangeLog: prevChangeLog,
102
+ newChangeLog: newChangeLog,
103
+ timestamp: new Date().toISOString()
104
+ };
105
+ this.detailChangeLog.push(changeLog);
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
+ });
125
+ }
126
+ regenReport() {
127
+ this.postDeviceData(deviceDataPropertySetId, { 'Algorithmhasrun': false, 'AlgorithmRun': true }).subscribe(() => {
128
+ this.status = 'generating'; // assume report starts generating immediately
129
+ });
130
+ }
131
+ resetModifications() {
132
+ this.formGroup.patchValue(this.initialFormState);
133
+ }
134
+ ngOnInit() {
135
+ if (!this.config?.owners?.[0]?.userId) {
136
+ this.status = 'nouser';
137
+ return;
138
+ }
139
+ this.loadDeviceData();
140
+ }
141
+ }
142
+ DetailEditComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, deps: [{ token: i1.SharedService }], target: i0.ɵɵFactoryTarget.Component });
143
+ 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 !== '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" }] });
144
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditComponent, decorators: [{
145
+ type: Component,
146
+ 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 !== '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"] }]
147
+ }], ctorParameters: function () { return [{ type: i1.SharedService }]; }, propDecorators: { config: [{
148
+ type: Input
149
+ }] } });
150
+ //# 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, switchMap, tap } from 'rxjs';\nimport { Config, SharedService } 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\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.config.owners[0].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.config.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\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"]}
@@ -0,0 +1,33 @@
1
+ import { CommonModule, JsonPipe } from '@angular/common';
2
+ import { NgModule } from '@angular/core';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { DetailEditComponent } from './detail-edit.component';
5
+ import { EntriesPipe } from './entries.pipe';
6
+ import * as i0 from "@angular/core";
7
+ export class DetailEditModule {
8
+ }
9
+ DetailEditModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
10
+ DetailEditModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.10", ngImport: i0, type: DetailEditModule, declarations: [DetailEditComponent,
11
+ EntriesPipe], imports: [JsonPipe,
12
+ ReactiveFormsModule,
13
+ CommonModule], exports: [DetailEditComponent] });
14
+ DetailEditModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditModule, imports: [ReactiveFormsModule,
15
+ CommonModule] });
16
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditModule, decorators: [{
17
+ type: NgModule,
18
+ args: [{
19
+ declarations: [
20
+ DetailEditComponent,
21
+ EntriesPipe
22
+ ],
23
+ imports: [
24
+ JsonPipe,
25
+ ReactiveFormsModule,
26
+ CommonModule
27
+ ],
28
+ exports: [
29
+ DetailEditComponent
30
+ ]
31
+ }]
32
+ }] });
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWVkaXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvZGV0YWlsLWVkaXQvc3JjL2xpYi9kZXRhaWwtZWRpdC5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxRQUFRLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQzlELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7QUFrQjdDLE1BQU0sT0FBTyxnQkFBZ0I7OzhHQUFoQixnQkFBZ0I7K0dBQWhCLGdCQUFnQixpQkFaekIsbUJBQW1CO1FBQ25CLFdBQVcsYUFHWCxRQUFRO1FBQ1IsbUJBQW1CO1FBQ25CLFlBQVksYUFHWixtQkFBbUI7K0dBR1YsZ0JBQWdCLFlBUHpCLG1CQUFtQjtRQUNuQixZQUFZOzRGQU1ILGdCQUFnQjtrQkFkNUIsUUFBUTttQkFBQztvQkFDUixZQUFZLEVBQUU7d0JBQ1osbUJBQW1CO3dCQUNuQixXQUFXO3FCQUNaO29CQUNELE9BQU8sRUFBRTt3QkFDUCxRQUFRO3dCQUNSLG1CQUFtQjt3QkFDbkIsWUFBWTtxQkFDYjtvQkFDRCxPQUFPLEVBQUU7d0JBQ1AsbUJBQW1CO3FCQUNwQjtpQkFDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSwgSnNvblBpcGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBEZXRhaWxFZGl0Q29tcG9uZW50IH0gZnJvbSAnLi9kZXRhaWwtZWRpdC5jb21wb25lbnQnO1xuaW1wb3J0IHsgRW50cmllc1BpcGUgfSBmcm9tICcuL2VudHJpZXMucGlwZSc7XG5cblxuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBEZXRhaWxFZGl0Q29tcG9uZW50LFxuICAgIEVudHJpZXNQaXBlXG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBKc29uUGlwZSxcbiAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgIENvbW1vbk1vZHVsZVxuICBdLFxuICBleHBvcnRzOiBbXG4gICAgRGV0YWlsRWRpdENvbXBvbmVudFxuICBdXG59KVxuZXhwb3J0IGNsYXNzIERldGFpbEVkaXRNb2R1bGUgeyB9XG4iXX0=
@@ -0,0 +1,14 @@
1
+ import { Injectable } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class DetailEditService {
4
+ constructor() { }
5
+ }
6
+ DetailEditService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
7
+ DetailEditService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditService, providedIn: 'root' });
8
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: DetailEditService, decorators: [{
9
+ type: Injectable,
10
+ args: [{
11
+ providedIn: 'root'
12
+ }]
13
+ }], ctorParameters: function () { return []; } });
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV0YWlsLWVkaXQuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2RldGFpbC1lZGl0L3NyYy9saWIvZGV0YWlsLWVkaXQuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUszQyxNQUFNLE9BQU8saUJBQWlCO0lBRTVCLGdCQUFnQixDQUFDOzsrR0FGTixpQkFBaUI7bUhBQWpCLGlCQUFpQixjQUZoQixNQUFNOzRGQUVQLGlCQUFpQjtrQkFIN0IsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIERldGFpbEVkaXRTZXJ2aWNlIHtcblxuICBjb25zdHJ1Y3RvcigpIHsgfVxufVxuIl19
@@ -0,0 +1,16 @@
1
+ import { Pipe } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class EntriesPipe {
4
+ transform(value) {
5
+ return Object.entries(value);
6
+ }
7
+ }
8
+ EntriesPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: EntriesPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe });
9
+ EntriesPipe.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "15.2.10", ngImport: i0, type: EntriesPipe, name: "entries" });
10
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: EntriesPipe, decorators: [{
11
+ type: Pipe,
12
+ args: [{
13
+ name: 'entries'
14
+ }]
15
+ }] });
16
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW50cmllcy5waXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvZGV0YWlsLWVkaXQvc3JjL2xpYi9lbnRyaWVzLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7O0FBS3BELE1BQU0sT0FBTyxXQUFXO0lBQ3RCLFNBQVMsQ0FBQyxLQUFhO1FBQ3JCLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvQixDQUFDOzt5R0FIVSxXQUFXO3VHQUFYLFdBQVc7NEZBQVgsV0FBVztrQkFIdkIsSUFBSTttQkFBQztvQkFDSixJQUFJLEVBQUUsU0FBUztpQkFDaEIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQaXBlLCBQaXBlVHJhbnNmb3JtIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBQaXBlKHtcbiAgbmFtZTogJ2VudHJpZXMnXG59KVxuZXhwb3J0IGNsYXNzIEVudHJpZXNQaXBlIGltcGxlbWVudHMgUGlwZVRyYW5zZm9ybSB7XG4gIHRyYW5zZm9ybSh2YWx1ZTogb2JqZWN0KTogW3N0cmluZywgYW55XVtdIHtcbiAgICByZXR1cm4gT2JqZWN0LmVudHJpZXModmFsdWUpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,48 @@
1
+ // strip the timezone offset to create format usable by <input>
2
+ export function localiseDateTime(datetime) {
3
+ const date = new Date(datetime);
4
+ const year = date.getFullYear();
5
+ const month = (date.getMonth() + 1).toString().padStart(2, '0');
6
+ const day = date.getDate().toString().padStart(2, '0');
7
+ const hour = date.getHours().toString().padStart(2, '0');
8
+ const minute = date.getMinutes().toString().padStart(2, '0');
9
+ const second = date.getSeconds().toString().padStart(2, '0');
10
+ return `${year}-${month}-${day}T${hour}:${minute}:${second}`;
11
+ }
12
+ // reconstruct original ISO format
13
+ export function globaliseDateTime(datetime) {
14
+ const date = new Date(datetime);
15
+ const offset = -date.getTimezoneOffset();
16
+ const offsetAbs = Math.abs(offset);
17
+ const sign = offset >= 0 ? '+' : '-';
18
+ const hours = String(Math.floor(offsetAbs / 60)).padStart(2, '0');
19
+ const minutes = String(offsetAbs % 60).padStart(2, '0');
20
+ const offsetString = `${sign}${hours}:${minutes}`;
21
+ const year = date.getFullYear();
22
+ const month = String(date.getMonth() + 1).padStart(2, '0');
23
+ const day = String(date.getDate()).padStart(2, '0');
24
+ const hour = String(date.getHours()).padStart(2, '0');
25
+ const minute = String(date.getMinutes()).padStart(2, '0');
26
+ const second = String(date.getSeconds()).padStart(2, '0');
27
+ const datePart = `${year}-${month}-${day}`;
28
+ const timePart = `${hour}:${minute}:${second}.000`;
29
+ return `${datePart}T${timePart}${offsetString}`;
30
+ }
31
+ export function getAge(dob) {
32
+ const then = new Date(dob);
33
+ const now = new Date();
34
+ let age = now.getFullYear() - then.getFullYear();
35
+ if (now.getMonth() < then.getMonth() ||
36
+ (now.getMonth() === then.getMonth() && now.getDate() < then.getDate())) {
37
+ age--;
38
+ }
39
+ return age;
40
+ }
41
+ export function titleise(camel) {
42
+ let spaced = camel.replace(/(?=[A-Z])/g, ' ');
43
+ return spaced.toLowerCase().replace(/\b\w/g, ch => ch.toUpperCase());
44
+ }
45
+ export function shallowCompare(a, b) {
46
+ return Object.entries(a).every(([k, v]) => b[k] === v);
47
+ }
48
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9kZXRhaWwtZWRpdC9zcmMvbGliL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLCtEQUErRDtBQUMvRCxNQUFNLFVBQVUsZ0JBQWdCLENBQUMsUUFBZ0I7SUFDL0MsTUFBTSxJQUFJLEdBQUcsSUFBSSxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7SUFFaEMsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQUMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUNqRyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN2RCxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN6RCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUM3RCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUMsUUFBUSxFQUFFLENBQUMsUUFBUSxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUU3RCxPQUFPLEdBQUcsSUFBSSxJQUFJLEtBQUssSUFBSSxHQUFHLElBQUksSUFBSSxJQUFJLE1BQU0sSUFBSSxNQUFNLEVBQUUsQ0FBQztBQUMvRCxDQUFDO0FBRUQsa0NBQWtDO0FBQ2xDLE1BQU0sVUFBVSxpQkFB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LENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUM7QUFDdkUsQ0FBQztBQUVELE1BQU0sVUFBVSxjQUFjLENBQUMsQ0FBUyxFQUFFLENBQVM7SUFDakQsT0FBTyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFDekQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIHN0cmlwIHRoZSB0aW1lem9uZSBvZmZzZXQgdG8gY3JlYXRlIGZvcm1hdCB1c2FibGUgYnkgPGlucHV0PlxuZXhwb3J0IGZ1bmN0aW9uIGxvY2FsaXNlRGF0ZVRpbWUoZGF0ZXRpbWU6IHN0cmluZykge1xuICBjb25zdCBkYXRlID0gbmV3IERhdGUoZGF0ZXRpbWUpO1xuXG4gIGNvbnN0IHllYXIgPSBkYXRlLmdldEZ1bGxZZWFyKCk7IGNvbnN0IG1vbnRoID0gKGRhdGUuZ2V0TW9udGgoKSArIDEpLnRvU3RyaW5nKCkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3QgZGF5ID0gZGF0ZS5nZXREYXRlKCkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBob3VyID0gZGF0ZS5nZXRIb3VycygpLnRvU3RyaW5nKCkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3QgbWludXRlID0gZGF0ZS5nZXRNaW51dGVzKCkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBzZWNvbmQgPSBkYXRlLmdldFNlY29uZHMoKS50b1N0cmluZygpLnBhZFN0YXJ0KDIsICcwJyk7XG5cbiAgcmV0dXJuIGAke3llYXJ9LSR7bW9udGh9LSR7ZGF5fVQke2hvdXJ9OiR7bWludXRlfToke3NlY29uZH1gO1xufVxuXG4vLyByZWNvbnN0cnVjdCBvcmlnaW5hbCBJU08gZm9ybWF0XG5leHBvcnQgZnVuY3Rpb24gZ2xvYmFsaXNlRGF0ZVRpbWUoZGF0ZXRpbWU6IHN0cmluZykge1xuICBjb25zdCBkYXRlID0gbmV3IERhdGUoZGF0ZXRpbWUpO1xuXG4gIGNvbnN0IG9mZnNldCA9IC1kYXRlLmdldFRpbWV6b25lT2Zmc2V0KCk7XG4gIGNvbnN0IG9mZnNldEFicyA9IE1hdGguYWJzKG9mZnNldCk7XG5cbiAgY29uc3Qgc2lnbiA9IG9mZnNldCA+PSAwID8gJysnIDogJy0nO1xuICBjb25zdCBob3VycyA9IFN0cmluZyhNYXRoLmZsb29yKG9mZnNldEFicyAvIDYwKSkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3QgbWludXRlcyA9IFN0cmluZyhvZmZzZXRBYnMgJSA2MCkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3Qgb2Zmc2V0U3RyaW5nID0gYCR7c2lnbn0ke2hvdXJzfToke21pbnV0ZXN9YDtcbiAgY29uc3QgeWVhciA9IGRhdGUuZ2V0RnVsbFllYXIoKTtcbiAgY29uc3QgbW9udGggPSBTdHJpbmcoZGF0ZS5nZXRNb250aCgpICsgMSkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3QgZGF5ID0gU3RyaW5nKGRhdGUuZ2V0RGF0ZSgpKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBob3VyID0gU3RyaW5nKGRhdGUuZ2V0SG91cnMoKSkucGFkU3RhcnQoMiwgJzAnKTtcbiAgY29uc3QgbWludXRlID0gU3RyaW5nKGRhdGUuZ2V0TWludXRlcygpKS5wYWRTdGFydCgyLCAnMCcpO1xuICBjb25zdCBzZWNvbmQgPSBTdHJpbmcoZGF0ZS5nZXRTZWNvbmRzKCkpLnBhZFN0YXJ0KDIsICcwJyk7XG5cbiAgY29uc3QgZGF0ZVBhcnQgPSBgJHt5ZWFyfS0ke21vbnRofS0ke2RheX1gO1xuICBjb25zdCB0aW1lUGFydCA9IGAke2hvdXJ9OiR7bWludXRlfToke3NlY29uZH0uMDAwYDtcblxuICByZXR1cm4gYCR7ZGF0ZVBhcnR9VCR7dGltZVBhcnR9JHtvZmZzZXRTdHJpbmd9YDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldEFnZShkb2I6IHN0cmluZykge1xuICBjb25zdCB0aGVuID0gbmV3IERhdGUoZG9iKTtcbiAgY29uc3Qgbm93ID0gbmV3IERhdGUoKTtcblxuICBsZXQgYWdlID0gbm93LmdldEZ1bGxZZWFyKCkgLSB0aGVuLmdldEZ1bGxZZWFyKCk7XG5cbiAgaWYgKG5vdy5nZXRNb250aCgpIDwgdGhlbi5nZXRNb250aCgpIHx8XG4gICAgKG5vdy5nZXRNb250aCgpID09PSB0aGVuLmdldE1vbnRoKCkgJiYgbm93LmdldERhdGUoKSA8IHRoZW4uZ2V0RGF0ZSgpKSkge1xuICAgIGFnZS0tO1xuICB9XG5cbiAgcmV0dXJuIGFnZTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHRpdGxlaXNlKGNhbWVsOiBzdHJpbmcpIHtcbiAgbGV0IHNwYWNlZCA9IGNhbWVsLnJlcGxhY2UoLyg/PVtBLVpdKS9nLCAnICcpO1xuICByZXR1cm4gc3BhY2VkLnRvTG93ZXJDYXNlKCkucmVwbGFjZSgvXFxiXFx3L2csIGNoID0+IGNoLnRvVXBwZXJDYXNlKCkpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2hhbGxvd0NvbXBhcmUoYTogb2JqZWN0LCBiOiBvYmplY3QpIHtcbiAgcmV0dXJuIE9iamVjdC5lbnRyaWVzKGEpLmV2ZXJ5KChbaywgdl0pID0+IGJba10gPT09IHYpO1xufVxuIl19
@@ -0,0 +1,99 @@
1
+ import { FormGroup } from "@angular/forms";
2
+ import { titleise } from "./utils";
3
+ function setControlError(control, errorKey, errorValue) {
4
+ const currentErrors = control.errors || {};
5
+ if (currentErrors[errorKey])
6
+ return;
7
+ control.setErrors({ ...currentErrors, [errorKey]: errorValue });
8
+ }
9
+ function clearControlError(control, errorKey) {
10
+ const currentErrors = control.errors || {};
11
+ if (!currentErrors[errorKey])
12
+ return;
13
+ control.setErrors({ ...currentErrors, [errorKey]: null });
14
+ }
15
+ export function dateComparison(a, b, comparator) {
16
+ return function (control) {
17
+ if (!(control instanceof FormGroup))
18
+ return null;
19
+ const formGroup = control;
20
+ const aControl = formGroup.get(a);
21
+ const bControl = formGroup.get(b);
22
+ if (!aControl?.value || !bControl?.value) {
23
+ clearControlError(aControl, 'dateRange');
24
+ return null;
25
+ }
26
+ const aDate = new Date(aControl.value);
27
+ const bDate = new Date(bControl.value);
28
+ const hasError = (comparator === 'after' && aDate <= bDate ||
29
+ comparator === 'before' && aDate >= bDate);
30
+ if (hasError) {
31
+ setControlError(aControl, 'dateRange', { against: b, comparator });
32
+ return { dateRange: true };
33
+ }
34
+ clearControlError(aControl, 'dateRange');
35
+ return null;
36
+ };
37
+ }
38
+ export function withinNYears(n, dir = 'both', of) {
39
+ return function (control) {
40
+ const timestamp = control?.value;
41
+ if (!timestamp || isNaN(new Date(timestamp).getTime()))
42
+ return null;
43
+ const date = new Date(timestamp);
44
+ const against = of ?? new Date();
45
+ const min = new Date(against);
46
+ min.setFullYear(against.getFullYear() - n);
47
+ const max = new Date(against);
48
+ max.setFullYear(against.getFullYear() + n);
49
+ let isValid = false;
50
+ if (dir === 'after')
51
+ isValid = date < max && date > against;
52
+ else if (dir === 'before')
53
+ isValid = date > min && date < against;
54
+ else
55
+ isValid = date > min && date < max;
56
+ return isValid ? null : { notWithinYears: { n, against, dir } };
57
+ };
58
+ }
59
+ export function withinNHours(n, dir = 'both', of) {
60
+ return function (control) {
61
+ const timestamp = control?.value;
62
+ if (!timestamp || isNaN(new Date(timestamp).getTime()))
63
+ return null;
64
+ const date = new Date(timestamp);
65
+ const against = of ?? new Date();
66
+ const min = new Date(against);
67
+ min.setHours(against.getHours() - n);
68
+ const max = new Date(against);
69
+ max.setHours(against.getHours() + n);
70
+ let isValid = false;
71
+ if (dir === 'after')
72
+ isValid = date < max && date > against;
73
+ else if (dir === 'before')
74
+ isValid = date > min && date < against;
75
+ else
76
+ isValid = date > min && date < max;
77
+ return isValid ? null : { notWithinHours: { n, against, dir } };
78
+ };
79
+ }
80
+ // translate validation errors into human readable strings
81
+ export function translateError(name, control) {
82
+ const { errors } = control;
83
+ if (control.hasError('required'))
84
+ return `${titleise(name)} is a required value.`;
85
+ else if (control.hasError('min'))
86
+ return `${titleise(name)} needs to be a minimum of ${errors.min.min}.`;
87
+ else if (control.hasError('max'))
88
+ return `${titleise(name)} needs to be a maximum of ${errors.max.max}.`;
89
+ else if (control.hasError('future'))
90
+ return `${titleise(name)} needs to be in the past.`;
91
+ else if (control.hasError('dateRange'))
92
+ return `${titleise(name)} needs to be ${errors.dateRange.comparator} ${titleise(errors.dateRange.against)}.`;
93
+ else if (control.hasError('notWithinYears'))
94
+ return `${titleise(name)} needs to be within ${errors.notWithinYears.n} years of ${errors.notWithinYears.against}`;
95
+ else if (control.hasError('notWithinHours'))
96
+ return `${titleise(name)} needs to be within ${errors.notWithinHours.n} hours of ${errors.notWithinHours.against}`;
97
+ return null;
98
+ }
99
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validators.js","sourceRoot":"","sources":["../../../../projects/detail-edit/src/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractControl, FormControl, FormGroup, ValidationErrors, ValidatorFn } from \"@angular/forms\";\nimport { titleise } from \"./utils\";\n\nfunction setControlError(control: AbstractControl, errorKey: string, errorValue: any) {\n  const currentErrors = control.errors || {};\n  if (currentErrors[errorKey]) return;\n  control.setErrors({ ...currentErrors, [errorKey]: errorValue });\n}\n\nfunction clearControlError(control: AbstractControl, errorKey: string) {\n  const currentErrors = control.errors || {};\n  if (!currentErrors[errorKey]) return;\n  control.setErrors({ ...currentErrors, [errorKey]: null });\n}\n\nexport function dateComparison(a: string, b: string, comparator: 'before' | 'after') {\n  return function(control: AbstractControl<any, any>): ValidationErrors | null {\n    if (!(control instanceof FormGroup)) return null;\n\n    const formGroup = control as FormGroup;\n\n    const aControl = formGroup.get(a);\n    const bControl = formGroup.get(b);\n\n    if (!aControl?.value || !bControl?.value) {\n      clearControlError(aControl, 'dateRange');\n      return null;\n    }\n\n    const aDate = new Date(aControl.value);\n    const bDate = new Date(bControl.value);\n\n    const hasError = (\n      comparator === 'after' && aDate <= bDate ||\n      comparator === 'before' && aDate >= bDate\n    );\n\n    if (hasError) {\n      setControlError(aControl, 'dateRange', { against: b, comparator });\n      return { dateRange: true };\n    }\n\n    clearControlError(aControl, 'dateRange');\n    return null;\n  }\n}\n\nexport function withinNYears(n: number, dir: 'both' | 'before' | 'after' = 'both', of?: Date): ValidatorFn {\n  return function(control: AbstractControl): ValidationErrors | null {\n    const timestamp = control?.value;\n    if (!timestamp || isNaN(new Date(timestamp).getTime())) return null;\n\n    const date = new Date(timestamp);\n    const against = of ?? new Date();\n\n    const min = new Date(against)\n    min.setFullYear(against.getFullYear() - n);\n    const max = new Date(against)\n    max.setFullYear(against.getFullYear() + n);\n\n    let isValid = false;\n    if (dir === 'after') isValid = date < max && date > against;\n    else if (dir === 'before') isValid = date > min && date < against;\n    else isValid = date > min && date < max;\n\n    return isValid ? null : { notWithinYears: { n, against, dir } };\n  }\n}\n\nexport function withinNHours(n: number, dir: 'both' | 'before' | 'after' = 'both', of?: Date): ValidatorFn {\n  return function(control: AbstractControl): ValidationErrors | null {\n    const timestamp = control?.value;\n    if (!timestamp || isNaN(new Date(timestamp).getTime())) return null;\n\n    const date = new Date(timestamp);\n    const against = of ?? new Date();\n\n    const min = new Date(against)\n    min.setHours(against.getHours() - n);\n    const max = new Date(against)\n    max.setHours(against.getHours() + n);\n\n    let isValid = false;\n    if (dir === 'after') isValid = date < max && date > against;\n    else if (dir === 'before') isValid = date > min && date < against;\n    else isValid = date > min && date < max;\n\n    return isValid ? null : { notWithinHours: { n, against, dir } };\n  }\n}\n\n// translate validation errors into human readable strings\nexport function translateError(name: string, control: AbstractControl) {\n  const { errors } = control;\n\n  if (control.hasError('required'))\n    return `${titleise(name)} is a required value.`;\n  else if (control.hasError('min'))\n    return `${titleise(name)} needs to be a minimum of ${errors.min.min}.`;\n  else if (control.hasError('max'))\n    return `${titleise(name)} needs to be a maximum of ${errors.max.max}.`;\n  else if (control.hasError('future'))\n    return `${titleise(name)} needs to be in the past.`;\n  else if (control.hasError('dateRange'))\n    return `${titleise(name)} needs to be ${errors.dateRange.comparator} ${titleise(errors.dateRange.against)}.`;\n  else if (control.hasError('notWithinYears'))\n    return `${titleise(name)} needs to be within ${errors.notWithinYears.n} years of ${errors.notWithinYears.against}`;\n  else if (control.hasError('notWithinHours'))\n    return `${titleise(name)} needs to be within ${errors.notWithinHours.n} hours of ${errors.notWithinHours.against}`;\n\n  return null;\n}\n"]}
@@ -0,0 +1,7 @@
1
+ /*
2
+ * Public API Surface of detail-edit
3
+ */
4
+ export * from './lib/detail-edit.service';
5
+ export * from './lib/detail-edit.component';
6
+ export * from './lib/detail-edit.module';
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2RldGFpbC1lZGl0L3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIGRldGFpbC1lZGl0XG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9saWIvZGV0YWlsLWVkaXQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9kZXRhaWwtZWRpdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvZGV0YWlsLWVkaXQubW9kdWxlJztcbiJdfQ==