wip: improve options typing

This commit is contained in:
Evan You 2019-09-05 16:09:30 -04:00
parent 369b9eb583
commit 9b90e673e8
6 changed files with 192 additions and 45 deletions

View File

@ -7,7 +7,7 @@ export interface ComputedRef<T> {
readonly effect: ReactiveEffect readonly effect: ReactiveEffect
} }
export interface ComputedOptions<T = any> { export interface ComputedOptions<T> {
get: () => T get: () => T
set: (v: T) => void set: (v: T) => void
} }

View File

@ -1,6 +1,7 @@
import { createComponent } from '../src/component' import { createComponent, ComponentRenderProxy } from '../src/component'
import { ref } from '@vue/reactivity' import { ref } from '@vue/reactivity'
import { PropType } from '../src/componentProps' import { PropType } from '../src/componentProps'
import { h } from '../src/h'
// mock React just for TSX testing purposes // mock React just for TSX testing purposes
const React = { const React = {
@ -55,6 +56,7 @@ test('createComponent type inference', () => {
this.d.e.slice() this.d.e.slice()
this.cc && this.cc.push('hoo') this.cc && this.cc.push('hoo')
this.dd.push('dd') this.dd.push('dd')
// return h('div', this.bb)
} }
}) })
// test TSX props inference // test TSX props inference
@ -71,9 +73,9 @@ test('type inference w/ optional props declaration', () => {
}, },
render() { render() {
this.$props.msg this.$props.msg
this.$data.a * 2
this.msg this.msg
this.a * 2 this.a * 2
// return h('div', this.msg)
} }
}) })
;(<Comp msg="hello"/>) ;(<Comp msg="hello"/>)
@ -99,7 +101,6 @@ test('type inference w/ array props declaration', () => {
render() { render() {
this.$props.a this.$props.a
this.$props.b this.$props.b
this.$data.c
this.a this.a
this.b this.b
this.c this.c
@ -107,3 +108,46 @@ test('type inference w/ array props declaration', () => {
}) })
;(<Comp a={1} b={2}/>) ;(<Comp a={1} b={2}/>)
}) })
test('with legacy options', () => {
createComponent({
props: { a: Number },
setup() {
return {
b: 123
}
},
data() {
this.a
this.b
return {
c: 234
}
},
computed: {
d(): number {
return this.b + 1
}
},
watch: {
a() {
this.b + 1
}
},
created() {
this.a && this.a * 2
this.b * 2
this.c * 2
this.d * 2
},
methods: {
doSomething() {
this.a && this.a * 2
this.b * 2
this.c * 2
this.d * 2
return (this.a || 0) + this.b + this.c + this.d
}
}
})
})

View File

