Skip to content

Commit

Permalink
feat(router): add params to navigate()
Browse files Browse the repository at this point in the history
Adds a `params` to the `navigate()` function, so that
```
router.navigate('#/main/:id/details', undefined, { id: 123, name: "onetwothree" })
```
results in navigation to
```
'#/main/123/details?name=onetwothree'
```

In other words, found named params are placed in the url, not found ones are added to the querystring.
  • Loading branch information
jwx committed Oct 18, 2017
1 parent dffe907 commit fb2c8a6
Showing 1 changed file with 44 additions and 4 deletions.
48 changes: 44 additions & 4 deletions src/router.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import {RouteRecognizer} from 'aurelia-route-recognizer';
import {Container} from 'aurelia-dependency-injection';
import {History} from 'aurelia-history';
import {buildQueryString, parseQueryString} from 'aurelia-path';
import {NavigationInstruction} from './navigation-instruction';
import {NavModel} from './nav-model';
import {RouterConfiguration} from './router-configuration';
Expand Down Expand Up @@ -164,12 +165,18 @@ export class Router {
*
* @param fragment The URL fragment to use as the navigation destination.
* @param options The navigation options.
* @param params The parameters to be used when populating the url pattern.
*/
navigate(fragment: string, options?: any): boolean {
navigate(fragment: string, options?: any, params?: any): boolean {
if (!this.isConfigured && this.parent) {
return this.parent.navigate(fragment, options);
}

if (params) {
let safeParams = Object.assign({}, params);
fragment = this.applyParams(fragment, safeParams);
}

this.isExplicitNavigation = true;
return this.history.navigate(_resolveUrl(fragment, this.baseUrl, this.history._hasPushState), options);
}
Expand Down Expand Up @@ -207,6 +214,39 @@ export class Router {
return childRouter;
}

/**
* Generates a URL fragment with parameters applied to it.
*
* @param fragment The route fragment whose pattern should have parameters applied to it.
* @param params The route params to be used to populate the route pattern.
* @returns {string} A string containing the generated URL fragment.
*/
applyParams(fragment: string, params: any): string {
let fragments = fragment.split('/');
let consumed = {};
for (let i = 0, ilen = fragments.length; i < ilen; ++i) {
let segment = fragments[i];

// Try to parse a parameter :param?
let match = segment.match(/^:([^?]+)(\?)?$/);
if (match) {
let [, name, optional] = match;
fragments[i] = params[name];
consumed[name] = true;
}
}
fragment = fragments.join('/');

for (let key of Object.keys(consumed)) {
delete params[key];
}
let query = buildQueryString(params);
if (query && query.length) {
fragment += (fragment.indexOf('?') < 0 ? '?' : '&') + query;
}
return fragment;
}

/**
* Generates a URL fragment matching the specified route pattern.
*
Expand Down Expand Up @@ -415,7 +455,7 @@ export class Router {
if (results && results.length) {
let first = results[0];
let instruction = new NavigationInstruction(Object.assign({}, instructionInit, {
params: first.params,
params: Object.assign({}, first.params, parseQueryString(queryString)),
queryParams: first.queryParams || results.queryParams,
config: first.config || first.handler
}));
Expand All @@ -429,7 +469,7 @@ export class Router {
return Promise.resolve(instruction);
} else if (this.catchAllHandler) {
let instruction = new NavigationInstruction(Object.assign({}, instructionInit, {
params: { path: fragment },
params: Object.assign({ path: fragment }, parseQueryString(queryString)),
queryParams: results ? results.queryParams : {},
config: null // config will be created by the catchAllHandler
}));
Expand All @@ -442,7 +482,7 @@ export class Router {
let newParentInstruction = this._findParentInstructionFromRouter(router, parentInstruction);

let instruction = new NavigationInstruction(Object.assign({}, instructionInit, {
params: { path: fragment },
params: Object.assign({ path: fragment }, parseQueryString(queryString)),
queryParams: results ? results.queryParams : {},
router: router,
parentInstruction: newParentInstruction,
Expand Down

0 comments on commit fb2c8a6

Please sign in to comment.