@ngx-formly/core
Version:
Formly is a dynamic (JSON powered) form library for Angular that bring unmatched maintainability to your application's forms.
144 lines • 21.3 kB
JavaScript
import { ComponentRef } from '@angular/core';
import { getFieldId, assignFieldValue, isUndefined, getFieldValue, reverseDeepMerge, defineHiddenProp, clone, getField, markFieldForCheck, hasKey, observe, isHiddenField, isSignalRequired, } from '../../utils';
import { Subject } from 'rxjs';
export class CoreExtension {
constructor(config) {
this.config = config;
this.formId = 0;
}
prePopulate(field) {
const root = field.parent;
this.initRootOptions(field);
this.initFieldProps(field);
if (root) {
Object.defineProperty(field, 'options', { get: () => root.options, configurable: true });
Object.defineProperty(field, 'model', {
get: () => (hasKey(field) && field.fieldGroup ? getFieldValue(field) : root.model),
configurable: true,
});
}
Object.defineProperty(field, 'get', {
value: (key) => getField(field, key),
configurable: true,
});
this.getFieldComponentInstance(field).prePopulate?.(field);
}
onPopulate(field) {
this.initFieldOptions(field);
this.getFieldComponentInstance(field).onPopulate?.(field);
if (field.fieldGroup) {
field.fieldGroup.forEach((f, index) => {
if (f) {
Object.defineProperty(f, 'parent', { get: () => field, configurable: true });
Object.defineProperty(f, 'index', { get: () => index, configurable: true });
}
this.formId++;
});
}
}
postPopulate(field) {
this.getFieldComponentInstance(field).postPopulate?.(field);
}
initFieldProps(field) {
field.props ??= field.templateOptions;
Object.defineProperty(field, 'templateOptions', {
get: () => field.props,
set: (props) => (field.props = props),
configurable: true,
});
}
initRootOptions(field) {
if (field.parent) {
return;
}
const options = field.options;
field.options.formState = field.options.formState || {};
if (!options.showError) {
options.showError = this.config.extras.showError;
}
if (!options.fieldChanges) {
defineHiddenProp(options, 'fieldChanges', new Subject());
}
if (!options._hiddenFieldsForCheck) {
options._hiddenFieldsForCheck = [];
}
options._detectChanges = (f) => {
if (f._componentRefs) {
markFieldForCheck(f);
}
f.fieldGroup?.forEach((f) => f && options._detectChanges(f));
};
options.detectChanges = (f) => {
f.options.checkExpressions?.(f);
options._detectChanges(f);
};
options.resetModel = (model) => {
model = clone(model ?? options._initialModel);
if (field.model) {
Object.keys(field.model).forEach((k) => delete field.model[k]);
Object.assign(field.model, model || {});
}
if (!isSignalRequired()) {
observe(options, ['parentForm', 'submitted']).setValue(false, false);
}
options.build(field);
field.form.reset(field.model);
};
options.updateInitialValue = (model) => (options._initialModel = clone(model ?? field.model));
field.options.updateInitialValue();
}
initFieldOptions(field) {
reverseDeepMerge(field, {
id: getFieldId(`formly_${this.formId}`, field, field.index),
hooks: {},
modelOptions: {},
validation: { messages: {} },
props: !field.type || !hasKey(field)
? {}
: {
label: '',
placeholder: '',
disabled: false,
},
});
if (this.config.extras.resetFieldOnHide && field.resetOnHide !== false) {
field.resetOnHide = true;
}
if (field.type !== 'formly-template' &&
(field.template || field.expressions?.template || field.expressionProperties?.template)) {
field.type = 'formly-template';
}
if (!field.type && field.fieldGroup) {
field.type = 'formly-group';
}
if (field.type) {
this.config.getMergedField(field);
}
if (hasKey(field) &&
!isUndefined(field.defaultValue) &&
isUndefined(getFieldValue(field)) &&
!isHiddenField(field)) {
assignFieldValue(field, field.defaultValue);
}
field.wrappers = field.wrappers || [];
}
getFieldComponentInstance(field) {
const componentRefInstance = () => {
let componentRef = this.config.resolveFieldTypeRef(field);
const fieldComponentRef = field._componentRefs?.slice(-1)[0];
if (fieldComponentRef instanceof ComponentRef &&
fieldComponentRef?.componentType === componentRef?.componentType) {
componentRef = fieldComponentRef;
}
return componentRef?.instance;
};
if (!field._proxyInstance) {
defineHiddenProp(field, '_proxyInstance', new Proxy({}, {
get: (_, prop) => componentRefInstance()?.[prop],
set: (_, prop, value) => (componentRefInstance()[prop] = value),
}));
}
return field._proxyInstance;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core.js","sourceRoot":"","sources":["../../../../../../../src/core/src/lib/extensions/core/core.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentRef } from '@angular/core';\nimport { FormlyConfig } from '../../services/formly.config';\nimport { FormlyFieldConfigCache, FormlyValueChangeEvent, FormlyExtension, FormlyFieldConfig } from '../../models';\nimport {\n  getFieldId,\n  assignFieldValue,\n  isUndefined,\n  getFieldValue,\n  reverseDeepMerge,\n  defineHiddenProp,\n  clone,\n  getField,\n  markFieldForCheck,\n  hasKey,\n  observe,\n  isHiddenField,\n  isSignalRequired,\n} from '../../utils';\nimport { Subject } from 'rxjs';\n\nexport class CoreExtension implements FormlyExtension {\n  private formId = 0;\n  constructor(private config: FormlyConfig) {}\n\n  prePopulate(field: FormlyFieldConfigCache) {\n    const root = field.parent;\n    this.initRootOptions(field);\n    this.initFieldProps(field);\n    if (root) {\n      Object.defineProperty(field, 'options', { get: () => root.options, configurable: true });\n      Object.defineProperty(field, 'model', {\n        get: () => (hasKey(field) && field.fieldGroup ? getFieldValue(field) : root.model),\n        configurable: true,\n      });\n    }\n\n    Object.defineProperty(field, 'get', {\n      value: (key: FormlyFieldConfig['key']) => getField(field, key),\n      configurable: true,\n    });\n\n    this.getFieldComponentInstance(field).prePopulate?.(field);\n  }\n\n  onPopulate(field: FormlyFieldConfigCache) {\n    this.initFieldOptions(field);\n    this.getFieldComponentInstance(field).onPopulate?.(field);\n    if (field.fieldGroup) {\n      field.fieldGroup.forEach((f, index) => {\n        if (f) {\n          Object.defineProperty(f, 'parent', { get: () => field, configurable: true });\n          Object.defineProperty(f, 'index', { get: () => index, configurable: true });\n        }\n        this.formId++;\n      });\n    }\n  }\n\n  postPopulate(field: FormlyFieldConfigCache) {\n    this.getFieldComponentInstance(field).postPopulate?.(field);\n  }\n\n  private initFieldProps(field: FormlyFieldConfigCache) {\n    field.props ??= field.templateOptions;\n    Object.defineProperty(field, 'templateOptions', {\n      get: () => field.props,\n      set: (props) => (field.props = props),\n      configurable: true,\n    });\n  }\n\n  private initRootOptions(field: FormlyFieldConfigCache) {\n    if (field.parent) {\n      return;\n    }\n\n    const options = field.options;\n    field.options.formState = field.options.formState || {};\n    if (!options.showError) {\n      options.showError = this.config.extras.showError;\n    }\n\n    if (!options.fieldChanges) {\n      defineHiddenProp(options, 'fieldChanges', new Subject<FormlyValueChangeEvent>());\n    }\n\n    if (!options._hiddenFieldsForCheck) {\n      options._hiddenFieldsForCheck = [];\n    }\n\n    options._detectChanges = (f: FormlyFieldConfigCache) => {\n      if (f._componentRefs) {\n        markFieldForCheck(f);\n      }\n\n      f.fieldGroup?.forEach((f) => f && options._detectChanges(f));\n    };\n\n    options.detectChanges = (f: FormlyFieldConfigCache) => {\n      f.options.checkExpressions?.(f);\n      options._detectChanges(f);\n    };\n\n    options.resetModel = (model?: any) => {\n      model = clone(model ?? options._initialModel);\n      if (field.model) {\n        Object.keys(field.model).forEach((k) => delete field.model[k]);\n        Object.assign(field.model, model || {});\n      }\n\n      if (!isSignalRequired()) {\n        observe(options, ['parentForm', 'submitted']).setValue(false, false);\n      }\n      options.build(field);\n      field.form.reset(field.model);\n    };\n\n    options.updateInitialValue = (model?: any) => (options._initialModel = clone(model ?? field.model));\n    field.options.updateInitialValue();\n  }\n\n  private initFieldOptions(field: FormlyFieldConfigCache) {\n    reverseDeepMerge(field, {\n      id: getFieldId(`formly_${this.formId}`, field, field.index),\n      hooks: {},\n      modelOptions: {},\n      validation: { messages: {} },\n      props:\n        !field.type || !hasKey(field)\n          ? {}\n          : {\n              label: '',\n              placeholder: '',\n              disabled: false,\n            },\n    });\n\n    if (this.config.extras.resetFieldOnHide && field.resetOnHide !== false) {\n      field.resetOnHide = true;\n    }\n\n    if (\n      field.type !== 'formly-template' &&\n      (field.template || field.expressions?.template || field.expressionProperties?.template)\n    ) {\n      field.type = 'formly-template';\n    }\n\n    if (!field.type && field.fieldGroup) {\n      field.type = 'formly-group';\n    }\n\n    if (field.type) {\n      this.config.getMergedField(field);\n    }\n\n    if (\n      hasKey(field) &&\n      !isUndefined(field.defaultValue) &&\n      isUndefined(getFieldValue(field)) &&\n      !isHiddenField(field)\n    ) {\n      assignFieldValue(field, field.defaultValue);\n    }\n\n    field.wrappers = field.wrappers || [];\n  }\n\n  private getFieldComponentInstance(field: FormlyFieldConfigCache) {\n    const componentRefInstance = () => {\n      let componentRef = this.config.resolveFieldTypeRef(field);\n\n      const fieldComponentRef = field._componentRefs?.slice(-1)[0];\n      if (\n        fieldComponentRef instanceof ComponentRef &&\n        fieldComponentRef?.componentType === componentRef?.componentType\n      ) {\n        componentRef = fieldComponentRef as any;\n      }\n\n      return componentRef?.instance as any;\n    };\n\n    if (!field._proxyInstance) {\n      defineHiddenProp(\n        field,\n        '_proxyInstance',\n        new Proxy({} as FormlyExtension, {\n          get: (_, prop) => componentRefInstance()?.[prop],\n          set: (_, prop, value) => (componentRefInstance()[prop] = value),\n        }),\n      );\n    }\n\n    return field._proxyInstance;\n  }\n}\n"]}