@hmcts/ccd-case-ui-toolkit 7.2.23-srt-rc2 → 7.2.23-srt-rc3

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.
@@ -112,8 +112,29 @@ export class CaseEditFormComponent {
112
112
  this.initial = JSON.stringify(this.formValueService.sanitise(this.formGroup.value));
113
113
  }
114
114
  detectChangesAndEmit(changes) {
115
- const current = JSON.stringify(this.formValueService.sanitise(changes));
116
- this.initial !== current ? this.valuesChanged.emit(true) : this.valuesChanged.emit(false);
115
+ const current = this.formValueService.sanitise(changes);
116
+ const isEqual = this.deepEqual(this.initial, current);
117
+ this.valuesChanged.emit(!isEqual);
118
+ }
119
+ deepEqual(obj1, obj2) {
120
+ // Simple deep equality check
121
+ if (obj1 === obj2) {
122
+ return true;
123
+ }
124
+ if (typeof obj1 !== 'object' || typeof obj2 !== 'object' || obj1 == null || obj2 == null) {
125
+ return false;
126
+ }
127
+ const keys1 = Object.keys(obj1);
128
+ const keys2 = Object.keys(obj2);
129
+ if (keys1.length !== keys2.length) {
130
+ return false;
131
+ }
132
+ for (const key of keys1) {
133
+ if (!this.deepEqual(obj1[key], obj2[key])) {
134
+ return false;
135
+ }
136
+ }
137
+ return true;
117
138
  }
118
139
  static ɵfac = function CaseEditFormComponent_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || CaseEditFormComponent)(i0.ɵɵdirectiveInject(i1.FormValueService)); };
119
140
  static ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: CaseEditFormComponent, selectors: [["ccd-case-edit-form"]], inputs: { fields: "fields", formGroup: "formGroup", caseFields: "caseFields", pageChangeSubject: "pageChangeSubject" }, outputs: { valuesChanged: "valuesChanged" }, decls: 1, vars: 1, consts: [["CompoundRow", ""], ["ccdConditionalShowForm", "", 3, "formGroup", "caseFields", "contextFields", 4, "ngFor", "ngForOf"], ["ccdConditionalShowForm", "", 3, "formGroup", "caseFields", "contextFields"], ["ccdLabelSubstitutor", "", 3, "caseField", "formGroup", "contextFields"], [3, "ngSwitch"], [3, "caseField", "caseFields", "withLabel", "formGroup", 4, "ngSwitchCase"], [4, "ngSwitchCase"], [3, "caseField", "caseFields", "withLabel", "formGroup"], [4, "ngIf", "ngIfElse"], [3, "caseField", "caseFields", "formGroup", "idPrefix", "hidden"]], template: function CaseEditFormComponent_Template(rf, ctx) { if (rf & 1) {
@@ -137,4 +158,4 @@ export class CaseEditFormComponent {
137
158
  type: Output
138
159
  }] }); })();
139
160
  (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(CaseEditFormComponent, { className: "CaseEditFormComponent", filePath: "lib/shared/components/case-editor/case-edit-form/case-edit-form.component.ts", lineNumber: 12 }); })();
