Antd - Table 父子表格Checkbox联动

news2025/1/23 23:19:11

Antd - Table 父子表格Checkbox联动

  • 前言
  • 一. 勾选父子组件联动
  • 二. 效果

前言

由于Antd中的父子组件之间,如果有多选功能,那么不会有联动的关系,需要自己实现。

一. 勾选父子组件联动

代码如下:

import React, { useState } from 'react';
import { Table } from 'antd';
const Column = Table.Column;
interface Parent {
    parentId: number;
    name: string;
    orderID: number;
    childList: Children[]
}
interface Children {
    name: string;
    childId: number;
}
const dataSource: Parent[] = [
    { parentId: 1, name: '张三', orderID: 123456, childList: [{ name: '吹风机', childId: 1001, }, { name: '牛奶', childId: 1002, }] },
    { parentId: 2, name: '李四', orderID: 987654, childList: [{ name: '鼠标', childId: 1003, }, { name: '键盘', childId: 1004, }] }
]
const Page = () => {
    const [parentSelectedRowKeys, setParentSelectedRowKeys] = useState<number[]>([])
    const [childSelectedRowKeys, setChildSelectedRowKeys] = useState<number[]>([])


    const onParentSelectChange = (record: Parent, selected: boolean) => {
        // 目前为止选择的父节点和子节点的RowKey
        const preParentRowKeys: number[] = [...(parentSelectedRowKeys || [])];
        let preChildRowKeys: number[] = [...(childSelectedRowKeys || [])];
        // 当前勾选的父节点,其对应的所有子节点的RowKey
        let currentChildRowKeys = dataSource.find(parent => parent.parentId === record.parentId)?.childList.map(item => item.childId) || [];
        // 判断是否选中,选中就加入,否则从老的RowKey中删除它(去重)
        if (selected) {
            preParentRowKeys.push(record.parentId)
            preChildRowKeys = Array.from(new Set([...currentChildRowKeys, ...preChildRowKeys]))
        } else {
            // 否则,父节点取消选中,子节点全部取消
            preParentRowKeys.splice(preParentRowKeys.findIndex(parentRowKey => parentRowKey === record.parentId), 1)
            preChildRowKeys = preChildRowKeys.filter(childRowKey => !currentChildRowKeys.some(rowkey => rowkey === childRowKey))
        }
        // 最后重新设置父,子的SelectedRowKeys
        setParentSelectedRowKeys(preParentRowKeys)
        setChildSelectedRowKeys(preChildRowKeys)
    }
    // 父节点选中全部
    const onParentSelectAll = (selected: true, selectedRows: Parent[], changeRows: Parent[]) => {
        let preParentRowKeys = [...(parentSelectedRowKeys || [])];
        let currentChildRowKeys: number[] = [];
        changeRows.forEach(e => {
            currentChildRowKeys = [...currentChildRowKeys, ...e.childList.map(child => child.childId)]
        });
        // 如果选中,那么所有子节点全部选中
        if (selected) {
            preParentRowKeys = Array.from(new Set([...preParentRowKeys, ...changeRows.map(item => item.parentId)]))
            setChildSelectedRowKeys(currentChildRowKeys)
        } else {
            // 否则所有子节点取消选中
            preParentRowKeys = preParentRowKeys.filter(item => !changeRows.some(e => e.parentId === item))
            setChildSelectedRowKeys([])
        }
        // 设置父节点RowKey
        setParentSelectedRowKeys(preParentRowKeys)
    }

    const parentRowSelection = {
        selectedRowKeys: parentSelectedRowKeys,
        onSelect: onParentSelectChange,
        onSelectAll: onParentSelectAll,
    }

    const onChildSelectChange = (record: Children, selected: true, selectedRows: Children[]) => {
        const preChildRowKeys: number[] = [...(childSelectedRowKeys || [])];
        // 判断当前子节点是 取消勾选/勾选 状态,对应维护
        if (selected) {
            preChildRowKeys.push(record.childId)
        } else {
            preChildRowKeys.splice(preChildRowKeys.findIndex(item => item === record.childId), 1)
        }
        selectedRows = selectedRows.filter(a => a)
        // 判断子节点选中的个数,和对应当前的父节点下的子节点个数是否相等,如果是,那么对应父节点也要勾选上
        for (const item of dataSource) {
            if (item.childList.find(d => d.childId === record.childId)) {
                const preParentRowKeys: number[] = [...(parentSelectedRowKeys || [])];
                if (item.childList.length === selectedRows.length) {
                    preParentRowKeys.push(item.parentId)
                } else {
                    if (preParentRowKeys.find(rowkey => rowkey === item.parentId)) {
                        preParentRowKeys.splice(preParentRowKeys.findIndex(rowkey => rowkey === item.parentId), 1)
                    }
                }
                setParentSelectedRowKeys(preParentRowKeys)
                break;
            }
        }
        setChildSelectedRowKeys(preChildRowKeys)
    }
    const onChildSelectAll = (selected: true, selectedRows: Children[], changeRows: Children[]) => {
        let preChildRowKeys: number[] = [...(childSelectedRowKeys || [])];
        if (selected) {
            preChildRowKeys = Array.from(new Set([...preChildRowKeys, ...changeRows.map(item => item.childId)]))
        } else {
            preChildRowKeys = preChildRowKeys.filter(item => !changeRows.some(child => child.childId === item))
        }
        // 子节点全部选中或者取消全部选中,那么对应父节点的状态也要勾选或者取消勾选
        for (const item of dataSource) {
            if (item.childList.find(d => d.childId === changeRows[0].childId)) {
                const preParentRowKeys: number[] = [...(parentSelectedRowKeys || [])];
                if (selected) {
                    //全选
                    preParentRowKeys.push(item.parentId)
                } else {
                    //取消全选
                    preParentRowKeys.splice(preParentRowKeys.findIndex(rowkey => rowkey === item.parentId), 1)
                }
                setParentSelectedRowKeys(preParentRowKeys)
                break;
            }
        }
        setChildSelectedRowKeys(preChildRowKeys)
    }

    const childRowSelection = {
        selectedRowKeys: childSelectedRowKeys,
        onSelect: onChildSelectChange,
        onSelectAll: onChildSelectAll
    }

    const expandedRowRender = (record: Parent) => {
        const { childList } = record;
        return <Table dataSource={childList} rowKey={'childId'} rowSelection={childRowSelection} pagination={false}>
            <Column key={'childId'} dataIndex={'childId'} title={'childId'} />
            <Column key={'name'} dataIndex={'name'} title={'购买产品'} />
        </Table>
    }

    return <>
        <Table dataSource={dataSource} rowKey={'parentId'} expandable={{ expandedRowRender }} rowSelection={parentRowSelection} pagination={false}>
            <Column key={'parentId'} dataIndex={'parentId'} title={'ID'} />
            <Column key={'name'} dataIndex={'name'} title={'姓名'} />
            <Column key={'orderID'} dataIndex={'orderID'} title={'订单号'} />
        </Table>
    </>
}

