2024-08-20 21:54:53 +08:00
|
|
|
import {$getSelection, COMMAND_PRIORITY_HIGH, FORMAT_TEXT_COMMAND, KEY_ENTER_COMMAND, LexicalEditor} from "lexical";
|
2024-08-20 20:07:33 +08:00
|
|
|
import {
|
|
|
|
cycleSelectionCalloutFormats,
|
2024-08-20 21:54:53 +08:00
|
|
|
formatCodeBlock, insertOrUpdateLink,
|
2024-08-20 20:07:33 +08:00
|
|
|
toggleSelectionAsBlockquote,
|
2024-08-20 21:54:53 +08:00
|
|
|
toggleSelectionAsHeading, toggleSelectionAsList,
|
2024-08-20 20:07:33 +08:00
|
|
|
toggleSelectionAsParagraph
|
|
|
|
} from "../utils/formats";
|
|
|
|
import {HeadingTagType} from "@lexical/rich-text";
|
2024-08-20 21:54:53 +08:00
|
|
|
import {EditorUiContext} from "../ui/framework/core";
|
|
|
|
import {$getNodeFromSelection} from "../utils/selection";
|
|
|
|
import {$isLinkNode, LinkNode} from "@lexical/link";
|
|
|
|
import {$showLinkForm} from "../ui/defaults/forms/objects";
|
|
|
|
import {showLinkSelector} from "../utils/links";
|
2024-08-20 20:07:33 +08:00
|
|
|
|
|
|
|
function headerHandler(editor: LexicalEditor, tag: HeadingTagType): boolean {
|
|
|
|
toggleSelectionAsHeading(editor, tag);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function wrapFormatAction(formatAction: (editor: LexicalEditor) => any): ShortcutAction {
|
|
|
|
return (editor: LexicalEditor) => {
|
|
|
|
formatAction(editor);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function toggleInlineCode(editor: LexicalEditor): boolean {
|
|
|
|
editor.dispatchCommand(FORMAT_TEXT_COMMAND, 'code');
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-08-20 21:54:53 +08:00
|
|
|
type ShortcutAction = (editor: LexicalEditor, context: EditorUiContext) => boolean;
|
2024-08-20 20:07:33 +08:00
|
|
|
|
2024-10-04 19:41:13 +08:00
|
|
|
/**
|
|
|
|
* List of action functions by their shortcut combo.
|
|
|
|
* We use "meta" as an abstraction for ctrl/cmd depending on platform.
|
|
|
|
*/
|
2024-08-20 20:07:33 +08:00
|
|
|
const actionsByKeys: Record<string, ShortcutAction> = {
|
2024-10-04 19:41:13 +08:00
|
|
|
'meta+s': () => {
|
2024-08-20 20:07:33 +08:00
|
|
|
window.$events.emit('editor-save-draft');
|
|
|
|
return true;
|
|
|
|
},
|
2024-10-04 19:41:13 +08:00
|
|
|
'meta+enter': () => {
|
2024-08-20 20:07:33 +08:00
|
|
|
window.$events.emit('editor-save-page');
|
|
|
|
return true;
|
|
|
|
},
|
2024-10-04 19:41:13 +08:00
|
|
|
'meta+1': (editor) => headerHandler(editor, 'h1'),
|
|
|
|
'meta+2': (editor) => headerHandler(editor, 'h2'),
|
|
|
|
'meta+3': (editor) => headerHandler(editor, 'h3'),
|
|
|
|
'meta+4': (editor) => headerHandler(editor, 'h4'),
|
|
|
|
'meta+5': wrapFormatAction(toggleSelectionAsParagraph),
|
|
|
|
'meta+d': wrapFormatAction(toggleSelectionAsParagraph),
|
|
|
|
'meta+6': wrapFormatAction(toggleSelectionAsBlockquote),
|
|
|
|
'meta+q': wrapFormatAction(toggleSelectionAsBlockquote),
|
|
|
|
'meta+7': wrapFormatAction(formatCodeBlock),
|
|
|
|
'meta+e': wrapFormatAction(formatCodeBlock),
|
|
|
|
'meta+8': toggleInlineCode,
|
|
|
|
'meta+shift+e': toggleInlineCode,
|
|
|
|
'meta+9': wrapFormatAction(cycleSelectionCalloutFormats),
|
2024-08-20 20:07:33 +08:00
|
|
|
|
2024-10-04 19:41:13 +08:00
|
|
|
'meta+o': wrapFormatAction((e) => toggleSelectionAsList(e, 'number')),
|
|
|
|
'meta+p': wrapFormatAction((e) => toggleSelectionAsList(e, 'bullet')),
|
|
|
|
'meta+k': (editor, context) => {
|
2024-08-20 21:54:53 +08:00
|
|
|
editor.getEditorState().read(() => {
|
|
|
|
const selectedLink = $getNodeFromSelection($getSelection(), $isLinkNode) as LinkNode | null;
|
|
|
|
$showLinkForm(selectedLink, context);
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
},
|
2024-10-04 19:41:13 +08:00
|
|
|
'meta+shift+k': (editor, context) => {
|
2024-08-20 21:54:53 +08:00
|
|
|
showLinkSelector(entity => {
|
|
|
|
insertOrUpdateLink(editor, {
|
|
|
|
text: entity.name,
|
|
|
|
title: entity.link,
|
|
|
|
target: '',
|
|
|
|
url: entity.link,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
},
|
2024-08-20 20:07:33 +08:00
|
|
|
};
|
|
|
|
|
2024-08-20 21:54:53 +08:00
|
|
|
function createKeyDownListener(context: EditorUiContext): (e: KeyboardEvent) => void {
|
2024-08-20 20:07:33 +08:00
|
|
|
return (event: KeyboardEvent) => {
|
2024-10-04 19:41:13 +08:00
|
|
|
const combo = keyboardEventToKeyComboString(event);
|
2024-08-20 21:54:53 +08:00
|
|
|
// console.log(`pressed: ${combo}`);
|
2024-08-20 20:07:33 +08:00
|
|
|
if (actionsByKeys[combo]) {
|
2024-08-20 21:54:53 +08:00
|
|
|
const handled = actionsByKeys[combo](context.editor, context);
|
2024-08-20 20:07:33 +08:00
|
|
|
if (handled) {
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-10-04 19:41:13 +08:00
|
|
|
function keyboardEventToKeyComboString(event: KeyboardEvent): string {
|
|
|
|
const metaKeyPressed = isMac() ? event.metaKey : event.ctrlKey;
|
|
|
|
|
|
|
|
const parts = [
|
|
|
|
metaKeyPressed ? 'meta' : '',
|
|
|
|
event.shiftKey ? 'shift' : '',
|
|
|
|
event.key,
|
|
|
|
];
|
|
|
|
|
|
|
|
return parts.filter(Boolean).join('+').toLowerCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
function isMac(): boolean {
|
|
|
|
return window.navigator.userAgent.includes('Mac OS X');
|
|
|
|
}
|
|
|
|
|
2024-08-20 20:07:33 +08:00
|
|
|
function overrideDefaultCommands(editor: LexicalEditor) {
|
|
|
|
// Prevent default ctrl+enter command
|
|
|
|
editor.registerCommand(KEY_ENTER_COMMAND, (event) => {
|
2024-10-04 19:41:13 +08:00
|
|
|
if (isMac()) {
|
|
|
|
return event?.metaKey || false;
|
|
|
|
}
|
|
|
|
return event?.ctrlKey || false;
|
2024-08-20 20:07:33 +08:00
|
|
|
}, COMMAND_PRIORITY_HIGH);
|
|
|
|
}
|
|
|
|
|
2024-08-20 21:54:53 +08:00
|
|
|
export function registerShortcuts(context: EditorUiContext) {
|
|
|
|
const listener = createKeyDownListener(context);
|
|
|
|
overrideDefaultCommands(context.editor);
|
2024-08-20 20:07:33 +08:00
|
|
|
|
2024-08-20 21:54:53 +08:00
|
|
|
return context.editor.registerRootListener((rootElement: null | HTMLElement, prevRootElement: null | HTMLElement) => {
|
2024-08-20 20:07:33 +08:00
|
|
|
// add the listener to the current root element
|
|
|
|
rootElement?.addEventListener('keydown', listener);
|
|
|
|
// remove the listener from the old root element
|
|
|
|
prevRootElement?.removeEventListener('keydown', listener);
|
|
|
|
});
|
|
|
|
}
|