window.ts 33.9 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

6
import * as path from 'vs/base/common/path';
J
Joao Moreno 已提交
7
import * as objects from 'vs/base/common/objects';
8
import * as nls from 'vs/nls';
9
import { URI } from 'vs/base/common/uri';
10
import { screen, BrowserWindow, systemPreferences, app, TouchBar, nativeImage, Rectangle, Display, TouchBarSegmentedControl, NativeImage, BrowserWindowConstructorOptions, SegmentedControlSegment } from 'electron';
J
Joao Moreno 已提交
11
import { IEnvironmentService, ParsedArgs } from 'vs/platform/environment/common/environment';
12
import { ILogService } from 'vs/platform/log/common/log';
13
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
14
import { parseArgs, OPTIONS } from 'vs/platform/environment/node/argv';
15
import product from 'vs/platform/product/common/product';
16
import { IWindowSettings, MenuBarVisibility, IWindowConfiguration, ReadyState, getTitleBarStyle } from 'vs/platform/windows/common/windows';
B
Benjamin Pasero 已提交
17
import { Disposable, toDisposable } from 'vs/base/common/lifecycle';
B
Benjamin Pasero 已提交
18
import { isLinux, isMacintosh, isWindows } from 'vs/base/common/platform';
19
import { ICodeWindow, IWindowState, WindowMode } from 'vs/platform/windows/electron-main/windows';
B
Benjamin Pasero 已提交
20 21
import { IWorkspaceIdentifier } from 'vs/platform/workspaces/common/workspaces';
import { IWorkspacesMainService } from 'vs/platform/workspaces/electron-main/workspacesMainService';
22
import { IBackupMainService } from 'vs/platform/backup/electron-main/backup';
B
Benjamin Pasero 已提交
23
import { ISerializableCommandAction } from 'vs/platform/actions/common/actions';
24
import * as perf from 'vs/base/common/performance';
25
import { resolveMarketplaceHeaders } from 'vs/platform/extensionManagement/common/extensionGalleryService';
26
import { IThemeMainService } from 'vs/platform/theme/electron-main/themeMainService';
27
import { endsWith } from 'vs/base/common/strings';
28
import { RunOnceScheduler } from 'vs/base/common/async';
29
import { IFileService } from 'vs/platform/files/common/files';
30

31 32
const RUN_TEXTMATE_IN_WORKER = false;

33 34
export interface IWindowCreationOptions {
	state: IWindowState;
35
	extensionDevelopmentPath?: string[];
36
	isExtensionTestHost?: boolean;
37 38
}

B
Benjamin Pasero 已提交
39
export const defaultWindowState = function (mode = WindowMode.Normal): IWindowState {
E
Erich Gamma 已提交
40 41 42
	return {
		width: 1024,
		height: 768,
43
		mode
E
Erich Gamma 已提交
44 45 46
	};
};

47
interface ITouchBarSegment extends SegmentedControlSegment {
B
Benjamin Pasero 已提交
48 49 50
	id: string;
}

B
Benjamin Pasero 已提交
51
export class CodeWindow extends Disposable implements ICodeWindow {
E
Erich Gamma 已提交
52

53 54
	private static readonly MIN_WIDTH = 200;
	private static readonly MIN_HEIGHT = 120;
E
Erich Gamma 已提交
55

B
Benjamin Pasero 已提交
56
	private static readonly MAX_URL_LENGTH = 2 * 1024 * 1024; // https://cs.chromium.org/chromium/src/url/url_constants.cc?l=32
57

B
Benjamin Pasero 已提交
58
	private hiddenTitleBarStyle: boolean;
59
	private showTimeoutHandle: NodeJS.Timeout;
B
Benjamin Pasero 已提交
60
	private _id: number;
61
	private _win: BrowserWindow;
E
Erich Gamma 已提交
62 63 64
	private _lastFocusTime: number;
	private _readyState: ReadyState;
	private windowState: IWindowState;
65
	private currentMenuBarVisibility: MenuBarVisibility;
66
	private representedFilename: string;
E
Erich Gamma 已提交
67

68
	private readonly whenReadyCallbacks: { (window: ICodeWindow): void }[];
E
Erich Gamma 已提交
69 70

	private currentConfig: IWindowConfiguration;
71
	private pendingLoadConfig?: IWindowConfiguration;
E
Erich Gamma 已提交
72

J
Johannes Rieken 已提交
73
	private marketplaceHeadersPromise: Promise<object>;
74

75
	private readonly touchBarGroups: TouchBarSegmentedControl[];
76

J
Joao Moreno 已提交
77 78
	constructor(
		config: IWindowCreationOptions,
79 80
		@ILogService private readonly logService: ILogService,
		@IEnvironmentService private readonly environmentService: IEnvironmentService,
81
		@IFileService private readonly fileService: IFileService,
82
		@IConfigurationService private readonly configurationService: IConfigurationService,
83
		@IThemeMainService private readonly themeMainService: IThemeMainService,
84 85
		@IWorkspacesMainService private readonly workspacesMainService: IWorkspacesMainService,
		@IBackupMainService private readonly backupMainService: IBackupMainService,
J
Joao Moreno 已提交
86
	) {
B
Benjamin Pasero 已提交
87 88
		super();

B
Benjamin Pasero 已提交
89
		this.touchBarGroups = [];
E
Erich Gamma 已提交
90 91 92 93
		this._lastFocusTime = -1;
		this._readyState = ReadyState.NONE;
		this.whenReadyCallbacks = [];

B
Benjamin Pasero 已提交
94 95 96 97 98 99
		// create browser window
		this.createBrowserWindow(config);

		// respect configured menu bar visibility
		this.onConfigurationUpdated();

B
Benjamin Pasero 已提交
100 101 102
		// macOS: touch bar support
		this.createTouchBar();

103 104 105
		// Request handling
		this.handleMarketplaceRequests();

B
Benjamin Pasero 已提交
106 107 108 109 110 111
		// Eventing
		this.registerListeners();
	}

	private createBrowserWindow(config: IWindowCreationOptions): void {

E
Erich Gamma 已提交
112
		// Load window state
B
Benjamin Pasero 已提交
113 114
		const [state, hasMultipleDisplays] = this.restoreWindowState(config.state);
		this.windowState = state;
E
Erich Gamma 已提交
115

116
		// in case we are maximized or fullscreen, only show later after the call to maximize/fullscreen (see below)
B
Benjamin Pasero 已提交
117
		const isFullscreenOrMaximized = (this.windowState.mode === WindowMode.Maximized || this.windowState.mode === WindowMode.Fullscreen);
118

119
		const options: BrowserWindowConstructorOptions = {
E
Erich Gamma 已提交
120 121 122 123
			width: this.windowState.width,
			height: this.windowState.height,
			x: this.windowState.x,
			y: this.windowState.y,
124
			backgroundColor: this.themeMainService.getBackgroundColor(),
B
Benjamin Pasero 已提交
125 126
			minWidth: CodeWindow.MIN_WIDTH,
			minHeight: CodeWindow.MIN_HEIGHT,
127
			show: !isFullscreenOrMaximized,
B
Benjamin Pasero 已提交
128
			title: product.nameLong,
129
			webPreferences: {
B
Benjamin Pasero 已提交
130 131 132 133
				// By default if Code is in the background, intervals and timeouts get throttled, so we
				// want to enforce that Code stays in the foreground. This triggers a disable_hidden_
				// flag that Electron provides via patch:
				// https://github.com/electron/libchromiumcontent/blob/master/patches/common/chromium/disable_hidden.patch
134 135
				backgroundThrottling: false,
				nodeIntegration: true,
136
				nodeIntegrationInWorker: RUN_TEXTMATE_IN_WORKER,
137
				webviewTag: true
138
			}
E
Erich Gamma 已提交
139 140
		};

141
		if (isLinux) {
142
			options.icon = path.join(this.environmentService.appRoot, 'resources/linux/code.png'); // Windows and Mac are better off using the embedded icon(s)
E
Erich Gamma 已提交
143 144
		}

145
		const windowConfig = this.configurationService.getValue<IWindowSettings>('window');
146

147 148 149 150
		if (isMacintosh && !this.useNativeFullScreen()) {
			options.fullscreenable = false; // enables simple fullscreen mode
		}

151 152 153 154 155 156 157 158
		if (isMacintosh) {
			options.acceptFirstMouse = true; // enabled by default

			if (windowConfig && windowConfig.clickThroughInactive === false) {
				options.acceptFirstMouse = false;
			}
		}

B
Benjamin Pasero 已提交
159 160
		const useNativeTabs = isMacintosh && windowConfig && windowConfig.nativeTabs === true;
		if (useNativeTabs) {
161 162 163
			options.tabbingIdentifier = product.nameShort; // this opts in to sierra tabs
		}

B
Benjamin Pasero 已提交
164
		const useCustomTitleStyle = getTitleBarStyle(this.configurationService, this.environmentService, !!config.extensionDevelopmentPath) === 'custom';
165 166 167
		if (useCustomTitleStyle) {
			options.titleBarStyle = 'hidden';
			this.hiddenTitleBarStyle = true;
R
Ryan Adolf 已提交
168
			if (!isMacintosh) {
R
Ryan Adolf 已提交
169 170
				options.frame = false;
			}
171 172
		}

E
Erich Gamma 已提交
173 174
		// Create the browser window.
		this._win = new BrowserWindow(options);
B
Benjamin Pasero 已提交
175
		this._id = this._win.id;
E
Erich Gamma 已提交
176

B
Benjamin Pasero 已提交
177
		if (isMacintosh && useCustomTitleStyle) {
178 179 180
			this._win.setSheetOffset(22); // offset dialogs by the height of the custom title bar if we have any
		}

B
Benjamin Pasero 已提交
181 182 183
		// TODO@Ben (Electron 4 regression): when running on multiple displays where the target display
		// to open the window has a larger resolution than the primary display, the window will not size
		// correctly unless we set the bounds again (https://github.com/microsoft/vscode/issues/74872)
184 185 186 187 188
		//
		// However, when running with native tabs with multiple windows we cannot use this workaround
		// because there is a potential that the new window will be added as native tab instead of being
		// a window on its own. In that case calling setBounds() would cause https://github.com/microsoft/vscode/issues/75830
		if (isMacintosh && hasMultipleDisplays && (!useNativeTabs || BrowserWindow.getAllWindows().length === 1)) {
B
Benjamin Pasero 已提交
189 190 191 192 193 194 195 196 197 198
			if ([this.windowState.width, this.windowState.height, this.windowState.x, this.windowState.y].every(value => typeof value === 'number')) {
				this._win.setBounds({
					width: this.windowState.width!,
					height: this.windowState.height!,
					x: this.windowState.x!,
					y: this.windowState.y!
				});
			}
		}

199
		if (isFullscreenOrMaximized) {
B
Benjamin Pasero 已提交
200
			this._win.maximize();
E
Erich Gamma 已提交
201

B
Benjamin Pasero 已提交
202
			if (this.windowState.mode === WindowMode.Fullscreen) {
203
				this.setFullScreen(true);
204 205
			}

B
Benjamin Pasero 已提交
206 207
			if (!this._win.isVisible()) {
				this._win.show(); // to reduce flicker from the default window size to maximize, we only show after maximize
E
Erich Gamma 已提交
208 209 210
			}
		}

211
		this._lastFocusTime = Date.now(); // since we show directly, we need to set the last focus time too
E
Erich Gamma 已提交
212 213
	}

B
Benjamin Pasero 已提交
214
	hasHiddenTitleBarStyle(): boolean {
B
Benjamin Pasero 已提交
215
		return this.hiddenTitleBarStyle;
216 217
	}

B
Benjamin Pasero 已提交
218
	get isExtensionDevelopmentHost(): boolean {
219
		return !!this.config.extensionDevelopmentPath;
220 221
	}

B
Benjamin Pasero 已提交
222
	get isExtensionTestHost(): boolean {
223
		return !!this.config.extensionTestsPath;
224 225
	}

B
Benjamin Pasero 已提交
226
	get config(): IWindowConfiguration {
E
Erich Gamma 已提交
227 228 229
		return this.currentConfig;
	}

B
Benjamin Pasero 已提交
230
	get id(): number {
B
Benjamin Pasero 已提交
231 232 233
		return this._id;
	}

234
	get win(): BrowserWindow {
E
Erich Gamma 已提交
235 236 237
		return this._win;
	}

B
Benjamin Pasero 已提交
238
	setRepresentedFilename(filename: string): void {
239
		if (isMacintosh) {
240 241 242 243 244 245
			this.win.setRepresentedFilename(filename);
		} else {
			this.representedFilename = filename;
		}
	}

B
Benjamin Pasero 已提交
246
	getRepresentedFilename(): string {
247
		if (isMacintosh) {
248 249 250 251 252 253
			return this.win.getRepresentedFilename();
		}

		return this.representedFilename;
	}

B
Benjamin Pasero 已提交
254
	focus(): void {
E
Erich Gamma 已提交
255 256 257 258
		if (!this._win) {
			return;
		}

B
Benjamin Pasero 已提交
259 260
		if (this._win.isMinimized()) {
			this._win.restore();
E
Erich Gamma 已提交
261 262
		}

B
Benjamin Pasero 已提交
263
		this._win.focus();
E
Erich Gamma 已提交
264 265
	}

B
Benjamin Pasero 已提交
266
	get lastFocusTime(): number {
E
Erich Gamma 已提交
267 268 269
		return this._lastFocusTime;
	}

270
	get backupPath(): string | undefined {
R
Rob Lourens 已提交
271
		return this.currentConfig ? this.currentConfig.backupPath : undefined;
E
Erich Gamma 已提交
272 273
	}

274
	get openedWorkspace(): IWorkspaceIdentifier | undefined {
R
Rob Lourens 已提交
275
		return this.currentConfig ? this.currentConfig.workspace : undefined;
276 277
	}

278
	get openedFolderUri(): URI | undefined {
R
Rob Lourens 已提交
279
		return this.currentConfig ? this.currentConfig.folderUri : undefined;
280 281
	}

282
	get remoteAuthority(): string | undefined {
R
Rob Lourens 已提交
283
		return this.currentConfig ? this.currentConfig.remoteAuthority : undefined;
M
Martin Aeschlimann 已提交
284 285
	}

B
Benjamin Pasero 已提交
286
	setReady(): void {
E
Erich Gamma 已提交
287 288 289 290
		this._readyState = ReadyState.READY;

		// inform all waiting promises that we are ready now
		while (this.whenReadyCallbacks.length) {
291
			this.whenReadyCallbacks.pop()!(this);
E
Erich Gamma 已提交
292 293 294
		}
	}

J
Johannes Rieken 已提交
295
	ready(): Promise<ICodeWindow> {
B
Benjamin Pasero 已提交
296 297 298
		return new Promise<ICodeWindow>(resolve => {
			if (this.isReady) {
				return resolve(this);
E
Erich Gamma 已提交
299 300 301
			}

			// otherwise keep and call later when we are ready
B
Benjamin Pasero 已提交
302
			this.whenReadyCallbacks.push(resolve);
E
Erich Gamma 已提交
303 304 305
		});
	}

B
Benjamin Pasero 已提交
306 307
	get isReady(): boolean {
		return this._readyState === ReadyState.READY;
E
Erich Gamma 已提交
308 309
	}

310 311 312
	private handleMarketplaceRequests(): void {

		// Resolve marketplace headers
313
		this.marketplaceHeadersPromise = resolveMarketplaceHeaders(product.version, this.environmentService, this.fileService);
E
Erich Gamma 已提交
314

315 316
		// Inject headers when requests are incoming
		const urls = ['https://marketplace.visualstudio.com/*', 'https://*.vsassets.io/*'];
317
		this._win.webContents.session.webRequest.onBeforeSendHeaders({ urls }, (details, cb) => {
318
			this.marketplaceHeadersPromise.then(headers => {
M
Matt Bierner 已提交
319
				const requestHeaders = objects.assign(details.requestHeaders, headers) as { [key: string]: string | undefined };
320 321 322
				if (!this.configurationService.getValue('extensions.disableExperimentalAzureSearch')) {
					requestHeaders['Cookie'] = `${requestHeaders['Cookie'] ? requestHeaders['Cookie'] + ';' : ''}EnableExternalSearchForVSCode=true`;
				}
323
				cb({ cancel: false, requestHeaders });
324
			});
325
		});
326 327 328
	}

