@ngx-formly/core
Version:
Formly is a dynamic (JSON powered) form library for Angular that bring unmatched maintainability to your application's forms.
275 lines • 43.5 kB
JavaScript
import { Component, Input, ViewContainerRef, ViewChild, ComponentRef, Optional, } from '@angular/core';
import { FormControl } from '@angular/forms';
import { defineHiddenProp, observe, observeDeep, getFieldValue, assignFieldValue, isObject, markFieldForCheck, hasKey, } from '../utils';
import { isObservable } from 'rxjs';
import { debounceTime, distinctUntilChanged, startWith } from 'rxjs/operators';
import * as i0 from "@angular/core";
import * as i1 from "../services/formly.config";
import * as i2 from "./formly.template";
/**
* The `<formly-field>` component is used to render the UI widget (layout + type) of a given `field`.
*/
export class FormlyField {
constructor(config, renderer, _elementRef, hostContainerRef, form) {
this.config = config;
this.renderer = renderer;
this._elementRef = _elementRef;
this.hostContainerRef = hostContainerRef;
this.form = form;
this.hostObservers = [];
this.componentRefs = [];
this.hooksObservers = [];
this.detectFieldBuild = false;
this.valueChangesUnsubscribe = () => { };
}
get containerRef() {
return this.config.extras.renderFormlyFieldElement ? this.viewContainerRef : this.hostContainerRef;
}
get elementRef() {
if (this.config.extras.renderFormlyFieldElement) {
return this._elementRef;
}
if (this.componentRefs?.[0] instanceof ComponentRef) {
return this.componentRefs[0].location;
}
return null;
}
ngAfterContentInit() {
this.triggerHook('afterContentInit');
}
ngAfterViewInit() {
this.triggerHook('afterViewInit');
}
ngDoCheck() {
if (this.detectFieldBuild && this.field && this.field.options) {
this.render();
}
}
ngOnInit() {
this.triggerHook('onInit');
}
ngOnChanges(changes) {
this.triggerHook('onChanges', changes);
}
ngOnDestroy() {
this.resetRefs(this.field);
this.hostObservers.forEach((hostObserver) => hostObserver.unsubscribe());
this.hooksObservers.forEach((unsubscribe) => unsubscribe());
this.valueChangesUnsubscribe();
this.triggerHook('onDestroy');
}
renderField(containerRef, f, wrappers = []) {
if (this.containerRef === containerRef) {
this.resetRefs(this.field);
this.containerRef.clear();
wrappers = this.field?.wrappers;
}
if (wrappers?.length > 0) {
const [wrapper, ...wps] = wrappers;
const { component } = this.config.getWrapper(wrapper);
const ref = containerRef.createComponent(component);
this.attachComponentRef(ref, f);
observe(ref.instance, ['fieldComponent'], ({ currentValue, previousValue, firstChange }) => {
if (currentValue) {
if (previousValue && previousValue._lContainer === currentValue._lContainer) {
return;
}
const viewRef = previousValue ? previousValue.detach() : null;
if (viewRef && !viewRef.destroyed) {
currentValue.insert(viewRef);
}
else {
this.renderField(currentValue, f, wps);
}
!firstChange && ref.changeDetectorRef.detectChanges();
}
});
}
else if (f?.type) {
const inlineType = this.form?.templates?.find((ref) => ref.name === f.type);
let ref;
if (inlineType) {
ref = containerRef.createEmbeddedView(inlineType.ref, { $implicit: f });
}
else {
const { component } = this.config.getType(f.type, true);
ref = containerRef.createComponent(component);
}
this.attachComponentRef(ref, f);
}
}
triggerHook(name, changes) {
if (name === 'onInit' || (name === 'onChanges' && changes.field && !changes.field.firstChange)) {
this.valueChangesUnsubscribe();
this.valueChangesUnsubscribe = this.fieldChanges(this.field);
}
if (this.field?.hooks?.[name]) {
if (!changes || changes.field) {
const r = this.field.hooks[name](this.field);
if (isObservable(r) && ['onInit', 'afterContentInit', 'afterViewInit'].indexOf(name) !== -1) {
const sub = r.subscribe();
this.hooksObservers.push(() => sub.unsubscribe());
}
}
}
if (name === 'onChanges' && changes.field) {
this.resetRefs(changes.field.previousValue);
this.render();
}
}
attachComponentRef(ref, field) {
this.componentRefs.push(ref);
field._componentRefs.push(ref);
if (ref instanceof ComponentRef) {
Object.assign(ref.instance, { field });
}
}
render() {
if (!this.field) {
return;
}
// require Formly build
if (!this.field.options) {
this.detectFieldBuild = true;
return;
}
this.detectFieldBuild = false;
this.hostObservers.forEach((hostObserver) => hostObserver.unsubscribe());
this.hostObservers = [
observe(this.field, ['hide'], ({ firstChange, currentValue }) => {
const containerRef = this.containerRef;
if (this.config.extras.lazyRender === false) {
firstChange && this.renderField(containerRef, this.field);
if (!firstChange || (firstChange && currentValue)) {
this.elementRef &&
this.renderer.setStyle(this.elementRef.nativeElement, 'display', currentValue ? 'none' : '');
}
}
else {
if (currentValue) {
containerRef.clear();
if (this.field.className) {
this.renderer.removeAttribute(this.elementRef.nativeElement, 'class');
}
}
else {
this.renderField(containerRef, this.field);
if (this.field.className) {
this.renderer.setAttribute(this.elementRef.nativeElement, 'class', this.field.className);
}
}
}
!firstChange && this.field.options.detectChanges(this.field);
}),
observe(this.field, ['className'], ({ firstChange, currentValue }) => {
if ((!firstChange || (firstChange && currentValue)) &&
(!this.config.extras.lazyRender || this.field.hide !== true)) {
this.elementRef && this.renderer.setAttribute(this.elementRef.nativeElement, 'class', currentValue);
}
}),
...['touched', 'pristine', 'status'].map((prop) => observe(this.field, ['formControl', prop], ({ firstChange }) => !firstChange && markFieldForCheck(this.field))),
];
}
resetRefs(field) {
if (field) {
if (field._localFields) {
field._localFields = [];
}
else {
defineHiddenProp(this.field, '_localFields', []);
}
if (field._componentRefs) {
field._componentRefs = field._componentRefs.filter((ref) => this.componentRefs.indexOf(ref) === -1);
}
else {
defineHiddenProp(this.field, '_componentRefs', []);
}
}
this.componentRefs = [];
}
fieldChanges(field) {
if (!field) {
return () => { };
}
const subscribes = [observeDeep(field, ['props'], () => field.options.detectChanges(field))];
if (field.options) {
subscribes.push(observeDeep(field.options, ['formState'], () => field.options.detectChanges(field)));
}
for (const key of Object.keys(field._expressions || {})) {
const expressionObserver = observe(field, ['_expressions', key], ({ currentValue, previousValue }) => {
if (previousValue?.subscription) {
previousValue.subscription.unsubscribe();
previousValue.subscription = null;
}
if (isObservable(currentValue.value$)) {
currentValue.subscription = currentValue.value$.subscribe();
}
});
subscribes.push(() => {
if (field._expressions[key]?.subscription) {
field._expressions[key].subscription.unsubscribe();
}
expressionObserver.unsubscribe();
});
}
for (const path of [['focus'], ['template'], ['fieldGroupClassName'], ['validation', 'show']]) {
const fieldObserver = observe(field, path, ({ firstChange }) => !firstChange && field.options.detectChanges(field));
subscribes.push(() => fieldObserver.unsubscribe());
}
if (field.formControl && !field.fieldGroup) {
const control = field.formControl;
let valueChanges = control.valueChanges.pipe(distinctUntilChanged((x, y) => {
if (x !== y || Array.isArray(x) || isObject(x)) {
return false;
}
return true;
}));
if (control.value !== getFieldValue(field)) {
valueChanges = valueChanges.pipe(startWith(control.value));
}
const { updateOn, debounce } = field.modelOptions;
if ((!updateOn || updateOn === 'change') && debounce?.default > 0) {
valueChanges = control.valueChanges.pipe(debounceTime(debounce.default));
}
const sub = valueChanges.subscribe((value) => {
// workaround for https://github.com/angular/angular/issues/13792
if (control._fields?.length > 1 && control instanceof FormControl) {
control.patchValue(value, { emitEvent: false, onlySelf: true });
}
field.parsers?.forEach((parserFn) => (value = parserFn(value)));
if (value !== field.formControl.value) {
field.formControl.setValue(value);
return;
}
if (hasKey(field)) {
assignFieldValue(field, value);
}
field.options.fieldChanges.next({ value, field, type: 'valueChanges' });
});
subscribes.push(() => sub.unsubscribe());
}
let templateFieldsSubs = [];
observe(field, ['_localFields'], ({ currentValue }) => {
templateFieldsSubs.forEach((unsubscribe) => unsubscribe());
templateFieldsSubs = (currentValue || []).map((f) => this.fieldChanges(f));
});
return () => {
subscribes.forEach((unsubscribe) => unsubscribe());
templateFieldsSubs.forEach((unsubscribe) => unsubscribe());
};
}
}
FormlyField.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: FormlyField, deps: [{ token: i1.FormlyConfig }, { token: i0.Renderer2 }, { token: i0.ElementRef }, { token: i0.ViewContainerRef }, { token: i2.FormlyFieldTemplates, optional: true }], target: i0.ɵɵFactoryTarget.Component });
FormlyField.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.12", type: FormlyField, selector: "formly-field", inputs: { field: "field" }, viewQueries: [{ propertyName: "viewContainerRef", first: true, predicate: ["container"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: '<ng-template #container></ng-template>', isInline: true, styles: [":host:empty{display:none}\n"] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.12", ngImport: i0, type: FormlyField, decorators: [{
type: Component,
args: [{ selector: 'formly-field', template: '<ng-template #container></ng-template>', styles: [":host:empty{display:none}\n"] }]
}], ctorParameters: function () { return [{ type: i1.FormlyConfig }, { type: i0.Renderer2 }, { type: i0.ElementRef }, { type: i0.ViewContainerRef }, { type: i2.FormlyFieldTemplates, decorators: [{
type: Optional
}] }]; }, propDecorators: { field: [{
type: Input
}], viewContainerRef: [{
type: ViewChild,
args: ['container', { read: ViewContainerRef, static: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formly.field.js","sourceRoot":"","sources":["../../../../../../src/core/src/lib/components/formly.field.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  ViewContainerRef,\n  ViewChild,\n  ComponentRef,\n  SimpleChanges,\n  DoCheck,\n  OnInit,\n  OnChanges,\n  OnDestroy,\n  AfterContentInit,\n  AfterViewInit,\n  Renderer2,\n  ElementRef,\n  EmbeddedViewRef,\n  Optional,\n} from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { FormlyConfig } from '../services/formly.config';\nimport { FormlyFieldConfig, FormlyFieldConfigCache, FormlyHookConfig } from '../models';\nimport {\n  defineHiddenProp,\n  observe,\n  observeDeep,\n  getFieldValue,\n  assignFieldValue,\n  isObject,\n  markFieldForCheck,\n  hasKey,\n  IObserver,\n} from '../utils';\nimport { FieldWrapper } from '../templates/field.wrapper';\nimport { FieldType } from '../templates/field.type';\nimport { isObservable } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, startWith } from 'rxjs/operators';\nimport { FormlyFieldTemplates } from './formly.template';\n\n/**\n * The `<formly-field>` component is used to render the UI widget (layout + type) of a given `field`.\n */\n@Component({\n  selector: 'formly-field',\n  template: '<ng-template #container></ng-template>',\n  styleUrls: ['./formly.field.scss'],\n})\nexport class FormlyField implements DoCheck, OnInit, OnChanges, AfterContentInit, AfterViewInit, OnDestroy {\n  /** The field config. */\n  @Input() field: FormlyFieldConfig;\n  @ViewChild('container', { read: ViewContainerRef, static: true }) viewContainerRef!: ViewContainerRef;\n\n  private hostObservers: IObserver<any>[] = [];\n  private componentRefs: (ComponentRef<FieldType> | EmbeddedViewRef<FieldType>)[] = [];\n  private hooksObservers: Function[] = [];\n  private detectFieldBuild = false;\n\n  private get containerRef() {\n    return this.config.extras.renderFormlyFieldElement ? this.viewContainerRef : this.hostContainerRef;\n  }\n\n  private get elementRef() {\n    if (this.config.extras.renderFormlyFieldElement) {\n      return this._elementRef;\n    }\n    if (this.componentRefs?.[0] instanceof ComponentRef) {\n      return this.componentRefs[0].location;\n    }\n\n    return null;\n  }\n\n  valueChangesUnsubscribe = () => {};\n\n  constructor(\n    private config: FormlyConfig,\n    private renderer: Renderer2,\n    private _elementRef: ElementRef,\n    private hostContainerRef: ViewContainerRef,\n    @Optional() private form: FormlyFieldTemplates,\n  ) {}\n\n  ngAfterContentInit() {\n    this.triggerHook('afterContentInit');\n  }\n\n  ngAfterViewInit() {\n    this.triggerHook('afterViewInit');\n  }\n\n  ngDoCheck() {\n    if (this.detectFieldBuild && this.field && this.field.options) {\n      this.render();\n    }\n  }\n\n  ngOnInit() {\n    this.triggerHook('onInit');\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    this.triggerHook('onChanges', changes);\n  }\n\n  ngOnDestroy() {\n    this.resetRefs(this.field);\n    this.hostObservers.forEach((hostObserver) => hostObserver.unsubscribe());\n    this.hooksObservers.forEach((unsubscribe) => unsubscribe());\n    this.valueChangesUnsubscribe();\n    this.triggerHook('onDestroy');\n  }\n\n  private renderField(\n    containerRef: ViewContainerRef,\n    f: FormlyFieldConfigCache,\n    wrappers: FormlyFieldConfig['wrappers'] = [],\n  ) {\n    if (this.containerRef === containerRef) {\n      this.resetRefs(this.field);\n      this.containerRef.clear();\n      wrappers = this.field?.wrappers;\n    }\n\n    if (wrappers?.length > 0) {\n      const [wrapper, ...wps] = wrappers;\n      const { component } = this.config.getWrapper(wrapper);\n\n      const ref = containerRef.createComponent<FieldWrapper>(component);\n      this.attachComponentRef(ref, f);\n      observe<ViewContainerRef & { _lContainer: any }>(\n        ref.instance,\n        ['fieldComponent'],\n        ({ currentValue, previousValue, firstChange }) => {\n          if (currentValue) {\n            if (previousValue && previousValue._lContainer === currentValue._lContainer) {\n              return;\n            }\n\n            const viewRef = previousValue ? previousValue.detach() : null;\n            if (viewRef && !viewRef.destroyed) {\n              currentValue.insert(viewRef);\n            } else {\n              this.renderField(currentValue, f, wps);\n            }\n\n            !firstChange && ref.changeDetectorRef.detectChanges();\n          }\n        },\n      );\n    } else if (f?.type) {\n      const inlineType = this.form?.templates?.find((ref) => ref.name === f.type);\n      let ref: ComponentRef<any> | EmbeddedViewRef<any>;\n      if (inlineType) {\n        ref = containerRef.createEmbeddedView(inlineType.ref, { $implicit: f });\n      } else {\n        const { component } = this.config.getType(f.type, true);\n        ref = containerRef.createComponent<FieldWrapper>(component as any);\n      }\n      this.attachComponentRef(ref, f);\n    }\n  }\n\n  private triggerHook(name: keyof FormlyHookConfig, changes?: SimpleChanges) {\n    if (name === 'onInit' || (name === 'onChanges' && changes.field && !changes.field.firstChange)) {\n      this.valueChangesUnsubscribe();\n      this.valueChangesUnsubscribe = this.fieldChanges(this.field);\n    }\n\n    if (this.field?.hooks?.[name]) {\n      if (!changes || changes.field) {\n        const r = this.field.hooks[name](this.field);\n        if (isObservable(r) && ['onInit', 'afterContentInit', 'afterViewInit'].indexOf(name) !== -1) {\n          const sub = r.subscribe();\n          this.hooksObservers.push(() => sub.unsubscribe());\n        }\n      }\n    }\n\n    if (name === 'onChanges' && changes.field) {\n      this.resetRefs(changes.field.previousValue);\n      this.render();\n    }\n  }\n\n  private attachComponentRef<T extends FieldType>(\n    ref: ComponentRef<T> | EmbeddedViewRef<T>,\n    field: FormlyFieldConfigCache,\n  ) {\n    this.componentRefs.push(ref);\n    field._componentRefs.push(ref);\n    if (ref instanceof ComponentRef) {\n      Object.assign(ref.instance, { field });\n    }\n  }\n\n  private render() {\n    if (!this.field) {\n      return;\n    }\n\n    // require Formly build\n    if (!this.field.options) {\n      this.detectFieldBuild = true;\n\n      return;\n    }\n\n    this.detectFieldBuild = false;\n    this.hostObservers.forEach((hostObserver) => hostObserver.unsubscribe());\n    this.hostObservers = [\n      observe<boolean>(this.field, ['hide'], ({ firstChange, currentValue }) => {\n        const containerRef = this.containerRef;\n        if (this.config.extras.lazyRender === false) {\n          firstChange && this.renderField(containerRef, this.field);\n          if (!firstChange || (firstChange && currentValue)) {\n            this.elementRef &&\n              this.renderer.setStyle(this.elementRef.nativeElement, 'display', currentValue ? 'none' : '');\n          }\n        } else {\n          if (currentValue) {\n            containerRef.clear();\n            if (this.field.className) {\n              this.renderer.removeAttribute(this.elementRef.nativeElement, 'class');\n            }\n          } else {\n            this.renderField(containerRef, this.field);\n            if (this.field.className) {\n              this.renderer.setAttribute(this.elementRef.nativeElement, 'class', this.field.className);\n            }\n          }\n        }\n\n        !firstChange && this.field.options.detectChanges(this.field);\n      }),\n      observe<string>(this.field, ['className'], ({ firstChange, currentValue }) => {\n        if (\n          (!firstChange || (firstChange && currentValue)) &&\n          (!this.config.extras.lazyRender || this.field.hide !== true)\n        ) {\n          this.elementRef && this.renderer.setAttribute(this.elementRef.nativeElement, 'class', currentValue);\n        }\n      }),\n      ...['touched', 'pristine', 'status'].map((prop) =>\n        observe<string>(\n          this.field,\n          ['formControl', prop],\n          ({ firstChange }) => !firstChange && markFieldForCheck(this.field),\n        ),\n      ),\n    ];\n  }\n\n  private resetRefs(field: FormlyFieldConfigCache) {\n    if (field) {\n      if (field._localFields) {\n        field._localFields = [];\n      } else {\n        defineHiddenProp(this.field, '_localFields', []);\n      }\n\n      if (field._componentRefs) {\n        field._componentRefs = field._componentRefs.filter((ref) => this.componentRefs.indexOf(ref) === -1);\n      } else {\n        defineHiddenProp(this.field, '_componentRefs', []);\n      }\n    }\n\n    this.componentRefs = [];\n  }\n\n  private fieldChanges(field: FormlyFieldConfigCache | undefined) {\n    if (!field) {\n      return () => {};\n    }\n\n    const subscribes = [observeDeep(field, ['props'], () => field.options.detectChanges(field))];\n\n    if (field.options) {\n      subscribes.push(observeDeep(field.options, ['formState'], () => field.options.detectChanges(field)));\n    }\n\n    for (const key of Object.keys(field._expressions || {})) {\n      const expressionObserver = observe<FormlyFieldConfigCache['_expressions']['key']>(\n        field,\n        ['_expressions', key],\n        ({ currentValue, previousValue }) => {\n          if (previousValue?.subscription) {\n            previousValue.subscription.unsubscribe();\n            previousValue.subscription = null;\n          }\n          if (isObservable(currentValue.value$)) {\n            currentValue.subscription = currentValue.value$.subscribe();\n          }\n        },\n      );\n      subscribes.push(() => {\n        if (field._expressions[key]?.subscription) {\n          field._expressions[key].subscription.unsubscribe();\n        }\n        expressionObserver.unsubscribe();\n      });\n    }\n\n    for (const path of [['focus'], ['template'], ['fieldGroupClassName'], ['validation', 'show']]) {\n      const fieldObserver = observe(\n        field,\n        path,\n        ({ firstChange }) => !firstChange && field.options.detectChanges(field),\n      );\n      subscribes.push(() => fieldObserver.unsubscribe());\n    }\n\n    if (field.formControl && !field.fieldGroup) {\n      const control = field.formControl;\n      let valueChanges = control.valueChanges.pipe(\n        distinctUntilChanged((x, y) => {\n          if (x !== y || Array.isArray(x) || isObject(x)) {\n            return false;\n          }\n\n          return true;\n        }),\n      );\n\n      if (control.value !== getFieldValue(field)) {\n        valueChanges = valueChanges.pipe(startWith(control.value));\n      }\n\n      const { updateOn, debounce } = field.modelOptions;\n      if ((!updateOn || updateOn === 'change') && debounce?.default > 0) {\n        valueChanges = control.valueChanges.pipe(debounceTime(debounce.default));\n      }\n\n      const sub = valueChanges.subscribe((value) => {\n        // workaround for https://github.com/angular/angular/issues/13792\n        if (control._fields?.length > 1 && control instanceof FormControl) {\n          control.patchValue(value, { emitEvent: false, onlySelf: true });\n        }\n\n        field.parsers?.forEach((parserFn) => (value = parserFn(value)));\n        if (value !== field.formControl.value) {\n          field.formControl.setValue(value);\n          return;\n        }\n\n        if (hasKey(field)) {\n          assignFieldValue(field, value);\n        }\n        field.options.fieldChanges.next({ value, field, type: 'valueChanges' });\n      });\n\n      subscribes.push(() => sub.unsubscribe());\n    }\n\n    let templateFieldsSubs: (() => void)[] = [];\n    observe(field, ['_localFields'], ({ currentValue }) => {\n      templateFieldsSubs.forEach((unsubscribe) => unsubscribe());\n      templateFieldsSubs = (currentValue || []).map((f: FormlyFieldConfigCache) => this.fieldChanges(f));\n    });\n\n    return () => {\n      subscribes.forEach((unsubscribe) => unsubscribe());\n      templateFieldsSubs.forEach((unsubscribe) => unsubscribe());\n    };\n  }\n}\n"]}