Skip to content

Commit 0ff44a6

Browse files
committed
CodeMapper support
1 parent 1488256 commit 0ff44a6

File tree

10 files changed

+139
-1
lines changed

10 files changed

+139
-1
lines changed

src/harness/client.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ import {
7676
UserPreferences,
7777
} from "./_namespaces/ts";
7878
import {
79-
protocol,
79+
protocol
8080
} from "./_namespaces/ts.server";
8181

8282
export interface SessionClientHost extends LanguageServiceHost {
@@ -781,6 +781,8 @@ export class SessionClient implements LanguageService {
781781
});
782782
}
783783

784+
mapCode = notImplemented;
785+
784786
private createFileLocationOrRangeRequestArgs(positionOrRange: number | TextRange, fileName: string): protocol.FileLocationOrRangeRequestArgs {
785787
return typeof positionOrRange === "number"
786788
? this.createFileLocationRequestArgs(fileName, positionOrRange)

src/harness/harnessLanguageService.ts

+3
Original file line numberDiff line numberDiff line change
@@ -698,6 +698,9 @@ class LanguageServiceShimProxy implements ts.LanguageService {
698698
provideInlayHints(fileName: string, span: ts.TextSpan, preference: ts.UserPreferences) {
699699
return unwrapJSONCallResult(this.shim.provideInlayHints(fileName, span, preference));
700700
}
701+
mapCode(fileName: string, contents: string[], selected?: ts.PrioritizedSpan[], updates?: ts.FileTextChanges[]): ts.FileTextChanges[] {
702+
return unwrapJSONCallResult(this.shim.mapCode(fileName, contents, selected, updates));
703+
}
701704
getEmitOutput(fileName: string): ts.EmitOutput {
702705
return unwrapJSONCallResult(this.shim.getEmitOutput(fileName));
703706
}

src/server/protocol.ts

+30
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,7 @@ export const enum CommandTypes {
173173
ProvideCallHierarchyIncomingCalls = "provideCallHierarchyIncomingCalls",
174174
ProvideCallHierarchyOutgoingCalls = "provideCallHierarchyOutgoingCalls",
175175
ProvideInlayHints = "provideInlayHints",
176+
MapCode = "mapCode",
176177
}
177178

178179
/**
@@ -2687,6 +2688,35 @@ export interface InlayHintsResponse extends Response {
26872688
body?: InlayHintItem[];
26882689
}
26892690

2691+
export interface MapCodeRequestArgs extends FileRequestArgs {
2692+
/// The specific code to map/insert/replace in the file.
2693+
contents: string[];
2694+
2695+
/// Areas of "focus" to inform the code mapper with. For example, cursor
2696+
/// location, current selection, viewport, etc.
2697+
focusLocations?: MapCodeFocusLocation[];
2698+
2699+
/// Edits to apply to the current workspace before performing the mapping.
2700+
updates?: FileCodeEdits[]
2701+
}
2702+
2703+
/// Area of "focus" to inform the code mapper with. For example, cursor
2704+
/// location, current selection, viewport, etc.
2705+
export interface MapCodeFocusLocation extends FileSpan {
2706+
/// The priority to give this focus location. Multiple FocusLocations can
2707+
/// share the same priority value.
2708+
priority: number;
2709+
}
2710+
2711+
export interface MapCodeRequest extends Request {
2712+
command: CommandTypes.MapCode,
2713+
arguments: MapCodeRequestArgs;
2714+
}
2715+
2716+
export interface MapCodeResponse extends Response {
2717+
body: FileCodeEdits[]
2718+
}
2719+
26902720
/**
26912721
* Synchronous request for semantic diagnostics of one file.
26922722
*/

src/server/session.ts

+46
Original file line numberDiff line numberDiff line change
@@ -1887,6 +1887,49 @@ export class Session<TMessage = string> implements EventSender {
18871887
});
18881888
}
18891889

