1. v-model实现自定义组件双向绑定

v-model其实是个语法糖,如果没按照相应的规范定义组件,直接写v-model是不会生效的。再说一遍,类似于v-on:click可以简写成@clickv-model是两个表达式合在一起的简写。记住这个,下面具体说明。

1.1 input双向绑定

子组件MyInput.vue

<template>
<div>输入
<input :value="value" @input="input"/>
</div>
</template> <script>
export default {
name: "MyInput",
props: {
value: {type: [String, Number]}
},
methods: {
input(e) {
this.$emit('input', e.target.value)
}
}
}
</script>

父组件App.vue中使用:

<template>
<div id="app">
<my-input v-model="haha" />
<div>{{haha}}</div>
</div>
</template> <script>
import MyInput from '@/components/MyInput'
export default {
name: 'App',
components: { MyInput },
data() {
return {
haha: '66666',
}
}
}
</script> <style>
#app {
font-family: Avenir, Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: center;
color: #2c3e50;
margin-top: 60px;
}
</style>

<my-input v-model="haha" />是一个语法糖(简写),它相当于:

<my-input
:value="haha"
@input="onInput"
/> onInput(e) {
this.haha = e
}

两者效果一样,v-model减少了很多代码,不用你手动把传过来的值赋值给haha了,少定义了一个方法。作用还是挺明显的。

MyInput组件中的value不可以叫别的名字,$emit中抛出的方法也只能叫input ,否则都不可以用v-model语法糖,用了也没啥效果。

另外,获取 input 输入值的的方式是:$event.target.value,写成方法的话,$event实参省略了。

1.2. checkbox双向绑定

到了checkbox这,又有点不一样,首先,它绑定的值不叫value,触发事件也不叫input,具体看代码。

子组件MyCheckBox代码:

<template>
<div>
<input type="checkbox" :checked="checked" @change="change"/>
</div>
</template> <script>
export default {
name: "MyCheckBox",
model: {
prop: 'checked',
event: 'zhuangbi'
},
props: {
checked: Boolean
},
methods: {
change(e) {
this.$emit('zhuangbi', e.target.checked)
}
}
}
</script>

父组件App.vue中使用:

<template>
<div id="app">
<my-check-box v-model="changeV"></my-check-box>
<div>{{changeV}}</div>
</div>
</template> <script>
import MyCheckBox from '@/components/MyCheckBox'
export default {
name: 'App',
components: { MyCheckBox },
data() {
return {
changeV: true,
}
}
}
</script>

除了绑定的value变成了checked@input变成了@change,传到父组件的参数是e.target.checked

注意: 绝大多数例子中,$emit中抛出的事件都是change,我这里写的是zhuangbi,其实只要和model模块中event属性值一样即可

此外,外面的props也不可少。

2. .sync方式实现双向绑定

如果需要对一个prop进行双向绑定,可以使用.sync语法糖。举一个使用场景的例子:别人封装好的 CheckBox 组件,需要做一些样式修改或者功能组合再使用,这就需要对 v-model 的值再来一次双向绑定。拿上面的 MyCheckBox 来说,<my-check-box v-model="checked"/>,给这个checked传值可以用 props,但想把checked的值传给父组件并赋值给props的值,就有点麻烦,需要定义一个方法,使用$emit,父组件监听事件并作赋值操作。现在用.sync可以一句搞定。

子组件DiyCheckBox代码:

<template>
<div>
<my-check-box v-model="diyCheck" @change="dChange"/>
</div>
</template> <script>
import MyCheckBox from './MyCheckBox'
export default {
name: "DiyCheckBox",
components: {MyCheckBox},
props: {
diyCheck: Boolean,
test: String
},
methods: {
dChange(e) {
this.$emit('update:diyCheck', e)
}
}
}
</script>

父组件App.vue中使用:

<template>
<div id="app">
<diy-check-box :diyCheck.sync="dCheck" />
<div>{{dCheck}}</div>
</div>
</template> <script>
import DiyCheckBox from '@/components/DiyCheckBox' export default {
name: 'App',
components: { DiyCheckBox },
data() {
return {
dCheck: true,
}
}
}
</script>

:diyCheck.sync="dCheck"这句代码相当于:

:diyCheck="dCheck"
@update:diyCheck="dCheck = $event"

语法糖作用很明显,大大简化了代码。

上面代码可以实现想要的功能,只是控制台会有一个警告:

[Vue warn]: Avoid mutating a prop directly since the value will be overwritten whenever the parent component re-renders. Instead, use a data or computed property based on the prop's value. Prop being mutated: "diyCheck"

found in

---> <DiyCheckBox> at src/components/DiyCheckBox.vue
<App>
<Root>

避免直接改变props,使用data or computed代替。所以优化一下,换个写法:

