|
1 | 1 | import type { LVal, Node, TSType } from '@babel/types'
|
2 | 2 | import type { ScriptCompileContext } from './context'
|
3 | 3 | import { inferRuntimeType } from './resolveType'
|
4 |
| -import { |
5 |
| - UNKNOWN_TYPE, |
6 |
| - concatStrings, |
7 |
| - isCallOf, |
8 |
| - toRuntimeTypeString, |
9 |
| -} from './utils' |
| 4 | +import { UNKNOWN_TYPE, isCallOf, toRuntimeTypeString } from './utils' |
10 | 5 | import { BindingTypes, unwrapTSNode } from '@vue/compiler-dom'
|
11 | 6 |
|
12 | 7 | export const DEFINE_MODEL = 'defineModel'
|
@@ -124,44 +119,50 @@ export function genModelProps(ctx: ScriptCompileContext) {
|
124 | 119 |
|
125 | 120 | const isProd = !!ctx.options.isProd
|
126 | 121 | let modelPropsDecl = ''
|
127 |
| - for (const [name, { type, options }] of Object.entries(ctx.modelDecls)) { |
| 122 | + for (const [name, { type, options: runtimeOptions }] of Object.entries( |
| 123 | + ctx.modelDecls, |
| 124 | + )) { |
128 | 125 | let skipCheck = false
|
129 |
| - |
| 126 | + let codegenOptions = `` |
130 | 127 | let runtimeTypes = type && inferRuntimeType(ctx, type)
|
131 | 128 | if (runtimeTypes) {
|
132 | 129 | const hasBoolean = runtimeTypes.includes('Boolean')
|
| 130 | + const hasFunction = runtimeTypes.includes('Function') |
133 | 131 | const hasUnknownType = runtimeTypes.includes(UNKNOWN_TYPE)
|
134 | 132 |
|
135 |
| - if (isProd || hasUnknownType) { |
136 |
| - runtimeTypes = runtimeTypes.filter( |
137 |
| - t => |
138 |
| - t === 'Boolean' || |
139 |
| - (hasBoolean && t === 'String') || |
140 |
| - (t === 'Function' && options), |
141 |
| - ) |
| 133 | + if (hasUnknownType) { |
| 134 | + if (hasBoolean || hasFunction) { |
| 135 | + runtimeTypes = runtimeTypes.filter(t => t !== UNKNOWN_TYPE) |
| 136 | + skipCheck = true |
| 137 | + } else { |
| 138 | + runtimeTypes = ['null'] |
| 139 | + } |
| 140 | + } |
142 | 141 |
|
143 |
| - skipCheck = !isProd && hasUnknownType && runtimeTypes.length > 0 |
| 142 | + if (!isProd) { |
| 143 | + codegenOptions = |
| 144 | + `type: ${toRuntimeTypeString(runtimeTypes)}` + |
| 145 | + (skipCheck ? ', skipCheck: true' : '') |
| 146 | + } else if (hasBoolean || (runtimeOptions && hasFunction)) { |
| 147 | + // preserve types if contains boolean, or |
| 148 | + // function w/ runtime options that may contain default |
| 149 | + codegenOptions = `type: ${toRuntimeTypeString(runtimeTypes)}` |
| 150 | + } else { |
| 151 | + // able to drop types in production |
144 | 152 | }
|
145 | 153 | }
|
146 | 154 |
|
147 |
| - let runtimeType = |
148 |
| - (runtimeTypes && |
149 |
| - runtimeTypes.length > 0 && |
150 |
| - toRuntimeTypeString(runtimeTypes)) || |
151 |
| - undefined |
152 |
| - |
153 |
| - const codegenOptions = concatStrings([ |
154 |
| - runtimeType && `type: ${runtimeType}`, |
155 |
| - skipCheck && 'skipCheck: true', |
156 |
| - ]) |
157 |
| - |
158 | 155 | let decl: string
|
159 |
| - if (runtimeType && options) { |
| 156 | + if (codegenOptions && runtimeOptions) { |
160 | 157 | decl = ctx.isTS
|
161 |
| - ? `{ ${codegenOptions}, ...${options} }` |
162 |
| - : `Object.assign({ ${codegenOptions} }, ${options})` |
| 158 | + ? `{ ${codegenOptions}, ...${runtimeOptions} }` |
| 159 | + : `Object.assign({ ${codegenOptions} }, ${runtimeOptions})` |
| 160 | + } else if (codegenOptions) { |
| 161 | + decl = `{ ${codegenOptions} }` |
| 162 | + } else if (runtimeOptions) { |
| 163 | + decl = runtimeOptions |
163 | 164 | } else {
|
164 |
| - decl = options || (runtimeType ? `{ ${codegenOptions} }` : '{}') |
| 165 | + decl = `{}` |
165 | 166 | }
|
166 | 167 | modelPropsDecl += `\n ${JSON.stringify(name)}: ${decl},`
|
167 | 168 |
|
|
0 commit comments