ElementUI 2x 动态修改级联选择器组件的 options 属性会输出异常?

问题描述

绑定动态 options 属性的级联选择,在切换绑定值时会控制台会输出异常:
[Vue warn]: Error in callback for watcher "options": "TypeError: Cannot read property 'level' of null

问题出现的环境背景及自己尝试过哪些方法

一开始这个报错信息是在弹窗关闭的时候出现的,所以我就简单的认为是关闭弹窗时使用 resetFields() 方法的把表单绑定的 model 重置导致的,以前遇到过使用 resetFields() 重置表单会时表单数据失去响应的情况。
所以修改为了手动置空表单再使用 clearValidate 清理校验结果,但是问题依旧任然会抛出 TypeError: Cannot read property 'level' of null 的错误信息,所以并不是数据重置导致的异常。

又觉得可能是因为 el-caseader 组件的 value 需要是一个数组,重置完表单之后绑定的 value 会变成 undefined 所以修改手动置空表单是声明绑定的属性为 [],并无果。

确认使用 v-model 绑定的变量在弹窗组件关闭后没有被错误赋值。

相关代码

<template>

<div>

<p>选择完两项后改变第一项</p>

<el-form :model="formData">

<el-form-item label="主要类别" prop="mainCategory">

<el-select v-model="formData.mainCategory" placeholder="请选择">

<el-option v-for="item in categoryList" :key="item.value" :label="item.label" :value="item.value"></el-option>

</el-select>

</el-form-item>

<el-form-item label="次要类别" prop="subCategory">

<el-cascader v-model="formData.subCategory" :options="subCategoryList"></el-cascader>

</el-form-item>

</el-form>

</div>

</template>

<script>

export default {

data() {

return {

formData:{

mainCategory: '',

subCategory: [],

},

categoryList: []

};

},

computed: {

subCategoryList() {

const category = this.categoryList.find(item => item.value === this.formData?.mainCategory)

return category?.children || []

}

},

mounted(){

this.categoryList = [{

value: 'zhinan',

label: '指南',

children: [{

value: 'shejiyuanze',

label: '设计原则',

children: [{

value: 'yizhi',

label: '一致'

}, {

value: 'fankui',

label: '反馈'

}, {

value: 'xiaolv',

label: '效率'

}, {

value: 'kekong',

label: '可控'

}]

}, {

value: 'daohang',

label: '导航',

children: [{

value: 'cexiangdaohang',

label: '侧向导航'

}, {

value: 'dingbudaohang',

label: '顶部导航'

}]

}]

}, {

value: 'zujian',

label: '组件',

children: [{

value: 'basic',

label: 'Basic',

children: [{

value: 'layout',

label: 'Layout 布局'

}, {

value: 'color',

label: 'Color 色彩'

}, {

value: 'typography',

label: 'Typography 字体'

}, {

value: 'icon',

label: 'Icon 图标'

}, {

value: 'button',

label: 'Button 按钮'

}]

}, {

value: 'form',

label: 'Form',

children: [{

value: 'radio',

label: 'Radio 单选框'

}, {

value: 'checkbox',

label: 'Checkbox 多选框'

}, {

value: 'input',

label: 'Input 输入框'

}, {

value: 'input-number',

label: 'InputNumber 计数器'

}, {

value: 'select',

label: 'Select 选择器'

}, {

value: 'cascader',

label: 'Cascader 级联选择器'

}, {

value: 'switch',

label: 'Switch 开关'

}, {

value: 'slider',

label: 'Slider 滑块'

}, {

value: 'time-picker',

label: 'TimePicker 时间选择器'

}, {

value: 'date-picker',

label: 'DatePicker 日期选择器'

}, {

value: 'datetime-picker',

label: 'DateTimePicker 日期时间选择器'

}, {

value: 'upload',

label: 'Upload 上传'

}, {

value: 'rate',

label: 'Rate 评分'

}, {

value: 'form',

label: 'Form 表单'

}]

}, {

value: 'data',

label: 'Data',

children: [{

value: 'table',

label: 'Table 表格'

}, {

value: 'tag',

label: 'Tag 标签'

}, {

value: 'progress',

label: 'Progress 进度条'

}, {

value: 'tree',

label: 'Tree 树形控件'

}, {

value: 'pagination',

label: 'Pagination 分页'

}, {

value: 'badge',

label: 'Badge 标记'

}]

}, {

value: 'notice',

label: 'Notice',

children: [{

value: 'alert',

label: 'Alert 警告'

}, {

value: 'loading',

label: 'Loading 加载'

}, {

value: 'message',

label: 'Message 消息提示'

}, {

value: 'message-box',

label: 'MessageBox 弹框'

}, {

value: 'notification',

label: 'Notification 通知'

}]

}, {

value: 'navigation',

label: 'Navigation',

children: [{

value: 'menu',

label: 'NavMenu 导航菜单'

}, {

value: 'tabs',

label: 'Tabs 标签页'

}, {

value: 'breadcrumb',

label: 'Breadcrumb 面包屑'

}, {

value: 'dropdown',

label: 'Dropdown 下拉菜单'

}, {

value: 'steps',

label: 'Steps 步骤条'

}]

}, {

value: 'others',

label: 'Others',

children: [{

value: 'dialog',

label: 'Dialog 对话框'

}, {

value: 'tooltip',

label: 'Tooltip 文字提示'

}, {

value: 'popover',

label: 'Popover 弹出框'

}, {

value: 'card',

label: 'Card 卡片'

}, {

value: 'carousel',

label: 'Carousel 走马灯'

}, {

value: 'collapse',

label: 'Collapse 折叠面板'

}]

}]

}, {

value: 'ziyuan',

label: '资源',

children: [{

value: 'axure',

label: 'Axure Components'

}, {

value: 'sketch',

label: 'Sketch Templates'

}, {

value: 'jiaohu',

label: '组件交互文档'

}]

}]

}

}