<template>
<div>
<my-check-box v-model="comDiyCheck"/>
</div>
</template> <script>
import MyCheckBox from './MyCheckBox'
export default {
name: "DiyCheckBox",
components: {MyCheckBox},
props: {
diyCheck: Boolean,
test: String
},
computed: {
comDiyCheck: {
get() {
return this.diyCheck
},
set(e) {
this.$emit('update:diyCheck', e)
}
}
}
}
</script>

使用计算属性后,@change事件都可以不要了,get()就是获取props传值,set(e)MyCheckBoxv-model值改变了会触发的方法,这里面做法是直接把改变后的值通过$emit方式发出去。父组件中仍然通过.sync绑定,代码没有变化。

.sync可不光能用来做checkbox的双向绑定,涉及到props双向绑定的场景都可以用sync实现。

.sync传整个对象

如果有许多props属性需要做双向绑定操作,标签写起来就很长,像这样:

<coverage-charge
v-for="(item, index) in chargingPiles"
:key="index + 'index'"
:code.sync="item.code"
:address.sync="item.address"
:addressType.sync="item.addressType"
:kind.sync="item.kind"
:yearLimitType.sync="item.yearLimitType"
>
</coverage-charge>

官方文档说可以简写成这样:

<text-document v-bind.sync="doc"></text-document>
<!--对应我们的例子就是:-->
<coverage-charge
v-for="(item, index) in chargingPiles"
:key="index + 'index'"
v-bind.sync='item'
>
</coverage-charge>

官方还说:

这样会把 doc 对象中的每一个 property (如 title) 都作为一个独立的 prop 传进去,然后各自添加用于更新的 v-on 监听器。

按照这个说法,item 的 5 个属性,都会通过 prop 传到子组件,我再在子组件中添加不同的computed属性即可:

<script>
export default {
name: 'CoverageCharge',
props: {},
computed: {
code: {
get() {
return this.code
},
set(val) {
this.$emit('update:code', val)
}
},
address: {
get() {
return this.address
},
set(val) {
this.$emit('update:address', val)
}
}
... // 其他属性值
}
}
</script>

真这样写,会发现,this.codethis.address这些都是undefined,这里需要在子组件的props中再定义一遍属性名才行,可以省略各属性的typedefault值:

 props: [ 'code', 'address', 'addressType', 'kind', 'yearLimitType' ]

这样就可以了,.sync功能强大,用的地方也挺多的。

props里面东西太多的话,也可以统一定义成一个对象,然后父组件通过v-bind或者v-model传进来:

2.1.1 v-bind 方式

<!--子组件-->
props: {
zb: {
type: Object,
default: () => {}
}
},
computed: {
code: {
get() {
return this.zb.code
},
set(val) {
this.$emit('update:code', val)
}
},
address: {
get() {
return this.zb.address
},
set(val) {
this.$emit('update:address', val)
}
}
}
<!--父组件-->
<coverage-charge
v-for="(item, index) in chargingPiles"
:key="index + 'index'"
v-bind.sync='item'
:zb='item'
>
</coverage-charge>

2.2.2 v-model方式

<!--子组件-->
export default {
model: {
prop: 'zb',
event: 'update'
},
props: {
zb: {
type: Object,
default: () => {}
}
},
computed: {
code: {
get() {
return this.zb.code
},
set(val) {
this.$emit('update:code', val)
}
},
address: {
get() {
return this.zb.address
},
set(val) {
this.$emit('update:address', val)
}
}
}
}
<!--父组件-->
<coverage-charge
v-for="(item, index) in chargingPiles"
:key="index + 'index'"
v-bind.sync='item'
v-model='item'
>
</coverage-charge>

注意modelevent值是update

3. v-model.sync比较

上面第 2 节第一个例子目的就是把v-model再包一层,照着第 1 节,用v-model的方法也能做到。这里只写主要点,不贴源码了:

  • 定义model模块:model: {prop: 'checked', event: 'zhuangbi' }
  • props中定义checked属性
  • computed中定义MyChecked: { get(){ return this.checked},set(val) { this.$emit('zhuangbi', val) } }
  • DiyCheckBox组件中使用:<my-check-box v-model="MyChecked"/>
  • 父组件中使用:<diy-check-box v-model=suibian" />suibian是父组件data中定义的变量

可见,v-model适用于双向绑定单个props(一个标签中不能有多个v-model),.sync适用于双向绑定一到多个props(一个标签中允许使用多个:xxx.sync=yyy)。

源码

点击查看源码

