2023-08-21 17:01:20 +02:00
|
|
|
import * as React from 'react';
|
|
|
|
import BaseModel from '@joplin/lib/BaseModel';
|
|
|
|
import Note from '@joplin/lib/models/Note';
|
|
|
|
import CommandService from '@joplin/lib/services/CommandService';
|
|
|
|
import { NoteEntity } from '@joplin/lib/services/database/types';
|
|
|
|
import { useCallback } from 'react';
|
|
|
|
import { Dispatch } from 'redux';
|
|
|
|
import { FocusNote } from './useFocusNote';
|
2023-09-18 18:40:36 +02:00
|
|
|
import { ItemFlow } from '@joplin/lib/services/plugins/api/noteListType';
|
2023-08-21 17:01:20 +02:00
|
|
|
import { KeyboardEventKey } from '@joplin/lib/dom';
|
2024-08-31 17:05:01 +02:00
|
|
|
import announceForAccessibility from '../../utils/announceForAccessibility';
|
|
|
|
import { _ } from '@joplin/lib/locale';
|
2023-08-21 17:01:20 +02:00
|
|
|
|
|
|
|
const useOnKeyDown = (
|
2024-08-31 17:05:01 +02:00
|
|
|
activeNoteId: string,
|
2023-08-21 17:01:20 +02:00
|
|
|
selectedNoteIds: string[],
|
|
|
|
moveNote: (direction: number, inc: number)=> void,
|
|
|
|
makeItemIndexVisible: (itemIndex: number)=> void,
|
|
|
|
focusNote: FocusNote,
|
|
|
|
notes: NoteEntity[],
|
|
|
|
dispatch: Dispatch,
|
|
|
|
visibleItemCount: number,
|
|
|
|
noteCount: number,
|
|
|
|
flow: ItemFlow,
|
2023-08-22 12:58:53 +02:00
|
|
|
itemsPerLine: number,
|
2023-08-21 17:01:20 +02:00
|
|
|
) => {
|
|
|
|
const scrollNoteIndex = useCallback((visibleItemCount: number, key: KeyboardEventKey, ctrlKey: boolean, metaKey: boolean, noteIndex: number) => {
|
|
|
|
if (flow === ItemFlow.TopToBottom) {
|
|
|
|
if (key === 'PageUp') {
|
|
|
|
noteIndex -= (visibleItemCount - 1);
|
|
|
|
} else if (key === 'PageDown') {
|
|
|
|
noteIndex += (visibleItemCount - 1);
|
|
|
|
} else if ((key === 'End' && ctrlKey) || (key === 'ArrowDown' && metaKey)) {
|
|
|
|
noteIndex = noteCount - 1;
|
|
|
|
} else if ((key === 'Home' && ctrlKey) || (key === 'ArrowUp' && metaKey)) {
|
|
|
|
noteIndex = 0;
|
|
|
|
} else if (key === 'ArrowUp' && !metaKey) {
|
|
|
|
noteIndex -= 1;
|
|
|
|
} else if (key === 'ArrowDown' && !metaKey) {
|
|
|
|
noteIndex += 1;
|
|
|
|
}
|
|
|
|
if (noteIndex < 0) noteIndex = 0;
|
|
|
|
if (noteIndex > noteCount - 1) noteIndex = noteCount - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flow === ItemFlow.LeftToRight) {
|
|
|
|
if (key === 'PageUp') {
|
|
|
|
noteIndex -= (visibleItemCount - itemsPerLine);
|
|
|
|
} else if (key === 'PageDown') {
|
|
|
|
noteIndex += (visibleItemCount - itemsPerLine);
|
|
|
|
} else if ((key === 'End' && ctrlKey) || (key === 'ArrowDown' && metaKey)) {
|
|
|
|
noteIndex = noteCount - 1;
|
|
|
|
} else if ((key === 'Home' && ctrlKey) || (key === 'ArrowUp' && metaKey)) {
|
|
|
|
noteIndex = 0;
|
|
|
|
} else if (key === 'ArrowUp' && !metaKey) {
|
|
|
|
noteIndex -= itemsPerLine;
|
|
|
|
} else if (key === 'ArrowDown' && !metaKey) {
|
|
|
|
noteIndex += itemsPerLine;
|
|
|
|
} else if (key === 'ArrowLeft' && !metaKey) {
|
|
|
|
noteIndex -= 1;
|
|
|
|
} else if (key === 'ArrowRight' && !metaKey) {
|
|
|
|
noteIndex += 1;
|
|
|
|
}
|
|
|
|
if (noteIndex < 0) noteIndex = 0;
|
|
|
|
if (noteIndex > noteCount - 1) noteIndex = noteCount - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return noteIndex;
|
|
|
|
}, [noteCount, flow, itemsPerLine]);
|
|
|
|
|
|
|
|
const onKeyDown: React.KeyboardEventHandler<HTMLDivElement> = useCallback(async (event) => {
|
|
|
|
const noteIds = selectedNoteIds;
|
|
|
|
const key = event.key as KeyboardEventKey;
|
|
|
|
|
|
|
|
if (['ArrowDown', 'ArrowUp', 'ArrowLeft', 'ArrowRight'].includes(key) && event.altKey) {
|
|
|
|
if (flow === ItemFlow.TopToBottom) {
|
|
|
|
await moveNote(key === 'ArrowDown' ? 1 : -1, 1);
|
|
|
|
} else {
|
|
|
|
if (key === 'ArrowRight') {
|
|
|
|
await moveNote(1, 1);
|
|
|
|
} else if (key === 'ArrowLeft') {
|
|
|
|
await moveNote(-1, 1);
|
|
|
|
} else if (key === 'ArrowUp') {
|
|
|
|
await moveNote(-1, itemsPerLine);
|
|
|
|
} else if (key === 'ArrowDown') {
|
|
|
|
await moveNote(1, itemsPerLine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
event.preventDefault();
|
2024-08-31 17:05:01 +02:00
|
|
|
} else if (notes.length > 0 && (key === 'ArrowDown' || key === 'ArrowUp' || key === 'ArrowLeft' || key === 'ArrowRight' || key === 'PageDown' || key === 'PageUp' || key === 'End' || key === 'Home')) {
|
|
|
|
const noteId = activeNoteId ?? notes[0]?.id;
|
2023-08-21 17:01:20 +02:00
|
|
|
let noteIndex = BaseModel.modelIndexById(notes, noteId);
|
|
|
|
|
|
|
|
noteIndex = scrollNoteIndex(visibleItemCount, key, event.ctrlKey, event.metaKey, noteIndex);
|
|
|
|
|
|
|
|
const newSelectedNote = notes[noteIndex];
|
|
|
|
|
2024-08-31 17:05:01 +02:00
|
|
|
if (event.shiftKey) {
|
|
|
|
if (selectedNoteIds.includes(newSelectedNote.id)) {
|
|
|
|
dispatch({
|
|
|
|
type: 'NOTE_SELECT_REMOVE',
|
|
|
|
id: noteId,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: 'NOTE_SELECT_ADD',
|
|
|
|
id: newSelectedNote.id,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
dispatch({
|
|
|
|
type: 'NOTE_SELECT',
|
|
|
|
id: newSelectedNote.id,
|
|
|
|
});
|
|
|
|
}
|
2023-08-21 17:01:20 +02:00
|
|
|
|
|
|
|
makeItemIndexVisible(noteIndex);
|
|
|
|
|
|
|
|
focusNote(newSelectedNote.id);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
|
2024-07-06 12:05:35 +02:00
|
|
|
if (noteIds.length) {
|
2024-09-16 23:16:51 +02:00
|
|
|
if (key === 'Delete' && event.shiftKey || (key === 'Backspace' && event.metaKey && event.altKey)) {
|
2024-07-06 12:05:35 +02:00
|
|
|
event.preventDefault();
|
|
|
|
if (CommandService.instance().isEnabled('permanentlyDeleteNote')) {
|
|
|
|
void CommandService.instance().execute('permanentlyDeleteNote', noteIds);
|
|
|
|
}
|
|
|
|
} else if (key === 'Delete' || (key === 'Backspace' && event.metaKey)) {
|
|
|
|
event.preventDefault();
|
|
|
|
if (CommandService.instance().isEnabled('deleteNote')) {
|
|
|
|
void CommandService.instance().execute('deleteNote', noteIds);
|
|
|
|
}
|
2024-03-02 16:25:27 +02:00
|
|
|
}
|
2023-08-21 17:01:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (noteIds.length && key === ' ') {
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
const selectedNotes = BaseModel.modelsByIds(notes, noteIds);
|
2024-08-31 17:05:01 +02:00
|
|
|
const todos = selectedNotes.filter(n => !!n.is_todo);
|
2023-08-21 17:01:20 +02:00
|
|
|
if (!todos.length) return;
|
|
|
|
|
|
|
|
for (let i = 0; i < todos.length; i++) {
|
|
|
|
const toggledTodo = Note.toggleTodoCompleted(todos[i]);
|
|
|
|
await Note.save(toggledTodo);
|
|
|
|
}
|
|
|
|
|
2024-08-31 17:05:01 +02:00
|
|
|
dispatch({ type: 'NOTE_SORT' });
|
2023-08-21 17:01:20 +02:00
|
|
|
focusNote(todos[0].id);
|
2024-08-31 17:05:01 +02:00
|
|
|
const wasCompleted = !!todos[0].todo_completed;
|
|
|
|
announceForAccessibility(!wasCompleted ? _('Complete') : _('Incomplete'));
|
2023-08-21 17:01:20 +02:00
|
|
|
}
|
|
|
|
|
2024-11-27 14:33:13 +02:00
|
|
|
// Check for isDefaultPrevented to allow plugins to call .preventDefault
|
|
|
|
if (key === 'Enter' && !event.isDefaultPrevented()) {
|
2023-08-21 17:01:20 +02:00
|
|
|
event.preventDefault();
|
2024-11-27 14:33:13 +02:00
|
|
|
|
|
|
|
if (event.shiftKey) {
|
|
|
|
void CommandService.instance().execute('focusElement', 'sideBar');
|
|
|
|
} else {
|
|
|
|
void CommandService.instance().execute('focusElement', 'noteTitle');
|
|
|
|
}
|
2023-08-21 17:01:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (key.toUpperCase() === 'A' && (event.ctrlKey || event.metaKey)) {
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: 'NOTE_SELECT_ALL',
|
|
|
|
});
|
|
|
|
}
|
2024-08-31 17:05:01 +02:00
|
|
|
}, [moveNote, focusNote, visibleItemCount, scrollNoteIndex, makeItemIndexVisible, notes, selectedNoteIds, activeNoteId, dispatch, flow, itemsPerLine]);
|
2023-08-21 17:01:20 +02:00
|
|
|
|
|
|
|
|
|
|
|
return onKeyDown;
|
|
|
|
};
|
|
|
|
|
|
|
|
export default useOnKeyDown;
|