content: ExpressionNode
}
-// always dynamic
export interface CompoundExpressionNode extends Node {
type: NodeTypes.COMPOUND_EXPRESSION
children: (
}
})
})
+
+ it('should not wrap normal guard if there is only keys guard', () => {
+ const [prop] = parseVOnProperties(`<div @keyup.enter="test"/>`, {
+ prefixIdentifiers: true
+ })
+ expect(prop).toMatchObject({
+ type: NodeTypes.JS_PROPERTY,
+ value: {
+ callee: V_ON_KEYS_GUARD,
+ arguments: [{ content: '_ctx.test' }, '["enter"]']
+ }
+ })
+ })
})
[V_MODEL_TEXT]: `vModelText`,
[V_MODEL_SELECT]: `vModelSelect`,
[V_MODEL_DYNAMIC]: `vModelDynamic`,
- [V_ON_MODIFIERS_GUARD]: `vOnModifiersGuard`,
- [V_ON_KEYS_GUARD]: `vOnKeysGuard`
+ [V_ON_MODIFIERS_GUARD]: `withModifiers`,
+ [V_ON_KEYS_GUARD]: `withKeys`
})
createCallExpression,
createObjectExpression,
createSimpleExpression,
- NodeTypes,
- CallExpression,
- ObjectExpression
+ NodeTypes
} from '@vue/compiler-core'
import { V_ON_MODIFIERS_GUARD, V_ON_KEYS_GUARD } from '../runtimeHelpers'
import { makeMap } from '@vue/shared'
const baseResult = baseTransform(dir, node, context)
if (!modifiers.length) return baseResult
- const { key, value } = baseResult.props[0]
+ let { key, value: handlerExp } = baseResult.props[0]
+
+ // modifiers for addEventListener() options, e.g. .passive & .capture
+ const eventOptionModifiers = modifiers.filter(isEventOptionModifier)
+ // modifiers that needs runtime guards
const runtimeModifiers = modifiers.filter(m => !isEventOptionModifier(m))
- let handler = createCallExpression(context.helper(V_ON_MODIFIERS_GUARD), [
- value,
- JSON.stringify(runtimeModifiers.filter(isNonKeyModifier))
- ])
+
+ // built-in modifiers that are not keys
+ const nonKeyModifiers = runtimeModifiers.filter(isNonKeyModifier)
+ if (nonKeyModifiers.length) {
+ handlerExp = createCallExpression(context.helper(V_ON_MODIFIERS_GUARD), [
+ handlerExp,
+ JSON.stringify(nonKeyModifiers)
+ ])
+ }
+
const keyModifiers = runtimeModifiers.filter(m => !isNonKeyModifier(m))
if (
keyModifiers.length &&
!key.isStatic ||
isKeyboardEvent(key.content))
) {
- handler = createCallExpression(context.helper(V_ON_KEYS_GUARD), [
- handler,
+ handlerExp = createCallExpression(context.helper(V_ON_KEYS_GUARD), [
+ handlerExp,
JSON.stringify(keyModifiers)
])
}
- let returnExp: CallExpression | ObjectExpression = handler
-
- const eventOptionModifiers = modifiers.filter(isEventOptionModifier)
if (eventOptionModifiers.length) {
- returnExp = createObjectExpression([
- createObjectProperty('handler', handler),
+ handlerExp = createObjectExpression([
+ createObjectProperty('handler', handlerExp),
createObjectProperty(
'options',
createObjectExpression(
}
return {
- props: [createObjectProperty(key, returnExp)],
+ props: [createObjectProperty(key, handlerExp)],
needRuntime: false
}
}
import { patchEvent } from '../../src/modules/events'
-import { vOnModifiersGuard, vOnKeysGuard } from '@vue/runtime-dom'
+import { withModifiers, withKeys } from '@vue/runtime-dom'
function triggerEvent(
target: Element,
const parent = document.createElement('div')
const child = document.createElement('input')
parent.appendChild(child)
- const childNextValue = vOnModifiersGuard(jest.fn(), ['prevent', 'stop'])
+ const childNextValue = withModifiers(jest.fn(), ['prevent', 'stop'])
patchEvent(child, 'click', null, childNextValue, null)
const parentNextValue = jest.fn()
patchEvent(parent, 'click', null, parentNextValue, null)
const child = document.createElement('input')
parent.appendChild(child)
const fn = jest.fn()
- const handler = vOnModifiersGuard(fn, ['self'])
+ const handler = withModifiers(fn, ['self'])
patchEvent(parent, 'click', null, handler, null)
triggerEvent(child, 'click')
expect(fn).not.toBeCalled()
const el = document.createElement('div')
const fn = jest.fn()
// <div @keyup.ctrl.esc="test"/>
- const nextValue = vOnKeysGuard(vOnModifiersGuard(fn, ['ctrl']), ['esc'])
+ const nextValue = withKeys(withModifiers(fn, ['ctrl']), ['esc'])
patchEvent(el, 'keyup', null, nextValue, null)
triggerEvent(el, 'keyup', e => (e.key = 'a'))
expect(fn).not.toBeCalled()
const el = document.createElement('div')
// Case 1: <div @keyup.exact="test"/>
const fn1 = jest.fn()
- const next1 = vOnModifiersGuard(fn1, ['exact'])
+ const next1 = withModifiers(fn1, ['exact'])
patchEvent(el, 'keyup', null, next1, null)
triggerEvent(el, 'keyup')
expect(fn1.mock.calls.length).toBe(1)
expect(fn1.mock.calls.length).toBe(1)
// Case 2: <div @keyup.ctrl.a.exact="test"/>
const fn2 = jest.fn()
- const next2 = vOnKeysGuard(vOnModifiersGuard(fn2, ['ctrl', 'exact']), ['a'])
+ const next2 = withKeys(withModifiers(fn2, ['ctrl', 'exact']), ['a'])
patchEvent(el, 'keyup', null, next2, null)
triggerEvent(el, 'keyup', e => (e.key = 'a'))
expect(fn2).not.toBeCalled()
buttons.forEach(button => {
const el = document.createElement('div')
const fn = jest.fn()
- const handler = vOnModifiersGuard(fn, [button])
+ const handler = withModifiers(fn, [button])
patchEvent(el, 'mousedown', null, handler, null)
buttons.filter(b => b !== button).forEach(button => {
triggerEvent(el, 'mousedown', e => (e.button = buttonCodes[button]))
const systemModifiers = ['ctrl', 'shift', 'alt', 'meta']
-type KeyedEvent = KeyboardEvent | MouseEvent | TouchEvent;
+type KeyedEvent = KeyboardEvent | MouseEvent | TouchEvent
const modifierGuards: Record<
string,
systemModifiers.some(m => (e as any)[`${m}Key`] && !modifiers.includes(m))
}
-export const vOnModifiersGuard = (fn: Function, modifiers: string[]) => {
+export const withModifiers = (fn: Function, modifiers: string[]) => {
return (event: Event) => {
for (let i = 0; i < modifiers.length; i++) {
const guard = modifierGuards[modifiers[i]]
delete: 'backspace'
}
-export const vOnKeysGuard = (fn: Function, modifiers: string[]) => {
+export const withKeys = (fn: Function, modifiers: string[]) => {
return (event: KeyboardEvent) => {
if (!('key' in event)) return
const eventKey = event.key.toLowerCase()
vModelDynamic
} from './directives/vModel'
-export { vOnModifiersGuard, vOnKeysGuard } from './directives/vOn'
+export { withModifiers, withKeys } from './directives/vOn'
// re-export everything from core
// h, Component, reactivity API, nextTick, flags & types