UNPKG

@ng-flexy/form

Version:

Flexy components and tools to build Angular 8+ applications

113 lines 16.5 kB
import { FlexyLayout } from '@ng-flexy/layout'; import { cloneDeep, merge } from 'lodash'; import { BehaviorSubject } from 'rxjs'; import { HIDDEN_CALC_GROUP_NAME } from '../services/json-mapper.utils'; import { calculate, clearEmptyArrayAndObjects, findErrors, findRemoved, findSchema, FlexyFormDataMode, getSchemaData } from './form.utils'; export class FlexyForm extends FlexyLayout { constructor(formGroup, schema, data) { super(schema); this.isStarted = false; this._calculatedRefs = {}; this._currentDataSubject = new BehaviorSubject(data); this.currentData$ = this._currentDataSubject.asObservable(); this.formGroup = formGroup; this.schema = schema; this._initCalculatedRefs(schema); this._originalData = cloneDeep(data); this._setCurrentData(); // refresh attributes this._setCurrentData(); // jump to next tick // setTimeout(() => { this._subscribeChangesAndCalculate(); // }); } get valid() { return !this.getAllErrors(); } getAllData() { const data = cloneDeep(getSchemaData(this.schema, this.currentData)); this._clearHiddenData(data); return data; } // @deprecated getDirtyData() { const data = cloneDeep(getSchemaData(this.schema, this.currentData, FlexyFormDataMode.Dirty)); this._clearHiddenData(data); clearEmptyArrayAndObjects(data); const allData = this.getAllData(); const removed = findRemoved(allData, this._originalData); clearEmptyArrayAndObjects(removed); return merge(data, removed); } getAllErrors() { return this._lastErrors; } containsFieldSchema(fieldName) { return !!findSchema(fieldName, this.schema); } getFieldSchema(fieldName) { return findSchema(fieldName, this.schema); } getFieldInstance(fieldName) { const schema = findSchema(fieldName, this.schema); if (schema && schema.componentRef) { return schema.componentRef.instance; } return null; } _subscribeChangesAndCalculate() { this._setCurrentData(); this.isStarted = true; // this._setCurrentData(); this._changesSubscription = this.formGroup.valueChanges.subscribe(data => { const hash = this.currentDataHash; this._setCurrentData(); if (hash !== this.currentDataHash) { this._currentDataSubject.next(this.currentData); } }); this._currentDataSubject.next(this.currentData); } _setCurrentData() { this.currentData = getSchemaData(this.schema, this.currentData); this.currentDataHash = JSON.stringify(this.currentData); this._calculate(); this.currentData = getSchemaData(this.schema, this.currentData); this.currentDataHash = JSON.stringify(this.currentData); const errors = findErrors(this.schema, this.currentData); this._lastErrors = errors && Object.keys(errors).length ? errors : null; } _initCalculatedRefs(schema) { if (schema) { schema.forEach((schemaItem) => { if (schemaItem.formName && schemaItem.formControl && schemaItem.calc) { this._calculatedRefs[schemaItem.formName] = { calc: schemaItem.calc, control: schemaItem.formControl }; } if (schemaItem.children) { this._initCalculatedRefs(schemaItem.children); } }); } } _calculate() { if (this._calculatedRefs) { Object.values(this._calculatedRefs).forEach(calc => { const value = calculate(calc.calc, this.currentData); if (value !== calc.control.value) { calc.control.setValue(value); calc.control.markAsDirty(); } }); } } _clearHiddenData(data) { if (data[HIDDEN_CALC_GROUP_NAME]) { delete data[HIDDEN_CALC_GROUP_NAME]; } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.model.js","sourceRoot":"","sources":["../../../../../projects/form/src/lib/models/form.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { FormControl, FormGroup } from '@angular/forms';\nimport { FlexyLayout } from '@ng-flexy/layout';\nimport { FlexyFormFieldLayoutSchema, FlexyFormLayoutSchema } from './layout-schema.model';\nimport { FlexyFormData } from './form.data';\nimport { cloneDeep, merge } from 'lodash';\nimport { BehaviorSubject, Observable, Subscription } from 'rxjs';\nimport { HIDDEN_CALC_GROUP_NAME } from '../services/json-mapper.utils';\nimport { calculate, clearEmptyArrayAndObjects, findErrors, findRemoved, findSchema, FlexyFormDataMode, getSchemaData } from './form.utils';\n\ninterface CalcRefs {\n  [name: string]: {\n    calc: string;\n    control: FormControl;\n    ifControl?: FormGroup;\n  };\n}\n\ninterface IfRefs {\n  if: string;\n  state: boolean;\n  ifControl?: FormGroup;\n}\n\nexport class FlexyForm extends FlexyLayout {\n  currentData: FlexyFormData;\n  currentDataHash: string;\n\n  readonly currentData$: Observable<FlexyFormData>;\n\n  get valid() {\n    return !this.getAllErrors();\n  }\n\n  isStarted = false;\n\n  // TODO to think change to private\n  readonly schema: FlexyFormLayoutSchema[];\n  readonly formGroup: FormGroup;\n\n  private readonly _originalData: FlexyFormData;\n  private readonly _currentDataSubject: BehaviorSubject<FlexyFormData>;\n\n  _calculatedRefs: CalcRefs = {};\n\n  _lastErrors: { [key: string]: any };\n\n  private _changesSubscription: Subscription;\n\n  constructor(formGroup: FormGroup, schema: FlexyFormLayoutSchema[], data: FlexyFormData) {\n    super(schema);\n\n    this._currentDataSubject = new BehaviorSubject(data);\n    this.currentData$ = this._currentDataSubject.asObservable();\n\n    this.formGroup = formGroup;\n    this.schema = schema;\n\n    this._initCalculatedRefs(schema);\n    this._originalData = cloneDeep(data);\n    this._setCurrentData();\n    // refresh attributes\n    this._setCurrentData();\n\n    // jump to next tick\n    // setTimeout(() => {\n    this._subscribeChangesAndCalculate();\n    // });\n  }\n\n  getAllData(): FlexyFormData {\n    const data = cloneDeep(getSchemaData(this.schema, this.currentData));\n    this._clearHiddenData(data);\n    return data;\n  }\n\n  // @deprecated\n  getDirtyData(): FlexyFormData {\n    const data = cloneDeep(getSchemaData(this.schema, this.currentData, FlexyFormDataMode.Dirty));\n    this._clearHiddenData(data);\n    clearEmptyArrayAndObjects(data);\n    const allData = this.getAllData();\n    const removed = findRemoved(allData, this._originalData);\n    clearEmptyArrayAndObjects(removed);\n    return merge(data, removed);\n  }\n\n  getAllErrors(): { [key: string]: any } {\n    return this._lastErrors;\n  }\n\n  containsFieldSchema(fieldName: string): boolean {\n    return !!findSchema(fieldName, this.schema);\n  }\n  getFieldSchema(fieldName: string): FlexyFormFieldLayoutSchema {\n    return findSchema(fieldName, this.schema);\n  }\n  getFieldInstance<T>(fieldName: string): T {\n    const schema: FlexyFormFieldLayoutSchema = findSchema(fieldName, this.schema);\n    if (schema && schema.componentRef) {\n      return schema.componentRef.instance;\n    }\n    return null;\n  }\n\n  private _subscribeChangesAndCalculate() {\n    this._setCurrentData();\n    this.isStarted = true;\n    // this._setCurrentData();\n    this._changesSubscription = this.formGroup.valueChanges.subscribe(data => {\n      const hash = this.currentDataHash;\n      this._setCurrentData();\n      if (hash !== this.currentDataHash) {\n        this._currentDataSubject.next(this.currentData);\n      }\n    });\n    this._currentDataSubject.next(this.currentData);\n  }\n\n  private _setCurrentData() {\n    this.currentData = getSchemaData(this.schema, this.currentData);\n    this.currentDataHash = JSON.stringify(this.currentData);\n    this._calculate();\n    this.currentData = getSchemaData(this.schema, this.currentData);\n    this.currentDataHash = JSON.stringify(this.currentData);\n    const errors = findErrors(this.schema, this.currentData);\n    this._lastErrors = errors && Object.keys(errors).length ? errors : null;\n  }\n\n  private _initCalculatedRefs(schema: FlexyFormLayoutSchema[]) {\n    if (schema) {\n      schema.forEach((schemaItem: FlexyFormFieldLayoutSchema) => {\n        if (schemaItem.formName && schemaItem.formControl && schemaItem.calc) {\n          this._calculatedRefs[schemaItem.formName] = {\n            calc: schemaItem.calc,\n            control: schemaItem.formControl as FormControl\n          };\n        }\n        if (schemaItem.children) {\n          this._initCalculatedRefs(schemaItem.children);\n        }\n      });\n    }\n  }\n\n  private _calculate() {\n    if (this._calculatedRefs) {\n      Object.values(this._calculatedRefs).forEach(calc => {\n        const value = calculate(calc.calc, this.currentData);\n        if (value !== calc.control.value) {\n          calc.control.setValue(value);\n          calc.control.markAsDirty();\n        }\n      });\n    }\n  }\n\n  private _clearHiddenData(data) {\n    if (data[HIDDEN_CALC_GROUP_NAME]) {\n      delete data[HIDDEN_CALC_GROUP_NAME];\n    }\n  }\n}\n"]}