2020-09-06 17:30:25 +05:30
|
|
|
import * as React from 'react';
|
|
|
|
import { useState } from 'react';
|
|
|
|
|
2020-11-07 15:59:37 +00:00
|
|
|
import KeymapService, { KeymapItem } from '@joplin/lib/services/KeymapService';
|
2020-09-06 17:30:25 +05:30
|
|
|
import { ShortcutRecorder } from './ShortcutRecorder';
|
|
|
|
import getLabel from './utils/getLabel';
|
|
|
|
import useKeymap from './utils/useKeymap';
|
|
|
|
import useCommandStatus from './utils/useCommandStatus';
|
|
|
|
import styles_ from './styles';
|
2020-11-07 15:59:37 +00:00
|
|
|
import { _ } from '@joplin/lib/locale';
|
2020-09-06 17:30:25 +05:30
|
|
|
|
2021-01-22 17:41:11 +00:00
|
|
|
import shim from '@joplin/lib/shim';
|
2024-10-26 13:06:18 -07:00
|
|
|
import bridge from '../../services/bridge';
|
2020-09-06 17:30:25 +05:30
|
|
|
|
|
|
|
const keymapService = KeymapService.instance();
|
|
|
|
|
|
|
|
export interface KeymapConfigScreenProps {
|
2020-11-12 19:29:22 +00:00
|
|
|
themeId: number;
|
2020-09-06 17:30:25 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
export const KeymapConfigScreen = ({ themeId }: KeymapConfigScreenProps) => {
|
|
|
|
const styles = styles_(themeId);
|
|
|
|
|
|
|
|
const [filter, setFilter] = useState('');
|
|
|
|
const [keymapItems, keymapError, overrideKeymapItems, setAccelerator, resetAccelerator] = useKeymap();
|
|
|
|
const [recorderError, setRecorderError] = useState<Error>(null);
|
|
|
|
const [editing, enableEditing, disableEditing] = useCommandStatus();
|
|
|
|
|
2020-11-12 19:29:22 +00:00
|
|
|
const handleSave = (event: { commandName: string; accelerator: string }) => {
|
2020-09-06 17:30:25 +05:30
|
|
|
const { commandName, accelerator } = event;
|
|
|
|
setAccelerator(commandName, accelerator);
|
|
|
|
disableEditing(commandName);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleReset = (event: { commandName: string }) => {
|
|
|
|
const { commandName } = event;
|
|
|
|
resetAccelerator(commandName);
|
|
|
|
disableEditing(commandName);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleCancel = (event: { commandName: string }) => {
|
|
|
|
const { commandName } = event;
|
|
|
|
disableEditing(commandName);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleError = (event: { recorderError: Error }) => {
|
|
|
|
const { recorderError } = event;
|
|
|
|
setRecorderError(recorderError);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleImport = async () => {
|
2021-11-01 07:38:06 +00:00
|
|
|
const filePath = await bridge().showOpenDialog({
|
2020-09-06 17:30:25 +05:30
|
|
|
properties: ['openFile'],
|
|
|
|
defaultPath: 'keymap-desktop',
|
|
|
|
filters: [{ name: 'Joplin Keymaps (keymap-desktop.json)', extensions: ['json'] }],
|
|
|
|
});
|
|
|
|
|
2022-10-29 12:28:05 +02:00
|
|
|
if (filePath && filePath.length !== 0) {
|
2020-09-06 17:30:25 +05:30
|
|
|
const actualFilePath = filePath[0];
|
|
|
|
try {
|
|
|
|
const keymapFile = await shim.fsDriver().readFile(actualFilePath, 'utf-8');
|
|
|
|
overrideKeymapItems(JSON.parse(keymapFile));
|
2023-02-16 10:55:24 +00:00
|
|
|
} catch (error) {
|
|
|
|
bridge().showErrorMessageBox(_('Error: %s', error.message));
|
2020-09-06 17:30:25 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleExport = async () => {
|
2021-11-01 07:38:06 +00:00
|
|
|
const filePath = await bridge().showSaveDialog({
|
2020-09-06 17:30:25 +05:30
|
|
|
defaultPath: 'keymap-desktop',
|
|
|
|
filters: [{ name: 'Joplin Keymaps (keymap-desktop.json)', extensions: ['json'] }],
|
|
|
|
});
|
|
|
|
|
|
|
|
if (filePath) {
|
|
|
|
try {
|
|
|
|
// KeymapService is already synchronized with the in-state keymap
|
|
|
|
await keymapService.saveCustomKeymap(filePath);
|
2023-02-16 10:55:24 +00:00
|
|
|
} catch (error) {
|
2024-02-26 10:16:23 +00:00
|
|
|
bridge().showErrorMessageBox(error.message);
|
2020-09-06 17:30:25 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const renderAccelerator = (accelerator: string) => {
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
{accelerator.split('+').map(part => <kbd style={styles.kbd} key={part}>{part}</kbd>).reduce(
|
|
|
|
(accumulator, part) => (accumulator.length ? [...accumulator, ' + ', part] : [part]),
|
2023-08-22 11:58:53 +01:00
|
|
|
[],
|
2020-09-06 17:30:25 +05:30
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const renderStatus = (commandName: string) => {
|
2024-10-26 13:06:18 -07:00
|
|
|
if (!editing[commandName]) {
|
|
|
|
const editLabel = _('Change shortcut for "%s"', getLabel(commandName));
|
|
|
|
return <i className="fa fa-pen" role='img' aria-label={editLabel} title={editLabel}/>;
|
|
|
|
} else if (recorderError) {
|
|
|
|
return <i className="fa fa-exclamation-triangle" role='img' aria-label={recorderError.message} title={recorderError.message} />;
|
2020-09-06 17:30:25 +05:30
|
|
|
}
|
2024-10-26 13:06:18 -07:00
|
|
|
|
|
|
|
return null;
|
2020-09-06 17:30:25 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
const renderError = (error: Error) => {
|
|
|
|
return (
|
2020-10-25 17:46:41 +00:00
|
|
|
<div style={{ ...styles.warning, position: 'absolute', top: 0 }}>
|
2020-09-06 17:30:25 +05:30
|
|
|
<p style={styles.text}>
|
|
|
|
<span>
|
|
|
|
{error.message}
|
|
|
|
</span>
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const renderKeymapRow = ({ command, accelerator }: KeymapItem) => {
|
2024-10-26 13:06:18 -07:00
|
|
|
const handleClick = () => {
|
|
|
|
if (!editing[command]) {
|
|
|
|
enableEditing(command);
|
|
|
|
} else if (recorderError) {
|
|
|
|
void bridge().showErrorMessageBox(recorderError.message);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
const statusContent = renderStatus(command);
|
2020-09-06 17:30:25 +05:30
|
|
|
const cellContent =
|
2024-10-26 13:06:18 -07:00
|
|
|
<div className='keymap-shortcut-row-content'>
|
2020-09-06 17:30:25 +05:30
|
|
|
{editing[command] ?
|
|
|
|
<ShortcutRecorder
|
|
|
|
onSave={handleSave}
|
|
|
|
onReset={handleReset}
|
|
|
|
onCancel={handleCancel}
|
|
|
|
onError={handleError}
|
|
|
|
initialAccelerator={accelerator || '' /* Because accelerator is null if disabled */}
|
|
|
|
commandName={command}
|
|
|
|
themeId={themeId}
|
|
|
|
/> :
|
|
|
|
<div style={styles.tableCellContent} onClick={handleClick}>
|
|
|
|
{accelerator
|
|
|
|
? renderAccelerator(accelerator)
|
|
|
|
: <div style={styles.disabled}>{_('Disabled')}</div>
|
|
|
|
}
|
|
|
|
</div>
|
|
|
|
}
|
2024-10-26 13:06:18 -07:00
|
|
|
<button
|
|
|
|
className={`flat-button edit ${editing[command] ? '-editing' : ''}`}
|
|
|
|
style={styles.tableCellStatus}
|
|
|
|
aria-live={recorderError ? 'polite' : null}
|
|
|
|
tabIndex={statusContent ? 0 : -1}
|
|
|
|
onClick={handleClick}
|
|
|
|
>
|
|
|
|
{statusContent}
|
|
|
|
</button>
|
2020-09-06 17:30:25 +05:30
|
|
|
</div>;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<tr key={command}>
|
|
|
|
<td style={styles.tableCommandColumn}>
|
|
|
|
{getLabel(command)}
|
|
|
|
</td>
|
|
|
|
<td style={styles.tableShortcutColumn}>
|
|
|
|
{cellContent}
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
{keymapError && renderError(keymapError)}
|
|
|
|
<div style={styles.container}>
|
|
|
|
<div style={styles.actionsContainer}>
|
|
|
|
<input
|
|
|
|
value={filter}
|
|
|
|
onChange={event => setFilter(event.target.value)}
|
|
|
|
placeholder={_('Search...')}
|
|
|
|
style={styles.filterInput}
|
|
|
|
/>
|
|
|
|
<button style={styles.inlineButton} onClick={handleImport}>{_('Import')}</button>
|
|
|
|
<button style={styles.inlineButton} onClick={handleExport}>{_('Export')}</button>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<table style={styles.table}>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th style={styles.tableCommandColumn}>{_('Command')}</th>
|
|
|
|
<th style={styles.tableShortcutColumn}>{_('Keyboard Shortcut')}</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
{keymapItems.filter(({ command }) => {
|
|
|
|
const filterLowerCase = filter.toLowerCase();
|
|
|
|
return (command.toLowerCase().includes(filterLowerCase) || getLabel(command).toLowerCase().includes(filterLowerCase));
|
|
|
|
}).map(item => renderKeymapRow(item))}
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|