用@update:modelValue方法,为啥子组件把数据置空,父组件不同步生效?

//父组件
<template>
  <div>
    <Search v-model="form" @getData="getData" />
    <el-divider />
    <el-form ref="formRef" :model="form" label-width="120px" class='flex'>
      <el-form-item label="项目名称">
        <el-input v-model="form.name" />
      </el-form-item>
      <el-form-item label="申请类型">
        <el-select v-model="form.region" placeholder="请选择">
          <el-option label="北京" value="北京" />
          <el-option label="广州" value="广州" />
        </el-select>
      </el-form-item>
      <el-form-item label="申请时间">
        <el-date-picker
          v-model="form.date"
          type="daterange"
          range-separator="至"
          start-placeholder="开始时间"
          end-placeholder="结束时间"
          :size="size"
        />
      </el-form-item>
      <!-- <el-form-item>
        <el-button @click="resetForm">重置</el-button>
        <el-button type="primary" @click="onSubmit">查询</el-button>
      </el-form-item> -->
  </el-form>
  </div>
</template>
<script lang='ts' setup>
import {defineComponent, reactive, toRefs, ref, onBeforeMount, onMounted } from 'vue';
import Search from './search.vue';

const form = reactive({
  name: '',
  region: '',
  date: '',
})
function getData(){
  console.log('form54',form);
  
}
</script>
<style scoped lang="scss">
.flex{
  display: flex;
  flex-wrap: wrap;
}
</style>
//子组件
<template>
  <el-form ref="formRef" :model="formData" label-width="120px" class='flex'>
    <el-form-item label="项目名称">
      <el-input v-model="formData.name" />
    </el-form-item>
    <el-form-item label="申请类型">
      <el-select v-model="formData.region" placeholder="请选择">
        <el-option label="北京" value="北京" />
        <el-option label="广州" value="广州" />
      </el-select>
    </el-form-item>
    <el-form-item label="申请时间">
      <el-date-picker
        v-model="formData.date"
        type="daterange"
        range-separator="至"
        start-placeholder="开始时间"
        end-placeholder="结束时间"
        :size="size"
      />
    </el-form-item>
    <el-form-item>
      <el-button @click="resetForm">重置</el-button>
      <el-button type="primary" @click="onSubmit">查询</el-button>
    </el-form-item>
  </el-form>
</template>
<script lang="ts" setup>
import { reactive, ref, watch, nextTick } from 'vue'

const props = defineProps({
    modelValue: {
      type: String,
      default: '',
    },
})

const formRef = ref();
const formData = ref<any>(props.modelValue);
const emit = defineEmits(['update:modelValue', 'getData']);
watch(
  () => props.modelValue,
  (d) => {
    formData.value = d
  }
)
watch(
  () => formData.value,
  (data) => {
    console.log('data56',data);
    emit('update:modelValue', data)
  },
  {
    deep: true,
  }
)
// 重置
function resetForm(){
    formData.value = {};
    // formData.value.region = '';
    emit('getData');
}
// 查询
const onSubmit = () => {
  console.log('submit!')
}
</script>
<style scoped lang="scss">
.flex{
  display: flex;
  flex-wrap: wrap;
}
</style>

1691242933539.png

问题:为啥子组件重置把 formData.value 置空后,父组件的 form 不同步生效?

阅读 5k
2 个回答
a = {a:1}
b = a
b = {b:2}

看上去 b 也不应该把 a 改变吧?下面这种应该是可以的

a = {a:1}
b = a
b.b = 2

这是一个典型的错误。

const formData = ref<any>(props.modelValue);

props.modelValue 是一个值,而不是 Ref,失去了响应性
上面这行代码的实际作用是以 setup 时 props.modelValue 的值初始化了一个新 Ref,之后就与 props 无关了。
因而 props 更新时,watch 得不到预期中的结果。

修改方案:

  1. 使用 toRefs 保持响应性

    const { modelValue } = toRefs(props)
  2. watch 整个 props

    watch(props, () => {
        formData.value = props.modelValue
    })

友情提示:
你的代码中其它一些可能有问题的地方

  1. defineComponent 是一个编译期宏,不应当被 import
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题