simonbengtsson / jsPDF-AutoTable

jsPDF plugin for generating PDF tables with javascript
https://simonbengtsson.github.io/jsPDF-AutoTable/
MIT License
2.33k stars 624 forks source link

rowspan and colspan in the table #311

Closed Genuris closed 6 years ago

Genuris commented 6 years ago

Can will I have use attributes in the table, for examples: rowspan, colspan? example issue https://github.com/simonbengtsson/jsPDF-AutoTable/issues/302 don`t helped me

TheOwl56 commented 6 years ago

Hi!

I've come with some sort of solution, at least for header with rowspan and colspan, I hope it helps you:

image

1st header's row column has rowspan 2. We use bootstrap table. here's table's settings:

`settingsBono: { striped: true, //pagination: true, //pageSize: 15, //pageList: [15, 25, 50, 100, 200], search: false, toolbarAlign: 'center', exportTypes: ['csv', 'excel'], formatShowingRows: function(pageFrom, pageTo, totalRows) { 'use strict'; return 'Mostrando ' + pageFrom + ' al ' + pageTo + ' de ' + totalRows + ' registros'; }, formatRecordsPerPage: function(pageNumber) { 'use strict'; return pageNumber + ' registros por página'; }, formatLoadingMessage: function() { 'use strict'; return 'Cargando, espere por favor...'; }, formatSearch: function() { 'use strict'; return 'Buscar'; }, formatNoMatches: function() { 'use strict'; return 'No se encontro información'; }, columns: [ [{ "field": "BANKER", "title": "BANKER", "colspan": 1, "rowspan": 2, "valign": "middle", }, { "title": "Franquicia GM Neta", "colspan": 5, "rowspan": 1,

  }, 
  {
    "title": "Franquicia GM Neta",
    "colspan": 5,
    "rowspan": 1,

  },
  {
    "title": "TAM",
    "colspan": 5,
    "rowspan": 1,

  }],
  [

// inicio de GM sin DCM
{ field: 'RATES_ACT', title: 'RATES', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'FOREX_ACT', title: 'FOREX', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'CREDIT_ACT', title: 'CREDIT', align: 'center', valign: 'middle', sortable: false, }, { field: 'EQUITY_ACT', title: 'EQUITY', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'TOTAL_ACT', title: 'TOTAL', align: 'center', valign: 'middle', sortable: false, visible: true, }, ///año anterior { field: 'RATES_ANT', title: 'RATES', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'FOREX_ANT', title: 'FOREX', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'CREDIT_ANT', title: 'CREDIT', align: 'center', valign: 'middle', sortable: false, }, { field: 'EQUITY_ANT', title: 'EQUITY', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'TOTAL_ANT', title: 'TOTAL', align: 'center', valign: 'middle', sortable: false, visible: true, },{ field: 'RATES_TAM', title: 'RATES', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'FOREX_TAM', title: 'FOREX', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'CREDIT_TAM', title: 'CREDIT', align: 'center', valign: 'middle', sortable: false, }, { field: 'EQUITY_TAM', title: 'EQUITY', align: 'center', valign: 'middle', sortable: false, visible: true, }, { field: 'TOTAL_TAM', title: 'TOTAL', align: 'center', valign: 'middle', sortable: false, visible: true, } ] ], data: [] }`

I just put a button to trigger a cutom method that retrieves table as htmlElement to overwrite tables header so autotable can draw the complete header, then it draws rects over that header with their colspan:

`crearPDFconTablas: function(tables){ 'use strict'; var imgData = ''; var doc = new jsPDF('l','pt'), tmpDate = new Date();

for(var j= 0; j<tables.length; j++){
  var currTbl = document.getElementById(tables[j]), tmpHeader = currTbl.tHead;
  var res = doc.autoTableHtmlToJson(currTbl), tmpRow, tmpHidden = $('#'+tables[j]).bootstrapTable('getHiddenRows');
  tmpHeader.fila0 = [];
  if (tmpHidden.length > 0) {
    tmpHidden.forEach(function(element){
      res.rows.push(Object.values(element));
    });
  }
  if(tmpHeader.rows.length > 1){
    var fila1, fila2 = 0, elapsedH = 0;
    var header = [];//= tmpHeader.rows[0] > tmpHeader.rows[1]  ? tmpHeader.rows[0] : tmpHeader.rows[1];
     for(fila1 in tmpHeader.rows[0].cells){
       if (fila1 === tmpHeader.rows[0].cells.length)
         break;

       console.info('en cell: ' + tmpHeader.rows[0].cells[fila1].innerText);
       console.dir(tmpHeader.rows[0].cells[fila1]);
       if(tmpHeader.rows[0].cells[fila1].colSpan > 1){
         for(var lclfila2 = 0; lclfila2 < tmpHeader.rows[0].cells[fila1].colSpan; lclfila2++){
           header.push(tmpHeader.rows[1].cells[fila2].innerText);
           fila2++;
         }
       }else if (tmpHeader.rows[0].cells[fila1].colSpan === 1){
         header.push(fila2=== 0 ? tmpHeader.rows[0].cells[fila1].innerText : tmpHeader.rows[1].cells[fila2].innerText);
         fila2 += fila2 === 0 ? 0 : 1;
       }
     }
     res.columns = header;
     res.rows = [];
     for(var j = 2; j < currTbl.rows.length; j++){
     var tmpFila=[]
       for(var tmpCell in currTbl.rows[j].cells) {
         tmpFila.push(currTbl.rows[j].cells[tmpCell].innerText);
       };
       res.rows.push(tmpFila);
     }
  }
  doc.autoTable(res.columns, res.rows,  {
    theme: 'striped',
    addPageContent:  function(data){
      var text = 'Este documento es propiedad de BBVA [generado por: ' + sessionStorage.NB_CORREO+']',
          tblName = currTbl.attributes.name ? $('#'+tables[j]).attr('name') : '';
       doc.setTextColor(7, 33, 70);
       doc.setFontSize(18);
       doc.addImage(imgData, 'JPEG', -5, -25, 150, 90);
       doc.text(0, 75,tblName);
       doc.setTextColor(189,189,189);
       doc.text(10, doc.internal.pageSize.height - 20, text);
       doc.setFontSize(10);
    },
    drawHeaderCell: function(cell, data){
      doc.setFillColor(0, 68, 129);
      doc.setTextColor(255,255,255);
      if(data.row.raw.length === data.column.index + 1){
        if(tmpHeader.rows.length > 1){
         var tmpCell, headerIdx = 0;
         for(tmpCell = 0 ; tmpCell < tmpHeader.rows[0].cells.length; tmpCell++){
           if(tmpHeader.rows[0].cells[tmpCell].colSpan === 1 ){
            doc.setFillColor(0, 68, 129);
            doc.setTextColor(255,255,255);
            doc.rect(data.row.cells[headerIdx].x, data.row.cells[headerIdx].y - 20, data.row.cells[headerIdx].width, 20, 'DF');
            doc.autoTableText(header[tmpCell] === tmpHeader.rows[0].cells[tmpCell].innerText ? '' : tmpHeader.rows[0].cells[tmpCell].innerText 
                , data.row.cells[headerIdx].x + data.row.cells[headerIdx].width /2, data.row.cells[headerIdx].y-20 + data.row.height / 2, {
                  halign: 'center',
                  valign: 'middle'
               });
            headerIdx++;
           }else{
             var widthAcumulado= 0, xdeColSpan = data.row.cells[headerIdx].x, txtAcumulado = tmpHeader.rows[0].cells[tmpCell].innerText,
                 lclIdx;
             for(lclIdx = 0 ; lclIdx < tmpHeader.rows[0].cells[tmpCell].colSpan; lclIdx++,headerIdx++){
               widthAcumulado += data.row.cells[headerIdx].width;
             }
             doc.setFillColor(0, 68, 129);
             doc.setTextColor(255,255,255);
             doc.rect(xdeColSpan, data.row.cells[headerIdx-1].y - 20, widthAcumulado, 20, 'DF');
             doc.autoTableText(txtAcumulado, xdeColSpan + widthAcumulado /2, data.row.cells[headerIdx-1].y-20 + data.row.height / 2, {
                halign: 'center',
                valign: 'middle'
             });
           }

         }
        }
      }
      doc.setFillColor(0, 68, 129);
      doc.setTextColor(255,255,255);
    },
    margin: { top: 90, horizontal:20 },
    styles: {
      fontSize: 8,
      lineWidth: 0.75,
      overflow: 'linebreak',
    },
    columnStyles:{
      text: {halign: 'right'}
    },

  });
  if (j !== tables.length-1 && j >= 0)
    doc.addPage()
 }

  doc.save(sessionStorage.currentPage+'_'+tmpDate.toLocaleString('en-GB').split(',')[0].toString().replace('/','_').replace('/','_') + ".pdf");

}`

pdf output: image