成人怡红院-成人怡红院视频在线观看-成人影视大全-成人影院203nnxyz-美女毛片在线看-美女免费黄

站長資訊網
最全最豐富的資訊網站

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,可以看出固定表頭的表格是由上下兩個<table>標簽組成的,它們分別嵌套在div內,上面的是表頭,只包含<thead>,下邊是表格內容,只包含<tbody>。應該是通過監聽下面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視頻教程》

贊(0)
分享到: 更多 (0)
?
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
中文成人久久久久影院免费观看| 上边一面亲下边一面膜的功效 | 小SAO货都湿掉奶头好硬男女| 性色AV一区二区三区无码| 亚洲国产在一区二区三区| 伊人久久大香线蕉无码不卡| 9L国产精品久久久久尤物| 被黑人猛烈30分钟视频| 国产SUV精品一区二区四| 国产亚洲色欲色一色WWW| 久久WWW成人_看片| 欧美黑人又粗又大久久久| 色妞精品AV一区二区三区| 小嫩模无套内谢第一次| 亚洲综合大片6999| JAPANESE少妇高潮潮喷| 国产精久久一区二区三区| 久久99久久99小草精品免视看| 欧美精产国品一二三区别| 天堂中文АⅤ在线| 亚洲熟伦熟女新五十路熟妇| www.xx欧美大鸡巴| 国产裸体XXXX视频在线播放| 久久香港三级台湾三级播放| 日本少妇XXX做受| 亚洲国产成人精品女人久久久| 中文字幕免费不卡二区| 丰满熟妇岳AV无码区HD| 精品无码人妻一区二区三区不卡| 欧美乱大交XXXXX疯狂俱乐部| 无码秘 蜜桃一区二区三区| 亚洲综合大片6999| 岛国无码AV不卡一区二区| 精品国产乱码久久久久久浪潮| 欧美超级乱婬视频播放| 午夜无码无遮挡在线视频| 中文WWW新版资源在线| 国产成人综合久久精品推最新| 久久五月丁香合缴情网| 熟妇人妻无码一区二区三区| 亚洲愉拍自拍欧美精品| 丰满少妇A级毛片野外| 久久国产精品香蕉成人APP| 人人玩人人添人人澡| 亚洲欧美日韩综合久久久久| 成 人 黄 色 网 站 18| 久久97人妻AⅤ无码一区 | 久久国产劲爆∧V内射-百度| 日本一区二区三区免费播放| 亚洲色欲综合一区二区三区小说| 菠萝蜜进口路线区二1688| 久久99精品国产99久久6| 三人一起玩弄娇妻高潮| 伊人涩涩涩涩久久久AV| 国产成人亚洲精品无码青| 猫咪WWW免费人成网站| 无码人妻一区二区三区免费看成人| 中字年轻漂亮的儿媳BD| 国产一产二产三精华液| 日本插槽X8插槽怎么用的| 亚洲中文字幕无码永久在线不卡| 隔壁邻居是巨爆乳寡妇| 免费观看羞羞的事情网站| 亚洲AV成人无码一二三在线观看| 把腿张开老子CAO烂你NP皇宫| 久久AV无码ΑV高潮ΑV喷吹| 四虎国产精品成人影院| 97久久国产露脸精品国产| 狠狠色欧美亚洲综合色| 少妇人妻在线视频| 99久久精品国产一区二区| 精品久久久久久中文字幕无码VR| 少妇爆乳无码AV无码专区| 69综合精品国产二区无码| 精品深夜AV无码一区二区| 天堂АⅤ在线最新版在线| 99热最新成人国产精品| 久久国国产免费999| 午夜福利YW在线观看2020| 边做边爱MP3在线播放免费观看| 久久天天躁狠狠躁夜夜躁2014| 亚州中文字幕午夜福利电影| 吃了继兄开的药后我做的梦更长了 | 东京热加勒比视频一区| 你真紧你这是要我的命吗什么意思 | 国产精品久久久久久精品三级| 人人澡人人妻人人爽人人蜜桃麻豆 | 精品日产A一卡2卡三卡4卡乱| 无码 制服 丝袜 国产 另类| 波多野结衣AV一区二区全免费观| 麻花豆传媒剧国产MV在线观看| 亚洲国产精品久久一线APP| 国产精品原创AV片国产日韩| 肉欲扒灰合集100篇| ZO2O女人另类ZO2O洗浴| 免费一本色道久久一区| 野花社区大全免费观看3| 狠狠躁夜夜躁青青草原软件| 无码国产色欲XXXX视频| 高潮VPSWINDOWS国产乱| 人人妻人人澡AV| ZOOM人与ZOOM视频| 欧美黑人巨大video粗暴| 18禁超污无遮挡无码免费动态图 | 一炕四女被窝交换| 久久国产精品成人无码网站| 亚洲精品成人无码中文毛片不卡 | 亚洲美女国产精品久久久久久久久| 国产精品无码一区二区在线| 熟女一区二区三区| 成熟闷骚女邻居引诱2| 人妻少妇伦在线麻豆M电影| BGMBGMBGM欧美老妇| 欧产日产国产精品| 51精产国品一二三产区| 奶头被几个流浪汉吃肿了| 在线观看成人片韩剧| 老头发狂的吸住她的乳尖| 一区二区清无吗视频| 久久婷婷人人澡人人爽人人爱| 亚洲欧洲日产V一个人免费观看视频WWW高清| 国语对白做受XXXXX在线中国| 香港三级精品三级在线专区| 国产精品任我爽爆在线播放| 无码中文字幕VA精品影院| 国产无套护士在线观看| 亚洲AⅤ成人精品无码| 国产偷自视频区视频| 亚洲AV无码一区二区三区在线播| 国自产拍偷拍精品啪啪模特| 亚洲VA久久久噜噜噜久久男同| 护士高潮喷水白浆| 亚洲中文字幕成人无码| 狂野欧美激情性XXXX在线观看| 中日AV乱码一区二区三区乱码| 男女乱婬免费视频黑人| MM131美女图片高清图片视频| 日本19禁啪啪无遮挡免费| 嗯好爽快点插我视频在线播放| 色一情一区二区三区四区| 国产国产精品人在线观看| 亚洲 欧美 国产 动漫 综合| 精品国产亚洲一区二区三区在线观| 亚洲色精品VR一区二区三区| 裸体欣赏ⅤIDE0SPH0TO| AV无码欧洲亚洲电影网| 日韩AV无码社区一区二区三区| 高雅人妻被迫沦为玩物| 午夜精品久久久久成人| 精品久久久久久狼人社区| 伊人久久久久熟女AV大片| 欧美高清性色生活片免费观看| 成年动作片AV免费网站| 无码人妻少妇色欲AV一区二区| 怀孕挺大肚子疯狂高潮AV毛片| 一二三四社区在线高清观看| 欧美军警GAY巨大粗长| 公翁大龟挺进秀婷全文免费阅读| 羞羞在线版免费阅读入口 | 欧美成人一区在线| 成人午夜视频精品一区| 性妇WBBBB搡BBBB嗓小说| 久久大香香蕉国产拍国| 337P西西人体大胆瓣开下部| 日韩AV毛片无码免费| 国产色母和进口色母区别| 野花免费观看日本韩国 | 久久久久99精品成人片试看 | 国产精选午睡沙发系列999| 亚洲欧美成人综合久久久| 女人与牛ZOZO| 国产大片内射1区2区| 亚洲性色成人AV天堂| 欧美又粗又大XXXXBBBB疯狂| 国产CHINASEX对白VID| 亚洲人成绝费网站色WWW吃脚| 妺妺坐在我腿上勃起弄了应用| 父母全家儿女大联欢第14集| 亚洲旡码A∨一区二区三区| 女朋友喊疼男生的心理活动知乎| 福利姬国产精品一区在线| 亚洲色偷偷综合亚洲AV伊人蜜桃| 欧美成人猛片AAAAAAA| 国产精品久久久久精品三级APP| 亚洲伊人色欲综合网| 人妻无码少妇一区二区| 国产综合无码一区二区色蜜蜜| 中文字幕无码成人片| 无码AⅤ精品一区二区三区 | 国产精品自在拍首页视频8| 幼儿HIPHOP张婉莹仙踪网| 伸进衣服里吃奶捏胸视频| 久久精品国产精品亚洲艾草网| 宝宝都湿透了还嘴硬疼怎么回事 | 狠狠色噜噜狠狠狠888米奇| 99久久久精品免费观看国产| 小SAO货都湿掉奶头好硬男女| 内射口爆少妇麻豆| 国产午夜福利亚洲第一| 9612黄桃网站进入页面直播|