	private registerListeners(): void {
329 330

		// Prevent loading of svgs
331
		this._win.webContents.session.webRequest.onBeforeRequest(null!, (details, callback) => {
332 333
			if (details.url.indexOf('.svg') > 0) {
				const uri = URI.parse(details.url);
334
				if (uri && !uri.scheme.match(/file/i) && endsWith(uri.path, '.svg')) {
335 336 337 338 339 340 341
					return callback({ cancel: true });
				}
			}

			return callback({});
		});

342
		this._win.webContents.session.webRequest.onHeadersReceived(null!, (details, callback) => {
B
Benjamin Pasero 已提交
343 344 345
			const responseHeaders = details.responseHeaders as { [key: string]: string[] };

			const contentType: string[] = (responseHeaders['content-type'] || responseHeaders['Content-Type']);
346 347 348 349
			if (contentType && Array.isArray(contentType) && contentType.some(x => x.toLowerCase().indexOf('image/svg') >= 0)) {
				return callback({ cancel: true });
			}

B
Benjamin Pasero 已提交
350
			return callback({ cancel: false, responseHeaders });
351 352
		});

E
Erich Gamma 已提交
353 354 355 356 357 358 359 360
		// Remember that we loaded
		this._win.webContents.on('did-finish-load', () => {
			this._readyState = ReadyState.LOADING;

			// Associate properties from the load request if provided
			if (this.pendingLoadConfig) {
				this.currentConfig = this.pendingLoadConfig;

361
				this.pendingLoadConfig = undefined;
E
Erich Gamma 已提交
362 363 364 365 366
			}
		});

		// Window Focus
		this._win.on('focus', () => {
B
Benjamin Pasero 已提交
367
			this._lastFocusTime = Date.now();
E
Erich Gamma 已提交
368 369
		});

B
Benjamin Pasero 已提交
370 371 372
		// Simple fullscreen doesn't resize automatically when the resolution changes so as a workaround
		// we need to detect when display metrics change or displays are added/removed and toggle the
		// fullscreen manually.
B
Benjamin Pasero 已提交
373
		if (isMacintosh) {
B
Benjamin Pasero 已提交
374 375 376 377 378 379
			const simpleFullScreenScheduler = this._register(new RunOnceScheduler(() => {
				if (!this._win) {
					return; // disposed
				}

				if (!this.useNativeFullScreen() && this.isFullScreen()) {
B
Benjamin Pasero 已提交
380 381 382
					this.setFullScreen(false);
					this.setFullScreen(true);
				}
B
Benjamin Pasero 已提交
383 384 385 386 387 388 389 390 391
			}, 100));

			const displayChangedListener = () => simpleFullScreenScheduler.schedule();

			screen.on('display-metrics-changed', displayChangedListener);
			this._register(toDisposable(() => screen.removeListener('display-metrics-changed', displayChangedListener)));

			screen.on('display-added', displayChangedListener);
			this._register(toDisposable(() => screen.removeListener('display-added', displayChangedListener)));
B
Benjamin Pasero 已提交
392

B
Benjamin Pasero 已提交
393 394
			screen.on('display-removed', displayChangedListener);
			this._register(toDisposable(() => screen.removeListener('display-removed', displayChangedListener)));
B
Benjamin Pasero 已提交
395
		}
396

S
SteVen Batten 已提交
397
		// Window (Un)Maximize
398
		this._win.on('maximize', (e: Event) => {
B
Benjamin Pasero 已提交
399 400 401 402
			if (this.currentConfig) {
				this.currentConfig.maximized = true;
			}

403 404 405
			app.emit('browser-window-maximize', e, this._win);
		});

406
		this._win.on('unmaximize', (e: Event) => {
B
Benjamin Pasero 已提交
407 408 409 410
			if (this.currentConfig) {
				this.currentConfig.maximized = false;
			}

411 412
			app.emit('browser-window-unmaximize', e, this._win);
		});
S
SteVen Batten 已提交
413

414 415 416 417 418 419 420 421 422
		// Window Fullscreen
		this._win.on('enter-full-screen', () => {
			this.sendWhenReady('vscode:enterFullScreen');
		});

		this._win.on('leave-full-screen', () => {
			this.sendWhenReady('vscode:leaveFullScreen');
		});

E
Erich Gamma 已提交
423
		// Window Failed to load
424
		this._win.webContents.on('did-fail-load', (event: Event, errorCode: number, errorDescription: string, validatedURL: string, isMainFrame: boolean) => {
B
Benjamin Pasero 已提交
425
			this.logService.warn('[electron event]: fail to load, ', errorDescription);
E
Erich Gamma 已提交
426 427
		});

B
Benjamin Pasero 已提交
428
		// Handle configuration changes
B
Benjamin Pasero 已提交
429
		this._register(this.configurationService.onDidChangeConfiguration(e => this.onConfigurationUpdated()));
430

431
		// Handle Workspace events
B
Benjamin Pasero 已提交
432
		this._register(this.workspacesMainService.onUntitledWorkspaceDeleted(e => this.onUntitledWorkspaceDeleted(e)));
433 434
	}

435
	private onUntitledWorkspaceDeleted(workspace: IWorkspaceIdentifier): void {
436 437 438

		// Make sure to update our workspace config if we detect that it
		// was deleted
439
		if (this.openedWorkspace && this.openedWorkspace.id === workspace.id) {
R
Rob Lourens 已提交
440
			this.currentConfig.workspace = undefined;
441
		}
E
Erich Gamma 已提交
442 443
	}

B
Benjamin Pasero 已提交
444
	private onConfigurationUpdated(): void {
445
		const newMenuBarVisibility = this.getMenuBarVisibility();
B
Benjamin Pasero 已提交
446 447 448 449
		if (newMenuBarVisibility !== this.currentMenuBarVisibility) {
			this.currentMenuBarVisibility = newMenuBarVisibility;
			this.setMenuBarVisibility(newMenuBarVisibility);
		}
B
Benjamin Pasero 已提交
450 451
	}

452 453 454 455 456 457
	addTabbedWindow(window: ICodeWindow): void {
		if (isMacintosh) {
			this._win.addTabbedWindow(window.win);
		}
	}

B
Benjamin Pasero 已提交
458
	load(config: IWindowConfiguration, isReload?: boolean, disableExtensions?: boolean): void {
E
Erich Gamma 已提交
459 460 461

		// If this is the first time the window is loaded, we associate the paths
		// directly with the window because we assume the loading will just work
B
Benjamin Pasero 已提交
462
		if (this._readyState === ReadyState.NONE) {
E
Erich Gamma 已提交
463 464 465 466 467 468 469 470 471 472 473 474
			this.currentConfig = config;
		}

		// Otherwise, the window is currently showing a folder and if there is an
		// unload handler preventing the load, we cannot just associate the paths
		// because the loading might be vetoed. Instead we associate it later when
		// the window load event has fired.
		else {
			this.pendingLoadConfig = config;
			this._readyState = ReadyState.NAVIGATING;
		}

475 476 477 478 479 480 481
		// Add disable-extensions to the config, but do not preserve it on currentConfig or
		// pendingLoadConfig so that it is applied only on this load
		const configuration = objects.assign({}, config);
		if (disableExtensions !== undefined) {
			configuration['disable-extensions'] = disableExtensions;
		}

482
		// Clear Document Edited if needed
483
		if (isMacintosh && this._win.isDocumentEdited()) {
B
Benjamin Pasero 已提交
484
			if (!isReload || !this.backupMainService.isHotExitEnabled()) {
485 486 487 488 489 490 491 492 493 494 495
				this._win.setDocumentEdited(false);
			}
		}

		// Clear Title and Filename if needed
		if (!isReload) {
			if (this.getRepresentedFilename()) {
				this.setRepresentedFilename('');
			}

			this._win.setTitle(product.nameLong);
496 497
		}

E
Erich Gamma 已提交
498
		// Load URL
499
		perf.mark('main:loadWindow');
500
		this._win.loadURL(this.getUrl(configuration));
E
Erich Gamma 已提交
501 502

		// Make window visible if it did not open in N seconds because this indicates an error
503
		// Only do this when running out of sources and not when running tests
504
		if (!this.environmentService.isBuilt && !this.environmentService.extensionTestsLocationURI) {
E
Erich Gamma 已提交
505 506 507 508
			this.showTimeoutHandle = setTimeout(() => {
				if (this._win && !this._win.isVisible() && !this._win.isMinimized()) {
					this._win.show();
					this._win.focus();
509
					this._win.webContents.openDevTools();
E
Erich Gamma 已提交
510 511 512 513 514
				}
			}, 10000);
		}
	}

515
	reload(configurationIn?: IWindowConfiguration, cli?: ParsedArgs): void {
B
Benjamin Pasero 已提交
516

517
		// If config is not provided, copy our current one
518
		const configuration = configurationIn ? configurationIn : objects.mixin({}, this.currentConfig);
519 520

		// Delete some properties we do not want during reload
521
		delete configuration.filesToOpenOrCreate;
522
		delete configuration.filesToDiff;
523
		delete configuration.filesToWait;
524

E
Erich Gamma 已提交
525
		// Some configuration things get inherited if the window is being reloaded and we are
B
Benjamin Pasero 已提交
526
		// in extension development mode. These options are all development related.
527
		if (this.isExtensionDevelopmentHost && cli) {
B
Benjamin Pasero 已提交
528
			configuration.verbose = cli.verbose;
529 530
			configuration['inspect-extensions'] = cli['inspect-extensions'];
			configuration['inspect-brk-extensions'] = cli['inspect-brk-extensions'];
531
			configuration.debugId = cli.debugId;
532
			configuration['extensions-dir'] = cli['extensions-dir'];
E
Erich Gamma 已提交
533 534
		}

535 536
		configuration.isInitialStartup = false; // since this is a reload

E
Erich Gamma 已提交
537
		// Load config
538 539
		const disableExtensions = cli ? cli['disable-extensions'] : undefined;
		this.load(configuration, true, disableExtensions);
E
Erich Gamma 已提交
540 541
	}

542
	private getUrl(windowConfiguration: IWindowConfiguration): string {
E
Erich Gamma 已提交
543

544 545
		// Set window ID
		windowConfiguration.windowId = this._win.id;
S
Sandeep Somavarapu 已提交
546
		windowConfiguration.logLevel = this.logService.getLevel();
547

548
		// Set zoomlevel
549
		const windowConfig = this.configurationService.getValue<IWindowSettings>('window');
550
		const zoomLevel = windowConfig && windowConfig.zoomLevel;
551 552 553 554
		if (typeof zoomLevel === 'number') {
			windowConfiguration.zoomLevel = zoomLevel;
		}

555
		// Set fullscreen state
556
		windowConfiguration.fullscreen = this.isFullScreen();
557

558
		// Set Accessibility Config
559 560 561 562 563
		let autoDetectHighContrast = true;
		if (windowConfig && windowConfig.autoDetectHighContrast === false) {
			autoDetectHighContrast = false;
		}
		windowConfiguration.highContrast = isWindows && autoDetectHighContrast && systemPreferences.isInvertedColorScheme();
564
		windowConfiguration.accessibilitySupport = app.isAccessibilitySupportEnabled();
565

B
Benjamin Pasero 已提交
566 567
		// Title style related
		windowConfiguration.maximized = this._win.isMaximized();
568
		windowConfiguration.frameless = this.hasHiddenTitleBarStyle() && !isMacintosh;
M
Martin Aeschlimann 已提交
569

570 571
		// Dump Perf Counters
		windowConfiguration.perfEntries = perf.exportEntries();
572

573
		// Parts splash
574
		windowConfiguration.partsSplashPath = path.join(this.environmentService.userDataPath, 'rapid_render.json');
575

576
		// Config (combination of process.argv and window configuration)
577
		const environment = parseArgs(process.argv, OPTIONS);
578
		const config = objects.assign(environment, windowConfiguration);
579 580 581 582
		for (const key in config) {
			const configValue = (config as any)[key];
			if (configValue === undefined || configValue === null || configValue === '' || configValue === false) {
				delete (config as any)[key]; // only send over properties that have a true value
583 584
			}
		}
585

586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604
		// In the unlikely event of the URL becoming larger than 2MB, remove parts of
		// it that are not under our control. Mainly, the user environment can be very
		// large depending on user configuration, so we can only remove it in that case.
		let configUrl = this.doGetUrl(config);
		if (configUrl.length > CodeWindow.MAX_URL_LENGTH) {
			delete config.userEnv;
			this.logService.warn('Application URL exceeds maximum of 2MB and was shortened.');

			configUrl = this.doGetUrl(config);

			if (configUrl.length > CodeWindow.MAX_URL_LENGTH) {
				this.logService.error('Application URL exceeds maximum of 2MB and cannot be loaded.');
			}
		}

		return configUrl;
	}

