wip: more consistent compiler-sfc usage + inline mode for ssr

This commit is contained in:
Evan You 2020-11-20 14:22:51 -05:00
parent 58227e88e9
commit 8ac2241b22
18 changed files with 213 additions and 93 deletions

View File

@ -9,7 +9,7 @@ const _hoisted_1 = /*#__PURE__*/_createVNode(\\"div\\", null, \\"hello\\", -1 /*
const _hoisted_2 = /*#__PURE__*/_createVNode(\\"div\\", null, \\"world\\", -1 /* HOISTED */) const _hoisted_2 = /*#__PURE__*/_createVNode(\\"div\\", null, \\"world\\", -1 /* HOISTED */)
_popScopeId() _popScopeId()
export const render = /*#__PURE__*/_withId(function render(_ctx, _cache) { export const render = /*#__PURE__*/_withId((_ctx, _cache) => {
return (_openBlock(), _createBlock(\\"div\\", null, [ return (_openBlock(), _createBlock(\\"div\\", null, [
_hoisted_1, _hoisted_1,
_createTextVNode(_toDisplayString(_ctx.foo), 1 /* TEXT */), _createTextVNode(_toDisplayString(_ctx.foo), 1 /* TEXT */),
@ -22,7 +22,7 @@ exports[`scopeId compiler support should wrap default slot 1`] = `
"import { createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\" "import { createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\"
const _withId = /*#__PURE__*/_withScopeId(\\"test\\") const _withId = /*#__PURE__*/_withScopeId(\\"test\\")
export const render = /*#__PURE__*/_withId(function render(_ctx, _cache) { export const render = /*#__PURE__*/_withId((_ctx, _cache) => {
const _component_Child = _resolveComponent(\\"Child\\") const _component_Child = _resolveComponent(\\"Child\\")
return (_openBlock(), _createBlock(_component_Child, null, { return (_openBlock(), _createBlock(_component_Child, null, {
@ -38,7 +38,7 @@ exports[`scopeId compiler support should wrap dynamic slots 1`] = `
"import { createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, renderList as _renderList, createSlots as _createSlots, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\" "import { createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, renderList as _renderList, createSlots as _createSlots, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\"
const _withId = /*#__PURE__*/_withScopeId(\\"test\\") const _withId = /*#__PURE__*/_withScopeId(\\"test\\")
export const render = /*#__PURE__*/_withId(function render(_ctx, _cache) { export const render = /*#__PURE__*/_withId((_ctx, _cache) => {
const _component_Child = _resolveComponent(\\"Child\\") const _component_Child = _resolveComponent(\\"Child\\")
return (_openBlock(), _createBlock(_component_Child, null, _createSlots({ _: 2 }, [ return (_openBlock(), _createBlock(_component_Child, null, _createSlots({ _: 2 }, [
@ -66,7 +66,7 @@ exports[`scopeId compiler support should wrap named slots 1`] = `
"import { toDisplayString as _toDisplayString, createTextVNode as _createTextVNode, createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\" "import { toDisplayString as _toDisplayString, createTextVNode as _createTextVNode, createVNode as _createVNode, resolveComponent as _resolveComponent, withCtx as _withCtx, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\"
const _withId = /*#__PURE__*/_withScopeId(\\"test\\") const _withId = /*#__PURE__*/_withScopeId(\\"test\\")
export const render = /*#__PURE__*/_withId(function render(_ctx, _cache) { export const render = /*#__PURE__*/_withId((_ctx, _cache) => {
const _component_Child = _resolveComponent(\\"Child\\") const _component_Child = _resolveComponent(\\"Child\\")
return (_openBlock(), _createBlock(_component_Child, null, { return (_openBlock(), _createBlock(_component_Child, null, {
@ -85,7 +85,7 @@ exports[`scopeId compiler support should wrap render function 1`] = `
"import { createVNode as _createVNode, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\" "import { createVNode as _createVNode, openBlock as _openBlock, createBlock as _createBlock, withScopeId as _withScopeId } from \\"vue\\"
const _withId = /*#__PURE__*/_withScopeId(\\"test\\") const _withId = /*#__PURE__*/_withScopeId(\\"test\\")
export const render = /*#__PURE__*/_withId(function render(_ctx, _cache) { export const render = /*#__PURE__*/_withId((_ctx, _cache) => {
return (_openBlock(), _createBlock(\\"div\\")) return (_openBlock(), _createBlock(\\"div\\"))
})" })"
`; `;

View File

@ -22,7 +22,7 @@ describe('scopeId compiler support', () => {
expect(ast.helpers).toContain(WITH_SCOPE_ID) expect(ast.helpers).toContain(WITH_SCOPE_ID)
expect(code).toMatch(`const _withId = /*#__PURE__*/_withScopeId("test")`) expect(code).toMatch(`const _withId = /*#__PURE__*/_withScopeId("test")`)
expect(code).toMatch( expect(code).toMatch(
`export const render = /*#__PURE__*/_withId(function render(` `export const render = /*#__PURE__*/_withId((_ctx, _cache) => {`
) )
expect(code).toMatchSnapshot() expect(code).toMatchSnapshot()
}) })

View File

@ -203,7 +203,7 @@ export function generate(
const hasHelpers = ast.helpers.length > 0 const hasHelpers = ast.helpers.length > 0
const useWithBlock = !prefixIdentifiers && mode !== 'module' const useWithBlock = !prefixIdentifiers && mode !== 'module'
const genScopeId = !__BROWSER__ && scopeId != null && mode === 'module' const genScopeId = !__BROWSER__ && scopeId != null && mode === 'module'
const isSetupInlined = !!options.inline const isSetupInlined = !__BROWSER__ && !!options.inline
// preambles // preambles
// in setup() inline mode, the preamble is generated in a sub context // in setup() inline mode, the preamble is generated in a sub context
@ -217,6 +217,8 @@ export function generate(
genFunctionPreamble(ast, preambleContext) genFunctionPreamble(ast, preambleContext)
} }
// enter render function
const functionName = ssr ? `ssrRender` : `render`
const args = ssr ? ['_ctx', '_push', '_parent', '_attrs'] : ['_ctx', '_cache'] const args = ssr ? ['_ctx', '_push', '_parent', '_attrs'] : ['_ctx', '_cache']
if (!__BROWSER__ && options.bindingMetadata && !options.inline) { if (!__BROWSER__ && options.bindingMetadata && !options.inline) {
// binding optimization args // binding optimization args
@ -226,24 +228,18 @@ export function generate(
!__BROWSER__ && options.isTS !__BROWSER__ && options.isTS
? args.map(arg => `${arg}: any`).join(',') ? args.map(arg => `${arg}: any`).join(',')
: args.join(', ') : args.join(', ')
// enter render function
if (!ssr) { if (genScopeId) {
if (isSetupInlined) { if (isSetupInlined) {
if (genScopeId) { push(`${PURE_ANNOTATION}_withId(`)
push(`${PURE_ANNOTATION}_withId(`)
}
push(`(${signature}) => {`)
} else { } else {
if (genScopeId) { push(`const ${functionName} = ${PURE_ANNOTATION}_withId(`)
push(`const render = ${PURE_ANNOTATION}_withId(`)
}
push(`function render(${signature}) {`)
} }
}
if (isSetupInlined || genScopeId) {
push(`(${signature}) => {`)
} else { } else {
if (genScopeId) { push(`function ${functionName}(${signature}) {`)
push(`const ssrRender = ${PURE_ANNOTATION}_withId(`)
}
push(`function ssrRender(${signature}) {`)
} }
indent() indent()

View File

@ -201,6 +201,40 @@ return (_ctx, _cache) => {
}" }"
`; `;
exports[`SFC compile <script setup> inlineTemplate mode ssr codegen 1`] = `
"import { useCssVars as _useCssVars, unref as _unref } from 'vue'
import { ssrRenderAttrs as _ssrRenderAttrs, ssrInterpolate as _ssrInterpolate } from \\"@vue/server-renderer\\"
import { ref } from 'vue'
export default {
expose: [],
__ssrInlineRender: true,
setup(__props) {
_useCssVars(_ctx => ({
\\"xxxxxxxx-count\\": (count.value)
}))
const count = ref(0)
return (_ctx, _push, _parent, _attrs) => {
const _cssVars = { style: {
\\"xxxxxxxx-count\\": (count.value)
}}
_push(\`<!--[--><div\${
_ssrRenderAttrs(_cssVars)
}>\${
_ssrInterpolate(count.value)
}</div><div\${
_ssrRenderAttrs(_cssVars)
}>static</div><!--]-->\`)
}
}
}"
`;
exports[`SFC compile <script setup> inlineTemplate mode template assignment expression codegen 1`] = ` exports[`SFC compile <script setup> inlineTemplate mode template assignment expression codegen 1`] = `
"import { createVNode as _createVNode, isRef as _isRef, Fragment as _Fragment, openBlock as _openBlock, createBlock as _createBlock } from \\"vue\\" "import { createVNode as _createVNode, isRef as _isRef, Fragment as _Fragment, openBlock as _openBlock, createBlock as _createBlock } from \\"vue\\"

View File

@ -1,4 +1,4 @@
import { BindingTypes } from '@vue/compiler-dom/src' import { BindingTypes } from '@vue/compiler-dom'
import { compileSFCScript as compile, assertCode } from './utils' import { compileSFCScript as compile, assertCode } from './utils'
describe('SFC compile <script setup>', () => { describe('SFC compile <script setup>', () => {
@ -297,6 +297,34 @@ const bar = 1
expect(content).toMatch(`{ lett: lett } = val`) expect(content).toMatch(`{ lett: lett } = val`)
assertCode(content) assertCode(content)
}) })
test('ssr codegen', () => {
const { content } = compile(
`
<script setup>
import { ref } from 'vue'
const count = ref(0)
</script>
<template>
<div>{{ count }}</div>
<div>static</div>
</template>
<style>
div { color: v-bind(count) }
</style>
`,
{
inlineTemplate: true,
templateOptions: {
ssr: true
}
}
)
expect(content).toMatch(`\n __ssrInlineRender: true,\n`)
expect(content).toMatch(`return (_ctx, _push`)
expect(content).toMatch(`ssrInterpolate`)
assertCode(content)
})
}) })
describe('with TypeScript', () => { describe('with TypeScript', () => {

View File

@ -16,7 +16,7 @@ export function compileScoped(
const res = compileStyle({ const res = compileStyle({
source, source,
filename: 'test.css', filename: 'test.css',
id: 'test', id: 'data-v-test',
scoped: true, scoped: true,
...options ...options
}) })
@ -32,61 +32,61 @@ export function compileScoped(
describe('SFC scoped CSS', () => { describe('SFC scoped CSS', () => {
test('simple selectors', () => { test('simple selectors', () => {
expect(compileScoped(`h1 { color: red; }`)).toMatch( expect(compileScoped(`h1 { color: red; }`)).toMatch(
`h1[test] { color: red;` `h1[data-v-test] { color: red;`
) )
expect(compileScoped(`.foo { color: red; }`)).toMatch( expect(compileScoped(`.foo { color: red; }`)).toMatch(
`.foo[test] { color: red;` `.foo[data-v-test] { color: red;`
) )
}) })
test('descendent selector', () => { test('descendent selector', () => {
expect(compileScoped(`h1 .foo { color: red; }`)).toMatch( expect(compileScoped(`h1 .foo { color: red; }`)).toMatch(
`h1 .foo[test] { color: red;` `h1 .foo[data-v-test] { color: red;`
) )
}) })
test('multiple selectors', () => { test('multiple selectors', () => {
expect(compileScoped(`h1 .foo, .bar, .baz { color: red; }`)).toMatch( expect(compileScoped(`h1 .foo, .bar, .baz { color: red; }`)).toMatch(
`h1 .foo[test], .bar[test], .baz[test] { color: red;` `h1 .foo[data-v-test], .bar[data-v-test], .baz[data-v-test] { color: red;`
) )
}) })
test('pseudo class', () => { test('pseudo class', () => {
expect(compileScoped(`.foo:after { color: red; }`)).toMatch( expect(compileScoped(`.foo:after { color: red; }`)).toMatch(
`.foo[test]:after { color: red;` `.foo[data-v-test]:after { color: red;`
) )
}) })
test('pseudo element', () => { test('pseudo element', () => {
expect(compileScoped(`::selection { display: none; }`)).toMatch( expect(compileScoped(`::selection { display: none; }`)).toMatch(
'[test]::selection {' '[data-v-test]::selection {'
) )
}) })
test('spaces before pseudo element', () => { test('spaces before pseudo element', () => {
const code = compileScoped(`.abc, ::selection { color: red; }`) const code = compileScoped(`.abc, ::selection { color: red; }`)
expect(code).toMatch('.abc[test],') expect(code).toMatch('.abc[data-v-test],')
expect(code).toMatch('[test]::selection {') expect(code).toMatch('[data-v-test]::selection {')
}) })
test('::v-deep', () => { test('::v-deep', () => {
expect(compileScoped(`:deep(.foo) { color: red; }`)).toMatchInlineSnapshot(` expect(compileScoped(`:deep(.foo) { color: red; }`)).toMatchInlineSnapshot(`
"[test] .foo { color: red; "[data-v-test] .foo { color: red;
}" }"
`) `)
expect(compileScoped(`::v-deep(.foo) { color: red; }`)) expect(compileScoped(`::v-deep(.foo) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
"[test] .foo { color: red; "[data-v-test] .foo { color: red;
}" }"
`) `)
expect(compileScoped(`::v-deep(.foo .bar) { color: red; }`)) expect(compileScoped(`::v-deep(.foo .bar) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
"[test] .foo .bar { color: red; "[data-v-test] .foo .bar { color: red;
}" }"
`) `)
expect(compileScoped(`.baz .qux ::v-deep(.foo .bar) { color: red; }`)) expect(compileScoped(`.baz .qux ::v-deep(.foo .bar) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".baz .qux[test] .foo .bar { color: red; ".baz .qux[data-v-test] .foo .bar { color: red;
}" }"
`) `)
}) })
@ -94,22 +94,22 @@ describe('SFC scoped CSS', () => {
test('::v-slotted', () => { test('::v-slotted', () => {
expect(compileScoped(`:slotted(.foo) { color: red; }`)) expect(compileScoped(`:slotted(.foo) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".foo[test-s] { color: red; ".foo[data-v-test-s] { color: red;
}" }"
`) `)
expect(compileScoped(`::v-slotted(.foo) { color: red; }`)) expect(compileScoped(`::v-slotted(.foo) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".foo[test-s] { color: red; ".foo[data-v-test-s] { color: red;
}" }"
`) `)
expect(compileScoped(`::v-slotted(.foo .bar) { color: red; }`)) expect(compileScoped(`::v-slotted(.foo .bar) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".foo .bar[test-s] { color: red; ".foo .bar[data-v-test-s] { color: red;
}" }"
`) `)
expect(compileScoped(`.baz .qux ::v-slotted(.foo .bar) { color: red; }`)) expect(compileScoped(`.baz .qux ::v-slotted(.foo .bar) { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".baz .qux .foo .bar[test-s] { color: red; ".baz .qux .foo .bar[data-v-test-s] { color: red;
}" }"
`) `)
}) })
@ -142,7 +142,7 @@ describe('SFC scoped CSS', () => {
expect(compileScoped(`@media print { .foo { color: red }}`)) expect(compileScoped(`@media print { .foo { color: red }}`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
"@media print { "@media print {
.foo[test] { color: red .foo[data-v-test] { color: red
}}" }}"
`) `)
}) })
@ -151,7 +151,7 @@ describe('SFC scoped CSS', () => {
expect(compileScoped(`@supports(display: grid) { .foo { display: grid }}`)) expect(compileScoped(`@supports(display: grid) { .foo { display: grid }}`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
"@supports(display: grid) { "@supports(display: grid) {
.foo[test] { display: grid .foo[data-v-test] { display: grid
}}" }}"
`) `)
}) })
@ -222,7 +222,7 @@ describe('SFC scoped CSS', () => {
// vue-loader/#1370 // vue-loader/#1370
test('spaces after selector', () => { test('spaces after selector', () => {
expect(compileScoped(`.foo , .bar { color: red; }`)).toMatchInlineSnapshot(` expect(compileScoped(`.foo , .bar { color: red; }`)).toMatchInlineSnapshot(`
".foo[test], .bar[test] { color: red; ".foo[data-v-test], .bar[data-v-test] { color: red;
}" }"
`) `)
}) })
@ -231,12 +231,12 @@ describe('SFC scoped CSS', () => {
test('::v-deep as combinator', () => { test('::v-deep as combinator', () => {
expect(compileScoped(`::v-deep .foo { color: red; }`)) expect(compileScoped(`::v-deep .foo { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
"[test] .foo { color: red; "[data-v-test] .foo { color: red;
}" }"
`) `)
expect(compileScoped(`.bar ::v-deep .foo { color: red; }`)) expect(compileScoped(`.bar ::v-deep .foo { color: red; }`))
.toMatchInlineSnapshot(` .toMatchInlineSnapshot(`
".bar[test] .foo { color: red; ".bar[data-v-test] .foo { color: red;
}" }"
`) `)
expect( expect(
@ -247,7 +247,7 @@ describe('SFC scoped CSS', () => {
test('>>> (deprecated syntax)', () => { test('>>> (deprecated syntax)', () => {
const code = compileScoped(`>>> .foo { color: red; }`) const code = compileScoped(`>>> .foo { color: red; }`)
expect(code).toMatchInlineSnapshot(` expect(code).toMatchInlineSnapshot(`
"[test] .foo { color: red; "[data-v-test] .foo { color: red;
}" }"
`) `)
expect( expect(
@ -258,7 +258,7 @@ describe('SFC scoped CSS', () => {
test('/deep/ (deprecated syntax)', () => { test('/deep/ (deprecated syntax)', () => {
const code = compileScoped(`/deep/ .foo { color: red; }`) const code = compileScoped(`/deep/ .foo { color: red; }`)
expect(code).toMatchInlineSnapshot(` expect(code).toMatchInlineSnapshot(`
"[test] .foo { color: red; "[data-v-test] .foo { color: red;
}" }"
`) `)
expect( expect(

View File

@ -1,10 +1,20 @@
import { compileTemplate } from '../src/compileTemplate' import {
compileTemplate,
SFCTemplateCompileOptions
} from '../src/compileTemplate'
import { parse, SFCTemplateBlock } from '../src/parse' import { parse, SFCTemplateBlock } from '../src/parse'
function compile(opts: Omit<SFCTemplateCompileOptions, 'id'>) {
return compileTemplate({
...opts,
id: ''
})
}
test('should work', () => { test('should work', () => {
const source = `<div><p>{{ render }}</p></div>` const source = `<div><p>{{ render }}</p></div>`
const result = compileTemplate({ filename: 'example.vue', source }) const result = compile({ filename: 'example.vue', source })
expect(result.errors.length).toBe(0) expect(result.errors.length).toBe(0)
expect(result.source).toBe(source) expect(result.source).toBe(source)
@ -25,7 +35,7 @@ body
{ filename: 'example.vue', sourceMap: true } { filename: 'example.vue', sourceMap: true }
).descriptor.template as SFCTemplateBlock ).descriptor.template as SFCTemplateBlock
const result = compileTemplate({ const result = compile({
filename: 'example.vue', filename: 'example.vue',
source: template.content, source: template.content,
preprocessLang: template.lang preprocessLang: template.lang
@ -40,7 +50,7 @@ test('warn missing preprocessor', () => {
sourceMap: true sourceMap: true
}).descriptor.template as SFCTemplateBlock }).descriptor.template as SFCTemplateBlock
const result = compileTemplate({ const result = compile({
filename: 'example.vue', filename: 'example.vue',
source: template.content, source: template.content,
preprocessLang: template.lang preprocessLang: template.lang
@ -52,7 +62,7 @@ test('warn missing preprocessor', () => {
test('transform asset url options', () => { test('transform asset url options', () => {
const input = { source: `<foo bar="~baz"/>`, filename: 'example.vue' } const input = { source: `<foo bar="~baz"/>`, filename: 'example.vue' }
// Object option // Object option
const { code: code1 } = compileTemplate({ const { code: code1 } = compile({
...input, ...input,
transformAssetUrls: { transformAssetUrls: {
tags: { foo: ['bar'] } tags: { foo: ['bar'] }
@ -61,7 +71,7 @@ test('transform asset url options', () => {
expect(code1).toMatch(`import _imports_0 from 'baz'\n`) expect(code1).toMatch(`import _imports_0 from 'baz'\n`)
// legacy object option (direct tags config) // legacy object option (direct tags config)
const { code: code2 } = compileTemplate({ const { code: code2 } = compile({
...input, ...input,
transformAssetUrls: { transformAssetUrls: {
foo: ['bar'] foo: ['bar']
@ -70,7 +80,7 @@ test('transform asset url options', () => {
expect(code2).toMatch(`import _imports_0 from 'baz'\n`) expect(code2).toMatch(`import _imports_0 from 'baz'\n`)
// false option // false option
const { code: code3 } = compileTemplate({ const { code: code3 } = compile({
...input, ...input,
transformAssetUrls: false transformAssetUrls: false
}) })
@ -87,7 +97,7 @@ test('source map', () => {
{ filename: 'example.vue', sourceMap: true } { filename: 'example.vue', sourceMap: true }
).descriptor.template as SFCTemplateBlock ).descriptor.template as SFCTemplateBlock
const result = compileTemplate({ const result = compile({
filename: 'example.vue', filename: 'example.vue',
source: template.content source: template.content
}) })
@ -96,7 +106,7 @@ test('source map', () => {
}) })
test('template errors', () => { test('template errors', () => {
const result = compileTemplate({ const result = compile({
filename: 'example.vue', filename: 'example.vue',
source: `<div :foo source: `<div :foo
:bar="a[" v-model="baz"/>` :bar="a[" v-model="baz"/>`
@ -114,7 +124,7 @@ test('preprocessor errors', () => {
{ filename: 'example.vue', sourceMap: true } { filename: 'example.vue', sourceMap: true }
).descriptor.template as SFCTemplateBlock ).descriptor.template as SFCTemplateBlock
const result = compileTemplate({ const result = compile({
filename: 'example.vue', filename: 'example.vue',
source: template.content, source: template.content,
preprocessLang: template.lang preprocessLang: template.lang

View File

@ -167,6 +167,7 @@ export function compileScript(
let optionsArg: ObjectExpression | undefined let optionsArg: ObjectExpression | undefined
let optionsType: TSTypeLiteral | undefined let optionsType: TSTypeLiteral | undefined
let hasAwait = false let hasAwait = false
let hasInlinedSsrRenderFn = false
// context types to generate // context types to generate
let propsType = `{}` let propsType = `{}`
let emitType = `(e: string, ...args: any[]) => void` let emitType = `(e: string, ...args: any[]) => void`
@ -820,15 +821,24 @@ export function compileScript(
// 10. generate return statement // 10. generate return statement
let returned let returned
if (options.inlineTemplate) { if (options.inlineTemplate) {
if (sfc.template) { if (sfc.template && !sfc.template.src) {
if (options.templateOptions && options.templateOptions.ssr) {
hasInlinedSsrRenderFn = true
}
// inline render function mode - we are going to compile the template and // inline render function mode - we are going to compile the template and
// inline it right here // inline it right here
const { code, ast, preamble, tips, errors } = compileTemplate({ const { code, ast, preamble, tips, errors } = compileTemplate({
...options.templateOptions,
filename, filename,
source: sfc.template.content, source: sfc.template.content,
inMap: sfc.template.map, inMap: sfc.template.map,
...options.templateOptions,
id: scopeId,
scoped: sfc.styles.some(s => s.scoped),
isProd: options.isProd,
ssrCssVars: sfc.cssVars,
compilerOptions: { compilerOptions: {
...(options.templateOptions &&
options.templateOptions.compilerOptions),
inline: true, inline: true,
isTS, isTS,
bindingMetadata bindingMetadata
@ -883,6 +893,9 @@ export function compileScript(
// 11. finalize default export // 11. finalize default export
// expose: [] makes <script setup> components "closed" by default. // expose: [] makes <script setup> components "closed" by default.
let runtimeOptions = `\n expose: [],` let runtimeOptions = `\n expose: [],`
if (hasInlinedSsrRenderFn) {
runtimeOptions += `\n __ssrInlineRender: true,`
}
if (optionsArg) { if (optionsArg) {
runtimeOptions += `\n ${scriptSetup.content runtimeOptions += `\n ${scriptSetup.content
.slice(optionsArg.start! + 1, optionsArg.end! - 1) .slice(optionsArg.start! + 1, optionsArg.end! - 1)

View File

@ -20,15 +20,19 @@ export interface SFCStyleCompileOptions {
source: string source: string
filename: string filename: string
id: string id: string
map?: RawSourceMap
scoped?: boolean scoped?: boolean
trim?: boolean trim?: boolean
isProd?: boolean isProd?: boolean
inMap?: RawSourceMap
preprocessLang?: PreprocessLang preprocessLang?: PreprocessLang
preprocessOptions?: any preprocessOptions?: any
preprocessCustomRequire?: (id: string) => any preprocessCustomRequire?: (id: string) => any
postcssOptions?: any postcssOptions?: any
postcssPlugins?: any[] postcssPlugins?: any[]
/**
* @deprecated
*/
map?: RawSourceMap
} }
export interface SFCAsyncStyleCompileOptions extends SFCStyleCompileOptions { export interface SFCAsyncStyleCompileOptions extends SFCStyleCompileOptions {
@ -92,16 +96,21 @@ export function doCompileStyle(
} = options } = options
const preprocessor = preprocessLang && processors[preprocessLang] const preprocessor = preprocessLang && processors[preprocessLang]
const preProcessedSource = preprocessor && preprocess(options, preprocessor) const preProcessedSource = preprocessor && preprocess(options, preprocessor)
const map = preProcessedSource ? preProcessedSource.map : options.map const map = preProcessedSource
? preProcessedSource.map
: options.inMap || options.map
const source = preProcessedSource ? preProcessedSource.code : options.source const source = preProcessedSource ? preProcessedSource.code : options.source
const shortId = id.replace(/^data-v-/, '')
const longId = `data-v-${shortId}`
const plugins = (postcssPlugins || []).slice() const plugins = (postcssPlugins || []).slice()
plugins.unshift(cssVarsPlugin({ id, isProd })) plugins.unshift(cssVarsPlugin({ id: shortId, isProd }))
if (trim) { if (trim) {
plugins.push(trimPlugin()) plugins.push(trimPlugin())
} }
if (scoped) { if (scoped) {
plugins.push(scopedPlugin(id)) plugins.push(scopedPlugin(longId))
} }
let cssModules: Record<string, string> | undefined let cssModules: Record<string, string> | undefined
if (modules) { if (modules) {

View File

@ -23,6 +23,7 @@ import * as CompilerDOM from '@vue/compiler-dom'
import * as CompilerSSR from '@vue/compiler-ssr' import * as CompilerSSR from '@vue/compiler-ssr'
import consolidate from 'consolidate' import consolidate from 'consolidate'
import { warnOnce } from './warn' import { warnOnce } from './warn'
import { genCssVarsFromList } from './cssVars'
export interface TemplateCompiler { export interface TemplateCompiler {
compile(template: string, options: CompilerOptions): CodegenResult compile(template: string, options: CompilerOptions): CodegenResult
@ -42,7 +43,11 @@ export interface SFCTemplateCompileResults {
export interface SFCTemplateCompileOptions { export interface SFCTemplateCompileOptions {
source: string source: string
filename: string filename: string
id: string
scoped?: boolean
isProd?: boolean
ssr?: boolean ssr?: boolean
ssrCssVars?: string[]
inMap?: RawSourceMap inMap?: RawSourceMap
compiler?: TemplateCompiler compiler?: TemplateCompiler
compilerOptions?: CompilerOptions compilerOptions?: CompilerOptions
@ -151,9 +156,13 @@ export function compileTemplate(
function doCompileTemplate({ function doCompileTemplate({
filename, filename,
id,
scoped,
inMap, inMap,
source, source,
ssr = false, ssr = false,
ssrCssVars,
isProd = false,
compiler = ssr ? (CompilerSSR as TemplateCompiler) : CompilerDOM, compiler = ssr ? (CompilerSSR as TemplateCompiler) : CompilerDOM,
compilerOptions = {}, compilerOptions = {},
transformAssetUrls transformAssetUrls
@ -171,19 +180,30 @@ function doCompileTemplate({
nodeTransforms = [transformAssetUrl, transformSrcset] nodeTransforms = [transformAssetUrl, transformSrcset]
} }
if (ssr && compilerOptions.ssrCssVars == null) { if (ssr && !ssrCssVars) {
warnOnce( warnOnce(
`compileTemplate is called with \`ssr: true\` but no ` + `compileTemplate is called with \`ssr: true\` but no ` +
`corresponding \`ssrCssVars\` option. The value can be generated by ` + `corresponding \`cssVars\` option.\`.`
`calling \`generateCssVars(sfcDescriptor, scopeId, isProduction)\`.`
) )
} }
if (!id) {
warnOnce(`compileTemplate now requires the \`id\` option.\`.`)
id = ''
}
const shortId = id.replace(/^data-v-/, '')
const longId = `data-v-${shortId}`
let { code, ast, preamble, map } = compiler.compile(source, { let { code, ast, preamble, map } = compiler.compile(source, {
mode: 'module', mode: 'module',
prefixIdentifiers: true, prefixIdentifiers: true,
hoistStatic: true, hoistStatic: true,
cacheHandlers: true, cacheHandlers: true,
ssrCssVars:
ssr && ssrCssVars && ssrCssVars.length
? genCssVarsFromList(ssrCssVars, shortId, isProd)
: '',
scopeId: scoped ? longId : undefined,
...compilerOptions, ...compilerOptions,
nodeTransforms: nodeTransforms.concat(compilerOptions.nodeTransforms || []), nodeTransforms: nodeTransforms.concat(compilerOptions.nodeTransforms || []),
filename, filename,

View File

@ -16,26 +16,13 @@ import hash from 'hash-sum'
export const CSS_VARS_HELPER = `useCssVars` export const CSS_VARS_HELPER = `useCssVars`
export const cssVarRE = /\bv-bind\(\s*(?:'([^']+)'|"([^"]+)"|([^'"][^)]*))\s*\)/g export const cssVarRE = /\bv-bind\(\s*(?:'([^']+)'|"([^"]+)"|([^'"][^)]*))\s*\)/g
/** export function genCssVarsFromList(
* Given an SFC descriptor, generate the CSS variables object string that can be
* passed to `compileTemplate` as `compilerOptions.ssrCssVars`.
* @public
*/
export function generateCssVars(
sfc: SFCDescriptor,
id: string,
isProd: boolean
): string {
return sfc.cssVars.length ? genCssVarsFromList(sfc.cssVars, id, isProd) : ''
}
function genCssVarsFromList(
vars: string[], vars: string[],
id: string, id: string,
isProd: boolean isProd: boolean
): string { ): string {
return `{\n ${vars return `{\n ${vars
.map(v => `"${genVarName(id, v, isProd)}": (${v})`) .map(key => `"${genVarName(id, key, isProd)}": (${key})`)
.join(',\n ')}\n}` .join(',\n ')}\n}`
} }
@ -68,12 +55,11 @@ export const cssVarsPlugin = postcss.plugin<CssVarsPluginOptions>(
'vue-scoped', 'vue-scoped',
opts => (root: Root) => { opts => (root: Root) => {
const { id, isProd } = opts! const { id, isProd } = opts!
const shortId = id.replace(/^data-v-/, '')
root.walkDecls(decl => { root.walkDecls(decl => {
// rewrite CSS variables // rewrite CSS variables
if (cssVarRE.test(decl.value)) { if (cssVarRE.test(decl.value)) {
decl.value = decl.value.replace(cssVarRE, (_, $1, $2, $3) => { decl.value = decl.value.replace(cssVarRE, (_, $1, $2, $3) => {
return `var(--${genVarName(shortId, $1 || $2 || $3, isProd)})` return `var(--${genVarName(id, $1 || $2 || $3, isProd)})`
}) })
} }
}) })

View File

@ -5,7 +5,6 @@ export { compileStyle, compileStyleAsync } from './compileStyle'
export { compileScript } from './compileScript' export { compileScript } from './compileScript'
export { rewriteDefault } from './rewriteDefault' export { rewriteDefault } from './rewriteDefault'
export { generateCodeFrame } from '@vue/compiler-core' export { generateCodeFrame } from '@vue/compiler-core'
export { generateCssVars } from './cssVars'
// Types // Types
export { export {

View File

@ -14,7 +14,7 @@ describe('ssr: scopeId', () => {
import { ssrRenderAttrs as _ssrRenderAttrs } from \\"@vue/server-renderer\\" import { ssrRenderAttrs as _ssrRenderAttrs } from \\"@vue/server-renderer\\"
const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\") const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\")
export const ssrRender = /*#__PURE__*/_withId(function ssrRender(_ctx, _push, _parent, _attrs) { export const ssrRender = /*#__PURE__*/_withId((_ctx, _push, _parent, _attrs) => {
_push(\`<div\${_ssrRenderAttrs(_attrs)} data-v-xxxxxxx><span data-v-xxxxxxx>hello</span></div>\`) _push(\`<div\${_ssrRenderAttrs(_attrs)} data-v-xxxxxxx><span data-v-xxxxxxx>hello</span></div>\`)
})" })"
`) `)
@ -32,7 +32,7 @@ describe('ssr: scopeId', () => {
import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\" import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\"
const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\") const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\")
export const ssrRender = /*#__PURE__*/_withId(function ssrRender(_ctx, _push, _parent, _attrs) { export const ssrRender = /*#__PURE__*/_withId((_ctx, _push, _parent, _attrs) => {
const _component_foo = _resolveComponent(\\"foo\\") const _component_foo = _resolveComponent(\\"foo\\")
_push(_ssrRenderComponent(_component_foo, _attrs, { _push(_ssrRenderComponent(_component_foo, _attrs, {
@ -62,7 +62,7 @@ describe('ssr: scopeId', () => {
import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\" import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\"
const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\") const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\")
export const ssrRender = /*#__PURE__*/_withId(function ssrRender(_ctx, _push, _parent, _attrs) { export const ssrRender = /*#__PURE__*/_withId((_ctx, _push, _parent, _attrs) => {
const _component_foo = _resolveComponent(\\"foo\\") const _component_foo = _resolveComponent(\\"foo\\")
_push(_ssrRenderComponent(_component_foo, _attrs, { _push(_ssrRenderComponent(_component_foo, _attrs, {
@ -92,7 +92,7 @@ describe('ssr: scopeId', () => {
import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\" import { ssrRenderComponent as _ssrRenderComponent } from \\"@vue/server-renderer\\"
const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\") const _withId = /*#__PURE__*/_withScopeId(\\"data-v-xxxxxxx\\")
export const ssrRender = /*#__PURE__*/_withId(function ssrRender(_ctx, _push, _parent, _attrs) { export const ssrRender = /*#__PURE__*/_withId((_ctx, _push, _parent, _attrs) => {
const _component_foo = _resolveComponent(\\"foo\\") const _component_foo = _resolveComponent(\\"foo\\")
const _component_bar = _resolveComponent(\\"bar\\") const _component_bar = _resolveComponent(\\"bar\\")

View File

@ -223,6 +223,11 @@ export interface ComponentInternalInstance {
* @internal * @internal
*/ */
render: InternalRenderFunction | null render: InternalRenderFunction | null
/**
* SSR render function
* @internal
*/
ssrRender?: Function | null
/** /**
* Object containing values this component provides for its descendents * Object containing values this component provides for its descendents
* @internal * @internal
@ -610,7 +615,13 @@ export function handleSetupResult(
) { ) {
if (isFunction(setupResult)) { if (isFunction(setupResult)) {
// setup returned an inline render function // setup returned an inline render function
instance.render = setupResult as InternalRenderFunction if (!__BROWSER__ && (instance.type as ComponentOptions).__ssrInlineRender) {
// when the function's name is `ssrRender` (compiled by SFC inline mode),
// set it as ssrRender instead.
instance.ssrRender = setupResult
} else {
instance.render = setupResult as InternalRenderFunction
}
} else if (isObject(setupResult)) { } else if (isObject(setupResult)) {
if (__DEV__ && isVNode(setupResult)) { if (__DEV__ && isVNode(setupResult)) {
warn( warn(

View File

@ -121,7 +121,6 @@ export interface ComponentOptionsBase<
/** /**
* SSR only. This is produced by compiler-ssr and attached in compiler-sfc * SSR only. This is produced by compiler-ssr and attached in compiler-sfc
* not user facing, so the typing is lax and for test only. * not user facing, so the typing is lax and for test only.
*
* @internal * @internal
*/ */
ssrRender?: ( ssrRender?: (
@ -136,6 +135,13 @@ export interface ComponentOptionsBase<
$options: ComponentInternalInstance['ctx'] $options: ComponentInternalInstance['ctx']
) => void ) => void
/**
* Only generated by compiler-sfc to mark a ssr render function inlined and
* returned from setup()
* @internal
*/
__ssrInlineRender?: boolean
/** /**
* marker for AsyncComponentWrapper * marker for AsyncComponentWrapper
* @internal * @internal

View File

@ -14,6 +14,8 @@ import { ShapeFlags } from '@vue/shared'
* @private * @private
*/ */
export function useCssVars(getter: (ctx: any) => Record<string, string>) { export function useCssVars(getter: (ctx: any) => Record<string, string>) {
if (!__BROWSER__ && !__TEST__) return
const instance = getCurrentInstance() const instance = getCurrentInstance()
/* istanbul ignore next */ /* istanbul ignore next */
if (!instance) { if (!instance) {

View File

@ -115,11 +115,17 @@ function renderComponentSubTree(
instance instance
) )
} else { } else {
if (!instance.render && !comp.ssrRender && isString(comp.template)) { if (
!instance.render &&
!instance.ssrRender &&
!comp.ssrRender &&
isString(comp.template)
) {
comp.ssrRender = ssrCompile(comp.template, instance) comp.ssrRender = ssrCompile(comp.template, instance)
} }
if (comp.ssrRender) { const ssrRender = instance.ssrRender || comp.ssrRender
if (ssrRender) {
// optimized // optimized
// resolve fallthrough attrs // resolve fallthrough attrs
let attrs = let attrs =
@ -138,7 +144,7 @@ function renderComponentSubTree(
// set current rendering instance for asset resolution // set current rendering instance for asset resolution
setCurrentRenderingInstance(instance) setCurrentRenderingInstance(instance)
comp.ssrRender( ssrRender(
instance.proxy, instance.proxy,
push, push,
instance, instance,

View File

@ -129,7 +129,7 @@ function createConfig(format, output, plugins = []) {
[ [
...Object.keys(pkg.dependencies || {}), ...Object.keys(pkg.dependencies || {}),
...Object.keys(pkg.peerDependencies || {}), ...Object.keys(pkg.peerDependencies || {}),
...['path', 'url'] // for @vue/compiler-sfc ...['path', 'url', 'stream'] // for @vue/compiler-sfc / server-renderer
] ]
// the browser builds of @vue/compiler-sfc requires postcss to be available // the browser builds of @vue/compiler-sfc requires postcss to be available