@theia/core
Version:
Theia is a cloud & desktop IDE framework implemented in TypeScript.
99 lines (85 loc) • 4.29 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2017 TypeFox and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
// *****************************************************************************
import * as electronRemote from '../../../electron-shared/@electron/remote';
import { injectable, inject, postConstruct } from 'inversify';
import * as electron from '../../../electron-shared/electron';
import { NewWindowOptions } from '../../common/window';
import { DefaultWindowService } from '../../browser/window/default-window-service';
import { ElectronMainWindowService } from '../../electron-common/electron-main-window-service';
import { ElectronWindowPreferences } from './electron-window-preferences';
import { CloseRequestArguments, CLOSE_REQUESTED_SIGNAL, RELOAD_REQUESTED_SIGNAL, StopReason } from '../../electron-common/messaging/electron-messages';
@injectable()
export class ElectronWindowService extends DefaultWindowService {
/**
* Lock to prevent multiple parallel executions of the `beforeunload` listener.
*/
protected isUnloading: boolean = false;
/**
* Close the window right away when `true`, else check if we can unload.
*/
protected closeOnUnload: boolean = false;
@inject(ElectronMainWindowService)
protected readonly delegate: ElectronMainWindowService;
@inject(ElectronWindowPreferences)
protected readonly electronWindowPreferences: ElectronWindowPreferences;
override openNewWindow(url: string, { external }: NewWindowOptions = {}): undefined {
this.delegate.openNewWindow(url, { external });
return undefined;
}
override openNewDefaultWindow(): void {
this.delegate.openNewDefaultWindow();
}
@postConstruct()
protected init(): void {
// Update the default zoom level on startup when the preferences event is fired.
this.electronWindowPreferences.onPreferenceChanged(e => {
if (e.preferenceName === 'window.zoomLevel') {
this.updateWindowZoomLevel();
}
});
}
protected override registerUnloadListeners(): void {
electron.ipcRenderer.on(CLOSE_REQUESTED_SIGNAL, (_event, closeRequestEvent: CloseRequestArguments) => this.handleCloseRequestedEvent(closeRequestEvent));
window.addEventListener('unload', () => this.onUnloadEmitter.fire());
}
/**
* Run when ElectronMain detects a `close` event and emits a `close-requested` event.
* Should send an event to `electron.ipcRenderer` on the event's `confirmChannel` if it is safe to exit
* after running FrontendApplication `onWillStop` handlers or on the `cancelChannel` if it is not safe to exit.
*/
protected async handleCloseRequestedEvent(event: CloseRequestArguments): Promise<void> {
const safeToClose = await this.isSafeToShutDown(event.reason);
if (safeToClose) {
console.debug(`Shutting down because of ${StopReason[event.reason]} request.`);
electron.ipcRenderer.send(event.confirmChannel);
} else {
electron.ipcRenderer.send(event.cancelChannel);
}
}
/**
* Updates the window zoom level based on the preference value.
*/
protected updateWindowZoomLevel(): void {
const preferredZoomLevel = this.electronWindowPreferences['window.zoomLevel'];
const webContents = electronRemote.getCurrentWindow().webContents;
if (webContents.getZoomLevel() !== preferredZoomLevel) {
webContents.setZoomLevel(preferredZoomLevel);
}
}
override reload(): void {
electron.ipcRenderer.send(RELOAD_REQUESTED_SIGNAL);
}
}