2021-02-14 02:56:55 +00:00
|
|
|
import { isMac, Key } from './Keyboard';
|
|
|
|
import SettingsStore from './settings/SettingsStore';
|
2021-02-11 09:18:10 +00:00
|
|
|
|
|
|
|
export enum KeyBindingContext {
|
2021-02-14 02:56:55 +00:00
|
|
|
SendMessageComposer = 'SendMessageComposer',
|
2021-02-11 09:18:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export enum KeyAction {
|
|
|
|
None = 'None',
|
2021-02-14 02:56:55 +00:00
|
|
|
// SendMessageComposer actions:
|
|
|
|
Send = 'Send',
|
|
|
|
SelectPrevSendHistory = 'SelectPrevSendHistory',
|
|
|
|
SelectNextSendHistory = 'SelectNextSendHistory',
|
|
|
|
EditLastMessage = 'EditLastMessage',
|
2021-02-11 09:18:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Represent a key combination.
|
|
|
|
*
|
2021-02-14 02:56:55 +00:00
|
|
|
* The combo is evaluated strictly, i.e. the KeyboardEvent must match exactly what is specified in the KeyCombo.
|
2021-02-11 09:18:10 +00:00
|
|
|
*/
|
|
|
|
export type KeyCombo = {
|
|
|
|
/** Currently only one `normal` key is supported */
|
|
|
|
keys: string[];
|
|
|
|
|
|
|
|
/** On PC: ctrl is pressed; on Mac: meta is pressed */
|
|
|
|
ctrlOrCmd?: boolean;
|
|
|
|
|
|
|
|
altKey?: boolean;
|
|
|
|
ctrlKey?: boolean;
|
|
|
|
metaKey?: boolean;
|
|
|
|
shiftKey?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type KeyBinding = {
|
|
|
|
action: KeyAction;
|
2021-02-14 02:56:55 +00:00
|
|
|
keyCombo: KeyCombo;
|
|
|
|
}
|
|
|
|
|
|
|
|
const messageComposerBindings = (): KeyBinding[] => {
|
|
|
|
const bindings: KeyBinding[] = [
|
|
|
|
{
|
|
|
|
action: KeyAction.SelectPrevSendHistory,
|
|
|
|
keyCombo: {
|
|
|
|
keys: [Key.ARROW_UP],
|
|
|
|
altKey: true,
|
|
|
|
ctrlKey: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
action: KeyAction.SelectNextSendHistory,
|
|
|
|
keyCombo: {
|
|
|
|
keys: [Key.ARROW_DOWN],
|
|
|
|
altKey: true,
|
|
|
|
ctrlKey: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
action: KeyAction.EditLastMessage,
|
|
|
|
keyCombo: {
|
|
|
|
keys: [Key.ARROW_UP],
|
|
|
|
}
|
|
|
|
},
|
|
|
|
];
|
|
|
|
if (SettingsStore.getValue('MessageComposerInput.ctrlEnterToSend')) {
|
|
|
|
bindings.push({
|
|
|
|
action: KeyAction.Send,
|
|
|
|
keyCombo: {
|
|
|
|
keys: [Key.ENTER],
|
|
|
|
ctrlOrCmd: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
bindings.push({
|
|
|
|
action: KeyAction.Send,
|
|
|
|
keyCombo: {
|
|
|
|
keys: [Key.ENTER],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return bindings;
|
2021-02-11 09:18:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper method to check if a KeyboardEvent matches a KeyCombo
|
|
|
|
*
|
|
|
|
* Note, this method is only exported for testing.
|
|
|
|
*/
|
|
|
|
export function isKeyComboMatch(ev: KeyboardEvent, combo: KeyCombo, onMac: boolean): boolean {
|
|
|
|
if (combo.keys.length > 0 && ev.key !== combo.keys[0]) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const comboCtrl = combo.ctrlKey ?? false;
|
|
|
|
const comboAlt = combo.altKey ?? false;
|
|
|
|
const comboShift = combo.shiftKey ?? false;
|
|
|
|
const comboMeta = combo.metaKey ?? false;
|
|
|
|
// When ctrlOrCmd is set, the keys need do evaluated differently on PC and Mac
|
|
|
|
if (combo.ctrlOrCmd) {
|
|
|
|
if (onMac) {
|
|
|
|
if (!ev.metaKey
|
|
|
|
|| ev.ctrlKey !== comboCtrl
|
|
|
|
|| ev.altKey !== comboAlt
|
|
|
|
|| ev.shiftKey !== comboShift) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!ev.ctrlKey
|
|
|
|
|| ev.metaKey !== comboMeta
|
|
|
|
|| ev.altKey !== comboAlt
|
|
|
|
|| ev.shiftKey !== comboShift) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ev.metaKey !== comboMeta
|
|
|
|
|| ev.ctrlKey !== comboCtrl
|
|
|
|
|| ev.altKey !== comboAlt
|
|
|
|
|| ev.shiftKey !== comboShift) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-02-14 02:56:55 +00:00
|
|
|
export type KeyBindingsGetter = () => KeyBinding[];
|
|
|
|
|
2021-02-11 09:18:10 +00:00
|
|
|
export class KeyBindingsManager {
|
2021-02-14 02:56:55 +00:00
|
|
|
/**
|
|
|
|
* Map of KeyBindingContext to a KeyBinding getter arrow function.
|
|
|
|
*
|
|
|
|
* Returning a getter function allowed to have dynamic bindings, e.g. when settings change the bindings can be
|
|
|
|
* recalculated.
|
|
|
|
*/
|
|
|
|
contextBindings: Record<KeyBindingContext, KeyBindingsGetter> = {
|
|
|
|
[KeyBindingContext.SendMessageComposer]: messageComposerBindings,
|
|
|
|
};
|
2021-02-11 09:18:10 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Finds a matching KeyAction for a given KeyboardEvent
|
|
|
|
*/
|
|
|
|
getAction(context: KeyBindingContext, ev: KeyboardEvent): KeyAction {
|
2021-02-14 02:56:55 +00:00
|
|
|
const bindings = this.contextBindings[context]?.();
|
2021-02-11 09:18:10 +00:00
|
|
|
if (!bindings) {
|
|
|
|
return KeyAction.None;
|
|
|
|
}
|
|
|
|
const binding = bindings.find(it => isKeyComboMatch(ev, it.keyCombo, isMac));
|
|
|
|
if (binding) {
|
|
|
|
return binding.action;
|
|
|
|
}
|
|
|
|
|
|
|
|
return KeyAction.None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const manager = new KeyBindingsManager();
|
|
|
|
|
|
|
|
export function getKeyBindingsManager(): KeyBindingsManager {
|
|
|
|
return manager;
|
|
|
|
}
|