mirror of https://github.com/vuejs/core.git
feat: ssr support for `<style vars>`
This commit is contained in:
parent
b6cdd5621e
commit
b9595e64cf
|
@ -204,10 +204,11 @@ export function generate(
|
||||||
genFunctionPreamble(ast, context)
|
genFunctionPreamble(ast, context)
|
||||||
}
|
}
|
||||||
|
|
||||||
// enter render function
|
// binding optimizations
|
||||||
const optimizeSources = options.bindingMetadata
|
const optimizeSources = options.bindingMetadata
|
||||||
? `, $props, $setup, $data, $options`
|
? `, $props, $setup, $data, $options`
|
||||||
: ``
|
: ``
|
||||||
|
// enter render function
|
||||||
if (!ssr) {
|
if (!ssr) {
|
||||||
if (genScopeId) {
|
if (genScopeId) {
|
||||||
push(`const render = ${PURE_ANNOTATION}_withId(`)
|
push(`const render = ${PURE_ANNOTATION}_withId(`)
|
||||||
|
|
|
@ -126,6 +126,11 @@ export interface TransformOptions {
|
||||||
* `ssrRender` option instead of `render`.
|
* `ssrRender` option instead of `render`.
|
||||||
*/
|
*/
|
||||||
ssr?: boolean
|
ssr?: boolean
|
||||||
|
/**
|
||||||
|
* SFC <style vars> injection string
|
||||||
|
* needed to render inline CSS variables on component root
|
||||||
|
*/
|
||||||
|
ssrCssVars?: string
|
||||||
/**
|
/**
|
||||||
* Optional binding metadata analyzed from script - used to optimize
|
* Optional binding metadata analyzed from script - used to optimize
|
||||||
* binding access when `prefixIdentifiers` is enabled.
|
* binding access when `prefixIdentifiers` is enabled.
|
||||||
|
|
|
@ -120,6 +120,7 @@ export function createTransformContext(
|
||||||
expressionPlugins = [],
|
expressionPlugins = [],
|
||||||
scopeId = null,
|
scopeId = null,
|
||||||
ssr = false,
|
ssr = false,
|
||||||
|
ssrCssVars = ``,
|
||||||
bindingMetadata = {},
|
bindingMetadata = {},
|
||||||
onError = defaultOnError
|
onError = defaultOnError
|
||||||
}: TransformOptions
|
}: TransformOptions
|
||||||
|
@ -136,6 +137,7 @@ export function createTransformContext(
|
||||||
expressionPlugins,
|
expressionPlugins,
|
||||||
scopeId,
|
scopeId,
|
||||||
ssr,
|
ssr,
|
||||||
|
ssrCssVars,
|
||||||
bindingMetadata,
|
bindingMetadata,
|
||||||
onError,
|
onError,
|
||||||
|
|
||||||
|
@ -148,7 +150,7 @@ export function createTransformContext(
|
||||||
imports: new Set(),
|
imports: new Set(),
|
||||||
temps: 0,
|
temps: 0,
|
||||||
cached: 0,
|
cached: 0,
|
||||||
identifiers: {},
|
identifiers: Object.create(null),
|
||||||
scopes: {
|
scopes: {
|
||||||
vFor: 0,
|
vFor: 0,
|
||||||
vSlot: 0,
|
vSlot: 0,
|
||||||
|
|
|
@ -591,6 +591,11 @@ export function compileScript(
|
||||||
Object.keys(setupExports).forEach(key => {
|
Object.keys(setupExports).forEach(key => {
|
||||||
bindings[key] = 'setup'
|
bindings[key] = 'setup'
|
||||||
})
|
})
|
||||||
|
Object.keys(typeDeclaredProps).forEach(key => {
|
||||||
|
bindings[key] = 'props'
|
||||||
|
})
|
||||||
|
// TODO analyze props if user declared props via `export default {}` inside
|
||||||
|
// <script setup>
|
||||||
|
|
||||||
s.trim()
|
s.trim()
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -0,0 +1,99 @@
|
||||||
|
import { compile } from '../src'
|
||||||
|
|
||||||
|
describe('ssr: inject <style vars>', () => {
|
||||||
|
test('basic', () => {
|
||||||
|
expect(
|
||||||
|
compile(`<div/>`, {
|
||||||
|
ssrCssVars: `{ color }`
|
||||||
|
}).code
|
||||||
|
).toMatchInlineSnapshot(`
|
||||||
|
"const { mergeProps: _mergeProps } = require(\\"vue\\")
|
||||||
|
const { ssrResolveCssVars: _ssrResolveCssVars, ssrRenderAttrs: _ssrRenderAttrs } = require(\\"@vue/server-renderer\\")
|
||||||
|
|
||||||
|
return function ssrRender(_ctx, _push, _parent, _attrs) {
|
||||||
|
const _cssVars = ssrResolveCssVars({ color: _ctx.color })
|
||||||
|
_push(\`<div\${_ssrRenderAttrs(_mergeProps(_attrs, _cssVars))}></div>\`)
|
||||||
|
}"
|
||||||
|
`)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('fragment', () => {
|
||||||
|
expect(
|
||||||
|
compile(`<div/><div/>`, {
|
||||||
|
ssrCssVars: `{ color }`
|
||||||
|
}).code
|
||||||
|
).toMatchInlineSnapshot(`
|
||||||
|
"const { ssrResolveCssVars: _ssrResolveCssVars, ssrRenderAttrs: _ssrRenderAttrs } = require(\\"@vue/server-renderer\\")
|
||||||
|
|
||||||
|
return function ssrRender(_ctx, _push, _parent, _attrs) {
|
||||||
|
const _cssVars = ssrResolveCssVars({ color: _ctx.color })
|
||||||
|
_push(\`<!--[--><div\${
|
||||||
|
_ssrRenderAttrs(_cssVars)
|
||||||
|
}></div><div\${
|
||||||
|
_ssrRenderAttrs(_cssVars)
|
||||||
|
}></div><!--]-->\`)
|
||||||
|
}"
|
||||||
|
`)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('passing on to components', () => {
|
||||||
|
expect(
|
||||||
|
compile(`<div/><foo/>`, {
|
||||||
|
ssrCssVars: `{ color }`
|
||||||
|
}).code
|
||||||
|
).toMatchInlineSnapshot(`
|
||||||
|
"const { resolveComponent: _resolveComponent } = require(\\"vue\\")
|
||||||
|
const { ssrResolveCssVars: _ssrResolveCssVars, ssrRenderAttrs: _ssrRenderAttrs, ssrRenderComponent: _ssrRenderComponent } = require(\\"@vue/server-renderer\\")
|
||||||
|
|
||||||
|
return function ssrRender(_ctx, _push, _parent, _attrs) {
|
||||||
|
const _component_foo = _resolveComponent(\\"foo\\")
|
||||||
|
|
||||||
|
const _cssVars = ssrResolveCssVars({ color: _ctx.color })
|
||||||
|
_push(\`<!--[--><div\${_ssrRenderAttrs(_cssVars)}></div>\`)
|
||||||
|
_push(_ssrRenderComponent(_component_foo, _cssVars, null, _parent))
|
||||||
|
_push(\`<!--]-->\`)
|
||||||
|
}"
|
||||||
|
`)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('v-if branches', () => {
|
||||||
|
expect(
|
||||||
|
compile(`<div v-if="ok"/><template v-else><div/><div/></template>`, {
|
||||||
|
ssrCssVars: `{ color }`
|
||||||
|
}).code
|
||||||
|
).toMatchInlineSnapshot(`
|
||||||
|
"const { mergeProps: _mergeProps } = require(\\"vue\\")
|
||||||
|
const { ssrResolveCssVars: _ssrResolveCssVars, ssrRenderAttrs: _ssrRenderAttrs } = require(\\"@vue/server-renderer\\")
|
||||||
|
|
||||||
|
return function ssrRender(_ctx, _push, _parent, _attrs) {
|
||||||
|
const _cssVars = ssrResolveCssVars({ color: _ctx.color })
|
||||||
|
if (_ctx.ok) {
|
||||||
|
_push(\`<div\${_ssrRenderAttrs(_mergeProps(_attrs, _cssVars))}></div>\`)
|
||||||
|
} else {
|
||||||
|
_push(\`<!--[--><div\${
|
||||||
|
_ssrRenderAttrs(_cssVars)
|
||||||
|
}></div><div\${
|
||||||
|
_ssrRenderAttrs(_cssVars)
|
||||||
|
}></div><!--]-->\`)
|
||||||
|
}
|
||||||
|
}"
|
||||||
|
`)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('w/ scopeId', () => {
|
||||||
|
expect(
|
||||||
|
compile(`<div/>`, {
|
||||||
|
ssrCssVars: `{ color }`,
|
||||||
|
scopeId: 'data-v-foo'
|
||||||
|
}).code
|
||||||
|
).toMatchInlineSnapshot(`
|
||||||
|
"const { mergeProps: _mergeProps } = require(\\"vue\\")
|
||||||
|
const { ssrResolveCssVars: _ssrResolveCssVars, ssrRenderAttrs: _ssrRenderAttrs } = require(\\"@vue/server-renderer\\")
|
||||||
|
|
||||||
|
return function ssrRender(_ctx, _push, _parent, _attrs) {
|
||||||
|
const _cssVars = ssrResolveCssVars({ color: _ctx.color }, \\"data-v-foo\\")
|
||||||
|
_push(\`<div\${_ssrRenderAttrs(_mergeProps(_attrs, _cssVars))} data-v-foo></div>\`)
|
||||||
|
}"
|
||||||
|
`)
|
||||||
|
})
|
||||||
|
})
|
|
@ -24,6 +24,7 @@ import { ssrTransformFor } from './transforms/ssrVFor'
|
||||||
import { ssrTransformModel } from './transforms/ssrVModel'
|
import { ssrTransformModel } from './transforms/ssrVModel'
|
||||||
import { ssrTransformShow } from './transforms/ssrVShow'
|
import { ssrTransformShow } from './transforms/ssrVShow'
|
||||||
import { ssrInjectFallthroughAttrs } from './transforms/ssrInjectFallthroughAttrs'
|
import { ssrInjectFallthroughAttrs } from './transforms/ssrInjectFallthroughAttrs'
|
||||||
|
import { ssrInjectCssVars } from './transforms/ssrInjectCssVars'
|
||||||
|
|
||||||
export function compile(
|
export function compile(
|
||||||
template: string,
|
template: string,
|
||||||
|
@ -57,6 +58,7 @@ export function compile(
|
||||||
transformExpression,
|
transformExpression,
|
||||||
ssrTransformSlotOutlet,
|
ssrTransformSlotOutlet,
|
||||||
ssrInjectFallthroughAttrs,
|
ssrInjectFallthroughAttrs,
|
||||||
|
ssrInjectCssVars,
|
||||||
ssrTransformElement,
|
ssrTransformElement,
|
||||||
ssrTransformComponent,
|
ssrTransformComponent,
|
||||||
trackSlotScopes,
|
trackSlotScopes,
|
||||||
|
|
|
@ -16,6 +16,7 @@ export const SSR_RENDER_DYNAMIC_MODEL = Symbol(`ssrRenderDynamicModel`)
|
||||||
export const SSR_GET_DYNAMIC_MODEL_PROPS = Symbol(`ssrGetDynamicModelProps`)
|
export const SSR_GET_DYNAMIC_MODEL_PROPS = Symbol(`ssrGetDynamicModelProps`)
|
||||||
export const SSR_RENDER_TELEPORT = Symbol(`ssrRenderTeleport`)
|
export const SSR_RENDER_TELEPORT = Symbol(`ssrRenderTeleport`)
|
||||||
export const SSR_RENDER_SUSPENSE = Symbol(`ssrRenderSuspense`)
|
export const SSR_RENDER_SUSPENSE = Symbol(`ssrRenderSuspense`)
|
||||||
|
export const SSR_RESOLVE_CSS_VARS = Symbol(`ssrResolveCssVars`)
|
||||||
|
|
||||||
export const ssrHelpers = {
|
export const ssrHelpers = {
|
||||||
[SSR_INTERPOLATE]: `ssrInterpolate`,
|
[SSR_INTERPOLATE]: `ssrInterpolate`,
|
||||||
|
@ -33,7 +34,8 @@ export const ssrHelpers = {
|
||||||
[SSR_RENDER_DYNAMIC_MODEL]: `ssrRenderDynamicModel`,
|
[SSR_RENDER_DYNAMIC_MODEL]: `ssrRenderDynamicModel`,
|
||||||
[SSR_GET_DYNAMIC_MODEL_PROPS]: `ssrGetDynamicModelProps`,
|
[SSR_GET_DYNAMIC_MODEL_PROPS]: `ssrGetDynamicModelProps`,
|
||||||
[SSR_RENDER_TELEPORT]: `ssrRenderTeleport`,
|
[SSR_RENDER_TELEPORT]: `ssrRenderTeleport`,
|
||||||
[SSR_RENDER_SUSPENSE]: `ssrRenderSuspense`
|
[SSR_RENDER_SUSPENSE]: `ssrRenderSuspense`,
|
||||||
|
[SSR_RESOLVE_CSS_VARS]: `ssrResolveCssVars`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: these are helpers imported from @vue/server-renderer
|
// Note: these are helpers imported from @vue/server-renderer
|
||||||
|
|
|
@ -11,10 +11,19 @@ import {
|
||||||
CompilerOptions,
|
CompilerOptions,
|
||||||
IfStatement,
|
IfStatement,
|
||||||
CallExpression,
|
CallExpression,
|
||||||
isText
|
isText,
|
||||||
|
processExpression,
|
||||||
|
createSimpleExpression,
|
||||||
|
createCompoundExpression,
|
||||||
|
createTransformContext,
|
||||||
|
createRoot
|
||||||
} from '@vue/compiler-dom'
|
} from '@vue/compiler-dom'
|
||||||
import { isString, escapeHtml } from '@vue/shared'
|
import { isString, escapeHtml } from '@vue/shared'
|
||||||
import { SSR_INTERPOLATE, ssrHelpers } from './runtimeHelpers'
|
import {
|
||||||
|
SSR_INTERPOLATE,
|
||||||
|
ssrHelpers,
|
||||||
|
SSR_RESOLVE_CSS_VARS
|
||||||
|
} from './runtimeHelpers'
|
||||||
import { ssrProcessIf } from './transforms/ssrVIf'
|
import { ssrProcessIf } from './transforms/ssrVIf'
|
||||||
import { ssrProcessFor } from './transforms/ssrVFor'
|
import { ssrProcessFor } from './transforms/ssrVFor'
|
||||||
import { ssrProcessSlotOutlet } from './transforms/ssrTransformSlotOutlet'
|
import { ssrProcessSlotOutlet } from './transforms/ssrTransformSlotOutlet'
|
||||||
|
@ -30,6 +39,25 @@ import { createSSRCompilerError, SSRErrorCodes } from './errors'
|
||||||
|
|
||||||
export function ssrCodegenTransform(ast: RootNode, options: CompilerOptions) {
|
export function ssrCodegenTransform(ast: RootNode, options: CompilerOptions) {
|
||||||
const context = createSSRTransformContext(ast, options)
|
const context = createSSRTransformContext(ast, options)
|
||||||
|
|
||||||
|
// inject <style vars> resolution
|
||||||
|
// we do this instead of inlining the expression to ensure the vars are
|
||||||
|
// only resolved once per render
|
||||||
|
if (options.ssrCssVars) {
|
||||||
|
const varsExp = processExpression(
|
||||||
|
createSimpleExpression(options.ssrCssVars, false),
|
||||||
|
createTransformContext(createRoot([]), options)
|
||||||
|
)
|
||||||
|
context.body.push(
|
||||||
|
createCompoundExpression([
|
||||||
|
`const _cssVars = ${ssrHelpers[SSR_RESOLVE_CSS_VARS]}(`,
|
||||||
|
varsExp,
|
||||||
|
options.scopeId ? `, ${JSON.stringify(options.scopeId)}` : ``,
|
||||||
|
`)`
|
||||||
|
])
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
const isFragment =
|
const isFragment =
|
||||||
ast.children.length > 1 && ast.children.some(c => !isText(c))
|
ast.children.length > 1 && ast.children.some(c => !isText(c))
|
||||||
processChildren(ast.children, context, isFragment)
|
processChildren(ast.children, context, isFragment)
|
||||||
|
|
|
@ -0,0 +1,57 @@
|
||||||
|
import {
|
||||||
|
NodeTransform,
|
||||||
|
NodeTypes,
|
||||||
|
ElementTypes,
|
||||||
|
locStub,
|
||||||
|
createSimpleExpression,
|
||||||
|
RootNode,
|
||||||
|
TemplateChildNode,
|
||||||
|
findDir
|
||||||
|
} from '@vue/compiler-dom'
|
||||||
|
import { SSR_RESOLVE_CSS_VARS } from '../runtimeHelpers'
|
||||||
|
|
||||||
|
export const ssrInjectCssVars: NodeTransform = (node, context) => {
|
||||||
|
if (!context.ssrCssVars) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// _cssVars is initailized once per render function
|
||||||
|
// the code is injected in ssrCodegenTrasnform when creating the
|
||||||
|
// ssr transform context
|
||||||
|
if (node.type === NodeTypes.ROOT) {
|
||||||
|
context.identifiers._cssVars = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
const parent = context.parent
|
||||||
|
if (!parent || parent.type !== NodeTypes.ROOT) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
context.helper(SSR_RESOLVE_CSS_VARS)
|
||||||
|
|
||||||
|
if (node.type === NodeTypes.IF_BRANCH) {
|
||||||
|
for (const child of node.children) {
|
||||||
|
injectCssVars(child)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
injectCssVars(node)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function injectCssVars(node: RootNode | TemplateChildNode) {
|
||||||
|
if (
|
||||||
|
node.type === NodeTypes.ELEMENT &&
|
||||||
|
(node.tagType === ElementTypes.ELEMENT ||
|
||||||
|
node.tagType === ElementTypes.COMPONENT) &&
|
||||||
|
!findDir(node, 'for')
|
||||||
|
) {
|
||||||
|
node.props.push({
|
||||||
|
type: NodeTypes.DIRECTIVE,
|
||||||
|
name: 'bind',
|
||||||
|
arg: undefined,
|
||||||
|
exp: createSimpleExpression(`_cssVars`, false),
|
||||||
|
modifiers: [],
|
||||||
|
loc: locStub
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,6 +1,6 @@
|
||||||
import {
|
import {
|
||||||
render,
|
render,
|
||||||
useCSSVars,
|
useCssVars,
|
||||||
h,
|
h,
|
||||||
reactive,
|
reactive,
|
||||||
nextTick,
|
nextTick,
|
||||||
|
@ -37,7 +37,7 @@ describe('useCssVars', () => {
|
||||||
await assertCssVars(state => ({
|
await assertCssVars(state => ({
|
||||||
setup() {
|
setup() {
|
||||||
// test receiving render context
|
// test receiving render context
|
||||||
useCSSVars((ctx: any) => ({
|
useCssVars((ctx: any) => ({
|
||||||
color: ctx.color
|
color: ctx.color
|
||||||
}))
|
}))
|
||||||
return state
|
return state
|
||||||
|
@ -51,7 +51,7 @@ describe('useCssVars', () => {
|
||||||
test('on fragment root', async () => {
|
test('on fragment root', async () => {
|
||||||
await assertCssVars(state => ({
|
await assertCssVars(state => ({
|
||||||
setup() {
|
setup() {
|
||||||
useCSSVars(() => state)
|
useCssVars(() => state)
|
||||||
return () => [h('div'), h('div')]
|
return () => [h('div'), h('div')]
|
||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
|
@ -62,7 +62,7 @@ describe('useCssVars', () => {
|
||||||
|
|
||||||
await assertCssVars(state => ({
|
await assertCssVars(state => ({
|
||||||
setup() {
|
setup() {
|
||||||
useCSSVars(() => state)
|
useCssVars(() => state)
|
||||||
return () => h(Child)
|
return () => h(Child)
|
||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
|
@ -75,7 +75,7 @@ describe('useCssVars', () => {
|
||||||
state => ({
|
state => ({
|
||||||
__scopeId: id,
|
__scopeId: id,
|
||||||
setup() {
|
setup() {
|
||||||
useCSSVars(() => state, true)
|
useCssVars(() => state, true)
|
||||||
return () => h('div')
|
return () => h('div')
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { warn, getCurrentInstance } from '@vue/runtime-core'
|
import { warn, getCurrentInstance } from '@vue/runtime-core'
|
||||||
import { EMPTY_OBJ } from '@vue/shared'
|
import { EMPTY_OBJ } from '@vue/shared'
|
||||||
|
|
||||||
export function useCSSModule(name = '$style'): Record<string, string> {
|
export function useCssModule(name = '$style'): Record<string, string> {
|
||||||
if (!__GLOBAL__) {
|
if (!__GLOBAL__) {
|
||||||
const instance = getCurrentInstance()!
|
const instance = getCurrentInstance()!
|
||||||
if (!instance) {
|
if (!instance) {
|
||||||
|
|
|
@ -9,7 +9,7 @@ import {
|
||||||
} from '@vue/runtime-core'
|
} from '@vue/runtime-core'
|
||||||
import { ShapeFlags } from '@vue/shared/src'
|
import { ShapeFlags } from '@vue/shared/src'
|
||||||
|
|
||||||
export function useCSSVars(
|
export function useCssVars(
|
||||||
getter: (ctx: ComponentPublicInstance) => Record<string, string>,
|
getter: (ctx: ComponentPublicInstance) => Record<string, string>,
|
||||||
scoped = false
|
scoped = false
|
||||||
) {
|
) {
|
||||||
|
|
|
@ -114,8 +114,8 @@ function normalizeContainer(container: Element | string): Element | null {
|
||||||
}
|
}
|
||||||
|
|
||||||
// SFC CSS utilities
|
// SFC CSS utilities
|
||||||
export { useCSSModule } from './helpers/useCssModule'
|
export { useCssModule } from './helpers/useCssModule'
|
||||||
export { useCSSVars } from './helpers/useCssVars'
|
export { useCssVars } from './helpers/useCssVars'
|
||||||
|
|
||||||
// DOM-only components
|
// DOM-only components
|
||||||
export { Transition, TransitionProps } from './components/Transition'
|
export { Transition, TransitionProps } from './components/Transition'
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
import { ssrResolveCssVars } from '../src'
|
||||||
|
|
||||||
|
describe('ssr: resolveCssVars', () => {
|
||||||
|
test('should work', () => {
|
||||||
|
expect(ssrResolveCssVars({ color: 'red' })).toMatchObject({
|
||||||
|
style: {
|
||||||
|
'--color': 'red'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should work with scopeId', () => {
|
||||||
|
expect(ssrResolveCssVars({ color: 'red' }, 'scoped')).toMatchObject({
|
||||||
|
style: {
|
||||||
|
'--scoped-color': 'red'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should strip data-v prefix', () => {
|
||||||
|
expect(ssrResolveCssVars({ color: 'red' }, 'data-v-123456')).toMatchObject({
|
||||||
|
style: {
|
||||||
|
'--123456-color': 'red'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
|
@ -0,0 +1,11 @@
|
||||||
|
export function ssrResolveCssVars(
|
||||||
|
source: Record<string, string>,
|
||||||
|
scopeId?: string
|
||||||
|
) {
|
||||||
|
const style: Record<string, string> = {}
|
||||||
|
const prefix = scopeId ? `${scopeId.replace(/^data-v-/, '')}-` : ``
|
||||||
|
for (const key in source) {
|
||||||
|
style[`--${prefix}${key}`] = source[key]
|
||||||
|
}
|
||||||
|
return { style }
|
||||||
|
}
|
|
@ -18,6 +18,7 @@ export {
|
||||||
export { ssrInterpolate } from './helpers/ssrInterpolate'
|
export { ssrInterpolate } from './helpers/ssrInterpolate'
|
||||||
export { ssrRenderList } from './helpers/ssrRenderList'
|
export { ssrRenderList } from './helpers/ssrRenderList'
|
||||||
export { ssrRenderSuspense } from './helpers/ssrRenderSuspense'
|
export { ssrRenderSuspense } from './helpers/ssrRenderSuspense'
|
||||||
|
export { ssrResolveCssVars } from './helpers/ssrResolveCssVars'
|
||||||
|
|
||||||
// v-model helpers
|
// v-model helpers
|
||||||
export {
|
export {
|
||||||
|
|
|
@ -9,7 +9,8 @@ export const compilerOptions: CompilerOptions = reactive({
|
||||||
optimizeImports: false,
|
optimizeImports: false,
|
||||||
hoistStatic: false,
|
hoistStatic: false,
|
||||||
cacheHandlers: false,
|
cacheHandlers: false,
|
||||||
scopeId: null
|
scopeId: null,
|
||||||
|
ssrCssVars: `{ color }`
|
||||||
})
|
})
|
||||||
|
|
||||||
const App = {
|
const App = {
|
||||||
|
|
Loading…
Reference in New Issue