【Javascript】Javascript橫向/縱向合並單元格TD


> 需求是這樣滴(>_<)

在報表系統中,涉及“HTML的TD單元格的合並”恐怕為數不少。

比如,從DB查得數據並經過后台的整理后,可能是這樣的:

Table1

    JOB TOTAL SAL INDEX EMPNO ENAME JOB MGR HIREDATE SAL COMM DEPTNO
1 ANALYST 6000 1 7788 SCOTT ANALYST 7566 4/19/1987 3000.00   20
1 ANALYST 6000 2 7902 FORD ANALYST 7566 12/3/1981 3000.00   20
2 CLERK 4150 3 7934 MILLER CLERK 7782 1/23/1982 1300.00   10
2 CLERK 4150 4 7900 JAMES CLERK 7698 12/3/1981 950.00   30
2 CLERK 4150 5 7369 SMITH CLERK 7902 12/17/1980 800.00   20
2 CLERK 4150 6 7876 ADAMS CLERK 7788 5/23/1987 1100.00   20
3 MANAGER 8275 7 7698 BLAKE MANAGER 7839 5/1/1981 2850.00   30
3 MANAGER 8275 8 7566 JONES MANAGER 7839 4/2/1981 2975.00   20
3 MANAGER 8275 9 7782 CLARK MANAGER 7839 6/9/1981 2450.00   10
4 PRESIDENT 5000 10 7839 KING PRESIDENT   11/17/1981 5000.00   10
5 SALESMAN 5600 11 7844 TURNER SALESMAN 7698 9/8/1981 1500.00 0.00 30
5 SALESMAN 5600 12 7654 MARTIN SALESMAN 7698 9/28/1981 1250.00 1400.00 30
5 SALESMAN 5600 13 7521 WARD SALESMAN 7698 2/22/1981 1250.00 500.00 30
5 SALESMAN 5600 14 7499 ALLEN SALESMAN 7698 2/20/1981 1600.00 300.00 30

 

為了讓用戶更清晰地查看報表,結果可能需要是這樣的(各職位的薪水總額及分布明細):

Table2

    JOB TOTAL SAL INDEX EMPNO ENAME JOB MGR HIREDATE SAL COMM DEPTNO
1 ANALYST 6000 1 7788 SCOTT ANALYST 7566 4/19/1987 3000.00   20
2 7902 FORD ANALYST 7566 12/3/1981 3000.00   20
2 CLERK 4150 3 7934 MILLER CLERK 7782 1/23/1982 1300.00   10
4 7900 JAMES CLERK 7698 12/3/1981 950.00   30
5 7369 SMITH CLERK 7902 12/17/1980 800.00   20
6 7876 ADAMS CLERK 7788 5/23/1987 1100.00   20
3 MANAGER 8275 7 7698 BLAKE MANAGER 7839 5/1/1981 2850.00   30
8 7566 JONES MANAGER 7839 4/2/1981 2975.00   20
9 7782 CLARK MANAGER 7839 6/9/1981 2450.00   10
4 PRESIDENT 5000 10 7839 KING PRESIDENT   11/17/1981 5000.00   10
5 SALESMAN 5600 11 7844 TURNER SALESMAN 7698 9/8/1981 1500.00 0.00 30
12 7654 MARTIN SALESMAN 7698 9/28/1981 1250.00 1400.00 30
13 7521 WARD SALESMAN 7698 2/22/1981 1250.00 500.00 30
14 7499 ALLEN SALESMAN 7698 2/20/1981 1600.00 300.00 30

 

 這就需要我們對單元格進行相應的合並。

一般來說,有兩種處理方法,

  1. 在編寫HTML時已設置單元格的合並;
  2. 在編寫HTML時未作合並處理,后期由Javascript完成單元格合並工作。

本文講的是第2種情況。

 

合並單元格的步驟

比如,我要把上表Table1中數值為ANALYST的兩個單元格合並。

JOB
ANALYST
ANALYST
  • 我們首先需要將第二個ANALYST的單元格刪除掉
  • 然后設置第一個ANALYST的單元格的rowSpan屬性為2(表示“跨越2行”)

 

> 寫得不好的公用方法

之前寫了一個公用Javascript方法,用以合並相鄰間文本相同的單元格。

這個方法寫得不好,有明顯的Bug。對已做過合並的Table,這個方法有可能導致合並單元格錯誤。

主要的原因在於,對於合並過單元格的表格,那么被合並的行就會相應的少一個TD,而此方法依舊按未合並過單元格的情況來獲取TD對象。

比如,獲取第二列的所有單元格,此程序用“獲取全部TR中的第二個TD”來實現,這在未合並過單元格的Table是正確的,但對於合並過單元格的Table,則未必,因為被合並的行相應地少一個單元格。

