mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-28 17:23:19 -05:00
225 lines
8.3 KiB
TypeScript
225 lines
8.3 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 { DriverChannel, WindowDriverChannelClient, IWindowDriverRegistry, WindowDriverRegistryChannel, IDriverOptions } from 'vs/platform/driver/node/driver';
|
|
import { IWindowsMainService } from 'vs/platform/windows/electron-main/windows';
|
|
import { serve as serveNet } from 'vs/base/parts/ipc/node/ipc.net';
|
|
import { combinedDisposable, IDisposable } from 'vs/base/common/lifecycle';
|
|
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
|
|
import { IPCServer, StaticRouter } from 'vs/base/parts/ipc/common/ipc';
|
|
import { SimpleKeybinding, KeyCode } from 'vs/base/common/keyCodes';
|
|
import { USLayoutResolvedKeybinding } from 'vs/platform/keybinding/common/usLayoutResolvedKeybinding';
|
|
import { OS } from 'vs/base/common/platform';
|
|
import { Emitter, Event } from 'vs/base/common/event';
|
|
import { INativeEnvironmentService } from 'vs/platform/environment/node/environmentService';
|
|
import { ScanCodeBinding } from 'vs/base/common/scanCode';
|
|
import { KeybindingParser } from 'vs/base/common/keybindingParser';
|
|
import { timeout } from 'vs/base/common/async';
|
|
import { IDriver, IElement, IWindowDriver } from 'vs/platform/driver/common/driver';
|
|
import { ILifecycleMainService } from 'vs/platform/lifecycle/electron-main/lifecycleMainService';
|
|
import { IElectronMainService } from 'vs/platform/electron/electron-main/electronMainService';
|
|
|
|
function isSilentKeyCode(keyCode: KeyCode) {
|
|
return keyCode < KeyCode.KEY_0;
|
|
}
|
|
|
|
export class Driver implements IDriver, IWindowDriverRegistry {
|
|
|
|
_serviceBrand: undefined;
|
|
|
|
private registeredWindowIds = new Set<number>();
|
|
private reloadingWindowIds = new Set<number>();
|
|
private readonly onDidReloadingChange = new Emitter<void>();
|
|
|
|
constructor(
|
|
private windowServer: IPCServer,
|
|
private options: IDriverOptions,
|
|
@IWindowsMainService private readonly windowsMainService: IWindowsMainService,
|
|
@ILifecycleMainService private readonly lifecycleMainService: ILifecycleMainService,
|
|
@IElectronMainService private readonly electronMainService: IElectronMainService
|
|
) { }
|
|
|
|
async registerWindowDriver(windowId: number): Promise<IDriverOptions> {
|
|
this.registeredWindowIds.add(windowId);
|
|
this.reloadingWindowIds.delete(windowId);
|
|
this.onDidReloadingChange.fire();
|
|
return this.options;
|
|
}
|
|
|
|
async reloadWindowDriver(windowId: number): Promise<void> {
|
|
this.reloadingWindowIds.add(windowId);
|
|
}
|
|
|
|
async getWindowIds(): Promise<number[]> {
|
|
return this.windowsMainService.getWindows()
|
|
.map(w => w.id)
|
|
.filter(id => this.registeredWindowIds.has(id) && !this.reloadingWindowIds.has(id));
|
|
}
|
|
|
|
async capturePage(windowId: number): Promise<string> {
|
|
await this.whenUnfrozen(windowId);
|
|
|
|
const window = this.windowsMainService.getWindowById(windowId);
|
|
if (!window) {
|
|
throw new Error('Invalid window');
|
|
}
|
|
const webContents = window.win.webContents;
|
|
const image = await webContents.capturePage();
|
|
return image.toPNG().toString('base64');
|
|
}
|
|
|
|
async reloadWindow(windowId: number): Promise<void> {
|
|
await this.whenUnfrozen(windowId);
|
|
|
|
const window = this.windowsMainService.getWindowById(windowId);
|
|
if (!window) {
|
|
throw new Error('Invalid window');
|
|
}
|
|
this.reloadingWindowIds.add(windowId);
|
|
this.lifecycleMainService.reload(window);
|
|
}
|
|
|
|
async exitApplication(): Promise<void> {
|
|
return this.electronMainService.quit(undefined);
|
|
}
|
|
|
|
async dispatchKeybinding(windowId: number, keybinding: string): Promise<void> {
|
|
await this.whenUnfrozen(windowId);
|
|
|
|
const parts = KeybindingParser.parseUserBinding(keybinding);
|
|
|
|
for (let part of parts) {
|
|
await this._dispatchKeybinding(windowId, part);
|
|
}
|
|
}
|
|
|
|
private async _dispatchKeybinding(windowId: number, keybinding: SimpleKeybinding | ScanCodeBinding): Promise<void> {
|
|
if (keybinding instanceof ScanCodeBinding) {
|
|
throw new Error('ScanCodeBindings not supported');
|
|
}
|
|
|
|
const window = this.windowsMainService.getWindowById(windowId);
|
|
if (!window) {
|
|
throw new Error('Invalid window');
|
|
}
|
|
const webContents = window.win.webContents;
|
|
const noModifiedKeybinding = new SimpleKeybinding(false, false, false, false, keybinding.keyCode);
|
|
const resolvedKeybinding = new USLayoutResolvedKeybinding(noModifiedKeybinding.toChord(), OS);
|
|
const keyCode = resolvedKeybinding.getElectronAccelerator();
|
|
|
|
const modifiers: string[] = [];
|
|
|
|
if (keybinding.ctrlKey) {
|
|
modifiers.push('ctrl');
|
|
}
|
|
|
|
if (keybinding.metaKey) {
|
|
modifiers.push('meta');
|
|
}
|
|
|
|
if (keybinding.shiftKey) {
|
|
modifiers.push('shift');
|
|
}
|
|
|
|
if (keybinding.altKey) {
|
|
modifiers.push('alt');
|
|
}
|
|
|
|
webContents.sendInputEvent({ type: 'keyDown', keyCode, modifiers } as any);
|
|
|
|
if (!isSilentKeyCode(keybinding.keyCode)) {
|
|
webContents.sendInputEvent({ type: 'char', keyCode, modifiers } as any);
|
|
}
|
|
|
|
webContents.sendInputEvent({ type: 'keyUp', keyCode, modifiers } as any);
|
|
|
|
await timeout(100);
|
|
}
|
|
|
|
async click(windowId: number, selector: string, xoffset?: number, yoffset?: number): Promise<void> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
await windowDriver.click(selector, xoffset, yoffset);
|
|
}
|
|
|
|
async doubleClick(windowId: number, selector: string): Promise<void> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
await windowDriver.doubleClick(selector);
|
|
}
|
|
|
|
async setValue(windowId: number, selector: string, text: string): Promise<void> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
await windowDriver.setValue(selector, text);
|
|
}
|
|
|
|
async getTitle(windowId: number): Promise<string> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
return await windowDriver.getTitle();
|
|
}
|
|
|
|
async isActiveElement(windowId: number, selector: string): Promise<boolean> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
return await windowDriver.isActiveElement(selector);
|
|
}
|
|
|
|
async getElements(windowId: number, selector: string, recursive: boolean): Promise<IElement[]> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
return await windowDriver.getElements(selector, recursive);
|
|
}
|
|
|
|
async getElementXY(windowId: number, selector: string, xoffset?: number, yoffset?: number): Promise<{ x: number; y: number; }> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
return await windowDriver.getElementXY(selector, xoffset, yoffset);
|
|
}
|
|
|
|
async typeInEditor(windowId: number, selector: string, text: string): Promise<void> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
await windowDriver.typeInEditor(selector, text);
|
|
}
|
|
|
|
async getTerminalBuffer(windowId: number, selector: string): Promise<string[]> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
return await windowDriver.getTerminalBuffer(selector);
|
|
}
|
|
|
|
async writeInTerminal(windowId: number, selector: string, text: string): Promise<void> {
|
|
const windowDriver = await this.getWindowDriver(windowId);
|
|
await windowDriver.writeInTerminal(selector, text);
|
|
}
|
|
|
|
private async getWindowDriver(windowId: number): Promise<IWindowDriver> {
|
|
await this.whenUnfrozen(windowId);
|
|
|
|
const id = `window:${windowId}`;
|
|
const router = new StaticRouter(ctx => ctx === id);
|
|
const windowDriverChannel = this.windowServer.getChannel('windowDriver', router);
|
|
return new WindowDriverChannelClient(windowDriverChannel);
|
|
}
|
|
|
|
private async whenUnfrozen(windowId: number): Promise<void> {
|
|
while (this.reloadingWindowIds.has(windowId)) {
|
|
await Event.toPromise(this.onDidReloadingChange.event);
|
|
}
|
|
}
|
|
}
|
|
|
|
export async function serve(
|
|
windowServer: IPCServer,
|
|
handle: string,
|
|
environmentService: INativeEnvironmentService,
|
|
instantiationService: IInstantiationService
|
|
): Promise<IDisposable> {
|
|
const verbose = environmentService.driverVerbose;
|
|
const driver = instantiationService.createInstance(Driver as any, windowServer, { verbose }) as Driver; // {{SQL CARBON EDIT}} strict-null-check...i guess?
|
|
|
|
const windowDriverRegistryChannel = new WindowDriverRegistryChannel(driver);
|
|
windowServer.registerChannel('windowDriverRegistry', windowDriverRegistryChannel);
|
|
|
|
const server = await serveNet(handle);
|
|
const channel = new DriverChannel(driver);
|
|
server.registerChannel('driver', channel);
|
|
|
|
return combinedDisposable(server, windowServer);
|
|
}
|