-
Notifications
You must be signed in to change notification settings - Fork 12.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Go to Implementation #10482
Go to Implementation #10482
Changes from 5 commits
111e509
457b67f
051c7b0
115141c
b0eff90
3ccc58c
66f30c9
9c562f8
b7071c1
5913a35
1cdd1d3
2069e1c
f91a123
4a37fd7
b6f7dd7
ea3752a
fd93699
4eef417
29d85cd
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1622,6 +1622,15 @@ namespace FourSlash { | |
assertFn(actualCount, expectedCount, this.messageAtLastKnownMarker("Type definitions Count")); | ||
} | ||
|
||
public verifyImplementationsCount(negative: boolean, expectedCount: number) { | ||
const assertFn = negative ? assert.notEqual : assert.equal; | ||
|
||
const implementations = this.languageService.getImplementationAtPosition(this.activeFile.fileName, this.currentCaretPosition); | ||
const actualCount = implementations && implementations.length || 0; | ||
|
||
assertFn(actualCount, expectedCount, this.messageAtLastKnownMarker("Implementations Count")); | ||
} | ||
|
||
public verifyDefinitionsName(negative: boolean, expectedName: string, expectedContainerName: string) { | ||
const definitions = this.languageService.getDefinitionAtPosition(this.activeFile.fileName, this.currentCaretPosition); | ||
const actualDefinitionName = definitions && definitions.length ? definitions[0].name : ""; | ||
|
@@ -1636,6 +1645,47 @@ namespace FourSlash { | |
} | ||
} | ||
|
||
public goToImplementation(implIndex: number) { | ||
const implementations = this.languageService.getImplementationAtPosition(this.activeFile.fileName, this.currentCaretPosition); | ||
if (!implementations || !implementations.length) { | ||
this.raiseError("goToImplementation failed - expected to at least one implementation location but got 0"); | ||
} | ||
|
||
if (implIndex >= implementations.length) { | ||
this.raiseError(`goToImplementation failed - implIndex value (${implIndex}) exceeds implementation list size (${implementations.length})`); | ||
} | ||
|
||
const implementation = implementations[implIndex]; | ||
this.openFile(implementation.fileName); | ||
this.currentCaretPosition = implementation.textSpan.start; | ||
} | ||
|
||
public verifyRangesInImplementationList() { | ||
const implementations = this.languageService.getImplementationAtPosition(this.activeFile.fileName, this.currentCaretPosition); | ||
if (!implementations || !implementations.length) { | ||
this.raiseError("verifyRangesInImplementationList failed - expected to at least one implementation location but got 0"); | ||
} | ||
|
||
const ranges = this.getRanges(); | ||
|
||
if (!ranges || !ranges.length) { | ||
this.raiseError("verifyRangesInImplementationList failed - expected to at least one range in test source"); | ||
} | ||
|
||
for (const range of ranges) { | ||
let rangeIsPresent = false; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. const matchingImpl = ts.find(implementations, impl =>
range.fileName === impl.fileName && range.start === impl.textSpan.start && length === impl.textSpan.length);
if (matchingImpl) {
matchingImpl.matched = true;
}
else {
unsatisfiedRanges.push(range);
} |
||
const length = range.end - range.start; | ||
for (const impl of implementations) { | ||
if (range.fileName === impl.fileName && range.start === impl.textSpan.start && length === impl.textSpan.length) { | ||
rangeIsPresent = true; | ||
break; | ||
} | ||
} | ||
assert.isTrue(rangeIsPresent, `No implementation found for range ${range.start}, ${range.end} in ${range.fileName}: ${this.rangeText(range)}`); | ||
} | ||
assert.equal(implementations.length, ranges.length, `Different number of implementations (${implementations.length}) and ranges (${ranges.length})`); | ||
} | ||
|
||
public getMarkers(): Marker[] { | ||
// Return a copy of the list | ||
return this.testData.markers.slice(0); | ||
|
@@ -2786,6 +2836,10 @@ namespace FourSlashInterface { | |
this.state.goToTypeDefinition(definitionIndex); | ||
} | ||
|
||
public implementation(implementationIndex = 0) { | ||
this.state.goToImplementation(implementationIndex); | ||
} | ||
|
||
public position(position: number, fileIndex?: number): void; | ||
public position(position: number, fileName?: string): void; | ||
public position(position: number, fileNameOrIndex?: any): void { | ||
|
@@ -2894,6 +2948,10 @@ namespace FourSlashInterface { | |
this.state.verifyTypeDefinitionsCount(this.negative, expectedCount); | ||
} | ||
|
||
public implementationCountIs(expectedCount: number) { | ||
this.state.verifyImplementationsCount(this.negative, expectedCount); | ||
} | ||
|
||
public definitionLocationExists() { | ||
this.state.verifyDefinitionLocationExists(this.negative); | ||
} | ||
|
@@ -3135,6 +3193,10 @@ namespace FourSlashInterface { | |
public ProjectInfo(expected: string[]) { | ||
this.state.verifyProjectInfo(expected); | ||
} | ||
|
||
public allRangesAppearInImplementationList() { | ||
this.state.verifyRangesInImplementationList(); | ||
} | ||
} | ||
|
||
export class Edit { | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -193,6 +193,14 @@ declare namespace ts.server.protocol { | |
export interface TypeDefinitionRequest extends FileLocationRequest { | ||
} | ||
|
||
/** | ||
* Go to implementation request; value of command field is | ||
* "implementation". Return response giving the file locations that | ||
* implement the symbol found in file at location line, col. | ||
*/ | ||
export interface ImplementationRequest extends FileLocationRequest { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. question: why you make this empty interface? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I was following what |
||
} | ||
|
||
/** | ||
* Location in source code expressed as (one-based) line and character offset. | ||
*/ | ||
|
@@ -240,6 +248,13 @@ declare namespace ts.server.protocol { | |
body?: FileSpan[]; | ||
} | ||
|
||
/** | ||
* Implementation response message. Gives text range for implementations. | ||
*/ | ||
export interface ImplementationResponse extends Response { | ||
body?: FileSpan[]; | ||
} | ||
|
||
/** | ||
* Get occurrences request; value of command field is | ||
* "occurrences". Return response giving spans that are relevant | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
expected to find
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've noticed that typo about a dozen times when reviewing my own code and always forget to fix it. Fixed now!