Skip to content

Commit 0739f89

Browse files
committedAug 30, 2022
fix(custom-element): fix event listeners with capital letter event names on custom elements
close vuejs/docs#1708 close vuejs/docs#1890
1 parent 9f8f07e commit 0739f89

File tree

7 files changed

+99
-12
lines changed

7 files changed

+99
-12
lines changed
 

‎packages/compiler-core/__tests__/transforms/transformElement.spec.ts

+33-1
Original file line numberDiff line numberDiff line change
@@ -314,6 +314,37 @@ describe('compiler: element transform', () => {
314314
)
315315
expect(root.helpers).toContain(MERGE_PROPS)
316316

317+
expect(node.props).toMatchObject({
318+
type: NodeTypes.JS_CALL_EXPRESSION,
319+
callee: MERGE_PROPS,
320+
arguments: [
321+
createObjectMatcher({
322+
id: 'foo'
323+
}),
324+
{
325+
type: NodeTypes.JS_CALL_EXPRESSION,
326+
callee: TO_HANDLERS,
327+
arguments: [
328+
{
329+
type: NodeTypes.SIMPLE_EXPRESSION,
330+
content: `obj`
331+
},
332+
`true`
333+
]
334+
},
335+
createObjectMatcher({
336+
class: 'bar'
337+
})
338+
]
339+
})
340+
})
341+
342+
test('v-on="obj" on component', () => {
343+
const { root, node } = parseWithElementTransform(
344+
`<Foo id="foo" v-on="obj" class="bar" />`
345+
)
346+
expect(root.helpers).toContain(MERGE_PROPS)
347+
317348
expect(node.props).toMatchObject({
318349
type: NodeTypes.JS_CALL_EXPRESSION,
319350
callee: MERGE_PROPS,
@@ -358,7 +389,8 @@ describe('compiler: element transform', () => {
358389
{
359390
type: NodeTypes.SIMPLE_EXPRESSION,
360391
content: `handlers`
361-
}
392+
},
393+
`true`
362394
]
363395
},
364396
{

‎packages/compiler-core/src/transforms/transformElement.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -647,7 +647,7 @@ export function buildProps(
647647
type: NodeTypes.JS_CALL_EXPRESSION,
648648
loc,
649649
callee: context.helper(TO_HANDLERS),
650-
arguments: [exp]
650+
arguments: isComponent ? [exp] : [exp, `true`]
651651
})
652652
}
653653
} else {

‎packages/compiler-core/src/transforms/vOn.ts

+11-6
Original file line numberDiff line numberDiff line change
@@ -47,12 +47,17 @@ export const transformOn: DirectiveTransform = (
4747
if (rawName.startsWith('vue:')) {
4848
rawName = `vnode-${rawName.slice(4)}`
4949
}
50-
// for all event listeners, auto convert it to camelCase. See issue #2249
51-
eventName = createSimpleExpression(
52-
toHandlerKey(camelize(rawName)),
53-
true,
54-
arg.loc
55-
)
50+
const eventString =
51+
node.tagType === ElementTypes.COMPONENT ||
52+
rawName.startsWith('vnode') ||
53+
!/[A-Z]/.test(rawName)
54+
? // for component and vnode lifecycle event listeners, auto convert
55+
// it to camelCase. See issue #2249
56+
toHandlerKey(camelize(rawName))
57+
// preserve case for plain element listeners that have uppercase
58+
// letters, as these may be custom elements' custom events
59+
: `on:${rawName}`
60+
eventName = createSimpleExpression(eventString, true, arg.loc)
5661
} else {
5762
// #2388
5863
eventName = createCompoundExpression([

‎packages/runtime-core/src/helpers/toHandlers.ts

+9-2
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,21 @@ import { warn } from '../warning'
55
* For prefixing keys in v-on="obj" with "on"
66
* @private
77
*/
8-
export function toHandlers(obj: Record<string, any>): Record<string, any> {
8+
export function toHandlers(
9+
obj: Record<string, any>,
10+
preserveCaseIfNecessary?: boolean
11+
): Record<string, any> {
912
const ret: Record<string, any> = {}
1013
if (__DEV__ && !isObject(obj)) {
1114
warn(`v-on with no argument expects an object value.`)
1215
return ret
1316
}
1417
for (const key in obj) {
15-
ret[toHandlerKey(key)] = obj[key]
18+
ret[
19+
preserveCaseIfNecessary && /[A-Z]/.test(key)
20+
? `on:${key}`
21+
: toHandlerKey(key)
22+
] = obj[key]
1623
}
1724
return ret
1825
}

‎packages/runtime-dom/src/modules/events.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -101,7 +101,8 @@ function parseName(name: string): [string, EventListenerOptions | undefined] {
101101
;(options as any)[m[0].toLowerCase()] = true
102102
}
103103
}
104-
return [hyphenate(name.slice(2)), options]
104+
const event = name[2] === ':' ? name.slice(3) : hyphenate(name.slice(2))
105+
return [event, options]
105106
}
106107

107108
function createInvoker(

‎packages/runtime-test/src/patchProp.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ export function patchProp(
1616
})
1717
el.props[key] = nextValue
1818
if (isOn(key)) {
19-
const event = key.slice(2).toLowerCase()
19+
const event = key[2] === ':' ? key.slice(3) : key.slice(2).toLowerCase()
2020
;(el.eventListeners || (el.eventListeners = {}))[event] = nextValue
2121
}
2222
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
import { createApp } from '../src'
2+
3+
// https://github.com/vuejs/docs/pull/1890
4+
// https://github.com/vuejs/core/issues/5401
5+
// https://github.com/vuejs/docs/issues/1708
6+
test('custom element event casing', () => {
7+
customElements.define(
8+
'custom-event-casing',
9+
class Foo extends HTMLElement {
10+
connectedCallback() {
11+
this.dispatchEvent(new Event('camelCase'))
12+
this.dispatchEvent(new Event('CAPScase'))
13+
this.dispatchEvent(new Event('PascalCase'))
14+
}
15+
}
16+
)
17+
18+
const container = document.createElement('div')
19+
document.body.appendChild(container)
20+
21+
const handler = jest.fn()
22+
const handler2 = jest.fn()
23+
createApp({
24+
template: `
25+
<custom-event-casing
26+
@camelCase="handler"
27+
@CAPScase="handler"
28+
@PascalCase="handler"
29+
v-on="{
30+
camelCase: handler2,
31+
CAPScase: handler2,
32+
PascalCase: handler2
33+
}" />`,
34+
methods: {
35+
handler,
36+
handler2
37+
}
38+
}).mount(container)
39+
40+
expect(handler).toHaveBeenCalledTimes(3)
41+
expect(handler2).toHaveBeenCalledTimes(3)
42+
})

0 commit comments

Comments
 (0)
Please sign in to comment.