Skip to content

Commit f3fe012

Browse files
committed
feat(v-model): support dynamic input type binding
1 parent 2876ed8 commit f3fe012

File tree

7 files changed

+254
-43
lines changed

7 files changed

+254
-43
lines changed

flow/compiler.js

+8-3
Original file line numberDiff line numberDiff line change
@@ -36,16 +36,19 @@ declare type CompiledResult = {
3636
};
3737

3838
declare type ModuleOptions = {
39-
preTransformNode: (el: ASTElement) => void;
40-
transformNode: (el: ASTElement) => void; // transform an element's AST node
39+
// returning an ASTElement from pre/transforms replaces the element
40+
preTransformNode: (el: ASTElement) => ?ASTElement;
41+
transformNode: (el: ASTElement) => ?ASTElement;
42+
// cannot return replacement in postTransform because tree is already finalized
4143
postTransformNode: (el: ASTElement) => void;
4244
genData: (el: ASTElement) => string; // generate extra data string for an element
4345
transformCode?: (el: ASTElement, code: string) => string; // further transform generated code for an element
4446
staticKeys?: Array<string>; // AST properties to be considered static
4547
};
4648

4749
declare type ASTModifiers = { [key: string]: boolean };
48-
declare type ASTIfConditions = Array<{ exp: ?string; block: ASTElement }>;
50+
declare type ASTIfCondition = { exp: ?string; block: ASTElement };
51+
declare type ASTIfConditions = Array<ASTIfCondition>;
4952

