mirror of
https://github.com/adelphes/android-dev-ext.git
synced 2025-12-22 09:29:38 +00:00
* replace mixpanel with amplitude * remove unused uuid require * include partial count fields to allow easier analysis * enable real logging * record extension deactivation
189 lines
7.4 KiB
JavaScript
189 lines
7.4 KiB
JavaScript
// The module 'vscode' contains the VS Code extensibility API
|
|
// Import the module and reference it with the alias vscode in your code below
|
|
const path = require('path');
|
|
const vscode = require('vscode');
|
|
const analytics = require('./langserver/analytics');
|
|
const package_json = require('./package.json');
|
|
const { LanguageClient, TransportKind, } = require('vscode-languageclient');
|
|
const { AndroidContentProvider } = require('./src/contentprovider');
|
|
const { openLogcatWindow } = require('./src/logcat');
|
|
const { selectAndroidProcessID } = require('./src/process-attach');
|
|
const { selectTargetDevice } = require('./src/utils/device');
|
|
|
|
/**
|
|
* @param {vscode.ExtensionContext} context
|
|
* @param {string} uid
|
|
* @param {number} session_id
|
|
* @param {*} vscode_props
|
|
*/
|
|
async function createLanguageClient(context, uid, session_id, vscode_props) {
|
|
// The server is implemented in node
|
|
let serverModule = context.asAbsolutePath(path.join('langserver', 'server.js'));
|
|
// The debug options for the server
|
|
// --inspect=6009: runs the server in Node's Inspector mode so VS Code can attach to the server for debugging
|
|
let debugOptions = { execArgv: ['--nolazy', '--inspect=6009'] };
|
|
|
|
// If the extension is launched in debug mode then the debug server options are used
|
|
// Otherwise the run options are used
|
|
/** @type {import('vscode-languageclient').ServerOptions} */
|
|
let serverOptions = {
|
|
run: {
|
|
module: serverModule,
|
|
transport: TransportKind.ipc,
|
|
},
|
|
debug: {
|
|
module: serverModule,
|
|
transport: TransportKind.ipc,
|
|
options: debugOptions
|
|
}
|
|
};
|
|
|
|
const config = vscode.workspace.getConfiguration('android-dev-ext');
|
|
const appSourceRoot = config.get('appSourceRoot', '');
|
|
let globSearchRoot = appSourceRoot;
|
|
if (globSearchRoot) {
|
|
// for findFiles to work properly, the path cannot begin with slash or have any relative components
|
|
globSearchRoot = path.normalize(appSourceRoot.replace(/(^[\\/]+)|([\\/]+$)/,''));
|
|
if (globSearchRoot) globSearchRoot += '/';
|
|
}
|
|
const sourceFiles = (await vscode.workspace.findFiles(`${globSearchRoot}**/*.java`, null, 1000, null)).map(uri => uri.toString());
|
|
|
|
// Options to control the language client
|
|
/** @type {import('vscode-languageclient').LanguageClientOptions} */
|
|
let clientOptions = {
|
|
// Register the server for Java source documents
|
|
documentSelector: [{
|
|
scheme: 'file', language: 'java'
|
|
}],
|
|
initializationOptions: {
|
|
// extensionPath points to the root of the extension (the folder where this file is)
|
|
extensionPath: context.extensionPath,
|
|
uid,
|
|
session_id,
|
|
vscode_props,
|
|
initialSettings: config,
|
|
sourceFiles,
|
|
workspaceFolders: (vscode.workspace.workspaceFolders || []).map(z => z.uri.toString()),
|
|
},
|
|
synchronize: {
|
|
// Notify the server about file changes to '.java files contained in the workspace
|
|
fileEvents: vscode.workspace.createFileSystemWatcher('**/*.java')
|
|
},
|
|
};
|
|
|
|
// Create the language client - this won't do anything until start() is called
|
|
return new LanguageClient(
|
|
'androidJavaLanguageServer',
|
|
'Android',
|
|
serverOptions,
|
|
clientOptions
|
|
);
|
|
}
|
|
|
|
/** @type {LanguageClient} */
|
|
let languageClient;
|
|
let languageSupportEnabled = false;
|
|
function refreshLanguageServerEnabledState() {
|
|
if (!languageClient) {
|
|
return;
|
|
}
|
|
let langSupport = vscode.workspace.getConfiguration('android-dev-ext').get('languageSupport', false);
|
|
if (langSupport === languageSupportEnabled) {
|
|
return;
|
|
}
|
|
if (langSupport) {
|
|
if (languageClient.needsStart()) {
|
|
languageClient.start();
|
|
}
|
|
languageSupportEnabled = true;
|
|
} else {
|
|
if (languageClient.needsStop()) {
|
|
languageClient.stop().then(() => {
|
|
languageSupportEnabled = false;
|
|
refreshLanguageServerEnabledState();
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @param {vscode.ExtensionContext} context
|
|
*/
|
|
function activate(context) {
|
|
|
|
/* Only the logcat stuff is configured here. The debugger is launched from src/debugMain.js */
|
|
AndroidContentProvider.register(context, vscode.workspace);
|
|
|
|
const { uid } = analytics.getIDs(context);
|
|
const session_id = Math.trunc(Math.random() * Number.MAX_SAFE_INTEGER);
|
|
const vscode_props = {
|
|
appName: vscode.env.appName,
|
|
language: vscode.env.language,
|
|
shell: vscode.env.shell,
|
|
uiKind: vscode.env.uiKind,
|
|
vscode_version: vscode.version,
|
|
}
|
|
analytics.init(undefined, uid, session_id, '', package_json, vscode_props, 'extension-start');
|
|
|
|
createLanguageClient(context, uid, session_id, vscode_props).then(client => {
|
|
languageClient = client;
|
|
refreshLanguageServerEnabledState();
|
|
});
|
|
|
|
// The commandId parameter must match the command field in package.json
|
|
const disposables = [
|
|
// add the view logcat handler
|
|
vscode.commands.registerCommand('android-dev-ext.view_logcat', () => {
|
|
openLogcatWindow(vscode);
|
|
}),
|
|
// add the device picker handler - used to choose a target device
|
|
vscode.commands.registerCommand('PickAndroidDevice', async (launchConfig) => {
|
|
// if the config has both PickAndroidDevice and PickAndroidProcess, ignore this
|
|
// request as PickAndroidProcess already includes chooosing a device...
|
|
if (launchConfig && launchConfig.processId === '${command:PickAndroidProcess}') {
|
|
return '';
|
|
}
|
|
const device = await selectTargetDevice(vscode, "Launch", { alwaysShow:true });
|
|
// the debugger requires a string value to be returned
|
|
return JSON.stringify(device);
|
|
}),
|
|
// add the process picker handler - used to choose a PID to attach to
|
|
vscode.commands.registerCommand('PickAndroidProcess', async (launchConfig) => {
|
|
// if the config has a targetDevice specified, use it instead of choosing a device...
|
|
let target_device = '';
|
|
if (launchConfig && typeof launchConfig.targetDevice === 'string') {
|
|
target_device = launchConfig.targetDevice;
|
|
}
|
|
const explicit_pick_device = target_device === '${command:PickAndroidDevice}';
|
|
if (!target_device || explicit_pick_device) {
|
|
// no targetDevice (or it's set to ${command:PickAndroidDevice})
|
|
const device = await selectTargetDevice(vscode, 'Attach', { alwaysShow: explicit_pick_device });
|
|
if (!device) {
|
|
return JSON.stringify({status: 'cancelled'});
|
|
}
|
|
target_device = device.serial;
|
|
}
|
|
const o = await selectAndroidProcessID(vscode, target_device);
|
|
// the debugger requires a string value to be returned
|
|
return JSON.stringify(o);
|
|
}),
|
|
|
|
vscode.workspace.onDidChangeConfiguration(e => {
|
|
// perform the refresh on the next tick to prevent spurious errors when
|
|
// trying to shut down the language server in the middle of a change-configuration request
|
|
process.nextTick(() => refreshLanguageServerEnabledState());
|
|
}),
|
|
];
|
|
|
|
context.subscriptions.splice(context.subscriptions.length, 0, ...disposables);
|
|
}
|
|
|
|
// this method is called when your extension is deactivated
|
|
function deactivate() {
|
|
analytics.event('extension-deactivate');
|
|
}
|
|
|
|
exports.activate = activate;
|
|
exports.deactivate = deactivate;
|