1
1
/* @flow */
2
2
3
+ import config from 'core/config'
3
4
import { isIE } from 'core/util/env'
4
5
import { addHandler , addProp , getBindingAttr , parseModel } from 'compiler/helpers'
5
6
@@ -40,8 +41,19 @@ export default function model (
40
41
genCheckboxModel ( el , value , modifiers )
41
42
} else if ( tag === 'input' && type === 'radio' ) {
42
43
genRadioModel ( el , value , modifiers )
43
- } else {
44
+ } else if ( tag === 'input' || tag === 'textarea' ) {
44
45
genDefaultModel ( el , value , modifiers )
46
+ } else if ( ! config . isReservedTag ( tag ) ) {
47
+ genComponentModel ( el , value , modifiers )
48
+ // component v-model doesn't need extra runtime
49
+ return false
50
+ } else if ( process . env . NODE_ENV !== 'production' ) {
51
+ warn (
52
+ `<${ el . tag } v-model="${ value } ">: ` +
53
+ `v-model is not supported on this element type. ` +
54
+ 'If you are working with contenteditable, it\'s recommended to ' +
55
+ 'wrap a library dedicated for that purpose inside a custom component.'
56
+ )
45
57
}
46
58
47
59
// ensure runtime directive metadata
@@ -107,6 +119,41 @@ function genRadioModel (
107
119
addHandler ( el , 'click' , genAssignmentCode ( value , valueBinding ) , null , true )
108
120
}
109
121
122
+ function genSelect (
123
+ el : ASTElement ,
124
+ value : string ,
125
+ modifiers : ?ASTModifiers
126
+ ) {
127
+ if ( process . env . NODE_ENV !== 'production' ) {
128
+ el . children . some ( checkOptionWarning )
129
+ }
130
+
131
+ const number = modifiers && modifiers . number
132
+ const selectedVal = `Array.prototype.filter` +
133
+ `.call($event.target.options,function(o){return o.selected})` +
134
+ `.map(function(o){var val = "_value" in o ? o._value : o.value;` +
135
+ `return ${ number ? '_n(val)' : 'val' } })`
136
+
137
+ const assignment = '$event.target.multiple ? $$selectedVal : $$selectedVal[0]'
138
+ let code = `var $$selectedVal = ${ selectedVal } ;`
139
+ code = `${ code } ${ genAssignmentCode ( value , assignment ) } `
140
+ addHandler ( el , 'change' , code , null , true )
141
+ }
142
+
143
+ function checkOptionWarning ( option : any ) : boolean {
144
+ if ( option . type === 1 &&
145
+ option . tag === 'option' &&
146
+ option . attrsMap . selected != null ) {
147
+ warn (
148
+ `<select v-model="${ option . parent . attrsMap [ 'v-model' ] } ">:\n` +
149
+ 'inline selected attributes on <option> will be ignored when using v-model. ' +
150
+ 'Declare initial values in the component\'s data option instead.'
151
+ )
152
+ return true
153
+ }
154
+ return false
155
+ }
156
+
110
157
function genDefaultModel (
111
158
el : ASTElement ,
112
159
value : string ,
@@ -133,60 +180,46 @@ function genDefaultModel (
133
180
const { lazy, number, trim } = modifiers || { }
134
181
const event = lazy || ( isIE && type === 'range' ) ? 'change' : 'input'
135
182
const needCompositionGuard = ! lazy && type !== 'range'
136
- const isNative = el . tag === 'input' || el . tag === 'textarea'
137
183
138
- let valueExpression = isNative
139
- ? `$event.target.value${ trim ? '.trim()' : '' } `
140
- : trim ? `(typeof $event === 'string' ? $event.trim() : $event)` : `$event`
141
- valueExpression = number || type === 'number'
142
- ? `_n(${ valueExpression } )`
143
- : valueExpression
184
+ let valueExpression = '$event.target.value'
185
+ if ( trim ) {
186
+ valueExpression = `$event.target.value.trim()`
187
+ }
188
+ if ( number ) {
189
+ valueExpression = `_n(${ valueExpression } )`
190
+ }
144
191
145
192
let code = genAssignmentCode ( value , valueExpression )
146
- if ( isNative && needCompositionGuard ) {
193
+ if ( needCompositionGuard ) {
147
194
code = `if($event.target.composing)return;${ code } `
148
195
}
149
196
150
- addProp ( el , 'value' , isNative ? `_s( ${ value } )` : `(${ value } )` )
197
+ addProp ( el , 'value' , `(${ value } )` )
151
198
addHandler ( el , event , code , null , true )
152
199
if ( trim || number || type === 'number' ) {
153
200
addHandler ( el , 'blur' , '$forceUpdate()' )
154
201
}
155
202
}
156
203
157
- function genSelect (
158
- el : ASTElement ,
159
- value : string ,
160
- modifiers : ?ASTModifiers
161
- ) {
162
- if ( process . env . NODE_ENV !== 'production' ) {
163
- el . children . some ( checkOptionWarning )
164
- }
165
-
166
- const number = modifiers && modifiers . number
167
- const selectedVal = `Array.prototype.filter` +
168
- `.call($event.target.options,function(o){return o.selected})` +
169
- `.map(function(o){var val = "_value" in o ? o._value : o.value;` +
170
- `return ${ number ? '_n(val)' : 'val' } })`
204
+ function genComponentModel (
205
+ el : ASTElement ,
206
+ value : string ,
207
+ modifiers : ?ASTModifiers
208
+ ) : ?boolean {
209
+ const { number, trim } = modifiers || { }
171
210
172
- const assignment = '$event.target.multiple ? $$selectedVal : $$selectedVal[0]'
173
- let code = `var $$selectedVal = ${ selectedVal } ;`
174
- code = `${ code } ${ genAssignmentCode ( value , assignment ) } `
175
- addHandler ( el , 'change' , code , null , true )
176
- }
211
+ let valueExpression = 'value'
212
+ if ( trim ) {
213
+ valueExpression = `(typeof value === 'string' ? value.trim() : value)`
214
+ }
215
+ if ( number ) {
216
+ valueExpression = `_n(${ valueExpression } )`
217
+ }
177
218
178
- function checkOptionWarning ( option : any ) : boolean {
179
- if ( option . type === 1 &&
180
- option . tag === 'option' &&
181
- option . attrsMap . selected != null ) {
182
- warn (
183
- `<select v-model="${ option . parent . attrsMap [ 'v-model' ] } ">:\n` +
184
- 'inline selected attributes on <option> will be ignored when using v-model. ' +
185
- 'Declare initial values in the component\'s data option instead.'
186
- )
187
- return true
219
+ el . model = {
220
+ value,
221
+ callback : `function (value) {${ genAssignmentCode ( value , valueExpression ) } }`
188
222
}
189
- return false
190
223
}
191
224
192
225
function genAssignmentCode ( value : string , assignment : string ) : string {
0 commit comments