Skip to content

Commit 9bb8233

Browse files
committed
CodeMapper support
1 parent 1488256 commit 9bb8233

File tree

10 files changed

+150
-1
lines changed

10 files changed

+150
-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[], formatOptions: ts.FormatCodeSettings, preferences: ts.UserPreferences = ts.emptyOptions, focusLocations?: ts.PrioritizedSpan[], updates?: ts.FileTextChanges[]): ts.FileTextChanges[] {
702+
return unwrapJSONCallResult(this.shim.mapCode(fileName, contents, formatOptions, preferences, focusLocations, 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

+48
Original file line numberDiff line numberDiff line change
@@ -1887,6 +1887,51 @@ 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 formatOptions = this.getHostFormatOptions();
1894+
const preferences = this.getHostPreferences();
1895+
const selected = args.focusLocations?.map(loc => {
1896+
const start = scriptInfo.lineOffsetToPosition(loc.start.line, loc.start.offset);
1897+
const end = scriptInfo.lineOffsetToPosition(loc.end.line, loc.end.offset);
1898+
return {
1899+
fileName: loc.file,
1900+
priority: loc.priority,
1901+
textSpan: {
1902+
start,
1903+
length: end - start,
1904+
}
1905+
};
1906+
});
1907+
const updates = args.updates?.map(edit => {
1908+
return {
1909+
fileName: edit.fileName,
1910+
textChanges: edit.textChanges.map(({ start, end, newText }) => {
1911+
const newStart = scriptInfo.lineOffsetToPosition(start.line, start.offset);
1912+
const newEnd = scriptInfo.lineOffsetToPosition(end.line, end.offset);
1913+
return {
1914+
span: { start: newStart, length: newEnd - newStart },
1915+
newText,
1916+
};
1917+
}),
1918+
};
1919+
});
1920+
return project.getLanguageService().mapCode(file, args.contents, formatOptions, preferences, selected, updates)?.map(change => {
1921+
return {
1922+
fileName: change.fileName,
1923+
textChanges: change.textChanges.map(({ span, newText }) => {
1924+
const newSpan = toProtocolTextSpan(span, scriptInfo);
1925+
return {
1926+
start: newSpan.start,
1927+
end: newSpan.end,
1928+
newText
1929+
};
1930+
}),
1931+
};
1932+
});
1933+
}
1934+
18901935
private setCompilerOptionsForInferredProjects(args: protocol.SetCompilerOptionsForInferredProjectsArgs): void {
18911936
this.projectService.setCompilerOptionsForInferredProjects(args.options, args.projectRootPath);
18921937
}
@@ -3556,6 +3601,9 @@ export class Session<TMessage = string> implements EventSender {
35563601
[protocol.CommandTypes.ProvideInlayHints]: (request: protocol.InlayHintsRequest) => {
35573602
return this.requiredResponse(this.provideInlayHints(request.arguments));
35583603
},
3604+
[protocol.CommandTypes.MapCode]: (request: protocol.MapCodeRequest) => {
3605+
return this.requiredResponse(this.mapCode(request.arguments));
3606+
},
35593607
}));
35603608

35613609
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

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

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[], formatOptions: FormatCodeSettings, preferences: UserPreferences = emptyOptions, focusLocations?: PrioritizedSpan[], updates?: FileTextChanges[]): FileTextChanges[] {
3076+
synchronizeHostData();
3077+
const sourceFile = getValidSourceFile(fileName);
3078+
return MapCode.mapCode(sourceFile, program, host, formatting.getFormatContext(formatOptions, host), preferences, sourceMapper, 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[], formatOptions: FormatCodeSettings, preferences: UserPreferences, focusLocations?: 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[], formatOptions: FormatCodeSettings, preferences: UserPreferences = emptyOptions, focusLocations?: PrioritizedSpan[], updates?: FileTextChanges[]): string {
1164+
return this.forwardJSONCall(
1165+
`mapCode('${fileName}', '${JSON.stringify(contents)}', '${JSON.stringify(formatOptions)}', '${JSON.stringify(preferences)}', '${JSON.stringify(focusLocations)}', '${JSON.stringify(updates)}')`,
1166+
() => this.languageService.mapCode(fileName, contents, formatOptions, preferences, 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[], formatOptions: FormatCodeSettings, preferences: UserPreferences, 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)