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

站長資訊網(wǎng)
最全最豐富的資訊網(wǎng)站

react怎么實(shí)現(xiàn)表頭固定

react實(shí)現(xiàn)表頭固定的方法:1、通過Ant Design的Table組件實(shí)現(xiàn)表格固定表頭;2、使用“rc-table”實(shí)現(xiàn)移動端表格表頭固定;3、通過監(jiān)聽div的onscroll事件,改變div的scrollLeft屬性。

react怎么實(shí)現(xiàn)表頭固定

本教程操作環(huán)境:Windows10系統(tǒng)、react18.0.0版、Dell G3電腦。

react怎么實(shí)現(xiàn)表頭固定?

React表格固定表頭/鎖定列

Ant Design的Table組件挺好用,固定表頭及鎖定列的功能不在話下,但Ant Design Mobile沒有Table組件。移動端要實(shí)現(xiàn)表格固定表頭及鎖定列的功能應(yīng)該可以使用rc-table,當(dāng)然也可以自己寫一個。

通過分析AntD的Table,可以看出固定表頭的表格是由上下兩個<table>標(biāo)簽組成的,它們分別嵌套在div內(nèi),上面的是表頭,只包含<thead>,下邊是表格內(nèi)容,只包含<tbody>。應(yīng)該是通過監(jiān)聽下面div的onscroll事件,改變上面div的scrollLeft屬性,這樣在水平滾動表格時(shí),表頭也會同步滾動。固定列是通過設(shè)置th及td的CSS屬性position為sticky并且設(shè)置left或right為0實(shí)現(xiàn),同時(shí)設(shè)置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;       }     }   } }
登錄后復(fù)制

然后可以這樣使用:

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: '醫(yī)療基金-1',     amount1: 153,     amount2: 150,     amount3: 155,     currency: '人民幣',     ca: 'Emily'   }, {     productName: '醫(yī)療基金-2',     amount1: 302,     amount2: 300,     amount3: 290,     currency: '美元',     ca: 'Baby'   }, {     productName: '醫(yī)療基金-3',     amount1: 108.8,     amount2: 100,     amount3: 130,     currency: '人民幣',     ca: 'Amy'   }, {     productName: '醫(yī)療基金-4',     amount1: 205,     amount2: 200,     amount3: 208,     currency: '美元',     ca: '吳丹'   }, {     productName: '醫(yī)療基金-5',     amount1: 315.5,     amount2: 300,     amount3: 280,     currency: '人民幣',     ca: 'Baby'   }, {     productName: '醫(yī)療基金-6',     amount1: 109,     amount2: 95,     amount3: 106,     currency: '人民幣',     ca: 'Emily'   }, {     productName: '恒大私募債',     amount1: 213,     amount2: 200,     amount3: 208,     currency: '港元',     ca: '吳丹'   }];   const totalRecord: any = {     productName: '合計(jì)',     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: '產(chǎn)品名稱',     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>上周預(yù)約金額<br/>(萬)</React.Fragment>,     width: 140,     align: 'center',     className: 'amount',     render: calculateTotal   }, {     dataKey: 'amount3',     title: <React.Fragment>待本周跟進(jìn)金額<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="銷售統(tǒng)計(jì)"         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;       }     }   } }
登錄后復(fù)制

推薦學(xué)習(xí):《react視頻教程》