	private doGetUrl(config: object): string {
605
		return `${require.toUrl('vs/code/electron-browser/workbench/workbench.html')}?config=${encodeURIComponent(JSON.stringify(config))}`;
E
Erich Gamma 已提交
606 607
	}

B
Benjamin Pasero 已提交
608
	serializeWindowState(): IWindowState {
609 610 611
		if (!this._win) {
			return defaultWindowState();
		}
612 613

		// fullscreen gets special treatment
614
		if (this.isFullScreen()) {
615 616
			const display = screen.getDisplayMatching(this.getBounds());

B
Benjamin Pasero 已提交
617 618 619
			const defaultState = defaultWindowState();

			const res = {
620
				mode: WindowMode.Fullscreen,
R
Rob Lourens 已提交
621
				display: display ? display.id : undefined,
622

B
Benjamin Pasero 已提交
623 624 625 626 627 628 629 630 631
				// Still carry over window dimensions from previous sessions
				// if we can compute it in fullscreen state.
				// does not seem possible in all cases on Linux for example
				// (https://github.com/Microsoft/vscode/issues/58218) so we
				// fallback to the defaults in that case.
				width: this.windowState.width || defaultState.width,
				height: this.windowState.height || defaultState.height,
				x: this.windowState.x || 0,
				y: this.windowState.y || 0
632
			};
B
Benjamin Pasero 已提交
633 634

			return res;
E
Erich Gamma 已提交
635 636
		}

637
		const state: IWindowState = Object.create(null);
E
Erich Gamma 已提交
638 639 640
		let mode: WindowMode;

		// get window mode
641
		if (!isMacintosh && this._win.isMaximized()) {
E
Erich Gamma 已提交
642 643 644 645 646 647 648 649
			mode = WindowMode.Maximized;
		} else {
			mode = WindowMode.Normal;
		}

		// we don't want to save minimized state, only maximized or normal
		if (mode === WindowMode.Maximized) {
			state.mode = WindowMode.Maximized;
650
		} else {
E
Erich Gamma 已提交
651 652 653 654 655
			state.mode = WindowMode.Normal;
		}

		// only consider non-minimized window states
		if (mode === WindowMode.Normal || mode === WindowMode.Maximized) {
656
			let bounds: Rectangle;
657 658 659 660 661
			if (mode === WindowMode.Normal) {
				bounds = this.getBounds();
			} else {
				bounds = this._win.getNormalBounds(); // make sure to persist the normal bounds when maximized to be able to restore them
			}
E
Erich Gamma 已提交
662

663 664 665 666
			state.x = bounds.x;
			state.y = bounds.y;
			state.width = bounds.width;
			state.height = bounds.height;
E
Erich Gamma 已提交
667 668 669 670 671
		}

		return state;
	}

B
Benjamin Pasero 已提交
672 673
	private restoreWindowState(state?: IWindowState): [IWindowState, boolean? /* has multiple displays */] {
		let hasMultipleDisplays = false;
E
Erich Gamma 已提交
674 675
		if (state) {
			try {
B
Benjamin Pasero 已提交
676 677 678 679
				const displays = screen.getAllDisplays();
				hasMultipleDisplays = displays.length > 1;

				state = this.validateWindowState(state, displays);
E
Erich Gamma 已提交
680
			} catch (err) {
J
Joao Moreno 已提交
681
				this.logService.warn(`Unexpected error validating window state: ${err}\n${err.stack}`); // somehow display API can be picky about the state to validate
E
Erich Gamma 已提交
682 683
			}
		}
B
Benjamin Pasero 已提交
684 685

		return [state || defaultWindowState(), hasMultipleDisplays];
E
Erich Gamma 已提交
686 687
	}

B
Benjamin Pasero 已提交
688
	private validateWindowState(state: IWindowState, displays: Display[]): IWindowState | undefined {
689 690 691 692 693
		if (typeof state.x !== 'number'
			|| typeof state.y !== 'number'
			|| typeof state.width !== 'number'
			|| typeof state.height !== 'number'
		) {
694
			return undefined;
E
Erich Gamma 已提交
695 696 697
		}

		if (state.width <= 0 || state.height <= 0) {
698
			return undefined;
E
Erich Gamma 已提交
699 700 701 702
		}

		// Single Monitor: be strict about x/y positioning
		if (displays.length === 1) {
B
Benjamin Pasero 已提交
703
			const displayWorkingArea = this.getWorkingArea(displays[0]);
704
			if (displayWorkingArea) {
B
Benjamin Pasero 已提交
705 706
				if (state.x < displayWorkingArea.x) {
					state.x = displayWorkingArea.x; // prevent window from falling out of the screen to the left
E
Erich Gamma 已提交
707 708
				}

B
Benjamin Pasero 已提交
709 710
				if (state.y < displayWorkingArea.y) {
					state.y = displayWorkingArea.y; // prevent window from falling out of the screen to the top
E
Erich Gamma 已提交
711 712
				}

B
Benjamin Pasero 已提交
713 714
				if (state.x > (displayWorkingArea.x + displayWorkingArea.width)) {
					state.x = displayWorkingArea.x; // prevent window from falling out of the screen to the right
E
Erich Gamma 已提交
715 716
				}

B
Benjamin Pasero 已提交
717 718
				if (state.y > (displayWorkingArea.y + displayWorkingArea.height)) {
					state.y = displayWorkingArea.y; // prevent window from falling out of the screen to the bottom
E
Erich Gamma 已提交
719 720
				}

B
Benjamin Pasero 已提交
721 722
				if (state.width > displayWorkingArea.width) {
					state.width = displayWorkingArea.width; // prevent window from exceeding display bounds width
E
Erich Gamma 已提交
723 724
				}

B
Benjamin Pasero 已提交
725 726
				if (state.height > displayWorkingArea.height) {
					state.height = displayWorkingArea.height; // prevent window from exceeding display bounds height
E
Erich Gamma 已提交
727 728 729 730 731 732
				}
			}

			return state;
		}

733 734 735 736 737 738 739 740 741 742 743 744 745
		// Multi Montior (fullscreen): try to find the previously used display
		if (state.display && state.mode === WindowMode.Fullscreen) {
			const display = displays.filter(d => d.id === state.display)[0];
			if (display && display.bounds && typeof display.bounds.x === 'number' && typeof display.bounds.y === 'number') {
				const defaults = defaultWindowState(WindowMode.Fullscreen); // make sure we have good values when the user restores the window
				defaults.x = display.bounds.x; // carefull to use displays x/y position so that the window ends up on the correct monitor
				defaults.y = display.bounds.y;

				return defaults;
			}
		}

		// Multi Monitor (non-fullscreen): be less strict because metrics can be crazy
746 747
		const bounds = { x: state.x, y: state.y, width: state.width, height: state.height };
		const display = screen.getDisplayMatching(bounds);
B
Benjamin Pasero 已提交
748
		const displayWorkingArea = this.getWorkingArea(display);
U
unknown 已提交
749
		if (
B
Benjamin Pasero 已提交
750 751 752 753 754 755
			display &&														// we have a display matching the desired bounds
			displayWorkingArea &&											// we have valid working area bounds
			bounds.x < displayWorkingArea.x + displayWorkingArea.width &&	// prevent window from falling out of the screen to the right
			bounds.y < displayWorkingArea.y + displayWorkingArea.height &&	// prevent window from falling out of the screen to the bottom
			bounds.x + bounds.width > displayWorkingArea.x &&				// prevent window from falling out of the screen to the left
			bounds.y + bounds.height > displayWorkingArea.y					// prevent window from falling out of the scree nto the top
U
unknown 已提交
756
		) {
E
Erich Gamma 已提交
757 758 759
			return state;
		}

760
		return undefined;
E
Erich Gamma 已提交
761
	}
B
Benjamin Pasero 已提交
762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779