此處將代碼貼出,希望大家引以為鑒

function mergeCell(tableObj, col) {
        var $tab = $(tableObj);
        var $trs = $tab.find("tr");
        
        var oldval;
        var firstTD;
        var counter = 0;
        $trs.each(function(index) {
            
            if (!oldval && !firstTD) {
                oldval = $(this).find("td:eq(" + col + ")").text();
                firstTD = $(this).find("td:eq(" + col + ")").get(0);
                counter = 0;
                counter++;
            } else {
                if ($(this).find("td:eq(" + col + ")").text() == oldval) {
                    $(this).find("td:eq(" + col + ")").remove();
                    counter++;
                } else {
                    $(firstTD).attr("rowSpan", counter);
                    oldval = $(this).find("td:eq(" + col + ")").text();
                    firstTD = $(this).find("td:eq(" + col + ")").get(0);
                    counter = 0;
                    counter++;
                }
            }
            
            if (index >= $trs.length - 1) {
                $(firstTD).attr("rowSpan", counter);
            }
            
        });
    }
mergeCell

 

 頁面DEMO如下(HTML代碼,較長):

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Insert title here</title>
</head>

<script type="text/javascript" src="../js/jquery.js"></script>
<script type="text/javascript" src="../js/tdmerger.js"></script>
<script type="text/javascript">
    $().ready(function() {

        mergeCell($("#list_table").get(0), 1);
        mergeCell($("#list_table").get(0), 2);
        
    });
    
    function mergeCell(tableObj, col) {
        var $tab = $(tableObj);
        var $trs = $tab.find("tr");
        
        var oldval;
        var firstTD;
        var counter = 0;
        $trs.each(function(index) {
            
            if (!oldval && !firstTD) {
                oldval = $(this).find("td:eq(" + col + ")").text();
                firstTD = $(this).find("td:eq(" + col + ")").get(0);
                counter = 0;
                counter++;
            } else {
                if ($(this).find("td:eq(" + col + ")").text() == oldval) {
                    $(this).find("td:eq(" + col + ")").remove();
                    counter++;
                } else {
                    $(firstTD).attr("rowSpan", counter);
                    oldval = $(this).find("td:eq(" + col + ")").text();
                    firstTD = $(this).find("td:eq(" + col + ")").get(0);
                    counter = 0;
                    counter++;
                }
            }
            
            if (index >= $trs.length - 1) {
                $(firstTD).attr("rowSpan", counter);
            }
            
        });
    }

</script>

<body>

    <TABLE id="list_table" BORDER="1">
        <TR>

            <TH>&nbsp;&nbsp;&nbsp;</TH>
            <TH>JOB</TH>
            <TH>TOTAL SAL</TH>

            <TH>INDEX</TH>
            <TH>EMPNO</TH>
            <TH>ENAME</TH>
            <TH>JOB</TH>
            <TH>MGR</TH>
            <TH>HIREDATE</TH>
            <TH>SAL</TH>
            <TH>COMM</TH>
            <TH>DEPTNO</TH>
        </TR>
        <TR>
            <TD>1</TD>
            <TD>ANALYST</TD>
            <TD>6000</TD>
            <TD>1</TD>
            <TD>7788</TD>
            <TD>SCOTT</TD>
            <TD>ANALYST</TD>
            <TD>7566</TD>
            <TD>4/19/1987</TD>
            <TD>3000.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>1</TD>
            <TD>ANALYST</TD>
            <TD>6000</TD>
            <TD>2</TD>
            <TD>7902</TD>
            <TD>FORD</TD>
            <TD>ANALYST</TD>
            <TD>7566</TD>
            <TD>12/3/1981</TD>
            <TD>3000.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>3</TD>
            <TD>7934</TD>
            <TD>MILLER</TD>
            <TD>CLERK</TD>
            <TD>7782</TD>
            <TD>1/23/1982</TD>
            <TD>1300.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>4</TD>
            <TD>7900</TD>
            <TD>JAMES</TD>
            <TD>CLERK</TD>
            <TD>7698</TD>
            <TD>12/3/1981</TD>
            <TD>950.00</TD>
            <TD>&nbsp;</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>5</TD>
            <TD>7369</TD>
            <TD>SMITH</TD>
            <TD>CLERK</TD>
            <TD>7902</TD>
            <TD>12/17/1980</TD>
            <TD>800.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>6</TD>
            <TD>7876</TD>
            <TD>ADAMS</TD>
            <TD>CLERK</TD>
            <TD>7788</TD>
            <TD>5/23/1987</TD>
            <TD>1100.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>7</TD>
            <TD>7698</TD>
            <TD>BLAKE</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>5/1/1981</TD>
            <TD>2850.00</TD>
            <TD>&nbsp;</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>8</TD>
            <TD>7566</TD>
            <TD>JONES</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>4/2/1981</TD>
            <TD>2975.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>9</TD>
            <TD>7782</TD>
            <TD>CLARK</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>6/9/1981</TD>
            <TD>2450.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>4</TD>
            <TD>PRESIDENT</TD>
            <TD>5000</TD>
            <TD>10</TD>
            <TD>7839</TD>
            <TD>KING</TD>
            <TD>PRESIDENT</TD>
            <TD>&nbsp;</TD>
            <TD>11/17/1981</TD>
            <TD>5000.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>4</TD>
            <TD>SUPER PRESIDENT</TD>
            <TD>5000</TD>
            <TD>10</TD>
            <TD>7839</TD>
            <TD>KING</TD>
            <TD>PRESIDENT</TD>
            <TD>&nbsp;</TD>
            <TD>11/17/1981</TD>
            <TD>5000.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>11</TD>
            <TD>7844</TD>
            <TD>TURNER</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>9/8/1981</TD>
            <TD>1500.00</TD>
            <TD>0.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>12</TD>
            <TD>7654</TD>
            <TD>MARTIN</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>9/28/1981</TD>
            <TD>1250.00</TD>
            <TD>1400.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>13</TD>
            <TD>7521</TD>
            <TD>WARD</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>2/22/1981</TD>
            <TD>1250.00</TD>
            <TD>500.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>14</TD>
            <TD>7499</TD>
            <TD>ALLEN</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>2/20/1981</TD>
            <TD>1600.00</TD>
            <TD>300.00</TD>
            <TD>30</TD>
        </TR>
    </TABLE>

