2023-10-22 13:00:19 +02:00
|
|
|
import { RefObject, useEffect } from 'react';
|
2024-09-26 12:35:32 +02:00
|
|
|
import { NoteBodyEditorRef, OnChangeEvent, ScrollOptionTypes } from './types';
|
2023-07-16 18:42:42 +02:00
|
|
|
import editorCommandDeclarations, { enabledCondition } from '../editorCommandDeclarations';
|
2021-01-23 17:51:19 +02:00
|
|
|
import CommandService, { CommandDeclaration, CommandRuntime, CommandContext } from '@joplin/lib/services/CommandService';
|
2021-01-22 19:41:11 +02:00
|
|
|
import time from '@joplin/lib/time';
|
2021-01-29 20:45:11 +02:00
|
|
|
import { reg } from '@joplin/lib/registry';
|
2020-07-03 23:32:39 +02:00
|
|
|
|
|
|
|
const commandsWithDependencies = [
|
|
|
|
require('../commands/showLocalSearch'),
|
|
|
|
require('../commands/focusElementNoteTitle'),
|
|
|
|
require('../commands/focusElementNoteBody'),
|
2023-02-13 21:16:33 +02:00
|
|
|
require('../commands/pasteAsText'),
|
2020-07-03 23:32:39 +02:00
|
|
|
];
|
2020-05-02 17:41:07 +02:00
|
|
|
|
2024-09-26 12:35:32 +02:00
|
|
|
type OnBodyChange = (event: OnChangeEvent)=> void;
|
2023-10-22 13:00:19 +02:00
|
|
|
|
2020-05-02 17:41:07 +02:00
|
|
|
interface HookDependencies {
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
setShowLocalSearch: Function;
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
dispatch: Function;
|
2024-04-05 13:16:49 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
noteSearchBarRef: any;
|
2023-10-22 13:00:19 +02:00
|
|
|
editorRef: RefObject<NoteBodyEditorRef>;
|
2024-05-21 02:28:19 +02:00
|
|
|
titleInputRef: RefObject<HTMLInputElement>;
|
2024-09-26 12:35:32 +02:00
|
|
|
onBodyChange: OnBodyChange;
|
2020-05-02 17:41:07 +02:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:00:19 +02:00
|
|
|
function editorCommandRuntime(
|
|
|
|
declaration: CommandDeclaration,
|
|
|
|
editorRef: RefObject<NoteBodyEditorRef>,
|
2024-09-26 12:35:32 +02:00
|
|
|
onBodyChange: OnBodyChange,
|
2023-10-22 13:00:19 +02:00
|
|
|
): CommandRuntime {
|
2020-07-03 23:32:39 +02:00
|
|
|
return {
|
2024-04-05 13:16:49 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2020-11-12 21:13:28 +02:00
|
|
|
execute: async (_context: CommandContext, ...args: any[]) => {
|
2021-01-08 00:03:13 +02:00
|
|
|
if (!editorRef.current) {
|
|
|
|
reg.logger().warn('Received command, but editor is gone', declaration.name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-03 23:32:39 +02:00
|
|
|
if (!editorRef.current.execCommand) {
|
|
|
|
reg.logger().warn('Received command, but editor cannot execute commands', declaration.name);
|
2020-10-18 22:52:10 +02:00
|
|
|
return;
|
2020-05-02 17:41:07 +02:00
|
|
|
}
|
2020-10-09 19:35:46 +02:00
|
|
|
|
2020-10-18 22:52:10 +02:00
|
|
|
if (declaration.name === 'insertDateTime') {
|
|
|
|
return editorRef.current.execCommand({
|
|
|
|
name: 'insertText',
|
|
|
|
value: time.formatMsToLocal(new Date().getTime()),
|
|
|
|
});
|
|
|
|
} else if (declaration.name === 'scrollToHash') {
|
|
|
|
return editorRef.current.scrollTo({
|
|
|
|
type: ScrollOptionTypes.Hash,
|
|
|
|
value: args[0],
|
|
|
|
});
|
2020-11-14 12:57:09 +02:00
|
|
|
} else if (declaration.name === 'editor.setText') {
|
2024-09-26 12:35:32 +02:00
|
|
|
onBodyChange({ content: args[0], changeId: 0 });
|
2020-10-18 22:52:10 +02:00
|
|
|
} else {
|
|
|
|
return editorRef.current.execCommand({
|
|
|
|
name: declaration.name,
|
|
|
|
value: args[0],
|
|
|
|
});
|
|
|
|
}
|
2020-07-03 23:32:39 +02:00
|
|
|
},
|
2021-12-31 10:20:29 +02:00
|
|
|
|
|
|
|
// We disable the editor commands whenever a modal dialog is visible,
|
|
|
|
// otherwise the user might type something in a dialog and accidentally
|
|
|
|
// change something in the editor. However, we still enable them for
|
|
|
|
// GotoAnything so that it's possible to type eg `textBold` and bold the
|
|
|
|
// currently selected text.
|
|
|
|
//
|
|
|
|
// https://github.com/laurent22/joplin/issues/5707
|
2023-07-16 18:42:42 +02:00
|
|
|
enabledCondition: enabledCondition(declaration.name),
|
2020-07-03 23:32:39 +02:00
|
|
|
};
|
|
|
|
}
|
2020-05-02 17:41:07 +02:00
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
export default function useWindowCommandHandler(dependencies: HookDependencies) {
|
2024-09-26 12:35:32 +02:00
|
|
|
const { setShowLocalSearch, noteSearchBarRef, editorRef, titleInputRef, onBodyChange } = dependencies;
|
2020-05-02 17:41:07 +02:00
|
|
|
|
2020-07-03 23:32:39 +02:00
|
|
|
useEffect(() => {
|
2021-02-06 18:01:06 +02:00
|
|
|
for (const declaration of editorCommandDeclarations) {
|
2024-09-26 12:35:32 +02:00
|
|
|
CommandService.instance().registerRuntime(declaration.name, editorCommandRuntime(declaration, editorRef, onBodyChange));
|
2020-07-03 23:32:39 +02:00
|
|
|
}
|
2020-05-02 17:41:07 +02:00
|
|
|
|
2020-07-03 23:32:39 +02:00
|
|
|
const dependencies = {
|
|
|
|
editorRef,
|
|
|
|
setShowLocalSearch,
|
|
|
|
noteSearchBarRef,
|
|
|
|
titleInputRef,
|
|
|
|
};
|
2020-05-02 17:41:07 +02:00
|
|
|
|
2020-07-03 23:32:39 +02:00
|
|
|
for (const command of commandsWithDependencies) {
|
|
|
|
CommandService.instance().registerRuntime(command.declaration.name, command.runtime(dependencies));
|
2020-05-02 17:41:07 +02:00
|
|
|
}
|
|
|
|
|
2020-07-03 23:32:39 +02:00
|
|
|
return () => {
|
2021-02-06 18:01:06 +02:00
|
|
|
for (const declaration of editorCommandDeclarations) {
|
2020-07-03 23:32:39 +02:00
|
|
|
CommandService.instance().unregisterRuntime(declaration.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const command of commandsWithDependencies) {
|
|
|
|
CommandService.instance().unregisterRuntime(command.declaration.name);
|
|
|
|
}
|
|
|
|
};
|
2024-09-26 12:35:32 +02:00
|
|
|
}, [editorRef, setShowLocalSearch, noteSearchBarRef, titleInputRef, onBodyChange]);
|
2020-05-02 17:41:07 +02:00
|
|
|
}
|