2024-08-20 03:28:28 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-09-05 09:21:08 +02:00
|
|
|
import { atoms, createBlock, getApi, getBlockComponentModel, globalStore, refocusNode, WOS } from "@/app/store/global";
|
2024-08-30 01:06:15 +02:00
|
|
|
import * as services from "@/app/store/services";
|
2024-08-26 20:56:00 +02:00
|
|
|
import {
|
|
|
|
deleteLayoutModelForTab,
|
|
|
|
getLayoutModelForActiveTab,
|
|
|
|
getLayoutModelForTab,
|
|
|
|
getLayoutModelForTabById,
|
|
|
|
NavigateDirection,
|
|
|
|
} from "@/layout/index";
|
2024-08-20 03:28:28 +02:00
|
|
|
import * as keyutil from "@/util/keyutil";
|
|
|
|
import * as jotai from "jotai";
|
|
|
|
|
|
|
|
const simpleControlShiftAtom = jotai.atom(false);
|
2024-08-30 01:06:15 +02:00
|
|
|
const globalKeyMap = new Map<string, (waveEvent: WaveKeyboardEvent) => boolean>();
|
|
|
|
|
2024-10-03 02:18:16 +02:00
|
|
|
function getFocusedBlockInActiveTab() {
|
|
|
|
const activeTabId = globalStore.get(atoms.activeTabId);
|
|
|
|
const layoutModel = getLayoutModelForTabById(activeTabId);
|
|
|
|
const focusedNode = globalStore.get(layoutModel.focusedNode);
|
|
|
|
return focusedNode.data?.blockId;
|
|
|
|
}
|
|
|
|
|
2024-08-30 01:06:15 +02:00
|
|
|
function getSimpleControlShiftAtom() {
|
|
|
|
return simpleControlShiftAtom;
|
|
|
|
}
|
2024-08-20 03:28:28 +02:00
|
|
|
|
|
|
|
function setControlShift() {
|
|
|
|
globalStore.set(simpleControlShiftAtom, true);
|
|
|
|
setTimeout(() => {
|
|
|
|
const simpleState = globalStore.get(simpleControlShiftAtom);
|
|
|
|
if (simpleState) {
|
|
|
|
globalStore.set(atoms.controlShiftDelayAtom, true);
|
|
|
|
}
|
|
|
|
}, 400);
|
|
|
|
}
|
|
|
|
|
|
|
|
function unsetControlShift() {
|
|
|
|
globalStore.set(simpleControlShiftAtom, false);
|
|
|
|
globalStore.set(atoms.controlShiftDelayAtom, false);
|
|
|
|
}
|
|
|
|
|
2024-09-03 01:48:10 +02:00
|
|
|
function shouldDispatchToBlock(e: WaveKeyboardEvent): boolean {
|
2024-08-30 01:06:15 +02:00
|
|
|
if (globalStore.get(atoms.modalOpen)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const activeElem = document.activeElement;
|
|
|
|
if (activeElem != null && activeElem instanceof HTMLElement) {
|
2024-09-03 01:48:10 +02:00
|
|
|
if (activeElem.tagName == "INPUT" || activeElem.tagName == "TEXTAREA" || activeElem.contentEditable == "true") {
|
2024-08-30 02:00:24 +02:00
|
|
|
if (activeElem.classList.contains("dummy-focus")) {
|
|
|
|
return true;
|
|
|
|
}
|
2024-09-03 01:48:10 +02:00
|
|
|
if (keyutil.isInputEvent(e)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-08-20 03:28:28 +02:00
|
|
|
function genericClose(tabId: string) {
|
|
|
|
const tabORef = WOS.makeORef("tab", tabId);
|
|
|
|
const tabAtom = WOS.getWaveObjectAtom<Tab>(tabORef);
|
|
|
|
const tabData = globalStore.get(tabAtom);
|
|
|
|
if (tabData == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (tabData.blockids == null || tabData.blockids.length == 0) {
|
|
|
|
// close tab
|
|
|
|
services.WindowService.CloseTab(tabId);
|
|
|
|
deleteLayoutModelForTab(tabId);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const layoutModel = getLayoutModelForTab(tabAtom);
|
2024-08-26 20:56:00 +02:00
|
|
|
layoutModel.closeFocusedNode();
|
2024-08-20 03:28:28 +02:00
|
|
|
}
|
|
|
|
|
2024-08-26 20:56:00 +02:00
|
|
|
function switchBlockByBlockNum(index: number) {
|
|
|
|
const layoutModel = getLayoutModelForActiveTab();
|
2024-08-22 02:43:11 +02:00
|
|
|
if (!layoutModel) {
|
2024-08-20 03:28:28 +02:00
|
|
|
return;
|
|
|
|
}
|
2024-08-26 20:56:00 +02:00
|
|
|
layoutModel.switchNodeFocusByBlockNum(index);
|
2024-08-20 03:28:28 +02:00
|
|
|
}
|
|
|
|
|
2024-08-26 20:56:00 +02:00
|
|
|
function switchBlockInDirection(tabId: string, direction: NavigateDirection) {
|
|
|
|
const layoutModel = getLayoutModelForTabById(tabId);
|
|
|
|
layoutModel.switchNodeFocusInDirection(direction);
|
2024-08-20 03:28:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function switchTabAbs(index: number) {
|
|
|
|
const ws = globalStore.get(atoms.workspace);
|
|
|
|
const newTabIdx = index - 1;
|
|
|
|
if (newTabIdx < 0 || newTabIdx >= ws.tabids.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const newActiveTabId = ws.tabids[newTabIdx];
|
|
|
|
services.ObjectService.SetActiveTab(newActiveTabId);
|
|
|
|
}
|
|
|
|
|
|
|
|
function switchTab(offset: number) {
|
|
|
|
const ws = globalStore.get(atoms.workspace);
|
|
|
|
const activeTabId = globalStore.get(atoms.tabAtom).oid;
|
|
|
|
let tabIdx = -1;
|
|
|
|
for (let i = 0; i < ws.tabids.length; i++) {
|
|
|
|
if (ws.tabids[i] == activeTabId) {
|
|
|
|
tabIdx = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tabIdx == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const newTabIdx = (tabIdx + offset + ws.tabids.length) % ws.tabids.length;
|
|
|
|
const newActiveTabId = ws.tabids[newTabIdx];
|
|
|
|
services.ObjectService.SetActiveTab(newActiveTabId);
|
|
|
|
}
|
|
|
|
|
2024-09-03 05:21:35 +02:00
|
|
|
function handleCmdI() {
|
|
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
|
|
const focusedNode = globalStore.get(layoutModel.focusedNode);
|
|
|
|
if (focusedNode == null) {
|
|
|
|
// focus a node
|
|
|
|
layoutModel.focusFirstNode();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const blockId = focusedNode?.data?.blockId;
|
|
|
|
if (blockId == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
refocusNode(blockId);
|
|
|
|
}
|
|
|
|
|
2024-08-26 20:56:00 +02:00
|
|
|
async function handleCmdN() {
|
2024-08-20 03:28:28 +02:00
|
|
|
const termBlockDef: BlockDef = {
|
|
|
|
meta: {
|
|
|
|
view: "term",
|
|
|
|
controller: "shell",
|
|
|
|
},
|
|
|
|
};
|
2024-08-26 20:56:00 +02:00
|
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
|
|
const focusedNode = globalStore.get(layoutModel.focusedNode);
|
|
|
|
if (focusedNode != null) {
|
|
|
|
const blockAtom = WOS.getWaveObjectAtom<Block>(WOS.makeORef("block", focusedNode.data?.blockId));
|
2024-08-20 03:28:28 +02:00
|
|
|
const blockData = globalStore.get(blockAtom);
|
|
|
|
if (blockData?.meta?.view == "term") {
|
|
|
|
if (blockData?.meta?.["cmd:cwd"] != null) {
|
|
|
|
termBlockDef.meta["cmd:cwd"] = blockData.meta["cmd:cwd"];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (blockData?.meta?.connection != null) {
|
|
|
|
termBlockDef.meta.connection = blockData.meta.connection;
|
|
|
|
}
|
|
|
|
}
|
2024-08-26 20:56:00 +02:00
|
|
|
await createBlock(termBlockDef);
|
2024-08-22 00:49:23 +02:00
|
|
|
}
|
|
|
|
|
2024-08-20 03:28:28 +02:00
|
|
|
function appHandleKeyDown(waveEvent: WaveKeyboardEvent): boolean {
|
2024-08-30 02:00:24 +02:00
|
|
|
const handled = handleGlobalWaveKeyboardEvents(waveEvent);
|
|
|
|
if (handled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
|
|
const focusedNode = globalStore.get(layoutModel.focusedNode);
|
|
|
|
const blockId = focusedNode?.data?.blockId;
|
2024-09-03 01:48:10 +02:00
|
|
|
if (blockId != null && shouldDispatchToBlock(waveEvent)) {
|
2024-09-05 09:21:08 +02:00
|
|
|
const bcm = getBlockComponentModel(blockId);
|
|
|
|
const viewModel = bcm?.viewModel;
|
2024-08-30 02:00:24 +02:00
|
|
|
if (viewModel?.keyDownHandler) {
|
|
|
|
const handledByBlock = viewModel.keyDownHandler(waveEvent);
|
|
|
|
if (handledByBlock) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2024-08-30 01:06:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function registerControlShiftStateUpdateHandler() {
|
|
|
|
getApi().onControlShiftStateUpdate((state: boolean) => {
|
|
|
|
if (state) {
|
2024-08-20 03:28:28 +02:00
|
|
|
setControlShift();
|
|
|
|
} else {
|
|
|
|
unsetControlShift();
|
|
|
|
}
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function registerElectronReinjectKeyHandler() {
|
|
|
|
getApi().onReinjectKey((event: WaveKeyboardEvent) => {
|
2024-08-30 02:00:24 +02:00
|
|
|
appHandleKeyDown(event);
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
}
|
2024-08-20 03:28:28 +02:00
|
|
|
|
2024-09-03 01:48:10 +02:00
|
|
|
function tryReinjectKey(event: WaveKeyboardEvent): boolean {
|
|
|
|
return appHandleKeyDown(event);
|
|
|
|
}
|
|
|
|
|
2024-08-30 01:06:15 +02:00
|
|
|
function registerGlobalKeys() {
|
|
|
|
globalKeyMap.set("Cmd:]", () => {
|
2024-08-20 03:28:28 +02:00
|
|
|
switchTab(1);
|
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Shift:Cmd:]", () => {
|
|
|
|
switchTab(1);
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
globalKeyMap.set("Cmd:[", () => {
|
2024-08-20 03:28:28 +02:00
|
|
|
switchTab(-1);
|
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Shift:Cmd:[", () => {
|
|
|
|
switchTab(-1);
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
globalKeyMap.set("Cmd:n", () => {
|
2024-08-26 20:56:00 +02:00
|
|
|
handleCmdN();
|
2024-08-22 00:49:23 +02:00
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Cmd:i", () => {
|
2024-09-03 05:21:35 +02:00
|
|
|
handleCmdI();
|
2024-08-30 01:06:15 +02:00
|
|
|
return true;
|
|
|
|
});
|
|
|
|
globalKeyMap.set("Cmd:t", () => {
|
2024-08-20 07:39:52 +02:00
|
|
|
const workspace = globalStore.get(atoms.workspace);
|
|
|
|
const newTabName = `T${workspace.tabids.length + 1}`;
|
|
|
|
services.ObjectService.AddTabToWorkspace(newTabName, true);
|
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Cmd:w", () => {
|
|
|
|
const tabId = globalStore.get(atoms.activeTabId);
|
|
|
|
genericClose(tabId);
|
|
|
|
return true;
|
|
|
|
});
|
2024-08-30 02:00:24 +02:00
|
|
|
globalKeyMap.set("Cmd:m", () => {
|
|
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
|
|
const focusedNode = globalStore.get(layoutModel.focusedNode);
|
|
|
|
if (focusedNode != null) {
|
|
|
|
layoutModel.magnifyNodeToggle(focusedNode.id);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
2024-08-30 01:06:15 +02:00
|
|
|
globalKeyMap.set("Ctrl:Shift:ArrowUp", () => {
|
|
|
|
const tabId = globalStore.get(atoms.activeTabId);
|
2024-08-26 20:56:00 +02:00
|
|
|
switchBlockInDirection(tabId, NavigateDirection.Up);
|
2024-08-20 03:28:28 +02:00
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Ctrl:Shift:ArrowDown", () => {
|
|
|
|
const tabId = globalStore.get(atoms.activeTabId);
|
2024-08-26 20:56:00 +02:00
|
|
|
switchBlockInDirection(tabId, NavigateDirection.Down);
|
2024-08-20 03:28:28 +02:00
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Ctrl:Shift:ArrowLeft", () => {
|
|
|
|
const tabId = globalStore.get(atoms.activeTabId);
|
2024-08-26 20:56:00 +02:00
|
|
|
switchBlockInDirection(tabId, NavigateDirection.Left);
|
2024-08-20 03:28:28 +02:00
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
|
|
|
globalKeyMap.set("Ctrl:Shift:ArrowRight", () => {
|
|
|
|
const tabId = globalStore.get(atoms.activeTabId);
|
2024-08-26 20:56:00 +02:00
|
|
|
switchBlockInDirection(tabId, NavigateDirection.Right);
|
2024-08-20 03:28:28 +02:00
|
|
|
return true;
|
2024-08-30 01:06:15 +02:00
|
|
|
});
|
2024-10-03 02:18:16 +02:00
|
|
|
globalKeyMap.set("Cmd:g", () => {
|
|
|
|
const bcm = getBlockComponentModel(getFocusedBlockInActiveTab());
|
|
|
|
if (bcm.openSwitchConnection != null) {
|
|
|
|
bcm.openSwitchConnection();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
2024-08-30 01:06:15 +02:00
|
|
|
for (let idx = 1; idx <= 9; idx++) {
|
|
|
|
globalKeyMap.set(`Cmd:${idx}`, () => {
|
|
|
|
switchTabAbs(idx);
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
globalKeyMap.set(`Ctrl:Shift:c{Digit${idx}}`, () => {
|
|
|
|
switchBlockByBlockNum(idx);
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
globalKeyMap.set(`Ctrl:Shift:c{Numpad${idx}}`, () => {
|
|
|
|
switchBlockByBlockNum(idx);
|
|
|
|
return true;
|
|
|
|
});
|
2024-08-20 03:28:28 +02:00
|
|
|
}
|
2024-08-30 01:06:15 +02:00
|
|
|
const allKeys = Array.from(globalKeyMap.keys());
|
|
|
|
// special case keys, handled by web view
|
|
|
|
allKeys.push("Cmd:l", "Cmd:r", "Cmd:ArrowRight", "Cmd:ArrowLeft");
|
|
|
|
getApi().registerGlobalWebviewKeys(allKeys);
|
|
|
|
}
|
|
|
|
|
2024-10-01 23:07:28 +02:00
|
|
|
function getAllGlobalKeyBindings(): string[] {
|
|
|
|
const allKeys = Array.from(globalKeyMap.keys());
|
|
|
|
return allKeys;
|
|
|
|
}
|
|
|
|
|
2024-08-30 01:06:15 +02:00
|
|
|
// these keyboard events happen *anywhere*, even if you have focus in an input or somewhere else.
|
|
|
|
function handleGlobalWaveKeyboardEvents(waveEvent: WaveKeyboardEvent): boolean {
|
|
|
|
for (const key of globalKeyMap.keys()) {
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, key)) {
|
|
|
|
const handler = globalKeyMap.get(key);
|
|
|
|
if (handler == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return handler(waveEvent);
|
|
|
|
}
|
2024-08-20 03:28:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-30 01:06:15 +02:00
|
|
|
export {
|
|
|
|
appHandleKeyDown,
|
2024-10-01 23:07:28 +02:00
|
|
|
getAllGlobalKeyBindings,
|
2024-08-30 01:06:15 +02:00
|
|
|
getSimpleControlShiftAtom,
|
|
|
|
registerControlShiftStateUpdateHandler,
|
|
|
|
registerElectronReinjectKeyHandler,
|
|
|
|
registerGlobalKeys,
|
2024-09-03 01:48:10 +02:00
|
|
|
tryReinjectKey,
|
2024-08-30 01:06:15 +02:00
|
|
|
unsetControlShift,
|
|
|
|
};
|