Skip to content

Commit 0f00f8f

Browse files
committed
fix(ssr): better handle v-html hydration
fix #6519
1 parent f323719 commit 0f00f8f

File tree

2 files changed

+64
-19
lines changed

2 files changed

+64
-19
lines changed

src/core/vdom/patch.js

+38-19
Original file line numberDiff line numberDiff line change
@@ -547,27 +547,46 @@ export function createPatchFunction (backend) {
547547
if (!elm.hasChildNodes()) {
548548
createChildren(vnode, children, insertedVnodeQueue)
549549
} else {
550-
let childrenMatch = true
551-
let childNode = elm.firstChild
552-
for (let i = 0; i < children.length; i++) {
553-
if (!childNode || !hydrate(childNode, children[i], insertedVnodeQueue)) {
554-
childrenMatch = false
555-
break
550+
// v-html and domProps: innerHTML
551+
if (isDef(i = data) && isDef(i = i.domProps) && isDef(i = i.innerHTML)) {
552+
if (i !== elm.innerHTML) {
553+
/* istanbul ignore if */
554+
if (process.env.NODE_ENV !== 'production' &&
555+
typeof console !== 'undefined' &&
556+
!bailed
557+
) {
558+
bailed = true
559+
console.warn('Parent: ', elm)
560+
console.warn('server innerHTML: ', i)
561+
console.warn('client innerHTML: ', elm.innerHTML)
562+
}
563+
return false
556564
}
557-
childNode = childNode.nextSibling
558-
}
559-
// if childNode is not null, it means the actual childNodes list is
560-
// longer than the virtual children list.
561-
if (!childrenMatch || childNode) {
562-
if (process.env.NODE_ENV !== 'production' &&
563-
typeof console !== 'undefined' &&
564-
!bailed
565-
) {
566-
bailed = true
567-
console.warn('Parent: ', elm)
568-
console.warn('Mismatching childNodes vs. VNodes: ', elm.childNodes, children)
565+
} else {
566+
// iterate and compare children lists
567+
let childrenMatch = true
568+
let childNode = elm.firstChild
569+
for (let i = 0; i < children.length; i++) {
570+
if (!childNode || !hydrate(childNode, children[i], insertedVnodeQueue)) {
571+
childrenMatch = false
572+
break
573+
}
574+
childNode = childNode.nextSibling
575+
}
576+
// if childNode is not null, it means the actual childNodes list is
577+
// longer than the virtual children list.
578+
if (!childrenMatch || childNode) {
579+
/* istanbul ignore if */
580+
if (process.env.NODE_ENV !== 'production' &&
581+
typeof console !== 'undefined' &&
582+
!bailed
583+
) {
584+
bailed = true
585+
console.warn('Parent: ', elm)
586+
console.warn('Mismatching childNodes vs. VNodes: ', elm.childNodes, children)
587+
}
588+
return false
569589
}
570-
return false
571590
}
572591
}
573592
}

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

+26
Original file line numberDiff line numberDiff line change
@@ -297,4 +297,30 @@ describe('vdom patch: hydration', () => {
297297
done()
298298
}, 50)
299299
})
300+
301+
it('should hydrate v-html with children', () => {
302+
const dom = createMockSSRDOM('<span>foo</span>')
303+
304+
new Vue({
305+
data: {
306+
html: `<span>foo</span>`
307+
},
308+
template: `<div v-html="html">hello</div>`
309+
}).$mount(dom)
310+
311+
expect('not matching server-rendered content').not.toHaveBeenWarned()
312+
})
313+
314+
it('should warn mismatching v-html', () => {
315+
const dom = createMockSSRDOM('<span>bar</span>')
316+
317+
new Vue({
318+
data: {
319+
html: `<span>foo</span>`
320+
},
321+
template: `<div v-html="html">hello</div>`
322+
}).$mount(dom)
323+
324+
expect('not matching server-rendered content').toHaveBeenWarned()
325+
})
300326
})

0 commit comments

Comments
 (0)