当前位置: 首页 > article >正文

React 如何封装一个可复用的 Ant Design 组件

文章目录

  • 前言
    • 一、为什么需要封装组件?
    • 二、 仿antd组件的Button按钮
    • 三、封装一个可复用的表格组件 (实战)
      • 1. 明确需求
      • 2. 设计组件 API
      • 3. 实现组件代码
      • 4. 使用组件
    • 三、封装组件的最佳实践
    • 四、进阶优化
  • 总结


前言

作为一名前端开发工程师,在日常项目中,我们经常会使用 UI 框架(如 Ant Design)来快速搭建界面。然而,直接使用框架组件可能会导致代码重复、样式不统一或功能扩展困难。本文将以封装一个 可复用的表格组件 为例,分享如何基于 Ant Design(antd)封装自己的业务组件,提升开发效率和代码质量。


一、为什么需要封装组件?

  1. 代码复用:避免重复编写相同的逻辑和样式。
  2. 统一风格:确保项目中的组件风格一致。
  3. 功能扩展:在基础组件上添加业务逻辑(如分页、搜索、权限控制等)。
  4. 降低维护成本:修改一处即可影响所有使用该组件的地方。

二、 仿antd组件的Button按钮

虽然有些人要问,antd组件库有现成的Button为啥不用,还要自己封装,难道你封装的比别人好?在此说明,本篇文章只是讲解封装思想,别人怎么封装的。

  1. 首先新建一个专门存放组件的文件夹—我的就叫做XLButton了
    index.tsx – 存放模板
    index.scss存放样式

在这里插入图片描述

  1. 确定模板和样式
import "./index.scss";interface XLButtonProps {type?: "primary" | "default" | "danger";size?: "small" | "middle" | "large";children?: React.ReactNode;
}const XLButton: React.FC<XLButtonProps> = (props) => {const typeStyle =props.type === "primary"? "xl-button-primary": props.type === "danger"? "xl-button-danger": "xl-button-default";const sizeStyle =props.size === "small"? "xl-button-small": props.size === "middle"? "xl-button-middle": "xl-button-large";return (<button className={`${typeStyle} ${sizeStyle}`}>{props.children || "Default Button"}</button>);
};export default XLButton;
// index.scss
button {color: #fff;width: 123px;height: 36px;border-radius: 6px;outline: none;border: none;margin-right: 20px;
}.xl-button-primary,
.xl-button-danger,
.xl-button-default {padding: 8px 16px;border: none;border-radius: 4px;cursor: pointer;transition: all 0.3s ease;
}// 尺寸样式
.xl-button-small {width: 60px;height: 30px;padding: 0 8px;font-size: 12px;
}.xl-button-middle {height: 32px;padding: 4px 16px;font-size: 14px;
}.xl-button-large {height: 40px;padding: 8px 24px;font-size: 16px;
}.xl-button-primary {background-color: #1890ff;color: white;
}.xl-button-danger {background-color: #ff4d4f;color: white;
}.xl-button-default {background-color: #fff;border: 1px solid #d9d9d9;color: #333;
}.xl-button-default:hover {border-color: #1890ff !important;color: #1890ff;
}.xl-button-danger:hover,
.xl-button-primary:hover {opacity: 0.8;box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
}
  1. 使用 -引入组件
import XLButton from "../Components/XLButton";
const Home = () => {return (<div><h1>仿antd按钮</h1><XLButton type="danger">我是danger</XLButton ><XLButton type="primary">我是primary</XLButton ><XLButton >我是默认</XLButton ></div>);
};export default Home;

在这里插入图片描述

三、封装一个可复用的表格组件 (实战)

有了上面的思想,现在让我们来封装一个自己的表格组件。进一步强化

1. 明确需求

假设我们需要封装一个支持以下功能的表格组件:

  • 分页(pagination)
  • 搜索(search)
  • 列排序(sortable columns)
  • 自定义列渲染
  • 加载状态(loading)
  • 空数据提示(empty text)

2. 设计组件 API

