fix(language-service): Make Definition and QuickInfo compatible with TS LS (#31972)

Now that the Angular LS is a proper tsserver plugin, it does not make
sense for it to maintain its own language service API.

This is part one of the effort to remove our custom LanguageService
interface.
This interface is cumbersome because we have to do two transformations:
  ng def -> ts def -> lsp definition

The TS LS interface is more comprehensive, so this allows the Angular LS
to return more information.

PR Close #31972
This commit is contained in:
Keen Yee Liau 2019-08-01 13:07:32 -07:00 committed by Alex Rickabaugh
parent e906a4f0d8
commit a8e2ee1343
10 changed files with 442 additions and 260 deletions

View File

@ -20,12 +20,12 @@
], ],
"textSpan": { "textSpan": {
"start": { "start": {
"line": 7, "line": 5,
"offset": 30 "offset": 26
}, },
"end": { "end": {
"line": 7, "line": 5,
"offset": 47 "offset": 30
} }
} }
} }

View File

@ -5,7 +5,7 @@
"request_seq": 2, "request_seq": 2,
"success": true, "success": true,
"body": { "body": {
"kind": "", "kind": "property",
"kindModifiers": "", "kindModifiers": "",
"start": { "start": {
"line": 5, "line": 5,
@ -15,7 +15,7 @@
"line": 5, "line": 5,
"offset": 30 "offset": 30
}, },
"displayString": "property name of AppComponent", "displayString": "(property) AppComponent.name",
"documentation": "", "documentation": "",
"tags": [] "tags": []
} }

View File

@ -6,28 +6,50 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import * as tss from 'typescript/lib/tsserverlibrary'; import * as ts from 'typescript'; // used as value and is provided at runtime
import {TemplateInfo} from './common'; import {TemplateInfo} from './common';
import {locateSymbol} from './locate_symbol'; import {locateSymbol} from './locate_symbol';
import {Location} from './types'; import {Span} from './types';
export function getDefinition(info: TemplateInfo): Location[]|undefined { /**
const result = locateSymbol(info); * Convert Angular Span to TypeScript TextSpan. Angular Span has 'start' and
return result && result.symbol.definition; * 'end' whereas TS TextSpan has 'start' and 'length'.
} * @param span Angular Span
*/
export function ngLocationToTsDefinitionInfo(loc: Location): tss.DefinitionInfo { function ngSpanToTsTextSpan(span: Span): ts.TextSpan {
return { return {
fileName: loc.fileName, start: span.start,
textSpan: { length: span.end - span.start,
start: loc.span.start, };
length: loc.span.end - loc.span.start, }
},
// TODO(kyliau): Provide more useful info for name, kind and containerKind export function getDefinitionAndBoundSpan(info: TemplateInfo): ts.DefinitionInfoAndBoundSpan|
name: '', // should be name of symbol but we don't have enough information here. undefined {
kind: tss.ScriptElementKind.unknown, const symbolInfo = locateSymbol(info);
containerName: loc.fileName, if (!symbolInfo) {
containerKind: tss.ScriptElementKind.unknown, return;
}
const textSpan = ngSpanToTsTextSpan(symbolInfo.span);
const {symbol} = symbolInfo;
const {container, definition: locations} = symbol;
if (!locations || !locations.length) {
// symbol.definition is really the locations of the symbol. There could be
// more than one. No meaningful info could be provided without any location.
return {textSpan};
}
const containerKind = container ? container.kind : ts.ScriptElementKind.unknown;
const containerName = container ? container.name : '';
const definitions = locations.map((location) => {
return {
kind: symbol.kind as ts.ScriptElementKind,
name: symbol.name,
containerKind: containerKind as ts.ScriptElementKind,
containerName: containerName,
textSpan: ngSpanToTsTextSpan(location.span),
fileName: location.fileName,
};
});
return {
definitions, textSpan,
}; };
} }

View File

