Skip to content

Commit e866434

Browse files
committed
feat(portal): SSR support for multi portal shared target
1 parent aafb880 commit e866434

File tree

7 files changed

+130
-32
lines changed

7 files changed

+130
-32
lines changed

packages/compiler-ssr/__tests__/ssrPortal.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ describe('ssr compile: portal', () => {
77
"const { ssrRenderPortal: _ssrRenderPortal } = require(\\"@vue/server-renderer\\")
88
99
return function ssrRender(_ctx, _push, _parent) {
10-
_ssrRenderPortal((_push) => {
10+
_ssrRenderPortal(_push, (_push) => {
1111
_push(\`<div></div>\`)
1212
}, _ctx.target, _parent)
1313
}"

packages/compiler-ssr/src/transforms/ssrTransformPortal.ts

+1
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@ export function ssrProcessPortal(
5252
contentRenderFn.body = processChildrenAsStatement(node.children, context)
5353
context.pushStatement(
5454
createCallExpression(context.helper(SSR_RENDER_PORTAL), [
55+
`_push`,
5556
contentRenderFn,
5657
target,
5758
`_parent`

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

+65-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
} from '@vue/runtime-dom'
1313
import { renderToString } from '@vue/server-renderer'
1414
import { mockWarn } from '@vue/shared'
15+
import { SSRContext } from 'packages/server-renderer/src/renderToString'
1516

1617
function mountWithHydration(html: string, render: () => any) {
1718
const container = document.createElement('div')
@@ -157,7 +158,7 @@ describe('SSR hydration', () => {
157158
const fn = jest.fn()
158159
const portalContainer = document.createElement('div')
159160
portalContainer.id = 'portal'
160-
portalContainer.innerHTML = `<span>foo</span><span class="foo"></span>`
161+
portalContainer.innerHTML = `<span>foo</span><span class="foo"></span><!---->`
161162
document.body.appendChild(portalContainer)
162163

163164
const { vnode, container } = mountWithHydration('<!--portal-->', () =>
@@ -182,7 +183,69 @@ describe('SSR hydration', () => {
182183
msg.value = 'bar'
183184
await nextTick()
184185
expect(portalContainer.innerHTML).toBe(
185-
`<span>bar</span><span class="bar"></span>`
186+
`<span>bar</span><span class="bar"></span><!---->`
187+
)
188+
})
189+
190+
test('Portal (multiple + integration)', async () => {
191+
const msg = ref('foo')
192+
const fn1 = jest.fn()
193+
const fn2 = jest.fn()
194+
195+
const Comp = () => [
196+
h(Portal, { target: '#portal2' }, [
197+
h('span', msg.value),
198+
h('span', { class: msg.value, onClick: fn1 })
199+
]),
200+
h(Portal, { target: '#portal2' }, [
201+
h('span', msg.value + '2'),
202+
h('span', { class: msg.value + '2', onClick: fn2 })
203+
])
204+
]
205+
206+
const portalContainer = document.createElement('div')
207+
portalContainer.id = 'portal2'
208+
const ctx: SSRContext = {}
209+
const mainHtml = await renderToString(h(Comp), ctx)
210+
expect(mainHtml).toMatchInlineSnapshot(
211+
`"<!--[--><!--portal--><!--portal--><!--]-->"`
212+
)
213+
214+
const portalHtml = ctx.portals!['#portal2']
215+
expect(portalHtml).toMatchInlineSnapshot(
216+
`"<span>foo</span><span class=\\"foo\\"></span><!----><span>foo2</span><span class=\\"foo2\\"></span><!---->"`
217+
)
218+
219+
portalContainer.innerHTML = portalHtml
220+
document.body.appendChild(portalContainer)
221+
222+
const { vnode, container } = mountWithHydration(mainHtml, Comp)
223+
expect(vnode.el).toBe(container.firstChild)
224+
const portalVnode1 = (vnode.children as VNode[])[0]
225+
const portalVnode2 = (vnode.children as VNode[])[1]
226+
expect(portalVnode1.el).toBe(container.childNodes[1])
227+
expect(portalVnode2.el).toBe(container.childNodes[2])
228+
229+
expect((portalVnode1 as any).children[0].el).toBe(
230+
portalContainer.childNodes[0]
231+
)
232+
expect(portalVnode1.anchor).toBe(portalContainer.childNodes[2])
233+
expect((portalVnode2 as any).children[0].el).toBe(
234+
portalContainer.childNodes[3]
235+
)
236+
expect(portalVnode2.anchor).toBe(portalContainer.childNodes[5])
237+
238+
// // event handler
239+
triggerEvent('click', portalContainer.querySelector('.foo')!)
240+
expect(fn1).toHaveBeenCalled()
241+
242+
triggerEvent('click', portalContainer.querySelector('.foo2')!)
243+
expect(fn2).toHaveBeenCalled()
244+
245+
msg.value = 'bar'
246+
await nextTick()
247+
expect(portalContainer.innerHTML).toMatchInlineSnapshot(
248+
`"<span>bar</span><span class=\\"bar\\"></span><!----><span>bar2</span><span class=\\"bar2\\"></span><!---->"`
186249
)
187250
})
188251

packages/runtime-core/src/hydration.ts

+10-2
Original file line numberDiff line numberDiff line change
@@ -366,6 +366,11 @@ export function createHydrationFunctions(
366366
}
367367
}
368368

369+
interface PortalTargetElement extends Element {
370+
// last portal target
371+
_lpa?: Node | null
372+
}
373+
369374
const hydratePortal = (
370375
vnode: VNode,
371376
parentComponent: ComponentInternalInstance | null,
@@ -377,14 +382,17 @@ export function createHydrationFunctions(
377382
? document.querySelector(targetSelector)
378383
: targetSelector)
379384
if (target && vnode.shapeFlag & ShapeFlags.ARRAY_CHILDREN) {
380-
hydrateChildren(
381-
target.firstChild,
385+
vnode.anchor = hydrateChildren(
386+
// if multiple portals rendered to the same target element, we need to
387+
// pick up from where the last portal finished instead of the first node
388+
(target as PortalTargetElement)._lpa || target.firstChild,
382389
vnode,
383390
target,
384391
parentComponent,
385392
parentSuspense,
386393
optimized
387394
)
395+
;(target as PortalTargetElement)._lpa = nextSibling(vnode.anchor as Node)
388396
} else if (__DEV__) {
389397
warn(
390398
`Attempting to hydrate portal but target ${targetSelector} does not ` +

packages/server-renderer/__tests__/ssrPortal.spec.ts

+29-7
Original file line numberDiff line numberDiff line change
@@ -4,16 +4,15 @@ import { ssrRenderPortal } from '../src/helpers/ssrRenderPortal'
44

55
describe('ssrRenderPortal', () => {
66
test('portal rendering (compiled)', async () => {
7-
const ctx = {
8-
portals: {}
9-
} as SSRContext
10-
await renderToString(
7+
const ctx: SSRContext = {}
8+
const html = await renderToString(
119
createApp({
1210
data() {
1311
return { msg: 'hello' }
1412
},
1513
ssrRender(_ctx, _push, _parent) {
1614
ssrRenderPortal(
15+
_push,
1716
_push => {
1817
_push(`<div>content</div>`)
1918
},
@@ -24,12 +23,13 @@ describe('ssrRenderPortal', () => {
2423
}),
2524
ctx
2625
)
27-
expect(ctx.portals!['#target']).toBe(`<div>content</div>`)
26+
expect(html).toBe('<!--portal-->')
27+
expect(ctx.portals!['#target']).toBe(`<div>content</div><!---->`)
2828
})
2929

3030
test('portal rendering (vnode)', async () => {
3131
const ctx: SSRContext = {}
32-
await renderToString(
32+
const html = await renderToString(
3333
h(
3434
Portal,
3535
{
@@ -39,6 +39,28 @@ describe('ssrRenderPortal', () => {
3939
),
4040
ctx
4141
)
42-
expect(ctx.portals!['#target']).toBe('<span>hello</span>')
42+
expect(html).toBe('<!--portal-->')
43+
expect(ctx.portals!['#target']).toBe('<span>hello</span><!---->')
44+
})
45+
46+
test('multiple portals with same target', async () => {
47+
const ctx: SSRContext = {}
48+
const html = await renderToString(
49+
h('div', [
50+
h(
51+
Portal,
52+
{
53+
target: `#target`
54+
},
55+
h('span', 'hello')
56+
),
57+
h(Portal, { target: `#target` }, 'world')
58+
]),
59+
ctx
60+
)
61+
expect(html).toBe('<div><!--portal--><!--portal--></div>')
62+
expect(ctx.portals!['#target']).toBe(
63+
'<span>hello</span><!---->world<!---->'
64+
)
4365
})
4466
})