贊(0)
分享到: 更多 (0)
?
網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號
班长哭了能不能再抠游戏里面的钱| 毛片无码免费无码播放| 国内美女推油按摩在线播放| 黑人与亚洲女人XXXXXXXⅩ| 精品无人区麻豆乱码1区2区| 鲁一鲁一鲁一鲁一澡| 国产无遮挡又黄又爽不要VIP网| 国产日产亚洲系列最新| 极品尤物一区二区三区| 国产熟女一区二区| 精品久久久BBBB人妻| 国产日韩亚洲大尺度高清| 激情无码人妻又粗又大| 老公和小三在车上做我想卖了车| 欧美成人少妇人妻精品视频| 久久欧美极品少妇XXXXⅩ| 男人扒开女人的腿做爽爽视频| 人体内射精一区二区三区| 太紧了夹得我的巴好爽欧美| 亚洲AV无码久久精品蜜桃播放| 亚洲最大天堂无码精品区| AV色综合久久天堂AV色综合在 | 蜜芽国产尤物AV尤物在线看| 人妻熟妇与黑人HDXⅩXX| 无码内射中文字幕岛国片| 亚洲熟妇色XXXXX中国少妇Y| AV一本大道香蕉大在线| 国产成人无码精品一区在线观看| 粗大挺进尤物人妻| 国内精品一线二线三线黄| 老熟女HDXXXX国产喷水| 日韩免费无码专区精品观看| 亚洲V欧美V国产V在线观看| 中文精品久久久久国产| 第九理论午夜电影院| 激情综合婷婷丁香五月情| 欧美高潮抽搐喷水大叫| 少妇特殊按摩高潮爽翻天| 亚洲欧美日韩国产成人| 锕锕锕锕锕锕锕好疼视频真人| 国产农村乱人伦精品视频| 美女露内裤扒开腿让男人桶无遮挡| 日韩成人精品久久网站| 亚洲精品99久久久久中文字幕 | 欧美黑人又大又粗XXXX| 午夜爽爽爽男女免费观看影院| 亚洲AⅤ永久无码毛片牛牛影视 | 亚洲乱码尤物193YW最新网站 | 强被迫伦姧在线观看无码| 亚洲AV无码久久精品蜜桃| C交人Z000Z000XXⅩ| 国自产精品手机在线观看视频| 欧美三级不卡在线观看| 亚洲成AV人片无码BT种子下载| PYTHON人狗大CSDN| 精品精品国产高清A级毛片| 人妻免费久久久久久久了| 亚洲精品国偷自产在线99正片| 爱丫爱丫影院在线视频| 精品一区二区三区无码视频| 色老板在线永久免费视频| 一本色道久久综合狠狠躁| 国产精华最好的产品人V中文| 奶头好大揉着好爽视频午夜院| 五月丁香综合激情六月久久| 亚洲国产精品成人AV在线| YY111111111少妇影院| 娇小BBW搡BBBB搡BBBB| 日产乱码一二三区别免费下载| 亚洲综合色一区二区三区| 国产成人一区二区三区影院| 女同学浮乱系列合集| 亚洲精品中文字幕无码蜜桃| 东北浪妇王梅娟偷人视频| 麻豆果冻传媒精品国产AV| 亚洲AV不卡一区二区三区| 波多野结衣亚洲AV手机在线| 久久中文字幕人妻熟AV女| 香蕉免费一区二区三区在| 波多野结衣人妻女教师4| 浪货趴办公桌~H揉秘书电影无码 浪货趴办公桌~H揉秘书电影 | 18禁H免费动漫无码网站| 狠狠色噜噜狠狠狠狠7777| 色欲AV伊人久久大香线蕉影院| 1000部无遮挡拍拍拍免费视频| 好吊妞人成视频在线观看强行| 少妇厨房愉情理伦BD在线观看| 97碰碰碰人妻无码视频| 公么大龟弄得我好舒服第一| 逆徒每天都想着欺师犯上 | 精品不卡一区二区| 天干天干夜天干天天爽| YOUJIZZ中国少妇| 毛片无码中文字幕| 亚洲人成无码网站在线观看野花| 国产毛片一区二区精品| 日日摸日日碰夜夜爽无码| 亚洲国产精品高清久久久| 国产AV免费一区二区三区| 人人妻人人澡人人爽人人精直播| 中文字幕无码久久一区| 久久久久久久精品成人热色戒| 亚洲AV成人无码一二三| 国产精品国产AV片国产| 日韩GAY小鲜肉啪啪18禁| XX性欧美肥妇精品久久久久久| 美女高潮无套内谢| 一面膜上边一面膜下边韩国| 精品久久亚洲中文无码| 亚洲AV片不卡无码一| BBWBBW欧美肥妇PICS| 开丫头小嫩苞疼死了| 亚洲午夜国产成人AV电影| 经典日韩成人网站在线观看| 午夜免费福利小电影| 国产成人免费AV片在线观看 | 成人午夜性A级毛片免费| 秋霞午夜久久午夜精品| AV天堂亚洲国产AV| 女人被第一次18毛片| 中文字幕无码人妻少妇免费| 久久亚洲国产精品成人AV秋霞| 亚洲图片小说激情综合| 久久国产加勒比精品无码| 亚洲国产成人爱AV在线播放| 韩国理论电费2023最| 亚洲AV综合AV一区二区三区| 国产台湾无码AV片在线观看| 午夜亚洲精品久久久久久| 国产熟妇与子伦HD| 性饥渴姓交HDSEX| 国产又色又爽又刺激在线播放 | 国语对白做受XXXXX在线中国| 亚洲AⅤ无码乱码在线观看性色| 国产偷V国产偷V亚洲高清| 亚洲AV无码成人精品国产| 黑人与日本XXXXXTV| 亚洲熟妇色XXXXX欧美老妇| 久久亚洲中文字幕精品一区| 中文无码人妻影音先锋| 女人爽到高潮的免费视频| AV无码久久久久不卡网站蜜桃| 欧美日产欧美日产国产精品| 啊轻点都日出水来了| 日韩一区二区三区无码人妻视频| 高清国产亚洲精品自在久久 | 久久99精品国产麻豆婷婷| 亚洲中文字幕无码久久综合网 | 日本一线和三线的区别| 国产98色在线 | 免费| 午夜精品久久久久9999高清| 黑人粗大猛烈进出高潮视频| 亚洲色成人网站WWW永久| 老外粗猛长爽的视频| AI换脸造梦JENNIE喷水| 日本理论片和搜子同居的日子| 国产AV高潮社区| 性一交一乱一色一视频| 精品人妻大屁股白浆无码| 一区二区三区乱码在线 | 中文 | 无码国模大尺度视频在线观看| 国产一区在线观看二区| 亚洲色成人网站www观看入口| 男人J桶进女人P无遮挡全过程| 波多野结系列18部无码观看A| 天堂AV无码AV一区二区三区| 好男人无码内射AV| 在线播放无码后入内射少妇| 秋霞国产午夜伦午夜无码灬 | GOOD电影网韩国三级无码| 色综合视频一区二区三区| 国色天香精品一卡2卡3卡4| 呦香8黝黝狖呦香8| 日本乱码伦午夜福利在线| 国产精品成人VA在线观看| 亚洲精品无码成人| 欧美高清视频手机在在线| 国产成人国拍亚洲精品| 亚洲精品自产拍在线观看动漫| 内射人妻深入内射| 国产999精品成人网站| 亚洲国产成人综合精品| 欧美激情综合色综合啪啪五月| 高清欧美性猛交XXXX黑人猛交| 亚洲AV无码一区二区三区DV| 免费人成视频网站在线18| 肥臀熟女一区二区三区| 亚洲另类春色国产精品| 琪琪电影网WWW888DVDC| 国产熟女乱子视频正在播放| 在线精品动漫一区二区无码| 色婷婷亚洲六月婷婷中文字幕| 韩日午夜在线资源一区二区| √最新版天堂资源网在线| 玩弄人妻少妇500系列视频| 久久人人做人人爽人人AV| 丰满人妻一区二区三区免费视频| 亚洲人成色777777在线观看| 日本55丰满熟妇厨房伦| 精品国产一区AV天美传媒 |