react实现表头固定的方法:1、通过Ant Design的Table组件实现表格固定表头;2、使用“rc-table”实现移动端表格表头固定;3、通过监听div的onscroll事件,改变div的scrollLeft属性。
本教程操作环境:Windows10系统、react18.0.0版、Dell G3电脑。
react怎么实现表头固定?
React表格固定表头/锁定列
Ant Design的Table组件挺好用,固定表头及锁定列的功能不在话下,但Ant Design Mobile没有Table组件。移动端要实现表格固定表头及锁定列的功能应该可以使用rc-table,当然也可以自己写一个。
通过分析AntD的Table,可以看出固定表头的表格是由上下两个f5d188ed2c074f8b944552db028f98a1标签组成的,它们分别嵌套在div内,上面的是表头,只包含ae20bdd317918ca68efdc799512a9b39,下边是表格内容,只包含92cee25da80fac49f6fb6eec5fd2c22a。应该是通过监听下面div的onscroll事件,改变上面div的scrollLeft属性,这样在水平滚动表格时,表头也会同步滚动。固定列是通过设置th及td的CSS属性position为sticky并且设置left或right为0实现,同时设置z-index,让锁定的列始终显示在上方。
原理整明白了,写代码就比较容易了。
components/ScrollableTable/interface.tsx import * as React from 'react'; export declare type AlignType = 'left' | 'center' | 'right'; export interface ColumnType { align?: AlignType; className?: string; dataKey?: string; fixed?: boolean; title?: React.ReactNode; width?: number; render?: (value: any, record: any, index: number) => React.ReactNode; } export interface TableProps { className?: string; style?: React.CSSProperties; columns?: ColumnType[]; dataSource?: any[]; width?: number; height?: number; } components/ScrollableTable/index.tsx import React, { FunctionComponent, useRef } from 'react'; import { TableProps, ColumnType } from './interface'; import './index.less'; const ScrollableTable: FunctionComponent<any> = (props: TableProps) => { const style: React.CSSProperties = props.style || {}; const maxHeight: string = props.width ? (props.height + 'px') : 'unset'; const columns: ColumnType[] = props.columns || []; const dataSource: any[] = props.dataSource || []; let maxWidth: number = 0; if (props.width) style.width = props.width; if (columns.length === 0) { columns.push({ dataKey: 'key' }); } columns.forEach((column: ColumnType) => { const width: number = column.width || 50; maxWidth += width; }); const fixedColumns: number[][] = getFixedColumns(columns); const leftFixedColumns: number[] = fixedColumns[0]; const rightFixedColumns: number[] = fixedColumns[1]; const tableBody: any = useRef(); const handleScroll = (target: any) => { const scrollLeft: number = target.scrollLeft; const tableHeaders: any = target.parentElement.getElementsByClassName('st-table-header'); if (tableHeaders.length > 0) { tableHeaders[0].scrollLeft = scrollLeft; } }; return ( <div className={classNames('st-table-container', props.className)} style={style} > <div className="st-table-header"> <table> <colgroup> { renderCols(columns) } </colgroup> <thead className="st-table-thead"> <tr> { columns.map((column: ColumnType, index: number) => { const align: any = column.align || undefined; const title: React.ReactNode = column.title || ''; const fixed: string = leftFixedColumns.includes(index) ? 'left' : (rightFixedColumns.includes(index) ? 'right' : ''); const fixedClassName: string = fixed ? ('st-table-cell-fix-' + fixed) : ''; return ( <th key={index} className={classNames('st-table-cell', fixedClassName, column.className)} style={{textAlign: align}} > {title} </th> ); }) } </tr> </thead> </table> </div> <div ref={tableBody} className="st-table-body" style={{maxHeight: maxHeight}} onScroll={(e: any) => handleScroll(e.currentTarget)} > <table style={{width: maxWidth, minWidth: '100%'}}> <colgroup> { renderCols(columns) } </colgroup> <tbody className="st-table-tbody"> { dataSource.map((record: any, index: number) => ( <tr key={index} className="st-table-row"> { renderCells(columns, leftFixedColumns, rightFixedColumns, record, index) } </tr> )) } </tbody> </table> </div> </div> ); }; function classNames(...names: (string | undefined)[]) { const currentNames: string[] = []; names.forEach((name: (string | undefined)) => { if (name) currentNames.push(name); }); return currentNames.join(' '); } function getFixedColumns(columns: ColumnType[]) { const total: number = columns.length; const leftFixedColumns: number[] = []; const rightFixedColumns: number[] = []; if (columns[0].fixed) { for (let i = 0; i < total; i++) { if (columns[i].fixed) { leftFixedColumns.push(i); } else { break; } } } if (columns[total - 1].fixed) { for (let i = total - 1; i >= 0; i--) { if (columns[i].fixed) { if (!leftFixedColumns.includes(i)) rightFixedColumns.push(i); } else { break; } } } return [leftFixedColumns, rightFixedColumns]; } function renderCols(columns: ColumnType[]) { return columns.map((column: ColumnType, index: number) => { const width: number = column.width || 50; return ( <col key={index} style={{width: width, minWidth: width}} /> ); }); } function renderCells(columns: ColumnType[], leftFixedColumns: number[], rightFixedColumns: number[], record: any, index: number) { return columns.map((column: ColumnType, index: number) => { const align: any = column.align || undefined; const fixed: string = leftFixedColumns.includes(index) ? 'left' : (rightFixedColumns.includes(index) ? 'right' : ''); const className: string = classNames('st-table-cell', column.className, fixed ? ('st-table-cell-fix-' + fixed) : ''); const rawValue: any = (column.dataKey && column.dataKey in record) ? record[column.dataKey] : undefined; let value: any = undefined; if (column.render) { value = column.render(rawValue, record, index); } else { value = (rawValue === undefined || rawValue === null) ? '' : String(rawValue); } return ( <td key={index} className={className} style={{textAlign: align}} > {value} </td> ); }); } export default ScrollableTable; components/ScrollableTable/index.less .st-table-container { border: 1px solid #f0f0f0; border-right: 0; border-bottom: 0; font-size: 14px; .st-table-header { border-right: 1px solid #f0f0f0; overflow: hidden; table { border-collapse: separate; border-spacing: 0; table-layout: fixed; width: 100%; thead.st-table-thead { tr { th.st-table-cell { background: #fafafa; border-bottom: 1px solid #f0f0f0; border-right: 1px solid #f0f0f0; color: rgba(0, 0, 0, .85); font-weight: 500; padding: 8px; text-align: left; &:last-child { border-right: 0; } } } } } } .st-table-body { overflow: auto scroll; border-bottom: 1px solid #f0f0f0; border-right: 1px solid #f0f0f0; table { border-collapse: separate; border-spacing: 0; table-layout: fixed; tbody.st-table-tbody { tr.st-table-row { td.st-table-cell { border-bottom: 1px solid #f0f0f0; border-right: 1px solid #f0f0f0; color: rgba(0, 0, 0, .65); padding: 8px; text-align: left; &:last-child { border-right: 0; } } &:last-child { td.st-table-cell { border-bottom: 0; } } } } } } table { .st-table-cell { &.st-table-cell-fix-left { background: #fff; position: sticky; left: 0; z-index: 2; } &.st-table-cell-fix-right { background: #fff; position: sticky; right: 0; z-index: 2; } } } }
然后可以这样使用:
views/Test/index.tsx import React, { FunctionComponent } from 'react'; import Page from '../../components/Page'; import ScrollableTable from '../../components/ScrollableTable'; import StoreProvider from '../../stores/products/context'; import './index.less'; const Test: FunctionComponent<any> = (props: any) => { let records: any[] = [{ id: 1, productName: '淡泰', amount1: 198, amount2: 200, amount3: 205.5, currency: '人民币', ca: 'Amy' }, { productName: '方润', amount1: 105.5, amount2: 100, amount3: 108, currency: '港元', ca: 'Baby' }, { productName: '医疗基金-1', amount1: 153, amount2: 150, amount3: 155, currency: '人民币', ca: 'Emily' }, { productName: '医疗基金-2', amount1: 302, amount2: 300, amount3: 290, currency: '美元', ca: 'Baby' }, { productName: '医疗基金-3', amount1: 108.8, amount2: 100, amount3: 130, currency: '人民币', ca: 'Amy' }, { productName: '医疗基金-4', amount1: 205, amount2: 200, amount3: 208, currency: '美元', ca: '吴丹' }, { productName: '医疗基金-5', amount1: 315.5, amount2: 300, amount3: 280, currency: '人民币', ca: 'Baby' }, { productName: '医疗基金-6', amount1: 109, amount2: 95, amount3: 106, currency: '人民币', ca: 'Emily' }, { productName: '恒大私募债', amount1: 213, amount2: 200, amount3: 208, currency: '港元', ca: '吴丹' }]; const totalRecord: any = { productName: '合计', amount1: {}, amount2: {}, amount3: {} }; records.forEach((record: any) => { const currency: string = record.currency; ['amount1', 'amount2', 'amount3'].forEach((key: string) => { const value: any = totalRecord[key]; if (!(currency in value)) value[currency] = 0; value[currency] += record[key]; }); }); records.push(totalRecord); const columns: any[] = [{ dataKey: 'productName', title: '产品名称', width: 90, fixed: true }, { dataKey: 'amount1', title: <React.Fragment>上周缴款金额<br/>(万)</React.Fragment>, width: 140, align: 'center', className: 'amount', render: calculateTotal }, { dataKey: 'amount2', title: <React.Fragment>上周预约金额<br/>(万)</React.Fragment>, width: 140, align: 'center', className: 'amount', render: calculateTotal }, { dataKey: 'amount3', title: <React.Fragment>待本周跟进金额<br/>(万)</React.Fragment>, width: 140, align: 'center', className: 'amount', render: calculateTotal }, { dataKey: 'currency', title: '币种', width: 80 }, { dataKey: 'ca', title: 'CA', width: 80 }]; return ( <StoreProvider> <Page {...props} title="销售统计" className="test" > <div style={{padding: 15}}> <ScrollableTable width={window.innerWidth - 30} height={196} columns={columns} dataSource={records} /> </div> </Page> </StoreProvider> ); }; function calculateTotal(value: any) { if (value instanceof Object) { const keys: any[] = Object.keys(value); return ( <React.Fragment> { keys.map((key: string, index: number) => ( <span key={index}> {`${value[key].toFixed(2)}万${key}`} </span> )) } </React.Fragment> ) } return value.toFixed(2); } export default Test; views/Test/index.less .st-table-container { .st-table-body { td.st-table-cell.amount { padding-right: 20px !important; text-align: right !important; span { display: block; } } } }
推荐学习:《react视频教程》
以上是react怎么实现表头固定的详细内容。更多信息请关注PHP中文网其他相关文章!

要将React集成到HTML中,需遵循以下步骤:1.在HTML文件中引入React和ReactDOM。2.定义一个React组件。3.使用ReactDOM将组件渲染到HTML元素中。通过这些步骤,可以将静态HTML页面转化为动态、交互式的体验。

React受欢迎的原因包括其性能优化、组件复用和丰富的生态系统。1.性能优化通过虚拟DOM和diffing机制实现高效更新。2.组件复用通过可复用组件减少重复代码。3.丰富的生态系统和单向数据流增强了开发体验。

React是构建动态和交互式用户界面的首选工具。1)组件化与JSX使UI拆分和复用变得简单。2)状态管理通过useState钩子实现,触发UI更新。3)事件处理机制响应用户交互,提升用户体验。

React是前端框架,用于构建用户界面;后端框架用于构建服务器端应用程序。React提供组件化和高效的UI更新,后端框架提供完整的后端服务解决方案。选择技术栈时需考虑项目需求、团队技能和可扩展性。

HTML和React的关系是前端开发的核心,它们共同构建现代Web应用的用户界面。1)HTML定义内容结构和语义,React通过组件化构建动态界面。2)React组件使用JSX语法嵌入HTML,实现智能渲染。3)组件生命周期管理HTML渲染,根据状态和属性动态更新。4)使用组件优化HTML结构,提高可维护性。5)性能优化包括避免不必要渲染,使用key属性,保持组件单一职责。

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

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

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


热AI工具

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

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

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。

WebStorm Mac版
好用的JavaScript开发工具

MinGW - 适用于 Windows 的极简 GNU
这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

VSCode Windows 64位 下载
微软推出的免费、功能强大的一款IDE编辑器