export default Page;

注意:

  1. 可以选择Redux去存储父子组件对应的RowKey,否则就要在同一个组件中维护状态。
  2. 建议把类型定义描述好,否则父子组件联动很容易出现问题,不要总是写any,否则很难维护的。

二. 效果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1974892.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

微搭如何设置隐私协议

目录 1 创建数据源2 创建应用3 校验隐私协议是否同意4 实际的效果 小程序里如果涉及收集用户隐私信息的&#xff0c;需要提供隐私协议&#xff0c;微搭中如何实现呢&#xff1f; 1 创建数据源 我们创建两个数据源&#xff0c;一个是个人信息采集表&#xff0c;一个是隐私协议表…

2024最全面软件测试2000+面试题附答案详解,卷起来

进大厂是大部分程序员的梦想&#xff0c;而进大厂的门槛也是比较高的&#xff0c;所以这里整理了一份阿里、美团、滴滴、头条等大厂面试大全&#xff0c;其中概括的知识点有&#xff1a;软件测试基础&#xff0c;MySQL&#xff0c;liunx&#xff0c;接口测试&#xff0c;APP测试…

C ++测试随机数

//输出0-15的数字 #include <iostream> #include <string> using namespace std; //更好的办法是使用数组来存储 方便索引 循环 void test01() {srand(time(0));int t0 0;int t1 0;int t2 0;int t3 0;int t4 0;int t5 0;int t6 0;int t7 0;int t8 0;in…

vs-2015安装教程

双击安装包 2-如图先选自定义&#xff0c;然后选安装路径&#xff08;英文路径&#xff09; 3-安装选项一个就够了&#xff0c;如图 4-点击下一步&#xff0c;之后如下图 5-点击安装 启动&#xff0c;如图则恭喜你成功安装

乐观锁与悲观锁:并发控制

乐观锁与悲观锁&#xff1a;并发控制 1、乐观锁2、悲观锁 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; 乐观锁和悲观锁作为两种关键的同步机制&#xff0c;以截然不同的方式应对数据冲突。 1、乐观锁 定义&#xff1a; 乐观锁认为数据冲…

医院不良事件监测预警上报系统,PHP不良事件管理系统源码

不良事件上报系统&#xff0c;支持医院进行10大类医疗安全&#xff08;不良&#xff09;事件的上报管理&#xff1b;帮助医院管理部门更好把控不良事件的发生趋势&#xff0c;分析医院内部潜在的问题和风险&#xff0c;采取适当的管理措施&#xff0c;有效加强质量控制&#xf…

从零开始的CPP(33)多种终止条件的回溯

leetcode39 给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target &#xff0c;找出 candidates 中可以使数字和为目标数 target 的 所有 不同组合 &#xff0c;并以列表形式返回。你可以按 任意顺序 返回这些组合。 candidates 中的 同一个 数字可以 无限制重复…