vue 双向绑定(v-model 双向绑定、.sync 双向绑定、.sync 传对象)的更多相关文章

  1. vue.js初级入门之最基础的双向绑定操作

    首先在页面引入vue.js以及其他需要用到的或者可能要用到的插件(这里我多引用了bootstrap和jquery) 引用的时候需要注意文件的路径,准备工作这样基本就完成了,下面正式开始入门. vue. ...

  2. 【vue】跟着老马学习vue-数据双向绑定

    学习了node.js教程,只能说是有了一定的了解,之前也了解了webpack和es6的核心内容,也看过vue2.0的官网教程,并结合视频看过项目,但是理解和运用仍然存在很多问题,接下来的一段时间,跟着 ...

  3. Vue.js之Vue计算属性、侦听器、样式绑定

    前言 上一篇介绍了Vue的基本概念,这一篇介绍一下Vue的基本使用. 一.搭建一个Vue程序 1.1 搭建Vue环境 搭建Vue的开发环境总共有三种方法: 引入CDN <script src=& ...

  4. Vue.js的类Class 与属性 Style如何绑定

    Vue.js的类Class 与属性 Style如何绑定 一.总结 一句话总结:数据绑定一个常见需求是操作元素的 class 列表和它的内联样式.因为它们都是属性,我们可以用 v-bind 处理它们:我 ...

  5. 理解ASP.NET Core - 模型绑定&验证(Model Binding and Validation)

    注:本文隶属于<理解ASP.NET Core>系列文章,请查看置顶博客或点击此处查看全文目录 模型绑定 什么是模型绑定?简单说就是将HTTP请求参数绑定到程序方法入参上,该变量可以是简单类 ...

  6. vue中如何在自定义组件上使用v-model和.sync

    自定义事件 tips 推荐始终使用 kebab-case 的事件名.(v-on会将事件名自动转换为小写,避免匹配不到) changeData × change-data √ 自定义组件的v-model ...

  7. v-bind绑定属性样式——class的三种绑定方式

    1.布尔值的绑定方式 <div id="demo"> <span v-bind:class="{‘class-a‘:isA ,‘class-b‘:isB ...

  8. input绑定datapicker控件后input再绑定blur或者mouseout等问题

    input绑定datapicker控件后input再绑定blur或者mouseout等问题 问题描述:今天在修改一个东西的时候需要给一个input输入域绑定blur事件,从而当它失去焦点后动态修改其中 ...

  9. python tips:类的绑定方法(bound)和非绑定方法(unbound)

    类属性只有类及其实例能够访问,可以理解为一个独立的命名空间. Python中类属性的引用方式有两种: 1. 通过类的实例进行属性引用,称为绑定方法(bound method),可以理解为方法与实例绑定 ...

随机推荐

  1. DC-1 靶机渗透测试

    DC-1靶机渗透测试 对着镜子大喊三声"太菜了""太菜了""太菜了" DC系列靶机的第一篇.边学习边日靶机边进步,摸爬滚打着前行. 内容不只 ...

  2. Tensor:Pytorch神经网络界的Numpy

    摘要:Tensor,它可以是0维.一维以及多维的数组,你可以将它看作为神经网络界的Numpy,它与Numpy相似,二者可以共享内存,且之间的转换非常方便. 本文分享自华为云社区<Tensor:P ...

  3. 使用scrapy搭建大型爬虫系统

    最近新项目准备启动,在开始前内容.词库这些都需要提前做好准备,所以就有了这篇文章.在开始动手,看了下行业核心词排在首页的站,发现内容都多得不要不要的,各种乱七八糟的频道.页面模板,心想,如果每个网站. ...

  4. 小白学习Vue第五天(v-model实用的地方)

    用法一radio单选项 <!-- 添加name男女选项互斥 --> <label for="male"> <input type="radi ...

  5. HCIA-数据链路层

    数据链路层 1.数据的差错检测 |FCS| 2.组帧|解帧 |数据帧帧头 帧尾| 3.标识身份 |MAC地址| 以太网络标准数据链路层的标准 数据链路层不仅仅只有以太网 地域来进行分类 局域网:小型地 ...

  6. C++ 封装类 2 设计一个学生类 属性有姓名学号 可以给姓名 和学号赋值 可以显示学生的姓名和学号

    1 //设计一个学生类 属性有姓名学号 可以给姓名 和学号赋值 可以显示学生的姓名和学号 2 #include <iostream> 3 #include<string> 4 ...

  7. Docker 实践及命令梳理

    文档 Docker Reference Documentation Docker 从入门到实践 [中文] 安装 安装 Docker,设置开机启动,然后配置阿里云镜像加速 1. 安装 Docker Do ...

  8. 响应式编程基础教程:Spring Boot 与 Lettuce 整合

    本文主要介绍响应式编程访问 Redis,以及 Spring Boot 与 Lettuce 的整合使用. Lettuce 是可扩展性线程安全的 Redis 客户端,用于同步.异步和响应式使用.如果多个线 ...

  9. Check Directory Existence in Shell

    The following command in one line can check if a directory exists. You can check the return value (& ...

  10. linux 源码搭建Kafka集群,100%有效

    kafka源码编译安装 准备三台服务器 192.168.xxx.xxx 192.168.xxx.xxx 192.168.xxx.xxx 安装kafka前需先安装JDK和zookeeper如下步骤: J ...