packages/server-renderer/src/helpers/ssrRenderPortal.ts

+8-3
Original file line numberDiff line numberDiff line change
@@ -2,19 +2,24 @@ import { ComponentInternalInstance, ssrContextKey } from 'vue'
22
import { SSRContext, createBuffer, PushFn } from '../renderToString'
33

44
export function ssrRenderPortal(
5+
parentPush: PushFn,
56
contentRenderFn: (push: PushFn) => void,
67
target: string,
78
parentComponent: ComponentInternalInstance
89
) {
10+
parentPush('<!--portal-->')
911
const { getBuffer, push } = createBuffer()
10-
1112
contentRenderFn(push)
13+
push(`<!---->`) // portal end anchor
1214

1315
const context = parentComponent.appContext.provides[
1416
ssrContextKey as any
1517
] as SSRContext
1618
const portalBuffers =
1719
context.__portalBuffers || (context.__portalBuffers = {})
18-
19-
portalBuffers[target] = getBuffer()
20+
if (portalBuffers[target]) {
21+
portalBuffers[target].push(getBuffer())
22+
} else {
23+
portalBuffers[target] = [getBuffer()]
24+
}
2025
}

packages/server-renderer/src/renderToString.ts

+16-17
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ import { compile } from '@vue/compiler-ssr'
3232
import { ssrRenderAttrs } from './helpers/ssrRenderAttrs'
3333
import { SSRSlots } from './helpers/ssrRenderSlot'
3434
import { CompilerError } from '@vue/compiler-dom'
35+
import { ssrRenderPortal } from './helpers/ssrRenderPortal'
3536