1890+
private mapCode(args: protocol.MapCodeRequestArgs): protocol.FileCodeEdits[] {
1891+
const { file, project } = this.getFileAndProject(args);
1892+
const scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file)!;
1893+
const selected = args.focusLocations?.map(loc => {
1894+
const start = scriptInfo.lineOffsetToPosition(loc.start.line, loc.start.offset);
1895+
const end = scriptInfo.lineOffsetToPosition(loc.end.line, loc.end.offset);
1896+
return {
1897+
fileName: loc.file,
1898+
priority: loc.priority,
1899+
textSpan: {
1900+
start,
1901+
length: end - start,
1902+
}
1903+
};
1904+
});
1905+
const updates = args.updates?.map(edit => {
1906+
return {
1907+
fileName: edit.fileName,
1908+
textChanges: edit.textChanges.map(({ start, end, newText }) => {
1909+
const newStart = scriptInfo.lineOffsetToPosition(start.line, start.offset);
1910+
const newEnd = scriptInfo.lineOffsetToPosition(end.line, end.offset);
1911+
return {
1912+
span: { start: newStart, length: newEnd - newStart },
1913+
newText,
1914+
};
1915+
}),
1916+
};
1917+
});
1918+
return project.getLanguageService().mapCode(file, args.contents, selected, updates)?.map(change => {
1919+
return {
1920+
fileName: change.fileName,
1921+
textChanges: change.textChanges.map(({ span, newText }) => {
1922+
const newSpan = toProtocolTextSpan(span, scriptInfo);
1923+
return {
1924+
start: newSpan.start,
1925+
end: newSpan.end,
1926+
newText
1927+
};
1928+
}),
1929+
};
1930+
});
1931+
}
1932+
18901933
private setCompilerOptionsForInferredProjects(args: protocol.SetCompilerOptionsForInferredProjectsArgs): void {
18911934
this.projectService.setCompilerOptionsForInferredProjects(args.options, args.projectRootPath);
18921935
}
@@ -3556,6 +3599,9 @@ export class Session<TMessage = string> implements EventSender {
35563599
[protocol.CommandTypes.ProvideInlayHints]: (request: protocol.InlayHintsRequest) => {
35573600
return this.requiredResponse(this.provideInlayHints(request.arguments));
35583601
},
3602+
[protocol.CommandTypes.MapCode]: (request: protocol.MapCodeRequest) => {
3603+
return this.requiredResponse(this.mapCode(request.arguments));
3604+
},
35593605
}));
35603606

