目录
需求分析
功能实现
递归方法
鼠标右键菜单
添加/修改/删除功能
搜索功能
Tab反向定位
拖拽移动
结束
首页 web前端 前端问答 Ant Design创建一个树形组件,实现编辑、搜索和定位功能

Ant Design创建一个树形组件,实现编辑、搜索和定位功能

Jan 13, 2022 pm 06:28 PM
ant design

怎么定制Ant Design树形组件实现编辑、搜索和反向定位功能?下面本篇文章给大家介绍一下创建树形组件,实现这些功能的方法,希望对大家有所帮助!

Ant Design创建一个树形组件,实现编辑、搜索和定位功能

这次在做了一个树形的展示功能,谁知产品意犹未尽,找我谈话:

PD: 什么?只有展开收起功能?这怎么行,咱们最基础的要支持编辑,支持搜索,如果可以的话还可以做个反向定位...

YY: 你咋不早说?需求文档上也没有啊...

PD: 你看谁家文档一次写到位的?哪家的PD不加需求?

YY: 话是这样说,可事情不是这么做的...

PD: 哎呀,别杵着浪费时间了,快去做吧!

YY: ...

以上故事纯属虚构,如有雷同请评论区留言...

树形数据在开发中算是比较常见了,文件夹、组织架构、生物分类、国家地区等等,世间万物的大多数结构都是树形结构。使用树控件可以完整展现其中的层级关系,并具有展开收起选择等交互功能。

需求分析

  • 编辑:添加/修改/删除/移动
  • 搜索功能:名称/创建人/ owner过滤
  • 定位:tab反向定位

项目仓库:https://github.com/speakice/editable-tree

1.png

功能实现

能实现以上功能的方法库和组件有很多种,这里只讲其中一种,都是Ant Design的组件:

  • Tree.DirectoryTree 目录树
  • Dropdown 右键菜单容器
  • Menu 菜单内容
  • Tabs 右侧Tab页
  • Input.Search 搜索框
  • Switch 切换关联状态
  • shortid 生成唯一id
import { Tree, Dropdown, Menu, Tabs, Input, Switch } from 'antd';import shortid from 'shortid';复制代码
登录后复制

递归方法

操作树行数据,最重要的前提是要有一个趁手的递归方法:

/**
 * 如果需要修改tree,action就返回修改后的item, 不修改就不返回
 */export const deepTree = (tree = [], action = () => {}) => {  return tree.map((item) => {    const newItem = action({ ...item }) || item;    if (newItem.children) {
      newItem.children = deepTree(newItem.children, action);
    }    return newItem;
  });
};复制代码
登录后复制

鼠标右键菜单

右键菜单作用在title上,需要把Dropdown写入树形组件的数据源上:

    <DirectoryTree
          style={{ width: 280 }}
          draggable
          onDrop={onDrop}
          defaultExpandAll
          onRightClick={({ node }) => setRightClickKey(node.key)}
          onSelect={onSelect}
          selectedKeys={rightConnect ? [activeTabKey] : selectedKeys}
          onExpand={onExpand}
          treeData={[
            ...deepTree(treeData, (item) => {              return {
                ...item,                titleWord: item.title,                title: (                  <Dropdown                    trigger="contextMenu"                    visible={rightClickKey === item.key}                    onVisibleChange={() => setRightClickKey()}
                    overlayStyle={{ width: 80 }}
                    overlay={menu(item)}
                  >                    <div                      style={                        searchWord && item.title.includes(searchWord)
                          ? { color: &#39;red&#39; }                          : {}
                      }
                    >
                      {item.title}                    </div>                  </Dropdown>
                ),
              };
            }),
          ]}
        />复制代码
登录后复制

关于右键菜单有几点需要补充说明一下:

  • Dropdown 的触发属性需要设置成contextMenu;
  • Dropdown 显示的位置是相对于title而言,需要设置外层容器宽度铺满剩余空间:
.ant-tree-node-content-wrapper {  display: flex;
}.ant-tree-title {  flex: 1;
}复制代码
登录后复制
  • Dropdown 的显示藏是通过右键点击记录的key来判断的;
  • Dropdown 的菜单需要传递当前item;
  const menu = (node) => (    <Menu      onClick={({ key, domEvent }) => {
        domEvent.stopPropagation();
        console.log('menuClick', node, key);
        // 如果要添加操作顶层文件夹,可以直接操作
        switch (key) {
          case 'add':
            setTreeData(
              deepTree(treeData, (item) => {
                if (item.children && item.key === node.key) {
                  return {
                    ...item,
                    children: [
                      ...item.children,
                      {
                        title: 'new add',
                        key: shortid.generate(),
                        isLeaf: true,
                      },
                    ],
                  };
                }
              })
            );
            break;
          case 'delete':
            const outer = treeData.find((item) => item.key === node.key);
            if (outer) {
              setTreeData(treeData.filter((item) => item.key !== node.key));
              return;
            }
            setTreeData(
              deepTree(treeData, (item) => {
                if (item.children) {
                  return {
                    ...item,
                    children: item.children.filter(
                      ({ key }) => key !== node.key
                    ),
                  };
                }
                return item;
              })
            );
            break;
          case 'edit':
            setTreeData(
              deepTree(treeData, (item) => {
                if (item.key === node.key) {
                  console.log('editle', {
                    ...item,
                    title: 'new edit',
                  });
                  return {
                    ...item,
                    title: 'new edit',
                  };
                }
                return item;
              })
            );
            break;
        }
      }}
    >      <Menu.Item key="add">新增</Menu.Item>      <Menu.Item key="delete" danger>
        删除      </Menu.Item>      <Menu.Item key="edit">编辑</Menu.Item>    </Menu>
  );复制代码
登录后复制

添加/修改/删除功能

添加功能默认只能给文件夹添加,通过key值判断添加,这里处理的比较简单,只做核心功能演示,代码见上一小节;

1-2.png

修改功能也做了简单的实例,在正式项目中一般需要弹窗编辑或者在树组件的title中嵌入输入框,可以使用变量记录正在编辑的item, 最后保存通过递归插入到树形数据中:

1-3.png

删除功能做了判断,如果是删除最外层,则直接通过filter过滤,⚠️否则删除功能是通过children来过滤的,这里要特别注意下。

搜索功能

搜索功能是通过titile颜色变红来提示的:

2.png

实现上也只是做了点击搜索之后搜索,没有实时搜索提示,也没有做搜索词区分,这里可以再截取下字符串来实现,可以见官方实例注意这个默认打开父节点的属性autoExpandParent,否则可能要费些功夫向上递归。

3.png

还有一种需求是要过滤数据源,可以对官方实例简单改造后实现;

Tab反向定位

4.png

点击Tree组件item,在右侧添加Tab,或者激活Tab,这可以算是正向定位;那反向定位就是当右侧Tab页切换时左侧Tree组件选中对应item,核心代码也就是指定selectedKeys,相比较而言也不难,难点在默认打开相关父节点,当然前面说过了控制好autoExpandParent这个属性,就好了。

5.png

拖拽移动

拖拽移动一是Tree组件本身支持,二是官方已经给出了拖拽移动实例,我也只是在官方实例稍微做了改造,这里也不多赘述:

6.png

结束

搜索和反向定位的难点其实是在,打开关联文件夹上,不过官方实例中使用了autoExpandParent这个属性,一下子简单了很多。

时候也不早了,今天就到这里了。

更多编程相关知识,请访问:编程视频!!

以上是Ant Design创建一个树形组件,实现编辑、搜索和定位功能的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

React在HTML中的作用:增强用户体验 React在HTML中的作用:增强用户体验 Apr 09, 2025 am 12:11 AM

React通过JSX与HTML结合,提升用户体验。1)JSX嵌入HTML,使开发更直观。2)虚拟DOM机制优化性能,减少DOM操作。3)组件化管理UI,提高可维护性。4)状态管理和事件处理增强交互性。

