vue3的v-model

基本用法

v-model可以在组件上实现双向绑定

从 Vue 3.4 开始,推荐的实现方式是使用defineModel()宏:

vue 复制代码
<script setup>
const model = defineModel()
</script>

<template>
  <span>My input</span> <input v-model="model">
</template>

在子组件中使用defineModel声明一个model变量,在模板中使用v-model绑定这个变量

vue 复制代码
// App.vue
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'

const msg = ref('Hello World!')
</script>

<template>
  <h1>{{ msg }}</h1>
  <Child v-model="msg" />
</template>

在父组件声明一个响应变量msg,再用v-model绑定msg,这样msg的值传给了子组件

效果:在子组件中修改model的值,父组件的msg也随之变化。

底层原理

vue 复制代码
<script setup>
const model = defineModel()
</script>

<template>
  <span>My input</span> <input v-model="model">
</template>

可以拆解成由defineProps()和defineEmit()的组合

vue 复制代码
<!-- Child.vue -->
<script setup>
const props = defineProps(['modelValue'])
const emit = defineEmits(['update:modelValue'])
</script>

<template>
  <input
    :value="props.modelValue"
    @input="emit('update:modelValue', $event.target.value)"
  />
</template>
vue 复制代码
// App.vue
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'

const msg = ref('Hello World!')
</script>

<template>
  <h1>{{ msg }}</h1>
  <Child v-model="msg" />
</template>

被拆解成

vue 复制代码
<!-- APP.vue -->
<Child
  :modelValue="msg"
  @update:modelValue="$event => (msg = $event)"
/>

在子组件中用props.modelValue接受父组件传来的msg的值,在模板中用:value进行响应式渲染。

当模板中input的value被修改的时候,使用emit中update:modelValue向父组件传value的值,然后父组件的msg的值被修改成value的值。

v-model的参数

在组件上的v-model可以接受一个参数:

vue 复制代码
// App.vue
<script setup>
import Child from './Child.vue'
import { ref } from 'vue'

const msg = ref('Hello World!')
</script>

<template>
  <h1>{{ msg }}</h1>
  <Child v-model:title="msg" />
</template>

在子组件中,我们可以通过将字符串作为第一个参数传递给 defineModel() 来支持相应的参数。

vue 复制代码
<script setup>
const title = defineModel('title')
</script>

<template>
  <span>My input</span> <input v-model="model">
</template>

如果需要额外的 prop 选项,应该在 model 名称之后传递

js 复制代码
const title = defineModel('title', { required: true })
相关推荐
CRMEB定制开发9 分钟前
CRMEB Pro版前端环境配置指南
前端·微信小程序·uni-app·商城源码·微信商城·crmeb
imber10 分钟前
Framer Motion & GSAP 实现酷炫动画
前端
刘政feng啊16 分钟前
[Vue2]从零实现一个 el-popover 气泡框组件
vue.js
ygming16 分钟前
Q43- code973- 最接近原点的 K 个点 + Q44- code347- 前 K 个高频元素
前端·算法
ygming18 分钟前
Hashmap/ Hashset- Q39~Q42内容
前端·算法
多啦C梦a18 分钟前
【前端必修】闭包、`this`、`箭头函数`、`bind`、节流,一篇文章全懂!
前端·javascript·html
归于尽18 分钟前
为什么别人用闭包那么溜?这 8 个场景照着用就对了
前端·javascript·面试
Hilaku21 分钟前
Vue 2与Vue 3响应式原理的对比与实现
前端·javascript·vue.js
自出洞来无敌手(曾令瑶)26 分钟前
浏览器 实时监听音量 实时语音识别 vue js
前端·javascript·vue.js·语音识别
在钱塘江43 分钟前
《你不知道的JavaScript-上卷》-笔记-5-作用域闭包
前端