defineModel的行为与文档不一致

先来看看文档对defineModel的描述和使用例

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

function update() {
  model.value++
}
</script>

<template>
  <div>Parent bound v-model is: {{ model }}</div>
  <button @click="update">Increment</button>
</template>
html 复制代码
<Child v-model="countModel" />

不过这里的简单例子中 v-model的类型是数字类型 对它的变更自然也只有直接赋值这一种方式

那么 如果v-model是对象呢?

html 复制代码
<script setup lang="ts">
const countObj = defineModel<{ count: number }>()
</script>
<template>
  <div class="m-4">
    {{ countObj.count }}
    <el-button
      @click="
        () => {
          countObj.count++
        }
      "
    >
      +1
    </el-button>
  </div>
</template>
html 复制代码
<script setup lang="ts">
import { ref } from 'vue'
import Child from './Child.vue'

const countObj = ref({
  count: 1,
})
</script>
<template>
  <Child v-model="countObj" />
</template>

可以正常使用 v-model的value也是很正常的ref

然而如果改一下写法

html 复制代码
  <Child
    :model-value="countObj"
    @update:model-value="
      (val) => {
        debugger
        countObj = val
      }
    "
  />

就会发现@update:model-value根本没调用 这一点和文档是不一致的

如果countObj自身还是shallowRef(虽然一般也不会用) 那defineModel就会完全出错

这里的v-model甚至是一个完全没有响应性的变量 对它的变更不会触发渲染 也不会调用update事件

而如果使用之前的写法model-value + @update 就可以正常使用

html 复制代码
<script setup lang="ts">
const props = defineProps<{ modelValue: { count: number } }>()
const emit = defineEmits<{
  'update:modelValue': [val: { count: number }]
}>()
</script>
<template>
  <div class="m-4">
    {{ modelValue.count }}
    <el-button
      @click="
        () => {
          emit('update:modelValue', { count: modelValue.count + 1 })
        }
      "
    >
      +1
    </el-button>
  </div>
</template>

总结:1.defineModel的行为与文档不一致 2.shallowRef+defineModel会导致完全失效

相关推荐
鱼毓屿御4 分钟前
如何给用户添加权限
前端·javascript·vue.js
Java新手村16 分钟前
基于 Vue 3 + Spring Boot 3 的 AI 面试辅助系统:实时语音识别 + 大模型智能回答
vue.js·人工智能·spring boot
雯0609~22 分钟前
hiprint:实现项目部署与打印3-vue版本-独立出模板设计与模板打印页面
前端·vue.js·arcgis
David凉宸1 小时前
Vue 3 + TS + Vite + Pinia vs Vue 2 + JS + Webpack + Vuex:对比分析
javascript·vue.js·webpack
滕青山1 小时前
Vue项目BMI计算器技术实现
前端·vue.js
boooooooom1 小时前
Pinia必学4大核心API:$patch/$reset/$subscribe/$onAction,用法封神!
javascript·vue.js·面试
wxin_VXbishe1 小时前
C#(asp.net)学员竞赛信息管理系统-计算机毕业设计源码28790
java·vue.js·spring boot·spring·django·c#·php
哈里谢顿2 小时前
Vue 3 入门完全指南:从零构建你的第一个响应式应用
vue.js
三十_A4 小时前
零基础通过 Vue 3 实现前端视频录制 —— 从原理到实战
前端·vue.js·音视频
爱问问题的小李4 小时前
ue 动态 Key 导致组件无限重置与 API 重复提交
前端·javascript·vue.js