140
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"case-edit-form.component.js","sourceRoot":"","sources":["../../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/components/case-editor/case-edit-form/case-edit-form.component.ts","../../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/components/case-editor/case-edit-form/case-edit-form.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, EventEmitter, Input, OnDestroy, Output } from '@angular/core';\nimport { FormGroup } from '@angular/forms';\nimport { Subject, Subscription } from 'rxjs';\nimport { debounceTime } from 'rxjs/operators';\nimport { CaseField } from '../../../domain/definition/case-field.model';\nimport { FormValueService } from '../../../services/form/form-value.service';\n\n@Component({\n  selector: 'ccd-case-edit-form',\n  templateUrl: 'case-edit-form.html'\n})\nexport class CaseEditFormComponent implements OnDestroy, AfterViewInit {\n\n  @Input()\n  public fields: CaseField[] = [];\n  @Input()\n  public formGroup: FormGroup;\n  @Input()\n  public caseFields: CaseField[] = [];\n  @Input()\n  public pageChangeSubject: Subject<boolean> = new Subject();\n  @Output()\n  public valuesChanged: EventEmitter<any> = new EventEmitter();\n\n  public initial: any;\n  public pageChangeSubscription: Subscription;\n  public formGroupChangeSubscription: Subscription;\n\n  constructor(private readonly formValueService: FormValueService) { }\n\n  public ngOnDestroy(): void {\n    if (this.pageChangeSubscription) {\n      this.pageChangeSubscription.unsubscribe();\n    }\n    if (this.formGroupChangeSubscription) {\n      this.formGroupChangeSubscription.unsubscribe();\n    }\n  }\n\n  // We need the below un/subscribe complexity as we do not have proper page component per page with its AfterViewInit hook\n  // being called on each page load. This is done for \"Cancel and return\" modal from RDM-2302.\n  public ngAfterViewInit(): void {\n    this.retrieveInitialFormValues();\n    this.pageChangeSubscription = this.pageChangeSubject.subscribe(() => {\n      if (this.formGroupChangeSubscription) {\n        this.formGroupChangeSubscription.unsubscribe();\n      }\n      // Timeout is required for the form to be rendered before subscription to form changes and initial form values retrieval.\n      setTimeout(() => {\n        this.subscribeToFormChanges();\n        this.retrieveInitialFormValues();\n      });\n    });\n    this.subscribeToFormChanges();\n  }\n\n  public subscribeToFormChanges() {\n    this.formGroupChangeSubscription = this.formGroup.valueChanges\n      .pipe(\n        debounceTime(200)\n      )\n      .subscribe(_ => this.detectChangesAndEmit(_));\n  }\n\n  public retrieveInitialFormValues() {\n    this.initial = JSON.stringify(this.formValueService.sanitise(this.formGroup.value));\n  }\n\n  public detectChangesAndEmit(changes) {\n    const current = JSON.stringify(this.formValueService.sanitise(changes));\n    this.initial !== current ? this.valuesChanged.emit(true) : this.valuesChanged.emit(false);\n  }\n\n  /*private deepEqual(obj1: any, obj2: any): boolean {\n    // Simple deep equality check\n    if (obj1 === obj2) {\n      return true;\n    }\n    if (typeof obj1 !== 'object' || typeof obj2 !== 'object' || obj1 == null || obj2 == null) {\n      return false;\n    }\n    const keys1 = Object.keys(obj1);\n    const keys2 = Object.keys(obj2);\n    if (keys1.length !== keys2.length) {\n      return false;\n    }\n    for (const key of keys1) {\n      if (!this.deepEqual(obj1[key], obj2[key])) {\n        return false;\n      }\n    }\n    return true;\n  }*/\n}\n","<ng-container ccdConditionalShowForm [formGroup]=\"formGroup\" [caseFields]=\"fields\" [contextFields]=\"caseFields\"\n              *ngFor=\"let field of fields\">\n\n  <div ccdLabelSubstitutor [caseField]=\"field\" [formGroup]=\"formGroup\" [contextFields]=\"caseFields\">\n    <ng-container [ngSwitch]=\"field | ccdIsReadOnlyAndNotCollection\">\n\n      <ccd-field-read *ngSwitchCase=\"true\" [caseField]=\"field\" [caseFields]=\"caseFields\" [withLabel]=\"true\"\n                      [formGroup]=\"formGroup\" [attr.field_id]=\"field.id\"\n                      [attr.field_type]=\"field.field_type.type\"></ccd-field-read>\n      <ng-container *ngSwitchCase=\"false\">\n\n        <ng-container *ngIf=\"!(field | ccdIsCompound); else CompoundRow\">\n          <ccd-field-write [caseField]=\"field\"\n                           [caseFields]=\"caseFields\"\n                           [formGroup]=\"formGroup\"\n                           [idPrefix]=\"\"\n                           [hidden]=\"field.hidden\"\n                           [attr.field_id]=\"field.id\"\n                           [attr.field_type]=\"field.field_type.type\">\n          </ccd-field-write>\n        </ng-container>\n\n        <ng-template #CompoundRow>\n          <ccd-field-write [caseField]=\"field\"\n                           [caseFields]=\"caseFields\"\n                           [formGroup]=\"formGroup\"\n                           [idPrefix]=\"field.id + '_'\"\n                           [hidden]=\"field.hidden\"\n                           [attr.field_id]=\"field.id\"\n                           [attr.field_type]=\"field.field_type.type\"></ccd-field-write>\n        </ng-template>\n      </ng-container>\n    </ng-container>\n  </div>\n</ng-container>\n"]}
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"case-edit-form.component.js","sourceRoot":"","sources":["../../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/components/case-editor/case-edit-form/case-edit-form.component.ts","../../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/components/case-editor/case-edit-form/case-edit-form.html"],"names":[],"mappings":"AAAA,OAAO,EAAiB,SAAS,EAAE,YAAY,EAAE,KAAK,EAAa,MAAM,EAAE,MAAM,eAAe,CAAC;AACjG,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAC3C,OAAO,EAAE,OAAO,EAAgB,MAAM,MAAM,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAE9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,2CAA2C,CAAC;;;;ICCvE,oCAE2E;;;;IAD3D,AADmE,AAA1B,AAApB,oCAAmB,iCAA0B,mBAAmB,+BAC9D;;;;IAIrC,6BAAiE;IAC/D,qCAOkB;;;;;IAPD,cAAmB;IAInB,AAFA,AADA,AADA,oCAAmB,iCACM,+BACF,2BAEA;;;;IAOxC,qCAM6E;;;;IAF5D,AADA,AADA,AADA,AADA,oCAAmB,iCACM,+BACF,+BACI,2BACJ;;;;IAlB5C,6BAAoC;IAElC,sHAAiE;;IAWjE,qJAA0B;;;;;IAXX,cAAgC;IAAA,AAAhC,sDAAgC,4BAAgB;;;IAXvE,gCAC2C;IAEzC,8BAAkG;IAChG,gCAAiE;;IAK/D,AAHA,2GAE0D,0FACtB;;IAwBxC,iBAAM;;;;;IAjC2E,AAAtB,AAAxB,4CAAuB,6BAAsB,oCAA6B;IAGpF,cAAmB;IAAyB,AAAxB,AAApB,oCAAmB,+BAAwB,oCAA6B;IACjF,cAAkD;IAAlD,yDAAkD;IAE7C,eAAkB;IAAlB,mCAAkB;IAGpB,cAAmB;IAAnB,oCAAmB;;ADExC,MAAM,OAAO,qBAAqB;IAiBH;IAdtB,MAAM,GAAgB,EAAE,CAAC;IAEzB,SAAS,CAAY;IAErB,UAAU,GAAgB,EAAE,CAAC;IAE7B,iBAAiB,GAAqB,IAAI,OAAO,EAAE,CAAC;IAEpD,aAAa,GAAsB,IAAI,YAAY,EAAE,CAAC;IAEtD,OAAO,CAAM;IACb,sBAAsB,CAAe;IACrC,2BAA2B,CAAe;IAEjD,YAA6B,gBAAkC;QAAlC,qBAAgB,GAAhB,gBAAgB,CAAkB;IAAI,CAAC;IAE7D,WAAW;QAChB,IAAI,IAAI,CAAC,sBAAsB,EAAE,CAAC;YAChC,IAAI,CAAC,sBAAsB,CAAC,WAAW,EAAE,CAAC;QAC5C,CAAC;QACD,IAAI,IAAI,CAAC,2BAA2B,EAAE,CAAC;YACrC,IAAI,CAAC,2BAA2B,CAAC,WAAW,EAAE,CAAC;QACjD,CAAC;IACH,CAAC;IAED,yHAAyH;IACzH,4FAA4F;IACrF,eAAe;QACpB,IAAI,CAAC,yBAAyB,EAAE,CAAC;QACjC,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,GAAG,EAAE;YAClE,IAAI,IAAI,CAAC,2BAA2B,EAAE,CAAC;gBACrC,IAAI,CAAC,2BAA2B,CAAC,WAAW,EAAE,CAAC;YACjD,CAAC;YACD,yHAAyH;YACzH,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,sBAAsB,EAAE,CAAC;gBAC9B,IAAI,CAAC,yBAAyB,EAAE,CAAC;YACnC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,sBAAsB,EAAE,CAAC;IAChC,CAAC;IAEM,sBAAsB;QAC3B,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY;aAC3D,IAAI,CACH,YAAY,CAAC,GAAG,CAAC,CAClB;aACA,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,CAAC;IAClD,CAAC;IAEM,yBAAyB;QAC9B,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;IACtF,CAAC;IAEM,oBAAoB,CAAC,OAAO;QACjC,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QACxD,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QACtD,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;IAEO,SAAS,CAAC,IAAS,EAAE,IAAS;QACpC,6BAA6B;QAC7B,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;YAClB,OAAO,IAAI,CAAC;QACd,CAAC;QACD,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;YACzF,OAAO,KAAK,CAAC;QACf,CAAC;QACD,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChC,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChC,IAAI,KAAK,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM,EAAE,CAAC;YAClC,OAAO,KAAK,CAAC;QACf,CAAC;QACD,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE,CAAC;YACxB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;gBAC1C,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;+GAlFU,qBAAqB;6DAArB,qBAAqB;YCXlC,yFAC2C;;YAAX,oCAAS;;;iFDU5B,qBAAqB;cAJjC,SAAS;2BACE,oBAAoB;iDAMvB,MAAM;kBADZ,KAAK;YAGC,SAAS;kBADf,KAAK;YAGC,UAAU;kBADhB,KAAK;YAGC,iBAAiB;kBADvB,KAAK;YAGC,aAAa;kBADnB,MAAM;;kFAVI,qBAAqB","sourcesContent":["import { AfterViewInit, Component, EventEmitter, Input, OnDestroy, Output } from '@angular/core';\nimport { FormGroup } from '@angular/forms';\nimport { Subject, Subscription } from 'rxjs';\nimport { debounceTime } from 'rxjs/operators';\nimport { CaseField } from '../../../domain/definition/case-field.model';\nimport { FormValueService } from '../../../services/form/form-value.service';\n\n@Component({\n  selector: 'ccd-case-edit-form',\n  templateUrl: 'case-edit-form.html'\n})\nexport class CaseEditFormComponent implements OnDestroy, AfterViewInit {\n\n  @Input()\n  public fields: CaseField[] = [];\n  @Input()\n  public formGroup: FormGroup;\n  @Input()\n  public caseFields: CaseField[] = [];\n  @Input()\n  public pageChangeSubject: Subject<boolean> = new Subject();\n  @Output()\n  public valuesChanged: EventEmitter<any> = new EventEmitter();\n\n  public initial: any;\n  public pageChangeSubscription: Subscription;\n  public formGroupChangeSubscription: Subscription;\n\n  constructor(private readonly formValueService: FormValueService) { }\n\n  public ngOnDestroy(): void {\n    if (this.pageChangeSubscription) {\n      this.pageChangeSubscription.unsubscribe();\n    }\n    if (this.formGroupChangeSubscription) {\n      this.formGroupChangeSubscription.unsubscribe();\n    }\n  }\n\n  // We need the below un/subscribe complexity as we do not have proper page component per page with its AfterViewInit hook\n  // being called on each page load. This is done for \"Cancel and return\" modal from RDM-2302.\n  public ngAfterViewInit(): void {\n    this.retrieveInitialFormValues();\n    this.pageChangeSubscription = this.pageChangeSubject.subscribe(() => {\n      if (this.formGroupChangeSubscription) {\n        this.formGroupChangeSubscription.unsubscribe();\n      }\n      // Timeout is required for the form to be rendered before subscription to form changes and initial form values retrieval.\n      setTimeout(() => {\n        this.subscribeToFormChanges();\n        this.retrieveInitialFormValues();\n      });\n    });\n    this.subscribeToFormChanges();\n  }\n\n  public subscribeToFormChanges() {\n    this.formGroupChangeSubscription = this.formGroup.valueChanges\n      .pipe(\n        debounceTime(200)\n      )\n      .subscribe(_ => this.detectChangesAndEmit(_));\n  }\n\n  public retrieveInitialFormValues() {\n    this.initial = JSON.stringify(this.formValueService.sanitise(this.formGroup.value));\n  }\n\n  public detectChangesAndEmit(changes) {\n    const current = this.formValueService.sanitise(changes);\n    const isEqual = this.deepEqual(this.initial, current);\n    this.valuesChanged.emit(!isEqual);\n  }\n\n  private deepEqual(obj1: any, obj2: any): boolean {\n    // Simple deep equality check\n    if (obj1 === obj2) {\n      return true;\n    }\n    if (typeof obj1 !== 'object' || typeof obj2 !== 'object' || obj1 == null || obj2 == null) {\n      return false;\n    }\n    const keys1 = Object.keys(obj1);\n    const keys2 = Object.keys(obj2);\n    if (keys1.length !== keys2.length) {\n      return false;\n    }\n    for (const key of keys1) {\n      if (!this.deepEqual(obj1[key], obj2[key])) {\n        return false;\n      }\n    }\n    return true;\n  }\n}\n","<ng-container ccdConditionalShowForm [formGroup]=\"formGroup\" [caseFields]=\"fields\" [contextFields]=\"caseFields\"\n              *ngFor=\"let field of fields\">\n\n  <div ccdLabelSubstitutor [caseField]=\"field\" [formGroup]=\"formGroup\" [contextFields]=\"caseFields\">\n    <ng-container [ngSwitch]=\"field | ccdIsReadOnlyAndNotCollection\">\n\n      <ccd-field-read *ngSwitchCase=\"true\" [caseField]=\"field\" [caseFields]=\"caseFields\" [withLabel]=\"true\"\n                      [formGroup]=\"formGroup\" [attr.field_id]=\"field.id\"\n                      [attr.field_type]=\"field.field_type.type\"></ccd-field-read>\n      <ng-container *ngSwitchCase=\"false\">\n\n        <ng-container *ngIf=\"!(field | ccdIsCompound); else CompoundRow\">\n          <ccd-field-write [caseField]=\"field\"\n                           [caseFields]=\"caseFields\"\n                           [formGroup]=\"formGroup\"\n                           [idPrefix]=\"\"\n                           [hidden]=\"field.hidden\"\n                           [attr.field_id]=\"field.id\"\n                           [attr.field_type]=\"field.field_type.type\">\n          </ccd-field-write>\n        </ng-container>\n\n        <ng-template #CompoundRow>\n          <ccd-field-write [caseField]=\"field\"\n                           [caseFields]=\"caseFields\"\n                           [formGroup]=\"formGroup\"\n                           [idPrefix]=\"field.id + '_'\"\n                           [hidden]=\"field.hidden\"\n                           [attr.field_id]=\"field.id\"\n                           [attr.field_type]=\"field.field_type.type\"></ccd-field-write>\n        </ng-template>\n      </ng-container>\n    </ng-container>\n  </div>\n</ng-container>\n"]}
@@ -10571,8 +10571,29 @@ class CaseEditFormComponent {
10571
10571
  this.initial = JSON.stringify(this.formValueService.sanitise(this.formGroup.value));
10572
10572
  }
