mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
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:
parent
c1be4112e4
commit
0603479ee4
@ -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)}
|
||||||
>
|
>
|
||||||
|
@ -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;"
|
||||||
>
|
>
|
||||||
|
@ -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">
|
||||||
|
@ -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;
|
||||||
|
@ -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>;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user