645 lines
17 KiB
TypeScript
645 lines
17 KiB
TypeScript
import {
|
|
generate,
|
|
NodeTypes,
|
|
RootNode,
|
|
SourceLocation,
|
|
createSimpleExpression,
|
|
Namespaces,
|
|
ElementTypes,
|
|
CallExpression,
|
|
createObjectExpression,
|
|
createObjectProperty,
|
|
createArrayExpression,
|
|
ElementNode,
|
|
createCompoundExpression,
|
|
createInterpolation
|
|
} from '../src'
|
|
import {
|
|
CREATE_VNODE,
|
|
COMMENT,
|
|
TO_STRING,
|
|
RENDER_LIST
|
|
} from '../src/runtimeConstants'
|
|
|
|
const mockLoc: SourceLocation = {
|
|
source: ``,
|
|
start: {
|
|
offset: 0,
|
|
line: 1,
|
|
column: 1
|
|
},
|
|
end: {
|
|
offset: 3,
|
|
line: 1,
|
|
column: 4
|
|
}
|
|
}
|
|
|
|
function createRoot(options: Partial<RootNode> = {}): RootNode {
|
|
return {
|
|
type: NodeTypes.ROOT,
|
|
children: [],
|
|
imports: [],
|
|
statements: [],
|
|
hoists: [],
|
|
loc: mockLoc,
|
|
...options
|
|
}
|
|
}
|
|
|
|
describe('compiler: codegen', () => {
|
|
test('module mode preamble', () => {
|
|
const root = createRoot({
|
|
imports: [`helperOne`, `helperTwo`]
|
|
})
|
|
const { code } = generate(root, { mode: 'module' })
|
|
expect(code).toMatch(`import { helperOne, helperTwo } from "vue"`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('function mode preamble', () => {
|
|
const root = createRoot({
|
|
imports: [`helperOne`, `helperTwo`]
|
|
})
|
|
const { code } = generate(root, { mode: 'function' })
|
|
expect(code).toMatch(`const _Vue = Vue`)
|
|
expect(code).toMatch(
|
|
`const { helperOne: _helperOne, helperTwo: _helperTwo } = _Vue`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('function mode preamble w/ prefixIdentifiers: true', () => {
|
|
const root = createRoot({
|
|
imports: [`helperOne`, `helperTwo`]
|
|
})
|
|
const { code } = generate(root, {
|
|
mode: 'function',
|
|
prefixIdentifiers: true
|
|
})
|
|
expect(code).not.toMatch(`const _Vue = Vue`)
|
|
expect(code).toMatch(`const { helperOne, helperTwo } = Vue`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('statements', () => {
|
|
const root = createRoot({
|
|
statements: [`const a = 1`, `const b = 2`]
|
|
})
|
|
const { code } = generate(root, { mode: 'function' })
|
|
expect(code).toMatch(`const a = 1\n`)
|
|
expect(code).toMatch(`const b = 2\n`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('hoists', () => {
|
|
const root = createRoot({
|
|
hoists: [
|
|
createSimpleExpression(`hello`, false, mockLoc),
|
|
createObjectExpression(
|
|
[
|
|
createObjectProperty(
|
|
createSimpleExpression(`id`, true, mockLoc),
|
|
createSimpleExpression(`foo`, true, mockLoc),
|
|
mockLoc
|
|
)
|
|
],
|
|
mockLoc
|
|
)
|
|
]
|
|
})
|
|
const { code } = generate(root)
|
|
expect(code).toMatch(`const _hoisted_1 = hello`)
|
|
expect(code).toMatch(`const _hoisted_2 = { id: "foo" }`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('prefixIdentifiers: true should inject _ctx statement', () => {
|
|
const { code } = generate(createRoot(), { prefixIdentifiers: true })
|
|
expect(code).toMatch(`const _ctx = this\n`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('static text', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.TEXT,
|
|
content: 'hello',
|
|
isEmpty: false,
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`return "hello"`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('interpolation', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [createInterpolation(`hello`, mockLoc)]
|
|
})
|
|
)
|
|
expect(code).toMatch(`return _${TO_STRING}(hello)`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('comment', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.COMMENT,
|
|
content: 'foo',
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`return _${CREATE_VNODE}(_${COMMENT}, 0, "foo")`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('text + comment + interpolation', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.TEXT,
|
|
content: 'foo',
|
|
isEmpty: false,
|
|
loc: mockLoc
|
|
},
|
|
createInterpolation(`hello`, mockLoc),
|
|
{
|
|
type: NodeTypes.COMMENT,
|
|
content: 'foo',
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`
|
|
return [
|
|
"foo",
|
|
_${TO_STRING}(hello),
|
|
_${CREATE_VNODE}(_${COMMENT}, 0, "foo")
|
|
]`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('text + comment + interpolation w/ prefixIdentifiers: true', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.TEXT,
|
|
content: 'foo',
|
|
isEmpty: false,
|
|
loc: mockLoc
|
|
},
|
|
createInterpolation(`hello`, mockLoc),
|
|
{
|
|
type: NodeTypes.COMMENT,
|
|
content: 'foo',
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
}),
|
|
{
|
|
prefixIdentifiers: true
|
|
}
|
|
)
|
|
expect(code).toMatch(`
|
|
return [
|
|
"foo",
|
|
${TO_STRING}(hello),
|
|
${CREATE_VNODE}(${COMMENT}, 0, "foo")
|
|
]`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('compound expression', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
createCompoundExpression(
|
|
[
|
|
`_ctx.`,
|
|
createSimpleExpression(`foo`, false, mockLoc),
|
|
` + `,
|
|
{
|
|
type: NodeTypes.INTERPOLATION,
|
|
loc: mockLoc,
|
|
content: createSimpleExpression(`bar`, false, mockLoc)
|
|
}
|
|
],
|
|
mockLoc
|
|
)
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`return _ctx.foo + _${TO_STRING}(bar)`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('ifNode', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.IF,
|
|
loc: mockLoc,
|
|
branches: [
|
|
{
|
|
type: NodeTypes.IF_BRANCH,
|
|
condition: createSimpleExpression('foo', false, mockLoc),
|
|
loc: mockLoc,
|
|
children: [
|
|
{
|
|
type: NodeTypes.TEXT,
|
|
content: 'foo',
|
|
isEmpty: false,
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
},
|
|
{
|
|
type: NodeTypes.IF_BRANCH,
|
|
condition: createSimpleExpression('a + b', false, mockLoc),
|
|
loc: mockLoc,
|
|
children: [createInterpolation(`bye`, mockLoc)]
|
|
},
|
|
{
|
|
type: NodeTypes.IF_BRANCH,
|
|
condition: undefined,
|
|
loc: mockLoc,
|
|
children: [
|
|
{
|
|
type: NodeTypes.COMMENT,
|
|
content: 'foo',
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`
|
|
return foo
|
|
? "foo"
|
|
: (a + b)
|
|
? _${TO_STRING}(bye)
|
|
: _${CREATE_VNODE}(_${COMMENT}, 0, "foo")`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('ifNode with no v-else', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.IF,
|
|
loc: mockLoc,
|
|
branches: [
|
|
{
|
|
type: NodeTypes.IF_BRANCH,
|
|
condition: createSimpleExpression('foo', false, mockLoc),
|
|
loc: mockLoc,
|
|
children: [
|
|
{
|
|
type: NodeTypes.TEXT,
|
|
content: 'foo',
|
|
isEmpty: false,
|
|
loc: mockLoc
|
|
}
|
|
]
|
|
},
|
|
{
|
|
type: NodeTypes.IF_BRANCH,
|
|
condition: createSimpleExpression('a + b', false, mockLoc),
|
|
loc: mockLoc,
|
|
children: [createInterpolation(`bye`, mockLoc)]
|
|
}
|
|
]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`
|
|
return foo
|
|
? "foo"
|
|
: (a + b)
|
|
? _${TO_STRING}(bye)
|
|
: null`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('forNode', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.FOR,
|
|
loc: mockLoc,
|
|
source: createSimpleExpression(`list`, false, mockLoc),
|
|
valueAlias: createSimpleExpression(`v`, false, mockLoc),
|
|
keyAlias: createSimpleExpression(`k`, false, mockLoc),
|
|
objectIndexAlias: createSimpleExpression(`i`, false, mockLoc),
|
|
children: [createInterpolation(`v`, mockLoc)]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(
|
|
`return _${RENDER_LIST}(list, (v, k, i) => {
|
|
return _${TO_STRING}(v)
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('forNode w/ prefixIdentifiers: true', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.FOR,
|
|
loc: mockLoc,
|
|
source: createSimpleExpression(`list`, false, mockLoc),
|
|
valueAlias: createSimpleExpression(`v`, false, mockLoc),
|
|
keyAlias: createSimpleExpression(`k`, false, mockLoc),
|
|
objectIndexAlias: createSimpleExpression(`i`, false, mockLoc),
|
|
children: [createInterpolation(`v`, mockLoc)]
|
|
}
|
|
]
|
|
}),
|
|
{
|
|
prefixIdentifiers: true
|
|
}
|
|
)
|
|
expect(code).toMatch(
|
|
`return ${RENDER_LIST}(list, (v, k, i) => {
|
|
return ${TO_STRING}(v)
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('forNode w/ skipped value alias', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.FOR,
|
|
loc: mockLoc,
|
|
source: createSimpleExpression(`list`, false, mockLoc),
|
|
valueAlias: undefined,
|
|
keyAlias: createSimpleExpression(`k`, false, mockLoc),
|
|
objectIndexAlias: createSimpleExpression(`i`, false, mockLoc),
|
|
children: [createInterpolation(`v`, mockLoc)]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(
|
|
`return _${RENDER_LIST}(list, (__value, k, i) => {
|
|
return _${TO_STRING}(v)
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('forNode w/ skipped key alias', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.FOR,
|
|
loc: mockLoc,
|
|
source: createSimpleExpression(`list`, false, mockLoc),
|
|
valueAlias: createSimpleExpression(`v`, false, mockLoc),
|
|
keyAlias: undefined,
|
|
objectIndexAlias: createSimpleExpression(`i`, false, mockLoc),
|
|
children: [createInterpolation(`v`, mockLoc)]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(
|
|
`return _${RENDER_LIST}(list, (v, __key, i) => {
|
|
return _${TO_STRING}(v)
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('forNode w/ skipped value and key aliases', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.FOR,
|
|
loc: mockLoc,
|
|
source: createSimpleExpression(`list`, false, mockLoc),
|
|
valueAlias: undefined,
|
|
keyAlias: undefined,
|
|
objectIndexAlias: createSimpleExpression(`i`, false, mockLoc),
|
|
children: [createInterpolation(`v`, mockLoc)]
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(
|
|
`return _${RENDER_LIST}(list, (__value, __key, i) => {
|
|
return _${TO_STRING}(v)
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('SlotFunctionExpression', () => {
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
{
|
|
type: NodeTypes.ELEMENT,
|
|
tagType: ElementTypes.COMPONENT,
|
|
ns: Namespaces.HTML,
|
|
isSelfClosing: false,
|
|
tag: `Comp`,
|
|
loc: mockLoc,
|
|
props: [],
|
|
children: [],
|
|
codegenNode: {
|
|
type: NodeTypes.JS_CALL_EXPRESSION,
|
|
loc: mockLoc,
|
|
callee: `_${CREATE_VNODE}`,
|
|
arguments: [
|
|
`Comp`,
|
|
`0`,
|
|
{
|
|
type: NodeTypes.JS_OBJECT_EXPRESSION,
|
|
loc: mockLoc,
|
|
properties: [
|
|
{
|
|
type: NodeTypes.JS_PROPERTY,
|
|
loc: mockLoc,
|
|
key: {
|
|
type: NodeTypes.SIMPLE_EXPRESSION,
|
|
isStatic: true,
|
|
content: `default`,
|
|
loc: mockLoc
|
|
},
|
|
value: {
|
|
type: NodeTypes.JS_SLOT_FUNCTION,
|
|
loc: mockLoc,
|
|
params: {
|
|
type: NodeTypes.SIMPLE_EXPRESSION,
|
|
isStatic: false,
|
|
content: `{ foo }`,
|
|
loc: mockLoc
|
|
},
|
|
returns: [
|
|
{
|
|
type: NodeTypes.INTERPOLATION,
|
|
loc: mockLoc,
|
|
content: {
|
|
type: NodeTypes.SIMPLE_EXPRESSION,
|
|
isStatic: false,
|
|
content: `foo`,
|
|
loc: mockLoc
|
|
}
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(
|
|
`return _createVNode(Comp, 0, {
|
|
default: ({ foo }) => [
|
|
_toString(foo)
|
|
]
|
|
})`
|
|
)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
|
|
test('callExpression + objectExpression + arrayExpression', () => {
|
|
function createElementWithCodegen(
|
|
args: CallExpression['arguments']
|
|
): ElementNode {
|
|
return {
|
|
type: NodeTypes.ELEMENT,
|
|
loc: mockLoc,
|
|
ns: Namespaces.HTML,
|
|
tag: 'div',
|
|
tagType: ElementTypes.ELEMENT,
|
|
isSelfClosing: false,
|
|
props: [],
|
|
children: [],
|
|
codegenNode: {
|
|
type: NodeTypes.JS_CALL_EXPRESSION,
|
|
loc: mockLoc,
|
|
callee: CREATE_VNODE,
|
|
arguments: args
|
|
}
|
|
}
|
|
}
|
|
|
|
const { code } = generate(
|
|
createRoot({
|
|
children: [
|
|
createElementWithCodegen([
|
|
// string
|
|
`"div"`,
|
|
// ObjectExpression
|
|
createObjectExpression(
|
|
[
|
|
createObjectProperty(
|
|
createSimpleExpression(`id`, true, mockLoc),
|
|
createSimpleExpression(`foo`, true, mockLoc),
|
|
mockLoc
|
|
),
|
|
createObjectProperty(
|
|
createSimpleExpression(`prop`, false, mockLoc),
|
|
createSimpleExpression(`bar`, false, mockLoc),
|
|
mockLoc
|
|
),
|
|
// compound expression as computed key
|
|
createObjectProperty(
|
|
{
|
|
type: NodeTypes.COMPOUND_EXPRESSION,
|
|
loc: mockLoc,
|
|
children: [
|
|
`foo + `,
|
|
createSimpleExpression(`bar`, false, mockLoc)
|
|
]
|
|
},
|
|
createSimpleExpression(`bar`, false, mockLoc),
|
|
mockLoc
|
|
)
|
|
],
|
|
mockLoc
|
|
),
|
|
// ChildNode[]
|
|
[
|
|
createElementWithCodegen([
|
|
`"p"`,
|
|
createObjectExpression(
|
|
[
|
|
createObjectProperty(
|
|
// should quote the key!
|
|
createSimpleExpression(`some-key`, true, mockLoc),
|
|
createSimpleExpression(`foo`, true, mockLoc),
|
|
mockLoc
|
|
)
|
|
],
|
|
mockLoc
|
|
)
|
|
])
|
|
],
|
|
// ArrayExpression
|
|
createArrayExpression(
|
|
[
|
|
'foo',
|
|
{
|
|
type: NodeTypes.JS_CALL_EXPRESSION,
|
|
loc: mockLoc,
|
|
callee: CREATE_VNODE,
|
|
arguments: [`"p"`]
|
|
}
|
|
],
|
|
mockLoc
|
|
)
|
|
])
|
|
]
|
|
})
|
|
)
|
|
expect(code).toMatch(`
|
|
return ${CREATE_VNODE}("div", {
|
|
id: "foo",
|
|
[prop]: bar,
|
|
[foo + bar]: bar
|
|
}, [
|
|
${CREATE_VNODE}("p", { "some-key": "foo" })
|
|
], [
|
|
foo,
|
|
${CREATE_VNODE}("p")
|
|
])`)
|
|
expect(code).toMatchSnapshot()
|
|
})
|
|
})
|