首先,设计组件的 props 接口,明确外部传入的参数:

	import React from 'react';import { Table, Input, Button, Space } from 'antd';import type { TableProps, ColumnsType } from 'antd';interface EnhancedTableProps<RecordType extends object = any> {columns: ColumnsType<RecordType>; // 表格列配置dataSource: RecordType[]; // 表格数据loading?: boolean; // 加载状态emptyText?: React.ReactNode; // 空数据提示pagination?: TableProps<RecordType>['pagination']; // 分页配置onSearch?: (value: string) => void; // 搜索回调onSort?: (field: string, order: 'ascend' | 'descend') => void; // 排序回调}

3. 实现组件代码

以下是完整的组件实现代码:

import React, { useState } from "react";
import { Table, Input, Space } from "antd";
import type { TableProps } from "antd";
import type { ColumnsType } from "antd/es/table";
import { SearchOutlined } from "@ant-design/icons";interface EnhancedTableProps<RecordType> {columns: ColumnsType<RecordType>;dataSource: RecordType[];loading?: boolean;emptyText?: React.ReactNode;pagination?: TableProps<RecordType>["pagination"];onSearch?: (value: string) => void;onSort?: (field: string, order: "ascend" | "descend") => void;
}const MyTable = <RecordType extends object>({columns,dataSource,loading = false,emptyText = "暂无数据",pagination,onSearch,onSort,
}: EnhancedTableProps<RecordType>) => {const [searchValue, setSearchValue] = useState("");// 处理搜索const handleSearch = () => {if (onSearch) {onSearch(searchValue.trim());}};// 处理排序const handleChange: TableProps<RecordType>["onChange"] = (pagination,filters,sorter) => {if (Array.isArray(sorter)) {// 处理多列排序的情况const firstSorter = sorter[0];if (firstSorter?.field && onSort) {onSort(firstSorter.field as string,firstSorter.order as "ascend" | "descend");}} else if (sorter?.field && onSort) {// 处理单列排序的情况onSort(sorter.field as string, sorter.order as "ascend" | "descend");}};return (<div>{/* 搜索框 */}<Space style={{ marginBottom: 16 }}><Input.Searchplaceholder="请输入搜索内容"value={searchValue}onChange={(e) => setSearchValue(e.target.value)}onSearch={handleSearch}enterButton={<SearchOutlined />}/></Space>{/* 表格 */}<Table<RecordType>columns={columns}dataSource={dataSource}loading={loading}pagination={pagination}onChange={handleChange}locale={{emptyText,}}/></div>);
};export default MyTable;

4. 使用组件

在其他地方使用封装好的 EnhancedTable 组件:

import XLBuuton from "../Components/XLButton";
import MyTable from "../Components/MyTable";
import type { ColumnsType } from "antd/es/table";interface User {key: number;name: string;age: number;address: string;
}const columns: ColumnsType<User> = [{title: "姓名",dataIndex: "name",key: "name",sorter: (a: User, b: User) => a.name.localeCompare(b.name),},{title: "年龄",dataIndex: "age",key: "age",sorter: (a: User, b: User) => a.age - b.age,},{title: "地址",dataIndex: "address",key: "address",},
];const dataSource: User[] = [{key: 1,name: "张三",age: 32,address: "北京市朝阳区",},{key: 2,name: "李四",age: 42,address: "上海市浦东新区",},{key: 3,name: "李四",age: 42,address: "上海市浦东新区",},{key: 4,name: "李四",age: 42,address: "上海市浦东新区",},{key: 5,name: "李四",age: 42,address: "上海市浦东新区",},{key: 6,name: "李四",age: 42,address: "上海市浦东新区",},{key: 7,name: "李四",age: 42,address: "上海市浦东新区",},
];const Home = () => {return (<div><h1>仿antd按钮</h1><XLBuuton type="danger">我是danger</XLBuuton><XLBuuton type="primary">我是primary</XLBuuton><XLBuuton>我是默认</XLBuuton><hr /><h1>自定义表格--包含分页和搜索排序</h1><MyTable<User>columns={columns}dataSource={dataSource}pagination={{ pageSize: 5 }}onSearch={(value) => console.log("搜索:", value)}onSort={(field, order) => console.log("排序:", field, order)}/></div>);
};export default Home;

在这里插入图片描述
在这里插入图片描述

三、封装组件的最佳实践

  1. 类型安全

    • 使用 TypeScript 定义组件的 props 和内部状态。
    • 避免使用 any 类型,确保类型推断正确。
  2. 可扩展性

    • 通过 props 暴露必要的配置项(如分页、搜索、排序等)。
    • 支持自定义渲染(如 render 函数)。
  3. 默认值

    • 为可选 props 提供合理的默认值,减少重复代码。
  4. 样式隔离

    • 使用 CSS Modules 或 CSS-in-JS 避免样式污染。
    • 通过 classNamestyle 允许外部覆盖样式。
  5. 文档和示例

    • 编写清晰的 README,说明组件的用途、API 和使用示例。
    • 提供 Storybook 或类似工具展示组件的交互效果。

四、进阶优化

  1. 国际化(i18n)

    • 支持多语言文本(如空数据提示)。
  2. 主题定制

    • 通过 CSS 变量或主题配置文件支持主题切换。
  3. 性能优化

    • 使用 React.memo 避免不必要的重渲染。
    • 对大数据量表格使用虚拟滚动(如 react-window)。
  4. 单元测试

    • 编写 Jest 或 React Testing Library 测试用例,确保组件行为符合预期。

总结

通过封装 Ant Design 组件,我们可以:

  1. 提升开发效率,减少重复代码。
  2. 统一项目风格,降低维护成本。
  3. 快速响应业务需求变化,扩展组件功能。

封装组件的核心思想是 抽象公共逻辑,暴露灵活配置。希望本文的分享能帮助你在实际项目中更好地复用和扩展 Ant Design 组件!如果你有其他封装组件的经验或问题,欢迎在评论区交流!

相关文章:

React 如何封装一个可复用的 Ant Design 组件

文章目录 前言一、为什么需要封装组件&#xff1f;二、 仿antd组件的Button按钮三、封装一个可复用的表格组件 (实战)1. 明确需求2. 设计组件 API3. 实现组件代码4. 使用组件 三、封装组件的最佳实践四、进阶优化 总结 前言 作为一名前端开发工程师&#xff0c;在日常项目中&a…...

CloudWeGo-Netpoll:高性能NIO网络库浅析

一、Netpoll 简介 Netpoll 是由字节跳动开发的高性能 NIO&#xff08;Non-blocking I/O&#xff09;网络库&#xff0c;专注于 RPC 场景。在 RPC 场景中&#xff0c;通常有较重的处理逻辑&#xff0c;无法串行处理 I/O。而 Go 的标准库 net 设计了 BIO&#xff08;Blocking I/…...

Mac的显卡架构种类

目录 一、Intel架构时期的Mac显卡&#xff08;2006 年至 2020 年&#xff09; 1. Intel 集成显卡&#xff08;iGPU&#xff09; 2. 独立显卡&#xff08;dGPU&#xff09;—— AMD 和 NVIDIA &#xff08;1&#xff09;AMD Radeon&#xff08;主流独显选择&#xff09; &a…...

HTTP基本概述

HTTP基本概述 报文格式 HTTP报文分为 请求报文 和 响应报文 一、请求报文 请求行&#xff08;Request Line&#xff09;请求头部&#xff08;Request Headers&#xff09;&#xff08;空行&#xff09;请求体&#xff08;Request Body&#xff09; ← 可选&#xff0c;如 P…...

Canvas SVG BpmnJS编辑器中Canvas与SVG职能详解

Canvas详解与常见API 一、Canvas基础 核心特性 • 像素级绘图&#xff1a;Canvas是基于位图的绘图技术&#xff0c;通过JavaScript操作像素实现图形渲染&#xff0c;适合动态、高性能场景&#xff08;如游戏、数据可视化&#xff09;。 • 即时模式&#xff1a;每次绘制需手动…...

dify多实例部署,一台机器部署多个dify实例

dify多实例部署 目的 实现在一台机器上&#xff0c;部署多个dify的实例。比如一个部署1.2版本&#xff0c;一个部署1.3版本。废话没有&#xff0c;直接上干货。 前提 你的电脑已经部署了一个dify实例&#xff0c;并成功运行。比如已经部署成功0.15.3版本。 步骤如下&#…...

ML 48.机器学习之临床生存树(rpartSurv)

简介机器学习中生存树&#xff08;Survival Tree&#xff09;的原理详解 生存树是结合决策树与生存分析的机器学习模型&#xff0c;主要用于处理带有时间-事件数据&#xff08;包含删失数据&#xff09;的预测问题。其核心目标是&#xff1a;通过树状结构对数据进行递归分割&am…...

HarmonyOS 应用开发,如何引入 Golang 编译的第三方 SO 库

本指南基于笔者临时修复的 ohos_golang_go 项目fork&#xff0c;解决HO 应用导入 cgo编译产物时的 crash 问题。 1. 下载 ohos_golang_go git clone https://gitcode.com/deslord/ohos_golang_go.git&#x1f4cc; 该仓库为笔者临时修复版本&#xff0c;修复了 CGO 编译模式下…...

Axure元件动作六:设置图片

亲爱的小伙伴,在您浏览之前,烦请关注一下,在此深表感谢!如有帮助请订阅专栏! Axure产品经理精品视频课已登录CSDN可点击学习https://edu.csdn.net/course/detail/40420 案例视频: Axure元件动作:设置图片 课程主题:设置图片 主要内容:图片悬停、鼠标按下时、选中...

一体化雷达波明渠流量计简介

一、技术定义与核心原理 一体化雷达波明渠流量计是基于微波技术的全自动流量监测设备&#xff0c;采用 24G K 波段平面雷达技术&#xff0c;通过非接触式测量方式实现对明渠、河道、排水管网等场景的水位、流速及流量监测。其核心原理是利用雷达发射高频电磁波&#xff0c;经水…...

Pr -- 耳机没有Pr输出的声音

问题 很久没更新视频号了&#xff0c;想用pr剪辑一下&#xff0c;结果使用Pr打开后发现耳机没有Pr输出的声音 解决方法 在编辑--首选项-音频硬件中设置音频硬件的输出为当前耳机设备...

白皮精读:2024年国家数据基础设施建设指引【附全文阅读】

《国家数据基础设施建设指引》提出建设覆盖数据采集至安全全链条的新型基础设施,目标到 2029 年形成横向联通、纵向贯通的格局,聚焦数据可信流通、算力协同、高速传输、安全保障四大功能,明确技术架构与重点方向,强调政府与市场协同,分阶段推进试点及规模化部署,为数字中…...

【信息系统项目管理师】第21章:项目管理科学基础 - 23个经典题目及详解

更多内容请见: 备考信息系统项目管理师-专栏介绍和目录 文章目录 【第1~2题】【第3题】【第4题】【第5题】【第6题】【第7题】【第8题】【第9题】【第10题】【第11题】【第12题】【第13题】【第14题】【第15题】【第16题】【第17题】【第18题】【第19题】【第20题】【第21题】…...

Mocha-Webpack 使用教程

Mocha-Webpack 使用教程 mocha-webpackmocha test runner with integrated webpack precompiler项目地址:https://gitcode.com/gh_mirrors/mo/mocha-webpack 项目介绍 Mocha-Webpack 是一个结合了 Mocha 和 Webpack 的测试工具&#xff0c;它允许开发者在使用 Webpack 打包的…...

AI硬件革命:OpenAI“伴侣设备”——从概念到亿级市场的生态重构

2025年5月23日&#xff0c;OpenAI宣布以65亿美元全股收购苹果前首席设计师Jony Ive创立的AI硬件公司io&#xff0c;并计划于2026年底前推出首款“AI伴侣设备”&#xff0c;目标出货量达1亿台。这一消息迅速成为全球AI领域的热点&#xff0c;标志着AI技术从云端大模型向端侧硬件…...

穿屏技巧:Mac-Windows一套鼠标键盘控制多台设备 (sharemouse6.0-Keygen)| KM-401A

文章目录 引言I sharemouse6.0介绍功能介绍关闭自动更新安装包II 安装系统对应的sharemouse软件Windowsmac版本III 知识扩展:SCP、FTP、SSH文件传输SCP配置SSH密钥免密登录FTP(File Transfer Protocal,文件传输协议)引言 基于USB进行同步键盘和鼠标事件,更流畅。 基于局域…...

【写在创作纪念日】基于SpringBoot和PostGIS的各省东西南北四至极点区县可视化

目录 前言 一、空间检索简介 1、空间表结构 2、四至空间检索 二、前后端实现 1、后端实现 2、前端集成 三、成果展示 1、东部省份 2、西部省份 3、南部省份 4、北部省份 5、中部省份 四、总结 前言 在当今数字化时代&#xff0c;地理信息数据的分析与可视化对于众…...

【C/C++】线程状态以及转换

文章目录 线程状态以及转换1 基本状态1.1 新建&#xff08;New&#xff09;1.2 就绪&#xff08;Ready / Runnable&#xff09;1.3 运行中&#xff08;Running&#xff09;1.4 阻塞/等待&#xff08;Blocked / Waiting / Sleeping&#xff09;1.5 挂起&#xff08;Suspended&am…...

从零开始:Python语言进阶之异常处理

一、认识异常&#xff1a;程序运行中的“意外事件” 在编写Python程序时&#xff0c;即使代码语法完全正确&#xff0c;运行过程中也可能遭遇各种意外情况。这些意外被称为异常&#xff0c;它们会打断程序的正常执行流程。例如&#xff0c;当我们尝试打开一个不存在的文件、用0…...

关于vue彻底删除node_modules文件夹

Vue彻底删除node_modules的命令 vue的node_modules文件夹非常大,常规手段根本无法删除. 解决方法: 在node_modules文件夹所在的路径运行命令窗口,并执行下面的命令. npm install rimraf -g rimraf node_modules说明&#xff1a; npm install rimraf -g 该命令是安装 node…...

如何制作可以本地联网搜索的MCP,并让本地Qwen3大模型调用搜索回答用户问题?

环境: SearXNG Qwen3-32B-FP8 vllm 0.8.5 问题描述: 如何制作可以本地联网搜索的MCP,并让本地Qwen3大模型调用搜索回答用户问题? 解决方案: 一、安装searxng 1.按需新建模型相关文件夹 mkdir MCP chmod 777 /mnt/program/MCP2.配置conda源 nano ~/.condarc nano…...

服务器硬盘虚拟卷的处理

目前的情况是需要删除逻辑卷&#xff0c;然后再重新来弄一遍。 数据已经备份好了&#xff0c;所以不用担心数据会丢失。 查看服务器的具体情况 使用 vgdisplay 操作查看服务器的卷组情况&#xff1a; --- Volume group ---VG Name vg01System IDFormat …...

一个国债交易策略思路

该国债交易策略的核心在于通过分析历史价格数据来识别市场趋势&#xff0c;并在趋势确认时进行开仓操作。策略的设计思路结合了价格波动范围的计算和市场波动性的评估&#xff0c;旨在捕捉市场的短期趋势并控制风险。 首先&#xff0c;策略通过对过去5根K线的最高价和最低价进行…...

Linux常用下载资源命令

wget命令 基本用法 wget -O http://example.com/file.zip-O 参数表示将文件保存为原始文件名。 如果需要指定文件名&#xff0c;可以使用 -o 参数&#xff1a; wget -o custom_name.zip http://example.com/file.zip-P &#xff1a;指定下载文件的保存路径。 wget -P /path/…...

Go语言爬虫系列教程(三)HTML解析技术

第3课&#xff1a;HTML解析技术 在上一章中&#xff0c;我们使用正则表达式提取网页内容&#xff0c;但这种方法有局限性。对于复杂的HTML结构&#xff0c;我们需要使用专门的HTML解析库。下面将介绍如何在Go中解析HTML。 1. HTML DOM树结构介绍 1.1 什么是DOM DOM&#xf…...

【三维重建】【3DGS系列】【深度学习】3DGS的理论基础知识之如何形成高斯椭球

【三维重建】【3DGS系列】【深度学习】3DGS的理论基础知识之如何形成高斯椭球 文章目录 【三维重建】【3DGS系列】【深度学习】3DGS的理论基础知识之如何形成高斯椭球前言高斯函数一维高斯多维高斯 椭球基本定义一般二次形式 3D高斯椭球3D高斯与椭球的关系各向同性(Isotropic)和…...

“夹子音”的发声原理和潜在风险

关于“夹子音”的发声原理和潜在风险&#xff0c;以下从科学角度和声乐实践出发&#xff0c;为你详细解析&#xff1a; 一、什么是夹子音&#xff1f; 夹子音是近年来网络流行的非专业术语&#xff0c;指通过刻意挤压喉部、改变共鸣腔形态发出的 尖细、嗲气、幼态化 的声音。常…...

思科硬件笔试面试题型解析

本专栏预计更新60期左右。当前第13期 这个系列通过在各类网上搜索大厂公开的笔试和面试题目,然后构造相关的知识点矩阵,让大家对核心的知识点有更深的认识,这个过程虽然耗时费力,但大厂的很多题目(包括模拟题)确实非常巧妙,很有代表性。由于官方没有发布过这样的题库,所…...

手写ES6 Promise() 相关函数

手写 Promise() 相关函数&#xff1a; Promise()、then()、catch()、finally() // 定义三种状态常量 const PENDING pending const FULFILLED fulfilled const REJECTED rejectedclass MyPromise {/*定义状态和结果两个私有属性:1.使用 # 语法&#xff08;ES2022 官方私有字…...

Windows 平台 TCP 通信开发指南

开篇介绍 在 Windows 平台进行 TCP 通信开发&#xff0c;是网络编程中的常见需求。本文将详细讲解在 Windows 平台下&#xff0c;如何利用 Winsock API 实现高效的 TCP 客户端与服务端通信。 使用示例 必须引入的头文件 #include <windows.h> #pragma comment(lib,&q…...