</script>

在线复现Demo,选择完两项后改变第一项

你期待的结果是什么?实际看到的错误信息又是什么?

应该怎么修复动态变更级联组件option值使会在控制台输出异常的问题?

本文参与了SegmentFault 思否面试闯关挑战赛,欢迎正在阅读的你也加入。


回答:

el-cascader 有选中项的绑定值 checkedValue 和激活路径 activePath 两个值,所以每次改变绑定的 subCategoryList 之前,重置掉绑定的 subCategory 值即可。

但是项目内使用的是计算属性,并不是通过事件去修改的,所以在 el-select@change 事件中重置掉 subCategory 并不是,除非说吧 subCategoryList 的计算属性改写在 @change 事件中赋值。

比如说:

<template>

<div>

<p>选择完两项后改变第一项</p>

<el-form :model="formData">

<el-form-item label="主要类别" prop="mainCategory">

<el-select v-model="formData.mainCategory" @change="onMainCategoryChange">

<el-option v-for="item in categoryList" :key="item.value" :label="item.label" :value="item.value"></el-option>

</el-select>

</el-form-item>

<el-form-item label="次要类别" prop="subCategory">

<el-cascader v-model="formData.subCategory" :options="subCategoryList"></el-cascader>

</el-form-item>

</el-form>

</div>

</template>

<script>