反应与前端:建立互动体验 反应与前端:建立互动体验 Apr 11, 2025 am 12:02 AM

React是构建交互式前端体验的首选工具。1)React通过组件化和虚拟DOM简化UI开发。2)组件分为函数组件和类组件,函数组件更简洁,类组件提供更多生命周期方法。3)React的工作原理依赖虚拟DOM和调和算法,提高性能。4)状态管理使用useState或this.state,生命周期方法如componentDidMount用于特定逻辑。5)基本用法包括创建组件和管理状态,高级用法涉及自定义钩子和性能优化。6)常见错误包括状态更新不当和性能问题,调试技巧包括使用ReactDevTools和优

VUE 2的反应性系统在数组和对象更改方面有什么局限性? VUE 2的反应性系统在数组和对象更改方面有什么局限性? Mar 25, 2025 pm 02:07 PM

VUE 2的反应性系统在直接阵列索引设置,长度修改和对象属性添加/删除方面挣扎。开发人员可以使用VUE的突变方法和vue.set()来确保反应性。

REACT组件:在HTML中创建可重复使用的元素 REACT组件:在HTML中创建可重复使用的元素 Apr 08, 2025 pm 05:53 PM

React组件可以通过函数或类定义,封装UI逻辑并通过props接受输入数据。1)定义组件:使用函数或类,返回React元素。2)渲染组件:React调用render方法或执行函数组件。3)复用组件:通过props传递数据,构建复杂UI。组件的生命周期方法允许在不同阶段执行逻辑,提升开发效率和代码可维护性。

与React一起使用打字稿有什么好处? 与React一起使用打字稿有什么好处? Mar 27, 2025 pm 05:43 PM

Typescript通过提供类型安全性,提高代码质量并提供更好的IDE支持来增强反应开发,从而降低错误并提高可维护性。

React和前端堆栈:工具和技术 React和前端堆栈:工具和技术 Apr 10, 2025 am 09:34 AM

React是一个用于构建用户界面的JavaScript库,其核心是组件化和状态管理。1)通过组件化和状态管理简化UI开发。2)工作原理包括调和和渲染,优化可通过React.memo和useMemo实现。3)基本用法是创建并渲染组件,高级用法包括使用Hooks和ContextAPI。4)常见错误如状态更新不当,可使用ReactDevTools调试。5)性能优化包括使用React.memo、虚拟化列表和CodeSplitting,保持代码可读性和可维护性是最佳实践。

vue.js中的功能组件是什么?它们什么时候有用? vue.js中的功能组件是什么?它们什么时候有用? Mar 25, 2025 pm 01:54 PM

vue.js中的功能组件无状态,轻量级且缺乏生命周期钩,非常适合呈现纯数据和优化性能。它们通过没有状态或反应性而与状态组件不同,使用渲染函数直接

如何将用户使用者用于复杂状态管理? 如何将用户使用者用于复杂状态管理? Mar 26, 2025 pm 06:29 PM

本文在React中使用UserDucer进行了复杂的状态管理解释,详细介绍了其对Usestate的好处,以及如何将其与副作用的使用效率集成在一起。

See all articles