</body>
</html>
WrongMergeTD.html

 

> 寫得不好怎么辦?卷土重來唄

於是,重新寫了兩個公用方法來合並單元格。

一個是合並相鄰值相同的單元格;

另外一個是合並指定序號的單元格,比如2-5,6-8,是“合並第2個至第5個單元格、第6個和至第8個單元格”。

這倆方法將“獲取需要合並的TD元素”的動作交由方法調用者,只負責對入參單元格作對應的合並。

JS

/**
 * Meger cells with same text
 * @param $tds TDs jquery object
 * @param type row/col
 */
function mergeCell4SameText($tds, type) {
    
    var oldval;
    var firstTD;
    var counter = 0;
    $tds.each(function(index) {
        
        if (index == 0) {
            oldval = $(this).text();
            firstTD = $(this).get(0);
            counter = 0;
            counter++;
        } else {
            if ($(this).text() == oldval) {
                $(this).remove();
                counter++;
            } else {
                if (type == 'col') {
                    $(firstTD).attr("rowSpan", counter);
                } else if (type == 'row') {
                    $(firstTD).attr("colSpan", counter);
                }
                
                oldval = $(this).text();
                firstTD = $(this).get(0);
                counter = 0;
                counter++;
            }
        }
        
        if (index >= $tds.length - 1) {
            
            if (type == 'col') {
                $(firstTD).attr("rowSpan", counter);
            } else if (type == 'row') {
                $(firstTD).attr("colSpan", counter);
            }
            
        }
        
    });
}

/**
 * Meger cells by the parameters "index"
 * @param $tds TDs jquery object
 * @param type row/col
 * @param index for example, 0-1,2-5,6-8,10-13
 */
function mergeCellByIndex($tds, type, index) {
    
    var indexArrs = index.split(',');
    
    var fromTo;
    var from;
    var to;
    
    for (var i in indexArrs) {
        fromTo = indexArrs[i];
        
        from = new Number(fromTo.split('-')[0]);
        to = new Number(fromTo.split('-')[1]);
        
        for (var j = 1 + from; j <= to; j++) {
            $($tds.get(j)).remove();
        }
        
        if (type == 'col') {
            $($tds.get(from)).attr("rowSpan", to - from + 1);
        } else if (type == 'row') {
            $($tds.get(from)).attr("colSpan", to - from + 1);
        }
        
    }
    
}
tdmerger

 

調用的頁面(HTML代碼,較長):

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Insert title here</title>
</head>

<script type="text/javascript" src="../js/jquery.js"></script>
<script type="text/javascript" src="../js/tdmerger.js"></script>
<script type="text/javascript">
    $().ready(function() {

        $col1tds = $("table tr td:nth-child(1)");
        $col2tds = $("table tr td:nth-child(2)");
        $col3tds = $("table tr td:nth-child(3)");
        
        mergeCellByIndex($col1tds, 'col', '0-1,2-5,6-8,10-13');
        mergeCellByIndex($col2tds, 'col', '0-1,2-5,6-8,10-13');
        mergeCellByIndex($col3tds, 'col', '0-1,2-5,6-8,10-13');
        
        /*
        $row2tds = $("table tr:eq(2) td");
        mergeCellByIndex($row2tds, 'row', '6-11');
        */

    });

