'use strict'; import {Component, View, OnInit, OnDestroy, ChangeDetectionStrategy} from 'angular2/core'; import {CORE_DIRECTIVES, JsonPipe, AsyncPipe} from 'angular2/common'; import SchemaManager from '../utils/SchemaManager'; import JsonPointer from '../utils/JsonPointer'; import {MarkedPipe, JsonPointerEscapePipe} from '../utils/pipes'; export { SchemaManager }; // common inputs for all components let commonInputs = ['pointer']; // json pointer to the schema chunk // internal helper function function safeConcat(a, b) { let res = a && a.slice() || []; b = (b == null) ? [] : b; return res.concat(b); } function defaults(target, src) { var props = Object.keys(src); var index = -1, length = props.length; while (++index < length) { var key = props[index]; if (target[key] === undefined) { target[key] = src[key]; } } return target; } function snapshot(obj) { if(obj == null || typeof(obj) != 'object') { return obj; } var temp = new obj.constructor(); for(var key in obj) { if (obj.hasOwnProperty(key)) { temp[key] = snapshot(obj[key]); } } return temp; } /** * Class decorator * Simplifies setup of component metainfo * All options are options from either Component or View angular2 decorator * For detailed info look angular2 doc * @param {Object} options - component options * @param {string[]} options.inputs - component inputs * @param {*[]} options.directives - directives used by component * (except CORE_DIRECTIVES) * @param {*[]} options.pipes - pipes used by component * @param {*[]} options.providers - component providers * @param {string} options.templateUrl - path to component template * @param {string} options.template - component template html * @param {string} options.styles - component css styles */ export function RedocComponent(options) { let inputs = safeConcat(options.inputs, commonInputs); let directives = safeConcat(options.directives, CORE_DIRECTIVES); let pipes = safeConcat(options.pipes, [JsonPointerEscapePipe, MarkedPipe, JsonPipe, AsyncPipe]); return function decorator(target) { let componentDecorator = Component({ selector: options.selector, inputs: inputs, outputs: options.outputs, lifecycle: [OnInit, OnDestroy], providers: options.providers, changeDetection: options.changeDetection || ChangeDetectionStrategy.Detached }); let viewDecorator = View({ templateUrl: options.templateUrl, template: options.template, styles: options.styles, directives: directives, pipes: pipes }); return componentDecorator(viewDecorator(target) || target) || target; }; } /** * Generic Component * @class */ @Reflect.metadata('parameters', [[SchemaManager]]) export class BaseComponent { constructor(schemaMgr) { this.schemaMgr = schemaMgr; this.componentSchema = null; } /** * onInit method is run by angular2 after all component inputs are resolved */ ngOnInit() { this.componentSchema = snapshot(this.schemaMgr.byPointer(this.pointer || '')); this.prepareModel(); this.init(); } ngOnDestroy() { this.destroy(); } /** * simple in-place schema dereferencing. Schema is already bundled so no need in global dereferencing. */ dereference(schema = Object.assign({}, this.componentSchema)) { let dereferencedCache = {}; let resolve = (schema) => { let resolvedRef; if (schema && schema.$ref) { resolvedRef = schema.$ref; let resolved = this.schemaMgr.byPointer(schema.$ref); let baseName = JsonPointer.baseName(schema.$ref); if (!dereferencedCache[schema.$ref]) { // if resolved schema doesn't have title use name from ref resolved = Object.assign({}, resolved); resolved._pointer = schema.$ref; } else { // for circular referenced save only title and type resolved = { title: resolved.title, type: resolved.type }; } dereferencedCache[schema.$ref] = dereferencedCache[schema.$ref] ? dereferencedCache[schema.$ref] + 1 : 1; resolved.title = resolved.title || baseName; let keysCount = Object.keys(schema).length; if ( keysCount > 2 || (keysCount === 2 && !schema.description) ) { // allow only description field on the same level as $ref because it is // common pattern over specs in the wild console.warn(`other properties defined at the same level as $ref at '${this.pointer}'. They are IGNORRED according to JsonSchema spec`); } schema = schema.description ? { description: schema.description } : {}; Object.assign(schema, resolved); } Object.keys(schema).forEach((key) => { let value = schema[key]; if (value && typeof value === 'object') { schema[key] = resolve(value); } }); if (resolvedRef) dereferencedCache[resolvedRef] = dereferencedCache[resolvedRef] ? dereferencedCache[resolvedRef] - 1 : 0; return schema; }; this.componentSchema = snapshot(resolve(schema, 1)); } static joinAllOf(schema, opts) { function merge(into, schemas) { for (let subSchema of schemas) { if (opts && opts.omitParent && subSchema.discriminator) continue; // TODO: add support for merge array schemas if (typeof subSchema !== 'object') { let errMessage = `Items of allOf should be Object: ${typeof subSchema} found ${subSchema}`; throw new Error(errMessage); } if (into.type && subSchema.type && into.type !== subSchema.type) { let errMessage = `allOf merging error: schemas with different types can't be merged`; throw new Error(errMessage); } into.type = into.type || subSchema.type; if (into.type === 'array') { console.warn('allOf: subschemas with type array are not supported yet'); } // TODO: add check if can be merged correctly (no different properties with the same name) if (into.type === 'object' && subSchema.properties) { into.properties || (into.properties = {}); Object.assign(into.properties, subSchema.properties); } if (into.type === 'object' && subSchema.required) { into.required || (into.required = []); into.required.push(...subSchema.required); } defaults(into, subSchema); } into.allOf = null; } function traverse(obj) { if (obj === null || typeof(obj) !== 'object') { return; } for(var key in obj) { if (obj.hasOwnProperty(key)) { traverse(obj[key]); } } if (obj.allOf) { merge(obj, obj.allOf); } } traverse(schema); } /** * Used to prepare model based on component schema * @abstract */ prepareModel() {} /** * Used to initialize component. Run after prepareModel * @abstract */ init() {} /** + Used to destroy component * @abstract */ destroy() {} }