fix(router): preserve specificity for redirects
Previously when comparing which of multiple possible routes to choose in an ambiguous case, we looked at the specificity of the target of redirect matches rather than the original match. This meant that if a redirect used a whilecard, but redirected to a target that was a static path, we'd cound the static path's specificity instead of the wildcard. This change stores the specificity of the redirect on the RedirectInstruction. Closes #5933
This commit is contained in:
parent
9d28147acb
commit
a038bb9ae3
|
@ -281,9 +281,11 @@ export class UnresolvedInstruction extends Instruction {
|
||||||
|
|
||||||
export class RedirectInstruction extends ResolvedInstruction {
|
export class RedirectInstruction extends ResolvedInstruction {
|
||||||
constructor(component: ComponentInstruction, child: Instruction,
|
constructor(component: ComponentInstruction, child: Instruction,
|
||||||
auxInstruction: {[key: string]: Instruction}) {
|
auxInstruction: {[key: string]: Instruction}, private _specificity: string) {
|
||||||
super(component, child, auxInstruction);
|
super(component, child, auxInstruction);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get specificity(): string { return this._specificity; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -205,7 +205,7 @@ export class RouteRegistry {
|
||||||
var instruction =
|
var instruction =
|
||||||
this.generate(candidate.redirectTo, ancestorInstructions.concat([null]));
|
this.generate(candidate.redirectTo, ancestorInstructions.concat([null]));
|
||||||
return new RedirectInstruction(instruction.component, instruction.child,
|
return new RedirectInstruction(instruction.component, instruction.child,
|
||||||
instruction.auxInstruction);
|
instruction.auxInstruction, candidate.specificity);
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,12 @@ import {
|
||||||
} from 'angular2/router';
|
} from 'angular2/router';
|
||||||
import {PromiseWrapper} from 'angular2/src/facade/async';
|
import {PromiseWrapper} from 'angular2/src/facade/async';
|
||||||
|
|
||||||
|
@Component({selector: 'goodbye-cmp', template: `{{farewell}}`})
|
||||||
|
export class GoodbyeCmp {
|
||||||
|
farewell: string;
|
||||||
|
constructor() { this.farewell = 'goodbye'; }
|
||||||
|
}
|
||||||
|
|
||||||
@Component({selector: 'hello-cmp', template: `{{greeting}}`})
|
@Component({selector: 'hello-cmp', template: `{{greeting}}`})
|
||||||
export class HelloCmp {
|
export class HelloCmp {
|
||||||
greeting: string;
|
greeting: string;
|
||||||
|
|
|
@ -25,7 +25,7 @@ import {
|
||||||
} from 'angular2/src/router/route_config_decorator';
|
} from 'angular2/src/router/route_config_decorator';
|
||||||
|
|
||||||
import {TEST_ROUTER_PROVIDERS, RootCmp, compile} from './util';
|
import {TEST_ROUTER_PROVIDERS, RootCmp, compile} from './util';
|
||||||
import {HelloCmp, RedirectToParentCmp} from './impl/fixture_components';
|
import {HelloCmp, GoodbyeCmp, RedirectToParentCmp} from './impl/fixture_components';
|
||||||
|
|
||||||
var cmpInstanceCount;
|
var cmpInstanceCount;
|
||||||
var childCmpInstanceCount;
|
var childCmpInstanceCount;
|
||||||
|
@ -117,5 +117,24 @@ export function main() {
|
||||||
async.done();
|
async.done();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
|
||||||
|
it('should not redirect when redirect is less specific than other matching routes',
|
||||||
|
inject([AsyncTestCompleter, Location], (async, location) => {
|
||||||
|
compile(tcb)
|
||||||
|
.then((rtc) => {rootTC = rtc})
|
||||||
|
.then((_) => rtr.config([
|
||||||
|
new Route({path: '/foo', component: HelloCmp, name: 'Hello'}),
|
||||||
|
new Route({path: '/:param', component: GoodbyeCmp, name: 'Goodbye'}),
|
||||||
|
new Redirect({path: '/*rest', redirectTo: ['/Hello']})
|
||||||
|
]))
|
||||||
|
.then((_) => rtr.navigateByUrl('/bye'))
|
||||||
|
.then((_) => {
|
||||||
|
rootTC.detectChanges();
|
||||||
|
expect(rootTC.debugElement.nativeElement).toHaveText('goodbye');
|
||||||
|
expect(location.urlChanges).toEqual(['/bye']);
|
||||||
|
async.done();
|
||||||
|
});
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue