vue3-yuanma/packages/compiler-core/src/codegen.ts

590 lines
15 KiB
TypeScript
Raw Normal View History

import {
RootNode,
ChildNode,
ElementNode,
IfNode,
ForNode,
TextNode,
CommentNode,
ExpressionNode,
2019-09-23 04:50:57 +08:00
NodeTypes,
JSChildNode,
CallExpression,
ArrayExpression,
ObjectExpression,
IfBranchNode,
SourceLocation,
Position,
InterpolationNode,
CompoundExpressionNode,
2019-09-28 08:29:20 +08:00
SimpleExpressionNode,
ElementTypes,
SlotFunctionExpression
} from './ast'
import { SourceMapGenerator, RawSourceMap } from 'source-map'
2019-09-25 10:39:20 +08:00
import {
advancePositionWithMutation,
assert,
isSimpleIdentifier
} from './utils'
2019-09-23 04:50:57 +08:00
import { isString, isArray } from '@vue/shared'
2019-09-25 03:49:02 +08:00
import {
RENDER_LIST,
TO_STRING,
CREATE_VNODE,
COMMENT
} from './runtimeConstants'
2019-09-23 04:50:57 +08:00
type CodegenNode = ChildNode | JSChildNode
export interface CodegenOptions {
2019-09-25 03:49:02 +08:00
// - Module mode will generate ES module import statements for helpers
// and export the render function as the default export.
// - Function mode will generate a single `const { helpers... } = Vue`
// statement and return the render function. It is meant to be used with
// `new Function(code)()` to generate a render function at runtime.
// Default: 'function'
2019-09-23 04:50:57 +08:00
mode?: 'module' | 'function'
2019-09-25 03:49:02 +08:00
// Prefix suitable identifiers with _ctx.
// If this option is false, the generated code will be wrapped in a
// `with (this) { ... }` block.
// Default: false
2019-09-24 01:29:41 +08:00
prefixIdentifiers?: boolean
2019-09-25 03:49:02 +08:00
// Generate source map?
// Default: false
sourceMap?: boolean
// Filename for source map generation.
2019-09-25 03:49:02 +08:00
// Default: `template.vue.html`
filename?: string
}
export interface CodegenResult {
code: string
ast: RootNode
map?: RawSourceMap
}
2019-09-22 03:47:26 +08:00
export interface CodegenContext extends Required<CodegenOptions> {
source: string
code: string
line: number
column: number
offset: number
2019-09-23 04:50:57 +08:00
indentLevel: number
map?: SourceMapGenerator
helper(name: string): string
push(code: string, node?: CodegenNode, openOnly?: boolean): void
resetMapping(loc: SourceLocation): void
2019-09-23 04:50:57 +08:00
indent(): void
deindent(withoutNewLine?: boolean): void
2019-09-23 04:50:57 +08:00
newline(): void
}
function createCodegenContext(
ast: RootNode,
2019-09-23 04:50:57 +08:00
{
mode = 'function',
2019-09-24 01:29:41 +08:00
prefixIdentifiers = false,
2019-09-25 03:49:02 +08:00
sourceMap = false,
2019-09-23 04:50:57 +08:00
filename = `template.vue.html`
}: CodegenOptions
): CodegenContext {
const context: CodegenContext = {
2019-09-23 04:50:57 +08:00
mode,
2019-09-24 01:29:41 +08:00
prefixIdentifiers,
2019-09-25 03:49:02 +08:00
sourceMap,
filename,
source: ast.loc.source,
code: ``,
column: 1,
line: 1,
offset: 0,
2019-09-23 04:50:57 +08:00
indentLevel: 0,
2019-09-22 03:47:26 +08:00
// lazy require source-map implementation, only in non-browser builds!
2019-09-25 03:49:02 +08:00
map:
__BROWSER__ || !sourceMap
? undefined
: new (require('source-map')).SourceMapGenerator(),
2019-09-22 03:47:26 +08:00
helper(name) {
return prefixIdentifiers ? name : `_${name}`
},
push(code, node, openOnly) {
2019-09-23 04:50:57 +08:00
context.code += code
if (!__BROWSER__ && context.map) {
if (node) {
let name
if (node.type === NodeTypes.SIMPLE_EXPRESSION && !node.isStatic) {
const content = node.content.replace(/^_ctx\./, '')
if (content !== node.content && isSimpleIdentifier(content)) {
name = content
}
}
addMapping(node.loc.start, name)
}
advancePositionWithMutation(context, code)
if (node && !openOnly) {
addMapping(node.loc.end)
}
}
2019-09-23 04:50:57 +08:00
},
resetMapping(loc: SourceLocation) {
if (!__BROWSER__ && context.map) {
addMapping(loc.start)
}
},
2019-09-23 04:50:57 +08:00
indent() {
newline(++context.indentLevel)
},
deindent(withoutNewLine = false) {
if (withoutNewLine) {
--context.indentLevel
} else {
newline(--context.indentLevel)
}
2019-09-23 04:50:57 +08:00
},
newline() {
newline(context.indentLevel)
}
}
function newline(n: number) {
context.push('\n' + ` `.repeat(n))
}
function addMapping(loc: Position, name?: string) {
context.map!.addMapping({
name,
source: context.filename,
original: {
line: loc.line,
column: loc.column - 1 // source-map column is 0 based
},
generated: {
line: context.line,
column: context.column - 1
}
})
}
2019-09-25 03:49:02 +08:00
if (!__BROWSER__ && context.map) {
context.map.setSourceContent(filename, context.source)
}
return context
}
2019-09-23 04:50:57 +08:00
export function generate(
ast: RootNode,
options: CodegenOptions = {}
): CodegenResult {
const context = createCodegenContext(ast, options)
2019-09-24 01:29:41 +08:00
const { mode, push, prefixIdentifiers, indent, deindent, newline } = context
const hasImports = ast.imports.length
2019-09-26 10:29:37 +08:00
const useWithBlock = !prefixIdentifiers && mode !== 'module'
// preambles
2019-09-23 04:50:57 +08:00
if (mode === 'function') {
// Generate const declaration for helpers
// In prefix mode, we place the const declaration at top so it's done
// only once; But if we not prefixing, we place the decalration inside the
// with block so it doesn't incur the `in` check cost for every helper access.
if (hasImports) {
if (prefixIdentifiers) {
push(`const { ${ast.imports.join(', ')} } = Vue\n`)
} else {
// save Vue in a separate variable to avoid collision
push(`const _Vue = Vue`)
}
}
genHoists(ast.hoists, context)
2019-09-23 04:50:57 +08:00
push(`return `)
} else {
// generate import statements for helpers
if (hasImports) {
2019-09-26 10:29:37 +08:00
push(`import { ${ast.imports.join(', ')} } from "vue"\n`)
}
genHoists(ast.hoists, context)
2019-09-23 04:50:57 +08:00
push(`export default `)
}
// enter render function
2019-09-23 04:50:57 +08:00
push(`function render() {`)
2019-09-23 14:52:54 +08:00
indent()
2019-09-26 10:29:37 +08:00
if (useWithBlock) {
push(`with (this) {`)
indent()
// function mode const declarations should be inside with block
// also they should be renamed to avoid collision with user properties
2019-09-26 10:29:37 +08:00
if (hasImports) {
push(`const { ${ast.imports.map(n => `${n}: _${n}`).join(', ')} } = _Vue`)
newline()
}
} else {
push(`const _ctx = this`)
newline()
}
// generate asset resolution statements
2019-09-23 14:52:54 +08:00
if (ast.statements.length) {
ast.statements.forEach(s => {
push(s)
newline()
})
newline()
}
// generate the VNode tree expression
2019-09-23 04:50:57 +08:00
push(`return `)
genChildren(ast.children, context, true)
2019-09-26 10:29:37 +08:00
if (useWithBlock) {
2019-09-23 04:50:57 +08:00
deindent()
push(`}`)
}
2019-09-26 10:29:37 +08:00
2019-09-23 04:50:57 +08:00
deindent()
push(`}`)
return {
ast,
2019-09-23 04:50:57 +08:00
code: context.code,
map: context.map ? context.map.toJSON() : undefined
}
}
function genHoists(hoists: JSChildNode[], context: CodegenContext) {
hoists.forEach((exp, i) => {
context.push(`const _hoisted_${i + 1} = `)
genNode(exp, context)
context.newline()
})
context.newline()
}
// This will generate a single vnode call if:
// - The target position explicitly allows a single node (root, if, for)
2019-09-28 08:29:20 +08:00
// - The list has length === 1, AND The only child is a:
// - text
// - expression
// - <slot> outlet, which always produces an array
function genChildren(
children: ChildNode[],
context: CodegenContext,
allowSingle: boolean = false
) {
2019-09-25 03:49:02 +08:00
if (!children.length) {
return context.push(`null`)
}
const child = children[0]
2019-09-28 08:29:20 +08:00
const type = child.type
if (
children.length === 1 &&
(allowSingle ||
2019-09-28 08:29:20 +08:00
type === NodeTypes.TEXT ||
type === NodeTypes.INTERPOLATION ||
(type === NodeTypes.ELEMENT &&
(child as ElementNode).tagType === ElementTypes.SLOT))
) {
genNode(child, context)
2019-09-23 04:50:57 +08:00
} else {
genNodeListAsArray(children, context)
}
2019-09-23 04:50:57 +08:00
}
function genNodeListAsArray(
nodes: (string | CodegenNode | ChildNode[])[],
context: CodegenContext
) {
const multilines = nodes.length > 1
context.push(`[`)
multilines && context.indent()
genNodeList(nodes, context, multilines)
multilines && context.deindent()
context.push(`]`)
}
2019-09-23 04:50:57 +08:00
function genNodeList(
nodes: (string | CodegenNode | ChildNode[])[],
context: CodegenContext,
multilines: boolean = false
) {
const { push, newline } = context
for (let i = 0; i < nodes.length; i++) {
const node = nodes[i]
if (isString(node)) {
push(node)
} else if (isArray(node)) {
genChildren(node, context)
2019-09-23 04:50:57 +08:00
} else {
genNode(node, context)
}
if (i < nodes.length - 1) {
if (multilines) {
push(',')
newline()
} else {
push(', ')
}
}
}
}
function genNode(node: CodegenNode, context: CodegenContext) {
switch (node.type) {
case NodeTypes.ELEMENT:
genElement(node, context)
break
case NodeTypes.TEXT:
genText(node, context)
break
case NodeTypes.SIMPLE_EXPRESSION:
genExpression(node, context)
break
case NodeTypes.INTERPOLATION:
genInterpolation(node, context)
break
case NodeTypes.COMPOUND_EXPRESSION:
genCompoundExpression(node, context)
break
case NodeTypes.COMMENT:
genComment(node, context)
break
case NodeTypes.IF:
genIf(node, context)
break
case NodeTypes.FOR:
genFor(node, context)
break
2019-09-23 04:50:57 +08:00
case NodeTypes.JS_CALL_EXPRESSION:
genCallExpression(node, context)
break
case NodeTypes.JS_OBJECT_EXPRESSION:
genObjectExpression(node, context)
break
case NodeTypes.JS_ARRAY_EXPRESSION:
genArrayExpression(node, context)
2019-09-23 14:52:54 +08:00
break
case NodeTypes.JS_SLOT_FUNCTION:
genSlotFunction(node, context)
break
2019-09-23 14:52:54 +08:00
default:
/* istanbul ignore if */
if (__DEV__) {
2019-09-23 14:52:54 +08:00
assert(false, `unhandled codegen node type: ${(node as any).type}`)
// make sure we exhaust all possible types
const exhaustiveCheck: never = node
return exhaustiveCheck
}
}
}
2019-09-23 04:50:57 +08:00
function genElement(node: ElementNode, context: CodegenContext) {
__DEV__ &&
assert(
node.codegenNode != null,
`AST is not transformed for codegen. ` +
`Apply appropriate transforms first.`
)
genCallExpression(node.codegenNode!, context, false)
}
function genText(
node: TextNode | SimpleExpressionNode,
context: CodegenContext
) {
context.push(JSON.stringify(node.content), node)
}
function genExpression(node: SimpleExpressionNode, context: CodegenContext) {
const { content, isStatic } = node
context.push(isStatic ? JSON.stringify(content) : content, node)
}
function genInterpolation(node: InterpolationNode, context: CodegenContext) {
const { push, helper } = context
push(`${helper(TO_STRING)}(`)
genNode(node.content, context)
push(`)`)
}
function genCompoundExpression(
node: CompoundExpressionNode,
context: CodegenContext
) {
for (let i = 0; i < node.children!.length; i++) {
const child = node.children![i]
if (isString(child)) {
context.push(child)
} else {
genExpression(child, context)
}
2019-09-23 14:52:54 +08:00
}
2019-09-23 04:50:57 +08:00
}
function genExpressionAsPropertyKey(
node: ExpressionNode,
context: CodegenContext
) {
2019-09-24 09:22:52 +08:00
const { push } = context
if (node.type === NodeTypes.COMPOUND_EXPRESSION) {
2019-09-24 09:22:52 +08:00
push(`[`)
genCompoundExpression(node, context)
push(`]`)
} else if (node.isStatic) {
2019-09-23 04:50:57 +08:00
// only quote keys if necessary
const text = isSimpleIdentifier(node.content)
? node.content
: JSON.stringify(node.content)
2019-09-24 09:22:52 +08:00
push(text, node)
2019-09-23 04:50:57 +08:00
} else {
push(`[${node.content}]`, node)
2019-09-23 14:52:54 +08:00
}
}
function genComment(node: CommentNode, context: CodegenContext) {
2019-09-25 03:49:02 +08:00
if (__DEV__) {
const { push, helper } = context
push(
`${helper(CREATE_VNODE)}(${helper(COMMENT)}, 0, ${JSON.stringify(
node.content
)})`,
2019-09-25 03:49:02 +08:00
node
)
}
}
// control flow
2019-09-23 04:50:57 +08:00
function genIf(node: IfNode, context: CodegenContext) {
genIfBranch(node.branches[0], node.branches, 1, context)
}
2019-09-23 04:50:57 +08:00
function genIfBranch(
{ condition, children }: IfBranchNode,
2019-09-23 04:50:57 +08:00
branches: IfBranchNode[],
nextIndex: number,
context: CodegenContext
) {
if (condition) {
// v-if or v-else-if
const { push, indent, deindent, newline } = context
if (condition.type === NodeTypes.SIMPLE_EXPRESSION) {
const needsQuote = !isSimpleIdentifier(condition.content)
needsQuote && push(`(`)
genExpression(condition, context)
needsQuote && push(`)`)
} else {
genCompoundExpression(condition, context)
}
indent()
context.indentLevel++
push(`? `)
genChildren(children, context, true)
context.indentLevel--
newline()
push(`: `)
2019-09-23 04:50:57 +08:00
if (nextIndex < branches.length) {
genIfBranch(branches[nextIndex], branches, nextIndex + 1, context)
} else {
context.push(`null`)
}
deindent(true /* without newline */)
2019-09-23 04:50:57 +08:00
} else {
// v-else
__DEV__ && assert(nextIndex === branches.length)
genChildren(children, context, true)
2019-09-23 04:50:57 +08:00
}
}
function genFor(node: ForNode, context: CodegenContext) {
const { push, helper, indent, deindent } = context
2019-09-23 04:50:57 +08:00
const { source, keyAlias, valueAlias, objectIndexAlias, children } = node
push(`${helper(RENDER_LIST)}(`, node, true)
genNode(source, context)
push(`, (`)
2019-09-23 04:50:57 +08:00
if (valueAlias) {
genExpression(valueAlias, context)
2019-09-23 04:50:57 +08:00
}
if (keyAlias) {
if (!valueAlias) {
2019-09-25 04:35:01 +08:00
push(`__value`)
2019-09-23 04:50:57 +08:00
}
push(`, `)
genExpression(keyAlias, context)
2019-09-23 04:50:57 +08:00
}
if (objectIndexAlias) {
if (!keyAlias) {
if (!valueAlias) {
2019-09-25 04:35:01 +08:00
push(`__value, __key`)
} else {
2019-09-25 04:35:01 +08:00
push(`, __key`)
2019-09-23 04:50:57 +08:00
}
}
push(`, `)
genExpression(objectIndexAlias, context)
2019-09-23 04:50:57 +08:00
}
push(`) => {`)
indent()
push(`return `)
genChildren(children, context, true)
deindent()
push(`})`)
2019-09-23 04:50:57 +08:00
}
// JavaScript
function genCallExpression(
node: CallExpression,
context: CodegenContext,
multilines = node.arguments.length > 2
2019-09-23 04:50:57 +08:00
) {
2019-09-28 08:29:20 +08:00
if (isString(node.callee)) {
context.push(node.callee + `(`, node, true)
} else {
genNode(node.callee, context)
context.push(`(`)
}
2019-09-23 04:50:57 +08:00
multilines && context.indent()
genNodeList(node.arguments, context, multilines)
multilines && context.deindent()
context.push(`)`)
}
function genObjectExpression(node: ObjectExpression, context: CodegenContext) {
const { push, indent, deindent, newline, resetMapping } = context
2019-09-23 04:50:57 +08:00
const { properties } = node
const multilines = properties.length > 1
push(multilines ? `{` : `{ `)
2019-09-23 04:50:57 +08:00
multilines && indent()
for (let i = 0; i < properties.length; i++) {
const { key, value, loc } = properties[i]
resetMapping(loc) // reset source mapping for every property.
2019-09-23 04:50:57 +08:00
// key
genExpressionAsPropertyKey(key, context)
push(`: `)
// value
2019-09-26 00:39:46 +08:00
genNode(value, context)
2019-09-23 04:50:57 +08:00
if (i < properties.length - 1) {
2019-09-25 04:35:01 +08:00
// will only reach this if it's multilines
push(`,`)
newline()
2019-09-23 04:50:57 +08:00
}
}
multilines && deindent()
push(multilines ? `}` : ` }`)
2019-09-23 04:50:57 +08:00
}
function genArrayExpression(node: ArrayExpression, context: CodegenContext) {
genNodeListAsArray(node.elements, context)
}
function genSlotFunction(
node: SlotFunctionExpression,
context: CodegenContext
) {
context.push(`(`, node)
if (node.params) genNode(node.params, context)
context.push(`) => `)
// pre-normalized slots should always return arrays
genNodeListAsArray(node.returns, context)
}