diff options
| author | Johannes Ranke <johannes.ranke@jrwb.de> | 2025-02-14 07:19:15 +0100 | 
|---|---|---|
| committer | Johannes Ranke <johannes.ranke@jrwb.de> | 2025-02-14 07:19:15 +0100 | 
| commit | b0f08271d1dae8ffaf57f557c27eba1314ece1d5 (patch) | |
| tree | 98da899d455d6945849d6f4b4e98adfb98dc8b2b /docs/coverage/lib/datatables-binding-0.33 | |
| parent | 7dc59c522d0639f6473463340e518e2e8074e364 (diff) | |
| parent | 55d9c2331e468efd364472555dbfae84603a4f73 (diff) | |
Merge branch 'main' into dev
Diffstat (limited to 'docs/coverage/lib/datatables-binding-0.33')
| -rw-r--r-- | docs/coverage/lib/datatables-binding-0.33/datatables.js | 1539 | 
1 files changed, 1539 insertions, 0 deletions
| diff --git a/docs/coverage/lib/datatables-binding-0.33/datatables.js b/docs/coverage/lib/datatables-binding-0.33/datatables.js new file mode 100644 index 00000000..765b53cb --- /dev/null +++ b/docs/coverage/lib/datatables-binding-0.33/datatables.js @@ -0,0 +1,1539 @@ +(function() { + +// some helper functions: using a global object DTWidget so that it can be used +// in JS() code, e.g. datatable(options = list(foo = JS('code'))); unlike R's +// dynamic scoping, when 'code' is eval'ed, JavaScript does not know objects +// from the "parent frame", e.g. JS('DTWidget') will not work unless it was made +// a global object +var DTWidget = {}; + +// 123456666.7890 -> 123,456,666.7890 +var markInterval = function(d, digits, interval, mark, decMark, precision) { +  x = precision ? d.toPrecision(digits) : d.toFixed(digits); +  if (!/^-?[\d.]+$/.test(x)) return x; +  var xv = x.split('.'); +  if (xv.length > 2) return x;  // should have at most one decimal point +  xv[0] = xv[0].replace(new RegExp('\\B(?=(\\d{' + interval + '})+(?!\\d))', 'g'), mark); +  return xv.join(decMark); +}; + +DTWidget.formatCurrency = function(data, currency, digits, interval, mark, decMark, before, zeroPrint) { +  var d = parseFloat(data); +  if (isNaN(d)) return ''; +  if (zeroPrint !== null && d === 0.0) return zeroPrint; +  var res = markInterval(d, digits, interval, mark, decMark); +  res = before ? (/^-/.test(res) ? '-' + currency + res.replace(/^-/, '') : currency + res) : +    res + currency; +  return res; +}; + +DTWidget.formatString = function(data, prefix, suffix) { +  var d = data; +  if (d === null) return ''; +  return prefix + d + suffix; +}; + +DTWidget.formatPercentage = function(data, digits, interval, mark, decMark, zeroPrint) { +  var d = parseFloat(data); +  if (isNaN(d)) return ''; +  if (zeroPrint !== null && d === 0.0) return zeroPrint; +  return markInterval(d * 100, digits, interval, mark, decMark) + '%'; +}; + +DTWidget.formatRound = function(data, digits, interval, mark, decMark, zeroPrint) { +  var d = parseFloat(data); +  if (isNaN(d)) return ''; +  if (zeroPrint !== null && d === 0.0) return zeroPrint; +  return markInterval(d, digits, interval, mark, decMark); +}; + +DTWidget.formatSignif = function(data, digits, interval, mark, decMark, zeroPrint) { +  var d = parseFloat(data); +  if (isNaN(d)) return ''; +  if (zeroPrint !== null && d === 0.0) return zeroPrint; +  return markInterval(d, digits, interval, mark, decMark, true); +}; + +DTWidget.formatDate = function(data, method, params) { +  var d = data; +  if (d === null) return ''; +  // (new Date('2015-10-28')).toDateString() may return 2015-10-27 because the +  // actual time created could be like 'Tue Oct 27 2015 19:00:00 GMT-0500 (CDT)', +  // i.e. the date-only string is treated as UTC time instead of local time +  if ((method === 'toDateString' || method === 'toLocaleDateString') && /^\d{4,}\D\d{2}\D\d{2}$/.test(d)) { +    d = d.split(/\D/); +    d = new Date(d[0], d[1] - 1, d[2]); +  } else { +    d = new Date(d); +  } +  return d[method].apply(d, params); +}; + +window.DTWidget = DTWidget; + +// A helper function to update the properties of existing filters +var setFilterProps = function(td, props) { +  // Update enabled/disabled state +  var $input = $(td).find('input').first(); +  var searchable = $input.data('searchable'); +  $input.prop('disabled', !searchable || props.disabled); + +  // Based on the filter type, set its new values +  var type = td.getAttribute('data-type'); +  if (['factor', 'logical'].includes(type)) { +    // Reformat the new dropdown options for use with selectize +    var new_vals = props.params.options.map(function(item) { +      return { text: item, value: item }; +    }); + +    // Find the selectize object +    var dropdown = $(td).find('.selectized').eq(0)[0].selectize; + +    // Note the current values +    var old_vals = dropdown.getValue(); + +    // Remove the existing values +    dropdown.clearOptions(); + +    // Add the new options +    dropdown.addOption(new_vals); + +    // Preserve the existing values +    dropdown.setValue(old_vals); + +  } else if (['number', 'integer', 'date', 'time'].includes(type)) { +    // Apply internal scaling to new limits. Updating scale not yet implemented. +    var slider = $(td).find('.noUi-target').eq(0); +    var scale = Math.pow(10, Math.max(0, +slider.data('scale') || 0)); +    var new_vals = [props.params.min * scale, props.params.max * scale]; + +    // Note what the new limits will be just for this filter +    var new_lims = new_vals.slice(); + +    // Determine the current values and limits +    var old_vals = slider.val().map(Number); +    var old_lims = slider.noUiSlider('options').range; +    old_lims = [old_lims.min, old_lims.max]; + +    // Preserve the current values if filters have been applied; otherwise, apply no filtering +    if (old_vals[0] != old_lims[0]) { +      new_vals[0] = Math.max(old_vals[0], new_vals[0]); +    } + +    if (old_vals[1] != old_lims[1]) { +      new_vals[1] = Math.min(old_vals[1], new_vals[1]); +    } + +    // Update the endpoints of the slider +    slider.noUiSlider({ +      start: new_vals, +      range: {'min': new_lims[0], 'max': new_lims[1]} +    }, true); +  } +}; + +var transposeArray2D = function(a) { +  return a.length === 0 ? a : HTMLWidgets.transposeArray2D(a); +}; + +var crosstalkPluginsInstalled = false; + +function maybeInstallCrosstalkPlugins() { +  if (crosstalkPluginsInstalled) +    return; +  crosstalkPluginsInstalled = true; + +  $.fn.dataTable.ext.afnFiltering.push( +    function(oSettings, aData, iDataIndex) { +      var ctfilter = oSettings.nTable.ctfilter; +      if (ctfilter && !ctfilter[iDataIndex]) +        return false; + +      var ctselect = oSettings.nTable.ctselect; +      if (ctselect && !ctselect[iDataIndex]) +        return false; + +      return true; +    } +  ); +} + +HTMLWidgets.widget({ +  name: "datatables", +  type: "output", +  renderOnNullValue: true, +  initialize: function(el, width, height) { +    // in order that the type=number inputs return a number +    $.valHooks.number = { +      get: function(el) { +        var value = parseFloat(el.value); +        return isNaN(value) ? "" : value; +      } +    }; +    $(el).html(' '); +    return { +      data: null, +      ctfilterHandle: new crosstalk.FilterHandle(), +      ctfilterSubscription: null, +      ctselectHandle: new crosstalk.SelectionHandle(), +      ctselectSubscription: null +    }; +  }, +  renderValue: function(el, data, instance) { +    if (el.offsetWidth === 0 || el.offsetHeight === 0) { +      instance.data = data; +      return; +    } +    instance.data = null; +    var $el = $(el); +    $el.empty(); + +    if (data === null) { +      $el.append(' '); +      // clear previous Shiny inputs (if any) +      for (var i in instance.clearInputs) instance.clearInputs[i](); +      instance.clearInputs = {}; +      return; +    } + +    var crosstalkOptions = data.crosstalkOptions; +    if (!crosstalkOptions) crosstalkOptions = { +      'key': null, 'group': null +    }; +    if (crosstalkOptions.group) { +      maybeInstallCrosstalkPlugins(); +      instance.ctfilterHandle.setGroup(crosstalkOptions.group); +      instance.ctselectHandle.setGroup(crosstalkOptions.group); +    } + +    // if we are in the viewer then we always want to fillContainer and +    // and autoHideNavigation (unless the user has explicitly set these) +    if (window.HTMLWidgets.viewerMode) { +      if (!data.hasOwnProperty("fillContainer")) +        data.fillContainer = true; +      if (!data.hasOwnProperty("autoHideNavigation")) +        data.autoHideNavigation = true; +    } + +    // propagate fillContainer to instance (so we have it in resize) +    instance.fillContainer = data.fillContainer; + +    var cells = data.data; + +    if (cells instanceof Array) cells = transposeArray2D(cells); + +    $el.append(data.container); +    var $table = $el.find('table'); +    if (data.class) $table.addClass(data.class); +    if (data.caption) $table.prepend(data.caption); + +    if (!data.selection) data.selection = { +      mode: 'none', selected: null, target: 'row', selectable: null +    }; +    if (HTMLWidgets.shinyMode && data.selection.mode !== 'none' && +        data.selection.target === 'row+column') { +      if ($table.children('tfoot').length === 0) { +        $table.append($('<tfoot>')); +        $table.find('thead tr').clone().appendTo($table.find('tfoot')); +      } +    } + +    // column filters +    var filterRow; +    switch (data.filter) { +      case 'top': +        $table.children('thead').append(data.filterHTML); +        filterRow = $table.find('thead tr:last td'); +        break; +      case 'bottom': +        if ($table.children('tfoot').length === 0) { +          $table.append($('<tfoot>')); +        } +        $table.children('tfoot').prepend(data.filterHTML); +        filterRow = $table.find('tfoot tr:first td'); +        break; +    } + +    var options = { searchDelay: 1000 }; +    if (cells !== null) $.extend(options, { +      data: cells +    }); + +    // options for fillContainer +    var bootstrapActive = typeof($.fn.popover) != 'undefined'; +    if (instance.fillContainer) { + +      // force scrollX/scrollY and turn off autoWidth +      options.scrollX = true; +      options.scrollY = "100px"; // can be any value, we'll adjust below + +      // if we aren't paginating then move around the info/filter controls +      // to save space at the bottom and rephrase the info callback +      if (data.options.paging === false) { + +        // we know how to do this cleanly for bootstrap, not so much +        // for other themes/layouts +        if (bootstrapActive) { +          options.dom = "<'row'<'col-sm-4'i><'col-sm-8'f>>" + +                        "<'row'<'col-sm-12'tr>>"; +        } + +        options.fnInfoCallback = function(oSettings, iStart, iEnd, +                                           iMax, iTotal, sPre) { +          return Number(iTotal).toLocaleString() + " records"; +        }; +      } +    } + +    // auto hide navigation if requested +    // Note, this only works on client-side processing mode as on server-side, +    // cells (data.data) is null; In addition, we require the pageLength option +    // being provided explicitly to enable this. Despite we may be able to deduce +    // the default value of pageLength, it may complicate things so we'd rather +    // put this responsiblity to users and warn them on the R side. +    if (data.autoHideNavigation === true && data.options.paging !== false) { +      // strip all nav if length >= cells +      if ((cells instanceof Array) && data.options.pageLength >= cells.length) +        options.dom = bootstrapActive ? "<'row'<'col-sm-12'tr>>" : "t"; +      // alternatively lean things out for flexdashboard mobile portrait +      else if (bootstrapActive && window.FlexDashboard && window.FlexDashboard.isMobilePhone()) +        options.dom = "<'row'<'col-sm-12'f>>" + +                      "<'row'<'col-sm-12'tr>>"  + +                      "<'row'<'col-sm-12'p>>"; +    } + +    $.extend(true, options, data.options || {}); + +    var searchCols = options.searchCols; +    if (searchCols) { +      searchCols = searchCols.map(function(x) { +        return x === null ? '' : x.search; +      }); +      // FIXME: this means I don't respect the escapeRegex setting +      delete options.searchCols; +    } + +    // server-side processing? +    var server = options.serverSide === true; + +    // use the dataSrc function to pre-process JSON data returned from R +    var DT_rows_all = [], DT_rows_current = []; +    if (server && HTMLWidgets.shinyMode && typeof options.ajax === 'object' && +        /^session\/[\da-z]+\/dataobj/.test(options.ajax.url) && !options.ajax.dataSrc) { +      options.ajax.dataSrc = function(json) { +        DT_rows_all = $.makeArray(json.DT_rows_all); +        DT_rows_current = $.makeArray(json.DT_rows_current); +        var data = json.data; +        if (!colReorderEnabled()) return data; +        var table = $table.DataTable(), order = table.colReorder.order(), flag = true, i, j, row; +        for (i = 0; i < order.length; ++i) if (order[i] !== i) flag = false; +        if (flag) return data; +        for (i = 0; i < data.length; ++i) { +          row = data[i].slice(); +          for (j = 0; j < order.length; ++j) data[i][j] = row[order[j]]; +        } +        return data; +      }; +    } + +    var thiz = this; +    if (instance.fillContainer) $table.on('init.dt', function(e) { +      thiz.fillAvailableHeight(el, $(el).innerHeight()); +    }); +    // If the page contains serveral datatables and one of which enables colReorder, +    // the table.colReorder.order() function will exist but throws error when called. +    // So it seems like the only way to know if colReorder is enabled or not is to +    // check the options. +    var colReorderEnabled = function() { return "colReorder" in options; }; +    var table = $table.DataTable(options); +    $el.data('datatable', table); + +    if ('rowGroup' in options) { +      // Maintain RowGroup dataSrc when columns are reordered (#1109) +      table.on('column-reorder', function(e, settings, details) { +        var oldDataSrc = table.rowGroup().dataSrc(); +        var newDataSrc = details.mapping[oldDataSrc]; +        table.rowGroup().dataSrc(newDataSrc); +      }); +    } + +    // Unregister previous Crosstalk event subscriptions, if they exist +    if (instance.ctfilterSubscription) { +      instance.ctfilterHandle.off("change", instance.ctfilterSubscription); +      instance.ctfilterSubscription = null; +    } +    if (instance.ctselectSubscription) { +      instance.ctselectHandle.off("change", instance.ctselectSubscription); +      instance.ctselectSubscription = null; +    } + +    if (!crosstalkOptions.group) { +      $table[0].ctfilter = null; +      $table[0].ctselect = null; +    } else { +      var key = crosstalkOptions.key; +      function keysToMatches(keys) { +        if (!keys) { +          return null; +        } else { +          var selectedKeys = {}; +          for (var i = 0; i < keys.length; i++) { +            selectedKeys[keys[i]] = true; +          } +          var matches = {}; +          for (var j = 0; j < key.length; j++) { +            if (selectedKeys[key[j]]) +              matches[j] = true; +          } +          return matches; +        } +      } + +      function applyCrosstalkFilter(e) { +        $table[0].ctfilter = keysToMatches(e.value); +        table.draw(); +      } +      instance.ctfilterSubscription = instance.ctfilterHandle.on("change", applyCrosstalkFilter); +      applyCrosstalkFilter({value: instance.ctfilterHandle.filteredKeys}); + +      function applyCrosstalkSelection(e) { +        if (e.sender !== instance.ctselectHandle) { +          table +            .rows('.' + selClass, {search: 'applied'}) +            .nodes() +            .to$() +            .removeClass(selClass); +          if (selectedRows) +            changeInput('rows_selected', selectedRows(), void 0, true); +        } + +        if (e.sender !== instance.ctselectHandle && e.value && e.value.length) { +          var matches = keysToMatches(e.value); + +          // persistent selection with plotly (& leaflet) +          var ctOpts = crosstalk.var("plotlyCrosstalkOpts").get() || {}; +          if (ctOpts.persistent === true) { +            var matches = $.extend(matches, $table[0].ctselect); +          } + +          $table[0].ctselect = matches; +          table.draw(); +        } else { +          if ($table[0].ctselect) { +            $table[0].ctselect = null; +            table.draw(); +          } +        } +      } +      instance.ctselectSubscription = instance.ctselectHandle.on("change", applyCrosstalkSelection); +      // TODO: This next line doesn't seem to work when renderDataTable is used +      applyCrosstalkSelection({value: instance.ctselectHandle.value}); +    } + +    var inArray = function(val, array) { +      return $.inArray(val, $.makeArray(array)) > -1; +    }; + +    // search the i-th column +    var searchColumn = function(i, value) { +      var regex = false, ci = true; +      if (options.search) { +        regex = options.search.regex, +        ci = options.search.caseInsensitive !== false; +      } +      // need to transpose the column index when colReorder is enabled +      if (table.colReorder) i = table.colReorder.transpose(i); +      return table.column(i).search(value, regex, !regex, ci); +    }; + +    if (data.filter !== 'none') { +      if (!data.hasOwnProperty('filterSettings')) data.filterSettings = {}; + +      filterRow.each(function(i, td) { + +        var $td = $(td), type = $td.data('type'), filter; +        var $input = $td.children('div').first().children('input'); +        var disabled = $input.prop('disabled'); +        var searchable = table.settings()[0].aoColumns[i].bSearchable; +        $input.prop('disabled', !searchable || disabled); +        $input.data('searchable', searchable); // for updating later +        $input.on('input blur', function() { +          $input.next('span').toggle(Boolean($input.val())); +        }); +        // Bootstrap sets pointer-events to none and we won't be able to click +        // the clear button +        $input.next('span').css('pointer-events', 'auto').hide().click(function() { +          $(this).hide().prev('input').val('').trigger('input').focus(); +        }); +        var searchCol;  // search string for this column +        if (searchCols && searchCols[i]) { +          searchCol = searchCols[i]; +          $input.val(searchCol).trigger('input'); +        } +        var $x = $td.children('div').last(); + +        // remove the overflow: hidden attribute of the scrollHead +        // (otherwise the scrolling table body obscures the filters) +        // The workaround and the discussion from +        // https://github.com/rstudio/DT/issues/554#issuecomment-518007347 +        // Otherwise the filter selection will not be anchored to the values +        // when the columns number is many and scrollX is enabled. +        var scrollHead = $(el).find('.dataTables_scrollHead,.dataTables_scrollFoot'); +        var cssOverflowHead = scrollHead.css('overflow'); +        var scrollBody = $(el).find('.dataTables_scrollBody'); +        var cssOverflowBody = scrollBody.css('overflow'); +        var scrollTable = $(el).find('.dataTables_scroll'); +        var cssOverflowTable = scrollTable.css('overflow'); +        if (cssOverflowHead === 'hidden') { +          $x.on('show hide', function(e) { +            if (e.type === 'show') { +              scrollHead.css('overflow', 'visible'); +              scrollBody.css('overflow', 'visible'); +              scrollTable.css('overflow-x', 'scroll'); +            } else { +              scrollHead.css('overflow', cssOverflowHead); +              scrollBody.css('overflow', cssOverflowBody); +              scrollTable.css('overflow-x', cssOverflowTable); +            } +          }); +          $x.css('z-index', 25); +        } + +        if (inArray(type, ['factor', 'logical'])) { +          $input.on({ +            click: function() { +              $input.parent().hide(); $x.show().trigger('show'); filter[0].selectize.focus(); +            }, +            input: function() { +              var v1 = JSON.stringify(filter[0].selectize.getValue()), v2 = $input.val(); +              if (v1 === '[]') v1 = ''; +              if (v1 !== v2) filter[0].selectize.setValue(v2 === '' ? [] : JSON.parse(v2)); +            } +          }); +          var $input2 = $x.children('select'); +          filter = $input2.selectize($.extend({ +            options: $input2.data('options').map(function(v, i) { +              return ({text: v, value: v}); +            }), +            plugins: ['remove_button'], +            hideSelected: true, +            onChange: function(value) { +              if (value === null) value = []; // compatibility with jQuery 3.0 +              $input.val(value.length ? JSON.stringify(value) : ''); +              if (value.length) $input.trigger('input'); +              $input.attr('title', $input.val()); +              if (server) { +                searchColumn(i, value.length ? JSON.stringify(value) : '').draw(); +                return; +              } +              // turn off filter if nothing selected +              $td.data('filter', value.length > 0); +              table.draw();  // redraw table, and filters will be applied +            } +          }, data.filterSettings.select)); +          filter[0].selectize.on('blur', function() { +            $x.hide().trigger('hide'); $input.parent().show(); $input.trigger('blur'); +          }); +          filter.next('div').css('margin-bottom', 'auto'); +        } else if (type === 'character') { +          var fun = function() { +            searchColumn(i, $input.val()).draw(); +          }; +          // throttle searching for server-side processing +          var throttledFun = $.fn.dataTable.util.throttle(fun, options.searchDelay); +          $input.on('input', function(e, immediate) { +            // always bypass throttling when immediate = true (via the updateSearch method) +            (immediate || !server) ? fun() : throttledFun(); +          }); +        } else if (inArray(type, ['number', 'integer', 'date', 'time'])) { +          var $x0 = $x; +          $x = $x0.children('div').first(); +          $x0.css({ +            'background-color': '#fff', +            'border': '1px #ddd solid', +            'border-radius': '4px', +            'padding': data.vertical ? '35px 20px': '20px 20px 10px 20px' +          }); +          var $spans = $x0.children('span').css({ +            'margin-top': data.vertical ? '0' : '10px', +            'white-space': 'nowrap' +          }); +          var $span1 = $spans.first(), $span2 = $spans.last(); +          var r1 = +$x.data('min'), r2 = +$x.data('max'); +          // when the numbers are too small or have many decimal places, the +          // slider may have numeric precision problems (#150) +          var scale = Math.pow(10, Math.max(0, +$x.data('scale') || 0)); +          r1 = Math.round(r1 * scale); r2 = Math.round(r2 * scale); +          var scaleBack = function(x, scale) { +            if (scale === 1) return x; +            var d = Math.round(Math.log(scale) / Math.log(10)); +            // to avoid problems like 3.423/100 -> 0.034230000000000003 +            return (x / scale).toFixed(d); +          }; +          var slider_min = function() { +            return filter.noUiSlider('options').range.min; +          }; +          var slider_max = function() { +            return filter.noUiSlider('options').range.max; +          }; +          $input.on({ +            focus: function() { +              $x0.show().trigger('show'); +              // first, make sure the slider div leaves at least 20px between +              // the two (slider value) span's +              $x0.width(Math.max(160, $span1.outerWidth() + $span2.outerWidth() + 20)); +              // then, if the input is really wide or slider is vertical, +              // make the slider the same width as the input +              if ($x0.outerWidth() < $input.outerWidth() || data.vertical) { +                $x0.outerWidth($input.outerWidth()); +              } +              // make sure the slider div does not reach beyond the right margin +              if ($(window).width() < $x0.offset().left + $x0.width()) { +                $x0.offset({ +                  'left': $input.offset().left + $input.outerWidth() - $x0.outerWidth() +                }); +              } +            }, +            blur: function() { +              $x0.hide().trigger('hide'); +            }, +            input: function() { +              if ($input.val() === '') filter.val([slider_min(), slider_max()]); +            }, +            change: function() { +              var v = $input.val().replace(/\s/g, ''); +              if (v === '') return; +              v = v.split('...'); +              if (v.length !== 2) { +                $input.parent().addClass('has-error'); +                return; +              } +              if (v[0] === '') v[0] = slider_min(); +              if (v[1] === '') v[1] = slider_max(); +              $input.parent().removeClass('has-error'); +              // treat date as UTC time at midnight +              var strTime = function(x) { +                var s = type === 'date' ? 'T00:00:00Z' : ''; +                var t = new Date(x + s).getTime(); +                // add 10 minutes to date since it does not hurt the date, and +                // it helps avoid the tricky floating point arithmetic problems, +                // e.g. sometimes the date may be a few milliseconds earlier +                // than the midnight due to precision problems in noUiSlider +                return type === 'date' ? t + 3600000 : t; +              }; +              if (inArray(type, ['date', 'time'])) { +                v[0] = strTime(v[0]); +                v[1] = strTime(v[1]); +              } +              if (v[0] != slider_min()) v[0] *= scale; +              if (v[1] != slider_max()) v[1] *= scale; +              filter.val(v); +            } +          }); +          var formatDate = function(d) { +            d = scaleBack(d, scale); +            if (type === 'number') return d; +            if (type === 'integer') return parseInt(d); +            var x = new Date(+d); +            if (type === 'date') { +              var pad0 = function(x) { +                return ('0' + x).substr(-2, 2); +              }; +              return x.getUTCFullYear() + '-' + pad0(1 + x.getUTCMonth()) +                      + '-' + pad0(x.getUTCDate()); +            } else { +              return x.toISOString(); +            } +          }; +          var opts = type === 'date' ? { step: 60 * 60 * 1000 } : +                     type === 'integer' ? { step: 1 } : {}; + +          opts.orientation = data.vertical ? 'vertical': 'horizontal'; +          opts.direction = data.vertical ? 'rtl': 'ltr'; + +          filter = $x.noUiSlider($.extend({ +            start: [r1, r2], +            range: {min: r1, max: r2}, +            connect: true +          }, opts, data.filterSettings.slider)); +          if (scale > 1) (function() { +            var t1 = r1, t2 = r2; +            var val = filter.val(); +            while (val[0] > r1 || val[1] < r2) { +              if (val[0] > r1) { +                t1 -= val[0] - r1; +              } +              if (val[1] < r2) { +                t2 += r2 - val[1]; +              } +              filter = $x.noUiSlider($.extend({ +                start: [t1, t2], +                range: {min: t1, max: t2}, +                connect: true +              }, opts, data.filterSettings.slider), true); +              val = filter.val(); +            } +            r1  = t1; r2 = t2; +          })(); +          // format with active column renderer, if defined +          var colDef = data.options.columnDefs.find(function(def) { +            return (def.targets === i || inArray(i, def.targets)) && 'render' in def; +          }); +          var updateSliderText = function(v1, v2) { +            // we only know how to use function renderers +            if (colDef && typeof colDef.render === 'function') { +              var restore = function(v) { +                v = scaleBack(v, scale); +                return inArray(type, ['date', 'time']) ? new Date(+v) : v; +              } +              $span1.text(colDef.render(restore(v1), 'display')); +              $span2.text(colDef.render(restore(v2), 'display')); +            } else { +              $span1.text(formatDate(v1)); +              $span2.text(formatDate(v2)); +            } +          }; +          updateSliderText(r1, r2); +          var updateSlider = function(e) { +            var val = filter.val(); +            // turn off filter if in full range +            $td.data('filter', val[0] > slider_min() || val[1] < slider_max()); +            var v1 = formatDate(val[0]), v2 = formatDate(val[1]), ival; +            if ($td.data('filter')) { +              ival = v1 + ' ... ' + v2; +              $input.attr('title', ival).val(ival).trigger('input'); +            } else { +              $input.attr('title', '').val(''); +            } +            updateSliderText(val[0], val[1]); +            if (e.type === 'slide') return;  // no searching when sliding only +            if (server) { +              searchColumn(i, $td.data('filter') ? ival : '').draw(); +              return; +            } +            table.draw(); +          }; +          filter.on({ +            set: updateSlider, +            slide: updateSlider +          }); +        } + +        // server-side processing will be handled by R (or whatever server +        // language you use); the following code is only needed for client-side +        // processing +        if (server) { +          // if a search string has been pre-set, search now +          if (searchCol) $input.trigger('input').trigger('change'); +          return; +        } + +        var customFilter = function(settings, data, dataIndex) { +          // there is no way to attach a search function to a specific table, +          // and we need to make sure a global search function is not applied to +          // all tables (i.e. a range filter in a previous table should not be +          // applied to the current table); we use the settings object to +          // determine if we want to perform searching on the current table, +          // since settings.sTableId will be different to different tables +          if (table.settings()[0] !== settings) return true; +          // no filter on this column or no need to filter this column +          if (typeof filter === 'undefined' || !$td.data('filter')) return true; + +          var r = filter.val(), v, r0, r1; +          var i_data = function(i) { +            if (!colReorderEnabled()) return i; +            var order = table.colReorder.order(), k; +            for (k = 0; k < order.length; ++k) if (order[k] === i) return k; +            return i; // in theory it will never be here... +          } +          v = data[i_data(i)]; +          if (type === 'number' || type === 'integer') { +            v = parseFloat(v); +            // how to handle NaN? currently exclude these rows +            if (isNaN(v)) return(false); +            r0 = parseFloat(scaleBack(r[0], scale)) +            r1 = parseFloat(scaleBack(r[1], scale)); +            if (v >= r0 && v <= r1) return true; +          } else if (type === 'date' || type === 'time') { +            v = new Date(v); +            r0 = new Date(r[0] / scale); r1 = new Date(r[1] / scale); +            if (v >= r0 && v <= r1) return true; +          } else if (type === 'factor') { +            if (r.length === 0 || inArray(v, r)) return true; +          } else if (type === 'logical') { +            if (r.length === 0) return true; +            if (inArray(v === '' ? 'na' : v, r)) return true; +          } +          return false; +        }; + +        $.fn.dataTable.ext.search.push(customFilter); + +        // search for the preset search strings if it is non-empty +        if (searchCol) $input.trigger('input').trigger('change'); + +      }); + +    } + +    // highlight search keywords +    var highlight = function() { +      var body = $(table.table().body()); +      // removing the old highlighting first +      body.unhighlight(); + +      // don't highlight the "not found" row, so we get the rows using the api +      if (table.rows({ filter: 'applied' }).data().length === 0) return; +      // highlight global search keywords +      body.highlight($.trim(table.search()).split(/\s+/)); +      // then highlight keywords from individual column filters +      if (filterRow) filterRow.each(function(i, td) { +        var $td = $(td), type = $td.data('type'); +        if (type !== 'character') return; +        var $input = $td.children('div').first().children('input'); +        var column = table.column(i).nodes().to$(), +            val = $.trim($input.val()); +        if (type !== 'character' || val === '') return; +        column.highlight(val.split(/\s+/)); +      }); +    }; + +    if (options.searchHighlight) { +      table +      .on('draw.dt.dth column-visibility.dt.dth column-reorder.dt.dth', highlight) +      .on('destroy', function() { +        // remove event handler +        table.off('draw.dt.dth column-visibility.dt.dth column-reorder.dt.dth'); +      }); + +      // Set the option for escaping regex characters in our search string.  This will be used +      // for all future matching. +      jQuery.fn.highlight.options.escapeRegex = (!options.search || !options.search.regex); + +      // initial highlight for state saved conditions and initial states +      highlight(); +    } + +    // run the callback function on the table instance +    if (typeof data.callback === 'function') data.callback(table); + +    // double click to edit the cell, row, column, or all cells +    if (data.editable) table.on('dblclick.dt', 'tbody td', function(e) { +      // only bring up the editor when the cell itself is dbclicked, and ignore +      // other dbclick events bubbled up (e.g. from the <input>) +      if (e.target !== this) return; +      var target = [], immediate = false; +      switch (data.editable.target) { +        case 'cell': +          target = [this]; +          immediate = true;  // edit will take effect immediately +          break; +        case 'row': +          target = table.cells(table.cell(this).index().row, '*').nodes(); +          break; +        case 'column': +          target = table.cells('*', table.cell(this).index().column).nodes(); +          break; +        case 'all': +          target = table.cells().nodes(); +          break; +        default: +          throw 'The editable parameter must be "cell", "row", "column", or "all"'; +      } +      var disableCols = data.editable.disable ? data.editable.disable.columns : null; +      var numericCols = data.editable.numeric; +      var areaCols = data.editable.area; +      var dateCols = data.editable.date; +      for (var i = 0; i < target.length; i++) { +        (function(cell, current) { +          var $cell = $(cell), html = $cell.html(); +          var _cell = table.cell(cell), value = _cell.data(), index = _cell.index().column; +          var $input; +          if (inArray(index, numericCols)) { +            $input = $('<input type="number">'); +          } else if (inArray(index, areaCols)) { +            $input = $('<textarea></textarea>'); +          } else if (inArray(index, dateCols)) { +            $input = $('<input type="date">'); +          } else { +            $input = $('<input type="text">'); +          } +          if (!immediate) { +            $cell.data('input', $input).data('html', html); +            $input.attr('title', 'Hit Ctrl+Enter to finish editing, or Esc to cancel'); +          } +          $input.val(value); +          if (inArray(index, disableCols)) { +            $input.attr('readonly', '').css('filter', 'invert(25%)'); +          } +          $cell.empty().append($input); +          if (cell === current) $input.focus(); +          $input.css('width', '100%'); + +          if (immediate) $input.on('blur', function(e) { +            var valueNew = $input.val(); +            if (valueNew !== value) { +              _cell.data(valueNew); +              if (HTMLWidgets.shinyMode) { +                changeInput('cell_edit', [cellInfo(cell)], 'DT.cellInfo', null, {priority: 'event'}); +              } +              // for server-side processing, users have to call replaceData() to update the table +              if (!server) table.draw(false); +            } else { +              $cell.html(html); +            } +          }).on('keyup', function(e) { +            // hit Escape to cancel editing +            if (e.keyCode === 27) $input.trigger('blur'); +          }); + +          // bulk edit (row, column, or all) +          if (!immediate) $input.on('keyup', function(e) { +            var removeInput = function($cell, restore) { +              $cell.data('input').remove(); +              if (restore) $cell.html($cell.data('html')); +            } +            if (e.keyCode === 27) { +              for (var i = 0; i < target.length; i++) { +                removeInput($(target[i]), true); +              } +            } else if (e.keyCode === 13 && e.ctrlKey) { +              // Ctrl + Enter +              var cell, $cell, _cell, cellData = []; +              for (var i = 0; i < target.length; i++) { +                cell = target[i]; $cell = $(cell); _cell = table.cell(cell); +                _cell.data($cell.data('input').val()); +                HTMLWidgets.shinyMode && cellData.push(cellInfo(cell)); +                removeInput($cell, false); +              } +              if (HTMLWidgets.shinyMode) { +                changeInput('cell_edit', cellData, 'DT.cellInfo', null, {priority: "event"}); +              } +              if (!server) table.draw(false); +            } +          }); +        })(target[i], this); +      } +    }); + +    // interaction with shiny +    if (!HTMLWidgets.shinyMode && !crosstalkOptions.group) return; + +    var methods = {}; +    var shinyData = {}; + +    methods.updateCaption = function(caption) { +      if (!caption) return; +      $table.children('caption').replaceWith(caption); +    } + +    // register clear functions to remove input values when the table is removed +    instance.clearInputs = {}; + +    var changeInput = function(id, value, type, noCrosstalk, opts) { +      var event = id; +      id = el.id + '_' + id; +      if (type) id = id + ':' + type; +      // do not update if the new value is the same as old value +      if (event !== 'cell_edit' && !/_clicked$/.test(event) && shinyData.hasOwnProperty(id) && shinyData[id] === JSON.stringify(value)) +        return; +      shinyData[id] = JSON.stringify(value); +      if (HTMLWidgets.shinyMode && Shiny.setInputValue) { +        Shiny.setInputValue(id, value, opts); +        if (!instance.clearInputs[id]) instance.clearInputs[id] = function() { +          Shiny.setInputValue(id, null); +        } +      } + +      // HACK +      if (event === "rows_selected" && !noCrosstalk) { +        if (crosstalkOptions.group) { +          var keys = crosstalkOptions.key; +          var selectedKeys = null; +          if (value) { +            selectedKeys = []; +            for (var i = 0; i < value.length; i++) { +              // The value array's contents use 1-based row numbers, so we must +              // convert to 0-based before indexing into the keys array. +              selectedKeys.push(keys[value[i] - 1]); +            } +          } +          instance.ctselectHandle.set(selectedKeys); +        } +      } +    }; + +    var addOne = function(x) { +      return x.map(function(i) { return 1 + i; }); +    }; + +    var unique = function(x) { +      var ux = []; +      $.each(x, function(i, el){ +        if ($.inArray(el, ux) === -1) ux.push(el); +      }); +      return ux; +    } + +    // change the row index of a cell +    var tweakCellIndex = function(cell) { +      var info = cell.index(); +      // some cell may not be valid. e.g, #759 +      // when using the RowGroup extension, datatables will +      // generate the row label and the cells are not part of +      // the data thus contain no row/col info +      if (info === undefined) +        return {row: null, col: null}; +      if (server) { +        info.row = DT_rows_current[info.row]; +      } else { +        info.row += 1; +      } +      return {row: info.row, col: info.column}; +    } + +    var cleanSelectedValues = function() { +      changeInput('rows_selected', []); +      changeInput('columns_selected', []); +      changeInput('cells_selected', transposeArray2D([]), 'shiny.matrix'); +    } +    // #828 we should clean the selection on the server-side when the table reloads +    cleanSelectedValues(); + +    // a flag to indicates if select extension is initialized or not +    var flagSelectExt = table.settings()[0]._select !== undefined; +    // the Select extension should only be used in the client mode and +    // when the selection.mode is set to none +    if (data.selection.mode === 'none' && !server && flagSelectExt) { +      var updateRowsSelected = function() { +        var rows = table.rows({selected: true}); +        var selected = []; +        $.each(rows.indexes().toArray(), function(i, v) { +          selected.push(v + 1); +        }); +        changeInput('rows_selected', selected); +      } +      var updateColsSelected = function() { +        var columns = table.columns({selected: true}); +        changeInput('columns_selected', columns.indexes().toArray()); +      } +      var updateCellsSelected = function() { +        var cells = table.cells({selected: true}); +        var selected = []; +        cells.every(function() { +          var row = this.index().row; +          var col = this.index().column; +          selected = selected.concat([[row + 1, col]]); +        }); +        changeInput('cells_selected', transposeArray2D(selected), 'shiny.matrix'); +      } +      table.on('select deselect', function(e, dt, type, indexes) { +        updateRowsSelected(); +        updateColsSelected(); +        updateCellsSelected(); +      }) +      updateRowsSelected(); +      updateColsSelected(); +      updateCellsSelected(); +    } + +    var selMode = data.selection.mode, selTarget = data.selection.target; +    var selDisable = data.selection.selectable === false; +    if (inArray(selMode, ['single', 'multiple'])) { +      var selClass = inArray(data.style, ['bootstrap', 'bootstrap4']) ? 'active' : 'selected'; +      // selected1: row indices; selected2: column indices +      var initSel = function(x) { +        if (x === null || typeof x === 'boolean' || selTarget === 'cell') { +          return {rows: [], cols: []}; +        } else if (selTarget === 'row') { +          return {rows: $.makeArray(x), cols: []}; +        } else if (selTarget === 'column') { +          return {rows: [], cols: $.makeArray(x)}; +        } else if (selTarget === 'row+column') { +          return {rows: $.makeArray(x.rows), cols: $.makeArray(x.cols)}; +        } +      } +      var selected = data.selection.selected; +      var selected1 = initSel(selected).rows, selected2 = initSel(selected).cols; +      // selectable should contain either all positive or all non-positive values, not both +      // positive values indicate "selectable" while non-positive values means "nonselectable" +      // the assertion is performed on R side. (only column indicides could be zero which indicates +      // the row name) +      var selectable = data.selection.selectable; +      var selectable1 = initSel(selectable).rows, selectable2 = initSel(selectable).cols; + +      // After users reorder the rows or filter the table, we cannot use the table index +      // directly. Instead, we need this function to find out the rows between the two clicks. +      // If user filter the table again between the start click and the end click, the behavior +      // would be undefined, but it should not be a problem. +      var shiftSelRowsIndex = function(start, end) { +        var indexes = server ? DT_rows_all : table.rows({ search: 'applied' }).indexes().toArray(); +        start = indexes.indexOf(start); end = indexes.indexOf(end); +        // if start is larger than end, we need to swap +        if (start > end) { +          var tmp = end; end = start; start = tmp; +        } +        return indexes.slice(start, end + 1); +      } + +      var serverRowIndex = function(clientRowIndex) { +        return server ? DT_rows_current[clientRowIndex] : clientRowIndex + 1; +      } + +      // row, column, or cell selection +      var lastClickedRow; +      if (inArray(selTarget, ['row', 'row+column'])) { +        // Get the current selected rows. It will also +        // update the selected1's value based on the current row selection state +        // Note we can't put this function inside selectRows() directly, +        // the reason is method.selectRows() will override selected1's value but this +        // function will add rows to selected1 (keep the existing selection), which is +        // inconsistent with column and cell selection. +        var selectedRows = function() { +          var rows = table.rows('.' + selClass); +          var idx = rows.indexes().toArray(); +          if (!server) { +            selected1 = addOne(idx); +            return selected1; +          } +          idx = idx.map(function(i) { +            return DT_rows_current[i]; +          }); +          selected1 = selMode === 'multiple' ? unique(selected1.concat(idx)) : idx; +          return selected1; +        } +        // Change selected1's value based on selectable1, then refresh the row state +        var onlyKeepSelectableRows = function() { +          if (selDisable) { // users can't select; useful when only want backend select +            selected1 = []; +            return; +          } +          if (selectable1.length === 0) return; +          var nonselectable = selectable1[0] <= 0; +          if (nonselectable) { +            // should make selectable1 positive +            selected1 = $(selected1).not(selectable1.map(function(i) { return -i; })).get(); +          } else { +            selected1 = $(selected1).filter(selectable1).get(); +          } +        } +        // Change selected1's value based on selectable1, then +        // refresh the row selection state according to values in selected1 +        var selectRows = function(ignoreSelectable) { +          if (!ignoreSelectable) onlyKeepSelectableRows(); +          table.$('tr.' + selClass).removeClass(selClass); +          if (selected1.length === 0) return; +          if (server) { +            table.rows({page: 'current'}).every(function() { +              if (inArray(DT_rows_current[this.index()], selected1)) { +                $(this.node()).addClass(selClass); +              } +            }); +          } else { +            var selected0 = selected1.map(function(i) { return i - 1; }); +            $(table.rows(selected0).nodes()).addClass(selClass); +          } +        } +        table.on('mousedown.dt', 'tbody tr', function(e) { +          var $this = $(this), thisRow = table.row(this); +          if (selMode === 'multiple') { +            if (e.shiftKey && lastClickedRow !== undefined) { +              // select or de-select depends on the last clicked row's status +              var flagSel = !$this.hasClass(selClass); +              var crtClickedRow = serverRowIndex(thisRow.index()); +              if (server) { +                var rowsIndex = shiftSelRowsIndex(lastClickedRow, crtClickedRow); +                // update current page's selClass +                rowsIndex.map(function(i) { +                  var rowIndex = DT_rows_current.indexOf(i); +                  if (rowIndex >= 0) { +                    var row = table.row(rowIndex).nodes().to$(); +                    var flagRowSel = !row.hasClass(selClass); +                    if (flagSel === flagRowSel) row.toggleClass(selClass); +                  } +                }); +                // update selected1 +                if (flagSel) { +                  selected1 = unique(selected1.concat(rowsIndex)); +                } else { +                  selected1 = selected1.filter(function(index) { +                    return !inArray(index, rowsIndex); +                  }); +                } +              } else { +                // js starts from 0 +                shiftSelRowsIndex(lastClickedRow - 1, crtClickedRow - 1).map(function(value) { +                  var row = table.row(value).nodes().to$(); +                  var flagRowSel = !row.hasClass(selClass); +                  if (flagSel === flagRowSel) row.toggleClass(selClass); +                }); +              } +              e.preventDefault(); +            } else { +              $this.toggleClass(selClass); +            } +          } else { +            if ($this.hasClass(selClass)) { +              $this.removeClass(selClass); +            } else { +              table.$('tr.' + selClass).removeClass(selClass); +              $this.addClass(selClass); +            } +          } +          if (server && !$this.hasClass(selClass)) { +            var id = DT_rows_current[thisRow.index()]; +            // remove id from selected1 since its class .selected has been removed +            if (inArray(id, selected1)) selected1.splice($.inArray(id, selected1), 1); +          } +          selectedRows(); // update selected1's value based on selClass +          selectRows(false); // only keep the selectable rows +          changeInput('rows_selected', selected1); +          changeInput('row_last_clicked', serverRowIndex(thisRow.index()), null, null, {priority: 'event'}); +          lastClickedRow = serverRowIndex(thisRow.index()); +        }); +        selectRows(false);  // in case users have specified pre-selected rows +        // restore selected rows after the table is redrawn (e.g. sort/search/page); +        // client-side tables will preserve the selections automatically; for +        // server-side tables, we have to *real* row indices are in `selected1` +        changeInput('rows_selected', selected1); +        if (server) table.on('draw.dt', function(e) { selectRows(false); }); +        methods.selectRows = function(selected, ignoreSelectable) { +          selected1 = $.makeArray(selected); +          selectRows(ignoreSelectable); +          changeInput('rows_selected', selected1); +        } +      } + +      if (inArray(selTarget, ['column', 'row+column'])) { +        if (selTarget === 'row+column') { +          $(table.columns().footer()).css('cursor', 'pointer'); +        } +        // update selected2's value based on selectable2 +        var onlyKeepSelectableCols = function() { +          if (selDisable) { // users can't select; useful when only want backend select +            selected2 = []; +            return; +          } +          if (selectable2.length === 0) return; +          var nonselectable = selectable2[0] <= 0; +          if (nonselectable) { +            // need to make selectable2 positive +            selected2 = $(selected2).not(selectable2.map(function(i) { return -i; })).get(); +          } else { +            selected2 = $(selected2).filter(selectable2).get(); +          } +        } +        // update selected2 and then +        // refresh the col selection state according to values in selected2 +        var selectCols = function(ignoreSelectable) { +          if (!ignoreSelectable) onlyKeepSelectableCols(); +          // if selected2 is not a valide index (e.g., larger than the column number) +          // table.columns(selected2) will fail and result in a blank table +          // this is different from the table.rows(), where the out-of-range indexes +          // doesn't affect at all +          selected2 = $(selected2).filter(table.columns().indexes()).get(); +          table.columns().nodes().flatten().to$().removeClass(selClass); +          if (selected2.length > 0) +            table.columns(selected2).nodes().flatten().to$().addClass(selClass); +        } +        var callback = function() { +          var colIdx = selTarget === 'column' ? table.cell(this).index().column : +              $.inArray(this, table.columns().footer()), +              thisCol = $(table.column(colIdx).nodes()); +          if (colIdx === -1) return; +          if (thisCol.hasClass(selClass)) { +            thisCol.removeClass(selClass); +            selected2.splice($.inArray(colIdx, selected2), 1); +          } else { +            if (selMode === 'single') $(table.cells().nodes()).removeClass(selClass); +            thisCol.addClass(selClass); +            selected2 = selMode === 'single' ? [colIdx] : unique(selected2.concat([colIdx])); +          } +          selectCols(false); // update selected2 based on selectable +          changeInput('columns_selected', selected2); +        } +        if (selTarget === 'column') { +          $(table.table().body()).on('click.dt', 'td', callback); +        } else { +          $(table.table().footer()).on('click.dt', 'tr th', callback); +        } +        selectCols(false);  // in case users have specified pre-selected columns +        changeInput('columns_selected', selected2); +        if (server) table.on('draw.dt', function(e) { selectCols(false); }); +        methods.selectColumns = function(selected, ignoreSelectable) { +          selected2 = $.makeArray(selected); +          selectCols(ignoreSelectable); +          changeInput('columns_selected', selected2); +        } +      } + +      if (selTarget === 'cell') { +        var selected3 = [], selectable3 = []; +        if (selected !== null) selected3 = selected; +        if (selectable !== null && typeof selectable !== 'boolean') selectable3 = selectable; +        var findIndex = function(ij, sel) { +          for (var i = 0; i < sel.length; i++) { +            if (ij[0] === sel[i][0] && ij[1] === sel[i][1]) return i; +          } +          return -1; +        } +         // Change selected3's value based on selectable3, then refresh the cell state +        var onlyKeepSelectableCells = function() { +          if (selDisable) { // users can't select; useful when only want backend select +            selected3 = []; +            return; +          } +          if (selectable3.length === 0) return; +          var nonselectable = selectable3[0][0] <= 0; +          var out = []; +          if (nonselectable) { +            selected3.map(function(ij) { +              // should make selectable3 positive +              if (findIndex([-ij[0], -ij[1]], selectable3) === -1) { out.push(ij); } +            }); +          } else { +            selected3.map(function(ij) { +              if (findIndex(ij, selectable3) > -1) { out.push(ij); } +            }); +          } +          selected3 = out; +        } +        // Change selected3's value based on selectable3, then +        // refresh the cell selection state according to values in selected3 +        var selectCells = function(ignoreSelectable) { +          if (!ignoreSelectable) onlyKeepSelectableCells(); +          table.$('td.' + selClass).removeClass(selClass); +          if (selected3.length === 0) return; +          if (server) { +            table.cells({page: 'current'}).every(function() { +              var info = tweakCellIndex(this); +              if (findIndex([info.row, info.col], selected3) > -1) +                $(this.node()).addClass(selClass); +            }); +          } else { +            selected3.map(function(ij) { +              $(table.cell(ij[0] - 1, ij[1]).node()).addClass(selClass); +            }); +          } +        }; +        table.on('click.dt', 'tbody td', function() { +          var $this = $(this), info = tweakCellIndex(table.cell(this)); +          if ($this.hasClass(selClass)) { +            $this.removeClass(selClass); +            selected3.splice(findIndex([info.row, info.col], selected3), 1); +          } else { +            if (selMode === 'single') $(table.cells().nodes()).removeClass(selClass); +            $this.addClass(selClass); +            selected3 = selMode === 'single' ? [[info.row, info.col]] : +              unique(selected3.concat([[info.row, info.col]])); +          } +          selectCells(false); // must call this to update selected3 based on selectable3 +          changeInput('cells_selected', transposeArray2D(selected3), 'shiny.matrix'); +        }); +        selectCells(false);  // in case users have specified pre-selected columns +        changeInput('cells_selected', transposeArray2D(selected3), 'shiny.matrix'); + +        if (server) table.on('draw.dt', function(e) { selectCells(false); }); +        methods.selectCells = function(selected, ignoreSelectable) { +          selected3 = selected ? selected : []; +          selectCells(ignoreSelectable); +          changeInput('cells_selected', transposeArray2D(selected3), 'shiny.matrix'); +        } +      } +    } + +    // expose some table info to Shiny +    var updateTableInfo = function(e, settings) { +      // TODO: is anyone interested in the page info? +      // changeInput('page_info', table.page.info()); +      var updateRowInfo = function(id, modifier) { +        var idx; +        if (server) { +          idx = modifier.page === 'current' ? DT_rows_current : DT_rows_all; +        } else { +          var rows = table.rows($.extend({ +            search: 'applied', +            page: 'all' +          }, modifier)); +          idx = addOne(rows.indexes().toArray()); +        } +        changeInput('rows' + '_' + id, idx); +      }; +      updateRowInfo('current', {page: 'current'}); +      updateRowInfo('all', {}); +    } +    table.on('draw.dt', updateTableInfo); +    updateTableInfo(); + +    // state info +    table.on('draw.dt column-visibility.dt', function() { +      changeInput('state', table.state()); +    }); +    changeInput('state', table.state()); + +    // search info +    var updateSearchInfo = function() { +      changeInput('search', table.search()); +      if (filterRow) changeInput('search_columns', filterRow.toArray().map(function(td) { +        return $(td).find('input').first().val(); +      })); +    } +    table.on('draw.dt', updateSearchInfo); +    updateSearchInfo(); + +    var cellInfo = function(thiz) { +      var info = tweakCellIndex(table.cell(thiz)); +      info.value = table.cell(thiz).data(); +      return info; +    } +    // the current cell clicked on +    table.on('click.dt', 'tbody td', function() { +      changeInput('cell_clicked', cellInfo(this), null, null, {priority: 'event'}); +    }) +    changeInput('cell_clicked', {}); + +    // do not trigger table selection when clicking on links unless they have classes +    table.on('mousedown.dt', 'tbody td a', function(e) { +      if (this.className === '') e.stopPropagation(); +    }); + +    methods.addRow = function(data, rowname, resetPaging) { +      var n = table.columns().indexes().length, d = n - data.length; +      if (d === 1) { +        data = rowname.concat(data) +      } else if (d !== 0) { +        console.log(data); +        console.log(table.columns().indexes()); +        throw 'New data must be of the same length as current data (' + n + ')'; +      }; +      table.row.add(data).draw(resetPaging); +    } + +    methods.updateSearch = function(keywords) { +      if (keywords.global !== null) +        $(table.table().container()).find('input[type=search]').first() +             .val(keywords.global).trigger('input'); +      var columns = keywords.columns; +      if (!filterRow || columns === null) return; +      filterRow.toArray().map(function(td, i) { +        var v = typeof columns === 'string' ? columns : columns[i]; +        if (typeof v === 'undefined') { +          console.log('The search keyword for column ' + i + ' is undefined') +          return; +        } +        // Update column search string and values on linked filter widgets. +        // 'input' for factor and char filters, 'change' for numeric filters. +        $(td).find('input').first().val(v).trigger('input', [true]).trigger('change'); +      }); +      table.draw(); +    } + +    methods.hideCols = function(hide, reset) { +      if (reset) table.columns().visible(true, false); +      table.columns(hide).visible(false); +    } + +    methods.showCols = function(show, reset) { +      if (reset) table.columns().visible(false, false); +      table.columns(show).visible(true); +    } + +    methods.colReorder = function(order, origOrder) { +      table.colReorder.order(order, origOrder); +    } + +    methods.selectPage = function(page) { +      if (table.page.info().pages < page || page < 1) { +        throw 'Selected page is out of range'; +      }; +      table.page(page - 1).draw(false); +    } + +    methods.reloadData = function(resetPaging, clearSelection) { +      // empty selections first if necessary +      if (methods.selectRows && inArray('row', clearSelection)) methods.selectRows([]); +      if (methods.selectColumns && inArray('column', clearSelection)) methods.selectColumns([]); +      if (methods.selectCells && inArray('cell', clearSelection)) methods.selectCells([]); +      table.ajax.reload(null, resetPaging); +    } + +    // update table filters (set new limits of sliders) +    methods.updateFilters = function(newProps) { +      // loop through each filter in the filter row +      filterRow.each(function(i, td) { +        var k = i; +        if (filterRow.length > newProps.length) { +          if (i === 0) return;  // first column is row names +          k = i - 1; +        } +        // Update the filters to reflect the updated data. +        // Allow "falsy" (e.g. NULL) to signify a no-op. +        if (newProps[k]) { +          setFilterProps(td, newProps[k]); +        } +      }); +    }; + +    table.shinyMethods = methods; +  }, +  resize: function(el, width, height, instance) { +    if (instance.data) this.renderValue(el, instance.data, instance); + +    // dynamically adjust height if fillContainer = TRUE +    if (instance.fillContainer) +      this.fillAvailableHeight(el, height); + +    this.adjustWidth(el); +  }, + +  // dynamically set the scroll body to fill available height +  // (used with fillContainer = TRUE) +  fillAvailableHeight: function(el, availableHeight) { + +    // see how much of the table is occupied by header/footer elements +    // and use that to compute a target scroll body height +    var dtWrapper = $(el).find('div.dataTables_wrapper'); +    var dtScrollBody = $(el).find($('div.dataTables_scrollBody')); +    var framingHeight = dtWrapper.innerHeight() - dtScrollBody.innerHeight(); +    var scrollBodyHeight = availableHeight - framingHeight; + +    // we need to set `max-height` to none as datatables library now sets this +    // to a fixed height, disabling the ability to resize to fill the window, +    // as it will be set to a fixed 100px under such circumstances, e.g., RStudio IDE, +    // or FlexDashboard +    // see https://github.com/rstudio/DT/issues/951#issuecomment-1026464509 +    dtScrollBody.css('max-height', 'none'); +    // set the height +    dtScrollBody.height(scrollBodyHeight + 'px'); +  }, + +  // adjust the width of columns; remove the hard-coded widths on table and the +  // scroll header when scrollX/Y are enabled +  adjustWidth: function(el) { +    var $el = $(el), table = $el.data('datatable'); +    if (table) table.columns.adjust(); +    $el.find('.dataTables_scrollHeadInner').css('width', '') +        .children('table').css('margin-left', ''); +  } +}); + +  if (!HTMLWidgets.shinyMode) return; + +  Shiny.addCustomMessageHandler('datatable-calls', function(data) { +    var id = data.id; +    var el = document.getElementById(id); +    var table = el ? $(el).data('datatable') : null; +    if (!table) { +      console.log("Couldn't find table with id " + id); +      return; +    } + +    var methods = table.shinyMethods, call = data.call; +    if (methods[call.method]) { +      methods[call.method].apply(table, call.args); +    } else { +      console.log("Unknown method " + call.method); +    } +  }); + +})(); | 
