Skip to content

Commit 83442f8

Browse files
committed
CodeMapper support
1 parent 628bf0e commit 83442f8

File tree

8 files changed

+476
-1
lines changed

8 files changed

+476
-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 {
@@ -790,6 +790,8 @@ export class SessionClient implements LanguageService {
790790
});
791791
}
792792

793+
mapCode = notImplemented;
794+
793795
private createFileLocationOrRangeRequestArgs(positionOrRange: number | TextRange, fileName: string): protocol.FileLocationOrRangeRequestArgs {
794796
return typeof positionOrRange === "number"
795797
? this.createFileLocationRequestArgs(fileName, positionOrRange)

src/server/protocol.ts

+37
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,7 @@ export const enum CommandTypes {
174174
ProvideCallHierarchyOutgoingCalls = "provideCallHierarchyOutgoingCalls",
175175
ProvideInlayHints = "provideInlayHints",
176176
WatchChange = "watchChange",
177+
MapCode = "mapCode",
177178
}
178179

179180
/**
@@ -2699,6 +2700,42 @@ export interface InlayHintsResponse extends Response {
26992700
body?: InlayHintItem[];
27002701
}
27012702

2703+
export interface MapCodeRequestArgs {
2704+
/// The files and changes to try and apply/map.
2705+
mappings: MapCodeRequestDocumentMapping[];
2706+
2707+
/// Edits to apply to the current workspace before performing the mapping.
2708+
updates?: FileCodeEdits[]
2709+
}
2710+
2711+
export interface MapCodeRequestDocumentMapping {
2712+
/// The file for the request (absolute pathname required). Null/undefined
2713+
/// if specific file is unknown.
2714+
file?: string;
2715+
2716+
/// Optional name of project that contains file
2717+
projectFileName?: string;
2718+
2719+
/// The specific code to map/insert/replace in the file.
2720+
contents: string[];
2721+
2722+
/// Areas of "focus" to inform the code mapper with. For example, cursor
2723+
/// location, current selection, viewport, etc. Nested arrays denote
2724+
/// priority: toplevel arrays are more important than inner arrays, and
2725+
/// inner array priorities are based on items within that array. Items
2726+
/// earlier in the arrays have higher priority.
2727+
focusLocations?: FileSpan[][];
2728+
}
2729+
2730+
export interface MapCodeRequest extends Request {
2731+
command: CommandTypes.MapCode,
2732+
arguments: MapCodeRequestArgs;
2733+
}
2734+
2735+
export interface MapCodeResponse extends Response {
2736+
body: FileCodeEdits[]
2737+
}
2738+
27022739
/**
27032740
* Synchronous request for semantic diagnostics of one file.
27042741
*/

src/server/session.ts

+69
Original file line numberDiff line numberDiff line change
@@ -85,6 +85,7 @@ import {
8585
LineAndCharacter,
8686
LinkedEditingInfo,
8787
map,
88+
MapCodeDocumentMapping,
8889
mapDefined,
8990
mapDefinedIterator,
9091
mapIterator,
@@ -1899,6 +1900,71 @@ export class Session<TMessage = string> implements EventSender {
18991900
});
19001901
}
19011902

1903+
private mapCode(args: protocol.MapCodeRequestArgs): protocol.FileCodeEdits[] {
1904+
const formatOptions = this.getHostFormatOptions();
1905+
const preferences = this.getHostPreferences();
1906+
const projects = new Map<Project, MapCodeDocumentMapping[]>();
1907+
args.mappings.forEach(mapping => {
1908+
if (!mapping.file) {
1909+
return { contents: mapping.contents };
1910+
}
1911+
const { file, project } = this.getFileAndProjectWorker(mapping.file, mapping.projectFileName);
1912+
const scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file)!;
1913+
const focusLocations = mapping.focusLocations?.map(spans => {
1914+
return spans.map(loc => {
1915+
const start = scriptInfo.lineOffsetToPosition(loc.start.line, loc.start.offset);
1916+
const end = scriptInfo.lineOffsetToPosition(loc.end.line, loc.end.offset);
1917+
return {
1918+
start,
1919+
length: end - start,
1920+
};
1921+
});
1922+
});
1923+
if (!projects.has(project)) {
1924+
projects.set(project, []);
1925+
}
1926+
projects.get(project)!.push({
1927+
contents: mapping.contents,
1928+
fileName: file,
1929+
focusLocations
1930+
});
1931+
});
1932+
const updates = args.updates?.map(edit => {
1933+
const file = this.getFileAndProjectWorker(edit.fileName, /*projectFileName*/ undefined).file;
1934+
const scriptInfo = this.projectService.getScriptInfoForNormalizedPath(file)!;
1935+
return {
1936+
fileName: edit.fileName,
1937+
textChanges: edit.textChanges.map(({ start, end, newText }) => {
1938+
const newStart = scriptInfo.lineOffsetToPosition(start.line, start.offset);
1939+
const newEnd = scriptInfo.lineOffsetToPosition(end.line, end.offset);
1940+
return {
1941+
span: { start: newStart, length: newEnd - newStart },
1942+
newText,
1943+
};
1944+
}),
1945+
};
1946+
});
1947+
1948+
return [...projects.entries()].map(([project, mappings]) => {
1949+
return project.getLanguageService().mapCode(mappings, formatOptions, preferences, updates);
1950+
}).filter(x => x).flatMap((changes) => {
1951+
return changes.map(change => {
1952+
const scriptInfo = this.projectService.getScriptInfoForNormalizedPath(toNormalizedPath(change.fileName))!;
1953+
return {
1954+
fileName: change.fileName,
1955+
textChanges: change.textChanges.map(({ span, newText }) => {
1956+
const newSpan = toProtocolTextSpan(span, scriptInfo);
1957+
return {
1958+
start: newSpan.start,
1959+
end: newSpan.end,
1960+
newText
1961+
};
1962+
}),
1963+
};
1964+
});
1965+
});
1966+
}
1967+
19021968
private setCompilerOptionsForInferredProjects(args: protocol.SetCompilerOptionsForInferredProjectsArgs): void {
19031969
this.projectService.setCompilerOptionsForInferredProjects(args.options, args.projectRootPath);
19041970
}
@@ -3575,6 +3641,9 @@ export class Session<TMessage = string> implements EventSender {
35753641
[protocol.CommandTypes.ProvideInlayHints]: (request: protocol.InlayHintsRequest) => {
35763642
return this.requiredResponse(this.provideInlayHints(request.arguments));
35773643
},
3644+
[protocol.CommandTypes.MapCode]: (request: protocol.MapCodeRequest) => {
3645+
return this.requiredResponse(this.mapCode(request.arguments));
3646+
},
35783647
}));
35793648

35803649
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
@@ -32,6 +32,8 @@ import * as GoToDefinition from "./ts.GoToDefinition";
3232
export { GoToDefinition };
3333
import * as InlayHints from "./ts.InlayHints";
3434
export { InlayHints };
35+
import * as MapCode from "./ts.MapCode";
36+
export { MapCode };
3537
import * as JsDoc from "./ts.JsDoc";
3638
export { JsDoc };
3739
import * as NavigateTo from "./ts.NavigateTo";

0 commit comments

Comments
 (0)