'use strict'; import { commands, Disposable } from 'vscode'; import { CommandContext, extensionId, setCommandContext } from './constants'; import { Logger } from './logger'; import { log } from './system/decorators/log'; export declare interface KeyCommand { onDidPressKey?(key: Keys): void | Promise; } const keyNoopCommand = Object.create(null) as KeyCommand; export { keyNoopCommand as KeyNoopCommand }; export const keys = [ 'left', 'alt+left', 'ctrl+left', 'right', 'alt+right', 'ctrl+right', 'alt+,', 'alt+.', 'escape' ] as const; export type Keys = typeof keys[number]; export type KeyMapping = { [K in Keys]?: KeyCommand | (() => Promise) }; type IndexableKeyMapping = KeyMapping & { [index: string]: KeyCommand | (() => Promise) | undefined; }; const mappings: KeyMapping[] = []; export class KeyboardScope implements Disposable { private readonly _mapping: IndexableKeyMapping; constructor(mapping: KeyMapping) { this._mapping = mapping; for (const key in this._mapping) { this._mapping[key] = this._mapping[key] || keyNoopCommand; } mappings.push(this._mapping); } @log({ args: false, prefix: context => `${context.prefix}[${mappings.length}]` }) async dispose() { const index = mappings.indexOf(this._mapping); const cc = Logger.getCorrelationContext(); if (cc) { cc.exitDetails = ` \u2022 index=${index}`; } if (index === mappings.length - 1) { mappings.pop(); await this.updateKeyCommandsContext(mappings[mappings.length - 1]); } else { mappings.splice(index, 1); } } private _paused = true; get paused() { return this._paused; } @log({ args: false, prefix: (context, key) => `${context.prefix}[${mappings.length}](${key})` }) async clearKeyCommand(key: Keys) { const cc = Logger.getCorrelationContext(); const mapping = mappings[mappings.length - 1]; if (mapping !== this._mapping || !mapping[key]) { if (cc) { cc.exitDetails = ' \u2022 skipped'; } return; } mapping[key] = undefined; await setCommandContext(`${CommandContext.Key}:${key}`, false); } @log({ args: false, prefix: context => `${context.prefix}(paused=${context.instance._paused})` }) async pause(keys?: Keys[]) { if (this._paused) return; this._paused = true; const mapping = (Object.keys(this._mapping) as Keys[]).reduce( (accumulator, key) => { accumulator[key] = keys === undefined ? false : keys.includes(key) ? false : this._mapping[key]; return accumulator; }, {} as any ); await this.updateKeyCommandsContext(mapping); } @log({ args: false, prefix: context => `${context.prefix}(paused=${context.instance._paused})` }) async resume() { if (!this._paused) return; this._paused = false; await this.updateKeyCommandsContext(this._mapping); } async start() { await this.resume(); } @log({ args: false, prefix: (context, key) => `${context.prefix}[${mappings.length}](${key})` }) async setKeyCommand(key: Keys, command: KeyCommand | (() => Promise)) { const cc = Logger.getCorrelationContext(); const mapping = mappings[mappings.length - 1]; if (mapping !== this._mapping) { if (cc) { cc.exitDetails = ' \u2022 skipped'; } return; } const set = Boolean(mapping[key]); mapping[key] = command; if (!set) { await setCommandContext(`${CommandContext.Key}:${key}`, true); } } private async updateKeyCommandsContext(mapping: KeyMapping) { await Promise.all( keys.map(key => setCommandContext(`${CommandContext.Key}:${key}`, Boolean(mapping && mapping[key]))) ); } } export class Keyboard implements Disposable { private _disposable: Disposable; constructor() { const subscriptions = keys.map(key => commands.registerCommand(`${extensionId}.key.${key}`, () => this.execute(key), this) ); this._disposable = Disposable.from(...subscriptions); } dispose() { this._disposable && this._disposable.dispose(); } @log({ args: false, prefix: (context, mapping) => `${context.prefix}[${mappings.length}](${mapping === undefined ? '' : Object.keys(mapping).join(',')})` }) createScope(mapping?: KeyMapping): KeyboardScope { return new KeyboardScope({ ...mapping }); } @log({ args: false, prefix: (context, mapping) => `${context.prefix}[${mappings.length}](${mapping === undefined ? '' : Object.keys(mapping).join(',')})` }) async beginScope(mapping?: KeyMapping): Promise { const scope = this.createScope(mapping); await scope.start(); return scope; } @log() async execute(key: Keys): Promise<{} | undefined> { const cc = Logger.getCorrelationContext(); if (!mappings.length) { if (cc) { cc.exitDetails = ' \u2022 skipped, no mappings'; } return undefined; } try { const mapping = mappings[mappings.length - 1]; let command = mapping[key] as KeyCommand | (() => Promise); if (typeof command === 'function') { command = await command(); } if (!command || typeof command.onDidPressKey !== 'function') { if (cc) { cc.exitDetails = ' \u2022 skipped, no callback'; } return undefined; } await command.onDidPressKey(key); return undefined; } catch (ex) { Logger.error(ex, cc); return undefined; } } }