Vue模式是向下的道具和向上的事件。这听起来很简单,但在编写定制组件时很容易忘记。
从Vue 2.2.0开始,您可以使用v-model(带有计算属性)。我发现这种组合可以在组件之间创建一个简单、干净和一致的接口:
传递给组件的任何道具都保持响应性(即,它不是克隆的,也不需要监视函数在检测到更改时更新本地副本)。
更改会自动发出到父节点。
可以与多级组件一起使用。
计算属性允许分别定义setter和getter。这允许Task组件被重写如下:
Vue.component('Task', {
template: '#task-template',
props: ['list'],
model: {
prop: 'list',
event: 'listchange'
},
computed: {
listLocal: {
get: function() {
return this.list
},
set: function(value) {
this.$emit('listchange', value)
}
}
}
})
model属性定义了哪个道具与v-model相关联,以及在发生更改时将触发哪个事件。然后你可以从父组件中调用这个组件,如下所示:
<Task v-model="parentList"></Task>
listLocal计算属性在组件中提供了一个简单的getter和setter接口(可以把它看作一个私有变量)。在# Task -template中,你可以呈现listLocal,它将保持响应性(即,如果parentList发生变化,它将更新Task组件)。你也可以通过调用setter来改变listLocal(例如this。listLocal = newList),它将把更改发送给父类。
这种模式的优点在于,您可以将listLocal传递给Task的子组件(使用v-model),并且来自子组件的更改将传播到顶层组件。
例如,假设我们有一个单独的EditTask组件,用于对任务数据进行某种类型的修改。通过使用相同的v-model和计算属性模式,我们可以将listLocal传递给组件(使用v-model):
<script type="text/x-template" id="task-template">
<div>
<EditTask v-model="listLocal"></EditTask>
</div>
</script>
如果EditTask发出更改,它将适当地调用listLocal上的set(),从而将事件传播到顶层。类似地,EditTask组件也可以使用v-model调用其他子组件(比如表单元素)。