当前位置:   article > 正文

element-ui —Cascader 级联选择器(选中方式处理)

cascader 级联选择器

目前Vue Element的 el-cascader 级联选择器,多选或者选择任意一级,需要点击左侧的checkbox才能选中。

目标:点击label选中,已选中状态再次点击label取消选中

有两种方式实现

  • 通过添加点击事件
  • 通过css样式控制

方法一:通过点击事件

  1. <!-- 改造前 -->
  2. <template>
  3. <div>
  4. <el-cascader
  5. v-model="cascaderValue"
  6. :options="options"
  7. :props="{ multiple: true,checkStrictly:true }"
  8. clearable>
  9. </el-cascader>
  10. {{cascaderValue}}
  11. </div>
  12. </template>

改造点

  • 可以通过 scoped slot 对级联选择器的备选项的节点内容进行自定义
  • 添加点击事件 @click="onClick(node)" 将选中的数据进行处理
  • 区分单选/多选 单选:
  1. 将选中的数据赋值到v-model

多选

  1. 将之前选择的数据保存
  2. 处理:未选勾选选中/已选取消选中
  3. 将选中的数据赋值到v-model
  1. <!-- 改造后 -->
  2. <template>
  3. <div>
  4. <el-cascader
  5. v-model="cascaderValue"
  6. :options="options"
  7. :props="{ multiple: true,checkStrictly:true }"
  8. clearable>
  9. <!-- 重点:第一步,增加slot-scope -->
  10. <template slot-scope="{ node, data }">
  11. <span style="display:block" @click="onClick(node)">{{ data.label }}</span>
  12. </template>
  13. </el-cascader>
  14. {{cascaderValue}}
  15. </div>
  16. </template>
  17. <script>
  18. export default {
  19. name: 'Test',
  20. data () {
  21. return {
  22. cascaderValue: [],
  23. options: [
  24. {
  25. value: 1,
  26. label: '东南',
  27. disabled: true,
  28. children: [
  29. {
  30. value: 2,
  31. label: '上海',
  32. disabled: true,
  33. children: [
  34. { value: 3, label: '普陀' },
  35. { value: 4, label: '黄埔' },
  36. { value: 5, label: '徐汇' }
  37. ]
  38. }
  39. ]
  40. },
  41. {
  42. value: 6,
  43. label: '西北',
  44. disabled: true,
  45. children: [
  46. { value: 7, label: '陕西' },
  47. { value: 8, label: '新疆' }
  48. ]
  49. },
  50. {
  51. value: 9,
  52. label: '东北'
  53. }
  54. ]
  55. }
  56. },
  57. methods: {
  58. // 单选的情况
  59. // onClick (node) {
  60. // if (!node.isDisabled) {
  61. // this.cascaderValue = node.path
  62. // }
  63. // },
  64. // 多选的情况
  65. onClick (node) {
  66. if (node.isDisabled) {
  67. return
  68. }
  69. // 定义一个数组,用于处理数据
  70. const data = []
  71. // 将之前选择的数据保存
  72. if (this.cascaderValue.length) {
  73. this.cascaderValue.forEach(n => {
  74. data.push(n)
  75. console.log('之前的值', data)
  76. })
  77. }
  78. // 处理:
  79. // 未选状态:将当前点击的node.path(是一个数组[1,2,3]),添加到data中
  80. // 已选状态:已选中的从data数组中查找node.path删除
  81. if (!node.checked) {
  82. data.push(node.path)
  83. console.log('未选->选中', data)
  84. } else {
  85. if (data.length) {
  86. // 将[ [ 1, 2, 3 ], [ 1, 2, 4 ], [ 1, 2, 5 ] ]数据转为["1,2,3", "1,2,4", "1,2,5"],因为数组不能比较
  87. const map = data.map(n => n.join(','))
  88. // 找出node.path删除
  89. const idx = map.findIndex(q => q === node.path.join(','))
  90. if (idx > -1) {
  91. data.splice(idx, 1)
  92. }
  93. console.log('已选->取消选中', data)
  94. }
  95. }
  96. // 最后双向绑定v-model的数据
  97. this.cascaderValue = data
  98. }
  99. }
  100. }
  101. </script>

多选存在问题:当使用el-cascader属性 clearable 的时候,会出现删除错误的问题,因为级联选择器绑定的数据是按照指定的格式添加。

