mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-18 09:35:39 -05:00
145 lines
4.4 KiB
TypeScript
145 lines
4.4 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import { assign } from 'vs/base/common/objects';
|
|
import { memoize } from 'vs/base/common/decorators';
|
|
import { IEnvironmentService } from 'vs/platform/environment/common/environment';
|
|
import { BrowserWindow, ipcMain } from 'electron';
|
|
import { ISharedProcess } from 'vs/platform/windows/electron-main/windows';
|
|
import { Barrier } from 'vs/base/common/async';
|
|
import { ILogService } from 'vs/platform/log/common/log';
|
|
import { ILifecycleService } from 'vs/platform/lifecycle/electron-main/lifecycleMain';
|
|
import { IStateService } from 'vs/platform/state/common/state';
|
|
import { getBackgroundColor } from 'vs/code/electron-main/theme';
|
|
import { dispose, toDisposable, IDisposable } from 'vs/base/common/lifecycle';
|
|
|
|
export class SharedProcess implements ISharedProcess {
|
|
|
|
private barrier = new Barrier();
|
|
|
|
private window: Electron.BrowserWindow | null;
|
|
|
|
constructor(
|
|
private readonly machineId: string,
|
|
private userEnv: NodeJS.ProcessEnv,
|
|
@IEnvironmentService private readonly environmentService: IEnvironmentService,
|
|
@ILifecycleService private readonly lifecycleService: ILifecycleService,
|
|
@IStateService private readonly stateService: IStateService,
|
|
@ILogService private readonly logService: ILogService
|
|
) { }
|
|
|
|
@memoize
|
|
private get _whenReady(): Promise<void> {
|
|
this.window = new BrowserWindow({
|
|
show: false,
|
|
backgroundColor: getBackgroundColor(this.stateService),
|
|
webPreferences: {
|
|
images: false,
|
|
webaudio: false,
|
|
webgl: false,
|
|
disableBlinkFeatures: 'Auxclick' // do NOT change, allows us to identify this window as shared-process in the process explorer
|
|
}
|
|
});
|
|
const config = assign({
|
|
appRoot: this.environmentService.appRoot,
|
|
machineId: this.machineId,
|
|
nodeCachedDataDir: this.environmentService.nodeCachedDataDir,
|
|
userEnv: this.userEnv
|
|
});
|
|
|
|
const url = `${require.toUrl('vs/code/electron-browser/sharedProcess/sharedProcess.html')}?config=${encodeURIComponent(JSON.stringify(config))}`;
|
|
this.window.loadURL(url);
|
|
|
|
// Prevent the window from dying
|
|
const onClose = (e: Event) => {
|
|
this.logService.trace('SharedProcess#close prevented');
|
|
|
|
// We never allow to close the shared process unless we get explicitly disposed()
|
|
e.preventDefault();
|
|
|
|
// Still hide the window though if visible
|
|
if (this.window && this.window.isVisible()) {
|
|
this.window.hide();
|
|
}
|
|
};
|
|
|
|
this.window.on('close', onClose);
|
|
|
|
const disposables: IDisposable[] = [];
|
|
|
|
this.lifecycleService.onWillShutdown(() => {
|
|
dispose(disposables);
|
|
|
|
// Shut the shared process down when we are quitting
|
|
//
|
|
// Note: because we veto the window close, we must first remove our veto.
|
|
// Otherwise the application would never quit because the shared process
|
|
// window is refusing to close!
|
|
//
|
|
if (this.window) {
|
|
this.window.removeListener('close', onClose);
|
|
}
|
|
|
|
// Electron seems to crash on Windows without this setTimeout :|
|
|
setTimeout(() => {
|
|
try {
|
|
if (this.window) {
|
|
this.window.close();
|
|
}
|
|
} catch (err) {
|
|
// ignore, as electron is already shutting down
|
|
}
|
|
|
|
this.window = null;
|
|
}, 0);
|
|
});
|
|
|
|
return new Promise<void>(c => {
|
|
ipcMain.once('handshake:hello', ({ sender }: { sender: any }) => {
|
|
sender.send('handshake:hey there', {
|
|
sharedIPCHandle: this.environmentService.sharedIPCHandle,
|
|
args: this.environmentService.args,
|
|
logLevel: this.logService.getLevel()
|
|
});
|
|
|
|
disposables.push(toDisposable(() => sender.send('handshake:goodbye')));
|
|
ipcMain.once('handshake:im ready', () => c(undefined));
|
|
});
|
|
});
|
|
}
|
|
|
|
spawn(userEnv: NodeJS.ProcessEnv): void {
|
|
this.userEnv = { ...this.userEnv, ...userEnv };
|
|
this.barrier.open();
|
|
}
|
|
|
|
async whenReady(): Promise<void> {
|
|
await this.barrier.wait();
|
|
await this._whenReady;
|
|
}
|
|
|
|
toggle(): void {
|
|
if (!this.window || this.window.isVisible()) {
|
|
this.hide();
|
|
} else {
|
|
this.show();
|
|
}
|
|
}
|
|
|
|
show(): void {
|
|
if (this.window) {
|
|
this.window.show();
|
|
this.window.webContents.openDevTools();
|
|
}
|
|
}
|
|
|
|
hide(): void {
|
|
if (this.window) {
|
|
this.window.webContents.closeDevTools();
|
|
this.window.hide();
|
|
}
|
|
}
|
|
}
|