Skip to content

Commit adf3ac8

Browse files
committed
feat(setup): support listeners on setup context + useListeners() helper
These are added because Vue 2 does not include listeners in `context.attrs` so there is no way to access the equivalent of `this.$listeners` in `setup()`.
1 parent 135d074 commit adf3ac8

File tree

7 files changed

+80
-32
lines changed

7 files changed

+80
-32
lines changed

src/core/instance/lifecycle.ts

+18-10
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ import {
1818
invokeWithErrorHandling
1919
} from '../util/index'
2020
import { currentInstance, setCurrentInstance } from 'v3/currentInstance'
21-
import { syncSetupAttrs } from 'v3/apiSetup'
21+
import { syncSetupProxy } from 'v3/apiSetup'
2222

2323
export let activeInstance: any = null
2424
export let isUpdatingChildComponent: boolean = false
@@ -288,19 +288,33 @@ export function updateChildComponent(
288288
// force update if attrs are accessed and has changed since it may be
289289
// passed to a child component.
290290
if (
291-
syncSetupAttrs(
291+
syncSetupProxy(
292292
vm._attrsProxy,
293293
attrs,
294294
(prevVNode.data && prevVNode.data.attrs) || emptyObject,
295-
vm
295+
vm,
296+
'$attrs'
296297
)
297298
) {
298299
needsForceUpdate = true
299300
}
300301
}
301302
vm.$attrs = attrs
302303

303-
vm.$listeners = listeners || emptyObject
304+
// update listeners
305+
listeners = listeners || emptyObject
306+
const prevListeners = vm.$options._parentListeners
307+
if (vm._listenersProxy) {
308+
syncSetupProxy(
309+
vm._listenersProxy,
310+
listeners,
311+
prevListeners || emptyObject,
312+
vm,
313+
'$listeners'
314+
)
315+
}
316+
vm.$listeners = vm.$options._parentListeners = listeners
317+
updateComponentListeners(vm, listeners, prevListeners)
304318

305319
// update props
306320
if (propsData && vm.$options.props) {
@@ -317,12 +331,6 @@ export function updateChildComponent(
317331
vm.$options.propsData = propsData
318332
}
319333

320-
// update listeners
321-
listeners = listeners || emptyObject
322-
const oldListeners = vm.$options._parentListeners
323-
vm.$options._parentListeners = listeners
324-
updateComponentListeners(vm, listeners, oldListeners)
325-
326334
// resolve slots + force update if has children
327335
if (needsForceUpdate) {
328336
vm.$slots = resolveSlots(renderChildren, parentVnode.context)

src/types/component.ts

+1
Original file line numberDiff line numberDiff line change
@@ -111,6 +111,7 @@ export declare class Component {
111111
_setupProxy?: Record<string, any>
112112
_setupContext?: SetupContext
113113
_attrsProxy?: Record<string, any>
114+
_listenersProxy?: Record<string, Function | Function[]>
114115
_slotsProxy?: Record<string, () => VNode[]>
115116
_preWatchers?: Watcher[]
116117

src/v3/apiSetup.ts

+33-19
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ import { proxyWithRefUnwrap } from './reactivity/ref'
1919
*/
2020
export interface SetupContext {
2121
attrs: Record<string, any>
22+
listeners: Record<string, Function | Function[]>
2223
slots: Record<string, () => VNode[]>
2324
emit: (event: string, ...args: any[]) => any
2425
expose: (exposed: Record<string, any>) => void
@@ -87,7 +88,19 @@ function createSetupContext(vm: Component): SetupContext {
8788
let exposeCalled = false
8889
return {
8990
get attrs() {
90-
return initAttrsProxy(vm)
91+
if (!vm._attrsProxy) {
92+
const proxy = (vm._attrsProxy = {})
93+
def(proxy, '_v_attr_proxy', true)
94+
syncSetupProxy(proxy, vm.$attrs, emptyObject, vm, '$attrs')
95+
}
96+
return vm._attrsProxy
97+
},
98+
get listeners() {
99+
if (!vm._listenersProxy) {
100+
const proxy = (vm._listenersProxy = {})
101+
syncSetupProxy(proxy, vm.$listeners, emptyObject, vm, '$listeners')
102+
}
103+
return vm._listenersProxy
91104
},
92105
get slots() {
93106
return initSlotsProxy(vm)
@@ -109,26 +122,18 @@ function createSetupContext(vm: Component): SetupContext {
109122
}
110123
}
111124

112-
function initAttrsProxy(vm: Component) {
113-
if (!vm._attrsProxy) {
114-
const proxy = (vm._attrsProxy = {})
115-
def(proxy, '_v_attr_proxy', true)
116-
syncSetupAttrs(proxy, vm.$attrs, emptyObject, vm)
117-
}
118-
return vm._attrsProxy
119-
}
120-
121-
export function syncSetupAttrs(
125+
export function syncSetupProxy(
122126
to: any,
123127
from: any,
124128
prev: any,
125-
instance: Component
129+
instance: Component,
130+
type: string
126131
) {
127132
let changed = false
128133
for (const key in from) {
129134
if (!(key in to)) {
130135
changed = true
131-
defineProxyAttr(to, key, instance)
136+
defineProxyAttr(to, key, instance, type)
132137
} else if (from[key] !== prev[key]) {
133138
changed = true
134139
}
@@ -142,12 +147,17 @@ export function syncSetupAttrs(
142147
return changed
143148
}
144149

145-
function defineProxyAttr(proxy: any, key: string, instance: Component) {
150+
function defineProxyAttr(
151+
proxy: any,
152+
key: string,
153+
instance: Component,
154+
type: string
155+
) {
146156
Object.defineProperty(proxy, key, {
147157
enumerable: true,
148158
configurable: true,
149159
get() {
150-
return instance.$attrs[key]
160+
return instance[type][key]
151161
}
152162
})
153163
}
@@ -171,19 +181,23 @@ export function syncSetupSlots(to: any, from: any) {
171181
}
172182

173183
/**
174-
* @internal use manual type def
184+
* @internal use manual type def because it relies on legacy VNode types
175185
*/
176186
export function useSlots(): SetupContext['slots'] {
177187
return getContext().slots
178188
}
179189

180-
/**
181-
* @internal use manual type def
182-
*/
183190
export function useAttrs(): SetupContext['attrs'] {
184191
return getContext().attrs
185192
}
186193

194+
/**
195+
* Vue 2 only
196+
*/
197+
export function useListeners(): SetupContext['listeners'] {
198+
return getContext().listeners
199+
}
200+
187201
function getContext(): SetupContext {
188202
if (__DEV__ && !currentInstance) {
189203
warn(`useContext() called without active instance.`)

src/v3/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ export { provide, inject, InjectionKey } from './apiInject'
7777

7878
export { h } from './h'
7979
export { getCurrentInstance } from './currentInstance'
80-
export { useSlots, useAttrs, mergeDefaults } from './apiSetup'
80+
export { useSlots, useAttrs, useListeners, mergeDefaults } from './apiSetup'
8181
export { nextTick } from 'core/util/next-tick'
8282
export { set, del } from 'core/observer'
8383

test/unit/features/v3/apiSetup.spec.ts

+23
Original file line numberDiff line numberDiff line change
@@ -297,4 +297,27 @@ describe('api: setup context', () => {
297297
await nextTick()
298298
expect(spy).toHaveBeenCalledTimes(1)
299299
})
300+
301+
it('context.listeners', async () => {
302+
let _listeners
303+
const Child = {
304+
setup(_, { listeners }) {
305+
_listeners = listeners
306+
return () => {}
307+
}
308+
}
309+
310+
const Parent = {
311+
data: () => ({ log: () => 1 }),
312+
template: `<Child @foo="log" />`,
313+
components: { Child }
314+
}
315+
316+
const vm = new Vue(Parent).$mount()
317+
318+
expect(_listeners.foo()).toBe(1)
319+
vm.log = () => 2
320+
await nextTick()
321+
expect(_listeners.foo()).toBe(2)
322+
})
300323
})

types/v3-manual-apis.d.ts

-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,4 @@ export function getCurrentInstance(): { proxy: Vue } | null
55

66
export const h: CreateElement
77

8-
export function useAttrs(): SetupContext['attrs']
9-
108
export function useSlots(): SetupContext['slots']

types/v3-setup-context.d.ts

+4
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,10 @@ export type EmitFn<
3131

3232
export interface SetupContext<E extends EmitsOptions = {}> {
3333
attrs: Data
34+
/**
35+
* Equivalent of `this.$listeners`, which is Vue 2 only.
36+
*/
37+
listeners: Record<string, Function | Function[]>
3438
slots: Slots
3539
emit: EmitFn<E>
3640
expose(exposed?: Record<string, any>): void

0 commit comments

Comments
 (0)