import { parse } from 'compiler/parser/index' import { optimize } from 'compiler/optimizer' import { generate } from 'compiler/codegen' import { isObject } from 'shared/util' import { isReservedTag } from 'web/util/index' import { baseOptions } from 'web/compiler/index' 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) } /* eslint-disable quotes */ describe('codegen', () => { it('generate directive', () => { assertCodegen( '

', `with(this){return _h('p',{directives:[{name:"custom1",rawName:"v-custom1:arg1.modifier",value:(value1),expression:"value1",arg:"arg1",modifiers:{"modifier":true}},{name:"custom2",rawName:"v-custom2",arg:"arg1"}]})}` ) }) it('generate filters', () => { assertCodegen( '

{{ d | e | f }}
', `with(this){return _h('div',{attrs:{"id":_f("c")(_f("b")(a))}},[_s(_f("f")(_f("e")(d)))])}` ) }) it('generate v-for directive', () => { assertCodegen( '
  • ', `with(this){return _l((items),function(item){return _h('li',{key:item.uid})})}` ) // iterator syntax assertCodegen( '
  • ', `with(this){return _l((items),function(item,i){return _h('li')})}` ) assertCodegen( '
  • ', `with(this){return _l((items),function(item,key,index){return _h('li')})}` ) }) it('generate v-if directive', () => { assertCodegen( '

    hello

    ', `with(this){return (show)?_h('p',["hello"]):_e()}` ) }) it('generate v-else directive', () => { assertCodegen( '

    hello

    world

    ', `with(this){return _h('div',[(show)?_h('p',["hello"]):_h('p',["world"])])}` ) }) it('generate ref', () => { assertCodegen( '

    ', `with(this){return _h('p',{ref:"component1"})}` ) }) it('generate ref on v-for', () => { assertCodegen( '', `with(this){return _h('ul',[_l((items),function(item){return _h('li',{ref:"component1",refInFor:true})})])}` ) }) it('generate v-bind directive', () => { assertCodegen( '

    ', `with(this){return _h('p',_b({},'p',test))}` ) }) it('generate template tag', () => { assertCodegen( '', `with(this){return [_h('p',[_s(hello)])]}` ) }) it('generate single slot', () => { assertCodegen( '', `with(this){return _t("default")}` ) }) it('generate named slot', () => { assertCodegen( '', `with(this){return _t("one")}` ) }) it('generate slot fallback content', () => { assertCodegen( '
    hi
    ', `with(this){return _t("default",[_h('div',["hi"])])}` ) }) it('generate slot target', () => { assertCodegen( '

    hello world

    ', `with(this){return _h('p',{slot:"one"},["hello world"])}` ) }) it('generate class binding', () => { // static assertCodegen( '

    hello world

    ', `with(this){return _h('p',{staticClass:"class1"},["hello world"])}`, ) // dynamic assertCodegen( '

    hello world

    ', `with(this){return _h('p',{class:class1},["hello world"])}` ) }) it('generate style binding', () => { assertCodegen( '

    hello world

    ', `with(this){return _h('p',{style:(error)},["hello world"])}` ) }) it('generate v-show directive', () => { assertCodegen( '

    hello world

    ', `with(this){return _h('p',{directives:[{name:"show",rawName:"v-show",value:(shown),expression:"shown"}]},["hello world"])}` ) }) it('generate DOM props with v-bind directive', () => { // input + value assertCodegen( '', `with(this){return _h('input',{domProps:{"value":msg}})}` ) // non input assertCodegen( '

    ', `with(this){return _h('p',{attrs:{"value":msg}})}` ) }) it('generate attrs with v-bind directive', () => { assertCodegen( '', `with(this){return _h('input',{attrs:{"name":field1}})}` ) }) it('generate static attrs', () => { assertCodegen( '', `with(this){return _h('input',{attrs:{"name":"field1"}})}` ) }) it('generate events with v-on directive', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":onInput}})}` ) }) it('generate events with keycode', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if($event.keyCode!==13)return;onInput($event)}}})}` ) // multiple keycodes (delete) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if($event.keyCode!==8&&$event.keyCode!==46)return;onInput($event)}}})}` ) // multiple keycodes (chained) assertCodegen( '', `with(this){return _h('input',{on:{"keydown":function($event){if($event.keyCode!==13&&$event.keyCode!==8&&$event.keyCode!==46)return;onInput($event)}}})}` ) // number keycode assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if($event.keyCode!==13)return;onInput($event)}}})}` ) // custom keycode assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if($event.keyCode!==_k("custom"))return;onInput($event)}}})}` ) }) it('generate events with modifiers', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){$event.stopPropagation();onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){$event.preventDefault();onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if($event.target !== $event.currentTarget)return;onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if(!$event.ctrlKey)return;onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if(!$event.shiftKey)return;onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if(!$event.altKey)return;onInput($event)}}})}` ) assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){if(!$event.metaKey)return;onInput($event)}}})}` ) }) it('generate events with multiple modifers', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){$event.stopPropagation();$event.preventDefault();if($event.target !== $event.currentTarget)return;onInput($event)}}})}` ) }) it('generate events with capture modifier', () => { assertCodegen( '', `with(this){return _h('input',{on:{"!input":function($event){onInput($event)}}})}` ) }) it('generate events with inline statement', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":function($event){curent++}}})}` ) }) // #3893 it('should not treat handler with unexpected whitespace as inline statement', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input": onInput }})}` ) }) it('generate unhandled events', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":function(){}}})}`, ast => { ast.events.input = undefined } ) }) it('generate multiple event handlers', () => { assertCodegen( '', `with(this){return _h('input',{on:{"input":[function($event){curent++},onInput]}})}` ) }) it('generate component', () => { assertCodegen( '

    hi
    ', `with(this){return _h('my-component',{attrs:{"name":"mycomponent1","msg":msg},on:{"notify":onNotify}},[_h('div',["hi"])])}` ) }) it('generate svg component with children', () => { assertCodegen( '', `with(this){return _h('svg',[_h('my-comp',[_h('circle',{attrs:{"r":10}})])])}` ) }) it('generate is attribute', () => { assertCodegen( '
    ', `with(this){return _h("component1",{tag:"div"})}` ) assertCodegen( '
    ', `with(this){return _h(component1,{tag:"div"})}` ) }) it('generate component with inline-template', () => { // have "inline-template'" assertCodegen( '

    hello world

    ', `with(this){return _h('my-component',{inlineTemplate:{render:function(){with(this){return _m(0)}},staticRenderFns:[function(){with(this){return _h('p',[_h('span',["hello world"])])}}]}})}` ) // "have inline-template attrs, but not having extactly one child element assertCodegen( '

    ', `with(this){return _h('my-component',{inlineTemplate:{render:function(){with(this){return _h('hr')}},staticRenderFns:[]}})}` ) expect('Inline-template components must have exactly one child element.').toHaveBeenWarned() }) it('generate static trees inside v-for', () => { assertCodegen( `

    `, `with(this){return _h('div',[_l((10),function(i){return _h('div',[_m(0,true)])})])}`, [`with(this){return _h('p',[_h('span')])}`] ) }) it('not specified ast type', () => { const res = generate(null, baseOptions) expect(res.render).toBe(`with(this){return _h("div")}`) expect(res.staticRenderFns).toEqual([]) }) it('not specified directives option', () => { assertCodegen( '

    hello world

    ', `with(this){return (show)?_h('p',["hello world"]):_e()}`, { isReservedTag } ) }) }) /* eslint-enable quotes */