首页 > 语言 > JavaScript > 正文

HTML+CSS+JS实现完美兼容各大浏览器的TABLE固定列

2024-05-06 16:18:44
字体:
来源:转载
供稿:网友

本文给大家分享的是使用HTML+CSS+JS实现完美兼容各大浏览器的TABLE固定列的方法和示例,非常的实用,特别是在BS架构的企业级应用,有需要的小伙伴可以参考下。

BS架构的企业级应用中,当一个表格列数较多时,用户一个常见的需求就是把前面几个重要的列固定住,这样拖动滚动条时固定的列会方便用户查看数据,用户体验很好。一些重量级的JS组件库也都有这个功能,那么有没有更简单的方法实现这个功能呢?

这个需求常见的解决方案是使用表格拼接的方法,这个方案如果要制作静态的网页,或者功能简单的动态页面,逻辑比较简单,技术上也不复杂,很容易实现,但是如果要做成组件,动态功能较多的话,就需要写大量的冗余代码,难以维护,甚至于一个简单的功能,都需要写很多的代码,比如事件处理等,这个方法就显得比较笨拙,灵活性很差,不是一个好的方案。

经过长时间的分析研究,各种场景的试验,我们找到了一个兼容性非常好的解决方案,总体上来讲采用的是定位计算的方法,下面贴出代码,然后做个解读。

 

 
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
  2. <html xmlns="http://www.w3.org/1999/xhtml"> 
  3. <head> 
  4. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
  5. <title>无标题文档</title> 
  6. <script type="text/javascript"> 
  7. function divScroll(scrollDiv){ 
  8. var scrollLeft = scrollDiv.scrollLeft; 
  9. document.getElementById("tableDiv_title").scrollLeft = scrollLeft; 
  10. document.getElementById("tableDiv_body").scrollLeft = scrollLeft;  
  11. function divYScroll(scrollYDiv){ 
  12. var scrollTop = scrollYDiv.scrollTop; 
  13. document.getElementById("tableDiv_y").scrollTop = scrollTop;  
  14. function onwheel(event){ 
  15. var evt = event||window.event; 
  16. var bodyDivY = document.getElementById("tableDiv_y"); 
  17. var scrollDivY = document.getElementById("scrollDiv_y"); 
  18. if (bodyDivY.scrollHeight>bodyDivY.offsetHeight){ 
  19. if (evt.deltaY){ 
  20. bodyDivYbodyDivY.scrollTop = bodyDivY.scrollTop + evt.deltaY*7; 
  21. scrollDivYscrollDivY.scrollTop = scrollDivY.scrollTop + evt.deltaY*7; 
  22. }else{ 
  23. bodyDivYbodyDivY.scrollTop = bodyDivY.scrollTop - evt.wheelDelta/5; 
  24. scrollDivYscrollDivY.scrollTop = scrollDivY.scrollTop - evt.wheelDelta/5; 
  25. </script> 
  26. <style type="text/css"> 
  27. body { 
  28. margin:0; 
  29. padding:0; 
  30. table { 
  31. border-collapse:collapse; 
  32. border:0; 
  33. border:none; 
  34.  
  35. table td { 
  36. border:1px solid #000; 
  37. overflow:hidden; 
  38. padding:0 2px; 
  39. </style> 
  40. </head> 
  41. <body> 
  42. <div style="width:500px; position:relative; padding-right:18px;"> 
  43. <div style="position:relative;height:368px;overflow:hidden;width:100%"> 
  44. <div style="padding-left:108px; width:auto; overflow:hidden; background:#f00;" id="tableDiv_title" > 
  45. <table border="0" cellspacing="0" cellpadding="0" > 
  46. <tr> 
  47. <td style="min-width:30px; max-width:30px; left:0; top:0; width:30px; overflow:hidden; background-color:#f00;position:absolute;z-index:1;">000</td> 
  48. <td style="min-width:74px; max-width:74px; left:30px; top:0; width:74px; overflow:hidden; background-color:#f00;position:absolute;z-index:1;">自动表格</td> 
  49. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  50. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  51. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  52. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  53. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  54. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  55. </tr> 
  56. </table> 
  57. </div>  
  58. <div style="overflow:hidden; position:absolute;height:128px; width:100%;" id="tableDiv_y" onmousewheel="onwheel(event);" onwheel="onwheel(event);"> 
  59. <div style="padding-left:108px; width:auto;overflow:hidden;" id="tableDiv_body"> 
  60. <table border="0" cellspacing="0" cellpadding="0" > 
  61. <tr> 
  62. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">001</td> 
  63. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  64. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  65. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  66. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  67. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  68. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  69. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  70. </tr> 
  71. <tr> 
  72. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff; position:absolute; z-index:1;">002</td> 
  73. <td style="min-width:74px; max-width:74px; left:30px; width:74px; overflow:hidden;background-color:#fff; position:absolute; z-index:1;">自动表格</td> 
  74. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  75. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  76. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  77. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  78. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  79. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  80. </tr> 
  81. <tr> 
  82. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">003</td> 
  83. <td style="min-width:74px; max-width:74px;left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  84. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  85. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  86. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  87. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  88. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  89. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  90. </tr> 
  91. <tr> 
  92. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">004</td> 
  93. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  94. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  95. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  96. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  97. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  98. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  99. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  100. </tr> 
  101. <tr> 
  102. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">005</td> 
  103. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  104. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  105. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  106. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  107. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  108. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  109. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  110. </tr> 
  111. <tr> 
  112. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">006</td> 
  113. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  114. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  115. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  116. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  117. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  118. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  119. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  120. </tr>  
  121. <tr> 
  122. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">007</td> 
  123. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  124. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  125. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  126. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  127. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  128. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  129. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  130. </tr> 
  131. <tr> 
  132. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">008</td> 
  133. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  134. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  135. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  136. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  137. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  138. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  139. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  140. </tr> 
  141. <tr> 
  142. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">009</td> 
  143. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  144. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  145. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  146. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  147. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  148. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  149. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  150. </tr> 
  151. <tr> 
  152. <td style="min-width:30px; max-width:30px; left:0; width:30px; overflow:hidden; background-color:#fff;position:absolute;z-index:1;">010</td> 
  153. <td style="min-width:74px; max-width:74px; left: 30px; width: 74px; overflow:hidden;background-color:#fff;position:absolute;z-index:1;">自动表格</td> 
  154. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  155. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  156. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  157. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  158. <td style="min-width: 100px; max-width: 100px; width: 100px;" >56454自动</td> 
  159. <td style="min-width: 100px; max-width: 100px; width: 100px;" >最后一列</td> 
  160. </tr>  
  161. </table> 
  162. </div>  
  163. </div>  
  164. <div style="background-color:#eee;overflow:hidden;top:150px; width:100%; z-index:2;position:absolute;"> 
  165. <div style="margin-left:108px; width:auto;overflow-x:scroll;overflow-y:hidden;" onscroll='divScroll(this);'> 
  166. <div style="width:630px; height:1px;"></div> 
  167. </div> 
  168. </div> 
  169. </div> 
  170. <div id="scrollDiv_y" style="display:block; overflow-x:hidden; overflow-y:scroll; position:absolute; top:22px; right:0px; height:118px; padding-bottom:10px;" onscroll='divYScroll(this);'> 
  171. <div style="width:1px; height:194px;"></div> 
  172. </div> 
  173. </div> 
  174. </div> 
  175. </body> 
  176. </html> 

一、总体结构:

页面基本元素为DIV+TABLE,固定的列采用绝对定位的方式固定,每一列都要指定固定宽度,为了解决横竖滚动条的问题,表头和表体的外面分别包裹两层DIV,滚动条采用虚拟的方式,固定在固定位置通过JS控制模拟正常DIV滚动条的效果。

二、定位:

固定的列要绝对定位,通过left属性控制左侧位移,为了保证固定列浮动在上方,设置z-index为1,。为了保证有竖滚动条时的正常显示,表体的外层DIV为绝对定位,由此导致滚动条也都要绝对定位。还有,表头和表体以及滚动条的内层DIV通过margin-left属性控制左侧外边距,把固定列的偏移量空余出来。

二、宽度计算:

每一列的宽度都要指定固定的值,并且要注意一个关键点,就是还要加上min-width和max-width属性,这两个属性和width值相等,表头表体的内层DIV,宽度为auto,自适应表格宽度,外层DIV宽度为100%,最外层的DIV通过padding-right属性控制右侧内边距,将竖滚动条的位置空余出来。

三、高度计算:

因为绝对定位的存在,整个表格组件的高度要指定,可以通过计算得出,竖滚动条的top值也需要进行计算。

四、滚动条:

本方案一个突出特点,就是虚拟的滚动条,就是通过一个和表格一样宽、高度为一个像素的DIV模拟出表体DIV的横向滚动条,竖滚动条同理。之所以采用这个形式,一个是横向滚动条这样处理比较美观,竖滚动条这样处理之后,表头和表体的外层DIV宽度不用计算了,都为100%,否则存在滚动条时,表头和横向滚动条要空出竖滚动条宽度的位移,否则无法对齐,这个计算倒不复杂,但是某些情况下存在问题,在此不展开了。

五、滚动事件:

因为表体的滚动条都隐藏了,导致鼠标滚轮不起作用了,这样就需要用JS处理鼠标滚轮事件,本文的样例代码兼容常见浏览器。这里的重点是同时写了onmousewheel和onwheel事件,onmousewheel兼容IE,在计算滚动距离时,注意deltaY和wheelDelta属性的差异即可。

六、优缺点分析:

本文的解决方案已经经过精简,重点是讲清楚原理,在我们的实际中,非常的复杂。这个设计考虑了非常多的兼容性,包括了浏览器的兼容性和各种场景的兼容性,如果需求简单,还有简化的空间。

这个方案的优点是,如果要做组件的话,因为HTML结构简单,表头和表体都是一个TABLE,JS控制代码非常干净,维护容易。缺点就是计算过多。我们认为该方案比较适用于开发组件的情况,静态页面就有点小题大作了。

以上所述就是本文的全部内容了,希望大家能够喜欢。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表

图片精选