refactor(compiler-cli): support namespaced references (#39346)
The compiler uses a `Reference` abstraction to refer to TS nodes that it needs to refer to from other parts of the source. Such references keep track of any identifiers that represent the referenced node. Prior to this commit, the compiler (and specifically `ReferenceEmitter` classes) assumed that the reference identifiers are always free standing. In other words a reference identifier would be an expression like `FooDirective` in the expression `class FooDirective {}`. But in UMD/CommonJS source, a reference can actually refer to an "exports" declaration of the form `exports.FooDirective = ...`. In such cases the `FooDirective` identifier is not free-standing since it is part of a property access, so the `ReferenceEmitter` should take this into account when emitting an expression that refers to such a `Reference`. This commit changes the `LocalIdentifierStrategy` reference emitter so that if the `node` being referenced is not a declaration itself and is in the current file, then it should be used directly, rather than trying to use one of its identifiers. PR Close #39346
This commit is contained in:
parent
413b55273b
commit
27a4adebcb
|
@ -104,13 +104,23 @@ export class ReferenceEmitter {
|
||||||
*/
|
*/
|
||||||
export class LocalIdentifierStrategy implements ReferenceEmitStrategy {
|
export class LocalIdentifierStrategy implements ReferenceEmitStrategy {
|
||||||
emit(ref: Reference<ts.Node>, context: ts.SourceFile, importFlags: ImportFlags): Expression|null {
|
emit(ref: Reference<ts.Node>, context: ts.SourceFile, importFlags: ImportFlags): Expression|null {
|
||||||
|
const refSf = getSourceFile(ref.node);
|
||||||
|
|
||||||
// If the emitter has specified ForceNewImport, then LocalIdentifierStrategy should not use a
|
// If the emitter has specified ForceNewImport, then LocalIdentifierStrategy should not use a
|
||||||
// local identifier at all, *except* in the source file where the node is actually declared.
|
// local identifier at all, *except* in the source file where the node is actually declared.
|
||||||
if (importFlags & ImportFlags.ForceNewImport &&
|
if (importFlags & ImportFlags.ForceNewImport && refSf !== context) {
|
||||||
getSourceFile(ref.node) !== getSourceFile(context)) {
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If referenced node is not an actual TS declaration (e.g. `class Foo` or `function foo() {}`,
|
||||||
|
// etc) and it is in the current file then just use it directly.
|
||||||
|
// This is important because the reference could be a property access (e.g. `exports.foo`). In
|
||||||
|
// such a case, the reference's `identities` property would be `[foo]`, which would result in an
|
||||||
|
// invalid emission of a free-standing `foo` identifier, rather than `exports.foo`.
|
||||||
|
if (!isDeclaration(ref.node) && refSf === context) {
|
||||||
|
return new WrappedNodeExpr(ref.node);
|
||||||
|
}
|
||||||
|
|
||||||
// A Reference can have multiple identities in different files, so it may already have an
|
// A Reference can have multiple identities in different files, so it may already have an
|
||||||
// Identifier in the requested context file.
|
// Identifier in the requested context file.
|
||||||
const identifier = ref.getIdentityIn(context);
|
const identifier = ref.getIdentityIn(context);
|
||||||
|
|
Loading…
Reference in New Issue