@@ -19,6 +19,7 @@ import { proxyWithRefUnwrap } from './reactivity/ref'
19
19
*/
20
20
export interface SetupContext {
21
21
attrs : Record < string , any >
22
+ listeners : Record < string , Function | Function [ ] >
22
23
slots : Record < string , ( ) => VNode [ ] >
23
24
emit : ( event : string , ...args : any [ ] ) => any
24
25
expose : ( exposed : Record < string , any > ) => void
@@ -87,7 +88,19 @@ function createSetupContext(vm: Component): SetupContext {
87
88
let exposeCalled = false
88
89
return {
89
90
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
91
104
} ,
92
105
get slots ( ) {
93
106
return initSlotsProxy ( vm )
@@ -109,26 +122,18 @@ function createSetupContext(vm: Component): SetupContext {
109
122
}
110
123
}
111
124
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 (
122
126
to : any ,
123
127
from : any ,
124
128
prev : any ,
125
- instance : Component
129
+ instance : Component ,
130
+ type : string
126
131
) {
127
132
let changed = false
128
133
for ( const key in from ) {
129
134
if ( ! ( key in to ) ) {
130
135
changed = true
131
- defineProxyAttr ( to , key , instance )
136
+ defineProxyAttr ( to , key , instance , type )
132
137
} else if ( from [ key ] !== prev [ key ] ) {
133
138
changed = true
134
139
}
@@ -142,12 +147,17 @@ export function syncSetupAttrs(
142
147
return changed
143
148
}
144
149
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
+ ) {
146
156
Object . defineProperty ( proxy , key , {
147
157
enumerable : true ,
148
158
configurable : true ,
149
159
get ( ) {
150
- return instance . $attrs [ key ]
160
+ return instance [ type ] [ key ]
151
161
}
152
162
} )
153
163
}
@@ -171,19 +181,23 @@ export function syncSetupSlots(to: any, from: any) {
171
181
}
172
182
173
183
/**
174
- * @internal use manual type def
184
+ * @internal use manual type def because it relies on legacy VNode types
175
185
*/
176
186
export function useSlots ( ) : SetupContext [ 'slots' ] {
177
187
return getContext ( ) . slots
178
188
}
179
189
180
- /**
181
- * @internal use manual type def
182
- */
183
190
export function useAttrs ( ) : SetupContext [ 'attrs' ] {
184
191
return getContext ( ) . attrs
185
192
}
186
193
194
+ /**
195
+ * Vue 2 only
196
+ */
197
+ export function useListeners ( ) : SetupContext [ 'listeners' ] {
198
+ return getContext ( ) . listeners
199
+ }
200
+
187
201
function getContext ( ) : SetupContext {
188
202
if ( __DEV__ && ! currentInstance ) {
189
203
warn ( `useContext() called without active instance.` )
0 commit comments