angular2
Version:
Angular 2 - a web framework for modern web apps
233 lines • 31.1 kB
JavaScript
import { global, isFunction, stringify } from 'angular2/src/facade/lang';
function extractAnnotation(annotation) {
if (isFunction(annotation) && annotation.hasOwnProperty('annotation')) {
// it is a decorator, extract annotation
annotation = annotation.annotation;
}
return annotation;
}
function applyParams(fnOrArray, key) {
if (fnOrArray === Object || fnOrArray === String || fnOrArray === Function ||
fnOrArray === Number || fnOrArray === Array) {
throw new Error(`Can not use native ${stringify(fnOrArray)} as constructor`);
}
if (isFunction(fnOrArray)) {
return fnOrArray;
}
else if (fnOrArray instanceof Array) {
var annotations = fnOrArray;
var fn = fnOrArray[fnOrArray.length - 1];
if (!isFunction(fn)) {
throw new Error(`Last position of Class method array must be Function in key ${key} was '${stringify(fn)}'`);
}
var annoLength = annotations.length - 1;
if (annoLength != fn.length) {
throw new Error(`Number of annotations (${annoLength}) does not match number of arguments (${fn.length}) in the function: ${stringify(fn)}`);
}
var paramsAnnotations = [];
for (var i = 0, ii = annotations.length - 1; i < ii; i++) {
var paramAnnotations = [];
paramsAnnotations.push(paramAnnotations);
var annotation = annotations[i];
if (annotation instanceof Array) {
for (var j = 0; j < annotation.length; j++) {
paramAnnotations.push(extractAnnotation(annotation[j]));
}
}
else if (isFunction(annotation)) {
paramAnnotations.push(extractAnnotation(annotation));
}
else {
paramAnnotations.push(annotation);
}
}
Reflect.defineMetadata('parameters', paramsAnnotations, fn);
return fn;
}
else {
throw new Error(`Only Function or Array is supported in Class definition for key '${key}' is '${stringify(fnOrArray)}'`);
}
}
/**
* Provides a way for expressing ES6 classes with parameter annotations in ES5.
*
* ## Basic Example
*
* ```
* var Greeter = ng.Class({
* constructor: function(name) {
* this.name = name;
* },
*
* greet: function() {
* alert('Hello ' + this.name + '!');
* }
* });
* ```
*
* is equivalent to ES6:
*
* ```
* class Greeter {
* constructor(name) {
* this.name = name;
* }
*
* greet() {
* alert('Hello ' + this.name + '!');
* }
* }
* ```
*
* or equivalent to ES5:
*
* ```
* var Greeter = function (name) {
* this.name = name;
* }
*
* Greeter.prototype.greet = function () {
* alert('Hello ' + this.name + '!');
* }
* ```
*
* ### Example with parameter annotations
*
* ```
* var MyService = ng.Class({
* constructor: [String, [new Query(), QueryList], function(name, queryList) {
* ...
* }]
* });
* ```
*
* is equivalent to ES6:
*
* ```
* class MyService {
* constructor(name: string, @Query() queryList: QueryList) {
* ...
* }
* }
* ```
*
* ### Example with inheritance
*
* ```
* var Shape = ng.Class({
* constructor: (color) {
* this.color = color;
* }
* });
*
* var Square = ng.Class({
* extends: Shape,
* constructor: function(color, size) {
* Shape.call(this, color);
* this.size = size;
* }
* });
* ```
*/
export function Class(clsDef) {
var constructor = applyParams(clsDef.hasOwnProperty('constructor') ? clsDef.constructor : undefined, 'constructor');
var proto = constructor.prototype;
if (clsDef.hasOwnProperty('extends')) {
if (isFunction(clsDef.extends)) {
constructor.prototype = proto =
Object.create(clsDef.extends.prototype);
}
else {
throw new Error(`Class definition 'extends' property must be a constructor function was: ${stringify(clsDef.extends)}`);
}
}
for (var key in clsDef) {
if (key != 'extends' && key != 'prototype' && clsDef.hasOwnProperty(key)) {
proto[key] = applyParams(clsDef[key], key);
}
}
if (this && this.annotations instanceof Array) {
Reflect.defineMetadata('annotations', this.annotations, constructor);
}
return constructor;
}
var Reflect = global.Reflect;
if (!(Reflect && Reflect.getMetadata)) {
throw 'reflect-metadata shim is required when using class decorators';
}
export function makeDecorator(annotationCls, chainFn = null) {
function DecoratorFactory(objOrType) {
var annotationInstance = new annotationCls(objOrType);
if (this instanceof annotationCls) {
return annotationInstance;
}
else {
var chainAnnotation = isFunction(this) && this.annotations instanceof Array ? this.annotations : [];
chainAnnotation.push(annotationInstance);
var TypeDecorator = function TypeDecorator(cls) {
var annotations = Reflect.getOwnMetadata('annotations', cls);
annotations = annotations || [];
annotations.push(annotationInstance);
Reflect.defineMetadata('annotations', annotations, cls);
return cls;
};
TypeDecorator.annotations = chainAnnotation;
TypeDecorator.Class = Class;
if (chainFn)
chainFn(TypeDecorator);
return TypeDecorator;
}
}
DecoratorFactory.prototype = Object.create(annotationCls.prototype);
return DecoratorFactory;
}
export function makeParamDecorator(annotationCls) {
function ParamDecoratorFactory(...args) {
var annotationInstance = Object.create(annotationCls.prototype);
annotationCls.apply(annotationInstance, args);
if (this instanceof annotationCls) {
return annotationInstance;
}
else {
ParamDecorator.annotation = annotationInstance;
return ParamDecorator;
}
function ParamDecorator(cls, unusedKey, index) {
var parameters = Reflect.getMetadata('parameters', cls);
parameters = parameters || [];
// there might be gaps if some in between parameters do not have annotations.
// we pad with nulls.
while (parameters.length <= index) {
parameters.push(null);
}
parameters[index] = parameters[index] || [];
var annotationsForParam = parameters[index];
annotationsForParam.push(annotationInstance);
Reflect.defineMetadata('parameters', parameters, cls);
return cls;
}
}
ParamDecoratorFactory.prototype = Object.create(annotationCls.prototype);
return ParamDecoratorFactory;
}
export function makePropDecorator(decoratorCls) {
function PropDecoratorFactory(...args) {
var decoratorInstance = Object.create(decoratorCls.prototype);
decoratorCls.apply(decoratorInstance, args);
if (this instanceof decoratorCls) {
return decoratorInstance;
}
else {
return function PropDecorator(target, name) {
var meta = Reflect.getOwnMetadata('propMetadata', target.constructor);
meta = meta || {};
meta[name] = meta[name] || [];
meta[name].unshift(decoratorInstance);
Reflect.defineMetadata('propMetadata', meta, target.constructor);
};
}
}
PropDecoratorFactory.prototype = Object.create(decoratorCls.prototype);
return PropDecoratorFactory;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["angular2/src/core/util/decorators.ts"],"names":["extractAnnotation","applyParams","Class","makeDecorator","makeDecorator.DecoratorFactory","makeDecorator.DecoratorFactory.TypeDecorator","makeParamDecorator","makeParamDecorator.ParamDecoratorFactory","makeParamDecorator.ParamDecoratorFactory.ParamDecorator","makePropDecorator","makePropDecorator.PropDecoratorFactory","makePropDecorator.PropDecoratorFactory.PropDecorator"],"mappings":"OAAO,EAAe,MAAM,EAAQ,UAAU,EAAE,SAAS,EAAC,MAAM,0BAA0B;AA4E1F,2BAA2B,UAAe;IACxCA,EAAEA,CAACA,CAACA,UAAUA,CAACA,UAAUA,CAACA,IAAIA,UAAUA,CAACA,cAAcA,CAACA,YAAYA,CAACA,CAACA,CAACA,CAACA;QACtEA,wCAAwCA;QACxCA,UAAUA,GAAGA,UAAUA,CAACA,UAAUA,CAACA;IACrCA,CAACA;IACDA,MAAMA,CAACA,UAAUA,CAACA;AACpBA,CAACA;AAED,qBAAqB,SAA6B,EAAE,GAAW;IAC7DC,EAAEA,CAACA,CAACA,SAASA,KAAKA,MAAMA,IAAIA,SAASA,KAAKA,MAAMA,IAAIA,SAASA,KAAKA,QAAQA;QACtEA,SAASA,KAAKA,MAAMA,IAAIA,SAASA,KAAKA,KAAKA,CAACA,CAACA,CAACA;QAChDA,MAAMA,IAAIA,KAAKA,CAACA,sBAAsBA,SAASA,CAACA,SAASA,CAACA,iBAAiBA,CAACA,CAACA;IAC/EA,CAACA;IACDA,EAAEA,CAACA,CAACA,UAAUA,CAACA,SAASA,CAACA,CAACA,CAACA,CAACA;QAC1BA,MAAMA,CAAWA,SAASA,CAACA;IAC7BA,CAACA;IAACA,IAAIA,CAACA,EAAEA,CAACA,CAACA,SAASA,YAAYA,KAAKA,CAACA,CAACA,CAACA;QACtCA,IAAIA,WAAWA,GAAUA,SAASA,CAACA;QACnCA,IAAIA,EAAEA,GAAaA,SAASA,CAACA,SAASA,CAACA,MAAMA,GAAGA,CAACA,CAACA,CAACA;QACnDA,EAAEA,CAACA,CAACA,CAACA,UAAUA,CAACA,EAAEA,CAACA,CAACA,CAACA,CAACA;YACpBA,MAAMA,IAAIA,KAAKA,CACXA,+DAA+DA,GAAGA,SAASA,SAASA,CAACA,EAAEA,CAACA,GAAGA,CAACA,CAACA;QACnGA,CAACA;QACDA,IAAIA,UAAUA,GAAGA,WAAWA,CAACA,MAAMA,GAAGA,CAACA,CAACA;QACxCA,EAAEA,CAACA,CAACA,UAAUA,IAAIA,EAAEA,CAACA,MAAMA,CAACA,CAACA,CAACA;YAC5BA,MAAMA,IAAIA,KAAKA,CACXA,0BAA0BA,UAAUA,yCAAyCA,EAAEA,CAACA,MAAMA,sBAAsBA,SAASA,CAACA,EAAEA,CAACA,EAAEA,CAACA,CAACA;QACnIA,CAACA;QACDA,IAAIA,iBAAiBA,GAAYA,EAAEA,CAACA;QACpCA,GAAGA,CAACA,CAACA,GAAGA,CAACA,CAACA,GAAGA,CAACA,EAAEA,EAAEA,GAAGA,WAAWA,CAACA,MAAMA,GAAGA,CAACA,EAAEA,CAACA,GAAGA,EAAEA,EAAEA,CAACA,EAAEA,EAAEA,CAACA;YACzDA,IAAIA,gBAAgBA,GAAUA,EAAEA,CAACA;YACjCA,iBAAiBA,CAACA,IAAIA,CAACA,gBAAgBA,CAACA,CAACA;YACzCA,IAAIA,UAAUA,GAAGA,WAAWA,CAACA,CAACA,CAACA,CAACA;YAChCA,EAAEA,CAACA,CAACA,UAAUA,YAAYA,KAAKA,CAACA,CAACA,CAACA;gBAChCA,GAAGA,CAACA,CAACA,GAAGA,CAACA,CAACA,GAAGA,CAACA,EAAEA,CAACA,GAAGA,UAAUA,CAACA,MAAMA,EAAEA,CAACA,EAAEA,EAAEA,CAACA;oBAC3CA,gBAAgBA,CAACA,IAAIA,CAACA,iBAAiBA,CAACA,UAAUA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBAC1DA,CAACA;YACHA,CAACA;YAACA,IAAIA,CAACA,EAAEA,CAACA,CAACA,UAAUA,CAACA,UAAUA,CAACA,CAACA,CAACA,CAACA;gBAClCA,gBAAgBA,CAACA,IAAIA,CAACA,iBAAiBA,CAACA,UAAUA,CAACA,CAACA,CAACA;YACvDA,CAACA;YAACA,IAAIA,CAACA,CAACA;gBACNA,gBAAgBA,CAACA,IAAIA,CAACA,UAAUA,CAACA,CAACA;YACpCA,CAACA;QACHA,CAACA;QACDA,OAAOA,CAACA,cAAcA,CAACA,YAAYA,EAAEA,iBAAiBA,EAAEA,EAAEA,CAACA,CAACA;QAC5DA,MAAMA,CAACA,EAAEA,CAACA;IACZA,CAACA;IAACA,IAAIA,CAACA,CAACA;QACNA,MAAMA,IAAIA,KAAKA,CACXA,oEAAoEA,GAAGA,SAASA,SAASA,CAACA,SAASA,CAACA,GAAGA,CAACA,CAACA;IAC/GA,CAACA;AACHA,CAACA;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgFG;AACH,sBAAsB,MAAuB;IAC3CC,IAAIA,WAAWA,GAAGA,WAAWA,CACzBA,MAAMA,CAACA,cAAcA,CAACA,aAAaA,CAACA,GAAGA,MAAMA,CAACA,WAAWA,GAAGA,SAASA,EAAEA,aAAaA,CAACA,CAACA;IAC1FA,IAAIA,KAAKA,GAAGA,WAAWA,CAACA,SAASA,CAACA;IAClCA,EAAEA,CAACA,CAACA,MAAMA,CAACA,cAAcA,CAACA,SAASA,CAACA,CAACA,CAACA,CAACA;QACrCA,EAAEA,CAACA,CAACA,UAAUA,CAACA,MAAMA,CAACA,OAAOA,CAACA,CAACA,CAACA,CAACA;YACpBA,WAAYA,CAACA,SAASA,GAAGA,KAAKA;gBACrCA,MAAMA,CAACA,MAAMA,CAAYA,MAAMA,CAACA,OAAQA,CAACA,SAASA,CAACA,CAACA;QAC1DA,CAACA;QAACA,IAAIA,CAACA,CAACA;YACNA,MAAMA,IAAIA,KAAKA,CACXA,2EAA2EA,SAASA,CAACA,MAAMA,CAACA,OAAOA,CAACA,EAAEA,CAACA,CAACA;QAC9GA,CAACA;IACHA,CAACA;IACDA,GAAGA,CAACA,CAACA,GAAGA,CAACA,GAAGA,IAAIA,MAAMA,CAACA,CAACA,CAACA;QACvBA,EAAEA,CAACA,CAACA,GAAGA,IAAIA,SAASA,IAAIA,GAAGA,IAAIA,WAAWA,IAAIA,MAAMA,CAACA,cAAcA,CAACA,GAAGA,CAACA,CAACA,CAACA,CAACA;YACzEA,KAAKA,CAACA,GAAGA,CAACA,GAAGA,WAAWA,CAACA,MAAMA,CAACA,GAAGA,CAACA,EAAEA,GAAGA,CAACA,CAACA;QAC7CA,CAACA;IACHA,CAACA;IAEDA,EAAEA,CAACA,CAACA,IAAIA,IAAIA,IAAIA,CAACA,WAAWA,YAAYA,KAAKA,CAACA,CAACA,CAACA;QAC9CA,OAAOA,CAACA,cAAcA,CAACA,aAAaA,EAAEA,IAAIA,CAACA,WAAWA,EAAEA,WAAWA,CAACA,CAACA;IACvEA,CAACA;IAEDA,MAAMA,CAAeA,WAAWA,CAACA;AACnCA,CAACA;AAED,IAAI,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;AAC7B,EAAE,CAAC,CAAC,CAAC,CAAC,OAAO,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;IACtC,MAAM,+DAA+D,CAAC;AACxE,CAAC;AAED,8BACI,aAAa,EAAE,OAAO,GAA2B,IAAI;IACvDC,0BAA0BA,SAASA;QACjCC,IAAIA,kBAAkBA,GAAGA,IAAUA,aAAcA,CAACA,SAASA,CAACA,CAACA;QAC7DA,EAAEA,CAACA,CAACA,IAAIA,YAAYA,aAAaA,CAACA,CAACA,CAACA;YAClCA,MAAMA,CAACA,kBAAkBA,CAACA;QAC5BA,CAACA;QAACA,IAAIA,CAACA,CAACA;YACNA,IAAIA,eAAeA,GACfA,UAAUA,CAACA,IAAIA,CAACA,IAAIA,IAAIA,CAACA,WAAWA,YAAYA,KAAKA,GAAGA,IAAIA,CAACA,WAAWA,GAAGA,EAAEA,CAACA;YAClFA,eAAeA,CAACA,IAAIA,CAACA,kBAAkBA,CAACA,CAACA;YACzCA,IAAIA,aAAaA,GAAiCA,uBAAuBA,GAAGA;gBAC1EC,IAAIA,WAAWA,GAAGA,OAAOA,CAACA,cAAcA,CAACA,aAAaA,EAAEA,GAAGA,CAACA,CAACA;gBAC7DA,WAAWA,GAAGA,WAAWA,IAAIA,EAAEA,CAACA;gBAChCA,WAAWA,CAACA,IAAIA,CAACA,kBAAkBA,CAACA,CAACA;gBACrCA,OAAOA,CAACA,cAAcA,CAACA,aAAaA,EAAEA,WAAWA,EAAEA,GAAGA,CAACA,CAACA;gBACxDA,MAAMA,CAACA,GAAGA,CAACA;YACbA,CAACA,CAACD;YACFA,aAAaA,CAACA,WAAWA,GAAGA,eAAeA,CAACA;YAC5CA,aAAaA,CAACA,KAAKA,GAAGA,KAAKA,CAACA;YAC5BA,EAAEA,CAACA,CAACA,OAAOA,CAACA;gBAACA,OAAOA,CAACA,aAAaA,CAACA,CAACA;YACpCA,MAAMA,CAACA,aAAaA,CAACA;QACvBA,CAACA;IACHA,CAACA;IACDD,gBAAgBA,CAACA,SAASA,GAAGA,MAAMA,CAACA,MAAMA,CAACA,aAAaA,CAACA,SAASA,CAACA,CAACA;IACpEA,MAAMA,CAACA,gBAAgBA,CAACA;AAC1BA,CAACA;AAED,mCAAmC,aAAa;IAC9CG,+BAA+BA,GAAGA,IAAIA;QACpCC,IAAIA,kBAAkBA,GAAGA,MAAMA,CAACA,MAAMA,CAACA,aAAaA,CAACA,SAASA,CAACA,CAACA;QAChEA,aAAaA,CAACA,KAAKA,CAACA,kBAAkBA,EAAEA,IAAIA,CAACA,CAACA;QAC9CA,EAAEA,CAACA,CAACA,IAAIA,YAAYA,aAAaA,CAACA,CAACA,CAACA;YAClCA,MAAMA,CAACA,kBAAkBA,CAACA;QAC5BA,CAACA;QAACA,IAAIA,CAACA,CAACA;YACAA,cAAeA,CAACA,UAAUA,GAAGA,kBAAkBA,CAACA;YACtDA,MAAMA,CAACA,cAAcA,CAACA;QACxBA,CAACA;QAGDA,wBAAwBA,GAAGA,EAAEA,SAASA,EAAEA,KAAKA;YAC3CC,IAAIA,UAAUA,GAAYA,OAAOA,CAACA,WAAWA,CAACA,YAAYA,EAAEA,GAAGA,CAACA,CAACA;YACjEA,UAAUA,GAAGA,UAAUA,IAAIA,EAAEA,CAACA;YAE9BA,6EAA6EA;YAC7EA,qBAAqBA;YACrBA,OAAOA,UAAUA,CAACA,MAAMA,IAAIA,KAAKA,EAAEA,CAACA;gBAClCA,UAAUA,CAACA,IAAIA,CAACA,IAAIA,CAACA,CAACA;YACxBA,CAACA;YAEDA,UAAUA,CAACA,KAAKA,CAACA,GAAGA,UAAUA,CAACA,KAAKA,CAACA,IAAIA,EAAEA,CAACA;YAC5CA,IAAIA,mBAAmBA,GAAUA,UAAUA,CAACA,KAAKA,CAACA,CAACA;YACnDA,mBAAmBA,CAACA,IAAIA,CAACA,kBAAkBA,CAACA,CAACA;YAE7CA,OAAOA,CAACA,cAAcA,CAACA,YAAYA,EAAEA,UAAUA,EAAEA,GAAGA,CAACA,CAACA;YACtDA,MAAMA,CAACA,GAAGA,CAACA;QACbA,CAACA;IACHD,CAACA;IACDD,qBAAqBA,CAACA,SAASA,GAAGA,MAAMA,CAACA,MAAMA,CAACA,aAAaA,CAACA,SAASA,CAACA,CAACA;IACzEA,MAAMA,CAACA,qBAAqBA,CAACA;AAC/BA,CAACA;AAED,kCAAkC,YAAY;IAC5CG,8BAA8BA,GAAGA,IAAIA;QACnCC,IAAIA,iBAAiBA,GAAGA,MAAMA,CAACA,MAAMA,CAACA,YAAYA,CAACA,SAASA,CAACA,CAACA;QAC9DA,YAAYA,CAACA,KAAKA,CAACA,iBAAiBA,EAAEA,IAAIA,CAACA,CAACA;QAE5CA,EAAEA,CAACA,CAACA,IAAIA,YAAYA,YAAYA,CAACA,CAACA,CAACA;YACjCA,MAAMA,CAACA,iBAAiBA,CAACA;QAC3BA,CAACA;QAACA,IAAIA,CAACA,CAACA;YACNA,MAAMA,CAACA,uBAAuBA,MAAWA,EAAEA,IAAYA;gBACrDC,IAAIA,IAAIA,GAAGA,OAAOA,CAACA,cAAcA,CAACA,cAAcA,EAAEA,MAAMA,CAACA,WAAWA,CAACA,CAACA;gBACtEA,IAAIA,GAAGA,IAAIA,IAAIA,EAAEA,CAACA;gBAClBA,IAAIA,CAACA,IAAIA,CAACA,GAAGA,IAAIA,CAACA,IAAIA,CAACA,IAAIA,EAAEA,CAACA;gBAC9BA,IAAIA,CAACA,IAAIA,CAACA,CAACA,OAAOA,CAACA,iBAAiBA,CAACA,CAACA;gBACtCA,OAAOA,CAACA,cAAcA,CAACA,cAAcA,EAAEA,IAAIA,EAAEA,MAAMA,CAACA,WAAWA,CAACA,CAACA;YACnEA,CAACA,CAACD;QACJA,CAACA;IACHA,CAACA;IACDD,oBAAoBA,CAACA,SAASA,GAAGA,MAAMA,CAACA,MAAMA,CAACA,YAAYA,CAACA,SAASA,CAACA,CAACA;IACvEA,MAAMA,CAACA,oBAAoBA,CAACA;AAC9BA,CAACA","sourcesContent":["import {ConcreteType, global, Type, isFunction, stringify} from 'angular2/src/facade/lang';\n\n/**\n * Declares the interface to be used with {@link Class}.\n */\nexport interface ClassDefinition {\n  /**\n   * Optional argument for specifying the superclass.\n   */\n  extends?: Type;\n\n  /**\n   * Required constructor function for a class.\n   *\n   * The function may be optionally wrapped in an `Array`, in which case additional parameter\n   * annotations may be specified.\n   * The number of arguments and the number of parameter annotations must match.\n   *\n   * See {@link Class} for example of usage.\n   */\n  constructor: Function | any[];\n\n  /**\n   * Other methods on the class. Note that values should have type 'Function' but TS requires\n   * all properties to have a narrower type than the index signature.\n   */\n  [x: string]: Type | Function | any[];\n}\n\n/**\n * An interface implemented by all Angular type decorators, which allows them to be used as ES7\n * decorators as well as\n * Angular DSL syntax.\n *\n * DSL syntax:\n *\n * ```\n * var MyClass = ng\n *   .Component({...})\n *   .View({...})\n *   .Class({...});\n * ```\n *\n * ES7 syntax:\n *\n * ```\n * @ng.Component({...})\n * @ng.View({...})\n * class MyClass {...}\n * ```\n */\nexport interface TypeDecorator {\n  /**\n   * Invoke as ES7 decorator.\n   */\n  <T extends Type>(type: T): T;\n\n  // Make TypeDecorator assignable to built-in ParameterDecorator type.\n  // ParameterDecorator is declared in lib.d.ts as a `declare type`\n  // so we cannot declare this interface as a subtype.\n  // see https://github.com/angular/angular/issues/3379#issuecomment-126169417\n  (target: Object, propertyKey?: string | symbol, parameterIndex?: number): void;\n\n  /**\n   * Storage for the accumulated annotations so far used by the DSL syntax.\n   *\n   * Used by {@link Class} to annotate the generated class.\n   */\n  annotations: any[];\n\n  /**\n   * Generate a class from the definition and annotate it with {@link TypeDecorator#annotations}.\n   */\n  Class(obj: ClassDefinition): ConcreteType;\n}\n\nfunction extractAnnotation(annotation: any): any {\n  if (isFunction(annotation) && annotation.hasOwnProperty('annotation')) {\n    // it is a decorator, extract annotation\n    annotation = annotation.annotation;\n  }\n  return annotation;\n}\n\nfunction applyParams(fnOrArray: (Function | any[]), key: string): Function {\n  if (fnOrArray === Object || fnOrArray === String || fnOrArray === Function ||\n      fnOrArray === Number || fnOrArray === Array) {\n    throw new Error(`Can not use native ${stringify(fnOrArray)} as constructor`);\n  }\n  if (isFunction(fnOrArray)) {\n    return <Function>fnOrArray;\n  } else if (fnOrArray instanceof Array) {\n    var annotations: any[] = fnOrArray;\n    var fn: Function = fnOrArray[fnOrArray.length - 1];\n    if (!isFunction(fn)) {\n      throw new Error(\n          `Last position of Class method array must be Function in key ${key} was '${stringify(fn)}'`);\n    }\n    var annoLength = annotations.length - 1;\n    if (annoLength != fn.length) {\n      throw new Error(\n          `Number of annotations (${annoLength}) does not match number of arguments (${fn.length}) in the function: ${stringify(fn)}`);\n    }\n    var paramsAnnotations: any[][] = [];\n    for (var i = 0, ii = annotations.length - 1; i < ii; i++) {\n      var paramAnnotations: any[] = [];\n      paramsAnnotations.push(paramAnnotations);\n      var annotation = annotations[i];\n      if (annotation instanceof Array) {\n        for (var j = 0; j < annotation.length; j++) {\n          paramAnnotations.push(extractAnnotation(annotation[j]));\n        }\n      } else if (isFunction(annotation)) {\n        paramAnnotations.push(extractAnnotation(annotation));\n      } else {\n        paramAnnotations.push(annotation);\n      }\n    }\n    Reflect.defineMetadata('parameters', paramsAnnotations, fn);\n    return fn;\n  } else {\n    throw new Error(\n        `Only Function or Array is supported in Class definition for key '${key}' is '${stringify(fnOrArray)}'`);\n  }\n}\n\n/**\n * Provides a way for expressing ES6 classes with parameter annotations in ES5.\n *\n * ## Basic Example\n *\n * ```\n * var Greeter = ng.Class({\n *   constructor: function(name) {\n *     this.name = name;\n *   },\n *\n *   greet: function() {\n *     alert('Hello ' + this.name + '!');\n *   }\n * });\n * ```\n *\n * is equivalent to ES6:\n *\n * ```\n * class Greeter {\n *   constructor(name) {\n *     this.name = name;\n *   }\n *\n *   greet() {\n *     alert('Hello ' + this.name + '!');\n *   }\n * }\n * ```\n *\n * or equivalent to ES5:\n *\n * ```\n * var Greeter = function (name) {\n *   this.name = name;\n * }\n *\n * Greeter.prototype.greet = function () {\n *   alert('Hello ' + this.name + '!');\n * }\n * ```\n *\n * ### Example with parameter annotations\n *\n * ```\n * var MyService = ng.Class({\n *   constructor: [String, [new Query(), QueryList], function(name, queryList) {\n *     ...\n *   }]\n * });\n * ```\n *\n * is equivalent to ES6:\n *\n * ```\n * class MyService {\n *   constructor(name: string, @Query() queryList: QueryList) {\n *     ...\n *   }\n * }\n * ```\n *\n * ### Example with inheritance\n *\n * ```\n * var Shape = ng.Class({\n *   constructor: (color) {\n *     this.color = color;\n *   }\n * });\n *\n * var Square = ng.Class({\n *   extends: Shape,\n *   constructor: function(color, size) {\n *     Shape.call(this, color);\n *     this.size = size;\n *   }\n * });\n * ```\n */\nexport function Class(clsDef: ClassDefinition): ConcreteType {\n  var constructor = applyParams(\n      clsDef.hasOwnProperty('constructor') ? clsDef.constructor : undefined, 'constructor');\n  var proto = constructor.prototype;\n  if (clsDef.hasOwnProperty('extends')) {\n    if (isFunction(clsDef.extends)) {\n      (<Function>constructor).prototype = proto =\n          Object.create((<Function>clsDef.extends).prototype);\n    } else {\n      throw new Error(\n          `Class definition 'extends' property must be a constructor function was: ${stringify(clsDef.extends)}`);\n    }\n  }\n  for (var key in clsDef) {\n    if (key != 'extends' && key != 'prototype' && clsDef.hasOwnProperty(key)) {\n      proto[key] = applyParams(clsDef[key], key);\n    }\n  }\n\n  if (this && this.annotations instanceof Array) {\n    Reflect.defineMetadata('annotations', this.annotations, constructor);\n  }\n\n  return <ConcreteType>constructor;\n}\n\nvar Reflect = global.Reflect;\nif (!(Reflect && Reflect.getMetadata)) {\n  throw 'reflect-metadata shim is required when using class decorators';\n}\n\nexport function makeDecorator(\n    annotationCls, chainFn: (fn: Function) => void = null): (...args: any[]) => (cls: any) => any {\n  function DecoratorFactory(objOrType): (cls: any) => any {\n    var annotationInstance = new (<any>annotationCls)(objOrType);\n    if (this instanceof annotationCls) {\n      return annotationInstance;\n    } else {\n      var chainAnnotation =\n          isFunction(this) && this.annotations instanceof Array ? this.annotations : [];\n      chainAnnotation.push(annotationInstance);\n      var TypeDecorator: TypeDecorator = <TypeDecorator>function TypeDecorator(cls) {\n        var annotations = Reflect.getOwnMetadata('annotations', cls);\n        annotations = annotations || [];\n        annotations.push(annotationInstance);\n        Reflect.defineMetadata('annotations', annotations, cls);\n        return cls;\n      };\n      TypeDecorator.annotations = chainAnnotation;\n      TypeDecorator.Class = Class;\n      if (chainFn) chainFn(TypeDecorator);\n      return TypeDecorator;\n    }\n  }\n  DecoratorFactory.prototype = Object.create(annotationCls.prototype);\n  return DecoratorFactory;\n}\n\nexport function makeParamDecorator(annotationCls): any {\n  function ParamDecoratorFactory(...args): any {\n    var annotationInstance = Object.create(annotationCls.prototype);\n    annotationCls.apply(annotationInstance, args);\n    if (this instanceof annotationCls) {\n      return annotationInstance;\n    } else {\n      (<any>ParamDecorator).annotation = annotationInstance;\n      return ParamDecorator;\n    }\n\n\n    function ParamDecorator(cls, unusedKey, index): any {\n      var parameters: any[][] = Reflect.getMetadata('parameters', cls);\n      parameters = parameters || [];\n\n      // there might be gaps if some in between parameters do not have annotations.\n      // we pad with nulls.\n      while (parameters.length <= index) {\n        parameters.push(null);\n      }\n\n      parameters[index] = parameters[index] || [];\n      var annotationsForParam: any[] = parameters[index];\n      annotationsForParam.push(annotationInstance);\n\n      Reflect.defineMetadata('parameters', parameters, cls);\n      return cls;\n    }\n  }\n  ParamDecoratorFactory.prototype = Object.create(annotationCls.prototype);\n  return ParamDecoratorFactory;\n}\n\nexport function makePropDecorator(decoratorCls): any {\n  function PropDecoratorFactory(...args): any {\n    var decoratorInstance = Object.create(decoratorCls.prototype);\n    decoratorCls.apply(decoratorInstance, args);\n\n    if (this instanceof decoratorCls) {\n      return decoratorInstance;\n    } else {\n      return function PropDecorator(target: any, name: string) {\n        var meta = Reflect.getOwnMetadata('propMetadata', target.constructor);\n        meta = meta || {};\n        meta[name] = meta[name] || [];\n        meta[name].unshift(decoratorInstance);\n        Reflect.defineMetadata('propMetadata', meta, target.constructor);\n      };\n    }\n  }\n  PropDecoratorFactory.prototype = Object.create(decoratorCls.prototype);\n  return PropDecoratorFactory;\n}\n"]}