fix(language-service): Function alias should be callable (#33782)

This commit fixes a long standing bug whereby a template variable that
gets initialized to a class method gets resolved to the Any type, thus
when it is called the language service produces error "Member X is not
callable".

PR closes https://github.com/angular/angular/issues/16643
PR closes https://github.com/angular/vscode-ng-language-service/issues/234

PR Close #33782
This commit is contained in:
Keen Yee Liau 2019-11-12 17:40:56 -08:00 committed by Alex Rickabaugh
parent ab0bcee144
commit 1d3aae6f92
2 changed files with 12 additions and 26 deletions

View File

@ -90,29 +90,6 @@ function getDefinitionOf(info: DiagnosticTemplateInfo, ast: TemplateAst): Defini
} }
} }
/**
* Resolve the specified `variable` from the `directives` list and return the
* corresponding symbol. If resolution fails, return the `any` type.
* @param variable template variable to resolve
* @param directives template context
* @param query
*/
function findSymbolForVariableInDirectives(
variable: VariableAst, directives: DirectiveAst[], query: SymbolQuery): Symbol {
for (const d of directives) {
// Get the symbol table for the directive's StaticSymbol
const table = query.getTemplateContext(d.directive.type.reference);
if (!table) {
continue;
}
const symbol = table.get(variable.value);
if (symbol) {
return symbol;
}
}
return query.getBuiltinType(BuiltinType.Any);
}
/** /**
* Resolve all variable declarations in a template by traversing the specified * Resolve all variable declarations in a template by traversing the specified
* `path`. * `path`.
@ -126,9 +103,8 @@ function getVarDeclarations(
if (!(current instanceof EmbeddedTemplateAst)) { if (!(current instanceof EmbeddedTemplateAst)) {
continue; continue;
} }
const {directives, variables} = current; for (const variable of current.variables) {
for (const variable of variables) { let symbol = info.members.get(variable.value) || info.query.getBuiltinType(BuiltinType.Any);
let symbol = findSymbolForVariableInDirectives(variable, directives, info.query);
const kind = info.query.getTypeKind(symbol); const kind = info.query.getTypeKind(symbol);
if (kind === BuiltinType.Any || kind === BuiltinType.Unbound) { if (kind === BuiltinType.Any || kind === BuiltinType.Unbound) {
// For special cases such as ngFor and ngIf, the any type is not very useful. // For special cases such as ngFor and ngIf, the any type is not very useful.

View File

@ -109,6 +109,16 @@ describe('diagnostics', () => {
.toBe(`Identifier 'age' is not defined. 'Hero' does not contain such a member`); .toBe(`Identifier 'age' is not defined. 'Hero' does not contain such a member`);
}); });
it('should not report error for variable initialized as class method', () => {
mockHost.override(TEST_TEMPLATE, `
<ng-template let-greet="myClick">
<span (click)="greet()"></span>
</ng-template>
`);
const diagnostics = ngLS.getDiagnostics(TEST_TEMPLATE);
expect(diagnostics).toEqual([]);
});
describe('in expression-cases.ts', () => { describe('in expression-cases.ts', () => {
it('should report access to an unknown field', () => { it('should report access to an unknown field', () => {
const diags = ngLS.getDiagnostics(EXPRESSION_CASES).map(d => d.messageText); const diags = ngLS.getDiagnostics(EXPRESSION_CASES).map(d => d.messageText);