3637
const {
3738
isVNode,
@@ -63,10 +64,7 @@ export type Props = Record<string, unknown>
6364
export type SSRContext = {
6465
[key: string]: any
6566
portals?: Record<string, string>
66-
__portalBuffers?: Record<
67-
string,
68-
ResolvedSSRBuffer | Promise<ResolvedSSRBuffer>
69-
>
67+
__portalBuffers?: Record<string, SSRBuffer>
7068
}
7169

7270
export function createBuffer() {
@@ -259,7 +257,7 @@ function renderVNode(
259257
} else if (shapeFlag & ShapeFlags.COMPONENT) {
260258
push(renderComponentVNode(vnode, parentComponent))
261259
} else if (shapeFlag & ShapeFlags.PORTAL) {
262-
renderPortalVNode(vnode, parentComponent)
260+
renderPortalVNode(push, vnode, parentComponent)
263261
} else if (shapeFlag & ShapeFlags.SUSPENSE) {
264262
renderVNode(
265263
push,
@@ -363,6 +361,7 @@ function applySSRDirectives(
363361
}
364362

365363
function renderPortalVNode(
364+
push: PushFn,
366365
vnode: VNode,
367366
parentComponent: ComponentInternalInstance
368367
) {
@@ -377,20 +376,18 @@ function renderPortalVNode(
377376
)
378377
return []
379378
}
380-
381-
const { getBuffer, push } = createBuffer()
382-
renderVNodeChildren(
379+
ssrRenderPortal(
383380
push,
384-
vnode.children as VNodeArrayChildren,
381+
push => {
382+
renderVNodeChildren(
383+
push,
384+
vnode.children as VNodeArrayChildren,
385+
parentComponent
386+
)
387+
},
388+
target,
385389
parentComponent
386390
)
387-
const context = parentComponent.appContext.provides[
388-
ssrContextKey as any
389-
] as SSRContext
390-
const portalBuffers =
391-
context.__portalBuffers || (context.__portalBuffers = {})
392-
393-
portalBuffers[target] = getBuffer()
394391
}
395392

396393
async function resolvePortals(context: SSRContext) {
@@ -399,7 +396,9 @@ async function resolvePortals(context: SSRContext) {
399396
for (const key in context.__portalBuffers) {
400397
// note: it's OK to await sequentially here because the Promises were
401398
// created eagerly in parallel.
402-
context.portals[key] = unrollBuffer(await context.__portalBuffers[key])
399+
context.portals[key] = unrollBuffer(
400+
await Promise.all(context.__portalBuffers[key])
401+
)
403402
}
404403
}
405404
}

0 commit comments

Comments
 (0)