Skip to content

Commit 665f2ae

Browse files
authoredNov 11, 2022
fix(custom-elements): fix event emitting for async custom elements (#5601)
fix #5599
1 parent 0b39e46 commit 665f2ae

File tree

4 files changed

+32
-20
lines changed

4 files changed

+32
-20
lines changed
 

‎packages/runtime-core/src/apiAsyncComponent.ts

+7-4
Original file line numberDiff line numberDiff line change
@@ -211,13 +211,16 @@ export function defineAsyncComponent<
211211

212212
function createInnerComp(
213213
comp: ConcreteComponent,
214-
{
215-
vnode: { ref, props, children, shapeFlag },
216-
parent
217-
}: ComponentInternalInstance
214+
parent: ComponentInternalInstance
218215
) {
216+
const { ref, props, children, ce } = parent.vnode
219217
const vnode = createVNode(comp, props, children)
220218
// ensure inner component inherits the async wrapper's ref owner
221219
vnode.ref = ref
220+
// pass the custom element callback on to the inner comp
221+
// and remove it from the async wrapper
222+
vnode.ce = ce
223+
delete parent.vnode.ce
224+
222225
return vnode
223226
}

‎packages/runtime-core/src/hmr.ts

-8
Original file line numberDiff line numberDiff line change
@@ -136,14 +136,6 @@ function reload(id: string, newComp: HMRComponent) {
136136
// components to be unmounted and re-mounted. Queue the update so that we
137137
// don't end up forcing the same parent to re-render multiple times.
138138
queueJob(instance.parent.update)
139-
// instance is the inner component of an async custom element
140-
// invoke to reset styles
141-
if (
142-
(instance.parent.type as ComponentOptions).__asyncLoader &&
143-
instance.parent.ceReload
144-
) {
145-
instance.parent.ceReload((newComp as any).styles)
146-
}
147139
} else if (instance.appContext.reload) {
148140
// root instance mounted via createApp() has a reload method
149141
instance.appContext.reload()

‎packages/runtime-dom/__tests__/customElement.spec.ts

+23-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import {
22
defineAsyncComponent,
3+
defineComponent,
34
defineCustomElement,
45
h,
56
inject,
@@ -227,7 +228,7 @@ describe('defineCustomElement', () => {
227228
})
228229

229230
describe('emits', () => {
230-
const E = defineCustomElement({
231+
const CompDef = defineComponent({
231232
setup(_, { emit }) {
232233
emit('created')
233234
return () =>
@@ -241,6 +242,7 @@ describe('defineCustomElement', () => {
241242
})
242243
}
243244
})
245+
const E = defineCustomElement(CompDef)
244246
customElements.define('my-el-emits', E)
245247

246248
test('emit on connect', () => {
@@ -277,6 +279,26 @@ describe('defineCustomElement', () => {
277279
expect(spy1).toHaveBeenCalledTimes(1)
278280
expect(spy2).toHaveBeenCalledTimes(1)
279281
})
282+
283+
test('emit from within async component wrapper', async () => {
284+
const E = defineCustomElement(
285+
defineAsyncComponent(
286+
() => new Promise<typeof CompDef>(res => res(CompDef as any))
287+
)
288+
)
289+
customElements.define('my-async-el-emits', E)
290+
container.innerHTML = `<my-async-el-emits></my-async-el-emits>`
291+
const e = container.childNodes[0] as VueElement
292+
const spy = jest.fn()
293+
e.addEventListener('my-click', spy)
294+
// this feels brittle but seems necessary to reach the node in the DOM.
295+
await customElements.whenDefined('my-async-el-emits')
296+
e.shadowRoot!.childNodes[0].dispatchEvent(new CustomEvent('click'))
297+
expect(spy).toHaveBeenCalled()
298+
expect(spy.mock.calls[0][0]).toMatchObject({
299+
detail: [1]
300+
})
301+
})
280302
})
281303

282304
describe('slots', () => {

‎packages/runtime-dom/src/apiCustomElement.ts

+2-7
Original file line numberDiff line numberDiff line change
@@ -341,13 +341,8 @@ export class VueElement extends BaseClass {
341341
this._styles.length = 0
342342
}
343343
this._applyStyles(newStyles)
344-
// if this is an async component, ceReload is called from the inner
345-
// component so no need to reload the async wrapper
346-
if (!(this._def as ComponentOptions).__asyncLoader) {
347-
// reload
348-
this._instance = null
349-
this._update()
350-
}
344+
this._instance = null
345+
this._update()
351346
}
352347
}
353348

0 commit comments

Comments
 (0)
Please sign in to comment.