mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-14 18:27:44 +02:00
104 lines
3.2 KiB
TypeScript
104 lines
3.2 KiB
TypeScript
import * as React from 'react';
|
|
import { useState, useEffect, KeyboardEvent } from 'react';
|
|
|
|
import KeymapService from '@joplin/lib/services/KeymapService';
|
|
import styles_ from './styles';
|
|
|
|
import { _ } from '@joplin/lib/locale';
|
|
const keymapService = KeymapService.instance();
|
|
|
|
export interface ShortcutRecorderProps {
|
|
onSave: (event: { commandName: string; accelerator: string })=> void;
|
|
onReset: (event: { commandName: string })=> void;
|
|
onCancel: (event: { commandName: string })=> void;
|
|
onError: (event: { recorderError: Error })=> void;
|
|
initialAccelerator: string;
|
|
commandName: string;
|
|
themeId: number;
|
|
}
|
|
|
|
export const ShortcutRecorder = ({ onSave, onReset, onCancel, onError, initialAccelerator, commandName, themeId }: ShortcutRecorderProps) => {
|
|
const styles = styles_(themeId);
|
|
|
|
const [accelerator, setAccelerator] = useState(initialAccelerator);
|
|
const [saveAllowed, setSaveAllowed] = useState(true);
|
|
|
|
useEffect(() => {
|
|
try {
|
|
// Only perform validations if there's an accelerator provided
|
|
// Otherwise performing a save means that it's going to be disabled
|
|
if (accelerator) {
|
|
keymapService.validateAccelerator(accelerator);
|
|
keymapService.validateKeymap({ accelerator, command: commandName });
|
|
}
|
|
|
|
// Discard previous errors
|
|
onError({ recorderError: null });
|
|
setSaveAllowed(true);
|
|
} catch (recorderError) {
|
|
onError({ recorderError });
|
|
setSaveAllowed(false);
|
|
}
|
|
// eslint-disable-next-line @seiyab/react-hooks/exhaustive-deps -- Old code before rule was applied
|
|
}, [accelerator]);
|
|
|
|
const handleKeyDown = (event: KeyboardEvent<HTMLDivElement>) => {
|
|
// Shift-tab and tab are needed for navigating the shortcuts screen with the keyboard. Do not
|
|
// .preventDefault.
|
|
if (event.code === 'Tab' && !event.metaKey && !event.altKey && !event.ctrlKey) {
|
|
return;
|
|
}
|
|
|
|
event.preventDefault();
|
|
const newAccelerator = keymapService.domToElectronAccelerator(event);
|
|
|
|
switch (newAccelerator) {
|
|
case 'Enter':
|
|
if (saveAllowed) return onSave({ commandName, accelerator });
|
|
break;
|
|
case 'Escape':
|
|
return onCancel({ commandName });
|
|
case 'Backspace':
|
|
case 'Delete':
|
|
return setAccelerator('');
|
|
default:
|
|
setAccelerator(newAccelerator);
|
|
}
|
|
};
|
|
|
|
const hintText = _('Press the shortcut and then press ENTER. Or, press BACKSPACE to clear the shortcut.');
|
|
const placeholderText = _('Press the shortcut');
|
|
|
|
return (
|
|
<div className='shortcut-recorder' style={styles.recorderContainer}>
|
|
<input
|
|
className='shortcut text-input'
|
|
|
|
value={accelerator}
|
|
aria-label={accelerator ? accelerator : placeholderText}
|
|
placeholder={placeholderText}
|
|
title={hintText}
|
|
aria-description={hintText}
|
|
aria-invalid={accelerator && !saveAllowed}
|
|
// With readOnly, aria-live polite seems necessary for screen readers to read
|
|
// the shortcut as it updates.
|
|
aria-live='polite'
|
|
|
|
onKeyDown={handleKeyDown}
|
|
readOnly
|
|
autoFocus
|
|
/>
|
|
|
|
<button style={styles.inlineButton} disabled={!saveAllowed} onClick={() => onSave({ commandName, accelerator })}>
|
|
{_('Save')}
|
|
</button>
|
|
<button style={styles.inlineButton} onClick={() => onReset({ commandName })}>
|
|
{_('Restore')}
|
|
</button>
|
|
<button style={styles.inlineButton} onClick={() => onCancel({ commandName })}>
|
|
{_('Cancel')}
|
|
</button>
|
|
</div>
|
|
);
|
|
};
|