【[C++BFS】1765. 地图中的最高点

本文涉及知识点 CBFS算法 LeetCoce1765. 地图中的最高点 给你一个大小为 m x n 的整数矩阵 isWater &#xff0c;它代表了一个由 陆地 和 水域 单元格组成的地图。 如果 isWater[i][j] 0 &#xff0c;格子 (i, j) 是一个 陆地 格子。 如果 isWater[i][j] 1 &#xff0c;格…

halcon深度学习语义分割预处理图片遇到的坑

1.最近使用halcon深度学习语义分割&#xff0c;做缺陷检测。 2.在使用halcon的深度学习标准工具&#xff0c;标注图片 3.标注好图片后&#xff0c;到处预处理&#xff0c;发现报错&#xff0c;[‘Multiple matching segmentation files for image /1.jpg’]意思是:[’ image /…

程序设计基础(c语言)

1、求n个数的平均数 float aver(float b[],int n) {int i;float ave0;for(i0;i<n;i){aveb[i];}ave/n; return ave; } 运行结果如下 2、本题要求实现一个函数&#xff0c;可求两个数的最大公约数&#xff0c;例如&#xff0c;12和8的最大公约数为4&#xff0c;则该函数应该…

打卡第31天------贪心算法

每天抓紧时间刷题,争取尽快上岸,不能再耽误一分一秒了,2024年已经过去大半年了。这个算法编程题是我的痛点。要尽快弥补。 卡尔在讲算法题的时候,思路比较清晰,通俗易懂,以前看见算法题就害怕,因为啥都不会,看懵了,跟了一个月了,每天坚持刷题,偶尔会回顾思路,也会…

计算机网络-PIM协议基础概念

一、PIM基础概念 组播网络回顾&#xff1a; 组播网络从网络结构上大体可以分为三个部分&#xff1a; 源端网络&#xff1a;将组播源产生的组播数据发送至组播网络。 组播转发网络&#xff1a;形成无环的组播转发路径&#xff0c;该转发路径也被称为组播分发树&#xff08;Multi…

Shell编程基本介绍

文章目录 Shell简介Shell环境第一个Shell脚本Shell脚本的运行方法 基础语法Shell变量定义变量使用变量修改变量的值单引号和双引号的区别将命令的结果赋值给变量只读变量删除变量 Shell传递参数特殊字符处理参数说明$* 与 $ 区别 Shell字符串三种形式的区别获取字符串长度Shell…

ajax图书管理项目

bootstrap弹框 不离开当前页面&#xff0c;显示单独内容&#xff0c;让用户操作 功能&#xff1a;不离开当前页面&#xff0c;显示单独内容&#xff0c;供用户操作步骤&#xff1a; 1.引入bootstrap.css和bootstrap.js …

Java关键字及保留字总结

文章目录 Java关键字及保留字总结&#xff08;按首字母字母顺序所排列&#xff09;1.abstract2.boolean3.break4.byte5.case6.catch7.char8.class9.continue10.default11.do12.double13.else14.enum15.extends16.final17.finally18.float19.for20.if21.implements22.import23.i…

delphi 12 学习如何登陆网站下载文件

启动时等待验证码. 输入验证码后,等待处理数据 处理完成后,显示数据 实现原理:利用已有的账号和密码登录后产生的cookie,向服务器请求数据.返回的数据是JSON格式,后期需要自己整理. 注意,请在程序中使用同一个TnetHttpClient控件来完成.因为里面保存了cookie信息 需要了解的知…

C#编写软件发布公告1——客户端

前言 软件或者生活中有时需要将信息同步至电子公告板上&#xff0c;利用C#可以快速实现这一目的&#xff0c;这里以软件公告场景设计&#xff0c;主要是将软件的版本号等相关信息同步至服务器&#xff0c;同步成功后&#xff0c;任务需要查找的人员只要有Web浏览器就可以快速查…

用前所未有的方式体验我们的现代 API 文档

增强您的 API 文档 对于开发人员 为内部和外部消费者创建从 API 定义文件自动生成的精美 API 文档。 只需 4 个简单步骤即可编写 API 文档 API 采用的成功取决于 API 文档的质量。 Baklib 从您的 API 定义文件创建漂亮的&#xff08;完全可定制的&#xff09;API 文档。帮助…

4个自定义倒计时

<!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><title>4个自定义倒计时</title><style>* {margin: 0;padding: 0;box-sizing: border-box;user-select: none;body {background: #0b1b2c;}}hea…

DedeCms 织梦系统 漏洞 上传webshell复现 四种方法 超详细

DedeCMS是织梦团队开发PHP 网站管理系统&#xff0c;它以简单、易用、高效为特色&#xff0c;组建出各种各样各具特色的网站&#xff0c;如地方门户、行业门户、政府及企事业站点等。 目录 方法一 &#xff1a;通过⽂件管理器上传WebShell 方法二&#xff1a;修改模板⽂件拿…