Vue实现树形下拉框

此为自己写的一个完整的前后分离项目: Vue完整前后台项目介绍

Vue自身并没有实现树形下拉框的组件,找了很多资料,最后在Github上找了个插件vue-treeselect,功能还是比较全的,模糊搜索、多选、延迟加载、异步搜索、排序,自定义、Vuex支持等等。这些功能在官网上都有详细的介绍:

vue-treeselect官网: vue-treeselect

vue-treeselect github地址: vue-treeselect

下面只做个简单的功能介绍,模糊搜索与树形结构展示:

当然,首先是下载安装插件:

npm install --save @riophae/vue-treeselect

安装成功之后,就可以进行测试了,下面是我的测试vue:

<template>
  <div>
    <Card>
      <tree-select
        :options="options"
        placeholder="请选择分类..."
        v-model="value"
      />
    </Card>
    <Card>
      {{ value }}
    </Card>
  </div>
</template>

<script>
  import { mapActions } from 'vuex'
  import TreeSelect from '@riophae/vue-treeselect'
  import '@riophae/vue-treeselect/dist/vue-treeselect.css'

  export default {
    name: 'CustomTreeSelect',
    components: { TreeSelect },
    data() {
      return {
        value: 0,
        options: [],
      }
    },
    methods: {
      ...mapActions([
        'albumCategoryList'
      ]),

      queryCategoryList () {
        this.albumCategoryList({
          parentId: 0,
          all: 1
        }).then((res) => {
          console.log(res.fields)
          if (res.status === 1) {
            let result = []
            const children = this.getTree(res.fields)
            let obj = {}
            obj.label = '顶级分类'
            obj.id = 0
            obj.children = children
            result.push(obj)
            this.options = result
          } else {
            this.$Notice.error({
              title: '错误',
              desc: res.msg
            })
          }
        }).catch(error => {
          this.$Notice.error({
            title: '错误',
            desc: '网络连接错误'
          })
          console.log(error)
        })
      },

      getTree (tree = []) {
        let arr = [];
        if (tree.length !== 0) {
          tree.forEach(item => {
            let obj = {};
            obj.label = item.categoryName;
            obj.id = item.id;
            if(item.child === 1) {
              obj.children = this.getTree(item.children);
            }
            arr.push(obj);
          });
        }
        return arr
      },
    },
    mounted () {
      this.queryCategoryList()
    }
  }
</script>

<style scoped>

</style>

这个插件默认的数据结构为:

{
    "id" : "",
    "label" : "",
    "children" : []
}

我们从数据库查出来的数据肯定不是这个样子的啦,所以需要在页面中转换一下数据结构,这个逻辑看一下代码中的函数 getTree()。在提醒一下,如果当前分类下面没有子分类的话,就不需要children这个属性了。

上述运行成功之后,在界面上就可以看到具体的结构了:

默认情况是这样的:

在这里插入图片描述

展开之后是这样的:

在这里插入图片描述

模糊搜索之后是这样的:

在这里插入图片描述

选中某个分类之后,与之绑定的data属性就会得到分类的id值:

在这里插入图片描述

这样一个简单的下拉树结构就出来了,更复杂的功能可以参考官网来写,其中例子都很全面

### Vue 实现树形下拉框的解决方案 在 Vue实现树形下拉框可以通过多种方式完成,以下是几种常见的方法及其特点: #### 方法一:使用 `vue-treeselect` 插件 `vue-treeselect` 是一个功能强大的第三方插件,能够满足大多数场景下的需求。它支持模糊搜索、多选、延迟加载、异步搜索等功能[^1]。该插件易于集成到项目中,并提供了丰富的配置选项来定制化行为。 安装命令如下: ```bash npm install @riophae/vue-treeselect ``` 基本用法示例: ```html <template> <div> <treeselect v-model="value" :options="options" /> </div> </template> <script> import { Treeselect } from '@riophae/vue-treeselect'; import '@riophae/vue-treeselect/dist/vue-treeselect.css'; export default { components: { Treeselect }, data() { return { value: null, options: [ { id: 'a', label: 'A' }, { id: 'b', label: 'B', children: [{ id: 'c', label: 'C' }] } ] }; } }; </script> ``` #### 方法二:基于 Element UI 自定义 TreeSelect 组件 对于基于 Vue 2 和 Element UI 的项目,可以借助 `el-tree` 和 `el-select` 嵌套的方式来自定义一个 TreeSelect 组件[^2]。这种方式无需引入额外依赖,适合已有项目的扩展开发。 代码结构大致如下: ```html <template> <el-popover placement="right-start" trigger="click"> <el-input slot="reference" readonly :value="selectedLabel"></el-input> <el-tree ref="tree" :data="treeData" node-key="id" highlight-current @node-click="handleNodeClick"></el-tree> </el-popover> </template> <script> export default { data() { return { selectedId: '', selectedLabel: '', treeData: [ { id: 1, label: 'Parent Node', children: [{ id: 2, label: 'Child Node' }] } ] }; }, methods: { handleNodeClick(node) { this.selectedId = node.id; this.selectedLabel = node.label; this.$refs.tree.setCurrentKey(this.selectedId); } } }; </script> ``` #### 方法三:结合自定义逻辑封装组件 如果希望更灵活地控制组件的行为,可以选择完全手动封装一个树形下拉框组件。通过监听输入事件并动态渲染树节点列表,可以实现高度可配置化的交互体验[^3]。 以下是一个简单的例子: ```html <template> <el-form-item label="父节点"> <list-box-f> <template #content> <treeselect class="treeSelect-option" v-model="formData.parentCategoryKey" :multiple="false" :normalizer="normalizer" :options="treeOptions" placeholder="请选择" @select="onSelect" /> </template> </list-box-f> </el-form-item> </template> <script> export default { data() { return { formData: { parentCategoryKey: '' }, treeOptions: [], normalizer(node) { return { ...node, label: node.name }; // 调整数据字段名适配 treeselect } }; }, methods: { onSelect(selectedOption) { console.log('Selected:', selectedOption); } } }; </script> ``` --- ### 总结 以上三种方案分别适用于不同场景: - 如果追求快速实现且不介意增加外部依赖,则推荐 **`vue-treeselect`**; - 若项目已采用 Element UI 并需保持一致性,建议尝试 **基于 ElTree/ElSelect 封装**; - 对于复杂业务需求或者特殊样式调整,考虑 **纯手写组件** 来获得最大灵活性。
评论 13
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值