简体   繁体   English

如果 Table 设置为 Responsive,则 Bootstrap DataTable Sparkline 不起作用

[英]Bootstrap DataTable Sparkline not working if Table set to Responsive

I have a Bootstrap DataTable with a sparkline in the last column, here is the full js:我在最后一列中有一个带有迷你图的 Bootstrap DataTable,这是完整的 js:

$(document).ready(function() {
  var groupColumn = 0;

  let table = $('#example').DataTable({
    //responsive: true,
    autoWidth: true,
    processing: true,
    ordering: true,
    scrollY: '50vh',
    scrollCollapse: true,
    paging: false,
    searching: true,

    ajax: {
      url: "api/ApartmentsAvailables",
      type: "GET",
      contentType: "application/json; charset=utf-8",
      dataType: "json",
    },
    columnDefs: [{
        visible: false,
        targets: groupColumn
      },
      {
        targets: 7,
        render: DataTable.render.datetime('YYYY-MM-DDT00:00:00', 'MMMM D, YYYY', 'en'),
      },
      {
        responsivePriority: 1,
        targets: 0
      },
    ],
    order: [
      [groupColumn, 'asc']
    ],

    drawCallback: function(settings) {
      $('.sparkline')
        .map(function() {
          return $('canvas', this).length ? null : this;
        })
        .sparkline('html', {
          type: 'line',
          width: '250px'
        })

      var api = this.api();
      var rows = api.rows({
        page: 'current'
      }).nodes();
      var last = null;

      api
        .column(groupColumn, {
          page: 'current'
        })
        .data()
        .each(function(group, i) {
          if (last !== group) {
            $(rows)
              .eq(i)
              .before('<tr class="group" style="background-color:DarkGray; text-align:center;font-weight: bold; color:white;"><td  colspan="8">' + group + '</td></tr>');

            last = group;
          }
        })
    },
    columns: [

      {
        data: "building"
      },
      {
        data: "floor_Plan"
      },
      {
        data: "apt_Number"
      },
      {
        data: "rent"
      },
      {
        data: "bedrooms"
      },
      {
        data: "bathrooms"
      },
      {
        data: "sqft"
      },
      {
        data: "available_Date"
      },
      {
        data: 'prices',
        render: function(data, type, row, meta) {

          return type === 'display' ?
            '<span class="sparkline">' + data.toString() + '</span>' :
            data;
        }
      },
    ]
  });
  new $.fn.dataTable.FixedHeader(table);
  // Order by the grouping
  $('#example tbody').on('click', 'tr.group', function() {
    var currentOrder = table.order()[0];
    if (currentOrder[0] === groupColumn && currentOrder[1] === 'asc') {
      table.order([groupColumn, 'desc']).draw();
    } else {
      table.order([groupColumn, 'asc']).draw();
    }
  });
});

The problem occurs when I enable responsive: true , the sparkline column becomes hidden and when I click to expand the row to show the hidden columns it shows the whole array of Value and not the sparkline.当我启用responsive: true时出现问题,迷你图列变为隐藏,当我单击展开行以显示隐藏列时,它显示整个值数组而不是迷你图。

I guess that the我猜那个

