Vue中的v-model理解

作者: DeeJay_Y | 来源:发表于2017-10-20 23:46 被阅读251次

    官方文档讲的比较模糊,自己通过例子进行理解。

    v-model是语法糖

    首先明确一点,v-model仅仅是语法糖。

            <input
                    type="text"
                    v-model="something">
            <!--等价于-->
            <input
                    type="text"
                    v-bind:value="something"
                    v-on:input="something = $event.target.value">
    

    还要理解$emit的用法:

    vm.$emit(event,[...args])

    触发当前实例上的事件。附加参数都会传给监听器回调。

    v-model在 普通input上

    一定要明确在给 <input /> 元素添加 v-model 属性时,默认会把 value 作为元素的属性,然后把 'input' 事件作为实时传递 value 的触发事件(具体参考语法糖的解释)

    v-model用在组件上

    这里是一个自定义组件currency-input,父组件的 price 的初始值是 100,更改子组件的值能实时更新父组件的 price

    html:

    <div id="demo">
     <currency-input v-model="price"></currentcy-input>
     <span>{{price}}</span>
    </div>
    

    js:

    Vue.component('currency-input', {
     template: `
      <span>
       <input
        ref="input"
        :value="value"
        <!--为什么这里把 'input' 作为触发事件的事件名?`input` 在哪定义的?-->
        @input="$emit('input', $event.target.value)"
       >
      </span>
     `,
     props: ['value'],// 为什么这里要用 value 属性,value在哪里定义的?貌似没找到啊?
    })
    
    var demo = new Vue({
     el: '#demo',
     data: {
      price: 100,
     }
    })
    

    这里的js代码,props中接受了value,但是在组件中<currency-input v-model="price"></currentcy-input>并没有传入value,而且在input中是监听了input事件,但是并没有在父组件中定义input。

    要理解这个,就要回到语法糖的问题上,
    <currency-input v-model="price"></currentcy-input>
    本质上等价于:
    <currency-input :value="price" @input="price = arguments[0]"></currency-input>
    所以可以看到value和input了

    理解v-model在组件中的实现,给组件添加 v-model 属性时,默认会把 value 作为组件的属性,然后把 'input' 值作为给组件绑定事件时的事件名

    代码分析

    所以这里在js中@input="$emit('input', $event.target.value)" 子组件在监听input事件发生的时候(即@input),向父组件传递了input($emit中的input)事件,并且传递了当前子组件的price值。

    而在父组件中,监听了自定义事件input,当自定义事件input触发后,将当前父组件自身的price值改为子组件中$emit上来的值($event.target.value 是作为$emit传递的参数,所以是arguments[0])

    v-model存在的问题

    对于复选框或者单选框的常见组件时,由于v-model默认传的是value,不是checked,触发事件也不是oninput而是onchange

    • 对于单纯的input type='checkbox'
      <input type="checkbox" :checked="status" @change="status = $event.target.checked" />
    • 当用到组件上时:
    <my-checkbox v-model="foo"></my-checkbox>
    
    Vue.component('my-checkbox', {
     tempalte: `<input 
            type="checkbox"
            @change="$emit('input', $event.target.checked)"
            :checked="value"
           />`
     props: ['value'],
    })
    

    这个时候需要Vue的model选项
    model选项可以指定当前的事件类型和传入的props

    所以可以通过这么改进:

    <my-checkbox v-model="foo"></my-checkbox>
    
    Vue.component('my-checkbox', {
     tempalte: `<input 
            type="checkbox"
            <!--这里就不用 input 了,而是 balabala-->
            @change="$emit('balabala', $event.target.checked)"
            :checked="value"
           />`
     props: ['checked'], //这里就不用 value 了,而是 checked
     model: { // model选项来指定
      prop: 'checked',
      event: 'balabala'
     },
    })
    

    相关文章

      网友评论

      • zwithz:解决了我一直以来的疑惑,但是最后的改进却有点看不太懂:dizzy_face:

      本文标题:Vue中的v-model理解

      本文链接:https://www.haomeiwen.com/subject/abybuxtx.html