Skip to content

Commit f44c3b3

Browse files
committedJul 16, 2024··
fix(hydration): handle consectuvie text nodes during hydration
close #7285 close #7301
1 parent ae52a37 commit f44c3b3

File tree

2 files changed

+41
-1
lines changed

2 files changed

+41
-1
lines changed
 

‎packages/runtime-core/__tests__/hydration.spec.ts

+20
Original file line numberDiff line numberDiff line change
@@ -148,6 +148,15 @@ describe('SSR hydration', () => {
148148
expect(container.innerHTML).toBe(`<div class="bar">bar</div>`)
149149
})
150150

151+
// #7285
152+
test('element with multiple continuous text vnodes', async () => {
153+
// should no mismatch warning
154+
const { container } = mountWithHydration('<div>fooo</div>', () =>
155+
h('div', ['fo', createTextVNode('o'), 'o']),
156+
)
157+
expect(container.textContent).toBe('fooo')
158+
})
159+
151160
test('element with elements children', async () => {
152161
const msg = ref('foo')
153162
const fn = vi.fn()
@@ -239,6 +248,17 @@ describe('SSR hydration', () => {
239248
)
240249
})
241250

251+
// #7285
252+
test('Fragment (multiple continuous text vnodes)', async () => {
253+
// should no mismatch warning
254+
const { container } = mountWithHydration('<!--[-->fooo<!--]-->', () => [
255+
'fo',
256+
createTextVNode('o'),
257+
'o',
258+
])
259+
expect(container.textContent).toBe('fooo')
260+
})
261+
242262
test('Teleport', async () => {
243263
const msg = ref('foo')
244264
const fn = vi.fn()

‎packages/runtime-core/src/hydration.ts

+21-1
Original file line numberDiff line numberDiff line change
@@ -531,7 +531,27 @@ export function createHydrationFunctions(
531531
const vnode = optimized
532532
? children[i]
533533
: (children[i] = normalizeVNode(children[i]))
534+
const isText = vnode.type === Text
534535
if (node) {
536+
if (isText && !optimized) {
537+
// #7285 possible consecutive text vnodes from manual render fns or
538+
// JSX-compiled fns, but on the client the browser parses only 1 text
539+
// node.
540+
// look ahead for next possible text vnode
541+
let next = children[i + 1]
542+
if (next && (next = normalizeVNode(next)).type === Text) {
543+
// create an extra TextNode on the client for the next vnode to
544+
// adopt
545+
insert(
546+
createText(
547+
(node as Text).data.slice((vnode.children as string).length),
548+
),
549+
container,
550+
nextSibling(node),
551+
)
552+
;(node as Text).data = vnode.children as string
553+
}
554+
}
535555
node = hydrateNode(
536556
node,
537557
vnode,
@@ -540,7 +560,7 @@ export function createHydrationFunctions(
540560
slotScopeIds,
541561
optimized,
542562
)
543-
} else if (vnode.type === Text && !vnode.children) {
563+
} else if (isText && !vnode.children) {
544564
// #7215 create a TextNode for empty text node
545565
// because server rendered HTML won't contain a text node
546566
insert((vnode.el = createText('')), container)

0 commit comments

Comments
 (0)
Please sign in to comment.