drawCallback: function (settings) {
            $('.sparkline')
                .map(function () {
                    return $('canvas', this).length ? null : this;
                })
                .sparkline('html', {
                    type: 'line',
                    width: '250px'
                })

is not able to be applied to a column that is hidden.不能应用于隐藏的列。

Without the responsive option the HTML generated for the td is:如果没有响应选项,为 td 生成的 HTML 是:

<td>
    <span class="sparkline">
        <canvas style="display: inline-block; width: 250px; height: 21px; vertical-align: top;"
                width="250"
                height="21"/>
    </span>
</td>

With the responsive set to true:将响应设置为 true:

<td style="display: none;"
    class="dtr-hidden">
    <span class="sparkline">3446,3446,3416,3416,3416,3546,3546,3546,3546,3546,3546,3561,3556,3551,3396,3396,3396,3346,3306,3306,3306</span>
</td>

I presume that I should somehow capture the mouse click on the expand icon and then re-inject the canvas but I don't know how to do that.我想我应该以某种方式捕获鼠标单击展开图标,然后重新注入 canvas 但我不知道该怎么做。

There is an event you can use for that, provided as part of the DataTables Responsive extension:您可以使用一个事件,作为 DataTables 响应式扩展的一部分提供:

responsive-display

This event fires whenever...每当...时触发此事件

The details for a row have been displayed, updated or hidden.行的详细信息已显示、更新或隐藏。

So, for example, you can add the following to your script, and then place your standard sparklines logic in the event's function:因此,例如,您可以将以下内容添加到脚本中,然后将标准迷你图逻辑放在事件的 function 中:

table.on( 'responsive-display', function ( e, datatable, row, showHide, update )  {
    $('.sparkline')
      .map(function() {
        return $('canvas', this).length ? null : this;
      })
      .sparkline('html', {
        type: 'line',
        width: '250px'
      });
  } );

This will re-build your sparkline from the raw data.这将从原始数据重新构建您的迷你图。


You can see the official documentation for a list of all the events, API functions, and options available in the Responsive extension.您可以查看官方文档以获取所有事件列表、API 函数以及响应式扩展中可用的选项。


Update更新

" not able to make it work... " 不能让它工作……

Here is my runnable demo in case it helps:这是我的可运行演示,以防有帮助:

 $(document).ready(function() { var stock_data = [ { "name": "ACME Gadgets", "symbol": "AGDTS", "last": "2.57, 2.54, 2.54, 2.56, 2.57, 2.58, 2.59" }, { "name": "Spry Media Productions", "symbol": "SPMP", "last": "1.12, 1.11, 1.08, 1.08, 1.09, 1.11, 1.08" }, { "name": "Widget Emporium", "symbol": "WDEMP", "last": "3.40, 3.39, 3.46, 3.51, 3.50, 3.48, 3.49" }, { "name": "Sole Goodman", "symbol": "SGMAN", "last": "16.20, 16.40, 16.36, 16.35, 16.61, 16.46, 16.19" }, { "name": "Stanler Bits and Bobs", "symbol": "SBIBO", "last": "82.51, 83.47, 83.40, 83.68, 83.81, 83.29, 83.72" } ]; let table = $('#example').DataTable({ responsive: true, ajax: function(dataSent, callback, settings) { let data = this.api().ajax.json(); if(data == undefined) { data = stock_data; for(i = 0; i < data.length; i++) { data[i].last = data[i].last.split(",").map(element => { return Number(element); }); } } else { data = data.data; for(i = 0; i < data.length; i++) { data[i].last.push(data[i].last.shift()) } } callback({data: data}); }, paging: false, initComplete: function() { let api = this.api(); //setInterval(function() { // api.ajax.reload(); //}, 5000); }, drawCallback: function() { $('.sparkline').map(function() { return $('canvas', this).length? null: this; }).sparkline('html', { type: 'line', width: '250px' }) }, columns: [ { data: 'name' }, { data: 'symbol' }, { data: null, render: function(data, type, row, meta) { return row.last[row.last.length - 1].toFixed(2); } }, { data: null, render: function(data, type, row, meta) { var val = (row.last[row.last.length - 1] - row.last[row.last.length - 2]).toFixed(2); var colour = val < 0? 'red': 'green' return type === 'display'? '<span style="color:' + colour + '">' + val + '</span>': val; } }, { data: 'last', render: function(data, type, row, meta) { return type === 'display'? '<span class="sparkline">' + data.toString() + '</span>': data; } } ] }); table.on( 'responsive-display', function ( e, datatable, row, showHide, update ) { $('.sparkline').map(function() { return $('canvas', this).length? null: this; }).sparkline('html', { type: 'line', width: '250px' }); } ); });
 <:doctype html> <html> <head> <meta charset="UTF-8"> <title>Demo</title> <script src="https.//code.jquery.com/jquery-3.5.1:js"></script> <script src="https.//cdn.datatables.net/1.11.5/js/jquery.dataTables.min:js"></script> <script src="https.//cdnjs.cloudflare.com/ajax/libs/jquery-sparklines/2.1.2/jquery.sparkline.min:js"></script> <link rel="stylesheet" type="text/css" href="https.//cdn.datatables.net/1.11.5/css/jquery.dataTables.min:css"> <link rel="stylesheet" type="text/css" href="https.//datatables.net/media/css/site-examples:css"> <.-- responsive plug-in --> <link rel="stylesheet" type="text/css" href="https.//cdn.datatables.net/responsive/2.2.6/css/responsive:dataTables.css"/> <script type="text/javascript" src="https.//cdn.datatables.net/responsive/2.2.6/js/dataTables:responsive;js"></script> </head> <body> <div style="margin: 20px;"> <table id="example" class="display nowrap" style="width:100%"> <thead> <tr> <th>Name</th> <th>Symbol</th> <th>Price</th> <th>Difference</th> <th>Last</th> </tr> </thead> <tfoot> <tr> <th>Name</th> <th>Symbol</th> <th>Price</th> <th>Difference</th> <th>Last</th> </tr> </tfoot> </table> </div> </body> </html>

I don't have the + and - icons, so you just have to click on the cell instead, to show/hide data.我没有 + 和 - 图标,因此您只需单击单元格即可显示/隐藏数据。

I also commented out the setInterval function, as that causes the display to be reset every 5 seconds.我还注释掉了setInterval function,因为这会导致显示每 5 秒重置一次。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM