react怎么实现表头固定

react实现表头固定的方法:1、通过Ant Design的Table组件实现表格固定表头;2、使用“rc-table”实现移动端表格表头固定;3、通过监听div的onscroll事件,改变div的scrollLeft属性。

react怎么实现表头固定

本教程操作环境:Windows10系统、react18.0.0版、Dell G3电脑。

react怎么实现表头固定?

React表格固定表头/锁定列

Ant Design的Table组件挺好用,固定表头及锁定列的功能不在话下,但Ant Design Mobile没有Table组件。移动端要实现表格固定表头及锁定列的功能应该可以使用rc-table,当然也可以自己写一个。

通过分析AntD的Table,可以看出固定表头的表格是由上下两个

标签组成的,它们分别嵌套在div内,上面的是表头,只包含

,下边是表格内容,只包含

。应该是通过监听下面div的onscroll事件,改变上面div的scrollLeft属性,这样在水平滚动表格时,表头也会同步滚动。固定列是通过设置th及td的CSS属性position为sticky并且设置left或right为0实现,同时设置z-index,让锁定的列始终显示在上方。

原理整明白了,写代码就比较容易了。

components/ScrollableTable/interface.tsximport * 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.tsximport React, { FunctionComponent, useRef } from 'react';import { TableProps, ColumnType } from './interface';import './index.less';const ScrollableTable: FunctionComponent = (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 (    
      
                              {              renderCols(columns)            }                                              {                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 (                                      );                })              }                              
                      {title}                    
      
      
 handleScroll(e.currentTarget)}      >                                {                renderCols(columns)              }                                      {                dataSource.map((record: any, index: number) => (                                      {                      renderCells(columns, leftFixedColumns, rightFixedColumns, record, index)                    }                                  ))              }                    
      
    
  );};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 = 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 (          );  });}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 (              {value}          );  });}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.tsximport 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 = (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: 上周缴款金额
(万),    width: 140,    align: 'center',    className: 'amount',    render: calculateTotal  }, {    dataKey: 'amount2',    title: 上周预约金额
(万),    width: 140,    align: 'center',    className: 'amount',    render: calculateTotal  }, {    dataKey: 'amount3',    title: 待本周跟进金额
(万),    width: 140,    align: 'center',    className: 'amount',    render: calculateTotal  }, {    dataKey: 'currency',    title: '币种',    width: 80  }, {    dataKey: 'ca',    title: 'CA',    width: 80  }];  return (                  
                  
            );};function calculateTotal(value: any) {  if (value instanceof Object) {    const keys: any[] = Object.keys(value);    return (              {          keys.map((key: string, index: number) => (                          {`${value[key].toFixed(2)}万${key}`}                      ))        }          )  }  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怎么实现表头固定的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2922051.html

(0)
上一篇 2025年3月11日 18:34:38
下一篇 2025年3月11日 18:34:45

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • react跳转传值怎么实现

    react跳转传值的实现方法:1、通过“history.push”或“NavLink”实现页面跳转;2、通过“this.props.history.push({pathname: ‘login’,query: {id…

    2025年3月11日
    100
  • react怎么实现显示隐藏div

    react实现显示隐藏div的方法:1、在函数式组件内容中通过“{showoverlay? ():null}”实现div的显示隐藏;2、在业务逻辑中判断visible的取值,并通过在组件样式中设置“style={{ display: `${…

    2025年3月11日
    200
  • react怎么给树增加节点

    react给树增加节点的方法:1、引入antd的树形控件,并通过“const treeData = […]”实现tree形组件;2、把当前节点的key值传递过去,然后遍历data数组;3、在遍历数组增加节点时,通过嵌套数组对象的…

    2025年3月11日
    200
  • react 安装插件就报错怎么办

    react安装插件就报错是由于从缓存中读取时不正确的完整性导致的,其解决办法:1、执行“yarn cache clean”命令;2、重新执行要安装的插件的指令操作即可。 本教程操作环境:Windows10系统、react18.0.0版、De…

    2025年3月11日
    200
  • react改变数组页面没有刷新怎么办

    react改变数组页面没有刷新是因为数组的赋值是引用传递的,其解决办法:1、打开相应的文件;2、找到“data.push(obj)”;3、使用新数组“let data = […this.state.data];”即可。 本教程操…

    2025年3月11日
    200
  • react路由跳转不刷新怎么办

    react路由跳转不刷新的解决办法:1、在路由组件最上层元素上加一个key增加路由的识别度;2、使用withRouter关联组件,代码如“render() {return (); }}export default withRouter(ro…

    2025年3月11日
    200
  • react打包图片路径错误怎么办

    react打包图片路径错误的解决办法:1、把build文件夹直接放到IIS上;2、修改index.html里面的引用资源,把“/static”前面的“/”符号去掉即可访问到页面内容。 本教程操作环境:Windows10系统、react18.…

    2025年3月11日
    200
  • react引入antd报错怎么办

    react引入antd报错的解决办法:1、通过“yarn add babel-plugin-import”下载“babel-plugin-import”;2、在babel选项中配置plugins;3、在“webpack.config.js”…

    2025年3月11日 编程技术
    200
  • react怎么改字体

    react改字体的方法:1、查看CSS3的“@font-face”属性在浏览器的兼容性;2、在assets目录下放置想要使用的字体名称;3、在“index.js”调用的“index.css”中定义“font”,然后就可以在全局使用新字体。 …

    2025年3月11日 编程技术
    200
  • react怎么移除节点

    react移除节点的方法:1、通过“import ReactDom from ‘react-dom’”引入“react-dom”;2、使用“ReactDom.render”添加节点;3、通过“let state=Re…

    2025年3月11日
    200

发表回复

登录后才能评论