admin管理员组文章数量:1135147
I wish to listen to an 'esc' key event in order to call a method in a Vue component. The docs shows this example:
<input v-on:keyup.enter="submit">
but i'm using a <div></div>
, and need to catch the event from outside.
However, I wish NOT to overload global handlers or anything like that.
Any suggestions?
I wish to listen to an 'esc' key event in order to call a method in a Vue component. The docs shows this example:
<input v-on:keyup.enter="submit">
but i'm using a <div></div>
, and need to catch the event from outside.
However, I wish NOT to overload global handlers or anything like that.
Any suggestions?
Share Improve this question asked Aug 8, 2017 at 12:48 ChenChen 3,0605 gold badges28 silver badges48 bronze badges 3- did you find your solution for this ? I'm trying to do the same ! – Vincent Wasteels Commented Nov 22, 2017 at 11:00
- 1 What is your end game here? I don't think there is such event as @keydown on non-input elements, especially on div. Can you please clarify what exactly do you have and what exactly needs to happen after you press ESC key? – yavor.vasilev Commented Sep 25, 2018 at 12:55
- 3 I want a modal to close @yavor.vasilev – vhflat Commented Feb 21, 2019 at 23:27
6 Answers
Reset to default 105For anyone who wanders here from google, in Vue 2...
<div @keydown.esc="something_in_your_methods"></div>
The secret for making keydown
events work on divs and other non-focusable elements is to add a tabindex
attribute:
<div tabindex="0"
@keydown.left="previousImage"
@keydown.right="nextImage" />
Now the div has become a focusable element and the key events will be triggered.
Here is more info on focusable elements and tabindex
What I did was go for a mixin.
The mixin in a file called close.js
export default {
created() {
let that = this;
document.addEventListener('keyup', function (evt) {
if (evt.keyCode === 27) {
that.close();
}
});
},
};
Import and use it in the desired component
import closeMixin from './../../mixins/close.js';
export default {
mixins: [closeMixin],
props: [],
computed: {}
methods: {
close(){
// closing logic
}
}
}
3 things to make sure of on the main element:
- It has a tabindex
- It or a descendant is focused
- It is listening for the event
Here is how I usually manage my modals:
<div ref="modal" @keyup.esc="close" tabindex="-1">
<!-- Modal content -->
</div>
mounted() {
this.$refs.modal.focus();
}
In my case, I created a directive, v-esc.ts. (※ This is Vue3 directive writing way)
import { Directive } from 'vue'
const directive: Directive = {
beforeMount(el, binding) {
el._keydownCallback = (event) => {
if (event.key === 'Escape') {
binding.value()
}
}
document.addEventListener('keydown', el._keydownCallback)
},
unmounted(el, binding) {
document.removeEventListener('keydown', el._keydownCallback)
delete el._keydownCallback
}
}
export const esc = { esc: directive }
Then I can use it in any component like this. (NOTE: you must pass a function param to v-esc, because the param executed as binding.value() in the directive)
<template>
<img
@click.prevent="close"
v-esc="close"
src="@/assets/icons/close.svg"
/>
</template>
<script lang="ts">
import { defineComponent } from 'vue'
import { esc } from '@/common/directives/v-esc'
export default defineComponent({
name: 'nitCloseButton',
...
methods: {
close() {
this.$emit('close')
}
},
directives: {
...esc
}
})
</script>
P.S One month after, I also need arrow left and arrow right keys. So, I've made this directive more general like this.
import { Directive } from 'vue'
const directive: Directive = {
beforeMount(el, binding) {
el._keydownCallback = event => {
console.log('keydown', event.key)
if (event.key === binding.arg) {
binding.value()
}
}
document.addEventListener('keydown', el._keydownCallback)
},
unmounted(el, binding) {
document.removeEventListener('keydown', el._keydownCallback)
delete el._keydownCallback
}
}
export const keydown = { keydown: directive }
You can detect any key's keydown by passing keyname as binding.args (v-keydown:{keyName} like below)
<button
v-keydown:ArrowLeft="moveToPreviousPage"
class="controller-button lo-center"
@click="moveToPreviousPage"
>
<arrow-icon :rotation="180" />
</button>
<button
v-keydown:ArrowRight="moveToNextPage"
class="controller-button lo-center"
@click="moveToNextPage"
export default defineComponent({
name: 'componentName',
directives: {
...keydown
}
...
})
You can't. Key events are dispatched from the body tag and Vue can't be mounted to the <body>
tag.
]
You'll have to set up your own event listener.
(image source & more info at When VueJS Can't Help You)
本文标签: javascriptListening to quotescquot key event on div component in VuejsStack Overflow
版权声明:本文标题:javascript - Listening to "esc" key event on div component in Vue.js - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1736912719a1956193.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论