test(runtime-dom): add test coverage for v-on runtime guards, fix "exact" guard (#298)
This commit is contained in:
parent
3385480ba7
commit
db5c343c33
@ -90,4 +90,17 @@ describe('compiler-dom: transform v-on', () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should not wrap keys guard if no key modifier is present', () => {
|
||||||
|
const [prop] = parseVOnProperties(`<div @keyup.exact="test"/>`, {
|
||||||
|
prefixIdentifiers: true
|
||||||
|
})
|
||||||
|
expect(prop).toMatchObject({
|
||||||
|
type: NodeTypes.JS_PROPERTY,
|
||||||
|
value: {
|
||||||
|
callee: V_ON_MODIFIERS_GUARD,
|
||||||
|
arguments: [{ content: '_ctx.test' }, '["exact"]']
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
@ -41,15 +41,17 @@ export const transformOn: DirectiveTransform = (dir, node, context) => {
|
|||||||
value,
|
value,
|
||||||
JSON.stringify(runtimeModifiers.filter(m => m in NOT_KEY_MODIFIERS))
|
JSON.stringify(runtimeModifiers.filter(m => m in NOT_KEY_MODIFIERS))
|
||||||
])
|
])
|
||||||
|
const keyModifiers = runtimeModifiers.filter(m => !(m in NOT_KEY_MODIFIERS))
|
||||||
if (
|
if (
|
||||||
|
keyModifiers.length &&
|
||||||
// if event name is dynamic, always wrap with keys guard
|
// if event name is dynamic, always wrap with keys guard
|
||||||
key.type === NodeTypes.COMPOUND_EXPRESSION ||
|
(key.type === NodeTypes.COMPOUND_EXPRESSION ||
|
||||||
!key.isStatic ||
|
!key.isStatic ||
|
||||||
key.content.toLowerCase() in KEYBOARD_EVENTS
|
key.content.toLowerCase() in KEYBOARD_EVENTS)
|
||||||
) {
|
) {
|
||||||
handler = createCallExpression(context.helper(V_ON_KEYS_GUARD), [
|
handler = createCallExpression(context.helper(V_ON_KEYS_GUARD), [
|
||||||
handler,
|
handler,
|
||||||
JSON.stringify(runtimeModifiers.filter(m => !(m in NOT_KEY_MODIFIERS)))
|
JSON.stringify(keyModifiers)
|
||||||
])
|
])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { patchEvent } from '../../src/modules/events'
|
import { patchEvent } from '../../src/modules/events'
|
||||||
import { vOnModifiersGuard } from '@vue/runtime-dom'
|
import { vOnModifiersGuard, vOnKeysGuard } from '@vue/runtime-dom'
|
||||||
|
|
||||||
function triggerEvent(
|
function triggerEvent(
|
||||||
target: Element,
|
target: Element,
|
||||||
@ -17,41 +17,93 @@ function triggerEvent(
|
|||||||
}
|
}
|
||||||
|
|
||||||
describe('runtime-dom: v-on directive', () => {
|
describe('runtime-dom: v-on directive', () => {
|
||||||
test('it should support stop and prevent', async () => {
|
test('it should support "stop" and "prevent"', () => {
|
||||||
const parent = document.createElement('div')
|
const parent = document.createElement('div')
|
||||||
const child = document.createElement('input')
|
const child = document.createElement('input')
|
||||||
parent.appendChild(child)
|
parent.appendChild(child)
|
||||||
const childNextValue = {
|
const childNextValue = vOnModifiersGuard(jest.fn(), ['prevent', 'stop'])
|
||||||
handler: vOnModifiersGuard(jest.fn(), ['prevent', 'stop']),
|
|
||||||
options: {}
|
|
||||||
}
|
|
||||||
patchEvent(child, 'click', null, childNextValue, null)
|
patchEvent(child, 'click', null, childNextValue, null)
|
||||||
const parentHandler = jest.fn()
|
const parentNextValue = jest.fn()
|
||||||
const parentNextValue = { handler: parentHandler, options: {} }
|
|
||||||
patchEvent(parent, 'click', null, parentNextValue, null)
|
patchEvent(parent, 'click', null, parentNextValue, null)
|
||||||
expect(triggerEvent(child, 'click').defaultPrevented).toBe(true)
|
expect(triggerEvent(child, 'click').defaultPrevented).toBe(true)
|
||||||
expect(parentHandler).not.toBeCalled()
|
expect(parentNextValue).not.toBeCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('it should support "self"', () => {
|
||||||
|
const parent = document.createElement('div')
|
||||||
|
const child = document.createElement('input')
|
||||||
|
parent.appendChild(child)
|
||||||
|
const fn = jest.fn()
|
||||||
|
const handler = vOnModifiersGuard(fn, ['self'])
|
||||||
|
patchEvent(parent, 'click', null, handler, null)
|
||||||
|
triggerEvent(child, 'click')
|
||||||
|
expect(fn).not.toBeCalled()
|
||||||
})
|
})
|
||||||
|
|
||||||
test('it should support key modifiers and system modifiers', () => {
|
test('it should support key modifiers and system modifiers', () => {
|
||||||
const el = document.createElement('div')
|
const el = document.createElement('div')
|
||||||
const fn = jest.fn()
|
const fn = jest.fn()
|
||||||
const nextValue = {
|
// <div @keyup.ctrl.esc="test"/>
|
||||||
handler: vOnModifiersGuard(fn, ['ctrl', 'esc']),
|
const nextValue = vOnKeysGuard(vOnModifiersGuard(fn, ['ctrl']), ['esc'])
|
||||||
options: {}
|
patchEvent(el, 'keyup', null, nextValue, null)
|
||||||
}
|
triggerEvent(el, 'keyup', e => (e.key = 'a'))
|
||||||
patchEvent(el, 'click', null, nextValue, null)
|
expect(fn).not.toBeCalled()
|
||||||
triggerEvent(el, 'click', e => {
|
triggerEvent(el, 'keyup', e => {
|
||||||
e.ctrlKey = false
|
e.ctrlKey = false
|
||||||
e.key = 'esc'
|
e.key = 'esc'
|
||||||
})
|
})
|
||||||
expect(fn).not.toBeCalled()
|
expect(fn).not.toBeCalled()
|
||||||
triggerEvent(el, 'click', e => {
|
triggerEvent(el, 'keyup', e => {
|
||||||
e.ctrlKey = true
|
e.ctrlKey = true
|
||||||
e.key = 'Escape'
|
e.key = 'Escape'
|
||||||
})
|
})
|
||||||
expect(fn).toBeCalled()
|
expect(fn).toBeCalled()
|
||||||
})
|
})
|
||||||
|
|
||||||
test('it should support "exact" modifier', () => {})
|
test('it should support "exact" modifier', () => {
|
||||||
|
const el = document.createElement('div')
|
||||||
|
// Case 1: <div @keyup.exact="test"/>
|
||||||
|
const fn1 = jest.fn()
|
||||||
|
const next1 = vOnModifiersGuard(fn1, ['exact'])
|
||||||
|
patchEvent(el, 'keyup', null, next1, null)
|
||||||
|
triggerEvent(el, 'keyup')
|
||||||
|
expect(fn1.mock.calls.length).toBe(1)
|
||||||
|
triggerEvent(el, 'keyup', e => (e.ctrlKey = true))
|
||||||
|
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'])
|
||||||
|
patchEvent(el, 'keyup', null, next2, null)
|
||||||
|
triggerEvent(el, 'keyup', e => (e.key = 'a'))
|
||||||
|
expect(fn2).not.toBeCalled()
|
||||||
|
triggerEvent(el, 'keyup', e => {
|
||||||
|
e.key = 'a'
|
||||||
|
e.ctrlKey = true
|
||||||
|
})
|
||||||
|
expect(fn2.mock.calls.length).toBe(1)
|
||||||
|
triggerEvent(el, 'keyup', e => {
|
||||||
|
// should not trigger if has other system modifiers
|
||||||
|
e.key = 'a'
|
||||||
|
e.ctrlKey = true
|
||||||
|
e.altKey = true
|
||||||
|
})
|
||||||
|
expect(fn2.mock.calls.length).toBe(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should support mouse modifiers', () => {
|
||||||
|
const buttons = ['left', 'middle', 'right'] as const
|
||||||
|
const buttonCodes = { left: 0, middle: 1, right: 2 }
|
||||||
|
buttons.forEach(button => {
|
||||||
|
const el = document.createElement('div')
|
||||||
|
const fn = jest.fn()
|
||||||
|
const handler = vOnModifiersGuard(fn, [button])
|
||||||
|
patchEvent(el, 'mousedown', null, handler, null)
|
||||||
|
buttons.filter(b => b !== button).forEach(button => {
|
||||||
|
triggerEvent(el, 'mousedown', e => (e.button = buttonCodes[button]))
|
||||||
|
})
|
||||||
|
expect(fn).not.toBeCalled()
|
||||||
|
triggerEvent(el, 'mousedown', e => (e.button = buttonCodes[button]))
|
||||||
|
expect(fn).toBeCalled()
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
const systemModifiers = new Set(['ctrl', 'shift', 'alt', 'meta'])
|
const systemModifiers = ['ctrl', 'shift', 'alt', 'meta']
|
||||||
|
|
||||||
type KeyedEvent = KeyboardEvent | MouseEvent | TouchEvent;
|
type KeyedEvent = KeyboardEvent | MouseEvent | TouchEvent;
|
||||||
|
|
||||||
@ -16,8 +16,8 @@ const modifierGuards: Record<
|
|||||||
left: e => 'button' in e && (e as MouseEvent).button !== 0,
|
left: e => 'button' in e && (e as MouseEvent).button !== 0,
|
||||||
middle: e => 'button' in e && (e as MouseEvent).button !== 1,
|
middle: e => 'button' in e && (e as MouseEvent).button !== 1,
|
||||||
right: e => 'button' in e && (e as MouseEvent).button !== 2,
|
right: e => 'button' in e && (e as MouseEvent).button !== 2,
|
||||||
exact: (e, modifiers) =>
|
exact: (e, modifiers: string[]) =>
|
||||||
modifiers!.some(m => systemModifiers.has(m) && (e as any)[`${m}Key`])
|
systemModifiers.some(m => (e as any)[`${m}Key`] && !modifiers.includes(m))
|
||||||
}
|
}
|
||||||
|
|
||||||
export const vOnModifiersGuard = (fn: Function, modifiers: string[]) => {
|
export const vOnModifiersGuard = (fn: Function, modifiers: string[]) => {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user