2022-06-06 14:23:48 +02:00
|
|
|
import { useEffect, useMemo } from 'react';
|
|
|
|
import { useIsAppleDevice } from './useIsAppleDevice';
|
|
|
|
|
|
|
|
export const useKeyboardShortcut = (
|
|
|
|
{
|
|
|
|
key,
|
|
|
|
modifiers = [],
|
|
|
|
preventDefault = false,
|
|
|
|
}: {
|
|
|
|
key: string;
|
|
|
|
modifiers?: Array<'ctrl' | 'alt' | 'shift'>;
|
|
|
|
preventDefault?: boolean;
|
|
|
|
},
|
2023-10-02 14:25:46 +02:00
|
|
|
callback: () => void,
|
2022-06-06 14:23:48 +02:00
|
|
|
) => {
|
|
|
|
const isAppleDevice = useIsAppleDevice();
|
|
|
|
useEffect(() => {
|
|
|
|
const onKeyDown = (event: KeyboardEvent) => {
|
|
|
|
if (key !== event.key) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (modifiers.includes('ctrl')) {
|
|
|
|
if (isAppleDevice) {
|
|
|
|
if (!event.metaKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!event.ctrlKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (modifiers.includes('alt') && !event.altKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (modifiers.includes('shift') && !event.shiftKey) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (preventDefault) {
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
};
|
|
|
|
|
|
|
|
window.addEventListener('keydown', onKeyDown);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
window.removeEventListener('keydown', onKeyDown);
|
|
|
|
};
|
|
|
|
}, [isAppleDevice, key, modifiers, preventDefault, callback]);
|
|
|
|
|
|
|
|
const formattedModifiers = useMemo(
|
|
|
|
() =>
|
|
|
|
modifiers.map(
|
2023-10-02 14:25:46 +02:00
|
|
|
(modifier) =>
|
2022-06-06 14:23:48 +02:00
|
|
|
({
|
|
|
|
ctrl: isAppleDevice ? '⌘' : 'Ctrl',
|
|
|
|
alt: 'Alt',
|
|
|
|
shift: 'Shift',
|
2023-10-02 14:25:46 +02:00
|
|
|
})[modifier],
|
2022-06-06 14:23:48 +02:00
|
|
|
),
|
2023-10-02 14:25:46 +02:00
|
|
|
[isAppleDevice, modifiers],
|
2022-06-06 14:23:48 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const hotkeyDescription = useMemo(
|
|
|
|
() =>
|
|
|
|
[
|
|
|
|
...formattedModifiers,
|
|
|
|
`${key[0].toUpperCase()}${key.slice(1)}`,
|
|
|
|
].join('+'),
|
2023-10-02 14:25:46 +02:00
|
|
|
[formattedModifiers, key],
|
2022-06-06 14:23:48 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return hotkeyDescription;
|
|
|
|
};
|