fix(Dialog): skip keyboard events of child elements (#10285)

* fix(Dialog): skip keyboard events of child elements

* test: fix snapshot
This commit is contained in:
neverland 2022-02-11 15:25:38 +08:00 committed by GitHub
parent c1be4112e4
commit 0603479ee4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 27 additions and 13 deletions

View File

@ -1,13 +1,15 @@
import { import {
ref,
reactive, reactive,
withKeys,
defineComponent, defineComponent,
type PropType, type PropType,
type ExtractPropTypes, type ExtractPropTypes,
withKeys,
} from 'vue'; } from 'vue';
// Utils // Utils
import { import {
noop,
pick, pick,
extend, extend,
addUnit, addUnit,
@ -20,6 +22,7 @@ import {
makeStringProp, makeStringProp,
callInterceptor, callInterceptor,
createNamespace, createNamespace,
type ComponentInstance,
} from '../utils'; } from '../utils';
import { popupSharedProps, popupSharedPropKeys } from '../popup/shared'; import { popupSharedProps, popupSharedPropKeys } from '../popup/shared';
@ -72,9 +75,10 @@ export default defineComponent({
props: dialogProps, props: dialogProps,
emits: ['confirm', 'cancel', 'update:show', 'keydown'], emits: ['confirm', 'cancel', 'keydown', 'update:show'],
setup(props, { emit, slots }) { setup(props, { emit, slots }) {
const root = ref<ComponentInstance>();
const loading = reactive({ const loading = reactive({
confirm: false, confirm: false,
cancel: false, cancel: false,
@ -116,13 +120,17 @@ export default defineComponent({
const onConfirm = getActionHandler('confirm'); const onConfirm = getActionHandler('confirm');
const onKeydown = withKeys( const onKeydown = withKeys(
(event: KeyboardEvent) => { (event: KeyboardEvent) => {
// skip keyboard events of child elements
if (event.target !== root.value?.popupRef?.value) {
return;
}
const onEventType: Record<string, () => void> = { const onEventType: Record<string, () => void> = {
Enter: props.showConfirmButton ? onConfirm : () => {}, Enter: props.showConfirmButton ? onConfirm : noop,
Escape: props.showCancelButton ? onCancel : () => {}, Escape: props.showCancelButton ? onCancel : noop,
}; };
onEventType[event.key](); onEventType[event.key]();
emit('keydown', event); emit('keydown', event);
}, },
['enter', 'esc'] ['enter', 'esc']
@ -243,12 +251,13 @@ export default defineComponent({
const { width, title, theme, message, className } = props; const { width, title, theme, message, className } = props;
return ( return (
<Popup <Popup
tabindex={0} ref={root}
onKeydown={onKeydown}
role="dialog" role="dialog"
class={[bem([theme]), className]} class={[bem([theme]), className]}
style={{ width: addUnit(width) }} style={{ width: addUnit(width) }}
tabindex={0}
aria-labelledby={title || message} aria-labelledby={title || message}
onKeydown={onKeydown}
onUpdate:show={updateShow} onUpdate:show={updateShow}
{...pick(props, popupInheritKeys)} {...pick(props, popupInheritKeys)}
> >

View File

@ -96,8 +96,8 @@ exports[`should render demo and match snapshot 1`] = `
</transition-stub> </transition-stub>
<transition-stub> <transition-stub>
<div class="van-popup van-popup--center van-dialog" <div class="van-popup van-popup--center van-dialog"
tabindex="0"
role="dialog" role="dialog"
tabindex="0"
aria-labelledby="Title" aria-labelledby="Title"
style="display: none;" style="display: none;"
> >

View File

@ -31,8 +31,8 @@ exports[`should render default slot correctly 1`] = `
exports[`should render footer slot correctly 1`] = ` exports[`should render footer slot correctly 1`] = `
<div class="van-popup van-popup--center van-dialog" <div class="van-popup van-popup--center van-dialog"
tabindex="0"
role="dialog" role="dialog"
tabindex="0"
aria-labelledby="message" aria-labelledby="message"
> >
<div class="van-dialog__content van-dialog__content--isolated"> <div class="van-dialog__content van-dialog__content--isolated">

View File

@ -181,7 +181,6 @@ export default defineComponent({
return ( return (
<div <div
onKeydown={onKeydown}
v-show={props.show} v-show={props.show}
ref={popupRef} ref={popupRef}
style={style.value} style={style.value}
@ -192,6 +191,7 @@ export default defineComponent({
}), }),
{ 'van-safe-area-bottom': safeAreaInsetBottom }, { 'van-safe-area-bottom': safeAreaInsetBottom },
]} ]}
onKeydown={onKeydown}
{...attrs} {...attrs}
> >
{slots.default?.()} {slots.default?.()}
@ -222,10 +222,11 @@ export default defineComponent({
if (show && !opened) { if (show && !opened) {
open(); open();
attrs.tabindex === 0 && if (attrs.tabindex === 0) {
nextTick(() => { nextTick(() => {
popupRef.value?.focus(); popupRef.value?.focus();
}); });
}
} }
if (!show && opened) { if (!show && opened) {
opened = false; opened = false;

View File

@ -1,4 +1,4 @@
import type { ComponentPublicInstance } from 'vue'; import type { Ref, ComponentPublicInstance } from 'vue';
import type { PopupProps } from './Popup'; import type { PopupProps } from './Popup';
export type PopupPosition = 'top' | 'left' | 'bottom' | 'right' | 'center' | ''; export type PopupPosition = 'top' | 'left' | 'bottom' | 'right' | 'center' | '';
@ -9,4 +9,8 @@ export type PopupCloseIconPosition =
| 'bottom-left' | 'bottom-left'
| 'bottom-right'; | 'bottom-right';
export type PopupInstance = ComponentPublicInstance<PopupProps>; export type PopupExpose = {
popupRef: Ref<HTMLElement>;
};
export type PopupInstance = ComponentPublicInstance<PopupProps, PopupExpose>;