UNPKG

@ngx-formly/core

Version:

Formly is a dynamic (JSON powered) form library for Angular that bring unmatched maintainability to your application's forms.

129 lines 16.9 kB
import { Component, ChangeDetectionStrategy, Input, EventEmitter, Output, ContentChildren, } from '@angular/core'; import { FormlyFormBuilder } from '../services/formly.builder'; import { clone, hasKey } from '../utils'; import { switchMap, filter, take } from 'rxjs/operators'; import { clearControl } from '../extensions/field-form/utils'; import { FormlyFieldTemplates, FormlyTemplate } from './formly.template'; import * as i0 from "@angular/core"; import * as i1 from "../services/formly.builder"; import * as i2 from "../services/formly.config"; import * as i3 from "./formly.template"; import * as i4 from "./formly.field"; /** * The `<form-form>` component is the main container of the form, * which takes care of managing the form state * and delegates the rendering of each field to `<formly-field>` component. */ export class FormlyForm { constructor(builder, config, ngZone, fieldTemplates) { this.builder = builder; this.config = config; this.ngZone = ngZone; this.fieldTemplates = fieldTemplates; /** Event that is emitted when the model value is changed */ this.modelChange = new EventEmitter(); this.field = { type: 'formly-group' }; this._modelChangeValue = {}; this.valueChangesUnsubscribe = () => { }; } /** The form instance which allow to track model value and validation status. */ set form(form) { this.field.form = form; } get form() { return this.field.form; } /** The model to be represented by the form. */ set model(model) { if (this.config.extras.immutable && this._modelChangeValue === model) { return; } this.setField({ model }); } get model() { return this.field.model; } /** The field configurations for building the form. */ set fields(fieldGroup) { this.setField({ fieldGroup }); } get fields() { return this.field.fieldGroup; } /** Options for the form. */ set options(options) { this.setField({ options }); } get options() { return this.field.options; } set templates(templates) { this.fieldTemplates.templates = templates; } ngDoCheck() { if (this.config.extras.checkExpressionOn === 'changeDetectionCheck') { this.checkExpressionChange(); } } ngOnChanges(changes) { if (changes.fields && this.form) { clearControl(this.form); } if (changes.fields || changes.form || (changes.model && this._modelChangeValue !== changes.model.currentValue)) { this.valueChangesUnsubscribe(); this.builder.build(this.field); this.valueChangesUnsubscribe = this.valueChanges(); } } ngOnDestroy() { this.valueChangesUnsubscribe(); } checkExpressionChange() { this.field.options.checkExpressions?.(this.field); } valueChanges() { this.valueChangesUnsubscribe(); const sub = this.field.options.fieldChanges .pipe(filter(({ field, type }) => hasKey(field) && type === 'valueChanges'), switchMap(() => this.ngZone.onStable.asObservable().pipe(take(1)))) .subscribe(() => this.ngZone.runGuarded(() => { // runGuarded is used to keep in sync the expression changes // https://github.com/ngx-formly/ngx-formly/issues/2095 this.checkExpressionChange(); this.modelChange.emit((this._modelChangeValue = clone(this.model))); })); return () => sub.unsubscribe(); } setField(field) { if (this.config.extras.immutable) { this.field = { ...this.field, ...clone(field) }; } else { Object.keys(field).forEach((p) => (this.field[p] = field[p])); } } } FormlyForm.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: FormlyForm, deps: [{ token: i1.FormlyFormBuilder }, { token: i2.FormlyConfig }, { token: i0.NgZone }, { token: i3.FormlyFieldTemplates }], target: i0.ɵɵFactoryTarget.Component }); FormlyForm.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: FormlyForm, selector: "formly-form", inputs: { form: "form", model: "model", fields: "fields", options: "options" }, outputs: { modelChange: "modelChange" }, providers: [FormlyFormBuilder, FormlyFieldTemplates], queries: [{ propertyName: "templates", predicate: FormlyTemplate }], usesOnChanges: true, ngImport: i0, template: '<formly-field [field]="field"></formly-field>', isInline: true, components: [{ type: i4.FormlyField, selector: "formly-field", inputs: ["field"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: FormlyForm, decorators: [{ type: Component, args: [{ selector: 'formly-form', template: '<formly-field [field]="field"></formly-field>', providers: [FormlyFormBuilder, FormlyFieldTemplates], changeDetection: ChangeDetectionStrategy.OnPush, }] }], ctorParameters: function () { return [{ type: i1.FormlyFormBuilder }, { type: i2.FormlyConfig }, { type: i0.NgZone }, { type: i3.FormlyFieldTemplates }]; }, propDecorators: { form: [{ type: Input }], model: [{ type: Input }], fields: [{ type: Input }], options: [{ type: Input }], modelChange: [{ type: Output }], templates: [{ type: ContentChildren, args: [FormlyTemplate] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formly.form.js","sourceRoot":"","sources":["../../../../../../src/core/src/lib/components/formly.form.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ChangeDetectionStrategy,\n  DoCheck,\n  OnChanges,\n  Input,\n  SimpleChanges,\n  EventEmitter,\n  Output,\n  OnDestroy,\n  NgZone,\n  ContentChildren,\n  QueryList,\n} from '@angular/core';\nimport { FormGroup, FormArray } from '@angular/forms';\nimport { FormlyFieldConfig, FormlyFormOptions, FormlyFieldConfigCache } from '../models';\nimport { FormlyFormBuilder } from '../services/formly.builder';\nimport { FormlyConfig } from '../services/formly.config';\nimport { clone, hasKey } from '../utils';\nimport { switchMap, filter, take } from 'rxjs/operators';\nimport { clearControl } from '../extensions/field-form/utils';\nimport { FormlyFieldTemplates, FormlyTemplate } from './formly.template';\n\n/**\n * The `<form-form>` component is the main container of the form,\n * which takes care of managing the form state\n * and delegates the rendering of each field to `<formly-field>` component.\n */\n@Component({\n  selector: 'formly-form',\n  template: '<formly-field [field]=\"field\"></formly-field>',\n  providers: [FormlyFormBuilder, FormlyFieldTemplates],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FormlyForm implements DoCheck, OnChanges, OnDestroy {\n  /** The form instance which allow to track model value and validation status. */\n  @Input()\n  set form(form: FormGroup | FormArray) {\n    this.field.form = form;\n  }\n  get form(): FormGroup | FormArray {\n    return this.field.form;\n  }\n\n  /** The model to be represented by the form. */\n  @Input()\n  set model(model: any) {\n    if (this.config.extras.immutable && this._modelChangeValue === model) {\n      return;\n    }\n\n    this.setField({ model });\n  }\n  get model(): any {\n    return this.field.model;\n  }\n\n  /** The field configurations for building the form. */\n  @Input()\n  set fields(fieldGroup: FormlyFieldConfig[]) {\n    this.setField({ fieldGroup });\n  }\n  get fields(): FormlyFieldConfig[] {\n    return this.field.fieldGroup;\n  }\n\n  /** Options for the form. */\n  @Input()\n  set options(options: FormlyFormOptions) {\n    this.setField({ options });\n  }\n  get options(): FormlyFormOptions {\n    return this.field.options;\n  }\n\n  /** Event that is emitted when the model value is changed */\n  @Output() modelChange = new EventEmitter<any>();\n  @ContentChildren(FormlyTemplate) set templates(templates: QueryList<FormlyTemplate>) {\n    this.fieldTemplates.templates = templates;\n  }\n\n  field: FormlyFieldConfigCache = { type: 'formly-group' };\n  private _modelChangeValue: any = {};\n  private valueChangesUnsubscribe = () => {};\n\n  constructor(\n    private builder: FormlyFormBuilder,\n    private config: FormlyConfig,\n    private ngZone: NgZone,\n    private fieldTemplates: FormlyFieldTemplates,\n  ) {}\n\n  ngDoCheck() {\n    if (this.config.extras.checkExpressionOn === 'changeDetectionCheck') {\n      this.checkExpressionChange();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.fields && this.form) {\n      clearControl(this.form);\n    }\n\n    if (changes.fields || changes.form || (changes.model && this._modelChangeValue !== changes.model.currentValue)) {\n      this.valueChangesUnsubscribe();\n      this.builder.build(this.field);\n      this.valueChangesUnsubscribe = this.valueChanges();\n    }\n  }\n\n  ngOnDestroy() {\n    this.valueChangesUnsubscribe();\n  }\n\n  private checkExpressionChange() {\n    this.field.options.checkExpressions?.(this.field);\n  }\n\n  private valueChanges() {\n    this.valueChangesUnsubscribe();\n\n    const sub = this.field.options.fieldChanges\n      .pipe(\n        filter(({ field, type }) => hasKey(field) && type === 'valueChanges'),\n        switchMap(() => this.ngZone.onStable.asObservable().pipe(take(1))),\n      )\n      .subscribe(() =>\n        this.ngZone.runGuarded(() => {\n          // runGuarded is used to keep in sync the expression changes\n          // https://github.com/ngx-formly/ngx-formly/issues/2095\n          this.checkExpressionChange();\n          this.modelChange.emit((this._modelChangeValue = clone(this.model)));\n        }),\n      );\n\n    return () => sub.unsubscribe();\n  }\n\n  private setField(field: FormlyFieldConfigCache) {\n    if (this.config.extras.immutable) {\n      this.field = { ...this.field, ...clone(field) };\n    } else {\n      Object.keys(field).forEach((p) => ((this.field as any)[p] = (field as any)[p]));\n    }\n  }\n}\n"]}