Skip to content

Commit 7173ad4

Browse files
committed
feat: types for <script setup> macros
1 parent 3c2707b commit 7173ad4

File tree

8 files changed

+337
-12
lines changed

8 files changed

+337
-12
lines changed

src/core/util/options.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -467,7 +467,7 @@ export function resolveAsset(
467467
// fallback to prototype chain
468468
const res = assets[id] || assets[camelizedId] || assets[PascalCaseId]
469469
if (__DEV__ && warnMissing && !res) {
470-
warn('Failed to resolve ' + type.slice(0, -1) + ': ' + id, options)
470+
warn('Failed to resolve ' + type.slice(0, -1) + ': ' + id)
471471
}
472472
return res
473473
}

src/types/options.ts

+7-5
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,9 @@ export type ComponentOptions = {
2626

2727
// data
2828
data: object | Function | void
29-
props?: { [key: string]: PropOptions }
29+
props?:
30+
| string[]
31+
| Record<string, Function | Array<Function> | null | PropOptions>
3032
propsData?: object
3133
computed?: {
3234
[key: string]:
@@ -105,8 +107,8 @@ export type ComponentOptions = {
105107
}
106108

107109
export type PropOptions = {
108-
type: Function | Array<Function> | null
109-
default: any
110-
required: boolean | null
111-
validator: Function | null
110+
type?: Function | Array<Function> | null
111+
default?: any
112+
required?: boolean | null
113+
validator?: Function | null
112114
}

src/v3/apiSetup.ts

+40-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,14 @@
11
import { Component } from 'types/component'
2+
import { PropOptions } from 'types/options'
23
import { def, invokeWithErrorHandling, isReserved, warn } from '../core/util'
34
import VNode from '../core/vdom/vnode'
4-
import { bind, emptyObject, isFunction, isObject } from '../shared/util'
5+
import {
6+
bind,
7+
emptyObject,
8+
isArray,
9+
isFunction,
10+
isObject
11+
} from '../shared/util'
512
import { currentInstance, setCurrentInstance } from './currentInstance'
613
import { isRef } from './reactivity/ref'
714

@@ -193,3 +200,35 @@ function getContext(): SetupContext {
193200
const vm = currentInstance!
194201
return vm._setupContext || (vm._setupContext = createSetupContext(vm))
195202
}
203+
204+
/**
205+
* Runtime helper for merging default declarations. Imported by compiled code
206+
* only.
207+
* @internal
208+
*/
209+
export function mergeDefaults(
210+
raw: string[] | Record<string, PropOptions>,
211+
defaults: Record<string, any>
212+
): Record<string, PropOptions> {
213+
const props = isArray(raw)
214+
? raw.reduce(
215+
(normalized, p) => ((normalized[p] = {}), normalized),
216+
{} as Record<string, PropOptions>
217+
)
218+
: raw
219+
for (const key in defaults) {
220+
const opt = props[key]
221+
if (opt) {
222+
if (isArray(opt) || isFunction(opt)) {
223+
props[key] = { type: opt, default: defaults[key] }
224+
} else {
225+
opt.default = defaults[key]
226+
}
227+
} else if (opt === null) {
228+
props[key] = { default: defaults[key] }
229+
} else if (__DEV__) {
230+
warn(`props default key "${key}" has no corresponding declaration.`)
231+
}
232+
}
233+
return props
234+
}

src/v3/apiWatch.ts

+2-4
Original file line numberDiff line numberDiff line change
@@ -179,10 +179,8 @@ function doWatch(
179179

180180
const warnInvalidSource = (s: unknown) => {
181181
warn(
182-
`Invalid watch source: `,
183-
s,
184-
`A watch source can only be a getter/effect function, a ref, ` +
185-
`a reactive object, or an array of these types.`
182+
`Invalid watch source: ${s}. A watch source can only be a getter/effect ` +
183+
`function, a ref, a reactive object, or an array of these types.`
186184
)
187185
}
188186

src/v3/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ export { provide, inject, InjectionKey } from './apiInject'
7070

7171
export { h } from './h'
7272
export { getCurrentInstance } from './currentInstance'
73-
export { useSlots, useAttrs } from './apiSetup'
73+
export { useSlots, useAttrs, mergeDefaults } from './apiSetup'
7474
export { nextTick } from 'core/util/next-tick'
7575
export { set, del } from 'core/observer'
7676

types/index.d.ts

+2
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,8 @@ export {
3535

3636
export * from './v3-manual-apis'
3737
export * from './v3-generated'
38+
// <script setup> helpers
39+
export * from './v3-setup-helpers'
3840

3941
export { Data } from './common'
4042
export { SetupContext } from './v3-setup-context'

types/test/setup-helpers-test.ts

+134
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,134 @@
1+
import { useAttrs, useSlots, SetupContext } from '../index'
2+
import { describe, expectType } from './utils'
3+
4+
describe('defineProps w/ type declaration', () => {
5+
// type declaration
6+
const props = defineProps<{
7+
foo: string
8+
}>()
9+
// explicitly declared type should be refined
10+
expectType<string>(props.foo)
11+
// @ts-expect-error
12+
props.bar
13+
})
14+
15+
describe('defineProps w/ type declaration + withDefaults', () => {
16+
const res = withDefaults(
17+
defineProps<{
18+
number?: number
19+
arr?: string[]
20+
obj?: { x: number }
21+
fn?: (e: string) => void
22+
x?: string
23+
genStr?: string
24+
}>(),
25+
{
26+
number: 123,
27+
arr: () => [],
28+
obj: () => ({ x: 123 }),
29+
fn: () => {},
30+
genStr: () => ''
31+
}
32+
)
33+
34+
res.number + 1
35+
res.arr.push('hi')
36+
res.obj.x
37+
res.fn('hi')
38+
// @ts-expect-error
39+
res.x.slice()
40+
res.genStr.slice()
41+
})
42+
43+
describe('defineProps w/ union type declaration + withDefaults', () => {
44+
withDefaults(
45+
defineProps<{
46+
union1?: number | number[] | { x: number }
47+
union2?: number | number[] | { x: number }
48+
union3?: number | number[] | { x: number }
49+
union4?: number | number[] | { x: number }
50+
}>(),
51+
{
52+
union1: 123,
53+
union2: () => [123],
54+
union3: () => ({ x: 123 }),
55+
union4: () => 123
56+
}
57+
)
58+
})
59+
60+
describe('defineProps w/ runtime declaration', () => {
61+
// runtime declaration
62+
const props = defineProps({
63+
foo: String,
64+
bar: {
65+
type: Number,
66+
default: 1
67+
},
68+
baz: {
69+
type: Array,
70+
required: true
71+
}
72+
})
73+
expectType<{
74+
foo?: string
75+
bar: number
76+
baz: unknown[]
77+
}>(props)
78+
79+
props.foo && props.foo + 'bar'
80+
props.bar + 1
81+
// @ts-expect-error should be readonly
82+
props.bar++
83+
props.baz.push(1)
84+
85+
const props2 = defineProps(['foo', 'bar'])
86+
props2.foo + props2.bar
87+
// @ts-expect-error
88+
props2.baz
89+
})
90+
91+
describe('defineEmits w/ type declaration', () => {
92+
const emit = defineEmits<(e: 'change') => void>()
93+
emit('change')
94+
// @ts-expect-error
95+
emit()
96+
// @ts-expect-error
97+
emit('bar')
98+
99+
type Emits = { (e: 'foo' | 'bar'): void; (e: 'baz', id: number): void }
100+
const emit2 = defineEmits<Emits>()
101+
102+
emit2('foo')
103+
emit2('bar')
104+
emit2('baz', 123)
105+
// @ts-expect-error
106+
emit2('baz')
107+
})
108+
109+
describe('defineEmits w/ runtime declaration', () => {
110+
const emit = defineEmits({
111+
foo: () => {},
112+
bar: null
113+
})
114+
emit('foo')
115+
emit('bar', 123)
116+
// @ts-expect-error
117+
emit('baz')
118+
119+
const emit2 = defineEmits(['foo', 'bar'])
120+
emit2('foo')
121+
emit2('bar', 123)
122+
// @ts-expect-error
123+
emit2('baz')
124+
})
125+
126+
describe('useAttrs', () => {
127+
const attrs = useAttrs()
128+
expectType<Record<string, unknown>>(attrs)
129+
})
130+
131+
describe('useSlots', () => {
132+
const slots = useSlots()
133+
expectType<SetupContext['slots']>(slots)
134+
})

0 commit comments

Comments
 (0)