UNPKG

34.5 kBJavaScriptView Raw
1import * as o from '../output/output_ast';
2import { Identifiers as R3 } from '../render3/r3_identifiers';
3import { typeWithParameters } from './util';
4export var R3FactoryDelegateType;
5(function (R3FactoryDelegateType) {
6 R3FactoryDelegateType[R3FactoryDelegateType["Class"] = 0] = "Class";
7 R3FactoryDelegateType[R3FactoryDelegateType["Function"] = 1] = "Function";
8})(R3FactoryDelegateType || (R3FactoryDelegateType = {}));
9export var FactoryTarget;
10(function (FactoryTarget) {
11 FactoryTarget[FactoryTarget["Directive"] = 0] = "Directive";
12 FactoryTarget[FactoryTarget["Component"] = 1] = "Component";
13 FactoryTarget[FactoryTarget["Injectable"] = 2] = "Injectable";
14 FactoryTarget[FactoryTarget["Pipe"] = 3] = "Pipe";
15 FactoryTarget[FactoryTarget["NgModule"] = 4] = "NgModule";
16})(FactoryTarget || (FactoryTarget = {}));
17/**
18 * Construct a factory function expression for the given `R3FactoryMetadata`.
19 */
20export function compileFactoryFunction(meta) {
21 const t = o.variable('t');
22 let baseFactoryVar = null;
23 // The type to instantiate via constructor invocation. If there is no delegated factory, meaning
24 // this type is always created by constructor invocation, then this is the type-to-create
25 // parameter provided by the user (t) if specified, or the current type if not. If there is a
26 // delegated factory (which is used to create the current type) then this is only the type-to-
27 // create parameter (t).
28 const typeForCtor = !isDelegatedFactoryMetadata(meta) ?
29 new o.BinaryOperatorExpr(o.BinaryOperator.Or, t, meta.internalType) :
30 t;
31 let ctorExpr = null;
32 if (meta.deps !== null) {
33 // There is a constructor (either explicitly or implicitly defined).
34 if (meta.deps !== 'invalid') {
35 ctorExpr = new o.InstantiateExpr(typeForCtor, injectDependencies(meta.deps, meta.target));
36 }
37 }
38 else {
39 // There is no constructor, use the base class' factory to construct typeForCtor.
40 baseFactoryVar = o.variable(${meta.name}_BaseFactory`);
41 ctorExpr = baseFactoryVar.callFn([typeForCtor]);
42 }
43 const body = [];
44 let retExpr = null;
45 function makeConditionalFactory(nonCtorExpr) {
46 const r = o.variable('r');
47 body.push(r.set(o.NULL_EXPR).toDeclStmt());
48 const ctorStmt = ctorExpr !== null ? r.set(ctorExpr).toStmt() :
49 o.importExpr(R3.invalidFactory).callFn([]).toStmt();
50 body.push(o.ifStmt(t, [ctorStmt], [r.set(nonCtorExpr).toStmt()]));
51 return r;
52 }
53 if (isDelegatedFactoryMetadata(meta)) {
54 // This type is created with a delegated factory. If a type parameter is not specified, call
55 // the factory instead.
56 const delegateArgs = injectDependencies(meta.delegateDeps, meta.target);
57 // Either call `new delegate(...)` or `delegate(...)` depending on meta.delegateType.
58 const factoryExpr = new (meta.delegateType === R3FactoryDelegateType.Class ?
59 o.InstantiateExpr :
60 o.InvokeFunctionExpr)(meta.delegate, delegateArgs);
61 retExpr = makeConditionalFactory(factoryExpr);
62 }
63 else if (isExpressionFactoryMetadata(meta)) {
64 // TODO(alxhub): decide whether to lower the value here or in the caller
65 retExpr = makeConditionalFactory(meta.expression);
66 }
67 else {
68 retExpr = ctorExpr;
69 }
70 if (retExpr === null) {
71 // The expression cannot be formed so render an `ɵɵinvalidFactory()` call.
72 body.push(o.importExpr(R3.invalidFactory).callFn([]).toStmt());
73 }
74 else if (baseFactoryVar !== null) {
75 // This factory uses a base factory, so call `ɵɵgetInheritedFactory()` to compute it.
76 const getInheritedFactoryCall = o.importExpr(R3.getInheritedFactory).callFn([meta.internalType]);
77 // Memoize the base factoryFn: `baseFactory || (baseFactory = ɵɵgetInheritedFactory(...))`
78 const baseFactory = new o.BinaryOperatorExpr(o.BinaryOperator.Or, baseFactoryVar, baseFactoryVar.set(getInheritedFactoryCall));
79 body.push(new o.ReturnStatement(baseFactory.callFn([typeForCtor])));
80 }
81 else {
82 // This is straightforward factory, just return it.
83 body.push(new o.ReturnStatement(retExpr));
84 }
85 let factoryFn = o.fn([new o.FnParam('t', o.DYNAMIC_TYPE)], body, o.INFERRED_TYPE, undefined, `${meta.name}_Factory`);
86 if (baseFactoryVar !== null) {
87 // There is a base factory variable so wrap its declaration along with the factory function into
88 // an IIFE.
89 factoryFn = o.fn([], [
90 new o.DeclareVarStmt(baseFactoryVar.name), new o.ReturnStatement(factoryFn)
91 ]).callFn([], /* sourceSpan */ undefined, /* pure */ true);
92 }
93 return {
94 expression: factoryFn,
95 statements: [],
96 type: createFactoryType(meta),
97 };
98}
99export function createFactoryType(meta) {
100 const ctorDepsType = meta.deps !== null && meta.deps !== 'invalid' ? createCtorDepsType(meta.deps) : o.NONE_TYPE;
101 return o.expressionType(o.importExpr(R3.FactoryDeclaration, [typeWithParameters(meta.type.type, meta.typeArgumentCount), ctorDepsType]));
102}
103function injectDependencies(deps, target) {
104 return deps.map((dep, index) => compileInjectDependency(dep, target, index));
105}
106function compileInjectDependency(dep, target, index) {
107 // Interpret the dependency according to its resolved type.
108 if (dep.token === null) {
109 return o.importExpr(R3.invalidFactoryDep).callFn([o.literal(index)]);
110 }
111 else if (dep.attributeNameType === null) {
112 // Build up the injection flags according to the metadata.
113 const flags = 0 /* Default */ | (dep.self ? 2 /* Self */ : 0) |
114 (dep.skipSelf ? 4 /* SkipSelf */ : 0) | (dep.host ? 1 /* Host */ : 0) |
115 (dep.optional ? 8 /* Optional */ : 0) |
116 (target === FactoryTarget.Pipe ? 16 /* ForPipe */ : 0);
117 // If this dependency is optional or otherwise has non-default flags, then additional
118 // parameters describing how to inject the dependency must be passed to the inject function
119 // that's being used.
120 let flagsParam = (flags !== 0 /* Default */ || dep.optional) ? o.literal(flags) : null;
121 // Build up the arguments to the injectFn call.
122 const injectArgs = [dep.token];
123 if (flagsParam) {
124 injectArgs.push(flagsParam);
125 }
126 const injectFn = getInjectFn(target);
127 return o.importExpr(injectFn).callFn(injectArgs);
128 }
129 else {
130 // The `dep.attributeTypeName` value is defined, which indicates that this is an `@Attribute()`
131 // type dependency. For the generated JS we still want to use the `dep.token` value in case the
132 // name given for the attribute is not a string literal. For example given `@Attribute(foo())`,
133 // we want to generate `ɵɵinjectAttribute(foo())`.
134 //
135 // The `dep.attributeTypeName` is only actually used (in `createCtorDepType()`) to generate
136 // typings.
137 return o.importExpr(R3.injectAttribute).callFn([dep.token]);
138 }
139}
140function createCtorDepsType(deps) {
141 let hasTypes = false;
142 const attributeTypes = deps.map(dep => {
143 const type = createCtorDepType(dep);
144 if (type !== null) {
145 hasTypes = true;
146 return type;
147 }
148 else {
149 return o.literal(null);
150 }
151 });
152 if (hasTypes) {
153 return o.expressionType(o.literalArr(attributeTypes));
154 }
155 else {
156 return o.NONE_TYPE;
157 }
158}
159function createCtorDepType(dep) {
160 const entries = [];
161 if (dep.attributeNameType !== null) {
162 entries.push({ key: 'attribute', value: dep.attributeNameType, quoted: false });
163 }
164 if (dep.optional) {
165 entries.push({ key: 'optional', value: o.literal(true), quoted: false });
166 }
167 if (dep.host) {
168 entries.push({ key: 'host', value: o.literal(true), quoted: false });
169 }
170 if (dep.self) {
171 entries.push({ key: 'self', value: o.literal(true), quoted: false });
172 }
173 if (dep.skipSelf) {
174 entries.push({ key: 'skipSelf', value: o.literal(true), quoted: false });
175 }
176 return entries.length > 0 ? o.literalMap(entries) : null;
177}
178export function isDelegatedFactoryMetadata(meta) {
179 return meta.delegateType !== undefined;
180}
181export function isExpressionFactoryMetadata(meta) {
182 return meta.expression !== undefined;
183}
184function getInjectFn(target) {
185 switch (target) {
186 case FactoryTarget.Component:
187 case FactoryTarget.Directive:
188 case FactoryTarget.Pipe:
189 return R3.directiveInject;
190 case FactoryTarget.NgModule:
191 case FactoryTarget.Injectable:
192 default:
193 return R3.inject;
194 }
195}
196//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_factory.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_factory.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport {InjectFlags} from '../core';\nimport * as o from '../output/output_ast';\nimport {Identifiers as R3} from '../render3/r3_identifiers';\nimport {R3CompiledExpression, R3Reference, typeWithParameters} from './util';\n\n\n/**\n * Metadata required by the factory generator to generate a `factory` function for a type.\n */\nexport interface R3ConstructorFactoryMetadata {\n  /**\n   * String name of the type being generated (used to name the factory function).\n   */\n  name: string;\n\n  /**\n   * An expression representing the interface type being constructed.\n   */\n  type: R3Reference;\n\n  /**\n   * An expression representing the constructor type, intended for use within a class definition\n   * itself.\n   *\n   * This can differ from the outer `type` if the class is being compiled by ngcc and is inside\n   * an IIFE structure that uses a different name internally.\n   */\n  internalType: o.Expression;\n\n  /** Number of arguments for the `type`. */\n  typeArgumentCount: number;\n\n  /**\n   * Regardless of whether `fnOrClass` is a constructor function or a user-defined factory, it\n   * may have 0 or more parameters, which will be injected according to the `R3DependencyMetadata`\n   * for those parameters. If this is `null`, then the type's constructor is nonexistent and will\n   * be inherited from `fnOrClass` which is interpreted as the current type. If this is `'invalid'`,\n   * then one or more of the parameters wasn't resolvable and any attempt to use these deps will\n   * result in a runtime error.\n   */\n  deps: R3DependencyMetadata[]|'invalid'|null;\n\n  /**\n   * Type of the target being created by the factory.\n   */\n  target: FactoryTarget;\n}\n\nexport enum R3FactoryDelegateType {\n  Class = 0,\n  Function = 1,\n}\n\nexport interface R3DelegatedFnOrClassMetadata extends R3ConstructorFactoryMetadata {\n  delegate: o.Expression;\n  delegateType: R3FactoryDelegateType;\n  delegateDeps: R3DependencyMetadata[];\n}\n\nexport interface R3ExpressionFactoryMetadata extends R3ConstructorFactoryMetadata {\n  expression: o.Expression;\n}\n\nexport type R3FactoryMetadata =\n    R3ConstructorFactoryMetadata|R3DelegatedFnOrClassMetadata|R3ExpressionFactoryMetadata;\n\nexport enum FactoryTarget {\n  Directive = 0,\n  Component = 1,\n  Injectable = 2,\n  Pipe = 3,\n  NgModule = 4,\n}\n\nexport interface R3DependencyMetadata {\n  /**\n   * An expression representing the token or value to be injected.\n   * Or `null` if the dependency could not be resolved - making it invalid.\n   */\n  token: o.Expression|null;\n\n  /**\n   * If an @Attribute decorator is present, this is the literal type of the attribute name, or\n   * the unknown type if no literal type is available (e.g. the attribute name is an expression).\n   * Otherwise it is null;\n   */\n  attributeNameType: o.Expression|null;\n\n  /**\n   * Whether the dependency has an @Host qualifier.\n   */\n  host: boolean;\n\n  /**\n   * Whether the dependency has an @Optional qualifier.\n   */\n  optional: boolean;\n\n  /**\n   * Whether the dependency has an @Self qualifier.\n   */\n  self: boolean;\n\n  /**\n   * Whether the dependency has an @SkipSelf qualifier.\n   */\n  skipSelf: boolean;\n}\n\n/**\n * Construct a factory function expression for the given `R3FactoryMetadata`.\n */\nexport function compileFactoryFunction(meta: R3FactoryMetadata): R3CompiledExpression {\n  const t = o.variable('t');\n  let baseFactoryVar: o.ReadVarExpr|null = null;\n\n  // The type to instantiate via constructor invocation. If there is no delegated factory, meaning\n  // this type is always created by constructor invocation, then this is the type-to-create\n  // parameter provided by the user (t) if specified, or the current type if not. If there is a\n  // delegated factory (which is used to create the current type) then this is only the type-to-\n  // create parameter (t).\n  const typeForCtor = !isDelegatedFactoryMetadata(meta) ?\n      new o.BinaryOperatorExpr(o.BinaryOperator.Or, t, meta.internalType) :\n      t;\n\n  let ctorExpr: o.Expression|null = null;\n  if (meta.deps !== null) {\n    // There is a constructor (either explicitly or implicitly defined).\n    if (meta.deps !== 'invalid') {\n      ctorExpr = new o.InstantiateExpr(typeForCtor, injectDependencies(meta.deps, meta.target));\n    }\n  } else {\n    // There is no constructor, use the base class' factory to construct typeForCtor.\n    baseFactoryVar = o.variable(`ɵ${meta.name}_BaseFactory`);\n    ctorExpr = baseFactoryVar.callFn([typeForCtor]);\n  }\n\n  const body: o.Statement[] = [];\n  let retExpr: o.Expression|null = null;\n\n  function makeConditionalFactory(nonCtorExpr: o.Expression): o.ReadVarExpr {\n    const r = o.variable('r');\n    body.push(r.set(o.NULL_EXPR).toDeclStmt());\n    const ctorStmt = ctorExpr !== null ? r.set(ctorExpr).toStmt() :\n                                         o.importExpr(R3.invalidFactory).callFn([]).toStmt();\n    body.push(o.ifStmt(t, [ctorStmt], [r.set(nonCtorExpr).toStmt()]));\n    return r;\n  }\n\n  if (isDelegatedFactoryMetadata(meta)) {\n    // This type is created with a delegated factory. If a type parameter is not specified, call\n    // the factory instead.\n    const delegateArgs = injectDependencies(meta.delegateDeps, meta.target);\n    // Either call `new delegate(...)` or `delegate(...)` depending on meta.delegateType.\n    const factoryExpr = new (\n        meta.delegateType === R3FactoryDelegateType.Class ?\n            o.InstantiateExpr :\n            o.InvokeFunctionExpr)(meta.delegate, delegateArgs);\n    retExpr = makeConditionalFactory(factoryExpr);\n  } else if (isExpressionFactoryMetadata(meta)) {\n    // TODO(alxhub): decide whether to lower the value here or in the caller\n    retExpr = makeConditionalFactory(meta.expression);\n  } else {\n    retExpr = ctorExpr;\n  }\n\n\n  if (retExpr === null) {\n    // The expression cannot be formed so render an `ɵɵinvalidFactory()` call.\n    body.push(o.importExpr(R3.invalidFactory).callFn([]).toStmt());\n  } else if (baseFactoryVar !== null) {\n    // This factory uses a base factory, so call `ɵɵgetInheritedFactory()` to compute it.\n    const getInheritedFactoryCall =\n        o.importExpr(R3.getInheritedFactory).callFn([meta.internalType]);\n    // Memoize the base factoryFn: `baseFactory || (baseFactory = ɵɵgetInheritedFactory(...))`\n    const baseFactory = new o.BinaryOperatorExpr(\n        o.BinaryOperator.Or, baseFactoryVar, baseFactoryVar.set(getInheritedFactoryCall));\n    body.push(new o.ReturnStatement(baseFactory.callFn([typeForCtor])));\n  } else {\n    // This is straightforward factory, just return it.\n    body.push(new o.ReturnStatement(retExpr));\n  }\n\n  let factoryFn: o.Expression = o.fn(\n      [new o.FnParam('t', o.DYNAMIC_TYPE)], body, o.INFERRED_TYPE, undefined,\n      `${meta.name}_Factory`);\n\n  if (baseFactoryVar !== null) {\n    // There is a base factory variable so wrap its declaration along with the factory function into\n    // an IIFE.\n    factoryFn = o.fn([], [\n                   new o.DeclareVarStmt(baseFactoryVar.name!), new o.ReturnStatement(factoryFn)\n                 ]).callFn([], /* sourceSpan */ undefined, /* pure */ true);\n  }\n\n  return {\n    expression: factoryFn,\n    statements: [],\n    type: createFactoryType(meta),\n  };\n}\n\nexport function createFactoryType(meta: R3FactoryMetadata) {\n  const ctorDepsType =\n      meta.deps !== null && meta.deps !== 'invalid' ? createCtorDepsType(meta.deps) : o.NONE_TYPE;\n  return o.expressionType(o.importExpr(\n      R3.FactoryDeclaration,\n      [typeWithParameters(meta.type.type, meta.typeArgumentCount), ctorDepsType]));\n}\n\nfunction injectDependencies(deps: R3DependencyMetadata[], target: FactoryTarget): o.Expression[] {\n  return deps.map((dep, index) => compileInjectDependency(dep, target, index));\n}\n\nfunction compileInjectDependency(\n    dep: R3DependencyMetadata, target: FactoryTarget, index: number): o.Expression {\n  // Interpret the dependency according to its resolved type.\n  if (dep.token === null) {\n    return o.importExpr(R3.invalidFactoryDep).callFn([o.literal(index)]);\n  } else if (dep.attributeNameType === null) {\n    // Build up the injection flags according to the metadata.\n    const flags = InjectFlags.Default | (dep.self ? InjectFlags.Self : 0) |\n        (dep.skipSelf ? InjectFlags.SkipSelf : 0) | (dep.host ? InjectFlags.Host : 0) |\n        (dep.optional ? InjectFlags.Optional : 0) |\n        (target === FactoryTarget.Pipe ? InjectFlags.ForPipe : 0);\n\n    // If this dependency is optional or otherwise has non-default flags, then additional\n    // parameters describing how to inject the dependency must be passed to the inject function\n    // that's being used.\n    let flagsParam: o.LiteralExpr|null =\n        (flags !== InjectFlags.Default || dep.optional) ? o.literal(flags) : null;\n\n    // Build up the arguments to the injectFn call.\n    const injectArgs = [dep.token];\n    if (flagsParam) {\n      injectArgs.push(flagsParam);\n    }\n    const injectFn = getInjectFn(target);\n    return o.importExpr(injectFn).callFn(injectArgs);\n  } else {\n    // The `dep.attributeTypeName` value is defined, which indicates that this is an `@Attribute()`\n    // type dependency. For the generated JS we still want to use the `dep.token` value in case the\n    // name given for the attribute is not a string literal. For example given `@Attribute(foo())`,\n    // we want to generate `ɵɵinjectAttribute(foo())`.\n    //\n    // The `dep.attributeTypeName` is only actually used (in `createCtorDepType()`) to generate\n    // typings.\n    return o.importExpr(R3.injectAttribute).callFn([dep.token]);\n  }\n}\n\nfunction createCtorDepsType(deps: R3DependencyMetadata[]): o.Type {\n  let hasTypes = false;\n  const attributeTypes = deps.map(dep => {\n    const type = createCtorDepType(dep);\n    if (type !== null) {\n      hasTypes = true;\n      return type;\n    } else {\n      return o.literal(null);\n    }\n  });\n\n  if (hasTypes) {\n    return o.expressionType(o.literalArr(attributeTypes));\n  } else {\n    return o.NONE_TYPE;\n  }\n}\n\nfunction createCtorDepType(dep: R3DependencyMetadata): o.LiteralMapExpr|null {\n  const entries: {key: string, quoted: boolean, value: o.Expression}[] = [];\n\n  if (dep.attributeNameType !== null) {\n    entries.push({key: 'attribute', value: dep.attributeNameType, quoted: false});\n  }\n  if (dep.optional) {\n    entries.push({key: 'optional', value: o.literal(true), quoted: false});\n  }\n  if (dep.host) {\n    entries.push({key: 'host', value: o.literal(true), quoted: false});\n  }\n  if (dep.self) {\n    entries.push({key: 'self', value: o.literal(true), quoted: false});\n  }\n  if (dep.skipSelf) {\n    entries.push({key: 'skipSelf', value: o.literal(true), quoted: false});\n  }\n\n  return entries.length > 0 ? o.literalMap(entries) : null;\n}\n\nexport function isDelegatedFactoryMetadata(meta: R3FactoryMetadata):\n    meta is R3DelegatedFnOrClassMetadata {\n  return (meta as any).delegateType !== undefined;\n}\n\nexport function isExpressionFactoryMetadata(meta: R3FactoryMetadata):\n    meta is R3ExpressionFactoryMetadata {\n  return (meta as any).expression !== undefined;\n}\n\nfunction getInjectFn(target: FactoryTarget): o.ExternalReference {\n  switch (target) {\n    case FactoryTarget.Component:\n    case FactoryTarget.Directive:\n    case FactoryTarget.Pipe:\n      return R3.directiveInject;\n    case FactoryTarget.NgModule:\n    case FactoryTarget.Injectable:\n    default:\n      return R3.inject;\n  }\n}\n"]}
\No newline at end of file