@ng-flexy/form
Version:
Flexy components and tools to build Angular 8+ applications
113 lines • 16.5 kB
JavaScript
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"]}