</script>

<body>

    <TABLE BORDER="1">
        <TR>

            <TH>&nbsp;&nbsp;&nbsp;</TH>
            <TH>JOB</TH>
            <TH>TOTAL SAL</TH>

            <TH>INDEX</TH>
            <TH>EMPNO</TH>
            <TH>ENAME</TH>
            <TH>JOB</TH>
            <TH>MGR</TH>
            <TH>HIREDATE</TH>
            <TH>SAL</TH>
            <TH>COMM</TH>
            <TH>DEPTNO</TH>
        </TR>
        <TR>
            <TD>1</TD>
            <TD>ANALYST</TD>
            <TD>6000</TD>
            <TD>1</TD>
            <TD>7788</TD>
            <TD>SCOTT</TD>
            <TD>ANALYST</TD>
            <TD>7566</TD>
            <TD>4/19/1987</TD>
            <TD>3000.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>1</TD>
            <TD>ANALYST</TD>
            <TD>6000</TD>
            <TD>2</TD>
            <TD>7902</TD>
            <TD>FORD</TD>
            <TD>ANALYST</TD>
            <TD>7566</TD>
            <TD>12/3/1981</TD>
            <TD>3000.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>3</TD>
            <TD>7934</TD>
            <TD>MILLER</TD>
            <TD>CLERK</TD>
            <TD>7782</TD>
            <TD>1/23/1982</TD>
            <TD>1300.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>4</TD>
            <TD>7900</TD>
            <TD>JAMES</TD>
            <TD>CLERK</TD>
            <TD>7698</TD>
            <TD>12/3/1981</TD>
            <TD>950.00</TD>
            <TD>&nbsp;</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>5</TD>
            <TD>7369</TD>
            <TD>SMITH</TD>
            <TD>CLERK</TD>
            <TD>7902</TD>
            <TD>12/17/1980</TD>
            <TD>800.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>2</TD>
            <TD>CLERK</TD>
            <TD>4150</TD>
            <TD>6</TD>
            <TD>7876</TD>
            <TD>ADAMS</TD>
            <TD>CLERK</TD>
            <TD>7788</TD>
            <TD>5/23/1987</TD>
            <TD>1100.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>7</TD>
            <TD>7698</TD>
            <TD>BLAKE</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>5/1/1981</TD>
            <TD>2850.00</TD>
            <TD>&nbsp;</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>8</TD>
            <TD>7566</TD>
            <TD>JONES</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>4/2/1981</TD>
            <TD>2975.00</TD>
            <TD>&nbsp;</TD>
            <TD>20</TD>
        </TR>
        <TR>
            <TD>3</TD>
            <TD>MANAGER</TD>
            <TD>8275</TD>
            <TD>9</TD>
            <TD>7782</TD>
            <TD>CLARK</TD>
            <TD>MANAGER</TD>
            <TD>7839</TD>
            <TD>6/9/1981</TD>
            <TD>2450.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>4</TD>
            <TD>PRESIDENT</TD>
            <TD>5000</TD>
            <TD>10</TD>
            <TD>7839</TD>
            <TD>KING</TD>
            <TD>PRESIDENT</TD>
            <TD>&nbsp;</TD>
            <TD>11/17/1981</TD>
            <TD>5000.00</TD>
            <TD>&nbsp;</TD>
            <TD>10</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>11</TD>
            <TD>7844</TD>
            <TD>TURNER</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>9/8/1981</TD>
            <TD>1500.00</TD>
            <TD>0.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>12</TD>
            <TD>7654</TD>
            <TD>MARTIN</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>9/28/1981</TD>
            <TD>1250.00</TD>
            <TD>1400.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>13</TD>
            <TD>7521</TD>
            <TD>WARD</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>2/22/1981</TD>
            <TD>1250.00</TD>
            <TD>500.00</TD>
            <TD>30</TD>
        </TR>
        <TR>
            <TD>5</TD>
            <TD>SALESMAN</TD>
            <TD>5600</TD>
            <TD>14</TD>
            <TD>7499</TD>
            <TD>ALLEN</TD>
            <TD>SALESMAN</TD>
            <TD>7698</TD>
            <TD>2/20/1981</TD>
            <TD>1600.00</TD>
            <TD>300.00</TD>
            <TD>30</TD>
        </TR>
    </TABLE>

</body>
</html>
mergeTD

 

對於合並單元格,這兩種方式都不太滿意,可是,也想不到更好的方法對此操作作封裝。

如果童靴有更好的方法,請指點!!

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM