Skip to content

Commit be21cfb

Browse files
committed
feat(types): feat(types): add ComponentCustomProperties interface (#982)
1 parent 4cf5e07 commit be21cfb

File tree

5 files changed

+59
-7
lines changed

5 files changed

+59
-7
lines changed

packages/runtime-core/src/apiWatch.ts

+5-4
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,6 @@ import {
2020
import {
2121
currentInstance,
2222
ComponentInternalInstance,
23-
Data,
2423
isInSSRComponentSetup,
2524
recordInstanceBoundEffect
2625
} from './component'
@@ -276,9 +275,11 @@ export function instanceWatch(
276275
cb: Function,
277276
options?: WatchOptions
278277
): StopHandle {
279-
const ctx = this.proxy as Data
280-
const getter = isString(source) ? () => ctx[source] : source.bind(ctx)
281-
const stop = watch(getter, cb.bind(ctx), options)
278+
const publicThis = this.proxy as any
279+
const getter = isString(source)
280+
? () => publicThis[source]
281+
: source.bind(publicThis)
282+
const stop = watch(getter, cb.bind(publicThis), options)
282283
onBeforeUnmount(stop, this)
283284
return stop
284285
}

packages/runtime-core/src/componentOptions.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -528,7 +528,7 @@ function createWatcher(
528528
publicThis: ComponentPublicInstance,
529529
key: string
530530
) {
531-
const getter = () => (publicThis as Data)[key]
531+
const getter = () => (publicThis as any)[key]
532532
if (isString(raw)) {
533533
const handler = ctx[raw]
534534
if (isFunction(handler)) {

packages/runtime-core/src/componentProxy.ts

+29-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,33 @@ import {
2424
} from './componentRenderUtils'
2525
import { warn } from './warning'
2626

27+
/**
28+
* Custom properties added to component instances in any way and can be accessed through `this`
29+
*
30+
* @example
31+
* Here is an example of adding a property `$router` to every component instance:
32+
* ```ts
33+
* import { createApp } from 'vue'
34+
* import { Router, createRouter } from 'vue-router'
35+
*
36+
* declare module '@vue/runtime-core' {
37+
* interface ComponentCustomProperties {
38+
* $router: Router
39+
* }
40+
* }
41+
*
42+
* // effectively adding the router to every component instance
43+
* const app = createApp({})
44+
* const router = createRouter()
45+
* app.config.globalProperties.$router = router
46+
*
47+
* const vm = app.mount('#app')
48+
* // we can access the router from the instance
49+
* vm.$router.push('/')
50+
* ```
51+
*/
52+
export interface ComponentCustomProperties {}
53+
2754
// public properties exposed on the proxy, which is used as the render context
2855
// in templates (as `this` in the render option)
2956
export type ComponentPublicInstance<
@@ -53,7 +80,8 @@ export type ComponentPublicInstance<
5380
UnwrapRef<B> &
5481
D &
5582
ExtractComputedReturns<C> &
56-
M
83+
M &
84+
ComponentCustomProperties
5785

5886
const publicPropertiesMap: Record<
5987
string,

packages/runtime-core/src/index.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -192,7 +192,10 @@ export {
192192
ComponentOptionsWithObjectProps as ComponentOptionsWithProps,
193193
ComponentOptionsWithArrayProps
194194
} from './componentOptions'
195-
export { ComponentPublicInstance } from './componentProxy'
195+
export {
196+
ComponentPublicInstance,
197+
ComponentCustomProperties
198+
} from './componentProxy'
196199
export {
197200
Renderer,
198201
RendererNode,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import { expectError } from 'tsd'
2+
import { defineComponent } from './index'
3+
4+
declare module '@vue/runtime-core' {
5+
interface ComponentCustomProperties {
6+
state: 'stopped' | 'running'
7+
}
8+
}
9+
10+
export const Custom = defineComponent({
11+
data: () => ({ counter: 0 }),
12+
methods: {
13+
aMethod() {
14+
expectError(this.notExisting)
15+
this.counter++
16+
this.state = 'running'
17+
expectError((this.state = 'not valid'))
18+
}
19+
}
20+
})

0 commit comments

Comments
 (0)