Skip to content

Commit

Permalink
fix(types): make return type of defineComponent assignable to `Comp…
Browse files Browse the repository at this point in the history
…onent` type (#1032)

fix #993
  • Loading branch information
pikax committed Apr 24, 2020
1 parent 28b4c31 commit f3a9b51
Show file tree
Hide file tree
Showing 4 changed files with 49 additions and 10 deletions.
5 changes: 3 additions & 2 deletions packages/runtime-core/__tests__/components/Suspense.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,8 @@ import {
watch,
watchEffect,
onUnmounted,
onErrorCaptured
onErrorCaptured,
Component
} from '@vue/runtime-test'

describe('Suspense', () => {
Expand All @@ -30,7 +31,7 @@ describe('Suspense', () => {
setup(props: any, { slots }: any) {
const p = new Promise(resolve => {
setTimeout(() => {
resolve(() => h(comp, props, slots))
resolve(() => h<Component>(comp, props, slots))
}, delay)
})
// in Node 12, due to timer/nextTick mechanism change, we have to wait
Expand Down
10 changes: 5 additions & 5 deletions packages/runtime-core/src/apiDefineComponent.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import {
ComponentOptionsWithArrayProps,
ComponentOptionsWithObjectProps
} from './componentOptions'
import { SetupContext, RenderFunction } from './component'
import { SetupContext, RenderFunction, FunctionalComponent } from './component'
import { ComponentPublicInstance } from './componentProxy'
import { ExtractPropTypes, ComponentPropsOptions } from './componentProps'
import { EmitsOptions } from './componentEmits'
Expand Down Expand Up @@ -34,7 +34,7 @@ export function defineComponent<Props, RawBindings = object>(
// public props
VNodeProps & Props
>
}
} & FunctionalComponent<Props>

// overload 2: object format with no props
// (uses user defined props interface)
Expand All @@ -59,7 +59,7 @@ export function defineComponent<
E,
VNodeProps & Props
>
}
} & ComponentOptionsWithoutProps<Props, RawBindings, D, C, M, E, EE>

// overload 3: object format with array props declaration
// props inferred as { [key in PropNames]?: any }
Expand All @@ -85,7 +85,7 @@ export function defineComponent<
): {
// array props technically doesn't place any contraints on props in TSX
new (): ComponentPublicInstance<VNodeProps, RawBindings, D, C, M, E>
}
} & ComponentOptionsWithArrayProps<PropNames, RawBindings, D, C, M, E, EE>

// overload 4: object format with object props declaration
// see `ExtractPropTypes` in ./componentProps.ts
Expand Down Expand Up @@ -119,7 +119,7 @@ export function defineComponent<
E,
VNodeProps & ExtractPropTypes<PropsOptions, false>
>
}
} & ComponentOptionsWithObjectProps<PropsOptions, RawBindings, D, C, M, E, EE>

// implementation, close to no-op
export function defineComponent(options: unknown) {
Expand Down
11 changes: 9 additions & 2 deletions packages/runtime-core/src/h.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,11 @@ interface Constructor<P = any> {
new (): { $props: P }
}

// Excludes Component type from returned `defineComponent`
type NotDefinedComponent<T extends Component> = T extends Constructor
? never
: T

// The following is a series of overloads for providing props validation of
// manually written render functions.

Expand Down Expand Up @@ -110,8 +115,10 @@ export function h<P>(

// catch-all for generic component types
export function h(type: Component, children?: RawChildren): VNode
export function h(
type: ComponentOptions | FunctionalComponent<{}>,

// exclude `defineComponent`
export function h<Options extends ComponentOptions | FunctionalComponent<{}>>(
type: NotDefinedComponent<Options>,
props?: RawProps | null,
children?: RawChildren | RawSlots
): VNode
Expand Down
33 changes: 32 additions & 1 deletion test-dts/h.test-d.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { expectError } from 'tsd'
import { expectError, expectAssignable } from 'tsd'
import {
describe,
h,
Expand Down Expand Up @@ -131,3 +131,34 @@ describe('h support for generic component type', () => {
}
foo({})
})

// #993
describe('describeComponent extends Component', () => {
// functional
expectAssignable<Component>(
defineComponent((_props: { foo?: string; bar: number }) => {})
)

// typed props
expectAssignable<Component>(defineComponent({}))

// prop arrays
expectAssignable<Component>(
defineComponent({
props: ['a', 'b']
})
)

// prop object
expectAssignable<Component>(
defineComponent({
props: {
foo: String,
bar: {
type: Number,
required: true
}
}
})
)
})

0 comments on commit f3a9b51

Please sign in to comment.