|
|
@@ -1,143 +1,141 @@
|
|
|
import { parse } from 'compiler/parser/index'
|
|
|
import { optimize } from 'compiler/optimizer'
|
|
|
import { generate } from 'compiler/codegen'
|
|
|
-import directives from 'web/compiler/directives/index'
|
|
|
-import { isReservedTag, isUnaryTag, mustUseProp, getTagNamespace } from 'web/util/index'
|
|
|
import { isObject } from 'shared/util'
|
|
|
+import directives from 'web/compiler/directives/index'
|
|
|
+import { isReservedTag } from 'web/util/index'
|
|
|
+import { baseOptions } from 'entries/web-compiler'
|
|
|
|
|
|
-/* eslint-disable quotes */
|
|
|
-describe('codegen', () => {
|
|
|
- const baseOptions = {
|
|
|
- expectHTML: true,
|
|
|
- preserveWhitespace: true,
|
|
|
- directives,
|
|
|
- isReservedTag,
|
|
|
- isUnaryTag,
|
|
|
- mustUseProp,
|
|
|
- getTagNamespace
|
|
|
- }
|
|
|
-
|
|
|
- function assertCodegen (template, generatedCode, ...args) {
|
|
|
- let staticRenderFnCodes = []
|
|
|
- let generateOptions = baseOptions
|
|
|
- let proc = null
|
|
|
- let len = args.length
|
|
|
- while (len--) {
|
|
|
- const arg = args[len]
|
|
|
- if (Array.isArray(arg)) {
|
|
|
- staticRenderFnCodes = arg
|
|
|
- } else if (isObject(arg)) {
|
|
|
- generateOptions = arg
|
|
|
- } else if (typeof arg === 'function') {
|
|
|
- proc = arg
|
|
|
- }
|
|
|
+function assertCodegen (template, generatedCode, ...args) {
|
|
|
+ let staticRenderFnCodes = []
|
|
|
+ let generateOptions = baseOptions
|
|
|
+ let proc = null
|
|
|
+ let len = args.length
|
|
|
+ while (len--) {
|
|
|
+ const arg = args[len]
|
|
|
+ if (Array.isArray(arg)) {
|
|
|
+ staticRenderFnCodes = arg
|
|
|
+ } else if (isObject(arg)) {
|
|
|
+ generateOptions = arg
|
|
|
+ } else if (typeof arg === 'function') {
|
|
|
+ proc = arg
|
|
|
}
|
|
|
-
|
|
|
- const ast = parse(template, baseOptions)
|
|
|
- optimize(ast, baseOptions)
|
|
|
- proc && proc(ast)
|
|
|
- const res = generate(ast, generateOptions)
|
|
|
- expect(res.render).toBe(generatedCode)
|
|
|
- expect(res.staticRenderFns).toEqual(staticRenderFnCodes)
|
|
|
}
|
|
|
+ const ast = parse(template, baseOptions)
|
|
|
+ optimize(ast, baseOptions)
|
|
|
+ proc && proc(ast)
|
|
|
+ const res = generate(ast, generateOptions)
|
|
|
+ expect(res.render).toBe(generatedCode)
|
|
|
+ expect(res.staticRenderFns).toEqual(staticRenderFnCodes)
|
|
|
+}
|
|
|
|
|
|
+/* eslint-disable quotes */
|
|
|
+describe('codegen', () => {
|
|
|
it('generate directive', () => {
|
|
|
assertCodegen(
|
|
|
'<p v-custom1:arg1.modifire="value1" v-custom2><p>',
|
|
|
- `with (this) { return __r__(__s__('p', {directives:[{name:"custom1",value:(value1),arg:"arg1",modifiers:{"modifire":true}},{name:"custom2",arg:"arg1"}]}), undefined)}`
|
|
|
+ `with(this){return _h(_e('p',{directives:[{name:"custom1",value:(value1),arg:"arg1",modifiers:{"modifire":true}},{name:"custom2",arg:"arg1"}]}))}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-pre', () => {
|
|
|
assertCodegen(
|
|
|
'<div v-pre><p>hello world</p></div>',
|
|
|
- 'with (this) { return __m__(0)}',
|
|
|
- [`with(this){return __r__(__s__('div', {pre:true}), [__r__(__s__('p', {}), [__t__("hello world")])])}`]
|
|
|
+ 'with(this){return _m(0)}',
|
|
|
+ [`with(this){return _h(_e('div',{pre:true}),[_h(_e('p'),[_t("hello world")])])}`]
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-for directive', () => {
|
|
|
assertCodegen(
|
|
|
- '<li v-for="item in items" track-by="uid"></li>',
|
|
|
- `with (this) { return (items)&&__renderList__((items), function(item,$index,$key){return __r__(__s__('li', {key:uid}), undefined)})}`
|
|
|
+ '<li v-for="item in items" track-by="item.uid"></li>',
|
|
|
+ `with(this){return (items)&&_l((items),function(item,$index,$key){return _h(_e('li',{key:item.uid}))})}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-if directive', () => {
|
|
|
assertCodegen(
|
|
|
'<p v-if="show">hello</p>',
|
|
|
- `with (this) { return (show) ? __r__(__s__('p', undefined), [__t__("hello")]) : null}`
|
|
|
+ `with(this){return (show)?_h(_e('p'),[_t("hello")]):void 0}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-else directive', () => {
|
|
|
assertCodegen(
|
|
|
'<div><p v-if="show">hello</p><p v-else>world</p></div>',
|
|
|
- `with (this) { return __r__(__s__('div', undefined), [(show) ? __r__(__s__('p', undefined), [__t__("hello")]) : __r__(__s__('p', undefined), [__t__("world")])])}`
|
|
|
+ `with(this){return _h(_e('div'),[(show)?_h(_e('p'),[_t("hello")]):_h(_e('p'),[_t("world")])])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-ref directive', () => {
|
|
|
assertCodegen(
|
|
|
'<p v-ref:component1></p>',
|
|
|
- `with (this) { return __r__(__s__('p', {hook:{"insert":function(n1,n2){__registerRef__("component1", n1.child || n1.elm, false), false},"destroy":function(n1,n2){__registerRef__("component1", n1.child || n1.elm, false, true)}}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('p',{hook:{"insert":function(n1,n2){_r("component1",n1.child||n1.elm,false,false},"destroy":function(n1,n2){_r("component1",n1.child||n1.elm,false,true)}}}))}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-ref directive on v-for', () => {
|
|
|
assertCodegen(
|
|
|
'<ul><li v-for="item in items" v-ref:component1></li></ul>',
|
|
|
- `with (this) { return __r__(__s__('ul', undefined), [(items)&&__renderList__((items), function(item,$index,$key){return __r__(__s__('li', {hook:{"insert":function(n1,n2){__registerRef__("component1", n1.child || n1.elm, true), false},"destroy":function(n1,n2){__registerRef__("component1", n1.child || n1.elm, true, true)}}}), undefined)})])}`
|
|
|
+ `with(this){return _h(_e('ul'),[(items)&&_l((items),function(item,$index,$key){return _h(_e('li',{hook:{"insert":function(n1,n2){_r("component1",n1.child||n1.elm,true,false},"destroy":function(n1,n2){_r("component1",n1.child||n1.elm,true,true)}}}))})])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate template tag', () => {
|
|
|
assertCodegen(
|
|
|
'<template><p>hello world</p></template>',
|
|
|
- `with (this) { return [__r__(__s__('p', undefined), [__t__("hello world")])]}`
|
|
|
+ `with(this){return [_h(_e('p'),[_t("hello world")])]}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate svg tag', () => {
|
|
|
assertCodegen(
|
|
|
'<svg><text>hello world</text></svg>',
|
|
|
- `with (this) { return __r__(__s__('svg', undefined,'svg'), function(){return [__r__(__s__('text', undefined,'svg'), function(){return [__t__("hello world")]})]})}`
|
|
|
+ `with(this){return _h(_e('svg',void 0,'svg'),[_h(_e('text',void 0,'svg'),[_t("hello world")])])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate render tag', () => {
|
|
|
assertCodegen(
|
|
|
'<render :method="onRender" :args="params"></render>',
|
|
|
- `with (this) { return onRender(params,undefined)}`
|
|
|
+ `with(this){return onRender(params)}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
- it('generate render tag that have childs', () => {
|
|
|
+ it('generate render tag that have children', () => {
|
|
|
assertCodegen(
|
|
|
'<render :method="onRender"><p>hello</p></render>',
|
|
|
- `with (this) { return onRender(null,[__m__(0)])}`,
|
|
|
- [`with(this){return __r__(__s__('p', undefined), [__t__("hello")])}`]
|
|
|
+ `with(this){return onRender([_m(0)])}`,
|
|
|
+ [`with(this){return _h(_e('p'),[_t("hello")])}`]
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate single slot', () => {
|
|
|
assertCodegen(
|
|
|
'<slot></slot>',
|
|
|
- `with (this) { return ($slots["default"] || undefined)}`
|
|
|
+ `with(this){return $slots["default"]}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate named slot', () => {
|
|
|
assertCodegen(
|
|
|
'<slot name="one"></slot>',
|
|
|
- `with (this) { return ($slots["one"] || undefined)}`
|
|
|
+ `with(this){return $slots["one"]}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate slot fallback content', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<slot><div>hi</div></slot>',
|
|
|
+ `with(this){return ($slots["default"]||[_m(0)])}`,
|
|
|
+ [`with(this){return _h(_e('div'),[_t("hi")])}`]
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate slot target', () => {
|
|
|
assertCodegen(
|
|
|
'<p slot="one">hello world</p>',
|
|
|
- `with (this) { return __r__(__s__('p', {slot:"one"}), [__t__("hello world")])}`
|
|
|
+ `with(this){return _h(_e('p',{slot:"one"}),[_t("hello world")])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
@@ -145,137 +143,192 @@ describe('codegen', () => {
|
|
|
// static
|
|
|
assertCodegen(
|
|
|
'<p class="class1">hello world</p>',
|
|
|
- 'with (this) { return __m__(0)}',
|
|
|
- [`with(this){return __r__(__s__('p', {staticAttrs:{"class":"class1"}}), [__t__("hello world")])}`]
|
|
|
+ 'with(this){return _m(0)}',
|
|
|
+ [`with(this){return _h(_e('p',{staticClass:"class1"}),[_t("hello world")])}`]
|
|
|
)
|
|
|
// dynamic
|
|
|
assertCodegen(
|
|
|
'<p :class="class1">hello world</p>',
|
|
|
- `with (this) { return __r__(__s__('p', {attrs:{"class":class1}}), [__t__("hello world")])}`
|
|
|
+ `with(this){return _h(_e('p',{class:class1}),[_t("hello world")])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate style binding', () => {
|
|
|
assertCodegen(
|
|
|
'<p :style="error">hello world</p>',
|
|
|
- `with (this) { return __r__(__s__('p', {attrs:{"style":error}}), [__t__("hello world")])}`
|
|
|
+ `with(this){return _h(_e('p',{style:error}),[_t("hello world")])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate transition', () => {
|
|
|
assertCodegen(
|
|
|
'<p transition="expand">hello world</p>',
|
|
|
- 'with (this) { return __m__(0)}',
|
|
|
- [`with(this){return __r__(__s__('p', {staticAttrs:{"transition":"expand"}}), [__t__("hello world")])}`]
|
|
|
+ `with(this){return _h(_e('p',{transition:{definition:("expand"),appear:false}}),[_t("hello world")])}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate dynamic transition with transition on appear', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<p :transition="expand" transition-on-appear>hello world</p>',
|
|
|
+ `with(this){return _h(_e('p',{transition:{definition:(expand),appear:true}}),[_t("hello world")])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate v-show directive', () => {
|
|
|
assertCodegen(
|
|
|
'<p v-show="shown">hello world</p>',
|
|
|
- `with (this) { return __r__(__s__('p', {directives:[{name:"show",value:(shown)}],show:true}), [__t__("hello world")])}`
|
|
|
+ `with(this){return _h(_e('p',{directives:[{name:"show",value:(shown)}],show:true}),[_t("hello world")])}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate props with v-bind directive', () => {
|
|
|
assertCodegen(
|
|
|
'<p :value="msg">',
|
|
|
- `with (this) { return __r__(__s__('p', {props:{"value":msg}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('p',{props:{"value":msg}}))}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate attrs with v-bind directive', () => {
|
|
|
assertCodegen(
|
|
|
'<input :name="field1">',
|
|
|
- `with (this) { return __r__(__s__('input', {attrs:{"name":field1}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{attrs:{"name":field1}}))}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate staticAttrs', () => {
|
|
|
assertCodegen(
|
|
|
'<input name="field1">',
|
|
|
- `with (this) { return __m__(0)}`,
|
|
|
- [`with(this){return __r__(__s__('input', {staticAttrs:{"name":"field1"}}), undefined)}`]
|
|
|
+ `with(this){return _m(0)}`,
|
|
|
+ [`with(this){return _h(_e('input',{staticAttrs:{"name":"field1"}}))}`]
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate events with v-on directive', () => {
|
|
|
assertCodegen(
|
|
|
'<input @input="onInput">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":onInput}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":onInput}}))}`
|
|
|
)
|
|
|
- // keycode
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate events with keycode', () => {
|
|
|
assertCodegen(
|
|
|
'<input @input.enter="onInput">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":function($event){if($event.keyCode!==13)return;onInput($event)}}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){if($event.keyCode!==13)return;onInput($event)}}}))}`
|
|
|
)
|
|
|
- // multiple keycode
|
|
|
+ // multiple keycodes (delete)
|
|
|
assertCodegen(
|
|
|
'<input @input.delete="onInput">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":function($event){if($event.keyCode!==8&&$event.keyCode!==46)return;onInput($event)}}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){if($event.keyCode!==8&&$event.keyCode!==46)return;onInput($event)}}}))}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate events with modifiers', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<input @input.stop="onInput">',
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){$event.stopPropagation();onInput($event)}}}))}`
|
|
|
)
|
|
|
- // inline statement
|
|
|
+ assertCodegen(
|
|
|
+ '<input @input.prevent="onInput">',
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){$event.preventDefault();onInput($event)}}}))}`
|
|
|
+ )
|
|
|
+ assertCodegen(
|
|
|
+ '<input @input.self="onInput">',
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){if($event.target !== $event.currentTarget)return;onInput($event)}}}))}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate events with multiple modifers', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<input @input.stop.prevent.self="onInput">',
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){$event.stopPropagation();$event.preventDefault();if($event.target !== $event.currentTarget)return;onInput($event)}}}))}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate events with capture modifier', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<input @input.capture="onInput">',
|
|
|
+ `with(this){return _h(_e('input',{on:{"!input":function($event){onInput($event)}}}))}`
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate events with inline statement', () => {
|
|
|
assertCodegen(
|
|
|
'<input @input="curent++">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":function($event){curent++}}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function($event){curent++}}}))}`
|
|
|
)
|
|
|
- // unhandled
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate unhandled events', () => {
|
|
|
assertCodegen(
|
|
|
'<input @input="curent++">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":function(){}}}), undefined)}`,
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":function(){}}}))}`,
|
|
|
ast => {
|
|
|
ast.events.input = undefined
|
|
|
}
|
|
|
)
|
|
|
- // multiple handlers
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate multiple event handlers', () => {
|
|
|
assertCodegen(
|
|
|
'<input @input="curent++" @input="onInput">',
|
|
|
- `with (this) { return __r__(__s__('input', {on:{"input":[function($event){curent++},onInput]}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('input',{on:{"input":[function($event){curent++},onInput]}}))}`
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('generate component', () => {
|
|
|
assertCodegen(
|
|
|
- '<my-component name="mycomponent1" :msg="msg" @notify="onNotify"></my-component>',
|
|
|
- `with (this) { return __r__(__s__('my-component', {attrs:{"msg":msg},staticAttrs:{"name":"mycomponent1"},on:{"notify":onNotify}}), undefined)}`
|
|
|
+ '<my-component name="mycomponent1" :msg="msg" @notify="onNotify"><div>hi</div></my-component>',
|
|
|
+ `with(this){return _h(_e('my-component',{attrs:{"msg":msg},staticAttrs:{"name":"mycomponent1"},on:{"notify":onNotify}}),function(){return [_m(0)]})}`,
|
|
|
+ [`with(this){return _h(_e('div'),[_t("hi")])}`]
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate is attribute', () => {
|
|
|
+ assertCodegen(
|
|
|
+ '<div is="component1"></div>',
|
|
|
+ `with(this){return _h(_e("component1",{}))}`
|
|
|
)
|
|
|
- // have "is" attribute
|
|
|
assertCodegen(
|
|
|
- '<my-component is="component1"></my-component>',
|
|
|
- `with (this) { return __r__(__s__("component1", {}), undefined)}`
|
|
|
+ '<div :is="component1"></div>',
|
|
|
+ `with(this){return _h(_e(component1,{}))}`
|
|
|
)
|
|
|
+ })
|
|
|
+
|
|
|
+ it('generate component with inline-template', () => {
|
|
|
// have "inline-template'"
|
|
|
assertCodegen(
|
|
|
'<my-component inline-template><p>hello world</p></my-component>',
|
|
|
- `with (this) { return __r__(__s__('my-component', {inlineTemplate:{render:function(){with (this) { return __m__(0)}},staticRenderFns:[function(){with(this){return __r__(__s__('p', undefined), [__t__("hello world")])}}]}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('my-component',{inlineTemplate:{render:function(){with(this){return _m(0)}},staticRenderFns:[function(){with(this){return _h(_e('p'),[_t("hello world")])}}]}}))}`
|
|
|
)
|
|
|
- // "have inline-template attrs, and not have child
|
|
|
+ // "have inline-template attrs, but not having extactly one child element
|
|
|
assertCodegen(
|
|
|
'<my-component inline-template><hr><hr></my-component>',
|
|
|
- `with (this) { return __r__(__s__('my-component', {inlineTemplate:{render:function(){with (this) { return __m__(0)}},staticRenderFns:[function(){with(this){return __r__(__s__('hr', undefined), undefined)}}]}}), undefined)}`
|
|
|
+ `with(this){return _h(_e('my-component',{inlineTemplate:{render:function(){with(this){return _m(0)}},staticRenderFns:[function(){with(this){return _h(_e('hr'))}}]}}))}`
|
|
|
)
|
|
|
expect('Inline-template components must have exactly one child element.').toHaveBeenWarned()
|
|
|
})
|
|
|
|
|
|
it('not specified ast type', () => {
|
|
|
const res = generate(null, baseOptions)
|
|
|
- expect(res.render).toBe(`with (this) { return __r__(__s__("div"))}`)
|
|
|
+ expect(res.render).toBe(`with(this){return _h(_e("div"))}`)
|
|
|
expect(res.staticRenderFns).toEqual([])
|
|
|
})
|
|
|
|
|
|
it('not specified directives option', () => {
|
|
|
assertCodegen(
|
|
|
'<p v-if="show">hello world</p>',
|
|
|
- `with (this) { return (show) ? __r__(__s__('p', undefined), [__t__("hello world")]) : null}`,
|
|
|
+ `with(this){return (show)?_h(_e('p'),[_t("hello world")]):void 0}`,
|
|
|
{ isReservedTag }
|
|
|
)
|
|
|
})
|
|
|
|
|
|
it('not specified isReservedTag option', () => {
|
|
|
+ // this causes all tags to be treated as components,
|
|
|
+ // thus all children are wrapped in thunks.
|
|
|
assertCodegen(
|
|
|
'<div><p>hello world</p></div>',
|
|
|
- `with (this) { return __m__(0)}`,
|
|
|
- [`with(this){return __r__(__s__('div', undefined), function(){return [__r__(__s__('p', undefined), function(){return [__t__("hello world")]})]})}`],
|
|
|
+ `with(this){return _m(0)}`,
|
|
|
+ [`with(this){return _h(_e('div'),function(){return [_h(_e('p'),function(){return [_t("hello world")]})]})}`],
|
|
|
{ directives }
|
|
|
)
|
|
|
})
|