	private getWorkingArea(display: Display): Rectangle | undefined {

		// Prefer the working area of the display to account for taskbars on the
		// desktop being positioned somewhere (https://github.com/Microsoft/vscode/issues/50830).
		//
		// Linux X11 sessions sometimes report wrong display bounds, so we validate
		// the reported sizes are positive.
		if (display.workArea.width > 0 && display.workArea.height > 0) {
			return display.workArea;
		}

		if (display.bounds.width > 0 && display.bounds.height > 0) {
			return display.bounds;
		}

		return undefined;
	}
E
Erich Gamma 已提交
780

781
	getBounds(): Rectangle {
B
Benjamin Pasero 已提交
782 783
		const pos = this._win.getPosition();
		const dimension = this._win.getSize();
E
Erich Gamma 已提交
784 785 786 787

		return { x: pos[0], y: pos[1], width: dimension[0], height: dimension[1] };
	}

B
Benjamin Pasero 已提交
788
	toggleFullScreen(): void {
789 790
		this.setFullScreen(!this.isFullScreen());
	}
E
Erich Gamma 已提交
791

792
	private setFullScreen(fullscreen: boolean): void {
793

794 795 796 797 798 799 800 801 802 803 804
		// Set fullscreen state
		if (this.useNativeFullScreen()) {
			this.setNativeFullScreen(fullscreen);
		} else {
			this.setSimpleFullScreen(fullscreen);
		}

		// Events
		this.sendWhenReady(fullscreen ? 'vscode:enterFullScreen' : 'vscode:leaveFullScreen');

		// Respect configured menu bar visibility or default to toggle if not set
805
		this.setMenuBarVisibility(this.currentMenuBarVisibility, false);
B
Benjamin Pasero 已提交
806 807
	}

808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829
	isFullScreen(): boolean {
		return this._win.isFullScreen() || this._win.isSimpleFullScreen();
	}

