UNPKG

@stratio/egeo

Version:
521 lines 47.4 kB
/** * @fileoverview added by tsickle * Generated from: lib/st-form/st-form.component.ts * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ /* * © 2017 Stratio Big Data Inc., Sucursal en España. * * This software is licensed under the Apache License, Version 2.0. * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the terms of the License for more details. * * SPDX-License-Identifier: Apache-2.0. */ import { ChangeDetectionStrategy, Component, EventEmitter, forwardRef, Input, Output, ViewChild } from '@angular/core'; import { NG_VALIDATORS, NG_VALUE_ACCESSOR, NgForm } from '@angular/forms'; import { FORM_UI_COMPONENT } from './st-form-field/st-form-field.interface'; /** * \@description {Component} [Dynamic form] * * The form component allows to generate forms dynamically using a JSON schema. * * \@model * * [Form Schema] {./st-form.model.ts#StFormSchema} * [Form UI Definition] {./st-form.model.ts#StFormUIDefinition} * * \@example * * {html} * * ``` * <st-form [schema]="jsonSchema" [(ngModel)]="model" #formModel="ngModel"> * </st-form> * * ``` * */ export class StFormComponent { constructor() { /** * \@Input {string} [nestingLevel=0] This informs about the nesting level of the form. This input is only used for design purposes */ this.nestingLevel = 0; /** * \@Input {string} [disabledSectionMessage='for this instance is disabled.'] * Message displayed when a section is disabled. This is always displayed after the section name */ this.disabledSectionMessage = 'for this instance is disabled.'; /** * \@Input {number} [sectionDescriptionLevel=-1] Level of the section to which description is displayed. * By default, section descriptions are not displayed. */ this.sectionDescriptionLevel = -1; /** * \@Input {boolean} [showTooltips=-1] Enable or disable displaying of tooltips * By default, tooltips are displayed */ this.showTooltips = true; /** * \@Output {any} [valueChange=] Event emitted when value is changed. This emits the current form value */ this.valueChange = new EventEmitter(); /** * \@Output {string} [clickLink=] Event emitted when link is clicked. It returns the field path */ this.clickLink = new EventEmitter(); this.showCollapsedSectionFields = false; this.innerValue = {}; this._value = {}; this._parentFieldSubscription = []; this.onTouched = (/** * @return {?} */ () => { }); } /** * \@Input {StFormSchema [schema=] Form schema needed to generate the form * @return {?} */ get schema() { return this._schema; } /** * @param {?} schema * @return {?} */ set schema(schema) { this._schema = schema; this._value = {}; if (this._schema.dependencies) { this._parentFields = Object.keys(this._schema.dependencies); } } /** * @return {?} */ ngAfterViewInit() { if (!this.forceValidations) { this.form.form.markAsPristine(); } } /** * @return {?} */ ngAfterViewChecked() { if (this._parentFields && this.form.control && this.form.control.controls) { for (let i = 0; i < this._parentFields.length; ++i) { /** @type {?} */ let parentField = this._parentFields[i]; if (!this._parentFieldSubscription[i] && this.form.control.controls[parentField]) { this._parentFieldSubscription[i] = this.form.control.controls[this._parentFields[i]].valueChanges.subscribe((/** * @param {?} value * @return {?} */ (value) => { if (!value) { /** @type {?} */ let childrenFields = this.schema.dependencies[parentField]; for (let j = 0; j < childrenFields.length; ++j) { if (this.form.controls[childrenFields[j]]) { this._value[childrenFields[j]] = undefined; } } } })); } } } } // Function to call when the value changes. /** * @param {?} _ * @return {?} */ onChange(_) { } /** * @param {?} control * @return {?} */ validate(control) { /** @type {?} */ let errors = null; if (this.form) { Object.keys(this.form.controls).forEach((/** * @param {?} propertyName * @return {?} */ (propertyName) => { if (this.form.controls[propertyName] && this.form.controls[propertyName].errors) { if (!errors) { errors = {}; } errors[propertyName] = this.form.controls[propertyName].errors; } })); this.form.control.setErrors(errors); } return errors; } /** * @param {?} propertyName * @return {?} */ isRequired(propertyName) { return propertyName && this.schema.required && this.schema.required.indexOf(propertyName) !== -1; } /** * @return {?} */ isCollapsedSection() { return this.schema.type === 'object' && this.schema.ui && (this.schema.ui.component === FORM_UI_COMPONENT.SHOW_MORE || this.schema.ui.component === FORM_UI_COMPONENT.ACCORDION); } /** * @return {?} */ iShowMoreSection() { return this.schema.type === 'object' && this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.SHOW_MORE; } /** * @param {?} propertyName * @return {?} */ isAParentField(propertyName) { return this.schema.dependencies && this.schema.dependencies[propertyName] && this.schema.dependencies[propertyName].length > 0; } /** * @return {?} */ getOptionalButtonLabel() { /** @type {?} */ let label = 'Additional options'; if (this.parentName || this.schema.title) { label += ' of '; label += this.parentName || this.schema.title; } return label; } /** * @return {?} */ onChangeOptionalFieldsVisibility() { this.showCollapsedSectionFields = !this.showCollapsedSectionFields; } /** * @param {?} propertyName * @return {?} */ getFieldClasses(propertyName) { return { 'hidden': this.isCollapsedSection() && !this.showCollapsedSectionFields, 'parent-field': this.isAParentField(propertyName) || (this.isASwitchSection() && this.isTheFirstField(propertyName)) }; } /** * @param {?} propertyName * @return {?} */ fieldHasToBeCreated(propertyName) { /** @type {?} */ let createField = true; /** @type {?} */ let parentField = this.getParentField(propertyName); if (((parentField && !this._value[parentField]) || (this.isInADisabledSection() && !this.isTheFirstField(propertyName))) || !this.fulfillDependencyVisibility(propertyName)) { createField = false; this._value[propertyName] = undefined; } return createField; } /** * @param {?} propertyName * @return {?} */ isRelatedField(propertyName) { return this._schema.properties[propertyName].ui && this.schema.properties[propertyName].ui.relatedTo && this.schema.properties[propertyName].ui.relatedTo.length > 0; } // When value is received from outside /** * @param {?} value * @return {?} */ writeValue(value) { if (value) { this.onChange(value); this.innerValue = value; } } /** * @param {?} value * @param {?} property * @return {?} */ onChangeProperty(value, property) { this._value[property] = value; this.valueChange.emit(this._value); setTimeout((/** * @return {?} */ () => { this.onChange(this._value); })); } // Registry the change function to propagate internal model changes /** * @param {?} fn * @return {?} */ registerOnChange(fn) { this.onChange = fn; } // Registry the touch function to propagate internal touch events TODO: make this function. /** * @param {?} fn * @return {?} */ registerOnTouched(fn) { this.onTouched = fn; } // Allows Angular to disable the form. /** * @param {?} disable * @return {?} */ setDisabledState(disable) { if (disable) { this.form.control.disable(); } else { this.form.control.enable(); } } /** * @return {?} */ ngOnDestroy() { if (this._parentFieldSubscription) { for (let i = 0; i < this._parentFieldSubscription.length; ++i) { this._parentFieldSubscription[i].unsubscribe(); } } } /** * @return {?} */ isInADisabledSection() { if (this.isASwitchSection()) { /** @type {?} */ let sectionEnabler = Object.keys(this.schema.properties)[0]; return !(this.form && this.form.controls[sectionEnabler] && this.form.controls[sectionEnabler].value); } return false; } /** * @return {?} */ isASwitchSection() { return this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.SWITCH; } /** * @return {?} */ isAnAccordion() { return this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.ACCORDION; } /** * @return {?} */ onClickTitle() { if (this.isAnAccordion()) { this.onChangeOptionalFieldsVisibility(); } } /** * @param {?} propertyName * @return {?} */ isTheFirstField(propertyName) { return propertyName === Object.keys(this.schema.properties)[0]; } /** * @param {?} fieldKey * @return {?} */ onClickLink(fieldKey) { this.clickLink.emit(fieldKey); } /** * @private * @param {?} propertyName * @return {?} */ getParentField(propertyName) { /** @type {?} */ let parentField = undefined; if (this.schema.dependencies) { Object.keys(this.schema.dependencies).forEach((/** * @param {?} key * @return {?} */ (key) => { if (this.schema.dependencies[key].indexOf(propertyName) !== -1) { parentField = key; } })); } return parentField; } /** * @private * @param {?} propertyName * @return {?} */ fulfillDependencyVisibility(propertyName) { /** @type {?} */ let propertySchema = this.schema.properties[propertyName]; /** @type {?} */ let fulfill = !propertySchema.ui || propertySchema.ui.visible !== false; if (propertySchema.ui && propertySchema.ui.visible) { /** @type {?} */ const keys = Object.keys(propertySchema.ui.visible); /** @type {?} */ let i = 0; while (fulfill && i < keys.length && propertySchema.ui.visible.hasOwnProperty(keys[i])) { if (this._value[keys[i]] !== propertySchema.ui.visible[keys[i]]) { fulfill = false; } ++i; } } return fulfill; } } StFormComponent.decorators = [ { type: Component, args: [{ selector: 'st-form', template: "<!--\n\n \u00A9 2017 Stratio Big Data Inc., Sucursal en Espa\u00F1a.\n\n This software is licensed under the Apache License, Version 2.0.\n This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;\n without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n See the terms of the License for more details.\n\n SPDX-License-Identifier: Apache-2.0.\n\n-->\n\n<form #form=\"ngForm\" *ngIf=\"innerValue\"\n [ngClass]=\"{'category': nestingLevel === 0,\n 'section': nestingLevel === 1,\n 'subsection': nestingLevel > 1,\n 'accordion': isAnAccordion(),\n 'accordion--open': isAnAccordion() && showCollapsedSectionFields}\"\n novalidate>\n <h1 *ngIf=\"schema.title && !isASwitchSection()\" class=\"title\" (click)=\"onClickTitle()\">{{schema.title}}</h1>\n <p *ngIf=\"schema.description && nestingLevel <= sectionDescriptionLevel\" class=\"description\">\n {{schema.description}}</p>\n <span class=\"optional-fields-button\" *ngIf=\"iShowMoreSection()\">\n <button class=\"button button-link\"\n (click)=\"onChangeOptionalFieldsVisibility()\">\n <span> <i\n [ngClass]=\"{'icon-arrow2_down': showCollapsedSectionFields, 'icon-arrow2_right': !showCollapsedSectionFields}\"></i> {{getOptionalButtonLabel()}}</span>\n </button>\n </span>\n <ng-container *ngFor=\"let property of schema.properties | stObjectToArray; let index = index;\">\n <div class=\"form-field-container\">\n <st-form-field *ngIf=\"property.value.type !== 'object' && fieldHasToBeCreated(property.key)\" class=\"form-field\"\n [schema]=\"property\"\n [name]=\"property.key\"\n [maxWidth]=\"textFieldMaxWidth\"\n [ngModel]=\"innerValue[property.key]\"\n [forceValidations]=\"forceValidations\"\n (ngModelChange)=\"onChangeProperty($event, property.key)\"\n [required]=\"isRequired(property.key)\"\n [hasDependencies]=\"isAParentField(property.key) || (isASwitchSection() && isTheFirstField(property.key))\"\n [ngClass]=\"getFieldClasses(property.key)\"\n [showTooltip]=\"showTooltips\"\n [qaTag]=\"property.key\"\n [errorMessages]=\"errorMessages\"\n [translations]=\"translations\"\n [attr.id]=\"property.key + '-form-field'\"\n (clickLink)=\"onClickLink($event)\">\n </st-form-field>\n\n <div *ngIf=\"property.value.type === 'object' && fieldHasToBeCreated(property.key)\">\n <st-form\n [hidden]=\"isCollapsedSection() && !showCollapsedSectionFields\"\n [schema]=\"property.value\"\n [parentName]=\"schema.title\"\n [name]=\"property.key\"\n [textFieldMaxWidth]=\"textFieldMaxWidth\"\n [ngModel]=\"innerValue[property.key]\"\n [attr.id]=\"property.key + '-section'\"\n [nestingLevel]=\"nestingLevel + 1\"\n [showTooltips]=\"showTooltips\"\n [sectionDescriptionLevel]=\"sectionDescriptionLevel\"\n [errorMessages]=\"errorMessages\"\n [translations]=\"translations\"\n (clickLink)=\"onClickLink(property.key + '.' + $event)\"\n (ngModelChange)=\"onChangeProperty($event, property.key)\">\n </st-form>\n </div>\n </div>\n <span class=\"line-break\" *ngIf=\"!isRelatedField(property.key)\"></span>\n </ng-container>\n\n <p *ngIf=\"isInADisabledSection()\" class=\"disabled-section-info\"> {{schema.title}} {{disabledSectionMessage}}</p>\n</form>\n", host: { class: 'st-form' }, changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef((/** * @return {?} */ () => StFormComponent)), multi: true }, { provide: NG_VALIDATORS, useExisting: forwardRef((/** * @return {?} */ () => StFormComponent)), multi: true } ], styles: ["@charset \"UTF-8\";.form-field{display:block}.form-field.hidden{visibility:hidden;position:absolute}.section-title{padding:10px 0 20px}"] }] } ]; StFormComponent.propDecorators = { parentName: [{ type: Input }], nestingLevel: [{ type: Input }], forceValidations: [{ type: Input }], disabledSectionMessage: [{ type: Input }], sectionDescriptionLevel: [{ type: Input }], textFieldMaxWidth: [{ type: Input }], errorMessages: [{ type: Input }], showTooltips: [{ type: Input }], translations: [{ type: Input }], valueChange: [{ type: Output }], clickLink: [{ type: Output }], form: [{ type: ViewChild, args: ['form', { static: false },] }], schema: [{ type: Input }] }; if (false) { /** * \@Input {string} [parentName=] Name of the parent section. By default, it is undefined * @type {?} */ StFormComponent.prototype.parentName; /** * \@Input {string} [nestingLevel=0] This informs about the nesting level of the form. This input is only used for design purposes * @type {?} */ StFormComponent.prototype.nestingLevel; /** * \@Input {boolean} [forceValidations=] Boolean to force the field validations * @type {?} */ StFormComponent.prototype.forceValidations; /** * \@Input {string} [disabledSectionMessage='for this instance is disabled.'] * Message displayed when a section is disabled. This is always displayed after the section name * @type {?} */ StFormComponent.prototype.disabledSectionMessage; /** * \@Input {number} [sectionDescriptionLevel=-1] Level of the section to which description is displayed. * By default, section descriptions are not displayed. * @type {?} */ StFormComponent.prototype.sectionDescriptionLevel; /** * \@Input {number} [textFieldMaxWidth=] Maximum width of a field needed to paint a input or textarea * @type {?} */ StFormComponent.prototype.textFieldMaxWidth; /** * \@Input {StInputError} [errorMessages=] Field error translations * @type {?} */ StFormComponent.prototype.errorMessages; /** * \@Input {boolean} [showTooltips=-1] Enable or disable displaying of tooltips * By default, tooltips are displayed * @type {?} */ StFormComponent.prototype.showTooltips; /** * \@Input {StFormFieldTranslations} [translations=] Field translations * @type {?} */ StFormComponent.prototype.translations; /** * \@Output {any} [valueChange=] Event emitted when value is changed. This emits the current form value * @type {?} */ StFormComponent.prototype.valueChange; /** * \@Output {string} [clickLink=] Event emitted when link is clicked. It returns the field path * @type {?} */ StFormComponent.prototype.clickLink; /** @type {?} */ StFormComponent.prototype.form; /** @type {?} */ StFormComponent.prototype.showCollapsedSectionFields; /** @type {?} */ StFormComponent.prototype.innerValue; /** * @type {?} * @private */ StFormComponent.prototype._value; /** * @type {?} * @private */ StFormComponent.prototype._parentFieldSubscription; /** * @type {?} * @private */ StFormComponent.prototype._parentFields; /** * @type {?} * @private */ StFormComponent.prototype._schema; /** @type {?} */ StFormComponent.prototype.onTouched; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"st-form.component.js","sourceRoot":"ng://@stratio/egeo/","sources":["lib/st-form/st-form.component.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;AAUA,OAAO,EAGJ,uBAAuB,EACvB,SAAS,EACT,YAAY,EACZ,UAAU,EACV,KAAK,EAEL,MAAM,EACN,SAAS,EACX,MAAM,eAAe,CAAC;AACvB,OAAO,EAAqC,aAAa,EAAE,iBAAiB,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAM7G,OAAO,EAAE,iBAAiB,EAA2B,MAAM,yCAAyC,CAAC;;;;;;;;;;;;;;;;;;;;;;AAmCrG,MAAM,OAAO,eAAe;IAZ5B;;;;QAgBY,iBAAY,GAAW,CAAC,CAAC;;;;;QAMzB,2BAAsB,GAAW,gCAAgC,CAAC;;;;;QAKlE,4BAAuB,GAAW,CAAC,CAAC,CAAC;;;;;QAWrC,iBAAY,GAAY,IAAI,CAAC;;;;QAM5B,gBAAW,GAAsB,IAAI,YAAY,EAAO,CAAC;;;;QAEzD,cAAS,GAAyB,IAAI,YAAY,EAAU,CAAC;QAIhE,+BAA0B,GAAY,KAAK,CAAC;QAC5C,eAAU,GAAQ,EAAE,CAAC;QACpB,WAAM,GAAQ,EAAE,CAAC;QACjB,6BAAwB,GAAmB,EAAE,CAAC;QAgDtD,cAAS;;;QAAG,GAAG,EAAE;QACjB,CAAC,EAAA;IA8KJ,CAAC;;;;;IA1NE,IAAa,MAAM;QAChB,OAAO,IAAI,CAAC,OAAO,CAAC;IACvB,CAAC;;;;;IAED,IAAI,MAAM,CAAC,MAAoB;QAC5B,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC;QACtB,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC;QACjB,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE;YAC5B,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;SAC9D;IACJ,CAAC;;;;IAED,eAAe;QACZ,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE;YACzB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;SAClC;IACJ,CAAC;;;;IAED,kBAAkB;QACf,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE;YACxE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE;;oBAC7C,WAAW,GAAW,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;gBAC/C,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE;oBAE/E,IAAI,CAAC,wBAAwB,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,SAAS;;;;oBAAC,CAAC,KAAK,EAAE,EAAE;wBACnH,IAAI,CAAC,KAAK,EAAE;;gCACL,cAAc,GAA2B,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,WAAW,CAAC;4BAClF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,cAAc,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE;gCAC7C,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,EAAE;oCACxC,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC;iCAC7C;6BACH;yBACH;oBACJ,CAAC,EAAC,CAAC;iBACL;aACH;SACH;IACJ,CAAC;;;;;;IAGD,QAAQ,CAAC,CAAM;IACf,CAAC;;;;;IAKD,QAAQ,CAAC,OAAoB;;YACtB,MAAM,GAAQ,IAAI;QACtB,IAAI,IAAI,CAAC,IAAI,EAAE;YACZ,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,OAAO;;;;YAAC,CAAC,YAAY,EAAE,EAAE;gBACtD,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC,MAAM,EAAE;oBAC9E,IAAI,CAAC,MAAM,EAAE;wBACV,MAAM,GAAG,EAAE,CAAC;qBACd;oBACD,MAAM,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC,MAAM,CAAC;iBACjE;YACJ,CAAC,EAAC,CAAC;YAEH,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;SACtC;QACD,OAAO,MAAM,CAAC;IACjB,CAAC;;;;;IAED,UAAU,CAAC,YAAoB;QAC5B,OAAO,YAAY,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;IACpG,CAAC;;;;IAED,kBAAkB;QACf,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE;YACnD,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,KAAK,iBAAiB,CAAC,SAAS,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,KAAK,iBAAiB,CAAC,SAAS,CAAC,CAAC;IAC7H,CAAC;;;;IAED,gBAAgB;QACb,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,KAAK,iBAAiB,CAAC,SAAS,CAAC;IACtH,CAAC;;;;;IAED,cAAc,CAAC,YAAoB;QAChC,OAAO,IAAI,CAAC,MAAM,CAAC,YAAY,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC;IAClI,CAAC;;;;IAED,sBAAsB;;YACf,KAAK,GAAW,oBAAoB;QAExC,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE;YACvC,KAAK,IAAI,MAAM,CAAC;YAChB,KAAK,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;SAChD;QAED,OAAO,KAAK,CAAC;IAChB,CAAC;;;;IAED,gCAAgC;QAC7B,IAAI,CAAC,0BAA0B,GAAG,CAAC,IAAI,CAAC,0BAA0B,CAAC;IACtE,CAAC;;;;;IAED,eAAe,CAAC,YAAoB;QACjC,OAAO;YACJ,QAAQ,EAAE,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,IAAI,CAAC,0BAA0B;YACvE,cAAc,EAAE,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,IAAI,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC;SACtH,CAAC;IACL,CAAC;;;;;IAED,mBAAmB,CAAC,YAAoB;;YACjC,WAAW,GAAY,IAAI;;YAC3B,WAAW,GAAW,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC;QAC3D,IAAI,CAAC,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC,CAAC;eAClH,CAAC,IAAI,CAAC,2BAA2B,CAAC,YAAY,CAAC,EAAE;YACpD,WAAW,GAAG,KAAK,CAAC;YACpB,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC;SACxC;QACD,OAAO,WAAW,CAAC;IACtB,CAAC;;;;;IAED,cAAc,CAAC,YAAoB;QAChC,OAAO,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC,EAAE,CAAC,SAAS;eAC9F,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC,EAAE,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC;IACtE,CAAC;;;;;;IAGD,UAAU,CAAC,KAAU;QAClB,IAAI,KAAK,EAAE;YACR,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACrB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;SAC1B;IACJ,CAAC;;;;;;IAED,gBAAgB,CAAC,KAAU,EAAE,QAAgB;QAC1C,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,KAAK,CAAC;QAC9B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACnC,UAAU;;;QAAC,GAAG,EAAE;YACb,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC9B,CAAC,EAAC,CAAC;IACN,CAAC;;;;;;IAGD,gBAAgB,CAAC,EAAoB;QAClC,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACtB,CAAC;;;;;;IAGD,iBAAiB,CAAC,EAAc;QAC7B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACvB,CAAC;;;;;;IAGD,gBAAgB,CAAC,OAAgB;QAC9B,IAAI,OAAO,EAAE;YACV,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;SAC9B;aAAM;YACJ,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;SAC7B;IACJ,CAAC;;;;IAED,WAAW;QACR,IAAI,IAAI,CAAC,wBAAwB,EAAE;YAChC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,wBAAwB,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE;gBAC5D,IAAI,CAAC,wBAAwB,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;aACjD;SACH;IACJ,CAAC;;;;IAED,oBAAoB;QACjB,IAAI,IAAI,CAAC,gBAAgB,EAAE,EAAE;;gBACtB,cAAc,GAAW,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YACnE,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC,KAAK,CAAC,CAAC;SACxG;QACD,OAAO,KAAK,CAAC;IAChB,CAAC;;;;IAED,gBAAgB;QACb,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,KAAK,iBAAiB,CAAC,MAAM,CAAC;IAClF,CAAC;;;;IAED,aAAa;QACV,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,KAAK,iBAAiB,CAAC,SAAS,CAAC;IACrF,CAAC;;;;IAED,YAAY;QACT,IAAI,IAAI,CAAC,aAAa,EAAE,EAAE;YACvB,IAAI,CAAC,gCAAgC,EAAE,CAAC;SAC1C;IACJ,CAAC;;;;;IAED,eAAe,CAAC,YAAoB;QACjC,OAAO,YAAY,KAAK,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;;;;;IAED,WAAW,CAAC,QAAgB;QACzB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACjC,CAAC;;;;;;IAEO,cAAc,CAAC,YAAoB;;YACpC,WAAW,GAAW,SAAS;QACnC,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE;YAC3B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,OAAO;;;;YAAC,CAAC,GAAW,EAAE,EAAE;gBAC3D,IAAI,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE;oBAC7D,WAAW,GAAG,GAAG,CAAC;iBACpB;YACJ,CAAC,EAAC,CAAC;SACL;QACD,OAAO,WAAW,CAAC;IACtB,CAAC;;;;;;IAEO,2BAA2B,CAAC,YAAoB;;YACjD,cAAc,GAAiB,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,YAAY,CAAC;;YACnE,OAAO,GAAY,CAAC,cAAc,CAAC,EAAE,IAAI,cAAc,CAAC,EAAE,CAAC,OAAO,KAAK,KAAK;QAChF,IAAI,cAAc,CAAC,EAAE,IAAI,cAAc,CAAC,EAAE,CAAC,OAAO,EAAE;;kBAC3C,IAAI,GAAa,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC,OAAO,CAAC;;gBACzD,CAAC,GAAG,CAAC;YACT,OAAO,OAAO,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,IAAI,cAAc,CAAC,EAAE,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE;gBACrF,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,cAAc,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE;oBAC9D,OAAO,GAAG,KAAK,CAAC;iBAClB;gBACD,EAAE,CAAC,CAAC;aACN;SACH;QACD,OAAO,OAAO,CAAC;IAClB,CAAC;;;YAnRH,SAAS,SAAC;gBACR,QAAQ,EAAE,SAAS;gBACnB,61HAAuC;gBAEvC,IAAI,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE;gBAC1B,eAAe,EAAE,uBAAuB,CAAC,MAAM;gBAC/C,SAAS,EAAE;oBACR,EAAE,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,UAAU;;;wBAAC,GAAG,EAAE,CAAC,eAAe,EAAC,EAAE,KAAK,EAAE,IAAI,EAAE;oBAC3F,EAAE,OAAO,EAAE,aAAa,EAAE,WAAW,EAAE,UAAU;;;wBAAC,GAAG,EAAE,CAAC,eAAe,EAAC,EAAE,KAAK,EAAE,IAAI,EAAE;iBACzF;;aACH;;;yBAIG,KAAK;2BAEL,KAAK;+BAEL,KAAK;qCAIL,KAAK;sCAKL,KAAK;gCAGL,KAAK;4BAGL,KAAK;2BAKL,KAAK;2BAGL,KAAK;0BAGL,MAAM;wBAEN,MAAM;mBAEN,SAAS,SAAC,MAAM,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE;qBAUnC,KAAK;;;;;;;IA5CN,qCAA4B;;;;;IAE5B,uCAAkC;;;;;IAElC,2CAAmC;;;;;;IAInC,iDAA2E;;;;;;IAK3E,kDAA8C;;;;;IAG9C,4CAAmC;;;;;IAGnC,wCAAqC;;;;;;IAKrC,uCAAsC;;;;;IAGtC,uCAAgD;;;;;IAGhD,sCAAmE;;;;;IAEnE,oCAAuE;;IAEvE,+BAAmD;;IAEnD,qDAAmD;;IACnD,qCAA4B;;;;;IAC5B,iCAAyB;;;;;IACzB,mDAAsD;;;;;IACtD,wCAAgC;;;;;IAChC,kCAA8B;;IA8C9B,oCACC","sourcesContent":["/*\n * © 2017 Stratio Big Data Inc., Sucursal en España.\n *\n * This software is licensed under the Apache License, Version 2.0.\n * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;\n * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n * See the terms of the License for more details.\n *\n * SPDX-License-Identifier: Apache-2.0.\n */\nimport {\n   AfterViewChecked,\n   AfterViewInit,\n   ChangeDetectionStrategy,\n   Component,\n   EventEmitter,\n   forwardRef,\n   Input,\n   OnDestroy,\n   Output,\n   ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, FormControl, NG_VALIDATORS, NG_VALUE_ACCESSOR, NgForm } from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { JSONSchema4 } from 'json-schema';\nimport { StInputError } from '../st-input/st-input.error.model';\nimport { StFormSchema } from './st-form.model';\nimport { FORM_UI_COMPONENT, StFormFieldTranslations } from './st-form-field/st-form-field.interface';\n\n/**\n * @description {Component} [Dynamic form]\n *\n * The form component allows to generate forms dynamically using a JSON schema.\n *\n * @model\n *\n *   [Form Schema] {./st-form.model.ts#StFormSchema}\n *   [Form UI Definition] {./st-form.model.ts#StFormUIDefinition}\n *\n * @example\n *\n * {html}\n *\n * ```\n * <st-form [schema]=\"jsonSchema\" [(ngModel)]=\"model\" #formModel=\"ngModel\">\n * </st-form>\n *\n * ```\n *\n */\n@Component({\n   selector: 'st-form',\n   templateUrl: './st-form.component.html',\n   styleUrls: ['./st-form.component.scss'],\n   host: { class: 'st-form' },\n   changeDetection: ChangeDetectionStrategy.OnPush,\n   providers: [\n      { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => StFormComponent), multi: true },\n      { provide: NG_VALIDATORS, useExisting: forwardRef(() => StFormComponent), multi: true }\n   ]\n})\n\nexport class StFormComponent implements AfterViewInit, AfterViewChecked, ControlValueAccessor, OnDestroy {\n   /** @Input {string} [parentName=] Name of the parent section. By default, it is undefined */\n   @Input() parentName: string;\n   /** @Input {string} [nestingLevel=0] This informs about the nesting level of the form. This input is only used for design purposes */\n   @Input() nestingLevel: number = 0;\n   /** @Input {boolean} [forceValidations=] Boolean to force the field validations */\n   @Input() forceValidations: boolean;\n   /** @Input {string} [disabledSectionMessage='for this instance is disabled.']\n    *  Message displayed when a section is disabled. This is always displayed after the section name\n    */\n   @Input() disabledSectionMessage: string = 'for this instance is disabled.';\n\n   /** @Input {number} [sectionDescriptionLevel=-1] Level of the section to which description is displayed.\n    * By default, section descriptions are not displayed.\n    */\n   @Input() sectionDescriptionLevel: number = -1;\n\n   /** @Input {number} [textFieldMaxWidth=] Maximum width of a field needed to paint a input or textarea */\n   @Input() textFieldMaxWidth: number;\n\n   /** @Input {StInputError} [errorMessages=] Field error translations */\n   @Input() errorMessages: StInputError;\n\n   /** @Input {boolean} [showTooltips=-1] Enable or disable displaying of tooltips\n    * By default, tooltips are displayed\n    */\n   @Input() showTooltips: boolean = true;\n\n   /** @Input {StFormFieldTranslations} [translations=] Field translations */\n   @Input() translations?: StFormFieldTranslations;\n\n   /** @Output {any} [valueChange=] Event emitted when value is changed. This emits the current form value */\n   @Output() valueChange: EventEmitter<any> = new EventEmitter<any>();\n   /** @Output {string} [clickLink=] Event emitted when link is clicked. It returns the field path */\n   @Output() clickLink: EventEmitter<string> = new EventEmitter<string>();\n\n   @ViewChild('form', { static: false }) form: NgForm;\n\n   public showCollapsedSectionFields: boolean = false;\n   public innerValue: any = {};\n   private _value: any = {};\n   private _parentFieldSubscription: Subscription[] = [];\n   private _parentFields: string[];\n   private _schema: StFormSchema;\n\n   /** @Input {StFormSchema [schema=] Form schema needed to generate the form */\n   @Input() get schema(): StFormSchema {\n      return this._schema;\n   }\n\n   set schema(schema: StFormSchema) {\n      this._schema = schema;\n      this._value = {};\n      if (this._schema.dependencies) {\n         this._parentFields = Object.keys(this._schema.dependencies);\n      }\n   }\n\n   ngAfterViewInit(): void {\n      if (!this.forceValidations) {\n         this.form.form.markAsPristine();\n      }\n   }\n\n   ngAfterViewChecked(): void {\n      if (this._parentFields && this.form.control && this.form.control.controls) {\n         for (let i = 0; i < this._parentFields.length; ++i) {\n            let parentField: string = this._parentFields[i];\n            if (!this._parentFieldSubscription[i] && this.form.control.controls[parentField]) {\n\n               this._parentFieldSubscription[i] = this.form.control.controls[this._parentFields[i]].valueChanges.subscribe((value) => {\n                  if (!value) {\n                     let childrenFields: JSONSchema4 | string[] = this.schema.dependencies[parentField];\n                     for (let j = 0; j < childrenFields.length; ++j) {\n                        if (this.form.controls[childrenFields[j]]) {\n                           this._value[childrenFields[j]] = undefined;\n                        }\n                     }\n                  }\n               });\n            }\n         }\n      }\n   }\n\n   // Function to call when the value changes.\n   onChange(_: any): void {\n   }\n\n   onTouched = () => {\n   }\n\n   validate(control: FormControl): any {\n      let errors: any = null;\n      if (this.form) {\n         Object.keys(this.form.controls).forEach((propertyName) => {\n            if (this.form.controls[propertyName] && this.form.controls[propertyName].errors) {\n               if (!errors) {\n                  errors = {};\n               }\n               errors[propertyName] = this.form.controls[propertyName].errors;\n            }\n         });\n\n         this.form.control.setErrors(errors);\n      }\n      return errors;\n   }\n\n   isRequired(propertyName: string): boolean {\n      return propertyName && this.schema.required && this.schema.required.indexOf(propertyName) !== -1;\n   }\n\n   isCollapsedSection(): boolean {\n      return this.schema.type === 'object' && this.schema.ui &&\n         (this.schema.ui.component === FORM_UI_COMPONENT.SHOW_MORE || this.schema.ui.component === FORM_UI_COMPONENT.ACCORDION);\n   }\n\n   iShowMoreSection(): boolean {\n      return this.schema.type === 'object' && this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.SHOW_MORE;\n   }\n\n   isAParentField(propertyName: string): boolean {\n      return this.schema.dependencies && this.schema.dependencies[propertyName] && this.schema.dependencies[propertyName].length > 0;\n   }\n\n   getOptionalButtonLabel(): string {\n      let label: string = 'Additional options';\n\n      if (this.parentName || this.schema.title) {\n         label += ' of ';\n         label += this.parentName || this.schema.title;\n      }\n\n      return label;\n   }\n\n   onChangeOptionalFieldsVisibility(): void {\n      this.showCollapsedSectionFields = !this.showCollapsedSectionFields;\n   }\n\n   getFieldClasses(propertyName: string): any {\n      return {\n         'hidden': this.isCollapsedSection() && !this.showCollapsedSectionFields,\n         'parent-field': this.isAParentField(propertyName) || (this.isASwitchSection() && this.isTheFirstField(propertyName))\n      };\n   }\n\n   fieldHasToBeCreated(propertyName: string): boolean {\n      let createField: boolean = true;\n      let parentField: string = this.getParentField(propertyName);\n      if (((parentField && !this._value[parentField]) || (this.isInADisabledSection() && !this.isTheFirstField(propertyName)))\n         || !this.fulfillDependencyVisibility(propertyName)) {\n         createField = false;\n         this._value[propertyName] = undefined;\n      }\n      return createField;\n   }\n\n   isRelatedField(propertyName: string): boolean {\n      return this._schema.properties[propertyName].ui && this.schema.properties[propertyName].ui.relatedTo\n         && this.schema.properties[propertyName].ui.relatedTo.length > 0;\n   }\n\n   // When value is received from outside\n   writeValue(value: any): void {\n      if (value) {\n         this.onChange(value);\n         this.innerValue = value;\n      }\n   }\n\n   onChangeProperty(value: any, property: string): void {\n      this._value[property] = value;\n      this.valueChange.emit(this._value);\n      setTimeout(() => {\n         this.onChange(this._value);\n      });\n   }\n\n   // Registry the change function to propagate internal model changes\n   registerOnChange(fn: (_: any) => void): void {\n      this.onChange = fn;\n   }\n\n   // Registry the touch function to propagate internal touch events TODO: make this function.\n   registerOnTouched(fn: () => void): void {\n      this.onTouched = fn;\n   }\n\n   // Allows Angular to disable the form.\n   setDisabledState(disable: boolean): void {\n      if (disable) {\n         this.form.control.disable();\n      } else {\n         this.form.control.enable();\n      }\n   }\n\n   ngOnDestroy(): void {\n      if (this._parentFieldSubscription) {\n         for (let i = 0; i < this._parentFieldSubscription.length; ++i) {\n            this._parentFieldSubscription[i].unsubscribe();\n         }\n      }\n   }\n\n   isInADisabledSection(): boolean {\n      if (this.isASwitchSection()) {\n         let sectionEnabler: string = Object.keys(this.schema.properties)[0];\n         return !(this.form && this.form.controls[sectionEnabler] && this.form.controls[sectionEnabler].value);\n      }\n      return false;\n   }\n\n   isASwitchSection(): boolean {\n      return this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.SWITCH;\n   }\n\n   isAnAccordion(): boolean {\n      return this.schema.ui && this.schema.ui.component === FORM_UI_COMPONENT.ACCORDION;\n   }\n\n   onClickTitle(): void {\n      if (this.isAnAccordion()) {\n         this.onChangeOptionalFieldsVisibility();\n      }\n   }\n\n   isTheFirstField(propertyName: string): boolean {\n      return propertyName === Object.keys(this.schema.properties)[0];\n   }\n\n   onClickLink(fieldKey: string): void {\n      this.clickLink.emit(fieldKey);\n   }\n\n   private getParentField(propertyName: string): string {\n      let parentField: string = undefined;\n      if (this.schema.dependencies) {\n         Object.keys(this.schema.dependencies).forEach((key: string) => {\n            if (this.schema.dependencies[key].indexOf(propertyName) !== -1) {\n               parentField = key;\n            }\n         });\n      }\n      return parentField;\n   }\n\n   private fulfillDependencyVisibility(propertyName: string): boolean {\n      let propertySchema: StFormSchema = this.schema.properties[propertyName];\n      let fulfill: boolean = !propertySchema.ui || propertySchema.ui.visible !== false;\n      if (propertySchema.ui && propertySchema.ui.visible) {\n         const keys: string[] = Object.keys(propertySchema.ui.visible);\n         let i = 0;\n         while (fulfill && i < keys.length && propertySchema.ui.visible.hasOwnProperty(keys[i])) {\n            if (this._value[keys[i]] !== propertySchema.ui.visible[keys[i]]) {\n               fulfill = false;\n            }\n            ++i;\n         }\n      }\n      return fulfill;\n   }\n}\n"]}