网上类似的jQuery和非jQuery表格控件都非常多了,由于项目需要,我没有时间去找一个合适且小巧的表格控件,于是自己写了一个。这个表格控件要实现如下的目标:

1.列宽可以控制

2.有鼠标滑过和选中高亮效果

3.标题列支持添加排序回调函数

4.事件绑定要少,因为表格数据是ajax动态加载的

5.支持多选

首先用HTML写一个表格框架,这里表扬一下特别好用的Zen Coding插件,写table.colortable>thead>tr>th*4就可以生成所需的结构

<table class="colortable">
<thead>
<tr>
<th>列A</th>
<th>列B</th>
<th>列C</th>
<th>列D</th>
</tr>
</thead>
</table>

表格内容是动态生成的,因此只需要一个tbody标签即可,但为了调试方便,先给两行数据。

<tbody>
<tr>
<td>1</td>
<td>2</td>
<td>3</td>
<td>4</td>
</tr>
<tr>
<td>5</td>
<td>6</td>
<td>7</td>
<td>8</td>
</tr>
</tbody>

接下来应用一些CSS让表格显得美观,由于要做成控件,因此CSS里也特意给每个选择器加了.colortable前缀。

.colortable{
    border-collapse:collapse;
    display:inline-block;
    cursor:pointer;
}
.colortable tr{
    text-align:center;
    height:22px;
    line-height:22px;
}
.colortable td,th{border:1px solid #000;overflow:hidden;}
.colortable th.orderA,th.orderD{
    background:url("../pic/common/table_order.gif") repeat-x;
    color:white;
}
.colortable tbody{background-color:#BBE0FF;}
.colortable thead{background:url("../pic/common/table.gif") repeat-x;}
.colortable tr.high{background-color:#E4F4FC;}
.colortable tr.sel,td.order{background-color:#39B2E4;color:white;}

这里用了两张图片:table_order.gif是在列排序时突出显示列标题用的,table.gif是标题的默认效果,我用了一点浅渐变来美化。

接下来就是js部分,考虑到要动态设置表格的列宽,因此要传入各个列宽的数组。再为排序传入两个回调函数。

代码的实现部分没有太多难度,主要就是在处理鼠标滑过效果时要区分该行是否被选中,选中行已经高亮了,就不需要滑过效果。

处理多选时也需要考虑两种情况,ctrl按下时是增加选择模式,不按则是单选模式。

处理排序时,对应每列的三种状态(默认,正序,逆序)要写三种不同的处理方式。

完整代码如下:

(function($){
$.fn.initTable=function(width,orderA,orderD){
var that=this;
        that.children("thead").click(function(event){
var target=event.target;
var index=$(target).index();
//干掉高亮行
            var sel=getByClass("sel","tr");
for(var i=0,len=sel.length;i<len;i++)
                sel[i].className="";
if(target.className==""){  //升序 
                //干掉其他th.order
                var thd=getByClass("order","td");
for(var i=0,len=thd.length;i<len;i++)
                    thd[i].className="";
                thd=getByClass("orderA","th");
if(thd.length>0)
                    thd[0].className="";
                thd=getByClass("orderD","th");
if(thd.length>0)
                    thd[0].className="";
//加类
                target.className="orderA";
                that.find("tbody tr").each(function(){
                    $(this).children("td").eq(index).attr("class","order");
                });
                orderA(index);
            }else if(target.className=="orderD"){  //取消排序
                target.className="";
var td=getByClass("order","td");
for(var i=0,len=td.length;i<len;i++)
                    td[i].className="";
            }else{  //降序
                target.className="orderD";
                orderD(index);
            }
        }).find("th").each(function(i){
this.style.width=width[i]+"px";
        }).end().end().children("tbody").click(function(event){
var target=event.target.parentNode;
if(event.ctrlKey) //多选
                target.className=(target.className=="sel")?"":"sel";
else{
if(target.className=="sel"){
var sel=getByClass("sel","tr");
for(var i=0,len=sel.length;i<len;i++)
                        sel[i].className="";
                }else{
var sel=getByClass("sel","tr");
for(var i=0,len=sel.length;i<len;i++)
                        sel[i].className="";
                    target.className="sel";
                }
            }
        }).mouseover(function(event){
var target=event.target.parentNode;
if(target.className!="sel")
                target.className="high";
        }).mouseout(function(event){
var target=event.target.parentNode;
if(target.className!="sel")
                target.className="";
        });
    };
})(jQuery);

代码里的getByClass是我写的按class查找元素的函数。运行效果如图:

jquery 表格样式代码 jquery表格控件_css

选中一行的效果

jquery 表格样式代码 jquery表格控件_jQuery_02

排序效果

jquery 表格样式代码 jquery表格控件_css_03

最后,设置表格内容不能直接设置tbody的innerHTML,因为IE不支持。replaceChild也不行,IE不支持。

直接createElement/appendChild方法可以。

用jQuery的.html()方法可以。

通过动态插入3000行比较性能,FF,OP,SAR,CHR两种方法都控制在100ms内。生成后,动态效果都很流畅。

IE6用第一种方法500ms,第二种方法1680ms。但是动态效果几乎瘫痪,反应极慢。诡异的是,目测下来,IE6对点击的反应要略快于滑过效果。但是我的滑过效果代码只有3行,没有任何查找和遍历,只改了一个class,而点击效果里还有一个全行遍历。我怀疑影响性能的是那个class切换导致css全部重新计算。因此我将滑过效果中的class切换去掉,改为直接赋值背景色,于是问题解决了,IE6中滑动流畅。
但是这影响了控件的独立性,我尝试恢复class切换,同时注释掉css里的规则,在js中改颜色。性能问题又重现了。可见性能瓶颈不是css计算,而在于class赋值本身。改用setAttribute赋值也无济于事。

好吧,一般的表格不会有3000行之巨,经测试,行数在500行之内,IE6的性能问题不明显,但快速滑动还是能看出高亮显示滞后,此时IE6生成表格的时间刚好降到100ms以内。行数继续降到100行,IE6才能达到Firefox的流畅度。这可能是因为我的Firefox是4.0的。

无论如何,让表格行数限制到100行,还是有点不人道。我也暂时想不到更好的解决办法,如果不考虑控件通用性,把颜色写到js里也未尝不可。