	private setNativeFullScreen(fullscreen: boolean): void {
		if (this._win.isSimpleFullScreen()) {
			this._win.setSimpleFullScreen(false);
		}

		this._win.setFullScreen(fullscreen);
	}

	private setSimpleFullScreen(fullscreen: boolean): void {
		if (this._win.isFullScreen()) {
			this._win.setFullScreen(false);
		}

		this._win.setSimpleFullScreen(fullscreen);
		this._win.webContents.focus(); // workaround issue where focus is not going into window
	}

	private useNativeFullScreen(): boolean {
830 831 832 833
		const windowConfig = this.configurationService.getValue<IWindowSettings>('window');
		if (!windowConfig || typeof windowConfig.nativeFullScreen !== 'boolean') {
			return true; // default
		}
834

835 836 837
		if (windowConfig.nativeTabs) {
			return true; // https://github.com/electron/electron/issues/16142
		}
B
Benjamin Pasero 已提交
838

839
		return windowConfig.nativeFullScreen !== false;
840 841
	}

B
Benjamin Pasero 已提交
842 843 844 845
	isMinimized(): boolean {
		return this._win.isMinimized();
	}

846
	private getMenuBarVisibility(): MenuBarVisibility {
847
		const windowConfig = this.configurationService.getValue<IWindowSettings>('window');
B
Benjamin Pasero 已提交
848
		if (!windowConfig || !windowConfig.menuBarVisibility) {
849
			return 'default';
B
Benjamin Pasero 已提交
850 851 852 853
		}

		let menuBarVisibility = windowConfig.menuBarVisibility;
		if (['visible', 'toggle', 'hidden'].indexOf(menuBarVisibility) < 0) {
854
			menuBarVisibility = 'default';
B
Benjamin Pasero 已提交
855 856 857
		}

		return menuBarVisibility;
E
Erich Gamma 已提交
858 859
	}

860
	private setMenuBarVisibility(visibility: MenuBarVisibility, notify: boolean = true): void {
861
		if (isMacintosh) {
B
Benjamin Pasero 已提交
862 863 864
			return; // ignore for macOS platform
		}

865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885
		if (visibility === 'toggle') {
			if (notify) {
				this.send('vscode:showInfoMessage', nls.localize('hiddenMenuBar', "You can still access the menu bar by pressing the Alt-key."));
			}
		}

		if (visibility === 'hidden') {
			// for some weird reason that I have no explanation for, the menu bar is not hiding when calling
			// this without timeout (see https://github.com/Microsoft/vscode/issues/19777). there seems to be
			// a timing issue with us opening the first window and the menu bar getting created. somehow the
			// fact that we want to hide the menu without being able to bring it back via Alt key makes Electron
			// still show the menu. Unable to reproduce from a simple Hello World application though...
			setTimeout(() => {
				this.doSetMenuBarVisibility(visibility);
			});
		} else {
			this.doSetMenuBarVisibility(visibility);
		}
	}

