@davebaol/angular-formio-editor
Version:
Angular component integrating Form.io builder and renderer with a json editor
241 lines • 43.4 kB
JavaScript
import { __decorate } from "tslib";
import { Component, ViewChild, Input } from '@angular/core';
import { BsModalService, BsModalRef } from 'ngx-bootstrap/modal';
import { merge, clone } from './clone-utils';
import { generateFormJsonSchema } from './resource-json-schema/json-schema-generator';
import { loose as formioJsonSchema } from './formio-json-schema';
var defaultJsonEditorOptions = {
enableSort: true,
enableTransform: true,
escapeUnicode: false,
expandAll: false,
history: true,
indentation: 2,
limitDragging: false,
mode: 'view',
modes: ['code', 'tree', 'view'],
schema: formioJsonSchema.schema,
schemaRefs: formioJsonSchema.schemaRefs,
search: true,
sortObjectKeys: false
};
var defaultRendererResourceJsonEditorOptions = merge(defaultJsonEditorOptions, {
mode: 'tree',
modes: ['code', 'tree'],
schema: undefined,
schemaRefs: undefined
});
var defaultRendererSchemaJsonEditorOptions = clone(defaultRendererResourceJsonEditorOptions);
var FormioEditorComponent = /** @class */ (function () {
function FormioEditorComponent(modalService) {
this.modalService = modalService;
this.builderDisplayChanged = false;
this.jsonEditorChanged = false;
// tslint:disable-next-line:variable-name
this._jsonEditorErrors = [];
this.jsonEditorWarningCounter = 0;
}
Object.defineProperty(FormioEditorComponent.prototype, "options", {
get: function () { return this._options; },
set: function (options) { this.setOptions(options); },
enumerable: true,
configurable: true
});
Object.defineProperty(FormioEditorComponent.prototype, "activeTab", {
get: function () { return this._activeTab; },
set: function (tab) {
this._activeTab = tab;
if (tab === 'renderer') {
this.submissionPanel = false; // Disable submission panel when the renderer tab becomes active
}
},
enumerable: true,
configurable: true
});
Object.defineProperty(FormioEditorComponent.prototype, "jsonEditorErrors", {
get: function () {
return this._jsonEditorErrors;
},
set: function (errors) {
var _this = this;
this._jsonEditorErrors = errors;
this.jsonEditorWarningCounter = 0;
errors.forEach(function (error) {
if (error.type === 'validation') {
_this.jsonEditorWarningCounter++;
}
});
},
enumerable: true,
configurable: true
});
FormioEditorComponent.prototype.ngOnInit = function () {
var _this = this;
if (!this.options) {
this.setOptions(); // Set default options
}
this.activeTab = ['builder', 'json', 'renderer']
.find(function (t) { var _a; return _this.options && ((_a = _this.options[t]) === null || _a === void 0 ? void 0 : _a.defaultTab) ? t : undefined; }) || 'builder';
if (this.reset) {
this.resetSubscription = this.reset.subscribe(function () { return _this.resetFormBuilder(); });
}
};
FormioEditorComponent.prototype.ngAfterViewInit = function () {
this.refreshJsonEditor();
};
FormioEditorComponent.prototype.ngOnDestroy = function () {
if (this.resetSubscription) {
this.resetSubscription.unsubscribe();
}
};
FormioEditorComponent.prototype.setOptions = function (options) {
if (options === void 0) { options = {}; }
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;
this._options = options;
this.jsonEditorOptions = merge(defaultJsonEditorOptions, (_b = (_a = options === null || options === void 0 ? void 0 : options.json) === null || _a === void 0 ? void 0 : _a.input) === null || _b === void 0 ? void 0 : _b.options);
this.rendererResourceJsonEditorOptions = merge(defaultRendererResourceJsonEditorOptions, (_f = (_e = (_d = (_c = options === null || options === void 0 ? void 0 : options.renderer) === null || _c === void 0 ? void 0 : _c.submissionPanel) === null || _d === void 0 ? void 0 : _d.resourceJsonEditor) === null || _e === void 0 ? void 0 : _e.input) === null || _f === void 0 ? void 0 : _f.options);
this.rendererSchemaJsonEditorOptions = merge(defaultRendererSchemaJsonEditorOptions, (_k = (_j = (_h = (_g = options === null || options === void 0 ? void 0 : options.renderer) === null || _g === void 0 ? void 0 : _g.submissionPanel) === null || _h === void 0 ? void 0 : _h.schemaJsonEditor) === null || _j === void 0 ? void 0 : _j.input) === null || _k === void 0 ? void 0 : _k.options);
this.fullSubmission = (_m = (_l = options === null || options === void 0 ? void 0 : options.renderer) === null || _l === void 0 ? void 0 : _l.submissionPanel) === null || _m === void 0 ? void 0 : _m.fullSubmission;
};
//
// Form Builder Tab
//
FormioEditorComponent.prototype.resetFormBuilder = function (fromJsonEditor) {
var _this = this;
if (fromJsonEditor === void 0) { fromJsonEditor = false; }
console.log('resetFormBuilder');
// Here we have to reset builder component through *ngIf="!builderDisplayChanged"
// See https://github.com/formio/angular-formio/issues/172#issuecomment-401876490
this.builderDisplayChanged = true;
setTimeout(function () {
_this.builderDisplayChanged = false;
if (!fromJsonEditor) {
_this.refreshJsonEditor(true);
}
_this.resetFormRendererIfActive();
});
};
FormioEditorComponent.prototype.onBuilderDiplayChange = function (event) {
console.log('onBuilderDiplayChange');
this.resetFormBuilder();
};
FormioEditorComponent.prototype.onBuilderChange = function (event) {
console.log('onBuilderChange: event', event);
this.refreshJsonEditor(true);
};
//
// JSON Tab
//
FormioEditorComponent.prototype.onJsonEditorError = function (errors) {
console.log('onJsonEditorError: found', errors.length, 'validation errors:');
this.jsonEditorErrors = errors;
};
FormioEditorComponent.prototype.onJsonEditorChange = function (event) {
console.log('onJsonEditorChange');
this.jsonEditorChanged = true;
};
FormioEditorComponent.prototype.onJsonEditorApply = function (template) {
console.log('Errors: ', this.jsonEditorErrors.length - this.jsonEditorWarningCounter, 'Warnings: ', this.jsonEditorWarningCounter);
if (this.jsonEditorWarningCounter === 0) {
this.jsonEditorApplyChanges();
}
else {
this.modalRef = this.modalService.show(template);
}
};
FormioEditorComponent.prototype.jsonEditorApplyChanges = function () {
var _this = this;
console.log('jsonEditorApplyChanges');
this.jsonEditorChanged = false;
// Remove all properties from this form
// then copy the properties of the edited json to this form
// and reset the builder
Object.getOwnPropertyNames(this.form).forEach(function (p) { return delete _this.form[p]; });
Object.assign(this.form, this.jsonEditor.get());
this.resetFormBuilder(true);
};
FormioEditorComponent.prototype.jsonEditorDiscardChanges = function () {
console.log('jsonEditorDiscardChanges');
this.refreshJsonEditor();
};
FormioEditorComponent.prototype.refreshJsonEditor = function (forceReset) {
if (forceReset === void 0) { forceReset = false; }
console.log('refreshJsonEditor');
if (forceReset) {
this.jsonEditor.reset(true);
}
// Here we use update instead of set to preserve the editor status
this.jsonEditor.update(this.form);
this.jsonEditorChanged = false;
};
//
// Form Renderer Tab
//
FormioEditorComponent.prototype.resetFormRendererIfActive = function () {
var _this = this;
console.log('resetFormRenderer');
// Here we recreate the renderer component through *ngIf="activeTab='renderer'"
// by changing the active tab and then restoring it.
// Although this is a rather dirty hack it is hardly noticeable to the eye :)
if (this.activeTab === 'renderer') {
this.activeTab = 'builder';
setTimeout(function () { _this.activeTab = 'renderer'; });
}
};
FormioEditorComponent.prototype.showSubmissionPanel = function (submission) {
var _this = this;
var _a, _b;
this.submissionPanel = !((_b = (_a = this.options.renderer) === null || _a === void 0 ? void 0 : _a.submissionPanel) === null || _b === void 0 ? void 0 : _b.disabled);
if (this.submissionPanel) {
if (submission) {
this.submission = submission;
}
setTimeout(function () {
var schema = generateFormJsonSchema(_this.form);
if (_this.fullSubmission) {
schema = { type: 'object', properties: { data: schema } };
}
_this.rendererResourceJsonEditor.setSchema(undefined);
_this.rendererResourceJsonEditor.set(_this.fullSubmission ? _this.submission : _this.submission.data);
_this.rendererSchemaJsonEditor.set(schema);
_this.rendererResourceJsonEditor.setSchema(schema);
});
}
};
FormioEditorComponent.prototype.applyResourceJsonSchema = function () {
var schema = this.rendererSchemaJsonEditor.get();
this.rendererResourceJsonEditor.setSchema(schema);
};
FormioEditorComponent.ctorParameters = function () { return [
{ type: BsModalService }
]; };
__decorate([
Input()
], FormioEditorComponent.prototype, "form", void 0);
__decorate([
Input()
], FormioEditorComponent.prototype, "reset", void 0);
__decorate([
Input()
], FormioEditorComponent.prototype, "options", null);
__decorate([
ViewChild('jsoneditor', { static: true })
], FormioEditorComponent.prototype, "jsonEditor", void 0);
__decorate([
ViewChild('renderer_resource_jsoneditor', { static: false })
], FormioEditorComponent.prototype, "rendererResourceJsonEditor", void 0);
__decorate([
ViewChild('renderer_schema_jsoneditor', { static: false })
], FormioEditorComponent.prototype, "rendererSchemaJsonEditor", void 0);
FormioEditorComponent = __decorate([
Component({
// tslint:disable-next-line:component-selector
selector: 'formio-editor',
template: "<!-- Modal -->\r\n<ng-template #saveModal class=\"modal fade\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"saveModalLabel\" aria-hidden=\"true\">\r\n\r\n <div class=\"modal-header\">\r\n <h5 class=\"modal-title\" id=\"exampleModalLabel\">Validation warning</h5>\r\n <button type=\"button\" class=\"close pull-right\" aria-label=\"Close\" (click)=\"modalRef.hide()\">\r\n <span aria-hidden=\"true\">×</span>\r\n </button>\r\n </div>\r\n <div class=\"modal-body\">\r\n Looks like the json is not a valid form.\r\n Do you want to apply changes anyways?\r\n </div>\r\n <div class=\"modal-footer\">\r\n <button type=\"button\" class=\"btn btn-secondary\" data-dismiss=\"modal\" (click)=\"modalRef.hide();\">NO</button>\r\n <button type=\"button\" class=\"btn btn-primary\" data-dismiss=\"modal\" (click)=\"modalRef.hide(); jsonEditorApplyChanges()\">YES</button>\r\n </div>\r\n</ng-template>\r\n<ul class=\"nav nav-tabs\" id=\"myTab\" role=\"tablist\">\r\n <li class=\"nav-item\" *ngIf=\"options?.builder?.hideTab !== true\">\r\n <span class=\"nav-link\" [ngClass]=\"{'active':activeTab=='builder'}\" (click)=\"activeTab='builder'\"\r\n id=\"builder-tab\" data-toggle=\"tab\" data-target=\"#builder\" role=\"tab\" aria-controls=\"builder\" aria-selected=\"true\">\r\n Builder\r\n </span>\r\n </li>\r\n <li class=\"nav-item\" *ngIf=\"options?.json?.hideTab !== true\">\r\n <span class=\"nav-link\" [ngClass]=\"{'active':activeTab=='json'}\" (click)=\"activeTab='json'\"\r\n id=\"json-editor-tab\" data-toggle=\"tab\" data-target=\"#json-editor\" role=\"tab\" aria-controls=\"json-editor\"\r\n aria-selected=\"false\">\r\n Json\r\n </span>\r\n </li>\r\n <li class=\"nav-item\" *ngIf=\"options?.renderer?.hideTab !== true\">\r\n <span class=\"nav-link\" [ngClass]=\"{'active':activeTab=='renderer'}\" (click)=\"activeTab='renderer'\"\r\n id=\"renderer-tab\" data-toggle=\"tab\" data-target=\"#renderer\" role=\"tab\" aria-controls=\"renderer\" aria-selected=\"false\">\r\n Renderer\r\n </span>\r\n </li>\r\n</ul>\r\n<div class=\"tab-content\" id=\"myTabContent\">\r\n <div class=\"tab-pane fade\" [ngClass]=\"{'show active':activeTab=='builder'}\" id=\"builder\" role=\"tabpanel\" aria-labelledby=\"builder-tab\">\r\n <div class=\"m-2\">\r\n <div *ngIf=\"options?.builder?.hideDisplaySelect !== true\">\r\n <label for=\"form-display-select\" class=\"mx-2\">Display as: </label>\r\n <select id=\"form-display-select\" class=\"form-control my-2\" style=\"display: inline-block; width: 150px;\"\r\n [(ngModel)]=\"form.display\" (ngModelChange)=\"onBuilderDiplayChange($event)\">\r\n <option value=\"form\">Form</option>\r\n <option value=\"wizard\">Wizard</option>\r\n </select>\r\n </div>\r\n <form-builder *ngIf=\"!builderDisplayChanged\" [form]=\"form\"\r\n [options]=\"options?.builder?.input?.options\"\r\n [formbuilder]=\"options?.builder?.input?.formbuilder\"\r\n [noeval]=\"options?.builder?.input?.noeval\"\r\n [refresh]=\"options?.builder?.input?.refresh\"\r\n (change)=\"onBuilderChange($event); options?.builder?.output?.change ? options.builder.output.change($event) : undefined\"\r\n ></form-builder>\r\n </div>\r\n </div>\r\n <div class=\"tab-pane fade\" [ngClass]=\"{'show active':activeTab=='json'}\" id=\"json-editor\" role=\"tabpanel\" aria-labelledby=\"json-editor-tab\">\r\n <div class=\"m-2 well\">\r\n <div class=\"row mx-0\" *ngIf=\"!options?.json?.changePanelLocations || options.json.changePanelLocations?.includes('top')\">\r\n <div class=\"m-2 p-0\">\r\n <button type=\"button\" class=\"btn btn-success mr-4\" (click)=\"onJsonEditorApply(saveModal)\"\r\n [disabled]=\"!jsonEditorChanged || jsonEditorErrors.length !== jsonEditorWarningCounter\">\r\n Apply\r\n </button>\r\n <button type=\"button\" class=\"btn btn-secondary\" (click)=\"jsonEditorDiscardChanges()\">Discard</button>\r\n </div>\r\n <alert class=\"col-6 mx-4 m-2 p-0\" type=\"danger\" [isOpen]=\"jsonEditorErrors.length !== jsonEditorWarningCounter\" >\r\n Cannot apply changes. Json is not well-formed.\r\n </alert>\r\n </div>\r\n <json-editor #jsoneditor \r\n [options]=\"jsonEditorOptions\"\r\n (dataChange)=\"onJsonEditorChange($event); options?.json?.output?.dataChange ? options.json.output.dataChange($event) : undefined\"\r\n (dataError)=\"onJsonEditorError($event); options?.json?.output?.dataError ? options.json.output.dataError($event) : undefined\"\r\n ></json-editor>\r\n <div class=\"row mx-0\" *ngIf=\"!options?.json?.changePanelLocations || options.json.changePanelLocations?.includes('bottom')\">\r\n <div class=\"m-2 p-0\">\r\n <button type=\"button\" class=\"btn btn-success mr-4\" (click)=\"onJsonEditorApply(saveModal)\"\r\n [disabled]=\"!jsonEditorChanged || jsonEditorErrors.length !== jsonEditorWarningCounter\">\r\n Apply\r\n </button>\r\n <button type=\"button\" class=\"btn btn-secondary\" (click)=\"jsonEditorDiscardChanges()\">Discard</button>\r\n </div>\r\n <alert class=\"col-6 mx-4 my-2 p-1 align-middle\" type=\"danger\" [isOpen]=\"jsonEditorErrors.length !== jsonEditorWarningCounter\" >\r\n Json is not well-formed. You won't be able to apply changes!\r\n </alert>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"tab-pane fade\" [ngClass]=\"{'show active':activeTab=='renderer'}\" id=\"renderer\" role=\"tabpanel\" aria-labelledby=\"renderer-tab\">\r\n <div class=\"m-2\">\r\n <formio *ngIf=\"activeTab=='renderer'\" [form]=\"form\"\r\n [submission]=\"options?.renderer?.input?.submission\"\r\n [src]=\"options?.renderer?.input?.src\"\r\n [url]=\"options?.renderer?.input?.url\"\r\n [service]=\"options?.renderer?.input?.service\"\r\n [options]=\"options?.renderer?.input?.options\"\r\n [noeval]=\"options?.renderer?.input?.noeval\"\r\n [formioOptions]=\"options?.renderer?.input?.formioOptions\"\r\n [renderOptions]=\"options?.renderer?.input?.renderOptions\"\r\n [readOnly]=\"options?.renderer?.input?.readOnly\"\r\n [viewOnly]=\"options?.renderer?.input?.viewOnly\"\r\n [hideComponents]=\"options?.renderer?.input?.hideComponents\"\r\n [refresh]=\"options?.renderer?.input?.refresh\"\r\n [success]=\"options?.renderer?.input?.success\"\r\n [language]=\"options?.renderer?.input?.language\"\r\n [hooks]=\"options?.renderer?.input?.hooks\"\r\n [renderer]=\"options?.renderer?.input?.renderer\"\r\n (render)=\"options?.renderer?.output?.render ? options.renderer.output.render($event) : undefined\"\r\n (customEvent)=\"options?.renderer?.output?.customEvent ? options.renderer.output.customEvent($event) : undefined\"\r\n (submit)=\"options?.renderer?.output?.submit ? options.renderer.output.submit($event) : undefined\"\r\n (prevPage)=\"options?.renderer?.output?.prevPage ? options.renderer.output.prevPage($event) : undefined\"\r\n (nextPage)=\"options?.renderer?.output?.nextPage ? options.renderer.output.nextPage($event) : undefined\"\r\n (beforeSubmit)=\"showSubmissionPanel($event); options?.renderer?.output?.beforeSubmit ? options.renderer.output.beforeSubmit($event) : undefined\"\r\n (change)=\"options?.renderer?.output?.change ? options.renderer.output.change($event) : undefined\"\r\n (invalid)=\"options?.renderer?.output?.invalid ? options.renderer.output.invalid($event) : undefined\"\r\n (errorChange)=\"options?.renderer?.output?.errorChange ? options.renderer.output.errorChange($event) : undefined\"\r\n (formLoad)=\"options?.renderer?.output?.formLoad ? options.renderer.output.formLoad($event) : undefined\"\r\n (submissionLoad)=\"options?.renderer?.output?.submissionLoad ? options.renderer.output.submissionLoad($event) : undefined\"\r\n (ready)=\"options?.renderer?.output?.ready ? options.renderer.output.ready($event) : undefined\"\r\n ></formio>\r\n </div>\r\n <div *ngIf=\"submissionPanel\" class=\"m-2 border-top border-primary\">\r\n <div class=\"row mx-0\">\r\n <div class=\"p-2\" [ngClass]=\"{'col-6':showResourceSchema,'col-12':!showResourceSchema}\">\r\n <p class=\"font-weight-bold text-center mb-1\">\r\n Summission (\r\n <label for=\"fullSubmitCheckbox\" class=\"my-0\">full</label>\r\n <input id=\"fullSubmitCheckbox\" class=\"mx-1\" type=\"checkbox\" [(ngModel)]=\"fullSubmission\" (change)=\"showSubmissionPanel(submission)\">\r\n )\r\n </p>\r\n <json-editor #renderer_resource_jsoneditor\r\n [options]=\"rendererResourceJsonEditorOptions\"\r\n (dataChange)=\"options?.renderer?.submissionPanel?.resourceJsonEditor?.output?.dataChange ? options.renderer.submissionPanel.resourceJsonEditor.output.dataChange($event) : undefined\"\r\n (dataError)=\"options?.renderer?.submissionPanel?.resourceJsonEditor?.output?.dataError ? options.renderer.submissionPanel.resourceJsonEditor.output.dataError($event) : undefined\"\r\n ></json-editor>\r\n </div>\r\n <div class=\"p-2 col-6\" [hidden]=!showResourceSchema>\r\n <p class=\"font-weight-bold text-center mb-1\">Json Schema Validator</p>\r\n <json-editor #renderer_schema_jsoneditor\r\n [options]=\"rendererSchemaJsonEditorOptions\"\r\n (dataChange)=\"options?.renderer?.submissionPanel?.schemaJsonEditor?.output?.dataChange ? options.renderer.submissionPanel.schemaJsonEditor.output.dataChange($event) : undefined\"\r\n (dataError)=\"options?.renderer?.submissionPanel?.schemaJsonEditor?.output?.dataError ? options.renderer.submissionPanel.schemaJsonEditor.output.dataError($event) : undefined\"\r\n ></json-editor>\r\n </div>\r\n </div>\r\n <button class=\"mx-3 btn btn-primary\" *ngIf=\"options?.renderer?.submissionPanel?.schemaJsonEditor?.enabled\" (click)=\"showResourceSchema = !showResourceSchema\">{{showResourceSchema? 'Hide Schema' : 'Show Schema'}}</button>\r\n <button class=\"mx-2 btn btn-primary\" *ngIf=\"options?.renderer?.submissionPanel?.schemaJsonEditor?.enabled && showResourceSchema\" (click)=\"applyResourceJsonSchema()\">Apply Schema</button>\r\n <button class=\"mx-2 btn btn-primary\" *ngIf=\"options?.renderer?.submissionPanel?.schemaJsonEditor?.enabled && showResourceSchema\" (click)=\"showSubmissionPanel(submission)\">Regenerate Schema</button>\r\n </div>\r\n </div>\r\n</div>\r\n",
styles: [""]
})
], FormioEditorComponent);
return FormioEditorComponent;
}());
export { FormioEditorComponent };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"formio-editor.component.js","sourceRoot":"ng://@davebaol/angular-formio-editor/","sources":["lib/formio-editor.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, OnInit, ViewChild, Input, TemplateRef, OnDestroy} from '@angular/core';\r\nimport { Observable, Subscription } from 'rxjs';\r\nimport { BsModalService, BsModalRef } from 'ngx-bootstrap/modal';\r\nimport { FormioEditorOptions, FormioEditorTab } from './formio-editor-options';\r\nimport { JsonEditorComponent } from './json-editor/json-editor.component';\r\nimport { JsonEditorValidationError, JsonEditorOptions } from './json-editor/json-editor-shapes';\r\nimport { merge, clone } from './clone-utils';\r\nimport { generateFormJsonSchema } from './resource-json-schema/json-schema-generator';\r\nimport { loose as formioJsonSchema } from './formio-json-schema';\r\n\r\nconst defaultJsonEditorOptions: JsonEditorOptions = {\r\n  enableSort: true,\r\n  enableTransform: true,\r\n  escapeUnicode: false,\r\n  expandAll: false,\r\n  history: true,\r\n  indentation: 2,\r\n  limitDragging: false,\r\n  mode: 'view', // set default mode\r\n  modes: ['code', 'tree', 'view'], // set allowed modes\r\n  schema: formioJsonSchema.schema,\r\n  schemaRefs: formioJsonSchema.schemaRefs,\r\n  search: true,\r\n  sortObjectKeys: false\r\n};\r\n\r\nconst defaultRendererResourceJsonEditorOptions: JsonEditorOptions = merge(defaultJsonEditorOptions, {\r\n  mode: 'tree', // set default mode\r\n  modes: ['code', 'tree'], // set allowed modes\r\n  schema: undefined,\r\n  schemaRefs: undefined\r\n});\r\n\r\nconst defaultRendererSchemaJsonEditorOptions: JsonEditorOptions = clone(defaultRendererResourceJsonEditorOptions);\r\n\r\n@Component({\r\n  // tslint:disable-next-line:component-selector\r\n  selector: 'formio-editor',\r\n  templateUrl: './formio-editor.component.html',\r\n  styleUrls: ['./formio-editor.component.css']\r\n})\r\nexport class FormioEditorComponent implements OnInit, AfterViewInit, OnDestroy  {\r\n  @Input() form: any;\r\n  builderDisplayChanged = false;\r\n\r\n  @Input() reset?: Observable<void>;\r\n  private resetSubscription: Subscription;\r\n\r\n  // tslint:disable-next-line:variable-name\r\n  private _options: FormioEditorOptions;\r\n  get options() { return this._options; }\r\n  @Input() set options(options: FormioEditorOptions) { this.setOptions(options); }\r\n  jsonEditorOptions: JsonEditorOptions;\r\n  jsonEditorChanged = false;\r\n  @ViewChild('jsoneditor', {static: true}) jsonEditor: JsonEditorComponent;\r\n\r\n  @ViewChild('renderer_resource_jsoneditor', {static: false}) rendererResourceJsonEditor: JsonEditorComponent;\r\n  @ViewChild('renderer_schema_jsoneditor', {static: false}) rendererSchemaJsonEditor: JsonEditorComponent;\r\n  rendererResourceJsonEditorOptions: JsonEditorOptions;\r\n  rendererSchemaJsonEditorOptions: JsonEditorOptions;\r\n  submissionPanel: boolean;\r\n  showResourceSchema: boolean;\r\n  submission: any;\r\n  fullSubmission: boolean;\r\n\r\n  // tslint:disable-next-line:variable-name\r\n  private _activeTab: FormioEditorTab;\r\n  get activeTab() { return this._activeTab; }\r\n  set activeTab(tab: FormioEditorTab) {\r\n    this._activeTab = tab;\r\n    if (tab === 'renderer') {\r\n      this.submissionPanel = false; // Disable submission panel when the renderer tab becomes active\r\n    }\r\n  }\r\n\r\n  modalRef: BsModalRef;\r\n\r\n  // tslint:disable-next-line:variable-name\r\n  private _jsonEditorErrors: JsonEditorValidationError[] = [];\r\n  get jsonEditorErrors() {\r\n    return this._jsonEditorErrors;\r\n  }\r\n  set jsonEditorErrors(errors: JsonEditorValidationError[]) {\r\n    this._jsonEditorErrors = errors;\r\n    this.jsonEditorWarningCounter = 0;\r\n    errors.forEach((error) => {\r\n      if (error.type === 'validation') {\r\n        this.jsonEditorWarningCounter++;\r\n      }\r\n    });\r\n  }\r\n\r\n  jsonEditorWarningCounter = 0;\r\n\r\n  constructor(private modalService: BsModalService) {\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (!this.options) {\r\n      this.setOptions(); // Set default options\r\n    }\r\n\r\n    this.activeTab = (['builder', 'json', 'renderer'] as FormioEditorTab[])\r\n          .find(t => this.options && this.options[t]?.defaultTab ? t : undefined) || 'builder';\r\n\r\n    if (this.reset) {\r\n      this.resetSubscription = this.reset.subscribe(() => this.resetFormBuilder());\r\n    }\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n    this.refreshJsonEditor();\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    if (this.resetSubscription) {\r\n      this.resetSubscription.unsubscribe();\r\n    }\r\n  }\r\n\r\n  private setOptions(options: FormioEditorOptions = {}) {\r\n    this._options = options;\r\n    this.jsonEditorOptions = merge(\r\n      defaultJsonEditorOptions,\r\n      options?.json?.input?.options\r\n    );\r\n    this.rendererResourceJsonEditorOptions = merge(\r\n      defaultRendererResourceJsonEditorOptions,\r\n      options?.renderer?.submissionPanel?.resourceJsonEditor?.input?.options\r\n    );\r\n    this.rendererSchemaJsonEditorOptions = merge(\r\n      defaultRendererSchemaJsonEditorOptions,\r\n      options?.renderer?.submissionPanel?.schemaJsonEditor?.input?.options\r\n    );\r\n    this.fullSubmission = options?.renderer?.submissionPanel?.fullSubmission;\r\n  }\r\n\r\n  //\r\n  // Form Builder Tab\r\n  //\r\n\r\n  resetFormBuilder(fromJsonEditor: boolean = false) {\r\n    console.log('resetFormBuilder');\r\n    // Here we have to reset builder component through *ngIf=\"!builderDisplayChanged\"\r\n    // See https://github.com/formio/angular-formio/issues/172#issuecomment-401876490\r\n    this.builderDisplayChanged = true;\r\n    setTimeout(() => {\r\n      this.builderDisplayChanged = false;\r\n      if (!fromJsonEditor) {\r\n        this.refreshJsonEditor(true);\r\n      }\r\n      this.resetFormRendererIfActive();\r\n    });\r\n  }\r\n\r\n  onBuilderDiplayChange(event) {\r\n    console.log('onBuilderDiplayChange');\r\n    this.resetFormBuilder();\r\n  }\r\n\r\n  onBuilderChange(event) {\r\n    console.log('onBuilderChange: event', event);\r\n    this.refreshJsonEditor(true);\r\n  }\r\n\r\n  //\r\n  // JSON Tab\r\n  //\r\n\r\n  onJsonEditorError(errors: any[]) {\r\n    console.log('onJsonEditorError: found', errors.length, 'validation errors:');\r\n    this.jsonEditorErrors = errors;\r\n  }\r\n\r\n  onJsonEditorChange(event) {\r\n    console.log('onJsonEditorChange');\r\n    this.jsonEditorChanged = true;\r\n  }\r\n\r\n  onJsonEditorApply(template: TemplateRef<any>) {\r\n    console.log('Errors: ', this.jsonEditorErrors.length - this.jsonEditorWarningCounter, 'Warnings: ', this.jsonEditorWarningCounter);\r\n    if (this.jsonEditorWarningCounter === 0) {\r\n      this.jsonEditorApplyChanges();\r\n    } else {\r\n      this.modalRef = this.modalService.show(template);\r\n    }\r\n  }\r\n\r\n  jsonEditorApplyChanges() {\r\n    console.log('jsonEditorApplyChanges');\r\n    this.jsonEditorChanged = false;\r\n    // Remove all properties from this form\r\n    // then copy the properties of the edited json to this form\r\n    // and reset the builder\r\n    Object.getOwnPropertyNames(this.form).forEach(p => delete this.form[p]);\r\n    Object.assign(this.form, this.jsonEditor.get());\r\n    this.resetFormBuilder(true);\r\n  }\r\n\r\n  jsonEditorDiscardChanges() {\r\n    console.log('jsonEditorDiscardChanges');\r\n    this.refreshJsonEditor();\r\n  }\r\n\r\n  refreshJsonEditor(forceReset: boolean = false) {\r\n    console.log('refreshJsonEditor');\r\n    if (forceReset) {\r\n      this.jsonEditor.reset(true);\r\n    }\r\n    // Here we use update instead of set to preserve the editor status\r\n    this.jsonEditor.update(this.form);\r\n    this.jsonEditorChanged = false;\r\n  }\r\n\r\n  //\r\n  // Form Renderer Tab\r\n  //\r\n\r\n  resetFormRendererIfActive() {\r\n    console.log('resetFormRenderer');\r\n    // Here we recreate the renderer component through *ngIf=\"activeTab='renderer'\"\r\n    // by changing the active tab and then restoring it.\r\n    // Although this is a rather dirty hack it is hardly noticeable to the eye :)\r\n    if (this.activeTab === 'renderer') {\r\n      this.activeTab = 'builder';\r\n      setTimeout(() => { this.activeTab = 'renderer'; });\r\n    }\r\n  }\r\n\r\n  showSubmissionPanel(submission: any) {\r\n    this.submissionPanel = !this.options.renderer?.submissionPanel?.disabled;\r\n    if (this.submissionPanel) {\r\n      if (submission) {\r\n        this.submission = submission;\r\n      }\r\n      setTimeout(() => {\r\n        let schema = generateFormJsonSchema(this.form);\r\n        if (this.fullSubmission) {\r\n          schema = { type: 'object', properties: { data: schema } };\r\n        }\r\n        this.rendererResourceJsonEditor.setSchema(undefined);\r\n        this.rendererResourceJsonEditor.set(this.fullSubmission ? this.submission : this.submission.data);\r\n        this.rendererSchemaJsonEditor.set(schema as JSON);\r\n        this.rendererResourceJsonEditor.setSchema(schema);\r\n      });\r\n    }\r\n  }\r\n  applyResourceJsonSchema() {\r\n    const schema = this.rendererSchemaJsonEditor.get();\r\n    this.rendererResourceJsonEditor.setSchema(schema);\r\n  }\r\n}\r\n"]}