export default {

data() {

return {

formData:{

mainCategory: '',

subCategory: [],

},

categoryList: [],

subCategoryList: []

};

},

methods:{

onMainCategoryChange(value){

this.formData.subCategory = []

// 需要加上 $nextTick 使复制操作在更新完毕后赋值

this.$nextTick(() => {

const category = this.categoryList.find(item => item.value === value)

this.subCategoryList = category?.children || []

})

}

},

mounted(){

this.categoryList = [{

value: 'zhinan',

label: '指南',

children: [{

value: 'shejiyuanze',

label: '设计原则',

children: [{

value: 'yizhi',

label: '一致'

}, {

value: 'fankui',

label: '反馈'

}, {

value: 'xiaolv',

label: '效率'

}, {

value: 'kekong',

label: '可控'

}]

}, {

value: 'daohang',

label: '导航',

children: [{

value: 'cexiangdaohang',

label: '侧向导航'

}, {

value: 'dingbudaohang',

label: '顶部导航'

}]

}]

}, {

value: 'zujian',

label: '组件',

children: [{

value: 'basic',

label: 'Basic',

children: [{

value: 'layout',

label: 'Layout 布局'

}, {

value: 'color',

label: 'Color 色彩'

}, {

value: 'typography',

label: 'Typography 字体'

}, {

value: 'icon',

label: 'Icon 图标'

}, {

value: 'button',

label: 'Button 按钮'

}]

}, {

value: 'form',

label: 'Form',

children: [{

value: 'radio',

label: 'Radio 单选框'

}, {

value: 'checkbox',

label: 'Checkbox 多选框'

}, {

value: 'input',

label: 'Input 输入框'

}, {

value: 'input-number',

label: 'InputNumber 计数器'

}, {

value: 'select',

label: 'Select 选择器'

}, {

value: 'cascader',

label: 'Cascader 级联选择器'

}, {

value: 'switch',

label: 'Switch 开关'

}, {

value: 'slider',

label: 'Slider 滑块'

}, {

value: 'time-picker',

label: 'TimePicker 时间选择器'

}, {

value: 'date-picker',

label: 'DatePicker 日期选择器'

}, {

value: 'datetime-picker',

label: 'DateTimePicker 日期时间选择器'

}, {

value: 'upload',

label: 'Upload 上传'

}, {

value: 'rate',

label: 'Rate 评分'

}, {

value: 'form',

label: 'Form 表单'

}]

}, {

value: 'data',

label: 'Data',

children: [{

value: 'table',

label: 'Table 表格'

}, {

value: 'tag',

label: 'Tag 标签'

}, {

value: 'progress',

label: 'Progress 进度条'

}, {

value: 'tree',

label: 'Tree 树形控件'

}, {

value: 'pagination',

label: 'Pagination 分页'

}, {

value: 'badge',

label: 'Badge 标记'

}]

}, {

value: 'notice',

label: 'Notice',

children: [{

value: 'alert',

label: 'Alert 警告'

}, {

value: 'loading',

label: 'Loading 加载'

}, {

value: 'message',

label: 'Message 消息提示'

}, {

value: 'message-box',

label: 'MessageBox 弹框'

}, {

value: 'notification',

label: 'Notification 通知'

}]

}, {

value: 'navigation',

label: 'Navigation',

children: [{

value: 'menu',

label: 'NavMenu 导航菜单'

}, {

value: 'tabs',

label: 'Tabs 标签页'

}, {

value: 'breadcrumb',

label: 'Breadcrumb 面包屑'

}, {

value: 'dropdown',

label: 'Dropdown 下拉菜单'

}, {

value: 'steps',

label: 'Steps 步骤条'

}]

}, {

value: 'others',

label: 'Others',

children: [{

value: 'dialog',

label: 'Dialog 对话框'

}, {

value: 'tooltip',

label: 'Tooltip 文字提示'

}, {

value: 'popover',

label: 'Popover 弹出框'

}, {

value: 'card',

label: 'Card 卡片'

}, {

value: 'carousel',

label: 'Carousel 走马灯'

}, {

value: 'collapse',

label: 'Collapse 折叠面板'

}]

}]

}, {

value: 'ziyuan',

label: '资源',

children: [{

value: 'axure',

label: 'Axure Components'

}, {

value: 'sketch',

label: 'Sketch Templates'

}, {

value: 'jiaohu',

label: '组件交互文档'

}]

}]

}

}

</script>

当然如果不想这样做,依旧想使用计算属性的话,可以通过给 el-cascader 组件绑定 key 来让每一次 subCategoryList 改变时渲染一个新的级联选择器组件,这就比较粗暴了,但是也很直接有效。具体代码如下:

<template>

<div>

<p>选择完两项后改变第一项</p>

<el-form :model="formData">

<el-form-item label="主要类别" prop="mainCategory">

<el-select v-model="formData.mainCategory" placeholder="请选择">

<el-option v-for="item in categoryList" :key="item.value" :label="item.label" :value="item.value"></el-option>

</el-select>

</el-form-item>

<el-form-item label="次要类别" prop="subCategory">

<!-- 给 el-cascader 组件增加一个 key -->

<el-cascader :key="formData.mainCategory" v-model="formData.subCategory" :options="subCategoryList"></el-cascader>

</el-form-item>

</el-form>

</div>

</template>

<script>

