Path: blob/main/src/vs/workbench/browser/actions/developerActions.ts
3296 views
/*---------------------------------------------------------------------------------------------1* Copyright (c) Microsoft Corporation. All rights reserved.2* Licensed under the MIT License. See License.txt in the project root for license information.3*--------------------------------------------------------------------------------------------*/45import './media/actions.css';67import { localize, localize2 } from '../../../nls.js';8import { IKeybindingService } from '../../../platform/keybinding/common/keybinding.js';9import { DomEmitter } from '../../../base/browser/event.js';10import { Color } from '../../../base/common/color.js';11import { Emitter, Event } from '../../../base/common/event.js';12import { IDisposable, toDisposable, dispose, DisposableStore, setDisposableTracker, DisposableTracker, DisposableInfo } from '../../../base/common/lifecycle.js';13import { getDomNodePagePosition, append, $, getActiveDocument, onDidRegisterWindow, getWindows } from '../../../base/browser/dom.js';14import { createCSSRule, createStyleSheet } from '../../../base/browser/domStylesheets.js';15import { IConfigurationService } from '../../../platform/configuration/common/configuration.js';16import { ContextKeyExpr, IContextKeyService, RawContextKey } from '../../../platform/contextkey/common/contextkey.js';17import { Context } from '../../../platform/contextkey/browser/contextKeyService.js';18import { StandardKeyboardEvent } from '../../../base/browser/keyboardEvent.js';19import { RunOnceScheduler } from '../../../base/common/async.js';20import { ILayoutService } from '../../../platform/layout/browser/layoutService.js';21import { Registry } from '../../../platform/registry/common/platform.js';22import { registerAction2, Action2, MenuRegistry } from '../../../platform/actions/common/actions.js';23import { IStorageService, StorageScope, StorageTarget } from '../../../platform/storage/common/storage.js';24import { clamp } from '../../../base/common/numbers.js';25import { KeyCode } from '../../../base/common/keyCodes.js';26import { IConfigurationRegistry, Extensions as ConfigurationExtensions } from '../../../platform/configuration/common/configurationRegistry.js';27import { ILogService } from '../../../platform/log/common/log.js';28import { IWorkingCopyService } from '../../services/workingCopy/common/workingCopyService.js';29import { ServicesAccessor } from '../../../platform/instantiation/common/instantiation.js';30import { Categories } from '../../../platform/action/common/actionCommonCategories.js';31import { IWorkingCopyBackupService } from '../../services/workingCopy/common/workingCopyBackup.js';32import { ResolutionResult, ResultKind } from '../../../platform/keybinding/common/keybindingResolver.js';33import { IDialogService } from '../../../platform/dialogs/common/dialogs.js';34import { IOutputService } from '../../services/output/common/output.js';35import { windowLogId } from '../../services/log/common/logConstants.js';36import { ByteSize } from '../../../platform/files/common/files.js';37import { IQuickInputService, IQuickPickItem } from '../../../platform/quickinput/common/quickInput.js';38import { IUserDataProfileService } from '../../services/userDataProfile/common/userDataProfile.js';39import { IEditorService } from '../../services/editor/common/editorService.js';40import product from '../../../platform/product/common/product.js';41import { CommandsRegistry } from '../../../platform/commands/common/commands.js';42import { IEnvironmentService } from '../../../platform/environment/common/environment.js';4344class InspectContextKeysAction extends Action2 {4546constructor() {47super({48id: 'workbench.action.inspectContextKeys',49title: localize2('inspect context keys', 'Inspect Context Keys'),50category: Categories.Developer,51f1: true52});53}5455run(accessor: ServicesAccessor): void {56const contextKeyService = accessor.get(IContextKeyService);5758const disposables = new DisposableStore();5960const stylesheet = createStyleSheet(undefined, undefined, disposables);61createCSSRule('*', 'cursor: crosshair !important;', stylesheet);6263const hoverFeedback = document.createElement('div');64const activeDocument = getActiveDocument();65activeDocument.body.appendChild(hoverFeedback);66disposables.add(toDisposable(() => hoverFeedback.remove()));6768hoverFeedback.style.position = 'absolute';69hoverFeedback.style.pointerEvents = 'none';70hoverFeedback.style.backgroundColor = 'rgba(255, 0, 0, 0.5)';71hoverFeedback.style.zIndex = '1000';7273const onMouseMove = disposables.add(new DomEmitter(activeDocument, 'mousemove', true));74disposables.add(onMouseMove.event(e => {75const target = e.target as HTMLElement;76const position = getDomNodePagePosition(target);7778hoverFeedback.style.top = `${position.top}px`;79hoverFeedback.style.left = `${position.left}px`;80hoverFeedback.style.width = `${position.width}px`;81hoverFeedback.style.height = `${position.height}px`;82}));8384const onMouseDown = disposables.add(new DomEmitter(activeDocument, 'mousedown', true));85Event.once(onMouseDown.event)(e => { e.preventDefault(); e.stopPropagation(); }, null, disposables);8687const onMouseUp = disposables.add(new DomEmitter(activeDocument, 'mouseup', true));88Event.once(onMouseUp.event)(e => {89e.preventDefault();90e.stopPropagation();9192const context = contextKeyService.getContext(e.target as HTMLElement) as Context;93console.log(context.collectAllValues());9495dispose(disposables);96}, null, disposables);97}98}99100interface IScreencastKeyboardOptions {101readonly showKeys?: boolean;102readonly showKeybindings?: boolean;103readonly showCommands?: boolean;104readonly showCommandGroups?: boolean;105readonly showSingleEditorCursorMoves?: boolean;106}107108class ToggleScreencastModeAction extends Action2 {109110static disposable: IDisposable | undefined;111112constructor() {113super({114id: 'workbench.action.toggleScreencastMode',115title: localize2('toggle screencast mode', 'Toggle Screencast Mode'),116category: Categories.Developer,117f1: true118});119}120121run(accessor: ServicesAccessor): void {122if (ToggleScreencastModeAction.disposable) {123ToggleScreencastModeAction.disposable.dispose();124ToggleScreencastModeAction.disposable = undefined;125return;126}127128const layoutService = accessor.get(ILayoutService);129const configurationService = accessor.get(IConfigurationService);130const keybindingService = accessor.get(IKeybindingService);131132const disposables = new DisposableStore();133134const container = layoutService.activeContainer;135136const mouseMarker = append(container, $('.screencast-mouse'));137disposables.add(toDisposable(() => mouseMarker.remove()));138139const keyboardMarker = append(container, $('.screencast-keyboard'));140disposables.add(toDisposable(() => keyboardMarker.remove()));141142const onMouseDown = disposables.add(new Emitter<MouseEvent>());143const onMouseUp = disposables.add(new Emitter<MouseEvent>());144const onMouseMove = disposables.add(new Emitter<MouseEvent>());145146function registerContainerListeners(container: HTMLElement, disposables: DisposableStore): void {147disposables.add(disposables.add(new DomEmitter(container, 'mousedown', true)).event(e => onMouseDown.fire(e)));148disposables.add(disposables.add(new DomEmitter(container, 'mouseup', true)).event(e => onMouseUp.fire(e)));149disposables.add(disposables.add(new DomEmitter(container, 'mousemove', true)).event(e => onMouseMove.fire(e)));150}151152for (const { window, disposables } of getWindows()) {153registerContainerListeners(layoutService.getContainer(window), disposables);154}155156disposables.add(onDidRegisterWindow(({ window, disposables }) => registerContainerListeners(layoutService.getContainer(window), disposables)));157158disposables.add(layoutService.onDidChangeActiveContainer(() => {159layoutService.activeContainer.appendChild(mouseMarker);160layoutService.activeContainer.appendChild(keyboardMarker);161}));162163const updateMouseIndicatorColor = () => {164mouseMarker.style.borderColor = Color.fromHex(configurationService.getValue<string>('screencastMode.mouseIndicatorColor')).toString();165};166167let mouseIndicatorSize: number;168const updateMouseIndicatorSize = () => {169mouseIndicatorSize = clamp(configurationService.getValue<number>('screencastMode.mouseIndicatorSize') || 20, 20, 100);170171mouseMarker.style.height = `${mouseIndicatorSize}px`;172mouseMarker.style.width = `${mouseIndicatorSize}px`;173};174175updateMouseIndicatorColor();176updateMouseIndicatorSize();177178disposables.add(onMouseDown.event(e => {179mouseMarker.style.top = `${e.clientY - mouseIndicatorSize / 2}px`;180mouseMarker.style.left = `${e.clientX - mouseIndicatorSize / 2}px`;181mouseMarker.style.display = 'block';182mouseMarker.style.transform = `scale(${1})`;183mouseMarker.style.transition = 'transform 0.1s';184185const mouseMoveListener = onMouseMove.event(e => {186mouseMarker.style.top = `${e.clientY - mouseIndicatorSize / 2}px`;187mouseMarker.style.left = `${e.clientX - mouseIndicatorSize / 2}px`;188mouseMarker.style.transform = `scale(${.8})`;189});190191Event.once(onMouseUp.event)(() => {192mouseMarker.style.display = 'none';193mouseMoveListener.dispose();194});195}));196197const updateKeyboardFontSize = () => {198keyboardMarker.style.fontSize = `${clamp(configurationService.getValue<number>('screencastMode.fontSize') || 56, 20, 100)}px`;199};200201const updateKeyboardMarker = () => {202keyboardMarker.style.bottom = `${clamp(configurationService.getValue<number>('screencastMode.verticalOffset') || 0, 0, 90)}%`;203};204205let keyboardMarkerTimeout!: number;206const updateKeyboardMarkerTimeout = () => {207keyboardMarkerTimeout = clamp(configurationService.getValue<number>('screencastMode.keyboardOverlayTimeout') || 800, 500, 5000);208};209210updateKeyboardFontSize();211updateKeyboardMarker();212updateKeyboardMarkerTimeout();213214disposables.add(configurationService.onDidChangeConfiguration(e => {215if (e.affectsConfiguration('screencastMode.verticalOffset')) {216updateKeyboardMarker();217}218219if (e.affectsConfiguration('screencastMode.fontSize')) {220updateKeyboardFontSize();221}222223if (e.affectsConfiguration('screencastMode.keyboardOverlayTimeout')) {224updateKeyboardMarkerTimeout();225}226227if (e.affectsConfiguration('screencastMode.mouseIndicatorColor')) {228updateMouseIndicatorColor();229}230231if (e.affectsConfiguration('screencastMode.mouseIndicatorSize')) {232updateMouseIndicatorSize();233}234}));235236const onKeyDown = disposables.add(new Emitter<KeyboardEvent>());237const onCompositionStart = disposables.add(new Emitter<CompositionEvent>());238const onCompositionUpdate = disposables.add(new Emitter<CompositionEvent>());239const onCompositionEnd = disposables.add(new Emitter<CompositionEvent>());240241function registerWindowListeners(window: Window, disposables: DisposableStore): void {242disposables.add(disposables.add(new DomEmitter(window, 'keydown', true)).event(e => onKeyDown.fire(e)));243disposables.add(disposables.add(new DomEmitter(window, 'compositionstart', true)).event(e => onCompositionStart.fire(e)));244disposables.add(disposables.add(new DomEmitter(window, 'compositionupdate', true)).event(e => onCompositionUpdate.fire(e)));245disposables.add(disposables.add(new DomEmitter(window, 'compositionend', true)).event(e => onCompositionEnd.fire(e)));246}247248for (const { window, disposables } of getWindows()) {249registerWindowListeners(window, disposables);250}251252disposables.add(onDidRegisterWindow(({ window, disposables }) => registerWindowListeners(window, disposables)));253254let length = 0;255let composing: Element | undefined = undefined;256let imeBackSpace = false;257258const clearKeyboardScheduler = new RunOnceScheduler(() => {259keyboardMarker.textContent = '';260composing = undefined;261length = 0;262}, keyboardMarkerTimeout);263264disposables.add(onCompositionStart.event(e => {265imeBackSpace = true;266}));267268disposables.add(onCompositionUpdate.event(e => {269if (e.data && imeBackSpace) {270if (length > 20) {271keyboardMarker.innerText = '';272length = 0;273}274composing = composing ?? append(keyboardMarker, $('span.key'));275composing.textContent = e.data;276} else if (imeBackSpace) {277keyboardMarker.innerText = '';278append(keyboardMarker, $('span.key', {}, `Backspace`));279}280clearKeyboardScheduler.schedule();281}));282283disposables.add(onCompositionEnd.event(e => {284composing = undefined;285length++;286}));287288disposables.add(onKeyDown.event(e => {289if (e.key === 'Process' || /[\uac00-\ud787\u3131-\u314e\u314f-\u3163\u3041-\u3094\u30a1-\u30f4\u30fc\u3005\u3006\u3024\u4e00-\u9fa5]/u.test(e.key)) {290if (e.code === 'Backspace') {291imeBackSpace = true;292} else if (!e.code.includes('Key')) {293composing = undefined;294imeBackSpace = false;295} else {296imeBackSpace = true;297}298clearKeyboardScheduler.schedule();299return;300}301302if (e.isComposing) {303return;304}305306const options = configurationService.getValue<IScreencastKeyboardOptions>('screencastMode.keyboardOptions');307const event = new StandardKeyboardEvent(e);308const shortcut = keybindingService.softDispatch(event, event.target);309310// Hide the single arrow key pressed311if (shortcut.kind === ResultKind.KbFound && shortcut.commandId && !(options.showSingleEditorCursorMoves ?? true) && (312['cursorLeft', 'cursorRight', 'cursorUp', 'cursorDown'].includes(shortcut.commandId))313) {314return;315}316317if (318event.ctrlKey || event.altKey || event.metaKey || event.shiftKey319|| length > 20320|| event.keyCode === KeyCode.Backspace || event.keyCode === KeyCode.Escape321|| event.keyCode === KeyCode.UpArrow || event.keyCode === KeyCode.DownArrow322|| event.keyCode === KeyCode.LeftArrow || event.keyCode === KeyCode.RightArrow323) {324keyboardMarker.innerText = '';325length = 0;326}327328const keybinding = keybindingService.resolveKeyboardEvent(event);329const commandDetails = (this._isKbFound(shortcut) && shortcut.commandId) ? this.getCommandDetails(shortcut.commandId) : undefined;330331let commandAndGroupLabel = commandDetails?.title;332let keyLabel: string | undefined | null = keybinding.getLabel();333334if (commandDetails) {335if ((options.showCommandGroups ?? false) && commandDetails.category) {336commandAndGroupLabel = `${commandDetails.category}: ${commandAndGroupLabel} `;337}338339if (this._isKbFound(shortcut) && shortcut.commandId) {340const keybindings = keybindingService.lookupKeybindings(shortcut.commandId)341.filter(k => k.getLabel()?.endsWith(keyLabel ?? ''));342343if (keybindings.length > 0) {344keyLabel = keybindings[keybindings.length - 1].getLabel();345}346}347}348349if ((options.showCommands ?? true) && commandAndGroupLabel) {350append(keyboardMarker, $('span.title', {}, `${commandAndGroupLabel} `));351}352353if ((options.showKeys ?? true) || ((options.showKeybindings ?? true) && this._isKbFound(shortcut))) {354// Fix label for arrow keys355keyLabel = keyLabel?.replace('UpArrow', '↑')356?.replace('DownArrow', '↓')357?.replace('LeftArrow', '←')358?.replace('RightArrow', '→');359360append(keyboardMarker, $('span.key', {}, keyLabel ?? ''));361}362363length++;364clearKeyboardScheduler.schedule();365}));366367ToggleScreencastModeAction.disposable = disposables;368}369370private _isKbFound(resolutionResult: ResolutionResult): resolutionResult is { kind: ResultKind.KbFound; commandId: string | null; commandArgs: any; isBubble: boolean } {371return resolutionResult.kind === ResultKind.KbFound;372}373374private getCommandDetails(commandId: string): { title: string; category?: string } | undefined {375const fromMenuRegistry = MenuRegistry.getCommand(commandId);376377if (fromMenuRegistry) {378return {379title: typeof fromMenuRegistry.title === 'string' ? fromMenuRegistry.title : fromMenuRegistry.title.value,380category: fromMenuRegistry.category ? (typeof fromMenuRegistry.category === 'string' ? fromMenuRegistry.category : fromMenuRegistry.category.value) : undefined381};382}383384const fromCommandsRegistry = CommandsRegistry.getCommand(commandId);385386if (fromCommandsRegistry && fromCommandsRegistry.metadata?.description) {387return { title: typeof fromCommandsRegistry.metadata.description === 'string' ? fromCommandsRegistry.metadata.description : fromCommandsRegistry.metadata.description.value };388}389390return undefined;391}392}393394class LogStorageAction extends Action2 {395396constructor() {397super({398id: 'workbench.action.logStorage',399title: localize2({ key: 'logStorage', comment: ['A developer only action to log the contents of the storage for the current window.'] }, "Log Storage Database Contents"),400category: Categories.Developer,401f1: true402});403}404405run(accessor: ServicesAccessor): void {406const storageService = accessor.get(IStorageService);407const dialogService = accessor.get(IDialogService);408409storageService.log();410411dialogService.info(localize('storageLogDialogMessage', "The storage database contents have been logged to the developer tools."), localize('storageLogDialogDetails', "Open developer tools from the menu and select the Console tab."));412}413}414415class LogWorkingCopiesAction extends Action2 {416417constructor() {418super({419id: 'workbench.action.logWorkingCopies',420title: localize2({ key: 'logWorkingCopies', comment: ['A developer only action to log the working copies that exist.'] }, "Log Working Copies"),421category: Categories.Developer,422f1: true423});424}425426async run(accessor: ServicesAccessor): Promise<void> {427const workingCopyService = accessor.get(IWorkingCopyService);428const workingCopyBackupService = accessor.get(IWorkingCopyBackupService);429const logService = accessor.get(ILogService);430const outputService = accessor.get(IOutputService);431432const backups = await workingCopyBackupService.getBackups();433434const msg = [435``,436`[Working Copies]`,437...(workingCopyService.workingCopies.length > 0) ?438workingCopyService.workingCopies.map(workingCopy => `${workingCopy.isDirty() ? '● ' : ''}${workingCopy.resource.toString(true)} (typeId: ${workingCopy.typeId || '<no typeId>'})`) :439['<none>'],440``,441`[Backups]`,442...(backups.length > 0) ?443backups.map(backup => `${backup.resource.toString(true)} (typeId: ${backup.typeId || '<no typeId>'})`) :444['<none>'],445];446447logService.info(msg.join('\n'));448449outputService.showChannel(windowLogId, true);450}451}452453class RemoveLargeStorageEntriesAction extends Action2 {454455private static SIZE_THRESHOLD = 1024 * 16; // 16kb456457constructor() {458super({459id: 'workbench.action.removeLargeStorageDatabaseEntries',460title: localize2('removeLargeStorageDatabaseEntries', 'Remove Large Storage Database Entries...'),461category: Categories.Developer,462f1: true463});464}465466async run(accessor: ServicesAccessor): Promise<void> {467const storageService = accessor.get(IStorageService);468const quickInputService = accessor.get(IQuickInputService);469const userDataProfileService = accessor.get(IUserDataProfileService);470const dialogService = accessor.get(IDialogService);471const environmentService = accessor.get(IEnvironmentService);472473interface IStorageItem extends IQuickPickItem {474readonly key: string;475readonly scope: StorageScope;476readonly target: StorageTarget;477readonly size: number;478}479480const items: IStorageItem[] = [];481482for (const scope of [StorageScope.APPLICATION, StorageScope.PROFILE, StorageScope.WORKSPACE]) {483if (scope === StorageScope.PROFILE && userDataProfileService.currentProfile.isDefault) {484continue; // avoid duplicates485}486487for (const target of [StorageTarget.MACHINE, StorageTarget.USER]) {488for (const key of storageService.keys(scope, target)) {489const value = storageService.get(key, scope);490if (value && (!environmentService.isBuilt /* show all keys in dev */ || value.length > RemoveLargeStorageEntriesAction.SIZE_THRESHOLD)) {491items.push({492key,493scope,494target,495size: value.length,496label: key,497description: ByteSize.formatSize(value.length),498detail: localize('largeStorageItemDetail', "Scope: {0}, Target: {1}", scope === StorageScope.APPLICATION ? localize('global', "Global") : scope === StorageScope.PROFILE ? localize('profile', "Profile") : localize('workspace', "Workspace"), target === StorageTarget.MACHINE ? localize('machine', "Machine") : localize('user', "User")),499});500}501}502}503}504505items.sort((itemA, itemB) => itemB.size - itemA.size);506507const selectedItems = await new Promise<readonly IStorageItem[]>(resolve => {508const disposables = new DisposableStore();509510const picker = disposables.add(quickInputService.createQuickPick<IStorageItem>());511picker.items = items;512picker.canSelectMany = true;513picker.ok = false;514picker.customButton = true;515picker.hideCheckAll = true;516picker.customLabel = localize('removeLargeStorageEntriesPickerButton', "Remove");517picker.placeholder = localize('removeLargeStorageEntriesPickerPlaceholder', "Select large entries to remove from storage");518519if (items.length === 0) {520picker.description = localize('removeLargeStorageEntriesPickerDescriptionNoEntries', "There are no large storage entries to remove.");521}522523picker.show();524525disposables.add(picker.onDidCustom(() => {526resolve(picker.selectedItems);527picker.hide();528}));529530disposables.add(picker.onDidHide(() => disposables.dispose()));531});532533if (selectedItems.length === 0) {534return;535}536537const { confirmed } = await dialogService.confirm({538type: 'warning',539message: localize('removeLargeStorageEntriesConfirmRemove', "Do you want to remove the selected storage entries from the database?"),540detail: localize('removeLargeStorageEntriesConfirmRemoveDetail', "{0}\n\nThis action is irreversible and may result in data loss!", selectedItems.map(item => item.label).join('\n')),541primaryButton: localize({ key: 'removeLargeStorageEntriesButtonLabel', comment: ['&& denotes a mnemonic'] }, "&&Remove")542});543544if (!confirmed) {545return;546}547548const scopesToOptimize = new Set<StorageScope>();549for (const item of selectedItems) {550storageService.remove(item.key, item.scope);551scopesToOptimize.add(item.scope);552}553554for (const scope of scopesToOptimize) {555await storageService.optimize(scope);556}557}558}559560let tracker: DisposableTracker | undefined = undefined;561let trackedDisposables = new Set<IDisposable>();562563const DisposablesSnapshotStateContext = new RawContextKey<'started' | 'pending' | 'stopped'>('dirtyWorkingCopies', 'stopped');564565class StartTrackDisposables extends Action2 {566567constructor() {568super({569id: 'workbench.action.startTrackDisposables',570title: localize2('startTrackDisposables', 'Start Tracking Disposables'),571category: Categories.Developer,572f1: true,573precondition: ContextKeyExpr.and(DisposablesSnapshotStateContext.isEqualTo('pending').negate(), DisposablesSnapshotStateContext.isEqualTo('started').negate())574});575}576577run(accessor: ServicesAccessor): void {578const disposablesSnapshotStateContext = DisposablesSnapshotStateContext.bindTo(accessor.get(IContextKeyService));579disposablesSnapshotStateContext.set('started');580581trackedDisposables.clear();582583tracker = new DisposableTracker();584setDisposableTracker(tracker);585}586}587588class SnapshotTrackedDisposables extends Action2 {589590constructor() {591super({592id: 'workbench.action.snapshotTrackedDisposables',593title: localize2('snapshotTrackedDisposables', 'Snapshot Tracked Disposables'),594category: Categories.Developer,595f1: true,596precondition: DisposablesSnapshotStateContext.isEqualTo('started')597});598}599600run(accessor: ServicesAccessor): void {601const disposablesSnapshotStateContext = DisposablesSnapshotStateContext.bindTo(accessor.get(IContextKeyService));602disposablesSnapshotStateContext.set('pending');603604trackedDisposables = new Set(tracker?.computeLeakingDisposables(1000)?.leaks.map(disposable => disposable.value));605}606}607608class StopTrackDisposables extends Action2 {609610constructor() {611super({612id: 'workbench.action.stopTrackDisposables',613title: localize2('stopTrackDisposables', 'Stop Tracking Disposables'),614category: Categories.Developer,615f1: true,616precondition: DisposablesSnapshotStateContext.isEqualTo('pending')617});618}619620run(accessor: ServicesAccessor): void {621const editorService = accessor.get(IEditorService);622623const disposablesSnapshotStateContext = DisposablesSnapshotStateContext.bindTo(accessor.get(IContextKeyService));624disposablesSnapshotStateContext.set('stopped');625626if (tracker) {627const disposableLeaks = new Set<DisposableInfo>();628629for (const disposable of new Set(tracker.computeLeakingDisposables(1000)?.leaks) ?? []) {630if (trackedDisposables.has(disposable.value)) {631disposableLeaks.add(disposable);632}633}634635const leaks = tracker.computeLeakingDisposables(1000, Array.from(disposableLeaks));636if (leaks) {637editorService.openEditor({ resource: undefined, contents: leaks.details });638}639}640641setDisposableTracker(null);642tracker = undefined;643trackedDisposables.clear();644}645}646647// --- Actions Registration648registerAction2(InspectContextKeysAction);649registerAction2(ToggleScreencastModeAction);650registerAction2(LogStorageAction);651registerAction2(LogWorkingCopiesAction);652registerAction2(RemoveLargeStorageEntriesAction);653if (!product.commit) {654registerAction2(StartTrackDisposables);655registerAction2(SnapshotTrackedDisposables);656registerAction2(StopTrackDisposables);657}658659// --- Configuration660661// Screen Cast Mode662const configurationRegistry = Registry.as<IConfigurationRegistry>(ConfigurationExtensions.Configuration);663configurationRegistry.registerConfiguration({664id: 'screencastMode',665order: 9,666title: localize('screencastModeConfigurationTitle', "Screencast Mode"),667type: 'object',668properties: {669'screencastMode.verticalOffset': {670type: 'number',671default: 20,672minimum: 0,673maximum: 90,674description: localize('screencastMode.location.verticalPosition', "Controls the vertical offset of the screencast mode overlay from the bottom as a percentage of the workbench height.")675},676'screencastMode.fontSize': {677type: 'number',678default: 56,679minimum: 20,680maximum: 100,681description: localize('screencastMode.fontSize', "Controls the font size (in pixels) of the screencast mode keyboard.")682},683'screencastMode.keyboardOptions': {684type: 'object',685description: localize('screencastMode.keyboardOptions.description', "Options for customizing the keyboard overlay in screencast mode."),686properties: {687'showKeys': {688type: 'boolean',689default: true,690description: localize('screencastMode.keyboardOptions.showKeys', "Show raw keys.")691},692'showKeybindings': {693type: 'boolean',694default: true,695description: localize('screencastMode.keyboardOptions.showKeybindings', "Show keyboard shortcuts.")696},697'showCommands': {698type: 'boolean',699default: true,700description: localize('screencastMode.keyboardOptions.showCommands', "Show command names.")701},702'showCommandGroups': {703type: 'boolean',704default: false,705description: localize('screencastMode.keyboardOptions.showCommandGroups', "Show command group names, when commands are also shown.")706},707'showSingleEditorCursorMoves': {708type: 'boolean',709default: true,710description: localize('screencastMode.keyboardOptions.showSingleEditorCursorMoves', "Show single editor cursor move commands.")711}712},713default: {714'showKeys': true,715'showKeybindings': true,716'showCommands': true,717'showCommandGroups': false,718'showSingleEditorCursorMoves': true719},720additionalProperties: false721},722'screencastMode.keyboardOverlayTimeout': {723type: 'number',724default: 800,725minimum: 500,726maximum: 5000,727description: localize('screencastMode.keyboardOverlayTimeout', "Controls how long (in milliseconds) the keyboard overlay is shown in screencast mode.")728},729'screencastMode.mouseIndicatorColor': {730type: 'string',731format: 'color-hex',732default: '#FF0000',733description: localize('screencastMode.mouseIndicatorColor', "Controls the color in hex (#RGB, #RGBA, #RRGGBB or #RRGGBBAA) of the mouse indicator in screencast mode.")734},735'screencastMode.mouseIndicatorSize': {736type: 'number',737default: 20,738minimum: 20,739maximum: 100,740description: localize('screencastMode.mouseIndicatorSize', "Controls the size (in pixels) of the mouse indicator in screencast mode.")741},742}743});744745746