Adding a custom search filter to jQuery DataTables
jQuery DataTables is the premier solution for adding sorting/
Start here
The documentation contains a good Custom filtering - range search example that adds two new INPUT
elements. These inputs filter a column in the table by min/max numbers. However…
- The
INPUT
elements are assumed to already be on the page - The filter function added applies to all DataTables on the page
There is one key takeaway from this example though. $.fn.dataTable.ext.search
is an array of functions that is run for each row of data. If a function returns false
for a row, that row is hidden on the table.
Also, be advised that the function for the DataTables default search input always runs before your custom functions. This means you can't just repurpose the value of the default search input when searching inside ranges. For example, say a colum of your table lists temperature ranges for a set of days. If you want to search the table for temperature values inside the listed ranges, you'll need to add your own custom INPUT
element where you can enter your search temperature. If you enter a temperature into the default search INPUT
, only temperature values that directly appear in the table will be displayed.
Add an INPUT
element dynamically
We can add a new INPUT
element to the page dynamically after we initialize the DataTable by doing the following:
// Get the table element and initialize it
let table = $('#myTable'),
settings = {
// Assign your settings here
},
dataTable = table.DataTable(settings);
// Now that the DataTable is initialized, the plugin has finished making DOM
// modifications. Let's search for their INPUT and add our own before it.
let tableId = table.attr('id'),
searchInput = table
.parents('.dataTables_wrapper')
.find('input[type=search]'),
ourInput = $(document.createElement('input'))
.attr({
type: 'search',
'class': 'form-control form-control-sm',
'aria-controls': tableId,
});
There are a few important gotchas to consider:
- If we try to hide the original DataTables search field by passing the setting
bFilter: false
, DataTables disables all filtering of data, including our custom function. Instead, hide the default search by using thedom
option to specify a custom DOM structure to use. Thedom
option documentation lists some standard layouts for Bootstrap, Foundation, etc. Choose one as the value ofdom
and remove thef
character (thef
character represents thef
ilter in the DOM option). - You may be tempted to pass the value of your custom
INPUT
into thedataTables.search(string)
method. If you do this, DataTables will also set the value you pass as the search value in the default search field. Because the DataTables search runs before your custom functions, this may prematurely filter out rows of data you care about.
Listen for keyup
events and filter the table
Now that we have our new INPUT
element, we can listen for keyup
events and tell the DataTable to redraw, filtering by the new input value:
let query = undefined;
ourInput.on('keyup', () => {
// Store the value of our INPUT in a higher scope so the filter function below
// can see the value. This way we do a DOM request for the value of the INPUT
// once, instead of in each execution of the filter function (which runs for
// each row of the table).
query = ourInput.val();
// If the INPUT is empty, we want to show all results, so store a value we can
// easily test for as empty later on
if (query === '') {
query = undefined;
}
// Redraw the DataTable, which then also runs all the
// $.fn.dataTable.ext.search functions
dataTable.draw();
});
// This is our filter that runs each time the table is redrawn
function filterDataTable(settings, data, dataIndex) {
// When undefined that means our INPUT is empty, so show all results
if (query === undefined) {
return true;
}
// We only want to use our INPUT on the TABLE where it is displayed, not
// other DataTables on the page, so we need to check that the ID of the table
// being drawn matches the tableId we stored in the last code block
if (settings.sTableId !== tableId) {
return true;
}
// Now we can run our own custom search logic. The below searches for exact
// strings on the first column in the table. This alone isn't too useful, so
// replace this portion with your own logic.
let columnIndex = 0;
return data[columnIndex] === query;
}
// Add our filter to the list of filters that is run
$.fn.dataTable.ext.search.push(filterDataTable);
Put it all together
I originally got turned onto this problem when a client asked if we could search inside a list of IP ranges. Their DataTable contained a column with IP ranges specified in CIDR notation. The npm package netmask can handle the CIDR notation, so we just needed to wire up netmask
into a custom DataTables search filter. Along the way I hit every single gotcha listed above :-p
AWS publishes a list of their public IP ranges which are loaded into the example below. The snippets of code above create and drive the "Search by IP" filter.