ngx-schema-forms
Version:
New features: - Ajv schema validator. - Angular forms compatible: Property tree is created using FormGroup, FormArray and FormControl classes. - Array now properly loads initial data from model. - WidgetTyep: WidgetRegistry now supports WidgetType, now wo
227 lines (226 loc) • 16.5 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
import * as tslib_1 from "tslib";
import { EventEmitter } from '@angular/core';
import { BehaviorSubject, combineLatest } from 'rxjs';
import { map, startWith } from 'rxjs/operators';
import { FormPropertyErrors } from './form-property-errors';
/** @typedef {?} */
var Constructor;
export { Constructor };
/**
* @template T
* @param {?} Base
* @return {?}
*/
export function ControlProperty(Base) {
/**
* @abstract
*/
var /**
* @abstract
*/
Property = /** @class */ (function (_super) {
tslib_1.__extends(Property, _super);
function Property() {
var args = [];
for (var _i = 0; _i < arguments.length; _i++) {
args[_i] = arguments[_i];
}
var _this = _super.apply(this, tslib_1.__spread(args)) || this;
_this.nonEmptyValueChanges = new EventEmitter();
_this.visibilityChanges = new BehaviorSubject(true);
_this._visible = true;
return _this;
}
Object.defineProperty(Property.prototype, "id", {
get: /**
* @return {?}
*/
function () {
return this.path.toLowerCase().slice(1).replace(/\//g, '-');
},
enumerable: true,
configurable: true
});
Object.defineProperty(Property.prototype, "isRoot", {
get: /**
* @return {?}
*/
function () {
return this === this.root;
},
enumerable: true,
configurable: true
});
Object.defineProperty(Property.prototype, "name", {
get: /**
* @return {?}
*/
function () {
return this.path.split('/').pop();
},
enumerable: true,
configurable: true
});
Object.defineProperty(Property.prototype, "visible", {
get: /**
* @return {?}
*/
function () {
return this._visible;
},
enumerable: true,
configurable: true
});
/**
* @return {?}
*/
Property.prototype.getErrors = /**
* @return {?}
*/
function () {
/** @type {?} */
var errors = this.errors;
if (!errors) {
return null;
}
return new FormPropertyErrors((_a = {}, _a[this.path] = errors, _a));
var _a;
};
/**
* @param {?} visible
* @param {?=} opts
* @return {?}
*/
Property.prototype.setVisible = /**
* @param {?} visible
* @param {?=} opts
* @return {?}
*/
function (visible, opts) {
if (opts === void 0) { opts = { disable: false }; }
this._visible = visible;
if (opts.disable) {
if (this.visible) {
this.enable();
}
else {
this.disable();
}
}
this.visibilityChanges.next(this.visible);
};
// visible if AT LEAST ONE of the properties it depends on is visible
// AND has a value in the list
/**
* @return {?}
*/
Property.prototype.bindVisibility = /**
* @return {?}
*/
function () {
var _this = this;
/** @type {?} */
var visibleIf = this.schema["visibleIf"];
if (visibleIf === undefined) {
return;
}
/** @type {?} */
var paths = Object.keys(visibleIf);
if (typeof visibleIf === 'object' && paths.length === 0) {
this.setVisible(false);
return;
}
/** @type {?} */
var observables = [];
var _loop_1 = function (path) {
if (!visibleIf.hasOwnProperty(path)) {
return "continue";
}
/** @type {?} */
var property = this_1.root.get(path);
if (!property) {
console.warn("Couldn't find property " + path + " for visibility check of " + this_1.path);
return "continue";
}
/** @type {?} */
var values = visibleIf[path];
/** @type {?} */
var observable = property.valueChanges.pipe(startWith(values.includes(property.value)), map(function (value) {
return values.includes('$ANY$') || values.includes(value);
}));
observables.push(observable);
};
var this_1 = this;
try {
for (var paths_1 = tslib_1.__values(paths), paths_1_1 = paths_1.next(); !paths_1_1.done; paths_1_1 = paths_1.next()) {
var path = paths_1_1.value;
_loop_1(path);
}
}
catch (e_1_1) { e_1 = { error: e_1_1 }; }
finally {
try {
if (paths_1_1 && !paths_1_1.done && (_a = paths_1.return)) _a.call(paths_1);
}
finally { if (e_1) throw e_1.error; }
}
// TODO unsubscribe
combineLatest(observables)
.subscribe(function (values) {
_this.setVisible(values.includes(true));
});
var e_1, _a;
};
/**
* @param {?} path
* @return {?}
*/
Property.prototype.get = /**
* @param {?} path
* @return {?}
*/
function (path) {
if (typeof path === 'string' && path.includes('/')) {
path = this.normalizePath(path);
}
return _super.prototype.get.call(this, path);
};
/**
* @param {?} path
* @return {?}
*/
Property.prototype.normalizePath = /**
* @param {?} path
* @return {?}
*/
function (path) {
if (path[0] === '/') {
path = path.slice(1);
}
return path.split('/');
};
return Property;
}(Base));
if (false) {
/** @type {?} */
Property.prototype.widgetInstance;
/** @type {?} */
Property.prototype.nonEmptyValue;
/** @type {?} */
Property.prototype.nonEmptyValueChanges;
/** @type {?} */
Property.prototype.visibilityChanges;
/** @type {?} */
Property.prototype.path;
/** @type {?} */
Property.prototype.schema;
/** @type {?} */
Property.prototype._visible;
}
return Property;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"control-property.js","sourceRoot":"ng://ngx-schema-forms/","sources":["lib/model/control-property.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from '@angular/core';\nimport {\n  AbstractControl,\n  ValidationErrors,\n  FormControl,\n  FormArray,\n  FormGroup\n} from '@angular/forms';\nimport { BehaviorSubject, combineLatest } from 'rxjs';\nimport { map, startWith  } from 'rxjs/operators';\n\nimport { FormProperty } from './form-property';\nimport { FormPropertyErrors } from './form-property-errors';\nimport { Schema, SchemaValidatorFn } from '../schema';\n\n\nexport type Constructor<T> = new (...args: any[]) => T;\n\n\nexport function ControlProperty<T extends Constructor<AbstractControl>>(\n  Base: T\n): T & Constructor<FormProperty> {\n\n  abstract class Property extends Base implements FormProperty {\n\n    widgetInstance: any;\n\n    nonEmptyValue: any;\n    nonEmptyValueChanges = new EventEmitter();\n    visibilityChanges = new BehaviorSubject<boolean>(true);\n\n    get id(): string {\n      return this.path.toLowerCase().slice(1).replace(/\\//g, '-');\n    }\n\n    get isRoot(): boolean {\n      return this === this.root;\n    }\n\n    readonly path: string;\n    readonly schema: Schema;\n\n    get name(): string {\n      return this.path.split('/').pop();\n    }\n\n    protected _visible = true;\n    get visible(): boolean {\n      return this._visible;\n    }\n\n    constructor(...args: any[]) {\n      super(...args);\n\n    }\n\n    getErrors(): FormPropertyErrors | null {\n      const errors = this.errors;\n\n      if (!errors) {\n        return null;\n      }\n\n      return new FormPropertyErrors({ [this.path]: errors });\n    }\n\n    setVisible(visible: boolean, opts = { disable: false }) {\n      this._visible = visible;\n      if (opts.disable) {\n        if (this.visible) {\n          this.enable();\n        } else {\n          this.disable();\n        }\n      }\n      this.visibilityChanges.next(this.visible);\n    }\n\n    // visible if AT LEAST ONE of the properties it depends on is visible\n    // AND has a value in the list\n    bindVisibility() {\n      // SHOULD ONLY BE CALLED AFTER ENTIRE PROPERTY TREE IS BUILT\n      const visibleIf = this.schema.visibleIf;\n\n      if (visibleIf === undefined) {\n        return;\n      }\n\n      const paths = Object.keys(visibleIf);\n      if (typeof visibleIf === 'object' && paths.length === 0) {\n        this.setVisible(false);\n        return;\n      }\n\n      const observables = [];\n      for (const path of paths) {\n        if (!visibleIf.hasOwnProperty(path)) {\n          continue;\n        }\n\n        const property = this.root.get(path);\n        if (!property) {\n          console.warn(\n            `Couldn't find property ${path} for visibility check of ` + this.path\n          );\n          continue;\n        }\n\n        const values = visibleIf[path];\n\n        const observable = property.valueChanges.pipe(\n          startWith(values.includes(property.value)),\n          map((value) => {\n            return values.includes('$ANY$') || values.includes(value);\n          })\n        );\n\n        observables.push(observable);\n      }\n\n      // TODO unsubscribe\n      combineLatest(observables)\n        .subscribe((values: boolean[]) => {\n          this.setVisible(values.includes(true));\n        });\n\n    }\n\n    get(path: Array<string|number>|string): AbstractControl|null {\n      if (typeof path === 'string' && path.includes('/')) {\n        path = this.normalizePath(path);\n      }\n      return super.get(path);\n    }\n\n    private normalizePath(path: string): string[] {\n      if (path[0] === '/') {\n        path = path.slice(1);\n      }\n      return path.split('/');\n    }\n\n  }\n\n  return Property;\n\n}\n\n\n"]}