I'm using DataTables with serverSide
processing. On popstate
the table is redrawn with loaded values. Everything is working nicely except for one problem: when I go from a state where the page for the loaded state isn't available, it won't load the correct page for the loaded state, if that makes sense.
So for example:
Edit: I realize now that this is because of this in the page() doc:
Note that if you attempt to show a page which does not exist using the set format of this method, DataTables will not throw an error, but rather reset the paging to the first page.
The problem is that with serverSide
processing, DataTables doesn't know how many pages there are until the table is drawn, so when the number of pages increases after doing a manual draw()
, this leads to the problem I'm describing. There doesn't seem to be a way to coerce DataTables to go to the page anyway, so the only option would be to draw
twice.
Here's the (simplified) code:
$(document).ready(function() {
myTable.on("draw", function() {
setState(); // history.replaceState or history.pushState
saveTableOrder(); // save to localStorage
saveTableLength(); // save to localStorage
});
$(window).on("popstate", function() {
myTable
.search(loadTableSearch()) // load from history.state
.order(loadTableOrder()) // load from history.state or localStorage
.page.len(loadTableLength()) // load from history.state or localStorage
.page(loadTablePage()-1) // zero-indexed; load from history.state
.draw(false); // same result with "full-hold" argument
});
});
It seems that, in these cases, the only way around this is to either draw the table twice, or to destroy and reinitialize the table. Below is an example of how to draw the table twice.
Edit: It's actually better to perform the redraw with the preDraw
event, because it will be faster and allows you to cancel the first draw
.
The issue is that preDraw
fires twice, before and after the ajax request is complete, and the new page count is only available the second time. So you will need to keep track of this and perform the redraw the second time.
$(document).ready(function() {
let redraw = false, predraw = false;
myTable
.on("preDraw", function() {
if (predraw) {
predraw = false;
redraw = false; // redraw is always true when predraw is true
// make sure we don't just reload the same page
let page = loadTablePage();
if (page > myTable.page.info().pages) page = 1;
if (page-1 != myTable.page()) {
myTable.page(page-1).draw("page");
return false; // cancel first draw
}
}
else if (redraw) predraw = true; // We only need to capture the second preDraw event if we need to redraw the table.
})
.on("draw", function() {
setState(); // history.replaceState or history.pushState
saveTableOrder(); // save to localStorage
saveTableLength(); // save to localStorage
});
$(window).on("popstate", function() {
let page = loadTablePage(); // load from history.state
redraw = page > 1 && page > myTable.page.info().pages;
myTable
.search(loadTableSearch()) // load from history.state
.order(loadTableOrder()) // load from history.state or localStorage
.page.len(loadTableLength()) // load from history.state or localStorage
.page(page-1)
.draw(false);
});
});