vue.js自定义组件使用v-model

vue.js自定义组件使用v-model

 <div id="app">
  <input v-model="price">
 </div>
new Vue({
  el: '#app',
 data: {
  price: ''
   }
 });

通过该语句实现price变量与输入值双向绑定

  • 实际上v-model只是一个语法糖,真正的实现是这样的
<input
  :value="something"
  @:input="something = $event.target.value">

以上代码分几个步骤:

将输入框的值绑定到price变量上,这个是单向绑定,意味着改变price变量的值可以改变input的value,但是改变value不能改变price
监听input事件(input输入框都有该事件,当输入内容时自动触发该事件),当输入框输入内容就单向改变price的值
这样就实现了双向绑定。

  • 自定义表单输入组件
<div id="app">
    <input-price v-model="price"></input-price>  //组件里面使用v-model
 </div>
 Vue.component('input-price', {
    template: '<input type='text'>'
});
new Vue({
    el: '#app',
    data: {
         price: ''
    }
});

上面的<input-price>是我们自定义的表单输入组件
首先根据我们的v-model语法糖来看
我们的子组件(input-price)的value需要绑定一个从父组件传来的值,通过子组件的props接收
在子组件上有新的输入时需要触发父组件的input事件,并将新的值作为参数传递给父组件

<div id="app"> 
    <!-- <price-input v-model="price"></price-input> -->
     <!-- 手动实现了v-model双向绑定 -->
     <!-- 3、父组件的input事件被触发,将传来的值赋给父组件的变量price -->
     <!-- 4、父组件value的值绑定到price -->
     <price-input :value="price" @input="onInput"></price-input>
     <p>{{price}}</p>
</div>
Vue.component('price-input', {
    // 5、将父组件的value值通过props传递给子组件
    // 1、当有数据输入时触发了该组件的input事件
    template: '<input :value="value" @input="updateVal($event.target.value)" type="text">',
    props: ["value"], 
    methods: {
         updateVal: function(val) {
            // 2、手动触发父组件的input事件并将值传给父组件
            this.$emit('input', val);  
         }
     }
});
var app = new Vue({
     el: '#app',
     data: {
         price: ''
     },
     methods: {
          onInput: function(val) {
               this.price = val;
          }
      }
 });

这里备注了几个步骤:

当有数据输入时触发了该组件的input事件
手动触发父组件的input事件并将值传给父组件
父组件的input事件被触发,将传来的值赋给父组件的变量price,实现输入框value到父元素的price的单向绑定
父组件value的值绑定到price
将父组件的value值通过props传递给子组件,实现了父组件的price到子组件value的单向绑定

小小的总结一下:

v-bind只能实现单向绑定
v-model(v-bind+触发的input事件)实现双向绑定

  • 例子:
     <div id="app">
          <p>我是父亲, 对儿子说: {{sthGiveChild}}</p>
          <model-child v-model='sthGiveChild'></model-child>
     </div>
     <script>
        Vue.component('model-child',{
          props: {
            give: String
          },
          model: {  //model实现数据双向绑定
            prop: 'give',  // 父组件向子组件传递的数据
            event: 'returnBack' //子组件向父组件传递的方法
          },
          methods: {
            returnBackFn() {
              this.$emit('returnBack', '还你200块');
            }
          },
          template: `
          <div>
              <p>我是儿子,父亲对我说: {{give}}</p>
              <button @click='returnBackFn'>点击我</button>
          </div>
          `
        })
         var app = new Vue ({
             el: '#app',
             data() {
               return {
                sthGiveChild:"给你100块"
               }
             }
         })
      </script>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值