@ -3,7 +3,8 @@ import {
Data, Data,
ComponentOptions, ComponentOptions,
currentRenderingInstance, currentRenderingInstance,
currentInstance currentInstance,
ComponentRenderProxy
} from './component' } from './component'
import { import {
isFunction, isFunction,
@ -15,8 +16,8 @@ import {
capitalize, capitalize,
camelize camelize
} from '@vue/shared' } from '@vue/shared'
import { computed, ComputedOptions } from './apiReactivity' import { computed } from './apiReactivity'
import { watch, WatchOptions } from './apiWatch' import { watch, WatchOptions, CleanupRegistrator } from './apiWatch'
import { provide, inject } from './apiInject' import { provide, inject } from './apiInject'
import { import {
onBeforeMount, onBeforeMount,
@ -34,20 +35,61 @@ import { warn } from './warning'
// TODO legacy component definition also supports constructors with .options // TODO legacy component definition also supports constructors with .options
type LegacyComponent = ComponentOptions type LegacyComponent = ComponentOptions
export interface ComputedOptions {
[key: string]:
| Function
| {
get: Function
set: Function
}
}
export interface MethodOptions {
[key: string]: Function
}
export type ExtracComputedReturns<T extends any> = {
[key in keyof T]: T[key] extends { get: Function }
? ReturnType<T[key]['get']>
: ReturnType<T[key]>
}
type WatchHandler = (
val: any,
oldVal: any,
onCleanup: CleanupRegistrator
) => void
// TODO type inference for these options // TODO type inference for these options
export interface LegacyOptions { export interface LegacyOptions<
Props,
RawBindings,
D,
C extends ComputedOptions,
M extends MethodOptions,
ThisContext = ThisType<ComponentRenderProxy<Props, D, RawBindings, C, M>>
> {
el?: any el?: any
// state // state
data?: Data | (() => Data) data?:
computed?: Record<string, (() => any) | ComputedOptions> | D
methods?: Record<string, Function> | (<This extends ComponentRenderProxy<Props, {}, RawBindings>>(
this: This
) => D)
computed?: C & ThisContext
methods?: M & ThisContext
// TODO watch array // TODO watch array
watch?: Record< watch?: Record<
string, string,
string | Function | { handler: Function } & WatchOptions string | WatchHandler | { handler: WatchHandler } & WatchOptions
> > &
provide?: Data | (() => Data) ThisContext
provide?:
| Data
| (<This extends ComponentRenderProxy<Props, D, RawBindings, C, M>>(
this: This
) => any)
inject?: inject?:
| string[] | string[]
| Record< | Record<
@ -60,8 +102,10 @@ export interface LegacyOptions {
extends?: LegacyComponent extends?: LegacyComponent
// lifecycle // lifecycle
beforeCreate?(): void beforeCreate?(this: ComponentRenderProxy): void
created?(): void created?<This extends ComponentRenderProxy<Props, D, RawBindings, C, M>>(
this: This
): void
beforeMount?(): void beforeMount?(): void
mounted?(): void mounted?(): void
beforeUpdate?(): void beforeUpdate?(): void
@ -138,7 +182,7 @@ export function applyOptions(
} }
if (computedOptions) { if (computedOptions) {
for (const key in computedOptions) { for (const key in computedOptions) {
const opt = computedOptions[key] const opt = (computedOptions as ComputedOptions)[key]
data[key] = isFunction(opt) data[key] = isFunction(opt)
? computed(opt.bind(ctx)) ? computed(opt.bind(ctx))
: computed({ : computed({
@ -149,7 +193,7 @@ export function applyOptions(
} }
if (methods) { if (methods) {
for (const key in methods) { for (const key in methods) {
data[key] = methods[key].bind(ctx) data[key] = (methods as MethodOptions)[key].bind(ctx)
} }
} }
if (watchOptions) { if (watchOptions) {

View File

@ -32,7 +32,7 @@ type MapSources<T> = {
[K in keyof T]: T[K] extends WatcherSource<infer V> ? V : never [K in keyof T]: T[K] extends WatcherSource<infer V> ? V : never
} }
type CleanupRegistrator = (invalidate: () => void) => void export type CleanupRegistrator = (invalidate: () => void) => void
type SimpleEffect = (onCleanup: CleanupRegistrator) => void type SimpleEffect = (onCleanup: CleanupRegistrator) => void

View File

@ -22,14 +22,28 @@ import {
} from './errorHandling' } from './errorHandling'
import { AppContext, createAppContext } from './apiApp' import { AppContext, createAppContext } from './apiApp'
import { Directive } from './directives' import { Directive } from './directives'
import { applyOptions, LegacyOptions, resolveAsset } from './apiOptions' import {
applyOptions,
LegacyOptions,
resolveAsset,
ComputedOptions,
MethodOptions,
ExtracComputedReturns
} from './apiOptions'
export type Data = { [key: string]: unknown } export type Data = { [key: string]: unknown }
// public properties exposed on the proxy, which is used as the render context // public properties exposed on the proxy, which is used as the render context
// in templates (as `this` in the render option) // in templates (as `this` in the render option)
export type ComponentRenderProxy<P = {}, S = {}, PublicProps = P> = { export type ComponentRenderProxy<
$data: S P = {},
D = {},
B = {},
C = {},
M = {},
PublicProps = P
> = {
$data: D
$props: PublicProps $props: PublicProps
$attrs: Data $attrs: Data
$refs: Data $refs: Data
@ -38,50 +52,70 @@ export type ComponentRenderProxy<P = {}, S = {}, PublicProps = P> = {
$parent: ComponentInstance | null $parent: ComponentInstance | null
$emit: (event: string, ...args: unknown[]) => void $emit: (event: string, ...args: unknown[]) => void
} & P & } & P &
S D &
UnwrapRef<B> &
ExtracComputedReturns<C> &
M
type RenderFunction<Props = {}, RawBindings = {}> = < type RenderFunction<P = {}, D = {}, B = {}, C = {}, M = {}> = <
Bindings extends UnwrapRef<RawBindings> This extends ComponentRenderProxy<P, D, B, C, M>
>( >(
this: ComponentRenderProxy<Props, Bindings> this: This
) => VNodeChild ) => VNodeChild
interface ComponentOptionsBase<Props, RawBindings> extends LegacyOptions { interface ComponentOptionsBase<
Props,
RawBindings,
D,
C extends ComputedOptions,
M extends MethodOptions
> extends LegacyOptions<Props, RawBindings, D, C, M> {
setup?: ( setup?: (
props: Props, props: Props,
ctx: SetupContext ctx: SetupContext
) => RawBindings | (() => VNodeChild) | void ) => RawBindings | (() => VNodeChild) | void
name?: string name?: string
template?: string template?: string
render?: RenderFunction<Props, RawBindings> render?: RenderFunction<Props, D, RawBindings, C, M>
components?: Record<string, Component> components?: Record<string, Component>
directives?: Record<string, Directive> directives?: Record<string, Directive>
} }
export interface ComponentOptionsWithoutProps<Props = {}, RawBindings = {}> export interface ComponentOptionsWithoutProps<
extends ComponentOptionsBase<Props, RawBindings> { Props = {},
RawBindings = {},
D = {},
C extends ComputedOptions = {},
M extends MethodOptions = {}
> extends ComponentOptionsBase<Props, RawBindings, D, C, M> {
props?: undefined props?: undefined
} }
export interface ComponentOptionsWithArrayProps< export interface ComponentOptionsWithArrayProps<
PropNames extends string = string, PropNames extends string = string,
RawBindings = {}, RawBindings = {},
D = {},
C extends ComputedOptions = {},
M extends MethodOptions = {},
Props = { [key in PropNames]?: unknown } Props = { [key in PropNames]?: unknown }
> extends ComponentOptionsBase<Props, RawBindings> { > extends ComponentOptionsBase<Props, RawBindings, D, C, M> {
props: PropNames[] props: PropNames[]
} }
export interface ComponentOptionsWithProps< export interface ComponentOptionsWithProps<
PropsOptions = ComponentPropsOptions, PropsOptions = ComponentPropsOptions,
RawBindings = {}, RawBindings = {},
D = {},
C extends ComputedOptions = {},
M extends MethodOptions = {},
Props = ExtractPropTypes<PropsOptions> Props = ExtractPropTypes<PropsOptions>
> extends ComponentOptionsBase<Props, RawBindings> { > extends ComponentOptionsBase<Props, RawBindings, D, C, M> {
props: PropsOptions props: PropsOptions
} }
export type ComponentOptions = export type ComponentOptions =
| ComponentOptionsWithProps
| ComponentOptionsWithoutProps | ComponentOptionsWithoutProps
| ComponentOptionsWithProps
| ComponentOptionsWithArrayProps | ComponentOptionsWithArrayProps
export interface FunctionalComponent<P = {}> { export interface FunctionalComponent<P = {}> {
@ -116,7 +150,7 @@ interface SetupContext {
emit: ((event: string, ...args: unknown[]) => void) emit: ((event: string, ...args: unknown[]) => void)
} }
export type ComponentInstance<P = Data, S = Data> = { export type ComponentInstance<P = Data, D = Data> = {
type: FunctionalComponent | ComponentOptions type: FunctionalComponent | ComponentOptions
parent: ComponentInstance | null parent: ComponentInstance | null
appContext: AppContext appContext: AppContext
@ -125,7 +159,7 @@ export type ComponentInstance<P = Data, S = Data> = {
next: VNode | null next: VNode | null
subTree: VNode subTree: VNode
update: ReactiveEffect update: ReactiveEffect
render: RenderFunction<P, S> | null render: RenderFunction<P, D> | null
effects: ReactiveEffect[] | null effects: ReactiveEffect[] | null
provides: Data provides: Data
@ -133,7 +167,7 @@ export type ComponentInstance<P = Data, S = Data> = {
directives: Record<string, Directive> directives: Record<string, Directive>
// the rest are only for stateful components // the rest are only for stateful components
data: S data: D
props: P props: P
renderProxy: ComponentRenderProxy | null renderProxy: ComponentRenderProxy | null
propsProxy: P | null propsProxy: P | null
@ -168,31 +202,55 @@ export function createComponent<Props>(
// overload 2: object format with no props // overload 2: object format with no props
// (uses user defined props interface) // (uses user defined props interface)
// return type is for Vetur and TSX support // return type is for Vetur and TSX support
export function createComponent<Props, RawBindings>( export function createComponent<
options: ComponentOptionsWithoutProps<Props, RawBindings> Props,
RawBindings,
D,
C extends ComputedOptions = {},
M extends MethodOptions = {}
>(
options: ComponentOptionsWithoutProps<Props, RawBindings, D, C, M>
): { ): {
new (): ComponentRenderProxy<Props, UnwrapRef<RawBindings>> new (): ComponentRenderProxy<Props, D, RawBindings, C, M>
} }
// overload 3: object format with array props declaration // overload 3: object format with array props declaration
// props inferred as { [key in PropNames]?: unknown } // props inferred as { [key in PropNames]?: unknown }
// return type is for Vetur and TSX support // return type is for Vetur and TSX support
export function createComponent<PropNames extends string, RawBindings>( export function createComponent<
options: ComponentOptionsWithArrayProps<PropNames, RawBindings> PropNames extends string,
RawBindings,
D,
C extends ComputedOptions = {},
M extends MethodOptions = {}
>(
options: ComponentOptionsWithArrayProps<PropNames, RawBindings, D, C, M>
): { ): {
new (): ComponentRenderProxy< new (): ComponentRenderProxy<
{ [key in PropNames]?: unknown }, { [key in PropNames]?: unknown },
UnwrapRef<RawBindings> D,
RawBindings,
C,
M
> >
} }
// overload 4: object format with object props declaration // overload 4: object format with object props declaration
// see `ExtractPropTypes` in ./componentProps.ts // see `ExtractPropTypes` in ./componentProps.ts
export function createComponent<PropsOptions, RawBindings>( export function createComponent<
options: ComponentOptionsWithProps<PropsOptions, RawBindings> PropsOptions,
RawBindings,
D,
C extends ComputedOptions = {},
M extends MethodOptions = {}
>(
options: ComponentOptionsWithProps<PropsOptions, RawBindings, D, C, M>
): { ): {
// for Vetur and TSX support // for Vetur and TSX support
new (): ComponentRenderProxy< new (): ComponentRenderProxy<
ExtractPropTypes<PropsOptions>, ExtractPropTypes<PropsOptions>,
UnwrapRef<RawBindings>, D,
RawBindings,
C,
M,
ExtractPropTypes<PropsOptions, false> ExtractPropTypes<PropsOptions, false>
> >
} }

View File

@ -10,6 +10,7 @@
"noUnusedLocals": true, "noUnusedLocals": true,
"strictNullChecks": true, "strictNullChecks": true,
"noImplicitAny": true, "noImplicitAny": true,
"noImplicitThis": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"esModuleInterop": true, "esModuleInterop": true,
"removeComments": false, "removeComments": false,