@formio-tmt/angular
Version:
This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 10.1.4.
144 lines • 22.1 kB
JavaScript
// @ts-nocheck
import { Components, Utils as FormioUtils } from '@formio-tmt/js';
import { clone, isNil, isArray } from 'lodash';
const BaseInputComponent = Components.components.input;
const TextfieldComponent = Components.components.textfield;
export function createCustomFormioComponent(customComponentOptions) {
return class CustomComponent extends BaseInputComponent {
component;
static editForm = customComponentOptions.editForm || TextfieldComponent.editForm;
id = FormioUtils.getRandomComponentId();
type = customComponentOptions.type;
_customAngularElement;
static schema() {
return BaseInputComponent.schema({
...customComponentOptions.schema,
type: customComponentOptions.type,
});
}
get defaultSchema() {
return CustomComponent.schema();
}
get emptyValue() {
return customComponentOptions.emptyValue || null;
}
static get builderInfo() {
return {
title: customComponentOptions.title,
group: customComponentOptions.group,
icon: customComponentOptions.icon,
weight: customComponentOptions.weight,
documentation: customComponentOptions.documentation,
schema: CustomComponent.schema(),
};
}
constructor(component, options, data) {
super(component, {
...options,
sanitizeConfig: {
addTags: [customComponentOptions.selector],
},
}, data);
this.component = component;
if (customComponentOptions.extraValidators) {
this.validators = this.validators.concat(customComponentOptions.extraValidators);
}
}
elementInfo() {
const info = super.elementInfo();
info.type = customComponentOptions.selector;
info.changeEvent = customComponentOptions.changeEvent || 'valueChange';
info.attr = {
...info.attr,
class: info.attr.class.replace('form-control', 'form-control-custom-field') // remove the form-control class as the custom angular component may look different
};
return info;
}
get inputInfo() {
const info = {
id: this.key,
...this.elementInfo()
};
return info;
}
renderElement(value, index) {
const info = this.inputInfo;
return this.renderTemplate(customComponentOptions.template || 'input', {
input: info,
value,
index
});
}
attach(element) {
let superAttach = super.attach(element);
this._customAngularElement = element.querySelector(customComponentOptions.selector);
// Bind the custom options and the validations to the Angular component's inputs (flattened)
if (this._customAngularElement) {
// To make sure we have working input in IE...
// IE doesn't render it properly if it's not visible on the screen
// due to the whole structure applied via innerHTML to the parent
// so we need to use appendChild
if (!this._customAngularElement.getAttribute('ng-version')) {
this._customAngularElement.removeAttribute('ref');
const newCustomElement = document.createElement(customComponentOptions.selector);
newCustomElement.setAttribute('ref', 'input');
Object.keys(this.inputInfo.attr).forEach((attr) => {
newCustomElement.setAttribute(attr, this.inputInfo.attr[attr]);
});
this._customAngularElement.appendChild(newCustomElement);
this._customAngularElement = newCustomElement;
superAttach = super.attach(element);
}
// Bind customOptions
for (const key in this.component.customOptions) {
if (this.component.customOptions.hasOwnProperty(key)) {
this._customAngularElement[key] = this.component.customOptions[key];
}
}
// Bind validate options
for (const key in this.component.validate) {
if (this.component.validate.hasOwnProperty(key)) {
this._customAngularElement[key] = this.component.validate[key];
}
}
// Bind options explicitly set
const fieldOptions = customComponentOptions.fieldOptions;
if (isArray(fieldOptions) && fieldOptions.length > 0) {
for (const key in fieldOptions) {
if (fieldOptions.hasOwnProperty(key)) {
this._customAngularElement[fieldOptions[key]] = this.component[fieldOptions[key]];
}
}
}
// Attach event listener for emit event
this._customAngularElement.addEventListener('formioEvent', (event) => {
this.emit(event.detail.eventName, {
...event.detail.data,
component: this.component
});
});
// Ensure we bind the value (if it isn't a multiple-value component with no wrapper)
if (!this._customAngularElement.value && !this.component.disableMultiValueWrapper) {
this.restoreValue();
}
}
return superAttach;
}
// Add extra option to support multiple value (e.g. datagrid) with single angular component (disableMultiValueWrapper)
useWrapper() {
return this.component.hasOwnProperty('multiple') && this.component.multiple && !this.component.disableMultiValueWrapper;
}
get defaultValue() {
let defaultValue = this.emptyValue;
// handle falsy default value
if (!isNil(this.component.defaultValue)) {
defaultValue = this.component.defaultValue;
}
if (this.component.customDefaultValue && !this.options.preview) {
defaultValue = this.evaluate(this.component.customDefaultValue, { value: '' }, 'value');
}
return clone(defaultValue);
}
};
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-custom-component.js","sourceRoot":"","sources":["../../../../projects/angular-formio/src/custom-component/create-custom-component.ts"],"names":[],"mappings":"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","sourcesContent":["// @ts-nocheck\r\nimport { BuilderInfo, Components, ExtendedComponentSchema, Utils as FormioUtils } from '@formio-tmt/js';\r\nimport { FormioCustomComponentInfo, FormioCustomElement, FormioEvent } from '../elements.common';\r\nimport { clone, isNil, isArray } from 'lodash';\r\n\r\nconst BaseInputComponent = Components.components.input;\r\nconst TextfieldComponent = Components.components.textfield;\r\n\r\nexport function createCustomFormioComponent(customComponentOptions: FormioCustomComponentInfo) {\r\n  return class CustomComponent extends BaseInputComponent {\r\n    static editForm = customComponentOptions.editForm || TextfieldComponent.editForm;\r\n    id = FormioUtils.getRandomComponentId();\r\n    type = customComponentOptions.type;\r\n    _customAngularElement: FormioCustomElement;\r\n\r\n    static schema() {\r\n      return BaseInputComponent.schema({\r\n        ...customComponentOptions.schema,\r\n        type: customComponentOptions.type,\r\n      });\r\n    }\r\n\r\n    get defaultSchema() {\r\n      return CustomComponent.schema();\r\n    }\r\n\r\n    get emptyValue() {\r\n      return customComponentOptions.emptyValue || null;\r\n    }\r\n\r\n    static get builderInfo(): BuilderInfo {\r\n      return {\r\n        title: customComponentOptions.title,\r\n        group: customComponentOptions.group,\r\n        icon: customComponentOptions.icon,\r\n        weight: customComponentOptions.weight,\r\n        documentation: customComponentOptions.documentation,\r\n        schema: CustomComponent.schema(),\r\n      };\r\n    }\r\n\r\n    constructor(public component: ExtendedComponentSchema, options: any, data: any) {\r\n      super(component, {\r\n        ...options,\r\n        sanitizeConfig: {\r\n          addTags: [customComponentOptions.selector],\r\n        },\r\n      }, data);\r\n\r\n      if (customComponentOptions.extraValidators) {\r\n        this.validators = this.validators.concat(customComponentOptions.extraValidators);\r\n      }\r\n    }\r\n\r\n    elementInfo() {\r\n      const info = super.elementInfo();\r\n      info.type = customComponentOptions.selector;\r\n      info.changeEvent = customComponentOptions.changeEvent || 'valueChange';\r\n      info.attr = {\r\n        ...info.attr,\r\n        class: info.attr.class.replace('form-control', 'form-control-custom-field') // remove the form-control class as the custom angular component may look different\r\n      };\r\n      return info;\r\n    }\r\n\r\n    get inputInfo() {\r\n      const info = {\r\n        id: this.key,\r\n        ...this.elementInfo()\r\n      }\r\n      return info;\r\n    }\r\n\r\n    renderElement(value: any, index: number) {\r\n      const info = this.inputInfo;\r\n      return this.renderTemplate(customComponentOptions.template || 'input', {\r\n        input: info,\r\n        value,\r\n        index\r\n      });\r\n    }\r\n\r\n    attach(element: HTMLElement) {\r\n      let superAttach = super.attach(element);\r\n\r\n      this._customAngularElement = element.querySelector(customComponentOptions.selector);\r\n\r\n      // Bind the custom options and the validations to the Angular component's inputs (flattened)\r\n      if (this._customAngularElement) {\r\n        // To make sure we have working input in IE...\r\n        // IE doesn't render it properly if it's not visible on the screen\r\n        // due to the whole structure applied via innerHTML to the parent\r\n        // so we need to use appendChild\r\n        if (!this._customAngularElement.getAttribute('ng-version')) {\r\n          this._customAngularElement.removeAttribute('ref');\r\n\r\n          const newCustomElement = document.createElement(customComponentOptions.selector) as FormioCustomElement;\r\n\r\n          newCustomElement.setAttribute('ref', 'input');\r\n          Object.keys(this.inputInfo.attr).forEach((attr: string) => {\r\n            newCustomElement.setAttribute(attr, this.inputInfo.attr[attr]);\r\n          });\r\n\r\n          this._customAngularElement.appendChild(newCustomElement);\r\n          this._customAngularElement = newCustomElement;\r\n\r\n          superAttach = super.attach(element);\r\n        }\r\n\r\n        // Bind customOptions\r\n        for (const key in this.component.customOptions) {\r\n          if (this.component.customOptions.hasOwnProperty(key)) {\r\n            this._customAngularElement[key] = this.component.customOptions[key];\r\n          }\r\n        }\r\n        // Bind validate options\r\n        for (const key in this.component.validate) {\r\n          if (this.component.validate.hasOwnProperty(key)) {\r\n            this._customAngularElement[key] = this.component.validate[key];\r\n          }\r\n        }\r\n        // Bind options explicitly set\r\n        const fieldOptions = customComponentOptions.fieldOptions;\r\n        if (isArray(fieldOptions) && fieldOptions.length > 0) {\r\n          for (const key in fieldOptions) {\r\n            if (fieldOptions.hasOwnProperty(key)) {\r\n              this._customAngularElement[fieldOptions[key]] = this.component[fieldOptions[key]];\r\n            }\r\n          }\r\n        }\r\n\r\n        // Attach event listener for emit event\r\n        this._customAngularElement.addEventListener('formioEvent', (event: CustomEvent<FormioEvent>) => {\r\n          this.emit(event.detail.eventName, {\r\n            ...event.detail.data,\r\n            component: this.component\r\n          });\r\n        });\r\n\r\n        // Ensure we bind the value (if it isn't a multiple-value component with no wrapper)\r\n        if (!this._customAngularElement.value && !this.component.disableMultiValueWrapper) {\r\n          this.restoreValue();\r\n        }\r\n\r\n      }\r\n      return superAttach;\r\n    }\r\n\r\n    // Add extra option to support multiple value (e.g. datagrid) with single angular component (disableMultiValueWrapper)\r\n    useWrapper() {\r\n      return this.component.hasOwnProperty('multiple') && this.component.multiple && !this.component.disableMultiValueWrapper;\r\n    }\r\n\r\n    get defaultValue() {\r\n      let defaultValue = this.emptyValue;\r\n\r\n      // handle falsy default value\r\n      if (!isNil(this.component.defaultValue)) {\r\n        defaultValue = this.component.defaultValue;\r\n      }\r\n\r\n      if (this.component.customDefaultValue && !this.options.preview) {\r\n        defaultValue = this.evaluate(\r\n          this.component.customDefaultValue,\r\n          { value: '' },\r\n          'value'\r\n        );\r\n      }\r\n\r\n      return clone(defaultValue);\r\n    }\r\n  };\r\n}\r\n"]}