Skip to content

Commit e1657fd

Browse files
committed
fix(ssr): properly render <select v-model> initial state
fix #6986
1 parent e80104e commit e1657fd

File tree

4 files changed

+138
-2
lines changed

4 files changed

+138
-2
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
import show from './show'
2+
import model from './model'
23

34
export default {
4-
show
5+
show,
6+
model
57
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
/* @flow */
2+
3+
import { looseEqual, looseIndexOf } from 'shared/util'
4+
5+
// this is only applied for <select v-model> because it is the only edge case
6+
// that must be done at runtime instead of compile time.
7+
export default function model (node: VNodeWithData, dir: VNodeDirective) {
8+
if (!node.children) return
9+
const value = dir.value
10+
const isMultiple = node.data.attrs && node.data.attrs.multiple
11+
for (let i = 0, l = node.children.length; i < l; i++) {
12+
const option = node.children[i]
13+
if (option.tag === 'option') {
14+
if (isMultiple) {
15+
const selected =
16+
Array.isArray(value) &&
17+
(looseIndexOf(value, getValue(option)) > -1)
18+
if (selected) {
19+
setSelected(option)
20+
}
21+
} else {
22+
if (looseEqual(value, getValue(option))) {
23+
setSelected(option)
24+
return
25+
}
26+
}
27+
}
28+
}
29+
}
30+
31+
function getValue (option) {
32+
const data = option.data || {}
33+
return (
34+
(data.attrs && data.attrs.value) ||
35+
(data.domProps && data.domProps.value) ||
36+
(option.children && option.children[0] && option.children[0].text)
37+
)
38+
}
39+
40+
function setSelected (option) {
41+
const data = option.data || (option.data = {})
42+
const attrs = data.attrs || (data.attrs = {})
43+
attrs.selected = ''
44+
}

src/server/optimizing-compiler/optimizer.js

+13-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,8 @@ function isUnOptimizableTree (node: ASTNode): boolean {
113113
return (
114114
isBuiltInTag(node.tag) || // built-in (slot, component)
115115
!isPlatformReservedTag(node.tag) || // custom component
116-
!!node.component // "is" component
116+
!!node.component || // "is" component
117+
isSelectWithModel(node) // <select v-model> requires runtime inspection
117118
)
118119
}
119120

@@ -126,3 +127,14 @@ function hasCustomDirective (node: ASTNode): ?boolean {
126127
node.directives.some(d => !isBuiltInDir(d.name))
127128
)
128129
}
130+
131+
// <select v-model> cannot be optimized because it requires a runtime check
132+
// to determine proper selected option
133+
function isSelectWithModel (node: ASTNode): ?boolean {
134+
return (
135+
node.type === 1 &&
136+
node.tag === 'select' &&
137+
node.directives &&
138+
node.directives.some(d => d.name === 'model')
139+
)
140+
}

test/ssr/ssr-string.spec.js

+78
Original file line numberDiff line numberDiff line change
@@ -1034,6 +1034,84 @@ describe('SSR: renderToString', () => {
10341034
done()
10351035
})
10361036
})
1037+
1038+
it('render v-model with <select> (value binding)', done => {
1039+
renderVmWithOptions({
1040+
data: {
1041+
selected: 2,
1042+
options: [
1043+
{ id: 1, label: 'one' },
1044+
{ id: 2, label: 'two' }
1045+
]
1046+
},
1047+
template: `
1048+
<div>
1049+
<select v-model="selected">
1050+
<option v-for="o in options" :value="o.id">{{ o.label }}</option>
1051+
</select>
1052+
</div>
1053+
`
1054+
}, result => {
1055+
expect(result).toContain(
1056+
'<select>' +
1057+
'<option value="1">one</option>' +
1058+
'<option selected="selected" value="2">two</option>' +
1059+
'</select>'
1060+
)
1061+
done()
1062+
})
1063+
})
1064+
1065+
it('render v-model with <select> (static value)', done => {
1066+
renderVmWithOptions({
1067+
data: {
1068+
selected: 2
1069+
},
1070+
template: `
1071+
<div>
1072+
<select v-model="selected">
1073+
<option value="1">one</option>
1074+
<option value="2">two</option>
1075+
</select>
1076+
</div>
1077+
`
1078+
}, result => {
1079+
expect(result).toContain(
1080+
'<select>' +
1081+
'<option value="1">one</option> ' +
1082+
'<option value="2" selected="selected">two</option>' +
1083+
'</select>'
1084+
)
1085+
done()
1086+
})
1087+
})
1088+
1089+
it('render v-model with <select> (text as value)', done => {
1090+
renderVmWithOptions({
1091+
data: {
1092+
selected: 2,
1093+
options: [
1094+
{ id: 1, label: 'one' },
1095+
{ id: 2, label: 'two' }
1096+
]
1097+
},
1098+
template: `
1099+
<div>
1100+
<select v-model="selected">
1101+
<option v-for="o in options">{{ o.id }}</option>
1102+
</select>
1103+
</div>
1104+
`
1105+
}, result => {
1106+
expect(result).toContain(
1107+
'<select>' +
1108+
'<option>1</option>' +
1109+
'<option selected="selected">2</option>' +
1110+
'</select>'
1111+
)
1112+
done()
1113+
})
1114+
})
10371115
})
10381116

10391117
function renderVmWithOptions (options, cb) {

0 commit comments

Comments
 (0)