fix(v-on): refactor DOM event options modifer handling
fix #1567 Previously multiple `v-on` handlers with different event attach option modifers (`.once`, `.capture` and `.passive`) are generated as an array of objects in the form of `[{ handler, options }]` - however, this makes it pretty complex for `runtime-dom` to properly handle all possible value permutations, as each handler may need to be attached with different options. With this commit, they are now generated as event props with different keys - e.g. `v-on:click.capture` is now generated as a prop named `onClick.capture`. This allows them to be patched as separate props which makes the runtime handling much simpler.
This commit is contained in:
parent
9152a89016
commit
380c6792d8
@ -6,7 +6,9 @@ import {
|
|||||||
CompilerOptions,
|
CompilerOptions,
|
||||||
ErrorCodes,
|
ErrorCodes,
|
||||||
NodeTypes,
|
NodeTypes,
|
||||||
VNodeCall
|
VNodeCall,
|
||||||
|
helperNameMap,
|
||||||
|
CAPITALIZE
|
||||||
} from '../../src'
|
} from '../../src'
|
||||||
import { transformOn } from '../../src/transforms/vOn'
|
import { transformOn } from '../../src/transforms/vOn'
|
||||||
import { transformElement } from '../../src/transforms/transformElement'
|
import { transformElement } from '../../src/transforms/transformElement'
|
||||||
@ -73,7 +75,11 @@ describe('compiler: transform v-on', () => {
|
|||||||
{
|
{
|
||||||
key: {
|
key: {
|
||||||
type: NodeTypes.COMPOUND_EXPRESSION,
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
||||||
children: [`"on" + (`, { content: `event` }, `)`]
|
children: [
|
||||||
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: `event` },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
},
|
},
|
||||||
value: {
|
value: {
|
||||||
type: NodeTypes.SIMPLE_EXPRESSION,
|
type: NodeTypes.SIMPLE_EXPRESSION,
|
||||||
@ -94,7 +100,11 @@ describe('compiler: transform v-on', () => {
|
|||||||
{
|
{
|
||||||
key: {
|
key: {
|
||||||
type: NodeTypes.COMPOUND_EXPRESSION,
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
||||||
children: [`"on" + (`, { content: `_ctx.event` }, `)`]
|
children: [
|
||||||
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: `_ctx.event` },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
},
|
},
|
||||||
value: {
|
value: {
|
||||||
type: NodeTypes.SIMPLE_EXPRESSION,
|
type: NodeTypes.SIMPLE_EXPRESSION,
|
||||||
@ -116,7 +126,7 @@ describe('compiler: transform v-on', () => {
|
|||||||
key: {
|
key: {
|
||||||
type: NodeTypes.COMPOUND_EXPRESSION,
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
||||||
children: [
|
children: [
|
||||||
`"on" + (`,
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
{ content: `_ctx.event` },
|
{ content: `_ctx.event` },
|
||||||
`(`,
|
`(`,
|
||||||
{ content: `_ctx.foo` },
|
{ content: `_ctx.foo` },
|
||||||
|
@ -5,16 +5,15 @@ import {
|
|||||||
ElementNode,
|
ElementNode,
|
||||||
ObjectExpression,
|
ObjectExpression,
|
||||||
NodeTypes,
|
NodeTypes,
|
||||||
VNodeCall
|
VNodeCall,
|
||||||
|
helperNameMap,
|
||||||
|
CAPITALIZE
|
||||||
} from '@vue/compiler-core'
|
} from '@vue/compiler-core'
|
||||||
import { transformOn } from '../../src/transforms/vOn'
|
import { transformOn } from '../../src/transforms/vOn'
|
||||||
import { V_ON_WITH_MODIFIERS, V_ON_WITH_KEYS } from '../../src/runtimeHelpers'
|
import { V_ON_WITH_MODIFIERS, V_ON_WITH_KEYS } from '../../src/runtimeHelpers'
|
||||||
import { transformElement } from '../../../compiler-core/src/transforms/transformElement'
|
import { transformElement } from '../../../compiler-core/src/transforms/transformElement'
|
||||||
import { transformExpression } from '../../../compiler-core/src/transforms/transformExpression'
|
import { transformExpression } from '../../../compiler-core/src/transforms/transformExpression'
|
||||||
import {
|
import { genFlagText } from '../../../compiler-core/__tests__/testUtils'
|
||||||
createObjectMatcher,
|
|
||||||
genFlagText
|
|
||||||
} from '../../../compiler-core/__tests__/testUtils'
|
|
||||||
import { PatchFlags } from '@vue/shared'
|
import { PatchFlags } from '@vue/shared'
|
||||||
|
|
||||||
function parseWithVOn(template: string, options: CompilerOptions = {}) {
|
function parseWithVOn(template: string, options: CompilerOptions = {}) {
|
||||||
@ -83,42 +82,37 @@ describe('compiler-dom: transform v-on', () => {
|
|||||||
})
|
})
|
||||||
expect(prop).toMatchObject({
|
expect(prop).toMatchObject({
|
||||||
type: NodeTypes.JS_PROPERTY,
|
type: NodeTypes.JS_PROPERTY,
|
||||||
value: createObjectMatcher({
|
key: {
|
||||||
handler: {
|
content: `onClick.capture.passive`
|
||||||
callee: V_ON_WITH_MODIFIERS,
|
},
|
||||||
arguments: [{ content: '_ctx.test' }, '["stop"]']
|
value: {
|
||||||
},
|
callee: V_ON_WITH_MODIFIERS,
|
||||||
options: createObjectMatcher({
|
arguments: [{ content: '_ctx.test' }, '["stop"]']
|
||||||
capture: { content: 'true', isStatic: false },
|
}
|
||||||
passive: { content: 'true', isStatic: false }
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should wrap keys guard for keyboard events or dynamic events', () => {
|
it('should wrap keys guard for keyboard events or dynamic events', () => {
|
||||||
const {
|
const {
|
||||||
props: [prop]
|
props: [prop]
|
||||||
} = parseWithVOn(`<div @keyDown.stop.capture.ctrl.a="test"/>`, {
|
} = parseWithVOn(`<div @keydown.stop.capture.ctrl.a="test"/>`, {
|
||||||
prefixIdentifiers: true
|
prefixIdentifiers: true
|
||||||
})
|
})
|
||||||
expect(prop).toMatchObject({
|
expect(prop).toMatchObject({
|
||||||
type: NodeTypes.JS_PROPERTY,
|
type: NodeTypes.JS_PROPERTY,
|
||||||
value: createObjectMatcher({
|
key: {
|
||||||
handler: {
|
content: `onKeydown.capture`
|
||||||
callee: V_ON_WITH_KEYS,
|
},
|
||||||
arguments: [
|
value: {
|
||||||
{
|
callee: V_ON_WITH_KEYS,
|
||||||
callee: V_ON_WITH_MODIFIERS,
|
arguments: [
|
||||||
arguments: [{ content: '_ctx.test' }, '["stop","ctrl"]']
|
{
|
||||||
},
|
callee: V_ON_WITH_MODIFIERS,
|
||||||
'["a"]'
|
arguments: [{ content: '_ctx.test' }, '["stop","ctrl"]']
|
||||||
]
|
},
|
||||||
},
|
'["a"]'
|
||||||
options: createObjectMatcher({
|
]
|
||||||
capture: { content: 'true', isStatic: false }
|
}
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -206,9 +200,21 @@ describe('compiler-dom: transform v-on', () => {
|
|||||||
type: NodeTypes.COMPOUND_EXPRESSION,
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
||||||
children: [
|
children: [
|
||||||
`(`,
|
`(`,
|
||||||
{ children: [`"on" + (`, { content: 'event' }, `)`] },
|
{
|
||||||
`).toLowerCase() === "onclick" ? "onContextmenu" : (`,
|
children: [
|
||||||
{ children: [`"on" + (`, { content: 'event' }, `)`] },
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: 'event' },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
|
},
|
||||||
|
`) === "onClick" ? "onContextmenu" : (`,
|
||||||
|
{
|
||||||
|
children: [
|
||||||
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: 'event' },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
|
},
|
||||||
`)`
|
`)`
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
@ -232,9 +238,21 @@ describe('compiler-dom: transform v-on', () => {
|
|||||||
type: NodeTypes.COMPOUND_EXPRESSION,
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
||||||
children: [
|
children: [
|
||||||
`(`,
|
`(`,
|
||||||
{ children: [`"on" + (`, { content: 'event' }, `)`] },
|
{
|
||||||
`).toLowerCase() === "onclick" ? "onMouseup" : (`,
|
children: [
|
||||||
{ children: [`"on" + (`, { content: 'event' }, `)`] },
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: 'event' },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
|
},
|
||||||
|
`) === "onClick" ? "onMouseup" : (`,
|
||||||
|
{
|
||||||
|
children: [
|
||||||
|
`"on" + _${helperNameMap[CAPITALIZE]}(`,
|
||||||
|
{ content: 'event' },
|
||||||
|
`)`
|
||||||
|
]
|
||||||
|
},
|
||||||
`)`
|
`)`
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
@ -254,24 +272,17 @@ describe('compiler-dom: transform v-on', () => {
|
|||||||
expect((root as any).children[0].codegenNode.patchFlag).toBe(
|
expect((root as any).children[0].codegenNode.patchFlag).toBe(
|
||||||
genFlagText(PatchFlags.HYDRATE_EVENTS)
|
genFlagText(PatchFlags.HYDRATE_EVENTS)
|
||||||
)
|
)
|
||||||
expect(prop.value).toMatchObject({
|
expect(prop).toMatchObject({
|
||||||
type: NodeTypes.JS_CACHE_EXPRESSION,
|
key: {
|
||||||
index: 1,
|
content: `onKeyup.capture`
|
||||||
|
},
|
||||||
value: {
|
value: {
|
||||||
type: NodeTypes.JS_OBJECT_EXPRESSION,
|
type: NodeTypes.JS_CACHE_EXPRESSION,
|
||||||
properties: [
|
index: 1,
|
||||||
{
|
value: {
|
||||||
key: { content: 'handler' },
|
type: NodeTypes.JS_CALL_EXPRESSION,
|
||||||
value: {
|
callee: V_ON_WITH_KEYS
|
||||||
type: NodeTypes.JS_CALL_EXPRESSION,
|
}
|
||||||
callee: V_ON_WITH_KEYS
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: { content: 'options' },
|
|
||||||
value: { type: NodeTypes.JS_OBJECT_EXPRESSION }
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -3,7 +3,6 @@ import {
|
|||||||
DirectiveTransform,
|
DirectiveTransform,
|
||||||
createObjectProperty,
|
createObjectProperty,
|
||||||
createCallExpression,
|
createCallExpression,
|
||||||
createObjectExpression,
|
|
||||||
createSimpleExpression,
|
createSimpleExpression,
|
||||||
NodeTypes,
|
NodeTypes,
|
||||||
createCompoundExpression,
|
createCompoundExpression,
|
||||||
@ -80,7 +79,7 @@ const transformClick = (key: ExpressionNode, event: string) => {
|
|||||||
? createCompoundExpression([
|
? createCompoundExpression([
|
||||||
`(`,
|
`(`,
|
||||||
key,
|
key,
|
||||||
`).toLowerCase() === "onclick" ? "${event}" : (`,
|
`) === "onClick" ? "${event}" : (`,
|
||||||
key,
|
key,
|
||||||
`)`
|
`)`
|
||||||
])
|
])
|
||||||
@ -126,20 +125,16 @@ export const transformOn: DirectiveTransform = (dir, node, context) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (eventOptionModifiers.length) {
|
if (eventOptionModifiers.length) {
|
||||||
handlerExp = createObjectExpression([
|
key = isStaticExp(key)
|
||||||
createObjectProperty('handler', handlerExp),
|
? createSimpleExpression(
|
||||||
createObjectProperty(
|
`${key.content}.${eventOptionModifiers.join(`.`)}`,
|
||||||
'options',
|
true
|
||||||
createObjectExpression(
|
|
||||||
eventOptionModifiers.map(modifier =>
|
|
||||||
createObjectProperty(
|
|
||||||
modifier,
|
|
||||||
createSimpleExpression('true', false)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
)
|
: createCompoundExpression([
|
||||||
])
|
`(`,
|
||||||
|
key,
|
||||||
|
`) + ".${eventOptionModifiers.join(`.`)}"`
|
||||||
|
])
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -160,30 +160,61 @@ describe('component: emit', () => {
|
|||||||
expect(`event validation failed for event "foo"`).toHaveBeenWarned()
|
expect(`event validation failed for event "foo"`).toHaveBeenWarned()
|
||||||
})
|
})
|
||||||
|
|
||||||
test('isEmitListener', () => {
|
test('.once', () => {
|
||||||
const def1 = { emits: ['click'] }
|
const Foo = defineComponent({
|
||||||
expect(isEmitListener(def1, 'onClick')).toBe(true)
|
render() {},
|
||||||
expect(isEmitListener(def1, 'onclick')).toBe(false)
|
emits: {
|
||||||
expect(isEmitListener(def1, 'onBlick')).toBe(false)
|
foo: null
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.$emit('foo')
|
||||||
|
this.$emit('foo')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const fn = jest.fn()
|
||||||
|
render(
|
||||||
|
h(Foo, {
|
||||||
|
'onFoo.once': fn
|
||||||
|
}),
|
||||||
|
nodeOps.createElement('div')
|
||||||
|
)
|
||||||
|
expect(fn).toHaveBeenCalledTimes(1)
|
||||||
|
})
|
||||||
|
|
||||||
const def2 = { emits: { click: null } }
|
describe('isEmitListener', () => {
|
||||||
expect(isEmitListener(def2, 'onClick')).toBe(true)
|
test('array option', () => {
|
||||||
expect(isEmitListener(def2, 'onclick')).toBe(false)
|
const def1 = { emits: ['click'] }
|
||||||
expect(isEmitListener(def2, 'onBlick')).toBe(false)
|
expect(isEmitListener(def1, 'onClick')).toBe(true)
|
||||||
|
expect(isEmitListener(def1, 'onclick')).toBe(false)
|
||||||
|
expect(isEmitListener(def1, 'onBlick')).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
const mixin1 = { emits: ['foo'] }
|
test('object option', () => {
|
||||||
const mixin2 = { emits: ['bar'] }
|
const def2 = { emits: { click: null } }
|
||||||
const extend = { emits: ['baz'] }
|
expect(isEmitListener(def2, 'onClick')).toBe(true)
|
||||||
const def3 = {
|
expect(isEmitListener(def2, 'onclick')).toBe(false)
|
||||||
emits: { click: null },
|
expect(isEmitListener(def2, 'onBlick')).toBe(false)
|
||||||
mixins: [mixin1, mixin2],
|
})
|
||||||
extends: extend
|
|
||||||
}
|
test('with mixins and extends', () => {
|
||||||
expect(isEmitListener(def3, 'onClick')).toBe(true)
|
const mixin1 = { emits: ['foo'] }
|
||||||
expect(isEmitListener(def3, 'onFoo')).toBe(true)
|
const mixin2 = { emits: ['bar'] }
|
||||||
expect(isEmitListener(def3, 'onBar')).toBe(true)
|
const extend = { emits: ['baz'] }
|
||||||
expect(isEmitListener(def3, 'onBaz')).toBe(true)
|
const def3 = {
|
||||||
expect(isEmitListener(def3, 'onclick')).toBe(false)
|
mixins: [mixin1, mixin2],
|
||||||
expect(isEmitListener(def3, 'onBlick')).toBe(false)
|
extends: extend
|
||||||
|
}
|
||||||
|
expect(isEmitListener(def3, 'onFoo')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onBar')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onBaz')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onclick')).toBe(false)
|
||||||
|
expect(isEmitListener(def3, 'onBlick')).toBe(false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('.once listeners', () => {
|
||||||
|
const def2 = { emits: { click: null } }
|
||||||
|
expect(isEmitListener(def2, 'onClick.once')).toBe(true)
|
||||||
|
expect(isEmitListener(def2, 'onclick.once')).toBe(false)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -246,6 +246,8 @@ export interface ComponentInternalInstance {
|
|||||||
slots: InternalSlots
|
slots: InternalSlots
|
||||||
refs: Data
|
refs: Data
|
||||||
emit: EmitFn
|
emit: EmitFn
|
||||||
|
// used for keeping track of .once event handlers on components
|
||||||
|
emitted: Record<string, boolean> | null
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* setup related
|
* setup related
|
||||||
@ -396,7 +398,8 @@ export function createComponentInstance(
|
|||||||
rtg: null,
|
rtg: null,
|
||||||
rtc: null,
|
rtc: null,
|
||||||
ec: null,
|
ec: null,
|
||||||
emit: null as any // to be set immediately
|
emit: null as any, // to be set immediately
|
||||||
|
emitted: null
|
||||||
}
|
}
|
||||||
if (__DEV__) {
|
if (__DEV__) {
|
||||||
instance.ctx = createRenderContext(instance)
|
instance.ctx = createRenderContext(instance)
|
||||||
|
@ -67,12 +67,21 @@ export function emit(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let handler = props[`on${capitalize(event)}`]
|
let handlerName = `on${capitalize(event)}`
|
||||||
|
let handler = props[handlerName]
|
||||||
// for v-model update:xxx events, also trigger kebab-case equivalent
|
// for v-model update:xxx events, also trigger kebab-case equivalent
|
||||||
// for props passed via kebab-case
|
// for props passed via kebab-case
|
||||||
if (!handler && event.startsWith('update:')) {
|
if (!handler && event.startsWith('update:')) {
|
||||||
event = hyphenate(event)
|
handlerName = `on${capitalize(hyphenate(event))}`
|
||||||
handler = props[`on${capitalize(event)}`]
|
handler = props[handlerName]
|
||||||
|
}
|
||||||
|
if (!handler) {
|
||||||
|
handler = props[handlerName + `.once`]
|
||||||
|
if (!instance.emitted) {
|
||||||
|
;(instance.emitted = {} as Record<string, boolean>)[handlerName] = true
|
||||||
|
} else if (instance.emitted[handlerName]) {
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (handler) {
|
if (handler) {
|
||||||
callWithAsyncErrorHandling(
|
callWithAsyncErrorHandling(
|
||||||
@ -123,13 +132,13 @@ function normalizeEmitsOptions(
|
|||||||
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
||||||
// both considered matched listeners.
|
// both considered matched listeners.
|
||||||
export function isEmitListener(comp: Component, key: string): boolean {
|
export function isEmitListener(comp: Component, key: string): boolean {
|
||||||
if (!isOn(key)) {
|
let emits: ObjectEmitsOptions | undefined
|
||||||
|
if (!isOn(key) || !(emits = normalizeEmitsOptions(comp))) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
const emits = normalizeEmitsOptions(comp)
|
key = key.replace(/\.once$/, '')
|
||||||
return (
|
return (
|
||||||
!!emits &&
|
hasOwn(emits, key[2].toLowerCase() + key.slice(3)) ||
|
||||||
(hasOwn(emits, key[2].toLowerCase() + key.slice(3)) ||
|
hasOwn(emits, key.slice(2))
|
||||||
hasOwn(emits, key.slice(2)))
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -57,17 +57,11 @@ describe(`runtime-dom: events patching`, () => {
|
|||||||
expect(fn).not.toHaveBeenCalled()
|
expect(fn).not.toHaveBeenCalled()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should support event options', async () => {
|
it('should support event option modifiers', async () => {
|
||||||
const el = document.createElement('div')
|
const el = document.createElement('div')
|
||||||
const event = new Event('click')
|
const event = new Event('click')
|
||||||
const fn = jest.fn()
|
const fn = jest.fn()
|
||||||
const nextValue = {
|
patchProp(el, 'onClick.once.capture', null, fn)
|
||||||
handler: fn,
|
|
||||||
options: {
|
|
||||||
once: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
patchProp(el, 'onClick', null, nextValue)
|
|
||||||
el.dispatchEvent(event)
|
el.dispatchEvent(event)
|
||||||
await timeout()
|
await timeout()
|
||||||
el.dispatchEvent(event)
|
el.dispatchEvent(event)
|
||||||
@ -75,39 +69,12 @@ describe(`runtime-dom: events patching`, () => {
|
|||||||
expect(fn).toHaveBeenCalledTimes(1)
|
expect(fn).toHaveBeenCalledTimes(1)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should support varying event options', async () => {
|
|
||||||
const el = document.createElement('div')
|
|
||||||
const event = new Event('click')
|
|
||||||
const prevFn = jest.fn()
|
|
||||||
const nextFn = jest.fn()
|
|
||||||
const nextValue = {
|
|
||||||
handler: nextFn,
|
|
||||||
options: {
|
|
||||||
once: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
patchProp(el, 'onClick', null, prevFn)
|
|
||||||
patchProp(el, 'onClick', prevFn, nextValue)
|
|
||||||
el.dispatchEvent(event)
|
|
||||||
await timeout()
|
|
||||||
el.dispatchEvent(event)
|
|
||||||
await timeout()
|
|
||||||
expect(prevFn).not.toHaveBeenCalled()
|
|
||||||
expect(nextFn).toHaveBeenCalledTimes(1)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should unassign event handler with options', async () => {
|
it('should unassign event handler with options', async () => {
|
||||||
const el = document.createElement('div')
|
const el = document.createElement('div')
|
||||||
const event = new Event('click')
|
const event = new Event('click')
|
||||||
const fn = jest.fn()
|
const fn = jest.fn()
|
||||||
const nextValue = {
|
patchProp(el, 'onClick.capture', null, fn)
|
||||||
handler: fn,
|
patchProp(el, 'onClick.capture', fn, null)
|
||||||
options: {
|
|
||||||
once: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
patchProp(el, 'onClick', null, nextValue)
|
|
||||||
patchProp(el, 'onClick', nextValue, null)
|
|
||||||
el.dispatchEvent(event)
|
el.dispatchEvent(event)
|
||||||
await timeout()
|
await timeout()
|
||||||
el.dispatchEvent(event)
|
el.dispatchEvent(event)
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { EMPTY_OBJ, isArray } from '@vue/shared'
|
import { isArray } from '@vue/shared'
|
||||||
import {
|
import {
|
||||||
ComponentInternalInstance,
|
ComponentInternalInstance,
|
||||||
callWithAsyncErrorHandling
|
callWithAsyncErrorHandling
|
||||||
@ -14,12 +14,6 @@ type EventValue = (Function | Function[]) & {
|
|||||||
invoker?: Invoker | null
|
invoker?: Invoker | null
|
||||||
}
|
}
|
||||||
|
|
||||||
type EventValueWithOptions = {
|
|
||||||
handler: EventValue
|
|
||||||
options: AddEventListenerOptions
|
|
||||||
invoker?: Invoker | null
|
|
||||||
}
|
|
||||||
|
|
||||||
// Async edge case fix requires storing an event listener's attach timestamp.
|
// Async edge case fix requires storing an event listener's attach timestamp.
|
||||||
let _getNow: () => number = Date.now
|
let _getNow: () => number = Date.now
|
||||||
|
|
||||||
@ -67,52 +61,43 @@ export function removeEventListener(
|
|||||||
export function patchEvent(
|
export function patchEvent(
|
||||||
el: Element,
|
el: Element,
|
||||||
rawName: string,
|
rawName: string,
|
||||||
prevValue: EventValueWithOptions | EventValue | null,
|
prevValue: EventValue | null,
|
||||||
nextValue: EventValueWithOptions | EventValue | null,
|
nextValue: EventValue | null,
|
||||||
instance: ComponentInternalInstance | null = null
|
instance: ComponentInternalInstance | null = null
|
||||||
) {
|
) {
|
||||||
const name = rawName.slice(2).toLowerCase()
|
|
||||||
const prevOptions = prevValue && 'options' in prevValue && prevValue.options
|
|
||||||
const nextOptions = nextValue && 'options' in nextValue && nextValue.options
|
|
||||||
const invoker = prevValue && prevValue.invoker
|
const invoker = prevValue && prevValue.invoker
|
||||||
const value =
|
if (nextValue && invoker) {
|
||||||
nextValue && 'handler' in nextValue ? nextValue.handler : nextValue
|
// patch
|
||||||
|
;(prevValue as EventValue).invoker = null
|
||||||
if (prevOptions || nextOptions) {
|
invoker.value = nextValue
|
||||||
const prev = prevOptions || EMPTY_OBJ
|
nextValue.invoker = invoker
|
||||||
const next = nextOptions || EMPTY_OBJ
|
} else {
|
||||||
if (
|
const [name, options] = parseName(rawName)
|
||||||
prev.capture !== next.capture ||
|
if (nextValue) {
|
||||||
prev.passive !== next.passive ||
|
addEventListener(el, name, createInvoker(nextValue, instance), options)
|
||||||
prev.once !== next.once
|
} else if (invoker) {
|
||||||
) {
|
// remove
|
||||||
if (invoker) {
|
removeEventListener(el, name, invoker, options)
|
||||||
removeEventListener(el, name, invoker, prev)
|
|
||||||
}
|
|
||||||
if (nextValue && value) {
|
|
||||||
const invoker = createInvoker(value, instance)
|
|
||||||
nextValue.invoker = invoker
|
|
||||||
addEventListener(el, name, invoker, next)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (nextValue && value) {
|
const optionsModifierRE = /\.(once|passive|capture)\b/g
|
||||||
if (invoker) {
|
|
||||||
;(prevValue as EventValue).invoker = null
|
function parseName(name: string): [string, EventListenerOptions | undefined] {
|
||||||
invoker.value = value
|
name = name.slice(2).toLowerCase()
|
||||||
nextValue.invoker = invoker
|
if (optionsModifierRE.test(name)) {
|
||||||
} else {
|
const options: EventListenerOptions = {}
|
||||||
addEventListener(
|
name = name.replace(
|
||||||
el,
|
optionsModifierRE,
|
||||||
name,
|
(_, key: keyof EventListenerOptions) => {
|
||||||
createInvoker(value, instance),
|
options[key] = true
|
||||||
nextOptions || void 0
|
return ''
|
||||||
)
|
}
|
||||||
}
|
)
|
||||||
} else if (invoker) {
|
return [name, options]
|
||||||
removeEventListener(el, name, invoker, prevOptions || void 0)
|
} else {
|
||||||
|
return [name, undefined]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user