mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-08 01:28:26 -05:00
* Merge from vscode a348d103d1256a06a2c9b3f9b406298a9fef6898 * Fixes and cleanup * Distro * Fix hygiene yarn * delete no yarn lock changes file * Fix hygiene * Fix layer check * Fix CI * Skip lib checks * Remove tests deleted in vs code * Fix tests * Distro * Fix tests and add removed extension point * Skip failing notebook tests for now * Disable broken tests and cleanup build folder * Update yarn.lock and fix smoke tests * Bump sqlite * fix contributed actions and file spacing * Fix user data path * Update yarn.locks Co-authored-by: ADS Merger <karlb@microsoft.com>
348 lines
12 KiB
TypeScript
348 lines
12 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 'vs/css!./media/actions';
|
|
|
|
import { localize } from 'vs/nls';
|
|
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
|
|
import { domEvent } from 'vs/base/browser/event';
|
|
import { Color } from 'vs/base/common/color';
|
|
import { Event } from 'vs/base/common/event';
|
|
import { IDisposable, toDisposable, dispose, Disposable, DisposableStore } from 'vs/base/common/lifecycle';
|
|
import { getDomNodePagePosition, createStyleSheet, createCSSRule, append, $ } from 'vs/base/browser/dom';
|
|
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
|
|
import { IContextKeyService } from 'vs/platform/contextkey/common/contextkey';
|
|
import { Context } from 'vs/platform/contextkey/browser/contextKeyService';
|
|
import { StandardKeyboardEvent } from 'vs/base/browser/keyboardEvent';
|
|
import { timeout } from 'vs/base/common/async';
|
|
import { ILayoutService } from 'vs/platform/layout/browser/layoutService';
|
|
import { Registry } from 'vs/platform/registry/common/platform';
|
|
import { registerAction2, Action2 } from 'vs/platform/actions/common/actions';
|
|
import { IStorageService } from 'vs/platform/storage/common/storage';
|
|
import { clamp } from 'vs/base/common/numbers';
|
|
import { KeyCode } from 'vs/base/common/keyCodes';
|
|
import { IConfigurationRegistry, Extensions as ConfigurationExtensions } from 'vs/platform/configuration/common/configurationRegistry';
|
|
import { ILogService } from 'vs/platform/log/common/log';
|
|
import { IWorkingCopyService } from 'vs/workbench/services/workingCopy/common/workingCopyService';
|
|
import { ServicesAccessor } from 'vs/platform/instantiation/common/instantiation';
|
|
import { CATEGORIES } from 'vs/workbench/common/actions';
|
|
import { IWorkingCopyBackupService } from 'vs/workbench/services/workingCopy/common/workingCopyBackup';
|
|
|
|
class InspectContextKeysAction extends Action2 {
|
|
|
|
constructor() {
|
|
super({
|
|
id: 'workbench.action.inspectContextKeys',
|
|
title: { value: localize('inspect context keys', "Inspect Context Keys"), original: 'Inspect Context Keys' },
|
|
category: CATEGORIES.Developer,
|
|
f1: true
|
|
});
|
|
}
|
|
|
|
run(accessor: ServicesAccessor): void {
|
|
const contextKeyService = accessor.get(IContextKeyService);
|
|
|
|
const disposables = new DisposableStore();
|
|
|
|
const stylesheet = createStyleSheet();
|
|
disposables.add(toDisposable(() => {
|
|
if (stylesheet.parentNode) {
|
|
stylesheet.parentNode.removeChild(stylesheet);
|
|
}
|
|
}));
|
|
createCSSRule('*', 'cursor: crosshair !important;', stylesheet);
|
|
|
|
const hoverFeedback = document.createElement('div');
|
|
document.body.appendChild(hoverFeedback);
|
|
disposables.add(toDisposable(() => document.body.removeChild(hoverFeedback)));
|
|
|
|
hoverFeedback.style.position = 'absolute';
|
|
hoverFeedback.style.pointerEvents = 'none';
|
|
hoverFeedback.style.backgroundColor = 'rgba(255, 0, 0, 0.5)';
|
|
hoverFeedback.style.zIndex = '1000';
|
|
|
|
const onMouseMove = domEvent(document.body, 'mousemove', true);
|
|
disposables.add(onMouseMove(e => {
|
|
const target = e.target as HTMLElement;
|
|
const position = getDomNodePagePosition(target);
|
|
|
|
hoverFeedback.style.top = `${position.top}px`;
|
|
hoverFeedback.style.left = `${position.left}px`;
|
|
hoverFeedback.style.width = `${position.width}px`;
|
|
hoverFeedback.style.height = `${position.height}px`;
|
|
}));
|
|
|
|
const onMouseDown = Event.once(domEvent(document.body, 'mousedown', true));
|
|
onMouseDown(e => { e.preventDefault(); e.stopPropagation(); }, null, disposables);
|
|
|
|
const onMouseUp = Event.once(domEvent(document.body, 'mouseup', true));
|
|
onMouseUp(e => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
|
|
const context = contextKeyService.getContext(e.target as HTMLElement) as Context;
|
|
console.log(context.collectAllValues());
|
|
|
|
dispose(disposables);
|
|
}, null, disposables);
|
|
}
|
|
}
|
|
|
|
class ToggleScreencastModeAction extends Action2 {
|
|
|
|
static disposable: IDisposable | undefined;
|
|
|
|
constructor() {
|
|
super({
|
|
id: 'workbench.action.toggleScreencastMode',
|
|
title: { value: localize('toggle screencast mode', "Toggle Screencast Mode"), original: 'Toggle Screencast Mode' },
|
|
category: CATEGORIES.Developer,
|
|
f1: true
|
|
});
|
|
}
|
|
|
|
run(accessor: ServicesAccessor): void {
|
|
if (ToggleScreencastModeAction.disposable) {
|
|
ToggleScreencastModeAction.disposable.dispose();
|
|
ToggleScreencastModeAction.disposable = undefined;
|
|
return;
|
|
}
|
|
|
|
const layoutService = accessor.get(ILayoutService);
|
|
const configurationService = accessor.get(IConfigurationService);
|
|
const keybindingService = accessor.get(IKeybindingService);
|
|
|
|
const disposables = new DisposableStore();
|
|
|
|
const container = layoutService.container;
|
|
const mouseMarker = append(container, $('.screencast-mouse'));
|
|
disposables.add(toDisposable(() => mouseMarker.remove()));
|
|
|
|
const onMouseDown = domEvent(container, 'mousedown', true);
|
|
const onMouseUp = domEvent(container, 'mouseup', true);
|
|
const onMouseMove = domEvent(container, 'mousemove', true);
|
|
|
|
const updateMouseIndicatorColor = () => {
|
|
mouseMarker.style.borderColor = Color.fromHex(configurationService.getValue<string>('screencastMode.mouseIndicatorColor')).toString();
|
|
};
|
|
|
|
let mouseIndicatorSize: number;
|
|
const updateMouseIndicatorSize = () => {
|
|
mouseIndicatorSize = clamp(configurationService.getValue<number>('screencastMode.mouseIndicatorSize') || 20, 20, 100);
|
|
|
|
mouseMarker.style.height = `${mouseIndicatorSize}px`;
|
|
mouseMarker.style.width = `${mouseIndicatorSize}px`;
|
|
};
|
|
|
|
updateMouseIndicatorColor();
|
|
updateMouseIndicatorSize();
|
|
|
|
disposables.add(onMouseDown(e => {
|
|
mouseMarker.style.top = `${e.clientY - mouseIndicatorSize / 2}px`;
|
|
mouseMarker.style.left = `${e.clientX - mouseIndicatorSize / 2}px`;
|
|
mouseMarker.style.display = 'block';
|
|
|
|
const mouseMoveListener = onMouseMove(e => {
|
|
mouseMarker.style.top = `${e.clientY - mouseIndicatorSize / 2}px`;
|
|
mouseMarker.style.left = `${e.clientX - mouseIndicatorSize / 2}px`;
|
|
});
|
|
|
|
Event.once(onMouseUp)(() => {
|
|
mouseMarker.style.display = 'none';
|
|
mouseMoveListener.dispose();
|
|
});
|
|
}));
|
|
|
|
const keyboardMarker = append(container, $('.screencast-keyboard'));
|
|
disposables.add(toDisposable(() => keyboardMarker.remove()));
|
|
|
|
const updateKeyboardFontSize = () => {
|
|
keyboardMarker.style.fontSize = `${clamp(configurationService.getValue<number>('screencastMode.fontSize') || 56, 20, 100)}px`;
|
|
};
|
|
|
|
const updateKeyboardMarker = () => {
|
|
keyboardMarker.style.bottom = `${clamp(configurationService.getValue<number>('screencastMode.verticalOffset') || 0, 0, 90)}%`;
|
|
};
|
|
|
|
let keyboardMarkerTimeout: number;
|
|
const updateKeyboardMarkerTimeout = () => {
|
|
keyboardMarkerTimeout = clamp(configurationService.getValue<number>('screencastMode.keyboardOverlayTimeout') || 800, 500, 5000);
|
|
};
|
|
|
|
updateKeyboardFontSize();
|
|
updateKeyboardMarker();
|
|
updateKeyboardMarkerTimeout();
|
|
|
|
disposables.add(configurationService.onDidChangeConfiguration(e => {
|
|
if (e.affectsConfiguration('screencastMode.verticalOffset')) {
|
|
updateKeyboardMarker();
|
|
}
|
|
|
|
if (e.affectsConfiguration('screencastMode.fontSize')) {
|
|
updateKeyboardFontSize();
|
|
}
|
|
|
|
if (e.affectsConfiguration('screencastMode.keyboardOverlayTimeout')) {
|
|
updateKeyboardMarkerTimeout();
|
|
}
|
|
|
|
if (e.affectsConfiguration('screencastMode.mouseIndicatorColor')) {
|
|
updateMouseIndicatorColor();
|
|
}
|
|
|
|
if (e.affectsConfiguration('screencastMode.mouseIndicatorSize')) {
|
|
updateMouseIndicatorSize();
|
|
}
|
|
}));
|
|
|
|
const onKeyDown = domEvent(window, 'keydown', true);
|
|
let keyboardTimeout: IDisposable = Disposable.None;
|
|
let length = 0;
|
|
|
|
disposables.add(onKeyDown(e => {
|
|
keyboardTimeout.dispose();
|
|
|
|
const event = new StandardKeyboardEvent(e);
|
|
const shortcut = keybindingService.softDispatch(event, event.target);
|
|
|
|
if (shortcut || !configurationService.getValue<boolean>('screencastMode.onlyKeyboardShortcuts')) {
|
|
if (
|
|
event.ctrlKey || event.altKey || event.metaKey || event.shiftKey
|
|
|| length > 20
|
|
|| event.keyCode === KeyCode.Backspace || event.keyCode === KeyCode.Escape
|
|
) {
|
|
keyboardMarker.innerText = '';
|
|
length = 0;
|
|
}
|
|
|
|
const keybinding = keybindingService.resolveKeyboardEvent(event);
|
|
const label = keybinding.getLabel();
|
|
const key = $('span.key', {}, label || '');
|
|
length++;
|
|
append(keyboardMarker, key);
|
|
}
|
|
|
|
const promise = timeout(keyboardMarkerTimeout);
|
|
keyboardTimeout = toDisposable(() => promise.cancel());
|
|
|
|
promise.then(() => {
|
|
keyboardMarker.textContent = '';
|
|
length = 0;
|
|
});
|
|
}));
|
|
|
|
ToggleScreencastModeAction.disposable = disposables;
|
|
}
|
|
}
|
|
|
|
class LogStorageAction extends Action2 {
|
|
|
|
constructor() {
|
|
super({
|
|
id: 'workbench.action.logStorage',
|
|
title: { value: localize({ key: 'logStorage', comment: ['A developer only action to log the contents of the storage for the current window.'] }, "Log Storage Database Contents"), original: 'Log Storage Database Contents' },
|
|
category: CATEGORIES.Developer,
|
|
f1: true
|
|
});
|
|
}
|
|
|
|
run(accessor: ServicesAccessor): void {
|
|
accessor.get(IStorageService).logStorage();
|
|
}
|
|
}
|
|
|
|
class LogWorkingCopiesAction extends Action2 {
|
|
|
|
constructor() {
|
|
super({
|
|
id: 'workbench.action.logWorkingCopies',
|
|
title: { value: localize({ key: 'logWorkingCopies', comment: ['A developer only action to log the working copies that exist.'] }, "Log Working Copies"), original: 'Log Working Copies' },
|
|
category: CATEGORIES.Developer,
|
|
f1: true
|
|
});
|
|
}
|
|
|
|
async run(accessor: ServicesAccessor): Promise<void> {
|
|
const workingCopyService = accessor.get(IWorkingCopyService);
|
|
const workingCopyBackupService = accessor.get(IWorkingCopyBackupService);
|
|
const logService = accessor.get(ILogService);
|
|
|
|
const backups = await workingCopyBackupService.getBackups();
|
|
|
|
const msg = [
|
|
``,
|
|
`[Working Copies]`,
|
|
...(workingCopyService.workingCopies.length > 0) ?
|
|
workingCopyService.workingCopies.map(workingCopy => `${workingCopy.isDirty() ? '● ' : ''}${workingCopy.resource.toString(true)} (typeId: ${workingCopy.typeId || '<no typeId>'})`) :
|
|
['<none>'],
|
|
``,
|
|
`[Backups]`,
|
|
...(backups.length > 0) ?
|
|
backups.map(backup => `${backup.resource.toString(true)} (typeId: ${backup.typeId || '<no typeId>'})`) :
|
|
['<none>'],
|
|
];
|
|
|
|
logService.info(msg.join('\n'));
|
|
}
|
|
}
|
|
|
|
// --- Actions Registration
|
|
registerAction2(InspectContextKeysAction);
|
|
registerAction2(ToggleScreencastModeAction);
|
|
registerAction2(LogStorageAction);
|
|
registerAction2(LogWorkingCopiesAction);
|
|
|
|
// --- Configuration
|
|
|
|
// Screen Cast Mode
|
|
const configurationRegistry = Registry.as<IConfigurationRegistry>(ConfigurationExtensions.Configuration);
|
|
configurationRegistry.registerConfiguration({
|
|
id: 'screencastMode',
|
|
order: 9,
|
|
title: localize('screencastModeConfigurationTitle', "Screencast Mode"),
|
|
type: 'object',
|
|
properties: {
|
|
'screencastMode.verticalOffset': {
|
|
type: 'number',
|
|
default: 20,
|
|
minimum: 0,
|
|
maximum: 90,
|
|
description: localize('screencastMode.location.verticalPosition', "Controls the vertical offset of the screencast mode overlay from the bottom as a percentage of the workbench height.")
|
|
},
|
|
'screencastMode.fontSize': {
|
|
type: 'number',
|
|
default: 56,
|
|
minimum: 20,
|
|
maximum: 100,
|
|
description: localize('screencastMode.fontSize', "Controls the font size (in pixels) of the screencast mode keyboard.")
|
|
},
|
|
'screencastMode.onlyKeyboardShortcuts': {
|
|
type: 'boolean',
|
|
description: localize('screencastMode.onlyKeyboardShortcuts', "Only show keyboard shortcuts in screencast mode."),
|
|
default: false
|
|
},
|
|
'screencastMode.keyboardOverlayTimeout': {
|
|
type: 'number',
|
|
default: 800,
|
|
minimum: 500,
|
|
maximum: 5000,
|
|
description: localize('screencastMode.keyboardOverlayTimeout', "Controls how long (in milliseconds) the keyboard overlay is shown in screencast mode.")
|
|
},
|
|
'screencastMode.mouseIndicatorColor': {
|
|
type: 'string',
|
|
format: 'color-hex',
|
|
default: '#FF0000',
|
|
description: localize('screencastMode.mouseIndicatorColor', "Controls the color in hex (#RGB, #RGBA, #RRGGBB or #RRGGBBAA) of the mouse indicator in screencast mode.")
|
|
},
|
|
'screencastMode.mouseIndicatorSize': {
|
|
type: 'number',
|
|
default: 20,
|
|
minimum: 20,
|
|
maximum: 100,
|
|
description: localize('screencastMode.mouseIndicatorSize', "Controls the size (in pixels) of the mouse indicator in screencast mode.")
|
|
},
|
|
}
|
|
});
|