	private doSetMenuBarVisibility(visibility: MenuBarVisibility): void {
886
		const isFullscreen = this.isFullScreen();
887

888
		switch (visibility) {
889
			case ('default'):
B
Benjamin Pasero 已提交
890 891
				this._win.setMenuBarVisibility(!isFullscreen);
				this._win.setAutoHideMenuBar(isFullscreen);
892 893
				break;

D
David Terry 已提交
894
			case ('visible'):
B
Benjamin Pasero 已提交
895 896
				this._win.setMenuBarVisibility(true);
				this._win.setAutoHideMenuBar(false);
897
				break;
D
David Terry 已提交
898 899

			case ('toggle'):
B
Benjamin Pasero 已提交
900 901
				this._win.setMenuBarVisibility(false);
				this._win.setAutoHideMenuBar(true);
902
				break;
D
David Terry 已提交
903 904

			case ('hidden'):
905 906
				this._win.setMenuBarVisibility(false);
				this._win.setAutoHideMenuBar(false);
907
				break;
908
		}
909 910
	}

911
	handleTitleDoubleClick(): void {
912 913

		// Respect system settings on mac with regards to title click on windows title
914
		if (isMacintosh) {
915 916 917 918 919 920 921 922 923
			const action = systemPreferences.getUserDefault('AppleActionOnDoubleClick', 'string');
			switch (action) {
				case 'Minimize':
					this.win.minimize();
					break;
				case 'None':
					break;
				case 'Maximize':
				default:
B
Benjamin Pasero 已提交
924 925 926 927 928
					if (this.win.isMaximized()) {
						this.win.unmaximize();
					} else {
						this.win.maximize();
					}
929 930 931 932 933 934 935 936 937 938 939 940 941
			}
		}

		// Linux/Windows: just toggle maximize/minimized state
		else {
			if (this.win.isMaximized()) {
				this.win.unmaximize();
			} else {
				this.win.maximize();
			}
		}
	}

B
Benjamin Pasero 已提交
942
	close(): void {
943 944 945 946 947
		if (this._win) {
			this._win.close();
		}
	}

B
Benjamin Pasero 已提交
948
	sendWhenReady(channel: string, ...args: any[]): void {
B
Benjamin Pasero 已提交
949
		if (this.isReady) {
950
			this.send(channel, ...args);
B
Benjamin Pasero 已提交
951 952 953
		} else {
			this.ready().then(() => this.send(channel, ...args));
		}
954 955
	}

B
Benjamin Pasero 已提交
956
	send(channel: string, ...args: any[]): void {
B
Benjamin Pasero 已提交
957 958 959
		if (this._win) {
			this._win.webContents.send(channel, ...args);
		}
960 961
	}

B
Benjamin Pasero 已提交
962
	updateTouchBar(groups: ISerializableCommandAction[][]): void {
963 964 965 966
		if (!isMacintosh) {
			return; // only supported on macOS
		}

B
Benjamin Pasero 已提交
967 968 969 970 971
		// Update segments for all groups. Setting the segments property
		// of the group directly prevents ugly flickering from happening
		this.touchBarGroups.forEach((touchBarGroup, index) => {
			const commands = groups[index];
			touchBarGroup.segments = this.createTouchBarGroupSegments(commands);
972
		});
B
Benjamin Pasero 已提交
973
	}
974

B
Benjamin Pasero 已提交
975 976 977 978
	private createTouchBar(): void {
		if (!isMacintosh) {
			return; // only supported on macOS
		}
979

B
Benjamin Pasero 已提交
980 981 982 983 984 985 986
		// To avoid flickering, we try to reuse the touch bar group
		// as much as possible by creating a large number of groups
		// for reusing later.
		for (let i = 0; i < 10; i++) {
			const groupTouchBar = this.createTouchBarGroup();
			this.touchBarGroups.push(groupTouchBar);
		}
987

B
Benjamin Pasero 已提交
988
		this._win.setTouchBar(new TouchBar({ items: this.touchBarGroups }));
989
	}
990

991
	private createTouchBarGroup(items: ISerializableCommandAction[] = []): TouchBarSegmentedControl {
992

B
Benjamin Pasero 已提交
993 994
		// Group Segments
		const segments = this.createTouchBarGroupSegments(items);
995

B
Benjamin Pasero 已提交
996 997 998 999 1000 1001 1002 1003 1004
		// Group Control
		const control = new TouchBar.TouchBarSegmentedControl({
			segments,
			mode: 'buttons',
			segmentStyle: 'automatic',
			change: (selectedIndex) => {
				this.sendWhenReady('vscode:runAction', { id: (control.segments[selectedIndex] as ITouchBarSegment).id, from: 'touchbar' });
			}
		});
1005

B
Benjamin Pasero 已提交
1006 1007 1008
		return control;
	}

B
Benjamin Pasero 已提交
1009
	private createTouchBarGroupSegments(items: ISerializableCommandAction[] = []): ITouchBarSegment[] {
B
Benjamin Pasero 已提交
1010
		const segments: ITouchBarSegment[] = items.map(item => {
1011
			let icon: NativeImage | undefined;
1012
			if (item.iconLocation && item.iconLocation.dark.scheme === 'file') {
B
Benjamin Pasero 已提交
1013
				icon = nativeImage.createFromPath(URI.revive(item.iconLocation.dark).fsPath);
1014
				if (icon.isEmpty()) {
R
Rob Lourens 已提交
1015
					icon = undefined;
1016
				}
1017
			}
1018

1019 1020 1021 1022 1023 1024 1025
			let title: string;
			if (typeof item.title === 'string') {
				title = item.title;
			} else {
				title = item.title.value;
			}

1026
			return {
B
Benjamin Pasero 已提交
1027
				id: item.id,
1028
				label: !icon ? title : undefined,
1029 1030
				icon
			};
1031 1032
		});

B
Benjamin Pasero 已提交
1033
		return segments;
1034 1035
	}

B
Benjamin Pasero 已提交
1036
	dispose(): void {
B
Benjamin Pasero 已提交
1037 1038
		super.dispose();

E
Erich Gamma 已提交
1039 1040 1041 1042
		if (this.showTimeoutHandle) {
			clearTimeout(this.showTimeoutHandle);
		}

1043
		this._win = null!; // Important to dereference the window object to allow for GC
E
Erich Gamma 已提交
1044
	}
1045
}