This provides an easy way to add keyboard shortcuts to any vue component via mixin or composition api. Why not start with the interactive demo with code examples? This package is a wrapper around mousetrap so additional information (key aliases, etc) can be found there.
npm install --save @unrest/vue-mousetrap
should install this package from npm. If that doesn't work for you or if there's another distribution method you'd prefer please open an issue on github.
<script>
import { ref } from 'vue'
import Mousetrap from '@unrest/vue-mousetrap'
export default {
setup(props) {
const game = {
paused: ref(false),
// pause, unpause, move, showHelp, hideHelp, and toggleCheat go here
}
Mousetrap.use(() => {
if (game.paused.value) { // 1
return {
esc: game.unpause
}
}
return {
esc: game.pause,
'space,x,enter': game.useItem, // 2
'up,down,left,right': (event) => game.move(event.key), // 3
'?': { // 4
keydown: game.showHelp,
keyup: game.hideHelp,
},
'up up down down left right left right b a,mod+shift+q': game.toggleCheat // 5
}
})
return { paused, game }
},
}
</script>
Notes:
-
Because
game.paused
is aref
and the action_map argument forMousetrap.use(action_map)
is a function, hotkeys will be recalculated and rebound every timegame.paused.value
changes. If actions are unchanging, then actions can be an object instead of a function (see mixin or action map for example). -
Multiple hotkeys can be specified if separated by a comma
,
. This delimiter can be change by usingMousetrap.use(action_map, { delimeter: '&&&' })
-
The first argument passed into an action is the native keyboard event. Also, arrow functions are the best.
-
The action can either be a function or a an object with keys
{ keydown, keyup, repeat, global }
. See full details below. -
Any valid mousetrap sequence will work for the keys. In this case the konami code or pressing ctrl+shift+q (command+shift+q on a mac) will
toggleCheat
.
Use the default import from this package and add a mousetrap computed property to a component to enable shortcuts. component.mousetrap
should return an object with a shortcut string as the key and an action or action object as the value.
<script>
import Mousetrap from '@unrest/vue-mousetrap'
export default {
mixins: [Mousetrap.Mixin],
computed: {
mousetrap() {
if (this.paused) { // 1
return {
esc: this.unpause
}
}
return {
esc: this.pause,
'space,x,enter': this.useItem, // 2
'up,down,left,right': (event) => this.move(event.key), // 3
'?': { // 4
keydown: this.showHelp,
keyup: this.hideHelp,
},
'up up down down left right left right b a,mod+shift+q': this.toggleCheat // 5
}
}
},
methods: {
... // pause, unpause, move, showHelp, hideHelp, and toggleCheat go here
}
}
// ...
Notes:
-
Because this is a computed property, hotkeys will be reloaded any time
this.paused
is changed. Other hotkeys (arrows and help menu) will not be available while paused. -
Multiple hotkeys can be specified if separated by a comma
,
. This delimiter can be set using mixin config options below. -
The first argument passed into an action is the native keyboard event. Also, arrow functions are the best.
-
The action can either be a function or a an object with keys
{ keydown, keyup, repeat, global }
. See full details below. -
Any valid mousetrap sequence will work for the keys. In this case the konami code or pressing ctrl+shift+q (command+shift+q on a mac) will
toggleCheat
.
If you want to use a different delimiter or namespace for the mousetrap, the mixin is configurable:
mixins: [MousetrapMixin.config({ delimeter: '&&&', namespace: 'myHotkeys' })],
computed: {
myHotkeys: {
', &&& enter': this.doSomething
}
}
If the action is specified as a function, the hotkey will default to executing that function on keydown. So esc: () => this.paused = false
and esc: { keydown: () => this.paused = false }
are equivalent. Full behavior can be customized by passing in an object instead. Available options are
-
keydown: Function(event)
-
keyup: Function(event)
-
keypress: Function(event)
-
repeat: Function(event)
- Will be fired whenevent.repeat === true
(ie user is holding down a key). Ifkeydown === undefined
, keydown will default to repeat. -
global: Boolean
- By default mousetrap will ignore hotkeys on input, textarea, and select elements. Setting global to true will capture these events (ie you want ctrl+s be captured even when a text area is focused).
mousetrap: {
a: {
keydown: this.doSomething,
keypress: this.doSomethingElse, // TBH I don't know what keypress is used for
repeat: this.doSomethingALot,
keyup: this.stopSomething,
global: true, // all the above work anywhere, including inputs
},
b: () => this.keyPressed('b'), // a function will be interpreted as { keydown: function }
'up,down,left,right': {
repeat: this.move, // holding down arrows spams repeat, keydown is this.move
},
'ctrl+s': {
keydown: this.save,
global: true, // save even in textarea
},
'?': {
keydown: this.showHelp,
keyup: this.hideHelp,
},
}