@ -6,23 +6,42 @@
* found in the LICENSE file at https://angular.io/license * found in the LICENSE file at https://angular.io/license
*/ */
import * as ts from 'typescript';
import {TemplateInfo} from './common'; import {TemplateInfo} from './common';
import {locateSymbol} from './locate_symbol'; import {locateSymbol} from './locate_symbol';
import {Hover, HoverTextSection, Symbol} from './types';
export function getHover(info: TemplateInfo): Hover|undefined { // Reverse mappings of enum would generate strings
const result = locateSymbol(info); const SYMBOL_SPACE = ts.SymbolDisplayPartKind[ts.SymbolDisplayPartKind.space];
if (result) { const SYMBOL_PUNC = ts.SymbolDisplayPartKind[ts.SymbolDisplayPartKind.punctuation];
return {text: hoverTextOf(result.symbol), span: result.span};
export function getHover(info: TemplateInfo): ts.QuickInfo|undefined {
const symbolInfo = locateSymbol(info);
if (!symbolInfo) {
return;
} }
const {symbol, span} = symbolInfo;
const containerDisplayParts: ts.SymbolDisplayPart[] = symbol.container ?
[
{text: symbol.container.name, kind: symbol.container.kind},
{text: '.', kind: SYMBOL_PUNC},
] :
[];
return {
kind: symbol.kind as ts.ScriptElementKind,
kindModifiers: '', // kindModifier info not available on 'ng.Symbol'
textSpan: {
start: span.start,
length: span.end - span.start,
},
// this would generate a string like '(property) ClassX.propY'
// 'kind' in displayParts does not really matter because it's dropped when
// displayParts get converted to string.
displayParts: [
{text: '(', kind: SYMBOL_PUNC}, {text: symbol.kind, kind: symbol.kind},
{text: ')', kind: SYMBOL_PUNC}, {text: ' ', kind: SYMBOL_SPACE}, ...containerDisplayParts,
{text: symbol.name, kind: symbol.kind},
// TODO: Append type info as well, but Symbol doesn't expose that!
// Ideally hover text should be like '(property) ClassX.propY: string'
],
};
} }
function hoverTextOf(symbol: Symbol): HoverTextSection[] {
const result: HoverTextSection[] =
[{text: symbol.kind}, {text: ' '}, {text: symbol.name, language: symbol.language}];
const container = symbol.container;
if (container) {
result.push({text: ' of '}, {text: container.name, language: container.language});
}
return result;
}

View File

