import {TextDocument} from 'vscode-languageserver-textdocument';
import {
CodeLens,
createConnection,
type InitializeParams,
type InitializeResult,
ProposedFeatures,
TextDocuments,
TextDocumentSyncKind,
} from 'vscode-languageserver/node';
import {compile, lastResult} from './compiler';
import {
type CompileSuccessEvent,
type LoggerEvent,
type PluginOptions,
defaultOptions,
} from 'babel-plugin-react-compiler';
import {babelLocationToRange, getRangeFirstCharacter} from './compiler/compat';
const SUPPORTED_LANGUAGE_IDS = new Set([
'javascript',
'javascriptreact',
'typescript',
'typescriptreact',
]);
const connection = createConnection(ProposedFeatures.all);
const documents = new TextDocuments(TextDocument);
let compilerOptions: PluginOptions | null = null;
let compiledFns: Set<CompileSuccessEvent> = new Set();
connection.onInitialize((_params: InitializeParams) => {
compilerOptions = defaultOptions;
compilerOptions = {
...compilerOptions,
logger: {
logEvent(_filename: string | null, event: LoggerEvent) {
connection.console.info(`Received event: ${event.kind}`);
connection.console.debug(JSON.stringify(event, null, 2));
if (event.kind === 'CompileSuccess') {
compiledFns.add(event);
}
},
},
};
const result: InitializeResult = {
capabilities: {
textDocumentSync: TextDocumentSyncKind.Full,
codeLensProvider: {resolveProvider: true},
},
};
return result;
});
connection.onInitialized(() => {
connection.console.log('initialized');
});
documents.onDidChangeContent(async event => {
connection.console.info(`Compiling: ${event.document.uri}`);
resetState();
if (SUPPORTED_LANGUAGE_IDS.has(event.document.languageId)) {
const text = event.document.getText();
try {
await compile({
text,
file: event.document.uri,
options: compilerOptions,
});
} catch (err) {
connection.console.error('Failed to compile');
if (err instanceof Error) {
connection.console.error(err.stack ?? err.message);
} else {
connection.console.error(JSON.stringify(err, null, 2));
}
}
}
});
connection.onDidChangeWatchedFiles(change => {
resetState();
connection.console.log(
change.changes.map(c => `File changed: ${c.uri}`).join('\n'),
);
});
connection.onCodeLens(params => {
connection.console.info(`Handling codelens for: ${params.textDocument.uri}`);
if (compiledFns.size === 0) {
return;
}
const lenses: Array<CodeLens> = [];
for (const compiled of compiledFns) {
if (compiled.fnLoc != null) {
const fnLoc = babelLocationToRange(compiled.fnLoc);
if (fnLoc === null) continue;
const lens = CodeLens.create(
getRangeFirstCharacter(fnLoc),
compiled.fnLoc,
);
if (lastResult?.code != null) {
lens.command = {
title: 'Optimized by React Compiler',
command: 'todo',
};
}
lenses.push(lens);
}
}
return lenses;
});
connection.onCodeLensResolve(lens => {
connection.console.info(`Resolving codelens for: ${JSON.stringify(lens)}`);
if (lastResult?.code != null) {
connection.console.log(lastResult.code);
}
return lens;
});
function resetState() {
connection.console.debug('Clearing state');
compiledFns.clear();
}
documents.listen(connection);
connection.listen();
connection.console.info(`React Analyzer running in node ${process.version}`);