vue2/test/unit/modules/compiler/optimizer.spec.ts

324 lines
9.7 KiB
TypeScript

import { parse } from 'compiler/parser/index'
import { extend } from 'shared/util'
import { optimize } from 'compiler/optimizer'
import { baseOptions } from 'web/compiler/options'
describe('optimizer', () => {
it('simple', () => {
const ast = parse(
'<h1 id="section1"><span>hello world</span></h1>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(true) // h1
expect(ast.staticRoot).toBe(true)
expect(ast.children[0].static).toBe(true) // span
})
it('simple with comment', () => {
const options = extend(
{
comments: true
},
baseOptions
)
const ast = parse(
'<h1 id="section1"><span>hello world</span><!--comment--></h1>',
options
)
optimize(ast, options)
expect(ast.static).toBe(true) // h1
expect(ast.staticRoot).toBe(true)
expect(ast.children.length).toBe(2)
expect(ast.children[0].static).toBe(true) // span
expect(ast.children[1].static).toBe(true) // comment
})
it('skip simple nodes', () => {
const ast = parse('<h1 id="section1">hello</h1>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(true)
expect(ast.staticRoot).toBe(false) // this is too simple to warrant a static tree
})
it('interpolation', () => {
const ast = parse('<h1>{{msg}}</h1>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false) // h1
expect(ast.children[0].static).toBe(false) // text node with interpolation
})
it('nested elements', () => {
const ast = parse('<ul><li>hello</li><li>world</li></ul>', baseOptions)
optimize(ast, baseOptions)
// ul
expect(ast.static).toBe(true)
expect(ast.staticRoot).toBe(true)
// li
expect(ast.children[0].static).toBe(true) // first
expect(ast.children[1].static).toBe(true) // second
// text node inside li
expect(ast.children[0].children[0].static).toBe(true) // first
expect(ast.children[1].children[0].static).toBe(true) // second
})
it('nested complex elements', () => {
const ast = parse(
'<ul><li>{{msg1}}</li><li>---</li><li>{{msg2}}</li></ul>',
baseOptions
)
optimize(ast, baseOptions)
// ul
expect(ast.static).toBe(false) // ul
// li
expect(ast.children[0].static).toBe(false) // first
expect(ast.children[1].static).toBe(true) // second
expect(ast.children[2].static).toBe(false) // third
// text node inside li
expect(ast.children[0].children[0].static).toBe(false) // first
expect(ast.children[1].children[0].static).toBe(true) // second
expect(ast.children[2].children[0].static).toBe(false) // third
})
it('v-if directive', () => {
const ast = parse(
'<div id="section1" v-if="show"><p><span>hello world</span></p></div>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(true)
})
it('v-else directive', () => {
const ast = parse(
'<div><p v-if="show">hello world</p><div v-else><p><span>foo bar</span></p></div></div>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(false)
expect(ast.children[0].ifConditions[0].block.static).toBe(false)
expect(ast.children[0].ifConditions[1].block.static).toBe(false)
expect(ast.children[0].ifConditions[0].block.children[0].static).toBe(true)
expect(ast.children[0].ifConditions[1].block.children[0].static).toBe(true)
})
it('v-pre directive', () => {
const ast = parse(
'<ul v-pre><li>{{msg}}</li><li>world</li></ul>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(true)
expect(ast.staticRoot).toBe(true)
expect(ast.children[0].static).toBe(true)
expect(ast.children[1].static).toBe(true)
expect(ast.children[0].children[0].static).toBe(true)
expect(ast.children[1].children[0].static).toBe(true)
})
it('v-for directive', () => {
const ast = parse(
'<ul><li v-for="item in items">hello world {{$index}}</li></ul>',
baseOptions
)
optimize(ast, baseOptions)
// ul
expect(ast.static).toBe(false)
// li with v-for
expect(ast.children[0].static).toBe(false)
expect(ast.children[0].children[0].static).toBe(false)
})
it('v-once directive', () => {
const ast = parse('<p v-once>{{msg}}</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false) // p
expect(ast.children[0].static).toBe(false) // text node
})
it('single slot', () => {
const ast = parse('<div><slot>hello</slot></div>', baseOptions)
optimize(ast, baseOptions)
expect(ast.children[0].static).toBe(false) // slot
expect(ast.children[0].children[0].static).toBe(true) // text node
})
it('named slot', () => {
const ast = parse(
'<div><slot name="one">hello world</slot></div>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.children[0].static).toBe(false) // slot
expect(ast.children[0].children[0].static).toBe(true) // text node
})
it('slot target', () => {
const ast = parse('<p slot="one">hello world</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false) // slot
expect(ast.children[0].static).toBe(true) // text node
})
it('component', () => {
const ast = parse('<my-component></my-component>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false) // component
})
it('component for inline-template', () => {
const ast = parse(
'<my-component inline-template><p>hello world</p><p>{{msg}}</p></my-component>',
baseOptions
)
optimize(ast, baseOptions)
// component
expect(ast.static).toBe(false) // component
// p
expect(ast.children[0].static).toBe(true) // first
expect(ast.children[1].static).toBe(false) // second
// text node inside p
expect(ast.children[0].children[0].static).toBe(true) // first
expect(ast.children[1].children[0].static).toBe(false) // second
})
it('class binding', () => {
const ast = parse('<p :class="class1">hello world</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(true)
})
it('style binding', () => {
const ast = parse('<p :style="error">{{msg}}</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(false)
})
it('key', () => {
const ast = parse('<p key="foo">hello world</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(true)
})
it('ref', () => {
const ast = parse('<p ref="foo">hello world</p>', baseOptions)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(true)
})
it('transition', () => {
const ast = parse(
'<p v-if="show" transition="expand">hello world</p>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(true)
})
it('v-bind directive', () => {
const ast = parse(
'<input type="text" name="field1" :value="msg">',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
})
it('v-on directive', () => {
const ast = parse(
'<input type="text" name="field1" :value="msg" @input="onInput">',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
})
it('custom directive', () => {
const ast = parse(
'<form><input type="text" name="field1" :value="msg" v-validate:field1="required"></form>',
baseOptions
)
optimize(ast, baseOptions)
expect(ast.static).toBe(false)
expect(ast.children[0].static).toBe(false)
})
it('not root ast', () => {
const ast = null
optimize(ast, baseOptions)
expect(ast).toBe(null)
})
it('not specified isReservedTag option', () => {
const ast = parse('<h1 id="section1">hello world</h1>', baseOptions)
optimize(ast, {})
expect(ast.static).toBe(false)
})
it('mark static trees inside v-for', () => {
const ast = parse(
`<div><div v-for="i in 10"><p><span>hi</span></p></div></div>`,
baseOptions
)
optimize(ast, baseOptions)
expect(ast.children[0].children[0].staticRoot).toBe(true)
expect(ast.children[0].children[0].staticInFor).toBe(true)
})
it('mark static trees inside v-for with nested v-else and v-once', () => {
const ast = parse(
`
<div v-if="1"></div>
<div v-else-if="2">
<div v-for="i in 10" :key="i">
<div v-if="1">{{ i }}</div>
<div v-else-if="2" v-once>{{ i }}</div>
<div v-else v-once>{{ i }}</div>
</div>
</div>
<div v-else>
<div v-for="i in 10" :key="i">
<div v-if="1">{{ i }}</div>
<div v-else v-once>{{ i }}</div>
</div>
</div>
`,
baseOptions
)
optimize(ast, baseOptions)
expect(
ast.ifConditions[1].block.children[0].children[0].ifConditions[1].block
.staticRoot
).toBe(false)
expect(
ast.ifConditions[1].block.children[0].children[0].ifConditions[1].block
.staticInFor
).toBe(true)
expect(
ast.ifConditions[1].block.children[0].children[0].ifConditions[2].block
.staticRoot
).toBe(false)
expect(
ast.ifConditions[1].block.children[0].children[0].ifConditions[2].block
.staticInFor
).toBe(true)
expect(
ast.ifConditions[2].block.children[0].children[0].ifConditions[1].block
.staticRoot
).toBe(false)
expect(
ast.ifConditions[2].block.children[0].children[0].ifConditions[1].block
.staticInFor
).toBe(true)
})
})