export default {

data() {

return {

formData:{

mainCategory: '',

subCategory: [],

},

categoryList: []

};

},

computed: {

subCategoryList() {

const category = this.categoryList.find(item => item.value === this.formData?.mainCategory)

return category?.children || []

}

},

mounted(){

this.categoryList = [{

value: 'zhinan',

label: '指南',

children: [{

value: 'shejiyuanze',

label: '设计原则',

children: [{

value: 'yizhi',

label: '一致'

}, {

value: 'fankui',

label: '反馈'

}, {

value: 'xiaolv',

label: '效率'

}, {

value: 'kekong',

label: '可控'

}]

}, {

value: 'daohang',

label: '导航',

children: [{

value: 'cexiangdaohang',

label: '侧向导航'

}, {

value: 'dingbudaohang',

label: '顶部导航'

}]

}]

}, {

value: 'zujian',

label: '组件',

children: [{

value: 'basic',

label: 'Basic',

children: [{

value: 'layout',

label: 'Layout 布局'

}, {

value: 'color',

label: 'Color 色彩'

}, {

value: 'typography',

label: 'Typography 字体'

}, {

value: 'icon',

label: 'Icon 图标'

}, {

value: 'button',

label: 'Button 按钮'

}]

}, {

value: 'form',

label: 'Form',

children: [{

value: 'radio',

label: 'Radio 单选框'

}, {

value: 'checkbox',

label: 'Checkbox 多选框'

}, {

value: 'input',

label: 'Input 输入框'

}, {

value: 'input-number',

label: 'InputNumber 计数器'

}, {

value: 'select',

label: 'Select 选择器'

}, {

value: 'cascader',

label: 'Cascader 级联选择器'

}, {

value: 'switch',

label: 'Switch 开关'

}, {

value: 'slider',

label: 'Slider 滑块'

}, {

value: 'time-picker',

label: 'TimePicker 时间选择器'

}, {

value: 'date-picker',

label: 'DatePicker 日期选择器'

}, {

value: 'datetime-picker',

label: 'DateTimePicker 日期时间选择器'

}, {

value: 'upload',

label: 'Upload 上传'

}, {

value: 'rate',

label: 'Rate 评分'

}, {

value: 'form',

label: 'Form 表单'

}]

}, {

value: 'data',

label: 'Data',

children: [{

value: 'table',

label: 'Table 表格'

}, {

value: 'tag',

label: 'Tag 标签'

}, {

value: 'progress',

label: 'Progress 进度条'

}, {

value: 'tree',

label: 'Tree 树形控件'

}, {

value: 'pagination',

label: 'Pagination 分页'

}, {

value: 'badge',

label: 'Badge 标记'

}]

}, {

value: 'notice',

label: 'Notice',

children: [{

value: 'alert',

label: 'Alert 警告'

}, {

value: 'loading',

label: 'Loading 加载'

}, {

value: 'message',

label: 'Message 消息提示'

}, {

value: 'message-box',

label: 'MessageBox 弹框'

}, {

value: 'notification',

label: 'Notification 通知'

}]

}, {

value: 'navigation',

label: 'Navigation',

children: [{

value: 'menu',

label: 'NavMenu 导航菜单'

}, {

value: 'tabs',

label: 'Tabs 标签页'

}, {

value: 'breadcrumb',

label: 'Breadcrumb 面包屑'

}, {

value: 'dropdown',

label: 'Dropdown 下拉菜单'

}, {

value: 'steps',

label: 'Steps 步骤条'

}]

}, {

value: 'others',

label: 'Others',

children: [{

value: 'dialog',

label: 'Dialog 对话框'

}, {

value: 'tooltip',

label: 'Tooltip 文字提示'

}, {

value: 'popover',

label: 'Popover 弹出框'

}, {

value: 'card',

label: 'Card 卡片'

}, {

value: 'carousel',

label: 'Carousel 走马灯'

}, {

value: 'collapse',

label: 'Collapse 折叠面板'

}]

}]

}, {

value: 'ziyuan',

label: '资源',

children: [{

value: 'axure',

label: 'Axure Components'

}, {

value: 'sketch',

label: 'Sketch Templates'

}, {

value: 'jiaohu',

label: '组件交互文档'

}]

}]

}

}

</script>

本文参与了SegmentFault 思否面试闯关挑战赛,欢迎正在阅读的你也加入。

以上是 ElementUI 2x 动态修改级联选择器组件的 options 属性会输出异常? 的全部内容, 来源链接: utcz.com/p/933764.html

回到顶部