1 | import * as o from '../output/output_ast';
|
---|
2 | import { Identifiers as R3 } from '../render3/r3_identifiers';
|
---|
3 | import { typeWithParameters } from './util';
|
---|
4 | export var R3FactoryDelegateType;
|
---|
5 | (function (R3FactoryDelegateType) {
|
---|
6 | R3FactoryDelegateType[R3FactoryDelegateType["Class"] = 0] = "Class";
|
---|
7 | R3FactoryDelegateType[R3FactoryDelegateType["Function"] = 1] = "Function";
|
---|
8 | })(R3FactoryDelegateType || (R3FactoryDelegateType = {}));
|
---|
9 | export 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 | */
|
---|
20 | export 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 | }
|
---|
99 | export 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 | }
|
---|
103 | function injectDependencies(deps, target) {
|
---|
104 | return deps.map((dep, index) => compileInjectDependency(dep, target, index));
|
---|
105 | }
|
---|
106 | function 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 | }
|
---|
140 | function 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 | }
|
---|
159 | function 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 | }
|
---|
178 | export function isDelegatedFactoryMetadata(meta) {
|
---|
179 | return meta.delegateType !== undefined;
|
---|
180 | }
|
---|
181 | export function isExpressionFactoryMetadata(meta) {
|
---|
182 | return meta.expression !== undefined;
|
---|
183 | }
|
---|
184 | function 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"]} |
---|