The small steps
But let me first go through the list of small changes that I’ve done:
Proper promise resolution/rejection
While makeRequest()
did return a promise, it was somewhat useless, as it carried no information about the
request; not even whether it was successful. That is no longer true: the returned promise is now resolved with
the parsed response body, or rejected with the thrown error.
Now if you dispatch the request manually, you can chain custom behavior after the response is processed by Naja:
naja.makeRequest(method, url)
.then((response) => {
// ...
})
.catch((error) => {
// ...
});
Aborted request is not an error
Another change concerns aborted requests. Naja used to (mistakenly) trigger the error handling for aborted requests, whereas the circumstances that lead to aborting the request do only rarely indicate an erroneous state. Usually it’s just the user navigating away, or Naja’s UniqueExtension kicking in because another request was dispatched.
Therefore aborted requests no longer trigger error
event, and should be processed – if ever needed – in a
specific abort
event. Note that aborted requests also trigger the complete
event.
History under your control
HistoryHandler
, a core component responsible for modifying the browser’s history, can now be configured on the
front-end side via a history
key in the options
object or data-naja-history
attribute on the AJAXified element.
By default, Naja pushes a new state to the history, but you can instruct it:
- to replace the current state instead of pushing a new one, by setting
options.history
to ‘replace’or
data-naja-history=“replace”`, or - to keep the request off-the-record and not modify the history state at all, by setting
options.history
tofalse
ordata-naja-history="off"
.
Custom selector
Last but not least, the bound selector was hard-coded to .ajax
, a default taken from nette.ajax.js library.
To accommodate for a wider range of use cases, the selector can now be changed:
naja.uiHandler.selector = '[data-naja]';
naja.uiHandler.selector = ':not(.synchronous)';
naja.uiHandler.selector = '';
As you can see, you can easily switch the scheme from opt-in asynchronicity to opt-out asynchronicity, which gives you far more opportunities to shoot yourself in the foot by dispatching cross-origin AJAX requests (and believe me, those are tricky).
To alleviate this, UIHandler now checks the target URL and does not dispatch the request asynchronously if it points to an external resource with a disallowed origin. You can of course configure the allowed origins:
naja.uiHandler.allowedOrigins.push('https://trusted.origin.com:4000');
The current origin is allowed by default, so you can point asynchronous links to absolute URLs as well.
The big leap
Now to the big change. It’s funny because it does not much affect the code of Naja itself, but still has a lot of practical implications: the tests now run in the browser.
Yes, in the end I’ve managed to make them run in the browser instead of Node. I’ve used Karma and set up an OSS account on SauceLabs so that the whole test suite is executed on a wide range of browsers including Internet Explorer and mobile device simulators, for every push.
I’ve invested a tremendous amount of time into it – something close to a whole weekend and a half – but the benefits have already outweighed the cost: it has helped me discover a few incompatibilities in quite critical sections of the code; surprisingly not only with old Internet Explorers, but also recent versions of Safari.
Besides, I’ve learnt so much in the process, which is always a good thing!
I’m really excited about this new release and I hope you will enjoy it as much as I do. Go and give it a try,
it’s just one npm update
away :)