UNPKG

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
/** * @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"]}