@ -8,9 +8,9 @@
import {CompileMetadataResolver, CompilePipeSummary} from '@angular/compiler'; import {CompileMetadataResolver, CompilePipeSummary} from '@angular/compiler';
import {DiagnosticTemplateInfo, getTemplateExpressionDiagnostics} from '@angular/compiler-cli/src/language_services'; import {DiagnosticTemplateInfo, getTemplateExpressionDiagnostics} from '@angular/compiler-cli/src/language_services';
import * as tss from 'typescript/lib/tsserverlibrary';
import {getTemplateCompletions} from './completions'; import {getTemplateCompletions} from './completions';
import {getDefinition} from './definitions'; import {getDefinitionAndBoundSpan} from './definitions';
import {getDeclarationDiagnostics} from './diagnostics'; import {getDeclarationDiagnostics} from './diagnostics';
import {getHover} from './hover'; import {getHover} from './hover';
import {Completion, Diagnostic, DiagnosticKind, Diagnostics, Hover, LanguageService, LanguageServiceHost, Location, Span, TemplateSource} from './types'; import {Completion, Diagnostic, DiagnosticKind, Diagnostics, Hover, LanguageService, LanguageServiceHost, Location, Span, TemplateSource} from './types';
@ -30,8 +30,6 @@ export function createLanguageService(host: LanguageServiceHost): LanguageServic
class LanguageServiceImpl implements LanguageService { class LanguageServiceImpl implements LanguageService {
constructor(private host: LanguageServiceHost) {} constructor(private host: LanguageServiceHost) {}
private get metadataResolver(): CompileMetadataResolver { return this.host.resolver; }
getTemplateReferences(): string[] { return this.host.getTemplateReferences(); } getTemplateReferences(): string[] { return this.host.getTemplateReferences(); }
getDiagnostics(fileName: string): Diagnostic[] { getDiagnostics(fileName: string): Diagnostic[] {
@ -65,14 +63,14 @@ class LanguageServiceImpl implements LanguageService {
} }
} }
getDefinitionAt(fileName: string, position: number): Location[]|undefined { getDefinitionAt(fileName: string, position: number): tss.DefinitionInfoAndBoundSpan|undefined {
let templateInfo = this.host.getTemplateAstAtPosition(fileName, position); let templateInfo = this.host.getTemplateAstAtPosition(fileName, position);
if (templateInfo) { if (templateInfo) {
return getDefinition(templateInfo); return getDefinitionAndBoundSpan(templateInfo);
} }
} }
getHoverAt(fileName: string, position: number): Hover|undefined { getHoverAt(fileName: string, position: number): tss.QuickInfo|undefined {
let templateInfo = this.host.getTemplateAstAtPosition(fileName, position); let templateInfo = this.host.getTemplateAstAtPosition(fileName, position);
if (templateInfo) { if (templateInfo) {
return getHover(templateInfo); return getHover(templateInfo);

View File

@ -9,9 +9,8 @@
import * as ts from 'typescript'; // used as value, passed in by tsserver at runtime import * as ts from 'typescript'; // used as value, passed in by tsserver at runtime
import * as tss from 'typescript/lib/tsserverlibrary'; // used as type only import * as tss from 'typescript/lib/tsserverlibrary'; // used as type only
import {ngLocationToTsDefinitionInfo} from './definitions';
import {createLanguageService} from './language_service'; import {createLanguageService} from './language_service';
import {Completion, Diagnostic, DiagnosticMessageChain, Location} from './types'; import {Completion, Diagnostic, DiagnosticMessageChain} from './types';
import {TypeScriptServiceHost} from './typescript_host'; import {TypeScriptServiceHost} from './typescript_host';
const projectHostMap = new WeakMap<tss.server.Project, TypeScriptServiceHost>(); const projectHostMap = new WeakMap<tss.server.Project, TypeScriptServiceHost>();
@ -76,13 +75,13 @@ export function create(info: tss.server.PluginCreateInfo): tss.LanguageService {
// This effectively disables native TS features and is meant for internal // This effectively disables native TS features and is meant for internal
// use only. // use only.
const angularOnly = config ? config.angularOnly === true : false; const angularOnly = config ? config.angularOnly === true : false;
const proxy: tss.LanguageService = Object.assign({}, tsLS);
const ngLSHost = new TypeScriptServiceHost(tsLSHost, tsLS); const ngLSHost = new TypeScriptServiceHost(tsLSHost, tsLS);
const ngLS = createLanguageService(ngLSHost); const ngLS = createLanguageService(ngLSHost);
projectHostMap.set(project, ngLSHost); projectHostMap.set(project, ngLSHost);
proxy.getCompletionsAtPosition = function( function getCompletionsAtPosition(
fileName: string, position: number, options: tss.GetCompletionsAtPositionOptions|undefined) { fileName: string, position: number,
options: tss.GetCompletionsAtPositionOptions | undefined) {
if (!angularOnly) { if (!angularOnly) {
const results = tsLS.getCompletionsAtPosition(fileName, position, options); const results = tsLS.getCompletionsAtPosition(fileName, position, options);
if (results && results.entries.length) { if (results && results.entries.length) {
@ -100,39 +99,20 @@ export function create(info: tss.server.PluginCreateInfo): tss.LanguageService {
isNewIdentifierLocation: false, isNewIdentifierLocation: false,
entries: results.map(completionToEntry), entries: results.map(completionToEntry),
}; };
}; }
proxy.getQuickInfoAtPosition = function(fileName: string, position: number): tss.QuickInfo | function getQuickInfoAtPosition(fileName: string, position: number): tss.QuickInfo|undefined {
undefined { if (!angularOnly) {
if (!angularOnly) { const result = tsLS.getQuickInfoAtPosition(fileName, position);
const result = tsLS.getQuickInfoAtPosition(fileName, position); if (result) {
if (result) { // If TS could answer the query, then return results immediately.
// If TS could answer the query, then return results immediately. return result;
return result; }
} }
} return ngLS.getHoverAt(fileName, position);
const result = ngLS.getHoverAt(fileName, position); }
if (!result) {
return;
}
return {
// TODO(kyliau): Provide more useful info for kind and kindModifiers
kind: ts.ScriptElementKind.unknown,
kindModifiers: ts.ScriptElementKindModifier.none,
textSpan: {
start: result.span.start,
length: result.span.end - result.span.start,
},
displayParts: result.text.map((part) => {
return {
text: part.text,
kind: part.language || 'angular',
};
}),
};
};
proxy.getSemanticDiagnostics = function(fileName: string): tss.Diagnostic[] { function getSemanticDiagnostics(fileName: string): tss.Diagnostic[] {
const results: tss.Diagnostic[] = []; const results: tss.Diagnostic[] = [];
if (!angularOnly) { if (!angularOnly) {
const tsResults = tsLS.getSemanticDiagnostics(fileName); const tsResults = tsLS.getSemanticDiagnostics(fileName);
@ -146,48 +126,43 @@ export function create(info: tss.server.PluginCreateInfo): tss.LanguageService {
const sourceFile = fileName.endsWith('.ts') ? ngLSHost.getSourceFile(fileName) : undefined; const sourceFile = fileName.endsWith('.ts') ? ngLSHost.getSourceFile(fileName) : undefined;
results.push(...ngResults.map(d => diagnosticToDiagnostic(d, sourceFile))); results.push(...ngResults.map(d => diagnosticToDiagnostic(d, sourceFile)));
return results; return results;
}; }
proxy.getDefinitionAtPosition = function(fileName: string, position: number): function getDefinitionAtPosition(
ReadonlyArray<tss.DefinitionInfo>| fileName: string, position: number): ReadonlyArray<tss.DefinitionInfo>|undefined {
undefined { if (!angularOnly) {
if (!angularOnly) { const results = tsLS.getDefinitionAtPosition(fileName, position);
const results = tsLS.getDefinitionAtPosition(fileName, position); if (results) {
if (results) { // If TS could answer the query, then return results immediately.
// If TS could answer the query, then return results immediately. return results;
return results; }
} }
} const result = ngLS.getDefinitionAt(fileName, position);
const results = ngLS.getDefinitionAt(fileName, position); if (!result || !result.definitions || !result.definitions.length) {
if (!results) { return;
return; }
} return result.definitions;
return results.map(ngLocationToTsDefinitionInfo); }
};
proxy.getDefinitionAndBoundSpan = function(fileName: string, position: number): function getDefinitionAndBoundSpan(
tss.DefinitionInfoAndBoundSpan | fileName: string, position: number): tss.DefinitionInfoAndBoundSpan|undefined {
undefined { if (!angularOnly) {
if (!angularOnly) { const result = tsLS.getDefinitionAndBoundSpan(fileName, position);
const result = tsLS.getDefinitionAndBoundSpan(fileName, position); if (result) {
if (result) { // If TS could answer the query, then return results immediately.
// If TS could answer the query, then return results immediately. return result;
return result; }
} }
} return ngLS.getDefinitionAt(fileName, position);
const results = ngLS.getDefinitionAt(fileName, position); }
if (!results || !results.length) {
return;
}
const {span} = results[0];
return {
definitions: results.map(ngLocationToTsDefinitionInfo),
textSpan: {
start: span.start,
length: span.end - span.start,
},
};
};
const proxy: tss.LanguageService = Object.assign(
// First clone the original TS language service
{}, tsLS,
// Then override the methods supported by Angular language service
{
getCompletionsAtPosition, getQuickInfoAtPosition, getSemanticDiagnostics,
getDefinitionAtPosition, getDefinitionAndBoundSpan,
});
return proxy; return proxy;
} }

View File

@ -8,6 +8,8 @@
import {CompileDirectiveMetadata, CompileMetadataResolver, CompilePipeSummary, NgAnalyzedModules, StaticSymbol} from '@angular/compiler'; import {CompileDirectiveMetadata, CompileMetadataResolver, CompilePipeSummary, NgAnalyzedModules, StaticSymbol} from '@angular/compiler';
import {BuiltinType, DeclarationKind, Definition, PipeInfo, Pipes, Signature, Span, Symbol, SymbolDeclaration, SymbolQuery, SymbolTable} from '@angular/compiler-cli/src/language_services'; import {BuiltinType, DeclarationKind, Definition, PipeInfo, Pipes, Signature, Span, Symbol, SymbolDeclaration, SymbolQuery, SymbolTable} from '@angular/compiler-cli/src/language_services';
import * as tss from 'typescript/lib/tsserverlibrary';
import {AstResult, TemplateInfo} from './common'; import {AstResult, TemplateInfo} from './common';
export { export {
@ -394,12 +396,12 @@ export interface LanguageService {
/** /**
* Return the definition location for the symbol at position. * Return the definition location for the symbol at position.
*/ */
getDefinitionAt(fileName: string, position: number): Location[]|undefined; getDefinitionAt(fileName: string, position: number): tss.DefinitionInfoAndBoundSpan|undefined;
/** /**
* Return the hover information for the symbol at position. * Return the hover information for the symbol at position.
*/ */
getHoverAt(fileName: string, position: number): Hover|undefined; getHoverAt(fileName: string, position: number): tss.QuickInfo|undefined;
/** /**
* Return the pipes that are available at the given position. * Return the pipes that are available at the given position.

View File

@ -9,159 +9,319 @@
import * as ts from 'typescript'; import * as ts from 'typescript';
import {createLanguageService} from '../src/language_service'; import {createLanguageService} from '../src/language_service';
import {Span} from '../src/types'; import {LanguageService} from '../src/types';
import {TypeScriptServiceHost} from '../src/typescript_host'; import {TypeScriptServiceHost} from '../src/typescript_host';
import {toh} from './test_data'; import {toh} from './test_data';
import {MockTypescriptHost} from './test_utils';
import {MockTypescriptHost,} from './test_utils';
describe('definitions', () => { describe('definitions', () => {
let documentRegistry = ts.createDocumentRegistry(); let mockHost: MockTypescriptHost;
let mockHost = new MockTypescriptHost(['/app/main.ts', '/app/parsing-cases.ts'], toh); let service: ts.LanguageService;
let service = ts.createLanguageService(mockHost, documentRegistry); let ngHost: TypeScriptServiceHost;
let ngHost = new TypeScriptServiceHost(mockHost, service); let ngService: LanguageService;
let ngService = createLanguageService(ngHost);
beforeEach(() => {
// Create a new mockHost every time to reset any files that are overridden.
mockHost = new MockTypescriptHost(['/app/main.ts', '/app/parsing-cases.ts'], toh);
service = ts.createLanguageService(mockHost);
ngHost = new TypeScriptServiceHost(mockHost, service);
ngService = createLanguageService(ngHost);
});
it('should be able to find field in an interpolation', () => { it('should be able to find field in an interpolation', () => {
localReference( const fileName = addCode(`
` @Component({template: '{{«name»}}'}) export class MyComponent { «ᐱnameᐱ: string;» }`); @Component({
template: '{{«name»}}'
})
export class MyComponent {
«ᐱnameᐱ: string;»
}`);
const marker = getReferenceMarkerFor(fileName, 'name');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
expect(textSpan).toEqual(marker);
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
expect(def.fileName).toBe(fileName);
expect(def.name).toBe('name');
expect(def.kind).toBe('property');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(fileName, 'name'));
}); });
it('should be able to find a field in a attribute reference', () => { it('should be able to find a field in a attribute reference', () => {
localReference( const fileName = addCode(`
` @Component({template: '<input [(ngModel)]="«name»">'}) export class MyComponent { «ᐱnameᐱ: string;» }`); @Component({
template: '<input [(ngModel)]="«name»">'
})
export class MyComponent {
«ᐱnameᐱ: string;»
}`);
const marker = getReferenceMarkerFor(fileName, 'name');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
expect(textSpan).toEqual(marker);
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
expect(def.fileName).toBe(fileName);
expect(def.name).toBe('name');
expect(def.kind).toBe('property');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(fileName, 'name'));
}); });
it('should be able to find a method from a call', () => { it('should be able to find a method from a call', () => {
localReference( const fileName = addCode(`
` @Component({template: '<div (click)="«myClick»();"></div>'}) export class MyComponent { «ᐱmyClickᐱ() { }»}`); @Component({
template: '<div (click)="~{start-my}«myClick»()~{end-my};"></div>'
})
export class MyComponent {
«myClickᐱ() { }»
}`);
const marker = getReferenceMarkerFor(fileName, 'myClick');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
expect(textSpan).toEqual(getLocationMarkerFor(fileName, 'my'));
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
expect(def.fileName).toBe(fileName);
expect(def.name).toBe('myClick');
expect(def.kind).toBe('method');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(fileName, 'myClick'));
}); });
it('should be able to find a field reference in an *ngIf', () => { it('should be able to find a field reference in an *ngIf', () => {
localReference( const fileName = addCode(`
` @Component({template: '<div *ngIf="«include»"></div>'}) export class MyComponent { «ᐱincludeᐱ = true;»}`); @Component({
template: '<div *ngIf="«include»"></div>'
})
export class MyComponent {
«includeᐱ = true;»
}`);
const marker = getReferenceMarkerFor(fileName, 'include');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
expect(textSpan).toEqual(marker);
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
expect(def.fileName).toBe(fileName);
expect(def.name).toBe('include');
expect(def.kind).toBe('property');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(fileName, 'include'));
}); });
it('should be able to find a reference to a component', () => { it('should be able to find a reference to a component', () => {
reference( const fileName = addCode(`
'parsing-cases.ts', @Component({
` @Component({template: '<«test-comp»></test-comp>'}) export class MyComponent { }`); template: '~{start-my}<«test-comp»></test-comp>~{end-my}'
})
export class MyComponent { }`);
// Get the marker for «test-comp» in the code added above.
const marker = getReferenceMarkerFor(fileName, 'test-comp');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
// Get the marker for bounded text in the code added above.
const boundedText = getLocationMarkerFor(fileName, 'my');
expect(textSpan).toEqual(boundedText);
// There should be exactly 1 definition
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
const refFileName = '/app/parsing-cases.ts';
expect(def.fileName).toBe(refFileName);
expect(def.name).toBe('TestComponent');
expect(def.kind).toBe('component');
expect(def.textSpan).toEqual(getLocationMarkerFor(refFileName, 'test-comp'));
}); });
it('should be able to find an event provider', () => { it('should be able to find an event provider', () => {
reference( const fileName = addCode(`
'/app/parsing-cases.ts', 'test', @Component({
` @Component({template: '<test-comp («test»)="myHandler()"></div>'}) export class MyComponent { myHandler() {} }`); template: '<test-comp ~{start-my}(«test»)="myHandler()"~{end-my}></div>'
})
export class MyComponent { myHandler() {} }`);
// Get the marker for «test» in the code added above.
const marker = getReferenceMarkerFor(fileName, 'test');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
// Get the marker for bounded text in the code added above
const boundedText = getLocationMarkerFor(fileName, 'my');
expect(textSpan).toEqual(boundedText);
// There should be exactly 1 definition
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
const refFileName = '/app/parsing-cases.ts';
expect(def.fileName).toBe(refFileName);
expect(def.name).toBe('testEvent');
expect(def.kind).toBe('event');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(refFileName, 'test'));
}); });
it('should be able to find an input provider', () => { it('should be able to find an input provider', () => {
reference( // '/app/parsing-cases.ts', 'tcName',
'/app/parsing-cases.ts', 'tcName', const fileName = addCode(`
` @Component({template: '<test-comp [«tcName»]="name"></div>'}) export class MyComponent { name = 'my name'; }`); @Component({
template: '<test-comp ~{start-my}[«tcName»]="name"~{end-my}></div>'
})
export class MyComponent {
name = 'my name';
}`);
// Get the marker for «test» in the code added above.
const marker = getReferenceMarkerFor(fileName, 'tcName');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
// Get the marker for bounded text in the code added above
const boundedText = getLocationMarkerFor(fileName, 'my');
expect(textSpan).toEqual(boundedText);
// There should be exactly 1 definition
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(1);
const def = definitions ![0];
const refFileName = '/app/parsing-cases.ts';
expect(def.fileName).toBe(refFileName);
expect(def.name).toBe('name');
expect(def.kind).toBe('property');
expect(def.textSpan).toEqual(getDefinitionMarkerFor(refFileName, 'tcName'));
}); });
it('should be able to find a pipe', () => { it('should be able to find a pipe', () => {
reference( const fileName = addCode(`
'common.d.ts', @Component({
` @Component({template: '<div *ngIf="input | «async»"></div>'}) export class MyComponent { input: EventEmitter; }`); template: '<div *ngIf="~{start-my}input | «async»~{end-my}"></div>'
})
export class MyComponent {
input: EventEmitter;
}`);
// Get the marker for «test» in the code added above.
const marker = getReferenceMarkerFor(fileName, 'async');
const result = ngService.getDefinitionAt(fileName, marker.start);
expect(result).toBeDefined();
const {textSpan, definitions} = result !;
// Get the marker for bounded text in the code added above
const boundedText = getLocationMarkerFor(fileName, 'my');
expect(textSpan).toEqual(boundedText);
expect(definitions).toBeDefined();
expect(definitions !.length).toBe(4);
const refFileName = '/node_modules/@angular/common/common.d.ts';
for (const def of definitions !) {
expect(def.fileName).toBe(refFileName);
expect(def.name).toBe('async');
expect(def.kind).toBe('pipe');
// Not asserting the textSpan of definition because it's external file
}
}); });
function localReference(code: string) { /**
addCode(code, fileName => { * Append a snippet of code to `app.component.ts` and return the file name.
const refResult = mockHost.getReferenceMarkers(fileName) !; * There must not be any name collision with existing code.
for (const name in refResult.references) { * @param code Snippet of code
const references = refResult.references[name]; */
const definitions = refResult.definitions[name]; function addCode(code: string) {
expect(definitions).toBeDefined(); // If this fails the test data is wrong.
for (const reference of references) {
const definition = ngService.getDefinitionAt(fileName, reference.start);
if (definition) {
definition.forEach(d => expect(d.fileName).toEqual(fileName));
const match = matchingSpan(definition.map(d => d.span), definitions);
if (!match) {
throw new Error(
`Expected one of ${stringifySpans(definition.map(d => d.span))} to match one of ${stringifySpans(definitions)}`);
}
} else {
throw new Error('Expected a definition');
}
}
}
});
}
function reference(referencedFile: string, code: string): void;
function reference(referencedFile: string, span: Span, code: string): void;
function reference(referencedFile: string, definition: string, code: string): void;
function reference(referencedFile: string, p1?: any, p2?: any): void {
const code: string = p2 ? p2 : p1;
const definition: string = p2 ? p1 : undefined;
let span: Span = p2 && p1.start != null ? p1 : undefined;
if (definition && !span) {
const referencedFileMarkers = mockHost.getReferenceMarkers(referencedFile) !;
expect(referencedFileMarkers).toBeDefined(); // If this fails the test data is wrong.
const spans = referencedFileMarkers.definitions[definition];
expect(spans).toBeDefined(); // If this fails the test data is wrong.
span = spans[0];
}
addCode(code, fileName => {
const refResult = mockHost.getReferenceMarkers(fileName) !;
let tests = 0;
for (const name in refResult.references) {
const references = refResult.references[name];
expect(reference).toBeDefined(); // If this fails the test data is wrong.
for (const reference of references) {
tests++;
const definition = ngService.getDefinitionAt(fileName, reference.start);
if (definition) {
definition.forEach(d => {
if (d.fileName.indexOf(referencedFile) < 0) {
throw new Error(
`Expected reference to file ${referencedFile}, received ${d.fileName}`);
}
if (span) {
expect(d.span).toEqual(span);
}
});
} else {
throw new Error('Expected a definition');
}
}
}
if (!tests) {
throw new Error('Expected at least one reference (test data error)');
}
});
}
function addCode(code: string, cb: (fileName: string, content?: string) => void) {
const fileName = '/app/app.component.ts'; const fileName = '/app/app.component.ts';
const originalContent = mockHost.getFileContent(fileName); const originalContent = mockHost.getFileContent(fileName);
const newContent = originalContent + code; const newContent = originalContent + code;
mockHost.override(fileName, originalContent + code); mockHost.override(fileName, newContent);
try { return fileName;
cb(fileName, newContent); }
} finally {
mockHost.override(fileName, undefined !); /**
} * Returns the definition marker selectorᐱ for the specified 'selector'.
* Asserts that marker exists.
* @param fileName name of the file
* @param selector name of the marker
*/
function getDefinitionMarkerFor(fileName: string, selector: string): ts.TextSpan {
const markers = mockHost.getReferenceMarkers(fileName);
expect(markers).toBeDefined();
expect(Object.keys(markers !.definitions)).toContain(selector);
expect(markers !.definitions[selector].length).toBe(1);
const marker = markers !.definitions[selector][0];
expect(marker.start).toBeLessThanOrEqual(marker.end);
return {
start: marker.start,
length: marker.end - marker.start,
};
}
/**
* Returns the reference marker «selector» for the specified 'selector'.
* Asserts that marker exists.
* @param fileName name of the file
* @param selector name of the marker
*/
function getReferenceMarkerFor(fileName: string, selector: string): ts.TextSpan {
const markers = mockHost.getReferenceMarkers(fileName);
expect(markers).toBeDefined();
expect(Object.keys(markers !.references)).toContain(selector);
expect(markers !.references[selector].length).toBe(1);
const marker = markers !.references[selector][0];
expect(marker.start).toBeLessThanOrEqual(marker.end);
return {
start: marker.start,
length: marker.end - marker.start,
};
}
/**
* Returns the location marker ~{selector} for the specified 'selector'.
* Asserts that marker exists.
* @param fileName name of the file
* @param selector name of the marker
*/
function getLocationMarkerFor(fileName: string, selector: string): ts.TextSpan {
const markers = mockHost.getMarkerLocations(fileName);
expect(markers).toBeDefined();
const start = markers ![`start-${selector}`];
expect(start).toBeDefined();
const end = markers ![`end-${selector}`];
expect(end).toBeDefined();
expect(start).toBeLessThanOrEqual(end);
return {
start: start,
length: end - start,
};
} }
}); });
function matchingSpan(aSpans: Span[], bSpans: Span[]): Span|undefined {
for (const a of aSpans) {
for (const b of bSpans) {
if (a.start == b.start && a.end == b.end) {
return a;
}
}
}
}
function stringifySpan(span: Span) {
return span ? `(${span.start}-${span.end})` : '<undefined>';
}
function stringifySpans(spans: Span[]) {
return spans ? `[${spans.map(stringifySpan).join(', ')}]` : '<empty>';
}

View File

@ -28,43 +28,43 @@ describe('hover', () => {
it('should be able to find field in an interpolation', () => { it('should be able to find field in an interpolation', () => {
hover( hover(
` @Component({template: '{{«name»}}'}) export class MyComponent { name: string; }`, ` @Component({template: '{{«name»}}'}) export class MyComponent { name: string; }`,
'property name of MyComponent'); '(property) MyComponent.name');
}); });
it('should be able to find a field in a attribute reference', () => { it('should be able to find a field in a attribute reference', () => {
hover( hover(
` @Component({template: '<input [(ngModel)]="«name»">'}) export class MyComponent { name: string; }`, ` @Component({template: '<input [(ngModel)]="«name»">'}) export class MyComponent { name: string; }`,
'property name of MyComponent'); '(property) MyComponent.name');
}); });
it('should be able to find a method from a call', () => { it('should be able to find a method from a call', () => {
hover( hover(
` @Component({template: '<div (click)="«ᐱmyClickᐱ()»;"></div>'}) export class MyComponent { myClick() { }}`, ` @Component({template: '<div (click)="«ᐱmyClickᐱ()»;"></div>'}) export class MyComponent { myClick() { }}`,
'method myClick of MyComponent'); '(method) MyComponent.myClick');
}); });
it('should be able to find a field reference in an *ngIf', () => { it('should be able to find a field reference in an *ngIf', () => {
hover( hover(
` @Component({template: '<div *ngIf="«include»"></div>'}) export class MyComponent { include = true;}`, ` @Component({template: '<div *ngIf="«include»"></div>'}) export class MyComponent { include = true;}`,
'property include of MyComponent'); '(property) MyComponent.include');
}); });
it('should be able to find a reference to a component', () => { it('should be able to find a reference to a component', () => {
hover( hover(
` @Component({template: '«<ᐱtestᐱ-comp></test-comp>»'}) export class MyComponent { }`, ` @Component({template: '«<ᐱtestᐱ-comp></test-comp>»'}) export class MyComponent { }`,
'component TestComponent'); '(component) TestComponent');
}); });
it('should be able to find an event provider', () => { it('should be able to find an event provider', () => {
hover( hover(
` @Component({template: '<test-comp «(ᐱtestᐱ)="myHandler()"»></div>'}) export class MyComponent { myHandler() {} }`, ` @Component({template: '<test-comp «(ᐱtestᐱ)="myHandler()"»></div>'}) export class MyComponent { myHandler() {} }`,
'event testEvent of TestComponent'); '(event) TestComponent.testEvent');
}); });
it('should be able to find an input provider', () => { it('should be able to find an input provider', () => {
hover( hover(
` @Component({template: '<test-comp «[ᐱtcNameᐱ]="name"»></div>'}) export class MyComponent { name = 'my name'; }`, ` @Component({template: '<test-comp «[ᐱtcNameᐱ]="name"»></div>'}) export class MyComponent { name = 'my name'; }`,
'property name of TestComponent'); '(property) TestComponent.name');
}); });
it('should be able to ignore a reference declaration', () => { it('should be able to ignore a reference declaration', () => {
@ -87,10 +87,13 @@ describe('hover', () => {
[]).concat(markers.definitions[referenceName] || []); []).concat(markers.definitions[referenceName] || []);
for (const reference of references) { for (const reference of references) {
tests++; tests++;
const hover = ngService.getHoverAt(fileName, reference.start); const quickInfo = ngService.getHoverAt(fileName, reference.start);
if (!hover) throw new Error(`Expected a hover at location ${reference.start}`); if (!quickInfo) throw new Error(`Expected a hover at location ${reference.start}`);
expect(hover.span).toEqual(reference); expect(quickInfo.textSpan).toEqual({
expect(toText(hover)).toEqual(hoverText); start: reference.start,
length: reference.end - reference.start,
});
expect(toText(quickInfo)).toEqual(hoverText);
} }
} }
expect(tests).toBeGreaterThan(0); // If this fails the test is wrong. expect(tests).toBeGreaterThan(0); // If this fails the test is wrong.
@ -109,5 +112,8 @@ describe('hover', () => {
} }
} }
function toText(hover: Hover): string { return hover.text.map(h => h.text).join(''); } function toText(quickInfo: ts.QuickInfo): string {
const displayParts = quickInfo.displayParts || [];
return displayParts.map(p => p.text).join('');
}
}); });

View File

@ -139,11 +139,11 @@ export class ForUsingComponent {
@Component({template: '<div #div> <test-comp #test1> {{~{test-comp-content}}} {{test1.~{test-comp-after-test}name}} {{div.~{test-comp-after-div}.innerText}} </test-comp> </div> <test-comp #test2></test-comp>'}) @Component({template: '<div #div> <test-comp #test1> {{~{test-comp-content}}} {{test1.~{test-comp-after-test}name}} {{div.~{test-comp-after-div}.innerText}} </test-comp> </div> <test-comp #test2></test-comp>'})
export class References {} export class References {}
@Component({selector: 'test-comp', template: '<div>Testing: {{name}}</div>'}) ~{start-test-comp}@Component({selector: 'test-comp', template: '<div>Testing: {{name}}</div>'})
export class TestComponent { export class TestComponent {
«@Input('ᐱtcNameᐱ') name = 'test';» «@Input('ᐱtcNameᐱ') name = 'test';»
«@Output('ᐱtestᐱ') testEvent = new EventEmitter();» «@Output('ᐱtestᐱ') testEvent = new EventEmitter();»
} }~{end-test-comp}
@Component({templateUrl: 'test.ng'}) @Component({templateUrl: 'test.ng'})
export class TemplateReference { export class TemplateReference {