Skip to content

Commit c1d5928

Browse files
committed
fix(types): accept generic Component type in h()
fix #922
1 parent 5dcc645 commit c1d5928

File tree

2 files changed

+19
-24
lines changed

2 files changed

+19
-24
lines changed

packages/runtime-core/src/h.ts

+5-19
Original file line numberDiff line numberDiff line change
@@ -10,14 +10,8 @@ import { Teleport, TeleportProps } from './components/Teleport'
1010
import { Suspense, SuspenseProps } from './components/Suspense'
1111
import { isObject, isArray } from '@vue/shared'
1212
import { RawSlots } from './componentSlots'
13-
import { FunctionalComponent } from './component'
14-
import {
15-
ComponentOptionsWithoutProps,
16-
ComponentOptionsWithArrayProps,
17-
ComponentOptionsWithObjectProps,
18-
ComponentOptions
19-
} from './componentOptions'
20-
import { ExtractPropTypes } from './componentProps'
13+
import { FunctionalComponent, Component } from './component'
14+
import { ComponentOptions } from './componentOptions'
2115

2216
// `h` is a more user-friendly version of `createVNode` that allows omitting the
2317
// props when possible. It is intended for manually written render functions.
@@ -108,27 +102,19 @@ export function h(
108102
): VNode
109103

110104
// functional component
111-
export function h(type: FunctionalComponent, children?: RawChildren): VNode
112105
export function h<P>(
113106
type: FunctionalComponent<P>,
114107
props?: (RawProps & P) | ({} extends P ? null : never),
115108
children?: RawChildren | RawSlots
116109
): VNode
117110

118-
// stateful component
119-
export function h(type: ComponentOptions, children?: RawChildren): VNode
111+
// catch-all for generic component types
112+
export function h(type: Component, children?: RawChildren): VNode
120113
export function h(
121-
type: ComponentOptionsWithoutProps | ComponentOptionsWithArrayProps,
114+
type: ComponentOptions | FunctionalComponent<{}>,
122115
props?: RawProps | null,
123116
children?: RawChildren | RawSlots
124117
): VNode
125-
export function h<O>(
126-
type: ComponentOptionsWithObjectProps<O>,
127-
props?:
128-
| (RawProps & ExtractPropTypes<O>)
129-
| ({} extends ExtractPropTypes<O> ? null : never),
130-
children?: RawChildren | RawSlots
131-
): VNode
132118

133119
// fake constructor type returned by `defineComponent` or class component
134120
export function h(type: Constructor, children?: RawChildren): VNode

test-dts/h.test-d.ts

+14-5
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,8 @@ import {
66
ref,
77
Fragment,
88
Teleport,
9-
Suspense
9+
Suspense,
10+
Component
1011
} from './index'
1112

1213
describe('h inference w/ element', () => {
@@ -58,17 +59,15 @@ describe('h inference w/ functional component', () => {
5859
expectError(h(Func, { bar: 123 }))
5960
})
6061

61-
describe('h inference w/ plain object component', () => {
62+
describe('h support w/ plain object component', () => {
6263
const Foo = {
6364
props: {
6465
foo: String
6566
}
6667
}
67-
6868
h(Foo, { foo: 'ok' })
6969
h(Foo, { foo: 'ok', class: 'extra' })
70-
// should fail on wrong type
71-
expectError(h(Foo, { foo: 1 }))
70+
// no inference in this case
7271
})
7372

7473
describe('h inference w/ defineComponent', () => {
@@ -122,3 +121,13 @@ describe('h inference w/ defineComponent + direct function', () => {
122121
// should fail on wrong type
123122
expectError(h(Foo, { bar: 1, foo: 1 }))
124123
})
124+
125+
// #922
126+
describe('h support for generic component type', () => {
127+
function foo(bar: Component) {
128+
h(bar)
129+
h(bar, 'hello')
130+
h(bar, { id: 'ok' }, 'hello')
131+
}
132+
foo({})
133+
})

0 commit comments

Comments
 (0)