vue3-yuanma/packages/compiler-ssr/src/ssrCodegenTransform.ts

145 lines
4.5 KiB
TypeScript
Raw Normal View History

import {
RootNode,
BlockStatement,
TemplateLiteral,
createCallExpression,
createTemplateLiteral,
NodeTypes,
TemplateChildNode,
ElementTypes,
createBlockStatement,
CompilerOptions,
2020-02-03 20:51:41 +00:00
isText,
IfStatement,
CallExpression
} from '@vue/compiler-dom'
2020-02-06 22:45:34 +00:00
import { isString, escapeHtml } from '@vue/shared'
import { SSR_INTERPOLATE, ssrHelpers } from './runtimeHelpers'
2020-02-06 02:04:40 +00:00
import { ssrProcessIf } from './transforms/ssrVIf'
import { ssrProcessFor } from './transforms/ssrVFor'
import { ssrProcessSlotOutlet } from './transforms/ssrTransformSlotOutlet'
2020-02-06 04:07:23 +00:00
import { ssrProcessComponent } from './transforms/ssrTransformComponent'
2020-02-06 22:45:34 +00:00
import { ssrProcessElement } from './transforms/ssrTransformElement'
// Because SSR codegen output is completely different from client-side output
// (e.g. multiple elements can be concatenated into a single template literal
// instead of each getting a corresponding call), we need to apply an extra
// transform pass to convert the template AST into a fresh JS AST before
// passing it to codegen.
export function ssrCodegenTransform(ast: RootNode, options: CompilerOptions) {
const context = createSSRTransformContext(options)
const isFragment =
2020-02-06 22:45:34 +00:00
ast.children.length > 1 && ast.children.some(c => !isText(c))
processChildren(ast.children, context, isFragment)
ast.codegenNode = createBlockStatement(context.body)
// Finalize helpers.
// We need to separate helpers imported from 'vue' vs. '@vue/server-renderer'
ast.ssrHelpers = [
...ast.helpers.filter(h => h in ssrHelpers),
...context.helpers
]
ast.helpers = ast.helpers.filter(h => !(h in ssrHelpers))
}
2020-02-03 20:51:41 +00:00
export type SSRTransformContext = ReturnType<typeof createSSRTransformContext>
2020-02-03 22:47:06 +00:00
function createSSRTransformContext(
options: CompilerOptions,
2020-02-06 22:45:34 +00:00
helpers: Set<symbol> = new Set(),
withSlotScopeId = false
2020-02-03 22:47:06 +00:00
) {
const body: BlockStatement['body'] = []
let currentString: TemplateLiteral | null = null
return {
options,
body,
2020-02-03 22:47:06 +00:00
helpers,
2020-02-06 22:45:34 +00:00
withSlotScopeId,
2020-02-03 22:47:06 +00:00
helper<T extends symbol>(name: T): T {
helpers.add(name)
return name
},
pushStringPart(part: TemplateLiteral['elements'][0]) {
if (!currentString) {
2020-02-03 20:51:41 +00:00
const currentCall = createCallExpression(`_push`)
body.push(currentCall)
currentString = createTemplateLiteral([])
currentCall.arguments.push(currentString)
}
const bufferedElements = currentString.elements
const lastItem = bufferedElements[bufferedElements.length - 1]
if (isString(part) && isString(lastItem)) {
bufferedElements[bufferedElements.length - 1] += part
} else {
bufferedElements.push(part)
}
2020-02-03 20:51:41 +00:00
},
pushStatement(statement: IfStatement | CallExpression) {
// close current string
currentString = null
body.push(statement)
}
}
}
2020-02-06 22:45:34 +00:00
function createChildContext(
parent: SSRTransformContext,
withSlotScopeId = parent.withSlotScopeId
2020-02-03 22:47:06 +00:00
): SSRTransformContext {
// ensure child inherits parent helpers
2020-02-06 22:45:34 +00:00
return createSSRTransformContext(
parent.options,
parent.helpers,
withSlotScopeId
)
2020-02-03 22:47:06 +00:00
}
2020-02-03 20:51:41 +00:00
export function processChildren(
children: TemplateChildNode[],
context: SSRTransformContext,
asFragment = false
) {
if (asFragment) {
context.pushStringPart(`<!---->`)
}
for (let i = 0; i < children.length; i++) {
const child = children[i]
if (child.type === NodeTypes.ELEMENT) {
if (child.tagType === ElementTypes.ELEMENT) {
2020-02-06 22:45:34 +00:00
ssrProcessElement(child, context)
} else if (child.tagType === ElementTypes.COMPONENT) {
2020-02-06 04:07:23 +00:00
ssrProcessComponent(child, context)
} else if (child.tagType === ElementTypes.SLOT) {
2020-02-06 02:04:40 +00:00
ssrProcessSlotOutlet(child, context)
}
} else if (child.type === NodeTypes.TEXT) {
context.pushStringPart(escapeHtml(child.content))
} else if (child.type === NodeTypes.INTERPOLATION) {
2020-02-03 22:47:06 +00:00
context.pushStringPart(
createCallExpression(context.helper(SSR_INTERPOLATE), [child.content])
)
} else if (child.type === NodeTypes.IF) {
2020-02-06 02:04:40 +00:00
ssrProcessIf(child, context)
} else if (child.type === NodeTypes.FOR) {
2020-02-06 02:04:40 +00:00
ssrProcessFor(child, context)
}
}
if (asFragment) {
context.pushStringPart(`<!---->`)
}
}
2020-02-06 22:45:34 +00:00
export function processChildrenAsStatement(
children: TemplateChildNode[],
parentContext: SSRTransformContext,
asFragment = false,
withSlotScopeId = parentContext.withSlotScopeId
): BlockStatement {
const childContext = createChildContext(parentContext, withSlotScopeId)
processChildren(children, childContext, asFragment)
return createBlockStatement(childContext.body)
}