Lifecycle hooks
Learn about the different routing hooks and how to leverage those in terms of dis/allow loading or unloading as well as performing setup and teardown of a view.
Inside your routable components which implement the IRouteViewModel
interface, there are certain methods that are called at different points of the routing lifecycle. These lifecycle hooks allow you to run code inside of your components such as fetch data or change the UI itself.
Router lifecycle hook methods are all completely optional. You only have to implement the methods you require. The router will only call a method if it has been specified inside of your routable component. All lifecycle hook methods also support returning a promise and can be asynchronous.
If you are working with components you are rendering, implementing IRouteViewModel
will ensure that your code editor provides you with intellisense to make working with these lifecycle hooks in the appropriate way a lot easier.
Using the canLoad
and canUnload
hooks you can determine whether to allow or disallow navigation to and from a route respectively. The loading
and unloading
hooks are meant to be used for performing setup and clean up activities respectively for a view. Note that all of these hooks can return a promise, which will be awaited by the router-lite pipeline. These hooks are discussed in details in the following section.
In case you are looking for the global/shared routing hooks, there is a separate documentation section dedicated for that.
canLoad
canLoad
The canLoad
method is called upon attempting to load the component. It allows you to determine if the component should be loaded or not. If your component relies on some precondition being fulfilled before being allowed to render, this is the method you would use.
To disallow loading the component you can return a boolean
false
. You can also return a navigation instruction to navigate the user to a different view. These are discussed in the following sections.
Allow or disallowed loading components
The following example shows that a parameterized route, such as /c1/:id?
, can only be loaded if the value of id
is an even number. Note that the value of the id
parameter can be grabbed from the the first argument (params
) to the canLoad
method.
You can also see this example in action below.
Redirect to another view from canLoad
canLoad
Not only can we allow or disallow the component to be loaded, but we can also redirect. The simplest way is to return a string path from canLoad
. In the following example, we re-write the previous example, but instead of returning false
, we return a path, where the user will be redirected.
You can also see this example in action below.
If you prefer a more structured navigation instructions then you can also do so. Following is the same example using route-id and parameters object.
Note that you can also choose to return a sibling navigation instructions. This can be done by returning an array of navigation instructions.
You can also see the example in action below.
Accessing fragment and query
Apart from accessing the route parameter, the query and the fragment associated with the URL can also be accessed inside the canLoad
hook. To this end, you can use the second argument (next
) to this method.
The following example shows that id
query parameter is checked whether that is an even number or not. If that condition does not hold, then user is redirected to a different view with the query and fragment.
You can also see the example in action below.
loading
loading
The loading
method is called when your component is navigated to. If your route has any parameters supplied, they will be provided to the loading
method as an object with one or more parameters as the first argument.
In many ways, the loading
method is the same as canLoad
with the exception that loading
cannot prevent the component from loading. Where canLoad
can be used to redirect users away from the component, the loading
method cannot.
This lifecycle hook can be utilized to perform setup; for example, fetching data from backend API etc.
All of the above code examples for canLoad
can be used with load
and will work the same with the exception of being able to return true
or false
boolean values to prevent the component being loaded.
One of the examples is refactored using loading
hook that is shown below.
Following is an additional example, that shows that you can use the next.title
property to dynamically set the route title from the loading
hook.
canUnload
canUnload
The canUnload
method is called when a user attempts to leave a routed view. The first argument (next
) of this hook is a RouteNode
which provides information about the next route.
This hook is like the canLoad
method but inverse. You can return a boolean false
from this method, to disallow the router-lite to navigate away from the current component.
The following example shows that before navigating away, the user is shown a confirmation prompt. If the user agrees to navigate way, then the navigation is performed. The navigation is cancelled, if the user does not confirm.
You can see this example in action below.
unloading
unloading
The unloading
hook is called when the user navigates away from the current component. The first argument (next
) of this hook is a RouteNode
which provides information about the next route.
This hook is like the loading
method but inverse.
The following example shows that a unloading
hook logs the event of unloading the component.
This can also be seen in the live example below.
Order of invocations
For completeness it needs to be noted that the canLoad
hook is invoked before loading
and canUnload
hook is invoked before unloading
. In the context of swapping two views/components it is as follows.
canUnload
hook (when present) of the current component is invoked.canLoad
hook (when present) of the next component (assuming that thecanUnload
returnedtrue
) is invoked.unloading
hook (when present) of the current component is invoked.loading
hook (when present) of the current component is invoked.
Note that the last 2 steps may run in parallel, if the hooks are asynchronous.
Order of invocations of component lifecycle hooks
The component lifecycle hooks are invoked bottom-up. As an example, let us assume that we have the following constellation of components.
In this case, the component lifecycle hooks are invoked in the following order.
component-two
attached
.component-one
attached
.app-root
attached
.
This is also the same for the router-lite, except for the "application root" component. Tweaking the example above slightly, let us assume that we have the following constellation of components.
In this case, the component lifecycle hooks are invoked in the following order.
app-root
attached
.component-two
attached
.component-one
attached
.routed-view
attached
.
Note that the application root is attached before any other components are attached. This happens because the router-lite starts loading the first route only after the app-root, and thereby the viewport(s) it is hosting, are fully activated/attached. In order to load a route, the router needs registered viewports. The registration process of a viewport only happens during the attaching
phase of a viewport. More details on this topic, can be found in this GitHub issue.
Last updated