10573
10573
  detectChangesAndEmit(changes) {
10574
- const current = JSON.stringify(this.formValueService.sanitise(changes));
10575
- this.initial !== current ? this.valuesChanged.emit(true) : this.valuesChanged.emit(false);
10574
+ const current = this.formValueService.sanitise(changes);
10575
+ const isEqual = this.deepEqual(this.initial, current);
10576
+ this.valuesChanged.emit(!isEqual);
10577
+ }
10578
+ deepEqual(obj1, obj2) {
10579
+ // Simple deep equality check
10580
+ if (obj1 === obj2) {
10581
+ return true;
10582
+ }
10583
+ if (typeof obj1 !== 'object' || typeof obj2 !== 'object' || obj1 == null || obj2 == null) {
10584
+ return false;
10585
+ }
10586
+ const keys1 = Object.keys(obj1);
10587
+ const keys2 = Object.keys(obj2);
10588
+ if (keys1.length !== keys2.length) {
10589
+ return false;
10590
+ }
10591
+ for (const key of keys1) {
10592
+ if (!this.deepEqual(obj1[key], obj2[key])) {
10593
+ return false;
10594
+ }
10595
+ }
10596
+ return true;
10576
10597
  }
10577
10598
  static ɵfac = function CaseEditFormComponent_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || CaseEditFormComponent)(i0.ɵɵdirectiveInject(FormValueService)); };
10578
10599
  static ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: CaseEditFormComponent, selectors: [["ccd-case-edit-form"]], inputs: { fields: "fields", formGroup: "formGroup", caseFields: "caseFields", pageChangeSubject: "pageChangeSubject" }, outputs: { valuesChanged: "valuesChanged" }, decls: 1, vars: 1, consts: [["CompoundRow", ""], ["ccdConditionalShowForm", "", 3, "formGroup", "caseFields", "contextFields", 4, "ngFor", "ngForOf"], ["ccdConditionalShowForm", "", 3, "formGroup", "caseFields", "contextFields"], ["ccdLabelSubstitutor", "", 3, "caseField", "formGroup", "contextFields"], [3, "ngSwitch"], [3, "caseField", "caseFields", "withLabel", "formGroup", 4, "ngSwitchCase"], [4, "ngSwitchCase"], [3, "caseField", "caseFields", "withLabel", "formGroup"], [4, "ngIf", "ngIfElse"], [3, "caseField", "caseFields", "formGroup", "idPrefix", "hidden"]], template: function CaseEditFormComponent_Template(rf, ctx) { if (rf & 1) {