35613607
public addProtocolHandler(command: string, handler: (request: protocol.Request) => HandlerResponse) {
+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
/* Generated file to emulate the ts.MapCode namespace. */
2+
3+
export * from "../mapCode";
4+

src/services/_namespaces/ts.ts

+2
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@ import * as GoToDefinition from "./ts.GoToDefinition";
3333
export { GoToDefinition };
3434
import * as InlayHints from "./ts.InlayHints";
3535
export { InlayHints };
36+
import * as MapCode from "./ts.MapCode";
37+
export { MapCode };
3638
import * as JsDoc from "./ts.JsDoc";
3739
export { JsDoc };
3840
import * as NavigateTo from "./ts.NavigateTo";

src/services/mapCode.ts

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import {
2+
FileTextChanges,
3+
PrioritizedSpan,
4+
Program,
5+
SourceFile,
6+
//textChanges,
7+
} from "./_namespaces/ts";
8+
9+
// useful stuff:
10+
// textChanges.ChangeTracker - lots of methods for inserting nodes in the right place.
11+
// textChanges.ChangeTracker.pushRaw() - to push `updates` before running our analysis.
12+
13+
/** @internal */
14+
export function mapCode(
15+
_file: SourceFile,
16+
_program: Program,
17+
_contents: string[],
18+
_focusLocations?: PrioritizedSpan[],
19+
_updates?: FileTextChanges[]
20+
): FileTextChanges[] {
21+
throw new Error("IT WORKED!!");
22+
// const checker = program.getTypeChecker();
23+
// return [];
24+
}

src/services/services.ts

+9
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,7 @@ import {
207207
LinkedEditingInfo,
208208
LiteralType,
209209
map,
210+
MapCode,
210211
mapDefined,
211212
MapLike,
212213
mapOneOrMany,
@@ -242,6 +243,7 @@ import {
242243
positionIsSynthesized,
243244
PossibleProgramFileInfo,
244245
PragmaMap,
246+
PrioritizedSpan,
245247
PrivateIdentifier,
246248
Program,
247249
PropertyName,
@@ -3070,6 +3072,12 @@ export function createLanguageService(
30703072
return InlayHints.provideInlayHints(getInlayHintsContext(sourceFile, span, preferences));
30713073
}
30723074

3075+
function mapCode(fileName: string, contents: string[], focusLocations?: PrioritizedSpan[], updates?: FileTextChanges[]): FileTextChanges[] {
3076+
synchronizeHostData();
3077+
const sourceFile = getValidSourceFile(fileName);
3078+
return MapCode.mapCode(sourceFile, program, contents, focusLocations, updates);
3079+
}
3080+
30733081
const ls: LanguageService = {
30743082
dispose,
30753083
cleanupSemanticCache,
@@ -3140,6 +3148,7 @@ export function createLanguageService(
31403148
uncommentSelection,
31413149
provideInlayHints,
31423150
getSupportedCodeFixes,
3151+
mapCode,
31433152
};
31443153

31453154
switch (languageServiceMode) {

src/services/shims.ts

+10
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import {
2121
Extension,
2222
extensionFromPath,
2323
FileReference,
24+
FileTextChanges,
2425
filter,
2526
flattenDiagnosticMessageText,
2627
FormatCodeOptions,
@@ -48,6 +49,7 @@ import {
4849
parseJsonSourceFileConfigFileContent,
4950
parseJsonText,
5051
preProcessFile,
52+
PrioritizedSpan,
5153
ResolvedModuleFull,
5254
ResolvedTypeReferenceDirective,
5355
resolveModuleName,
@@ -361,6 +363,7 @@ export interface LanguageServiceShim extends Shim {
361363
provideCallHierarchyIncomingCalls(fileName: string, position: number): string;
362364
provideCallHierarchyOutgoingCalls(fileName: string, position: number): string;
363365
provideInlayHints(fileName: string, span: TextSpan, preference: UserPreferences | undefined): string;
366+
mapCode(fileName: string, contents: string[], selected?: PrioritizedSpan[], updates?: FileTextChanges[]): string;
364367
getEmitOutput(fileName: string): string;
365368
getEmitOutputObject(fileName: string): EmitOutput;
366369

@@ -1157,6 +1160,13 @@ class LanguageServiceShimObject extends ShimBase implements LanguageServiceShim
11571160
);
11581161
}
11591162

1163+
public mapCode(fileName: string, contents: string[], focusLocations?: PrioritizedSpan[], updates?: FileTextChanges[]): string {
1164+
return this.forwardJSONCall(
1165+
`mapCode('${fileName}', '${JSON.stringify(contents)}', '${JSON.stringify(focusLocations)}', '${JSON.stringify(updates)}')`,
1166+
() => this.languageService.mapCode(fileName, contents, focusLocations, updates),
1167+
)
1168+
}
1169+
11601170
/// Emit
11611171
public getEmitOutput(fileName: string): string {
11621172
return this.forwardJSONCall(

src/services/types.ts

+8
Original file line numberDiff line numberDiff line change
@@ -676,6 +676,8 @@ export interface LanguageService {
676676

677677
getSupportedCodeFixes(fileName?: string): readonly string[];
678678

679+
mapCode(fileName: string, contents: string[], focusLocations?: PrioritizedSpan[], updates?: FileTextChanges[]): FileTextChanges[];
680+
679681
dispose(): void;
680682
}
681683

@@ -1117,6 +1119,12 @@ export interface EditorSettings {
11171119
trimTrailingWhitespace?: boolean;
11181120
}
11191121

1122+
export interface PrioritizedSpan {
1123+
fileName: string,
1124+
textSpan: TextSpan,
1125+
priority: number,
1126+
}
1127+
11201128
/** @deprecated - consider using FormatCodeSettings instead */
11211129
export interface FormatCodeOptions extends EditorOptions {
11221130
InsertSpaceAfterCommaDelimiter: boolean;

0 commit comments

Comments
 (0)