From 7c1f2123e95b3d8419298577013cae47c5fcd7fc Mon Sep 17 00:00:00 2001 From: Robert Knight Date: Tue, 23 Apr 2024 14:22:42 +0100 Subject: [PATCH] Centralize keyboard shortcuts --- package-lock.json | 10 ++++ package.json | 1 + src/common/GenericDialog.tsx | 3 +- src/common/InputDialog.tsx | 3 +- src/common/PostSaveDialog.tsx | 3 +- src/common/keyboard-shortcuts.ts | 14 ++++++ src/documentation/api/ApiNode.tsx | 21 +++------ src/documentation/common/CodeEmbed.tsx | 24 ++++------ src/project/SaveButton.tsx | 13 ++++- src/project/SendButton.tsx | 10 ++++ src/project/project-actions.tsx | 14 +++--- src/workbench/SideBarHeader.tsx | 47 ++++++++----------- .../connect-dialogs/TransferHexDialog.tsx | 3 +- 13 files changed, 95 insertions(+), 71 deletions(-) create mode 100644 src/common/keyboard-shortcuts.ts diff --git a/package-lock.json b/package-lock.json index 862699247..093933d50 100644 --- a/package-lock.json +++ b/package-lock.json @@ -50,6 +50,7 @@ "mobile-drag-drop": "^2.3.0-rc.2", "react": "^18.0.0", "react-dom": "^18.0.0", + "react-hotkeys-hook": "^4.5.0", "react-icons": "^4.8.0", "react-intl": "^6.2.10", "vite": "^5.1.5", @@ -8808,6 +8809,15 @@ } } }, + "node_modules/react-hotkeys-hook": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/react-hotkeys-hook/-/react-hotkeys-hook-4.5.0.tgz", + "integrity": "sha512-Samb85GSgAWFQNvVt3PS90LPPGSf9mkH/r4au81ZP1yOIFayLC3QAvqTgGtJ8YEDMXtPmaVBs6NgipHO6h4Mug==", + "peerDependencies": { + "react": ">=16.8.1", + "react-dom": ">=16.8.1" + } + }, "node_modules/react-icons": { "version": "4.12.0", "resolved": "https://registry.npmjs.org/react-icons/-/react-icons-4.12.0.tgz", diff --git a/package.json b/package.json index c4df05d20..6e0eff4f0 100644 --- a/package.json +++ b/package.json @@ -46,6 +46,7 @@ "mobile-drag-drop": "^2.3.0-rc.2", "react": "^18.0.0", "react-dom": "^18.0.0", + "react-hotkeys-hook": "^4.5.0", "react-icons": "^4.8.0", "react-intl": "^6.2.10", "vite": "^5.1.5", diff --git a/src/common/GenericDialog.tsx b/src/common/GenericDialog.tsx index c1e92a1d5..ba7e54790 100644 --- a/src/common/GenericDialog.tsx +++ b/src/common/GenericDialog.tsx @@ -16,6 +16,7 @@ import { ThemingProps } from "@chakra-ui/styled-system"; import { ReactNode } from "react"; import { FormattedMessage } from "react-intl"; import ModalCloseButton from "./ModalCloseButton"; +import { FinalFocusRef } from "../project/project-actions"; export interface GenericDialogProps { header?: ReactNode; @@ -24,7 +25,7 @@ export interface GenericDialogProps { size?: ThemingProps<"Button">["size"]; onClose: () => void; returnFocusOnClose?: boolean; - finalFocusRef?: React.RefObject; + finalFocusRef?: FinalFocusRef; } export const GenericDialog = ({ diff --git a/src/common/InputDialog.tsx b/src/common/InputDialog.tsx index ca258d13b..8d13f8e45 100644 --- a/src/common/InputDialog.tsx +++ b/src/common/InputDialog.tsx @@ -16,6 +16,7 @@ import { import { ThemeTypings } from "@chakra-ui/styled-system"; import { ReactNode, useCallback, useState } from "react"; import { FormattedMessage } from "react-intl"; +import { FinalFocusRef } from "../project/project-actions"; export interface InputValidationResult { ok: boolean; @@ -39,7 +40,7 @@ export interface InputDialogProps { actionLabel: string; size?: ThemeTypings["components"]["Modal"]["sizes"]; validate?: (input: T) => InputValidationResult; - finalFocusRef?: React.RefObject; + finalFocusRef?: FinalFocusRef; callback: (value: ValueOrCancelled) => void; } diff --git a/src/common/PostSaveDialog.tsx b/src/common/PostSaveDialog.tsx index 7a6d76a0f..5fdbcc14e 100644 --- a/src/common/PostSaveDialog.tsx +++ b/src/common/PostSaveDialog.tsx @@ -8,6 +8,7 @@ import { ReactNode, useCallback } from "react"; import { FormattedMessage } from "react-intl"; import { GenericDialog, GenericDialogFooter } from "../common/GenericDialog"; import { useProject } from "../project/project-hooks"; +import { FinalFocusRef } from "../project/project-actions"; export const enum PostSaveChoice { ShowTransferHexHelp, @@ -18,7 +19,7 @@ export const enum PostSaveChoice { interface PostSaveDialogProps { callback: (value: PostSaveChoice) => void; dialogNormallyHidden: boolean; - finalFocusRef: React.RefObject; + finalFocusRef: FinalFocusRef; } export const PostSaveDialog = ({ diff --git a/src/common/keyboard-shortcuts.ts b/src/common/keyboard-shortcuts.ts new file mode 100644 index 000000000..dbc213970 --- /dev/null +++ b/src/common/keyboard-shortcuts.ts @@ -0,0 +1,14 @@ +// Shortcuts are global unless noted otherwise. +export const keyboardShortcuts = { + // This is scoped by keyboard focus. + copyCode: ["ctrl+c", "meta+c", "enter"], + search: ["ctrl+shift+f", "meta+shift+f"], + sendToMicrobit: ["ctrl+shift+e", "meta+shift+e"], + saveProject: ["ctrl+shift+s", "meta+shift+s"], +}; + +export const globalShortcutConfig = { + preventDefault: true, + enableOnContentEditable: true, + enableOnFormTags: true, +}; diff --git a/src/documentation/api/ApiNode.tsx b/src/documentation/api/ApiNode.tsx index d245123d2..9c1c6eae2 100644 --- a/src/documentation/api/ApiNode.tsx +++ b/src/documentation/api/ApiNode.tsx @@ -41,6 +41,8 @@ import ShowMoreButton from "../common/ShowMoreButton"; import { allowWrapAtPeriods } from "../common/wrap"; import { useCodeDragImage } from "../documentation-hooks"; import Highlight from "../reference/Highlight"; +import { useHotkeys } from "react-hotkeys-hook"; +import { keyboardShortcuts } from "../../common/keyboard-shortcuts"; const kindToFontSize: Record = { module: "2xl", @@ -422,20 +424,9 @@ const DraggableSignature = ({ onCopy(); await actions?.copyCode(code, codeWithImports, type, id); }, [actions, code, codeWithImports, onCopy, type, id]); - const isMac = /Mac/.test(navigator.platform); - const handleKeyDown = useCallback( - async (e: React.KeyboardEvent) => { - if (e.key === "Enter") { - e.preventDefault(); - handleCopyCode(); - } - if ((e.key === "c" || e.key === "C") && (isMac ? e.metaKey : e.ctrlKey)) { - e.preventDefault(); - handleCopyCode(); - } - }, - [handleCopyCode, isMac] - ); + const hotKeysRef = useHotkeys(keyboardShortcuts.copyCode, handleCopyCode, { + preventDefault: true, + }); const intl = useIntl(); const [{ dragDropSuccess }] = useSessionSettings(); return ( @@ -448,6 +439,7 @@ const DraggableSignature = ({ isDisabled={dragDropSuccess} > diff --git a/src/documentation/common/CodeEmbed.tsx b/src/documentation/common/CodeEmbed.tsx index 531cbe042..90db24da8 100644 --- a/src/documentation/common/CodeEmbed.tsx +++ b/src/documentation/common/CodeEmbed.tsx @@ -13,6 +13,7 @@ import { } from "@chakra-ui/react"; import { forwardRef } from "@chakra-ui/system"; import React, { + LegacyRef, Ref, useCallback, useEffect, @@ -33,6 +34,8 @@ import { useSessionSettings } from "../../settings/session-settings"; import DragHandle from "../common/DragHandle"; import { useCodeDragImage } from "../documentation-hooks"; import CodeActionButton from "./CodeActionButton"; +import { useHotkeys } from "react-hotkeys-hook"; +import { keyboardShortcuts } from "../../common/keyboard-shortcuts"; interface CodeEmbedProps { code: string; @@ -132,20 +135,9 @@ const CodeEmbed = ({ const textHeight = lineCount * 1.375 + "em"; const codeHeight = `calc(${textHeight} + var(--chakra-space-2) + var(--chakra-space-2))`; const codePopUpHeight = `calc(${codeHeight} + 2px)`; // Account for border. - const isMac = /Mac/.test(navigator.platform); - const handleKeyDown = useCallback( - async (e: React.KeyboardEvent) => { - if (e.key === "Enter") { - e.preventDefault(); - handleCopyCode(); - } - if ((e.key === "c" || e.key === "C") && (isMac ? e.metaKey : e.ctrlKey)) { - e.preventDefault(); - handleCopyCode(); - } - }, - [handleCopyCode, isMac] - ); + const hotKeysRef = useHotkeys(keyboardShortcuts.copyCode, handleCopyCode, { + preventDefault: true, + }) as LegacyRef; const determineBackground = () => { if ( (toolkitType === "ideas" && state === "highlighted") || @@ -157,7 +149,7 @@ const CodeEmbed = ({ }; return ( - + {state === "raised" && ( diff --git a/src/project/SaveButton.tsx b/src/project/SaveButton.tsx index 4d6528ee5..d9b4c964a 100644 --- a/src/project/SaveButton.tsx +++ b/src/project/SaveButton.tsx @@ -4,13 +4,18 @@ * SPDX-License-Identifier: MIT */ import { Tooltip } from "@chakra-ui/react"; -import { useRef } from "react"; +import { useCallback, useRef } from "react"; import { RiDownload2Line } from "react-icons/ri"; import { useIntl } from "react-intl"; import CollapsibleButton, { CollapsibleButtonProps, } from "../common/CollapsibleButton"; import { useProjectActions } from "./project-hooks"; +import { useHotkeys } from "react-hotkeys-hook"; +import { + globalShortcutConfig, + keyboardShortcuts, +} from "../common/keyboard-shortcuts"; interface SaveButtonProps extends Omit {} @@ -27,6 +32,12 @@ const SaveButton = (props: SaveButtonProps) => { const actions = useProjectActions(); const intl = useIntl(); const menuButtonRef = useRef(null); + const ref = useRef(null); + const handleSave = useCallback(() => { + ref.current = document.activeElement as HTMLElement; + actions.save(ref); + }, [actions]); + useHotkeys(keyboardShortcuts.saveProject, handleSave, globalShortcutConfig); return ( (null); + useHotkeys( + keyboardShortcuts.sendToMicrobit, + handleSendToMicrobit, + globalShortcutConfig + ); return ( diff --git a/src/project/project-actions.tsx b/src/project/project-actions.tsx index 31c8346a4..6f7edc7a8 100644 --- a/src/project/project-actions.tsx +++ b/src/project/project-actions.tsx @@ -76,6 +76,8 @@ import ProjectNameQuestion from "./ProjectNameQuestion"; */ export type LoadType = "drop-load" | "file-upload"; +export type FinalFocusRef = React.RefObject; + export interface MainScriptChoice { main: string | undefined; } @@ -547,7 +549,7 @@ export class ProjectActions { * Trigger a browser download with a universal hex file. */ save = async ( - finalFocusRef: React.RefObject, + finalFocusRef: FinalFocusRef, saveViaWebUsbNotSupported?: boolean ) => { this.logging.event({ @@ -728,7 +730,7 @@ export class ProjectActions { isDefaultProjectName = (): boolean => this.fs.project.name === undefined; ensureProjectName = async ( - finalFocusRef: React.RefObject + finalFocusRef: FinalFocusRef ): Promise => { if (this.isDefaultProjectName()) { return await this.editProjectName(true, finalFocusRef); @@ -738,7 +740,7 @@ export class ProjectActions { editProjectName = async ( isSave: boolean = false, - finalFocusRef?: React.RefObject + finalFocusRef?: FinalFocusRef ) => { const name = await this.dialogs.show((callback) => ( - ) { + private async handlePostSaveDialog(finalFocusRef: FinalFocusRef) { const showPostSaveHelpSetting = this.settings.values.showPostSaveHelp; if (!showPostSaveHelpSetting) { return; @@ -966,7 +966,7 @@ export class ProjectActions { private async handleTransferHexDialog( forceTransferHexHelp: boolean, - finalFocusRef: React.RefObject + finalFocusRef: FinalFocusRef ) { const showTransferHexHelpSetting = this.settings.values.showTransferHexHelp; if (!forceTransferHexHelp && !showTransferHexHelpSetting) { diff --git a/src/workbench/SideBarHeader.tsx b/src/workbench/SideBarHeader.tsx index a3e812df2..511059111 100644 --- a/src/workbench/SideBarHeader.tsx +++ b/src/workbench/SideBarHeader.tsx @@ -33,6 +33,11 @@ import SearchDialog from "../documentation/search/SearchDialog"; import { useLogging } from "../logging/logging-hooks"; import { RouterState, useRouterState } from "../router-hooks"; import { useSettings } from "../settings/settings"; +import { useHotkeys } from "react-hotkeys-hook"; +import { + globalShortcutConfig, + keyboardShortcuts, +} from "../common/keyboard-shortcuts"; interface SideBarHeaderProps { sidebarShown: boolean; @@ -72,35 +77,21 @@ const SideBarHeader = ({ const [{ languageId }] = useSettings(); const searchAvailable = supportedSearchLanguages.includes(languageId); - // When we add more keyboard shortcuts, we should pull this up and have a CM-like model of the - // available actions and their shortcuts, with a hook used here to register a handler for the action. - useEffect(() => { - const isMac = /Mac/.test(navigator.platform); - const keydown = (e: KeyboardEvent) => { - if ( - (e.key === "F" || e.key === "f") && - (isMac ? e.metaKey : e.ctrlKey) && - e.shiftKey && - !e.repeat && - searchAvailable - ) { - handleModalOpened(); - if (!sidebarShown) { - onSidebarToggled(); - } + + const handleSearchShortcut = useCallback(() => { + if (searchAvailable) { + handleModalOpened(); + if (!sidebarShown) { + onSidebarToggled(); } - }; - document.addEventListener("keydown", keydown); - return () => { - document.removeEventListener("keydown", keydown); - }; - }, [ - onSidebarToggled, - searchModal, - sidebarShown, - handleModalOpened, - searchAvailable, - ]); + } + }, [handleModalOpened, onSidebarToggled, searchAvailable, sidebarShown]); + + useHotkeys( + keyboardShortcuts.search, + handleSearchShortcut, + globalShortcutConfig + ); const handleQueryChange: React.ChangeEventHandler = useCallback( diff --git a/src/workbench/connect-dialogs/TransferHexDialog.tsx b/src/workbench/connect-dialogs/TransferHexDialog.tsx index 779bc97ed..6abd79107 100644 --- a/src/workbench/connect-dialogs/TransferHexDialog.tsx +++ b/src/workbench/connect-dialogs/TransferHexDialog.tsx @@ -10,6 +10,7 @@ import { FormattedMessage } from "react-intl"; import { GenericDialog, GenericDialogFooter } from "../../common/GenericDialog"; import transferHexMac from "./transfer-hex-mac.gif"; import transferHexWin from "./transfer-hex-win.gif"; +import { FinalFocusRef } from "../../project/project-actions"; export const enum TransferHexChoice { CloseDontShowAgain, @@ -20,7 +21,7 @@ interface TransferHexDialogProps { callback: (value: TransferHexChoice) => void; dialogNormallyHidden: boolean; shownByRequest: boolean; - finalFocusRef: React.RefObject; + finalFocusRef: FinalFocusRef; } export const TransferHexDialog = ({