Skip to content

Commit b080a14

Browse files
committed
fix(ssr): fix hydration mismatch with adjacent text node from slots
fix vuejs/vue-loader#974
1 parent 94512f3 commit b080a14

File tree

2 files changed

+32
-6
lines changed

2 files changed

+32
-6
lines changed

src/core/vdom/helpers/normalize-children.js

+13-6
Original file line numberDiff line numberDiff line change
@@ -42,28 +42,35 @@ function isTextNode (node): boolean {
4242

4343
function normalizeArrayChildren (children: any, nestedIndex?: string): Array<VNode> {
4444
const res = []
45-
let i, c, last
45+
let i, c, lastIndex, last
4646
for (i = 0; i < children.length; i++) {
4747
c = children[i]
4848
if (isUndef(c) || typeof c === 'boolean') continue
49-
last = res[res.length - 1]
49+
lastIndex = res.length - 1
50+
last = res[lastIndex]
5051
// nested
51-
if (Array.isArray(c)) {
52-
res.push.apply(res, normalizeArrayChildren(c, `${nestedIndex || ''}_${i}`))
52+
if (Array.isArray(c) && c.length > 0) {
53+
c = normalizeArrayChildren(c, `${nestedIndex || ''}_${i}`)
54+
// merge adjacent text nodes
55+
if (isTextNode(c[0]) && isTextNode(last)) {
56+
res[lastIndex] = createTextVNode(last.text + (c[0]: any).text)
57+
c.shift()
58+
}
59+
res.push.apply(res, c)
5360
} else if (isPrimitive(c)) {
5461
if (isTextNode(last)) {
5562
// merge adjacent text nodes
5663
// this is necessary for SSR hydration because text nodes are
5764
// essentially merged when rendered to HTML strings
58-
(last: any).text += String(c)
65+
res[lastIndex] = createTextVNode(last.text + c)
5966
} else if (c !== '') {
6067
// convert primitive to vnode
6168
res.push(createTextVNode(c))
6269
}
6370
} else {
6471
if (isTextNode(c) && isTextNode(last)) {
6572
// merge adjacent text nodes
66-
res[res.length - 1] = createTextVNode(last.text + c.text)
73+
res[lastIndex] = createTextVNode(last.text + c.text)
6774
} else {
6875
// default key for nested array children (likely generated by v-for)
6976
if (isTrue(children._isVList) &&

test/unit/modules/vdom/patch/hydration.spec.js

+19
Original file line numberDiff line numberDiff line change
@@ -323,4 +323,23 @@ describe('vdom patch: hydration', () => {
323323

324324
expect('not matching server-rendered content').toHaveBeenWarned()
325325
})
326+
327+
it('should hydrate with adjacent text nodes from array children (e.g. slots)', () => {
328+
const dom = createMockSSRDOM('<div>foo</div> hello')
329+
330+
new Vue({
331+
template: `<test>hello</test>`,
332+
components: {
333+
test: {
334+
template: `
335+
<div>
336+
<div>foo</div>
337+
<slot/>
338+
</div>
339+
`
340+
}
341+
}
342+
}).$mount(dom)
343+
expect('not matching server-rendered content').not.toHaveBeenWarned()
344+
})
326345
})

0 commit comments

Comments
 (0)