UNPKG

19.9 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright Google LLC All Rights Reserved.
4 *
5 * Use of this source code is governed by an MIT-style license that can be
6 * found in the LICENSE file at https://angular.io/license
7 */
8import * as o from './output/output_ast';
9import { compileFactoryFunction, FactoryTarget, R3FactoryDelegateType } from './render3/r3_factory';
10import { Identifiers } from './render3/r3_identifiers';
11import { convertFromMaybeForwardRefExpression, typeWithParameters } from './render3/util';
12import { DefinitionMap } from './render3/view/util';
13export function compileInjectable(meta, resolveForwardRefs) {
14 let result = null;
15 const factoryMeta = {
16 name: meta.name,
17 type: meta.type,
18 internalType: meta.internalType,
19 typeArgumentCount: meta.typeArgumentCount,
20 deps: [],
21 target: FactoryTarget.Injectable,
22 };
23 if (meta.useClass !== undefined) {
24 // meta.useClass has two modes of operation. Either deps are specified, in which case `new` is
25 // used to instantiate the class with dependencies injected, or deps are not specified and
26 // the factory of the class is used to instantiate it.
27 //
28 // A special case exists for useClass: Type where Type is the injectable type itself and no
29 // deps are specified, in which case 'useClass' is effectively ignored.
30 const useClassOnSelf = meta.useClass.expression.isEquivalent(meta.internalType);
31 let deps = undefined;
32 if (meta.deps !== undefined) {
33 deps = meta.deps;
34 }
35 if (deps !== undefined) {
36 // factory: () => new meta.useClass(...deps)
37 result = compileFactoryFunction({
38 ...factoryMeta,
39 delegate: meta.useClass.expression,
40 delegateDeps: deps,
41 delegateType: R3FactoryDelegateType.Class,
42 });
43 }
44 else if (useClassOnSelf) {
45 result = compileFactoryFunction(factoryMeta);
46 }
47 else {
48 result = {
49 statements: [],
50 expression: delegateToFactory(meta.type.value, meta.useClass.expression, resolveForwardRefs)
51 };
52 }
53 }
54 else if (meta.useFactory !== undefined) {
55 if (meta.deps !== undefined) {
56 result = compileFactoryFunction({
57 ...factoryMeta,
58 delegate: meta.useFactory,
59 delegateDeps: meta.deps || [],
60 delegateType: R3FactoryDelegateType.Function,
61 });
62 }
63 else {
64 result = {
65 statements: [],
66 expression: o.fn([], [new o.ReturnStatement(meta.useFactory.callFn([]))])
67 };
68 }
69 }
70 else if (meta.useValue !== undefined) {
71 // Note: it's safe to use `meta.useValue` instead of the `USE_VALUE in meta` check used for
72 // client code because meta.useValue is an Expression which will be defined even if the actual
73 // value is undefined.
74 result = compileFactoryFunction({
75 ...factoryMeta,
76 expression: meta.useValue.expression,
77 });
78 }
79 else if (meta.useExisting !== undefined) {
80 // useExisting is an `inject` call on the existing token.
81 result = compileFactoryFunction({
82 ...factoryMeta,
83 expression: o.importExpr(Identifiers.inject).callFn([meta.useExisting.expression]),
84 });
85 }
86 else {
87 result = {
88 statements: [],
89 expression: delegateToFactory(meta.type.value, meta.internalType, resolveForwardRefs)
90 };
91 }
92 const token = meta.internalType;
93 const injectableProps = new DefinitionMap();
94 injectableProps.set('token', token);
95 injectableProps.set('factory', result.expression);
96 // Only generate providedIn property if it has a non-null value
97 if (meta.providedIn.expression.value !== null) {
98 injectableProps.set('providedIn', convertFromMaybeForwardRefExpression(meta.providedIn));
99 }
100 const expression = o.importExpr(Identifiers.ɵɵdefineInjectable)
101 .callFn([injectableProps.toLiteralMap()], undefined, true);
102 return {
103 expression,
104 type: createInjectableType(meta),
105 statements: result.statements,
106 };
107}
108export function createInjectableType(meta) {
109 return new o.ExpressionType(o.importExpr(Identifiers.InjectableDeclaration, [typeWithParameters(meta.type.type, meta.typeArgumentCount)]));
110}
111function delegateToFactory(type, internalType, unwrapForwardRefs) {
112 if (type.node === internalType.node) {
113 // The types are the same, so we can simply delegate directly to the type's factory.
114 // ```
115 // factory: type.ɵfac
116 // ```
117 return internalType.prop('ɵfac');
118 }
119 if (!unwrapForwardRefs) {
120 // The type is not wrapped in a `forwardRef()`, so we create a simple factory function that
121 // accepts a sub-type as an argument.
122 // ```
123 // factory: function(t) { return internalType.ɵfac(t); }
124 // ```
125 return createFactoryFunction(internalType);
126 }
127 // The internalType is actually wrapped in a `forwardRef()` so we need to resolve that before
128 // calling its factory.
129 // ```
130 // factory: function(t) { return core.resolveForwardRef(type).ɵfac(t); }
131 // ```
132 const unwrappedType = o.importExpr(Identifiers.resolveForwardRef).callFn([internalType]);
133 return createFactoryFunction(unwrappedType);
134}
135function createFactoryFunction(type) {
136 return o.fn([new o.FnParam('t', o.DYNAMIC_TYPE)], [new o.ReturnStatement(type.prop('ɵfac').callFn([o.variable('t')]))]);
137}
138//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"injectable_compiler_2.js","sourceRoot":"","sources":["../../../../../../packages/compiler/src/injectable_compiler_2.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 */\n\nimport * as o from './output/output_ast';\nimport {compileFactoryFunction, FactoryTarget, R3DependencyMetadata, R3FactoryDelegateType, R3FactoryMetadata} from './render3/r3_factory';\nimport {Identifiers} from './render3/r3_identifiers';\nimport {convertFromMaybeForwardRefExpression, ForwardRefHandling, generateForwardRef, MaybeForwardRefExpression, R3CompiledExpression, R3Reference, typeWithParameters} from './render3/util';\nimport {DefinitionMap} from './render3/view/util';\n\nexport interface R3InjectableMetadata {\n  name: string;\n  type: R3Reference;\n  internalType: o.Expression;\n  typeArgumentCount: number;\n  providedIn: MaybeForwardRefExpression;\n  useClass?: MaybeForwardRefExpression;\n  useFactory?: o.Expression;\n  useExisting?: MaybeForwardRefExpression;\n  useValue?: MaybeForwardRefExpression;\n  deps?: R3DependencyMetadata[];\n}\n\nexport function compileInjectable(\n    meta: R3InjectableMetadata, resolveForwardRefs: boolean): R3CompiledExpression {\n  let result: {expression: o.Expression, statements: o.Statement[]}|null = null;\n\n  const factoryMeta: R3FactoryMetadata = {\n    name: meta.name,\n    type: meta.type,\n    internalType: meta.internalType,\n    typeArgumentCount: meta.typeArgumentCount,\n    deps: [],\n    target: FactoryTarget.Injectable,\n  };\n\n  if (meta.useClass !== undefined) {\n    // meta.useClass has two modes of operation. Either deps are specified, in which case `new` is\n    // used to instantiate the class with dependencies injected, or deps are not specified and\n    // the factory of the class is used to instantiate it.\n    //\n    // A special case exists for useClass: Type where Type is the injectable type itself and no\n    // deps are specified, in which case 'useClass' is effectively ignored.\n\n    const useClassOnSelf = meta.useClass.expression.isEquivalent(meta.internalType);\n    let deps: R3DependencyMetadata[]|undefined = undefined;\n    if (meta.deps !== undefined) {\n      deps = meta.deps;\n    }\n\n    if (deps !== undefined) {\n      // factory: () => new meta.useClass(...deps)\n      result = compileFactoryFunction({\n        ...factoryMeta,\n        delegate: meta.useClass.expression,\n        delegateDeps: deps,\n        delegateType: R3FactoryDelegateType.Class,\n      });\n    } else if (useClassOnSelf) {\n      result = compileFactoryFunction(factoryMeta);\n    } else {\n      result = {\n        statements: [],\n        expression: delegateToFactory(\n            meta.type.value as o.WrappedNodeExpr<any>,\n            meta.useClass.expression as o.WrappedNodeExpr<any>, resolveForwardRefs)\n      };\n    }\n  } else if (meta.useFactory !== undefined) {\n    if (meta.deps !== undefined) {\n      result = compileFactoryFunction({\n        ...factoryMeta,\n        delegate: meta.useFactory,\n        delegateDeps: meta.deps || [],\n        delegateType: R3FactoryDelegateType.Function,\n      });\n    } else {\n      result = {\n        statements: [],\n        expression: o.fn([], [new o.ReturnStatement(meta.useFactory.callFn([]))])\n      };\n    }\n  } else if (meta.useValue !== undefined) {\n    // Note: it's safe to use `meta.useValue` instead of the `USE_VALUE in meta` check used for\n    // client code because meta.useValue is an Expression which will be defined even if the actual\n    // value is undefined.\n    result = compileFactoryFunction({\n      ...factoryMeta,\n      expression: meta.useValue.expression,\n    });\n  } else if (meta.useExisting !== undefined) {\n    // useExisting is an `inject` call on the existing token.\n    result = compileFactoryFunction({\n      ...factoryMeta,\n      expression: o.importExpr(Identifiers.inject).callFn([meta.useExisting.expression]),\n    });\n  } else {\n    result = {\n      statements: [],\n      expression: delegateToFactory(\n          meta.type.value as o.WrappedNodeExpr<any>, meta.internalType as o.WrappedNodeExpr<any>,\n          resolveForwardRefs)\n    };\n  }\n\n  const token = meta.internalType;\n\n  const injectableProps =\n      new DefinitionMap<{token: o.Expression, factory: o.Expression, providedIn: o.Expression}>();\n  injectableProps.set('token', token);\n  injectableProps.set('factory', result.expression);\n\n  // Only generate providedIn property if it has a non-null value\n  if ((meta.providedIn.expression as o.LiteralExpr).value !== null) {\n    injectableProps.set('providedIn', convertFromMaybeForwardRefExpression(meta.providedIn));\n  }\n\n  const expression = o.importExpr(Identifiers.ɵɵdefineInjectable)\n                         .callFn([injectableProps.toLiteralMap()], undefined, true);\n  return {\n    expression,\n    type: createInjectableType(meta),\n    statements: result.statements,\n  };\n}\n\nexport function createInjectableType(meta: R3InjectableMetadata) {\n  return new o.ExpressionType(o.importExpr(\n      Identifiers.InjectableDeclaration,\n      [typeWithParameters(meta.type.type, meta.typeArgumentCount)]));\n}\n\nfunction delegateToFactory(\n    type: o.WrappedNodeExpr<any>, internalType: o.WrappedNodeExpr<any>,\n    unwrapForwardRefs: boolean): o.Expression {\n  if (type.node === internalType.node) {\n    // The types are the same, so we can simply delegate directly to the type's factory.\n    // ```\n    // factory: type.ɵfac\n    // ```\n    return internalType.prop('ɵfac');\n  }\n\n  if (!unwrapForwardRefs) {\n    // The type is not wrapped in a `forwardRef()`, so we create a simple factory function that\n    // accepts a sub-type as an argument.\n    // ```\n    // factory: function(t) { return internalType.ɵfac(t); }\n    // ```\n    return createFactoryFunction(internalType);\n  }\n\n  // The internalType is actually wrapped in a `forwardRef()` so we need to resolve that before\n  // calling its factory.\n  // ```\n  // factory: function(t) { return core.resolveForwardRef(type).ɵfac(t); }\n  // ```\n  const unwrappedType = o.importExpr(Identifiers.resolveForwardRef).callFn([internalType]);\n  return createFactoryFunction(unwrappedType);\n}\n\nfunction createFactoryFunction(type: o.Expression): o.FunctionExpr {\n  return o.fn(\n      [new o.FnParam('t', o.DYNAMIC_TYPE)],\n      [new o.ReturnStatement(type.prop('ɵfac').callFn([o.variable('t')]))]);\n}\n"]}
\No newline at end of file