因此需要在 “数据赋值到v-model” 以前对数据进行处理

  1. 将之前选择的数据保存
  2. 处理:未选勾选选中/已选取消选中
  3. 将options树形数据转为数组数据
  4. 将满足条件的数据过滤出来
  5. 将选中的数据赋值到v-model
  1. <template>
  2. <div>
  3. <!-- 多选 -->
  4. <el-cascader
  5. v-model="cascaderValue"
  6. :options="options"
  7. :props="{ multiple: true,checkStrictly:true }"
  8. clearable>
  9. <!-- 重点:增加slot-scope -->
  10. <template slot-scope="{ node, data }">
  11. <span style="display:block" @click="onClick(node)">{{ data.label }}</span>
  12. </template>
  13. </el-cascader>
  14. {{cascaderValue}}
  15. </div>
  16. </template>
  17. <script>
  18. export default {
  19. name: 'Test',
  20. data () {
  21. return {
  22. cascaderValue: [],
  23. options: [
  24. {
  25. value: 1,
  26. label: '东南',
  27. disabled: true,
  28. children: [
  29. {
  30. value: 2,
  31. label: '上海',
  32. disabled: true,
  33. children: [
  34. { value: 3, label: '普陀' },
  35. { value: 4, label: '黄埔' },
  36. { value: 5, label: '徐汇' }
  37. ]
  38. }
  39. ]
  40. },
  41. {
  42. value: 6,
  43. label: '西北',
  44. disabled: true,
  45. children: [
  46. { value: 7, label: '陕西' },
  47. { value: 8, label: '新疆' }
  48. ]
  49. },
  50. {
  51. value: 9,
  52. label: '东北'
  53. }
  54. ]
  55. }
  56. },
  57. computed: {
  58. // 1.第一步,将树形数据转数组数据
  59. // ["1,2,3", "1,2,4", "1,2,5", "6,7", "6,8", "9"]
  60. gradeLabelToList () {
  61. return this.treeToList(this.options)
  62. }
  63. },
  64. methods: {
  65. // 单选的情况
  66. // onClick (node) {
  67. // if (!node.isDisabled) {
  68. // this.cascaderValue = node.path
  69. // }
  70. // },
  71. // 多选的情况
  72. onClick (node) {
  73. if (node.isDisabled) {
  74. return
  75. }
  76. // 定义一个数组,用于处理数据
  77. const data = []
  78. // 将之前选择的数据保存
  79. if (this.cascaderValue.length) {
  80. this.cascaderValue.forEach(n => {
  81. data.push(n)
  82. console.log('之前的值', data)
  83. })
  84. }
  85. // 处理:
  86. // 未选状态:将当前点击的node.path(是一个数组[1,2,3]),添加到data中
  87. // 已选状态:已选中的从data数组中查找node.path删除
  88. if (!node.checked) {
  89. data.push(node.path)
  90. console.log('未选->选中', data)
  91. } else {
  92. if (data.length) {
  93. // 将[ [ 1, 2, 3 ], [ 1, 2, 4 ], [ 1, 2, 5 ] ]数据转为["1,2,3", "1,2,4", "1,2,5"],因为数组不能比较
  94. const map = data.map(n => n.join(','))
  95. // 找出node.path删除
  96. const idx = map.findIndex(q => q === node.path.join(','))
  97. if (idx > -1) {
  98. data.splice(idx, 1)
  99. }
  100. console.log('已选->取消选中', data)
  101. }
  102. }
  103. // 如果不添加以下代码,当使用组件的属性 clearable 的时候,会出现删除错误的问题,因为数据是有顺序的
  104. const joinData = data.map(n => n.join(','))
  105. // 将满足条件的数据过滤出来
  106. const tempData = this.gradeLabelToList.filter(n => joinData.find(m => m === n))
  107. let retData = []
  108. // 再将数据数据构造回[ [ 1, 2, 3 ], [ 1, 2, 4 ], [ 1, 2, 5 ] ]
  109. retData = tempData.map(n => n.split(',').map(Number))
  110. // 最后双向绑定v-model的数据
  111. this.cascaderValue = retData
  112. },
  113. treeToList (tree, out, pid) {
  114. if (out === undefined) {
  115. out = []
  116. }
  117. tree.forEach(n => {
  118. const path = pid ? `${pid},` : ''
  119. const data = `${path}${n.value}`
  120. if (n.children) {
  121. this.treeToList(n.children, out, data)
  122. } else {
  123. out.push(data)
  124. }
  125. })
  126. return out
  127. }
  128. }
  129. }
  130. </script>

方法二:通过CSS样式

还有另一种方法,仅根据css样式,扩大radio或者checkbox的区域达到选中效果

存在问题:无法展开下一级

  1. // css样式
  2. <style lang="less">
  3. .el-cascader-panel .el-radio,
  4. .el-checkbox {
  5. width: 100%;
  6. height: 100%;
  7. z-index: 10;
  8. position: absolute;
  9. top: 10px;
  10. right: 10px;
  11. }
  12. .el-cascader-panel .el-radio__input,
  13. .el-checkbox__input {
  14. visibility: hidden;
  15. }
  16. </style>

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/120774
推荐阅读
相关标签
  

闽ICP备14008679号