2021-07-16 14:40:06 +00:00
|
|
|
/**
|
|
|
|
* @jest-environment node
|
|
|
|
*/
|
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
import { createApp, h, Teleport } from 'vue'
|
2020-06-26 15:09:47 +00:00
|
|
|
import { renderToString } from '../src/renderToString'
|
2022-05-17 04:41:40 +00:00
|
|
|
import { renderToSimpleStream } from '../src/renderToStream'
|
2020-06-26 15:09:47 +00:00
|
|
|
import { SSRContext } from '../src/render'
|
2020-03-31 14:52:42 +00:00
|
|
|
import { ssrRenderTeleport } from '../src/helpers/ssrRenderTeleport'
|
2020-02-26 19:59:53 +00:00
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
describe('ssrRenderTeleport', () => {
|
|
|
|
test('teleport rendering (compiled)', async () => {
|
2020-03-28 00:49:01 +00:00
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const html = await renderToString(
|
2020-02-26 19:59:53 +00:00
|
|
|
createApp({
|
|
|
|
data() {
|
|
|
|
return { msg: 'hello' }
|
|
|
|
},
|
|
|
|
ssrRender(_ctx, _push, _parent) {
|
2020-03-31 14:52:42 +00:00
|
|
|
ssrRenderTeleport(
|
2020-03-28 00:49:01 +00:00
|
|
|
_push,
|
2020-02-26 19:59:53 +00:00
|
|
|
_push => {
|
|
|
|
_push(`<div>content</div>`)
|
|
|
|
},
|
|
|
|
'#target',
|
2020-03-28 03:45:50 +00:00
|
|
|
false,
|
2020-02-26 19:59:53 +00:00
|
|
|
_parent
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
ctx
|
|
|
|
)
|
2020-03-31 14:52:42 +00:00
|
|
|
expect(html).toBe('<!--teleport start--><!--teleport end-->')
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(
|
|
|
|
`<div>content</div><!--teleport anchor-->`
|
|
|
|
)
|
2020-02-26 19:59:53 +00:00
|
|
|
})
|
2020-03-10 19:23:14 +00:00
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
test('teleport rendering (compiled + disabled)', async () => {
|
2020-03-28 03:45:50 +00:00
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const html = await renderToString(
|
|
|
|
createApp({
|
|
|
|
data() {
|
|
|
|
return { msg: 'hello' }
|
|
|
|
},
|
|
|
|
ssrRender(_ctx, _push, _parent) {
|
2020-03-31 14:52:42 +00:00
|
|
|
ssrRenderTeleport(
|
2020-03-28 03:45:50 +00:00
|
|
|
_push,
|
|
|
|
_push => {
|
|
|
|
_push(`<div>content</div>`)
|
|
|
|
},
|
|
|
|
'#target',
|
|
|
|
true,
|
|
|
|
_parent
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}),
|
|
|
|
ctx
|
|
|
|
)
|
2020-03-31 14:52:42 +00:00
|
|
|
expect(html).toBe(
|
|
|
|
'<!--teleport start--><div>content</div><!--teleport end-->'
|
|
|
|
)
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(`<!--teleport anchor-->`)
|
2020-03-28 03:45:50 +00:00
|
|
|
})
|
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
test('teleport rendering (vnode)', async () => {
|
2020-03-10 19:23:14 +00:00
|
|
|
const ctx: SSRContext = {}
|
2020-03-28 00:49:01 +00:00
|
|
|
const html = await renderToString(
|
2020-03-10 19:23:14 +00:00
|
|
|
h(
|
2020-03-31 14:52:42 +00:00
|
|
|
Teleport,
|
2020-03-10 19:23:14 +00:00
|
|
|
{
|
2020-03-31 14:52:42 +00:00
|
|
|
to: `#target`
|
2020-03-10 19:23:14 +00:00
|
|
|
},
|
|
|
|
h('span', 'hello')
|
|
|
|
),
|
|
|
|
ctx
|
|
|
|
)
|
2020-03-31 14:52:42 +00:00
|
|
|
expect(html).toBe('<!--teleport start--><!--teleport end-->')
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(
|
|
|
|
'<span>hello</span><!--teleport anchor-->'
|
|
|
|
)
|
2020-03-28 00:49:01 +00:00
|
|
|
})
|
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
test('teleport rendering (vnode + disabled)', async () => {
|
2020-03-28 03:45:50 +00:00
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const html = await renderToString(
|
|
|
|
h(
|
2020-03-31 14:52:42 +00:00
|
|
|
Teleport,
|
2020-03-28 03:45:50 +00:00
|
|
|
{
|
2020-03-31 14:52:42 +00:00
|
|
|
to: `#target`,
|
2020-03-28 03:45:50 +00:00
|
|
|
disabled: true
|
|
|
|
},
|
|
|
|
h('span', 'hello')
|
|
|
|
),
|
|
|
|
ctx
|
|
|
|
)
|
2020-03-31 14:52:42 +00:00
|
|
|
expect(html).toBe(
|
|
|
|
'<!--teleport start--><span>hello</span><!--teleport end-->'
|
|
|
|
)
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(`<!--teleport anchor-->`)
|
2020-03-28 03:45:50 +00:00
|
|
|
})
|
|
|
|
|
2020-03-31 14:52:42 +00:00
|
|
|
test('multiple teleports with same target', async () => {
|
2020-03-28 00:49:01 +00:00
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const html = await renderToString(
|
|
|
|
h('div', [
|
|
|
|
h(
|
2020-03-31 14:52:42 +00:00
|
|
|
Teleport,
|
2020-03-28 00:49:01 +00:00
|
|
|
{
|
2020-03-31 14:52:42 +00:00
|
|
|
to: `#target`
|
2020-03-28 00:49:01 +00:00
|
|
|
},
|
|
|
|
h('span', 'hello')
|
|
|
|
),
|
2020-03-31 14:52:42 +00:00
|
|
|
h(Teleport, { to: `#target` }, 'world')
|
2020-03-28 00:49:01 +00:00
|
|
|
]),
|
|
|
|
ctx
|
|
|
|
)
|
2020-03-28 03:45:50 +00:00
|
|
|
expect(html).toBe(
|
2020-03-31 14:52:42 +00:00
|
|
|
'<div><!--teleport start--><!--teleport end--><!--teleport start--><!--teleport end--></div>'
|
2020-03-28 03:45:50 +00:00
|
|
|
)
|
2020-03-31 14:52:42 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(
|
2022-05-18 10:13:08 +00:00
|
|
|
'<span>hello</span><!--teleport anchor-->world<!--teleport anchor-->'
|
2020-03-28 00:49:01 +00:00
|
|
|
)
|
2020-03-10 19:23:14 +00:00
|
|
|
})
|
2022-05-13 07:57:10 +00:00
|
|
|
|
|
|
|
test('teleport inside async component', async () => {
|
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const asyncComponent = {
|
|
|
|
template: '<teleport to="#target"><div>content</div></teleport>',
|
|
|
|
async setup() {}
|
|
|
|
}
|
|
|
|
const html = await renderToString(
|
|
|
|
h({
|
|
|
|
template: '<async-component />',
|
|
|
|
components: { asyncComponent }
|
|
|
|
}),
|
|
|
|
ctx
|
|
|
|
)
|
|
|
|
expect(html).toBe('<!--teleport start--><!--teleport end-->')
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(
|
|
|
|
`<div>content</div><!--teleport anchor-->`
|
|
|
|
)
|
2022-05-13 07:57:10 +00:00
|
|
|
})
|
2022-05-17 04:41:40 +00:00
|
|
|
|
|
|
|
test('teleport inside async component (stream)', async () => {
|
|
|
|
const ctx: SSRContext = {}
|
|
|
|
const asyncComponent = {
|
|
|
|
template: '<teleport to="#target"><div>content</div></teleport>',
|
|
|
|
async setup() {}
|
|
|
|
}
|
|
|
|
let html = ''
|
|
|
|
let resolve: any
|
|
|
|
const p = new Promise(r => (resolve = r))
|
|
|
|
renderToSimpleStream(
|
|
|
|
h({
|
|
|
|
template: '<async-component />',
|
|
|
|
components: { asyncComponent }
|
|
|
|
}),
|
|
|
|
ctx,
|
|
|
|
{
|
|
|
|
push(chunk) {
|
|
|
|
if (chunk === null) {
|
|
|
|
resolve()
|
|
|
|
} else {
|
|
|
|
html += chunk
|
|
|
|
}
|
|
|
|
},
|
|
|
|
destroy(err) {
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
await p
|
|
|
|
expect(html).toBe('<!--teleport start--><!--teleport end-->')
|
2022-05-18 10:13:08 +00:00
|
|
|
expect(ctx.teleports!['#target']).toBe(
|
|
|
|
`<div>content</div><!--teleport anchor-->`
|
|
|
|
)
|
2022-05-17 04:41:40 +00:00
|
|
|
})
|
2020-02-26 19:59:53 +00:00
|
|
|
})
|