Angular Router provides powerful extension points that allow you to customize how routes behave in your application. While the default routing behavior works well for most applications, specific requirements often demand custom implementations for performance optimization, specialized URL handling, or complex routing logic.
Route customization can become valuable when your application needs:
- Component state preservation across navigations to avoid re-fetching data
- Strategic lazy module loading based on user behavior or network conditions
- External URL integration or handling Angular routes alongside legacy systems
- Dynamic route matching based on runtime conditions beyond simple path patterns
NOTE: Before implementing custom strategies, ensure the default router behavior doesn't meet your needs. Angular's default routing is optimized for common use cases and provides the best balance of performance and simplicity. Customizing route strategies can create additional code complexity and have performance implications on memory usage if not carefully managed.
Router configuration options
The withRouterConfig or RouterModule.forRoot allows providing additional RouterConfigOptions to adjust the Router’s behavior.
Handle canceled navigations
canceledNavigationResolution controls how the Router restores browser history when a navigation is canceled. The default value is 'replace', which reverts to the pre-navigation URL with location.replaceState. In practice, this means that any time the address bar has already been updated for the navigation, such as with the browser back or forward buttons, the history entry is overwritten with the "rollback" if the navigation fails or is rejected by a guard.
Switching to 'computed' keeps the in-flight history index in sync with the Angular navigation, so canceling a back button navigation triggers a forward navigation (and vice versa) to return to the original page.
This setting is most helpful when your app uses urlUpdateStrategy: 'eager' or when guards frequently cancel popstate navigations initiated by the browser.
provideRouter(routes, withRouterConfig({ canceledNavigationResolution: 'computed' }));
React to same-URL navigations
onSameUrlNavigation configures what should happen when the user asks to navigate to the current URL. The default 'ignore' skips work, while 'reload' re-runs guards and resolvers and refreshes component instances.
This is useful when you want repeated clicks on a list filter, left-nav item, or refresh button to trigger new data retrieval even though the URL does not change.
provideRouter(routes, withRouterConfig({ onSameUrlNavigation: 'reload' }));
You can also control this behavior on individual navigations rather than globally. This allows you to keep the keep the default of 'ignore' while selectively enabling reload behavior for specific use cases:
router.navigate(['/some-path'], { onSameUrlNavigation: 'reload' });
Control parameter inheritance
paramsInheritanceStrategy defines how route parameters and data flow from parent routes.
With the default 'emptyOnly', child routes inherit params only when their path is empty or the parent does not declare a component.
provideRouter(routes, withRouterConfig({ paramsInheritanceStrategy: 'always' }));
export const routes: Routes = [ { path: 'org/:orgId', component: Organization, children: [ { path: 'projects/:projectId', component: Project, children: [ { path: 'customers/:customerId', component: Customer } ] } ] }];
@Component({ /* ... */})export class CustomerComponent { private route = inject(ActivatedRoute); orgId = this.route.parent?.parent?.snapshot.params['orgId']; projectId = this.route.parent?.snapshot.params['projectId']; customerId = this.route.snapshot.params['customerId'];}
Using 'always' ensures matrix parameters, route data, and resolved values are available further down the route tree—handy when you share contextual identifiers across feature areas such as /org/:orgId/projects/:projectId/customers/:customerId.
@Component({ /* ... */})export class CustomerComponent { private route = inject(ActivatedRoute); // All parent parameters are available directly orgId = this.route.snapshot.params['orgId']; projectId = this.route.snapshot.params['projectId']; customerId = this.route.snapshot.params['customerId'];}
Decide when the URL updates
urlUpdateStrategy determines when Angular writes to the browser address bar. The default 'deferred' waits for a successful navigation before changing the URL. Use 'eager' to update immediately when navigation starts. Eager updates make it easier to surface the attempted URL if navigation fails due to guards or errors, but can briefly show an in-progress URL if you have long-running guards.
Consider this when your analytics pipeline needs to see the attempted route even if guards block it.
provideRouter(routes, withRouterConfig({ urlUpdateStrategy: 'eager' }));
Choose default query parameter handling
defaultQueryParamsHandling sets the fallback behavior for Router.createUrlTree when the call does not specify queryParamsHandling. 'replace' is the default and swaps out the existing query string. 'merge' combines the provided values with the current ones, and 'preserve' keeps the existing query parameters unless you explicitly supply new ones.
provideRouter(routes, withRouterConfig({ defaultQueryParamsHandling: 'merge' }));
This is especially helpful for search and filter pages to automatically retain existing filters when additional parameters are provided.
Angular Router exposes four main areas for customization:
Route reuse strategy
Route reuse strategy controls whether Angular destroys and recreates components during navigation or preserves them for reuse. By default, Angular destroys component instances when navigating away from a route and creates new instances when navigating back.
When to implement route reuse
Custom route reuse strategies benefit applications that need:
- Form state preservation - Keep partially completed forms when users navigate away and return
- Expensive data retention - Avoid re-fetching large datasets or complex calculations
- Scroll position maintenance - Preserve scroll positions in long lists or infinite scroll implementations
- Tab-like interfaces - Maintain component state when switching between tabs
Creating a custom route reuse strategy
Angular's RouteReuseStrategy class allows you to customize navigation behavior through the concept of "detached route handles."
"Detached route handles" are Angular's way of storing component instances and their entire view hierarchy. When a route is detached, Angular preserves the component instance, its child components, and all associated state in memory. This preserved state can later be reattached when navigating back to the route.
The RouteReuseStrategy class provides five methods that control the lifecycle of route components:
| Method | Description |
|---|---|
shouldDetach |
Determines if a route should be stored for later reuse when navigating away |
store |
Stores the detached route handle when shouldDetach returns true |
shouldAttach |
Determines if a stored route should be reattached when navigating to it |
retrieve |
Returns the previously stored route handle for reattachment |
shouldReuseRoute |
Determines if the router should reuse the current route instance instead of destroying it during navigation |
The following example demonstrates a custom route reuse strategy that selectively preserves component state based on route metadata:
import { RouteReuseStrategy, ActivatedRouteSnapshot, DetachedRouteHandle } from '@angular/router';import { Injectable } from '@angular/core';@Injectable()export class CustomRouteReuseStrategy implements RouteReuseStrategy { private handlers = new Map<string, DetachedRouteHandle>(); shouldDetach(route: ActivatedRouteSnapshot): boolean { // Determines if a route should be stored for later reuse return route.data['reuse'] === true; } store(route: ActivatedRouteSnapshot, handle: DetachedRouteHandle | null): void { // Stores the detached route handle when shouldDetach returns true if (handle && route.data['reuse'] === true) { const key = this.getRouteKey(route); this.handlers.set(key, handle); } } shouldAttach(route: ActivatedRouteSnapshot): boolean { // Checks if a stored route should be reattached const key = this.getRouteKey(route); return route.data['reuse'] === true && this.handlers.has(key); } retrieve(route: ActivatedRouteSnapshot): DetachedRouteHandle | null { // Returns the stored route handle for reattachment const key = this.getRouteKey(route); return route.data['reuse'] === true ? this.handlers.get(key) ?? null : null; } shouldReuseRoute(future: ActivatedRouteSnapshot, curr: ActivatedRouteSnapshot): boolean { // Determines if the router should reuse the current route instance return future.routeConfig === curr.routeConfig; } private getRouteKey(route: ActivatedRouteSnapshot): string { return route.routeConfig ?? ''; }}
Configuring a route to use a custom route reuse strategy
Routes can opt into reuse behavior through route configuration metadata. This approach keeps the reuse logic separate from component code, making it easy to adjust behavior without modifying components:
export const routes: Routes = [ { path: 'products', component: ProductListComponent, data: { reuse: true } // Component state persists across navigations }, { path: 'products/:id', component: ProductDetailComponent, // No reuse flag - component recreates on each navigation }, { path: 'search', component: SearchComponent, data: { reuse: true } // Preserves search results and filter state }];
You can also configure a custom route reuse strategy at the application level through Angular's dependency injection system. In this case, Angular creates a single instance of the strategy that manages all route reuse decisions throughout the application:
export const appConfig: ApplicationConfig = { providers: [ provideRouter(routes), { provide: RouteReuseStrategy, useClass: CustomRouteReuseStrategy } ]};
Preloading strategy
Preloading strategies determine when Angular loads lazy-loaded route modules in the background. While lazy loading improves initial load time by deferring module downloads, users still experience a delay when first navigating to a lazy route. Preloading strategies eliminate this delay by loading modules before users request them.
Built-in preloading strategies
Angular provides two preloading strategies out of the box:
| Strategy | Description |
|---|---|
NoPreloading |
The default strategy that disables all preloading. In other words, modules only load when users navigate to them |
PreloadAllModules |
Loads all lazy-loaded modules immediately after the initial navigation |
The PreloadAllModules strategy can be configured as follows:
import { ApplicationConfig } from '@angular/core';import { provideRouter, withPreloading, PreloadAllModules } from '@angular/router';import { routes } from './app.routes';export const appConfig: ApplicationConfig = { providers: [ provideRouter( routes, withPreloading(PreloadAllModules) ) ]};
The PreloadAllModules strategy works well for small to medium applications where downloading all modules doesn't significantly impact performance. However, larger applications with many feature modules might benefit from more selective preloading.
Creating a custom preloading strategy
Custom preloading strategies implement the PreloadingStrategy interface, which requires a single preload method. This method receives the route configuration and a function that triggers the actual module load. The strategy returns an Observable that emits when preloading completes or an empty Observable to skip preloading:
import { Injectable } from '@angular/core';import { PreloadingStrategy, Route } from '@angular/router';import { Observable, of, timer } from 'rxjs';import { mergeMap } from 'rxjs/operators';@Injectable()export class SelectivePreloadingStrategy implements PreloadingStrategy { preload(route: Route, load: () => Observable<any>): Observable<any> { // Only preload routes marked with data: { preload: true } if (route.data?.['preload']) { return load(); } return of(null); }}
This selective strategy checks route metadata to determine preloading behavior. Routes can opt into preloading through their configuration:
import { Routes } from '@angular/router';export const routes: Routes = [ { path: 'dashboard', loadChildren: () => import('./dashboard/dashboard.routes'), data: { preload: true } // Preload immediately after initial navigation }, { path: 'reports', loadChildren: () => import('./reports/reports.routes'), data: { preload: false } // Only load when user navigates to reports }, { path: 'admin', loadChildren: () => import('./admin/admin.routes') // No preload flag - won't be preloaded }];
Performance considerations for preloading
Preloading impacts both network usage and memory consumption. Each preloaded module consumes bandwidth and increases the application's memory footprint. Mobile users on metered connections might prefer minimal preloading, while desktop users on fast networks can handle aggressive preloading strategies.
The timing of preloading also matters. Immediate preloading after initial load might compete with other critical resources like images or API calls. Strategies should consider the application's post-load behavior and coordinate with other background tasks to avoid performance degradation.
Browser resource limits also affect preloading behavior. Browsers limit concurrent HTTP connections, so aggressive preloading might queue behind other requests. Service workers can help by providing fine-grained control over caching and network requests, complementing the preloading strategy.
URL handling strategy
URL handling strategies determine which URLs the Angular router processes versus which ones it ignores. By default, Angular attempts to handle all navigation events within the application, but real-world applications often need to coexist with other systems, handle external links, or integrate with legacy applications that manage their own routes.
The UrlHandlingStrategy class gives you control over this boundary between Angular-managed routes and external URLs. This becomes essential when migrating applications to Angular incrementally or when Angular applications need to share URL space with other frameworks.
Implementing a custom URL handling strategy
Custom URL handling strategies extend the UrlHandlingStrategy class and implement three methods. The shouldProcessUrl method determines whether Angular should handle a given URL, extract returns the portion of the URL that Angular should process, and merge combines the URL fragment with the rest of the URL:
import { Injectable } from '@angular/core';import { UrlHandlingStrategy, UrlTree } from '@angular/router';@Injectable()export class CustomUrlHandlingStrategy implements UrlHandlingStrategy { shouldProcessUrl(url: UrlTree): boolean { // Only handle URLs that start with /app or /admin return url.toString().startsWith('/app') || url.toString().startsWith('/admin'); } extract(url: UrlTree): UrlTree { // Return the URL unchanged if we should process it return url; } merge(newUrlPart: UrlTree, rawUrl: UrlTree): UrlTree { // Combine the URL fragment with the rest of the URL return newUrlPart; }}
This strategy creates clear boundaries in the URL space. Angular handles /app and /admin paths while ignoring everything else. This pattern works well when migrating legacy applications where Angular controls specific sections while the legacy system maintains others.
Configuring a custom URL handling strategy
You can register a custom strategy through Angular's dependency injection system:
import { ApplicationConfig } from '@angular/core';import { provideRouter } from '@angular/router';import { UrlHandlingStrategy } from '@angular/router';export const appConfig: ApplicationConfig = { providers: [ provideRouter(routes), { provide: UrlHandlingStrategy, useClass: CustomUrlHandlingStrategy } ]};
Custom route matchers
By default, Angular's router iterates through routes in the order they're defined, attempting to match the URL path against each route's path pattern. It supports static segments, parameterized segments (:id), and wildcards (**). The first route that matches wins, and the router stops searching.
When applications require more sophisticated matching logic based on runtime conditions, complex URL patterns, or other custom rules, custom matchers provide this flexibility without compromising the simplicity of standard routes.
The router evaluates custom matchers during the route matching phase, before path matching occurs. When a matcher returns a successful match, it can also extract parameters from the URL, making them available to the activated component just like standard route parameters.
Creating a custom matcher
A custom matcher is a function that receives URL segments and returns either a match result with consumed segments and parameters, or null to indicate no match. The matcher function runs before Angular evaluates the route's path property:
import { Route, UrlSegment, UrlSegmentGroup, UrlMatchResult } from '@angular/router';export function customMatcher( segments: UrlSegment[], group: UrlSegmentGroup, route: Route): UrlMatchResult | null { // Matching logic here if (matchSuccessful) { return { consumed: segments, posParams: { paramName: new UrlSegment('paramValue', {}) } }; } return null;}
Implementing version-based routing
Consider an API documentation site that needs to route based on version numbers in the URL. Different versions might have different component structures or feature sets:
import { Routes, UrlSegment, UrlMatchResult } from '@angular/router';export function versionMatcher(segments: UrlSegment[]): UrlMatchResult | null { // Match patterns like /v1/docs, /v2.1/docs, /v3.0.1/docs if (segments.length >= 2 && segments[0].path.match(/^v\d+(\.\d+)*$/)) { return { consumed: segments.slice(0, 2), // Consume version and 'docs' posParams: { version: segments[0], // Make version available as a parameter section: segments[1] // Make section available too } }; } return null;}// Route configurationexport const routes: Routes = [ { matcher: versionMatcher, component: DocumentationComponent }, { path: 'latest/docs', redirectTo: 'v3/docs' }];
The component receives the extracted parameters through route inputs:
import { Component, input, inject } from '@angular/core';import { resource } from '@angular/core';@Component({ selector: 'app-documentation', template: ` @if (documentation.isLoading()) { <div>Loading documentation...</div> } @else if (documentation.error()) { <div>Error loading documentation</div> } @else if (documentation.value(); as docs) { <article>{{ docs.content }}</article> } `})export class DocumentationComponent { // Route parameters are automatically bound to signal inputs version = input.required<string>(); // Receives the version parameter section = input.required<string>(); // Receives the section parameter private docsService = inject(DocumentationService); // Resource automatically loads documentation when version or section changes documentation = resource({ params: () => { if (!this.version() || !this.section()) return; return { version: this.version(), section: this.section() } }, loader: ({ params }) => { return this.docsService.loadDocumentation(params.version, params.section); } })}
Locale-aware routing
International applications often encode locale information in URLs. A custom matcher can extract locale codes and route to appropriate components while making the locale available as a parameter:
// Supported localesconst locales = ['en', 'es', 'fr', 'de', 'ja', 'zh'];export function localeMatcher(segments: UrlSegment[]): UrlMatchResult | null { if (segments.length > 0) { const potentialLocale = segments[0].path; if (locales.includes(potentialLocale)) { // This is a locale prefix, consume it and continue matching return { consumed: [segments[0]], posParams: { locale: segments[0] } }; } else { // No locale prefix, use default locale return { consumed: [], // Don't consume any segments posParams: { locale: new UrlSegment('en', {}) } }; } } return null;}
Complex business logic matching
Custom matchers excel at implementing business rules that would be awkward to express in path patterns. Consider an e-commerce site where product URLs follow different patterns based on product type:
export function productMatcher(segments: UrlSegment[]): UrlMatchResult | null { if (segments.length === 0) return null; const firstSegment = segments[0].path; // Books: /isbn-1234567890 if (firstSegment.startsWith('isbn-')) { return { consumed: [segments[0]], posParams: { productType: new UrlSegment('book', {}), identifier: new UrlSegment(firstSegment.substring(5), {}) } }; } // Electronics: /sku/ABC123 if (firstSegment === 'sku' && segments.length > 1) { return { consumed: segments.slice(0, 2), posParams: { productType: new UrlSegment('electronics', {}), identifier: segments[1] } }; } // Clothing: /style/BRAND/ITEM if (firstSegment === 'style' && segments.length > 2) { return { consumed: segments.slice(0, 3), posParams: { productType: new UrlSegment('clothing', {}), brand: segments[1], identifier: segments[2] } }; } return null;}
Performance considerations for custom matchers
Custom matchers run for every navigation attempt until a match is found. As a result, complex matching logic can impact navigation performance, especially in applications with many routes. Keep matchers focused and efficient:
- Return early when a match is impossible
- Avoid expensive operations like API calls or complex regular expressions
- Consider caching results for repeated URL patterns
While custom matchers solve complex routing requirements elegantly, overuse can make route configuration harder to understand and maintain. Reserve custom matchers for scenarios where standard path matching genuinely falls short.