From 90770d236fe50a4f1874c4934d7158a82b5b3808 Mon Sep 17 00:00:00 2001 From: Tereza Tomcova Date: Wed, 5 Oct 2016 17:22:58 +0200 Subject: [PATCH] Call AllowSetForegroundWindow before sending IPC to the running instance on Windows (Fixes #929) --- build/gulpfile.vscode.js | 1 + npm-shrinkwrap.json | 5 +++++ package.json | 1 + src/typings/windows-foreground-love.d.ts | 10 ++++++++++ src/vs/code/electron-main/launch.ts | 20 +++++++++++++++++--- src/vs/code/electron-main/main.ts | 13 ++++++++++++- 6 files changed, 46 insertions(+), 4 deletions(-) create mode 100644 src/typings/windows-foreground-love.d.ts diff --git a/build/gulpfile.vscode.js b/build/gulpfile.vscode.js index dabac01f5ab39..af5a5fe1fc3ea 100644 --- a/build/gulpfile.vscode.js +++ b/build/gulpfile.vscode.js @@ -268,6 +268,7 @@ function packageTask(platform, arch, opts) { .pipe(util.cleanNodeModule('oniguruma', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['**/*.node'])) .pipe(util.cleanNodeModule('windows-mutex', ['binding.gyp', 'build/**', 'src/**'], ['**/*.node'])) .pipe(util.cleanNodeModule('native-keymap', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['**/*.node'])) + .pipe(util.cleanNodeModule('windows-foreground-love', ['binding.gyp', 'build/**', 'src/**'], ['**/*.node'])) .pipe(util.cleanNodeModule('gc-signals', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['**/*.node', 'src/index.js'])) .pipe(util.cleanNodeModule('pty.js', ['binding.gyp', 'build/**', 'src/**', 'deps/**'], ['build/Release/**'])); diff --git a/npm-shrinkwrap.json b/npm-shrinkwrap.json index 0b3fd12858e0a..0a9e33916cff0 100644 --- a/npm-shrinkwrap.json +++ b/npm-shrinkwrap.json @@ -414,6 +414,11 @@ "from": "vscode-textmate@2.2.0", "resolved": "https://registry.npmjs.org/vscode-textmate/-/vscode-textmate-2.2.0.tgz" }, + "windows-foreground-love": { + "version": "0.1.0", + "from": "windows-foreground-love@0.1.0", + "resolved": "https://registry.npmjs.org/windows-foreground-love/-/windows-foreground-love-0.1.0.tgz" + }, "windows-mutex": { "version": "0.2.0", "from": "windows-mutex@>=0.2.0 <0.3.0", diff --git a/package.json b/package.json index 8ad3c9f600fda..98f8977c72632 100644 --- a/package.json +++ b/package.json @@ -108,6 +108,7 @@ } }, "optionalDependencies": { + "windows-foreground-love": "0.1.0", "windows-mutex": "^0.2.0", "fsevents": "0.3.8" } diff --git a/src/typings/windows-foreground-love.d.ts b/src/typings/windows-foreground-love.d.ts new file mode 100644 index 0000000000000..f7d20f13aef8e --- /dev/null +++ b/src/typings/windows-foreground-love.d.ts @@ -0,0 +1,10 @@ +/*--------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All rights reserved. + * Licensed under the MIT License. See License.txt in the project root for license information. + *--------------------------------------------------------------------------------------------*/ + +declare module 'windows-foreground-love' { + + export function allowSetForegroundWindow(pid?: number): boolean; + +} \ No newline at end of file diff --git a/src/vs/code/electron-main/launch.ts b/src/vs/code/electron-main/launch.ts index c708a942011ad..185cd639486e0 100644 --- a/src/vs/code/electron-main/launch.ts +++ b/src/vs/code/electron-main/launch.ts @@ -21,10 +21,12 @@ export interface IStartArguments { export interface ILaunchService { start(args: ParsedArgs, userEnv: IProcessEnvironment): TPromise; + getMainProcessId(): TPromise; } export interface ILaunchChannel extends IChannel { call(command: 'start', arg: IStartArguments): TPromise; + call(command: 'get-main-process-id', arg: null): TPromise; call(command: string, arg: any): TPromise; } @@ -33,10 +35,13 @@ export class LaunchChannel implements ILaunchChannel { constructor(private service: ILaunchService) { } call(command: string, arg: any): TPromise { - const { args, userEnv } = arg as IStartArguments; - switch (command) { - case 'start': return this.service.start(args, userEnv); + case 'start': + const { args, userEnv } = arg as IStartArguments; + return this.service.start(args, userEnv); + + case 'get-main-process-id': + return this.service.getMainProcessId(); } } } @@ -48,6 +53,10 @@ export class LaunchChannelClient implements ILaunchService { start(args: ParsedArgs, userEnv: IProcessEnvironment): TPromise { return this.channel.call('start', { args, userEnv }); } + + getMainProcessId(): TPromise { + return this.channel.call('get-main-process-id', null); + } } export class LaunchService implements ILaunchService { @@ -105,4 +114,9 @@ export class LaunchService implements ILaunchService { return TPromise.as(null); } + + getMainProcessId(): TPromise { + this.logService.log('Received request for process ID from other instance.'); + return TPromise.as(process.pid); + } } \ No newline at end of file diff --git a/src/vs/code/electron-main/main.ts b/src/vs/code/electron-main/main.ts index 8a1798c023dbf..8248fb807ce22 100644 --- a/src/vs/code/electron-main/main.ts +++ b/src/vs/code/electron-main/main.ts @@ -25,6 +25,7 @@ import { AskpassChannel } from 'vs/workbench/parts/git/common/gitIpc'; import { GitAskpassService } from 'vs/workbench/parts/git/electron-main/askpassService'; import { spawnSharedProcess } from 'vs/code/node/sharedProcess'; import { Mutex } from 'windows-mutex'; +import { allowSetForegroundWindow } from 'windows-foreground-love'; import { LaunchService, ILaunchChannel, LaunchChannel, LaunchChannelClient } from './launch'; import { ServicesAccessor, IInstantiationService } from 'vs/platform/instantiation/common/instantiation'; import { InstantiationService } from 'vs/platform/instantiation/common/instantiationService'; @@ -299,7 +300,17 @@ function setupIPC(accessor: ServicesAccessor): TPromise { const channel = client.getChannel('launch'); const service = new LaunchChannelClient(channel); - return service.start(environmentService.args, process.env) + let promise = TPromise.as(null); + if (platform.isWindows) { + promise = service.getMainProcessId() + .then(processId => { + logService.log('Sending some foreground love to the running instance:', processId); + allowSetForegroundWindow(processId); + }); + } + + return promise + .then(() => service.start(environmentService.args, process.env)) .then(() => client.dispose()) .then(() => TPromise.wrapError('Sent env to running instance. Terminating...')); },