5053
declare type ASTElementHandler = {
5154
value: string;
@@ -74,6 +77,8 @@ declare type ASTElement = {
7477
parent: ASTElement | void;
7578
children: Array<ASTNode>;
7679

80+
processed?: true;
81+
7782
static?: boolean;
7883
staticRoot?: boolean;
7984
staticInFor?: boolean;

src/compiler/helpers.js

+12-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,15 @@ export function getBindingAttr (
104104
}
105105
}
106106

107-
export function getAndRemoveAttr (el: ASTElement, name: string): ?string {
107+
// note: this only removes the attr from the Array (attrsList) so that it
108+
// doesn't get processed by processAttrs.
109+
// By default it does NOT remove it from the map (attrsMap) because the map is
110+
// needed during codegen.
111+
export function getAndRemoveAttr (
112+
el: ASTElement,
113+
name: string,
114+
removeFromMap?: boolean
115+
): ?string {
108116
let val
109117
if ((val = el.attrsMap[name]) != null) {
110118
const list = el.attrsList
@@ -115,5 +123,8 @@ export function getAndRemoveAttr (el: ASTElement, name: string): ?string {
115123
}
116124
}
117125
}
126+
if (removeFromMap) {
127+
delete el.attrsMap[name]
128+
}
118129
return val
119130
}

src/compiler/parser/index.js

+40-25
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,22 @@ let platformIsPreTag
4040
let platformMustUseProp
4141
let platformGetTagNamespace
4242

43+
type Attr = { name: string; value: string }
44+
export function createASTElement (
45+
tag: string,
46+
attrs: Array<Attr>,
47+
parent: ASTElement | void
48+
): ASTElement {
49+
return {
50+
type: 1,
51+
tag,
52+
attrsList: attrs,
53+
attrsMap: makeAttrsMap(attrs),
54+
parent,
55+
children: []
56+
}
57+
}
58+
4359
/**
4460
* Convert HTML string to AST.
4561
*/
@@ -102,14 +118,7 @@ export function parse (
102118
attrs = guardIESVGBug(attrs)
103119
}
104120

105-
const element: ASTElement = {
106-
type: 1,
107-
tag,
108-
attrsList: attrs,
109-
attrsMap: makeAttrsMap(attrs),
110-
parent: currentParent,
111-
children: []
112-
}
121+
let element: ASTElement = createASTElement(tag, attrs, currentParent)
113122
if (ns) {
114123
element.ns = ns
115124
}
@@ -125,7 +134,7 @@ export function parse (
125134

126135
// apply pre-transforms
127136
for (let i = 0; i < preTransforms.length; i++) {
128-
preTransforms[i](element, options)
137+
element = preTransforms[i](element, options) || element
129138
}
130139

131140
if (!inVPre) {
@@ -139,23 +148,13 @@ export function parse (
139148
}
140149
if (inVPre) {
141150
processRawAttrs(element)
142-
} else {
151+
} else if (!element.processed) {
152+
// structural directives
143153
processFor(element)
144154
processIf(element)
145155
processOnce(element)
146-
processKey(element)
147-
148-
// determine whether this is a plain element after
149-
// removing structural attributes
150-
element.plain = !element.key && !attrs.length
151-
152-
processRef(element)
153-
processSlot(element)
154-
processComponent(element)
155-
for (let i = 0; i < transforms.length; i++) {
156-
transforms[i](element, options)
157-
}
158-
processAttrs(element)
156+
// element-scope stuff
157+
processElement(element, options)
159158
}
160159

161160
function checkRootConstraints (el) {
@@ -309,6 +308,22 @@ function processRawAttrs (el) {
309308
}
310309
}
311310

311+
export function processElement (element: ASTElement, options: CompilerOptions) {
312+
processKey(element)
313+
314+
// determine whether this is a plain element after
315+
// removing structural attributes
316+
element.plain = !element.key && !element.attrsList.length
317+
318+
processRef(element)
319+
processSlot(element)
320+
processComponent(element)
321+
for (let i = 0; i < transforms.length; i++) {
322+
element = transforms[i](element, options) || element
323+
}
324+
processAttrs(element)
325+
}
326+
312327
function processKey (el) {
313328
const exp = getBindingAttr(el, 'key')
314329
if (exp) {
@@ -327,7 +342,7 @@ function processRef (el) {
327342
}
328343
}
329344

330-
function processFor (el) {
345+
export function processFor (el: ASTElement) {
331346
let exp
332347
if ((exp = getAndRemoveAttr(el, 'v-for'))) {
333348
const inMatch = exp.match(forAliasRE)
@@ -403,7 +418,7 @@ function findPrevElement (children: Array<any>): ASTElement | void {
403418
}
404419
}
405420

406-
function addIfCondition (el, condition) {
421+
export function addIfCondition (el: ASTElement, condition: ASTIfCondition) {
407422
if (!el.ifConditions) {
408423
el.ifConditions = []
409424
}

src/platforms/web/compiler/directives/model.js

-7
Original file line numberDiff line numberDiff line change
@@ -22,13 +22,6 @@ export default function model (
2222
const type = el.attrsMap.type
2323

2424
if (process.env.NODE_ENV !== 'production') {
25-
const dynamicType = el.attrsMap['v-bind:type'] || el.attrsMap[':type']
26-
if (tag === 'input' && dynamicType) {
27-
warn(
28-
`<input :type="${dynamicType}" v-model="${value}">:\n` +
29-
`v-model does not support dynamic input types. Use v-if branches instead.`
30-
)
31-
}
3225
// inputs with type="file" are read only and setting the input's
3326
// value will throw an error.
3427
if (tag === 'input' && type === 'file') {
+3-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
import klass from './class'
22
import style from './style'
3+
import model from './model'
34

45
export default [
56
klass,
6-
style
7+
style,
8+
model
79
]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
/* @flow */
2+
3+
/**
4+
* Expand input[v-model] with dyanmic type bindings into v-if-else chains
5+
* Turn this:
6+
* <input v-model="data[type]" :type="type">
7+
* into this:
8+
* <input v-if="type === 'checkbox'" type="checkbox" v-model="data[type]">
9+
* <input v-else-if="type === 'radio'" type="radio" v-model="data[type]">
10+
* <input v-else :type="type" v-model="data[type]">
11+
*/
12+
13+
import {
14+
getBindingAttr,
15+
getAndRemoveAttr
16+
} from 'compiler/helpers'
17+
18+
import {
19+
processFor,
20+
processElement,
21+
addIfCondition,
22+
createASTElement
23+
} from 'compiler/parser/index'
24+
25+
function preTransformNode (el: ASTElement, options: CompilerOptions) {
26+
if (el.tag === 'input') {
27+
const map = el.attrsMap
28+
if (map['v-model'] && (map['v-bind:type'] || map[':type'])) {
29+
const typeBinding: any = getBindingAttr(el, 'type')
30+
const ifCondition = getAndRemoveAttr(el, 'v-if', true)
31+
// 1. checkbox
32+
const branch0 = cloneASTElement(el)
33+
// process for on the main node
34+
processFor(branch0)
35+
addRawAttr(branch0, 'type', 'checkbox')
36+
processElement(branch0, options)
37+
branch0.processed = true // prevent it from double-processed
38+
branch0.if = `type==='checkbox'` + (ifCondition ? `&&(${ifCondition})` : ``)
39+
addIfCondition(branch0, {
40+
exp: branch0.if,
41+
block: branch0
42+
})
43+
// 2. add radio else-if condition
44+
const branch1 = cloneASTElement(el)
45+
getAndRemoveAttr(branch1, 'v-for', true)
46+
addRawAttr(branch1, 'type', 'radio')
47+
processElement(branch1, options)
48+
addIfCondition(branch0, {
49+
exp: `type==='radio'` + (ifCondition ? `&&(${ifCondition})` : ``),
50+
block: branch1
51+
})
52+
// 3. other
53+
const branch2 = cloneASTElement(el)
54+
getAndRemoveAttr(branch2, 'v-for', true)
55+
addRawAttr(branch2, ':type', typeBinding)
56+
processElement(branch2, options)
57+
addIfCondition(branch0, {
58+
exp: ifCondition,
59+
block: branch2
60+
})
61+
return branch0
62+
}
63+
}
64+
}
65+
66+
function cloneASTElement (el) {
67+
return createASTElement(el.tag, el.attrsList.slice(), el.parent)
68+
}
69+
70+
function addRawAttr (el, name, value) {
71+
el.attrsMap[name] = value
72+
el.attrsList.push({ name, value })
73+
}
74+
75+
export default {
76+
preTransformNode
77+
}

0 commit comments

Comments
 (0)