fix(router): fix lazy loading of aux routes (#23459)
Fixes #10981 PR Close #23459
This commit is contained in:
parent
70ef061fa6
commit
5731d0741a
22
packages/router/src/components/empty_outlet.ts
Normal file
22
packages/router/src/components/empty_outlet.ts
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
/**
|
||||||
|
* @license
|
||||||
|
* Copyright Google Inc. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style license that can be
|
||||||
|
* found in the LICENSE file at https://angular.io/license
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This component is used internally within the router to be a placeholder when an empty
|
||||||
|
* router-outlet is needed. For example, with a config such as:
|
||||||
|
*
|
||||||
|
* `{path: 'parent', outlet: 'nav', children: [...]}`
|
||||||
|
*
|
||||||
|
* In order to render, there needs to be a component on this config, which will default
|
||||||
|
* to this `EmptyOutletComponent`.
|
||||||
|
*/
|
||||||
|
@Component({template: `<router-outlet></router-outlet>`})
|
||||||
|
export class EmptyOutletComponent {
|
||||||
|
}
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
import {NgModuleFactory, NgModuleRef, Type} from '@angular/core';
|
import {NgModuleFactory, NgModuleRef, Type} from '@angular/core';
|
||||||
import {Observable} from 'rxjs';
|
import {Observable} from 'rxjs';
|
||||||
|
import {EmptyOutletComponent} from './components/empty_outlet';
|
||||||
import {PRIMARY_OUTLET} from './shared';
|
import {PRIMARY_OUTLET} from './shared';
|
||||||
import {UrlSegment, UrlSegmentGroup} from './url_tree';
|
import {UrlSegment, UrlSegmentGroup} from './url_tree';
|
||||||
|
|
||||||
@ -412,9 +413,10 @@ function validateNode(route: Route, fullPath: string): void {
|
|||||||
if (Array.isArray(route)) {
|
if (Array.isArray(route)) {
|
||||||
throw new Error(`Invalid configuration of route '${fullPath}': Array cannot be specified`);
|
throw new Error(`Invalid configuration of route '${fullPath}': Array cannot be specified`);
|
||||||
}
|
}
|
||||||
if (!route.component && (route.outlet && route.outlet !== PRIMARY_OUTLET)) {
|
if (!route.component && !route.children && !route.loadChildren &&
|
||||||
|
(route.outlet && route.outlet !== PRIMARY_OUTLET)) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
`Invalid configuration of route '${fullPath}': a componentless route cannot have a named outlet set`);
|
`Invalid configuration of route '${fullPath}': a componentless route without children or loadChildren cannot have a named outlet set`);
|
||||||
}
|
}
|
||||||
if (route.redirectTo && route.children) {
|
if (route.redirectTo && route.children) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
@ -477,8 +479,14 @@ function getFullPath(parentPath: string, currentRoute: Route): string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
export function copyConfig(r: Route): Route {
|
* Makes a copy of the config and adds any default required properties.
|
||||||
const children = r.children && r.children.map(copyConfig);
|
*/
|
||||||
return children ? {...r, children} : {...r};
|
export function standardizeConfig(r: Route): Route {
|
||||||
|
const children = r.children && r.children.map(standardizeConfig);
|
||||||
|
const c = children ? {...r, children} : {...r};
|
||||||
|
if (!c.component && (children || c.loadChildren) && (c.outlet && c.outlet !== PRIMARY_OUTLET)) {
|
||||||
|
c.component = EmptyOutletComponent;
|
||||||
|
}
|
||||||
|
return c;
|
||||||
}
|
}
|
||||||
|
@ -7,5 +7,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
export {EmptyOutletComponent as ɵEmptyOutletComponent} from './components/empty_outlet';
|
||||||
export {ROUTER_PROVIDERS as ɵROUTER_PROVIDERS} from './router_module';
|
export {ROUTER_PROVIDERS as ɵROUTER_PROVIDERS} from './router_module';
|
||||||
export {flatten as ɵflatten} from './utils/collection';
|
export {flatten as ɵflatten} from './utils/collection';
|
||||||
|
@ -12,7 +12,7 @@ import {BehaviorSubject, Observable, Subject, Subscription, of } from 'rxjs';
|
|||||||
import {concatMap, map, mergeMap} from 'rxjs/operators';
|
import {concatMap, map, mergeMap} from 'rxjs/operators';
|
||||||
|
|
||||||
import {applyRedirects} from './apply_redirects';
|
import {applyRedirects} from './apply_redirects';
|
||||||
import {LoadedRouterConfig, QueryParamsHandling, Route, Routes, copyConfig, validateConfig} from './config';
|
import {LoadedRouterConfig, QueryParamsHandling, Route, Routes, standardizeConfig, validateConfig} from './config';
|
||||||
import {createRouterState} from './create_router_state';
|
import {createRouterState} from './create_router_state';
|
||||||
import {createUrlTree} from './create_url_tree';
|
import {createUrlTree} from './create_url_tree';
|
||||||
import {ActivationEnd, ChildActivationEnd, Event, GuardsCheckEnd, GuardsCheckStart, NavigationCancel, NavigationEnd, NavigationError, NavigationStart, NavigationTrigger, ResolveEnd, ResolveStart, RouteConfigLoadEnd, RouteConfigLoadStart, RoutesRecognized} from './events';
|
import {ActivationEnd, ChildActivationEnd, Event, GuardsCheckEnd, GuardsCheckStart, NavigationCancel, NavigationEnd, NavigationError, NavigationStart, NavigationTrigger, ResolveEnd, ResolveStart, RouteConfigLoadEnd, RouteConfigLoadStart, RoutesRecognized} from './events';
|
||||||
@ -357,7 +357,7 @@ export class Router {
|
|||||||
*/
|
*/
|
||||||
resetConfig(config: Routes): void {
|
resetConfig(config: Routes): void {
|
||||||
validateConfig(config);
|
validateConfig(config);
|
||||||
this.config = config.map(copyConfig);
|
this.config = config.map(standardizeConfig);
|
||||||
this.navigated = false;
|
this.navigated = false;
|
||||||
this.lastSuccessfulId = -1;
|
this.lastSuccessfulId = -1;
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ import {Compiler, InjectionToken, Injector, NgModuleFactory, NgModuleFactoryLoad
|
|||||||
// TODO(i): switch to fromPromise once it's expored in rxjs
|
// TODO(i): switch to fromPromise once it's expored in rxjs
|
||||||
import {Observable, from, of } from 'rxjs';
|
import {Observable, from, of } from 'rxjs';
|
||||||
import {map, mergeMap} from 'rxjs/operators';
|
import {map, mergeMap} from 'rxjs/operators';
|
||||||
import {LoadChildren, LoadedRouterConfig, Route, copyConfig} from './config';
|
import {LoadChildren, LoadedRouterConfig, Route, standardizeConfig} from './config';
|
||||||
import {flatten, wrapIntoObservable} from './utils/collection';
|
import {flatten, wrapIntoObservable} from './utils/collection';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -39,7 +39,8 @@ export class RouterConfigLoader {
|
|||||||
|
|
||||||
const module = factory.create(parentInjector);
|
const module = factory.create(parentInjector);
|
||||||
|
|
||||||
return new LoadedRouterConfig(flatten(module.injector.get(ROUTES)).map(copyConfig), module);
|
return new LoadedRouterConfig(
|
||||||
|
flatten(module.injector.get(ROUTES)).map(standardizeConfig), module);
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -11,6 +11,7 @@ import {ANALYZE_FOR_ENTRY_COMPONENTS, APP_BOOTSTRAP_LISTENER, APP_INITIALIZER, A
|
|||||||
import {ɵgetDOM as getDOM} from '@angular/platform-browser';
|
import {ɵgetDOM as getDOM} from '@angular/platform-browser';
|
||||||
import {Subject, of } from 'rxjs';
|
import {Subject, of } from 'rxjs';
|
||||||
|
|
||||||
|
import {EmptyOutletComponent} from './components/empty_outlet';
|
||||||
import {Route, Routes} from './config';
|
import {Route, Routes} from './config';
|
||||||
import {RouterLink, RouterLinkWithHref} from './directives/router_link';
|
import {RouterLink, RouterLinkWithHref} from './directives/router_link';
|
||||||
import {RouterLinkActive} from './directives/router_link_active';
|
import {RouterLinkActive} from './directives/router_link_active';
|
||||||
@ -36,7 +37,8 @@ import {flatten} from './utils/collection';
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
const ROUTER_DIRECTIVES = [RouterOutlet, RouterLink, RouterLinkWithHref, RouterLinkActive];
|
const ROUTER_DIRECTIVES =
|
||||||
|
[RouterOutlet, RouterLink, RouterLinkWithHref, RouterLinkActive, EmptyOutletComponent];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @description
|
* @description
|
||||||
@ -128,7 +130,11 @@ export function routerNgProbeToken() {
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@NgModule({declarations: ROUTER_DIRECTIVES, exports: ROUTER_DIRECTIVES})
|
@NgModule({
|
||||||
|
declarations: ROUTER_DIRECTIVES,
|
||||||
|
exports: ROUTER_DIRECTIVES,
|
||||||
|
entryComponents: [EmptyOutletComponent]
|
||||||
|
})
|
||||||
export class RouterModule {
|
export class RouterModule {
|
||||||
// Note: We are injecting the Router so it gets created eagerly...
|
// Note: We are injecting the Router so it gets created eagerly...
|
||||||
constructor(@Optional() @Inject(ROUTER_FORROOT_GUARD) guard: any, @Optional() router: Router) {}
|
constructor(@Optional() @Inject(ROUTER_FORROOT_GUARD) guard: any, @Optional() router: Router) {}
|
||||||
|
@ -123,21 +123,27 @@ describe('config', () => {
|
|||||||
}).toThrowError(/Invalid configuration of route '{path: "", redirectTo: "b"}'/);
|
}).toThrowError(/Invalid configuration of route '{path: "", redirectTo: "b"}'/);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when pathPatch is invalid', () => {
|
it('should throw when pathMatch is invalid', () => {
|
||||||
expect(() => { validateConfig([{path: 'a', pathMatch: 'invalid', component: ComponentB}]); })
|
expect(() => { validateConfig([{path: 'a', pathMatch: 'invalid', component: ComponentB}]); })
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
/Invalid configuration of route 'a': pathMatch can only be set to 'prefix' or 'full'/);
|
/Invalid configuration of route 'a': pathMatch can only be set to 'prefix' or 'full'/);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should throw when pathPatch is invalid', () => {
|
it('should throw when path/outlet combination is invalid', () => {
|
||||||
expect(() => { validateConfig([{path: 'a', outlet: 'aux', children: []}]); })
|
expect(() => { validateConfig([{path: 'a', outlet: 'aux'}]); })
|
||||||
.toThrowError(
|
.toThrowError(
|
||||||
/Invalid configuration of route 'a': a componentless route cannot have a named outlet set/);
|
/Invalid configuration of route 'a': a componentless route without children or loadChildren cannot have a named outlet set/);
|
||||||
|
|
||||||
expect(() => validateConfig([{path: 'a', outlet: '', children: []}])).not.toThrow();
|
expect(() => validateConfig([{path: 'a', outlet: '', children: []}])).not.toThrow();
|
||||||
expect(() => validateConfig([{path: 'a', outlet: PRIMARY_OUTLET, children: []}]))
|
expect(() => validateConfig([{path: 'a', outlet: PRIMARY_OUTLET, children: []}]))
|
||||||
.not.toThrow();
|
.not.toThrow();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should not throw when path/outlet combination is valid', () => {
|
||||||
|
expect(() => { validateConfig([{path: 'a', outlet: 'aux', children: []}]); }).not.toThrow();
|
||||||
|
expect(() => {
|
||||||
|
validateConfig([{path: 'a', outlet: 'aux', loadChildren: 'child'}]);
|
||||||
|
}).not.toThrow();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -3366,6 +3366,72 @@ describe('Integration', () => {
|
|||||||
expect(location.path()).toEqual('/lazy2/loaded');
|
expect(location.path()).toEqual('/lazy2/loaded');
|
||||||
})));
|
})));
|
||||||
|
|
||||||
|
it('should allow lazy loaded module in named outlet',
|
||||||
|
fakeAsync(inject(
|
||||||
|
[Router, NgModuleFactoryLoader], (router: Router, loader: SpyNgModuleFactoryLoader) => {
|
||||||
|
|
||||||
|
@Component({selector: 'lazy', template: 'lazy-loaded'})
|
||||||
|
class LazyComponent {
|
||||||
|
}
|
||||||
|
|
||||||
|
@NgModule({
|
||||||
|
declarations: [LazyComponent],
|
||||||
|
imports: [RouterModule.forChild([{path: '', component: LazyComponent}])]
|
||||||
|
})
|
||||||
|
class LazyLoadedModule {
|
||||||
|
}
|
||||||
|
|
||||||
|
loader.stubbedModules = {lazyModule: LazyLoadedModule};
|
||||||
|
|
||||||
|
const fixture = createRoot(router, RootCmp);
|
||||||
|
|
||||||
|
router.resetConfig([{
|
||||||
|
path: 'team/:id',
|
||||||
|
component: TeamCmp,
|
||||||
|
children: [
|
||||||
|
{path: 'user/:name', component: UserCmp},
|
||||||
|
{path: 'lazy', loadChildren: 'lazyModule', outlet: 'right'},
|
||||||
|
]
|
||||||
|
}]);
|
||||||
|
|
||||||
|
|
||||||
|
router.navigateByUrl('/team/22/user/john');
|
||||||
|
advance(fixture);
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('team 22 [ user john, right: ]');
|
||||||
|
|
||||||
|
router.navigateByUrl('/team/22/(user/john//right:lazy)');
|
||||||
|
advance(fixture);
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('team 22 [ user john, right: lazy-loaded ]');
|
||||||
|
})));
|
||||||
|
|
||||||
|
it('should allow componentless named outlet to render children',
|
||||||
|
fakeAsync(inject(
|
||||||
|
[Router, NgModuleFactoryLoader], (router: Router, loader: SpyNgModuleFactoryLoader) => {
|
||||||
|
|
||||||
|
const fixture = createRoot(router, RootCmp);
|
||||||
|
|
||||||
|
router.resetConfig([{
|
||||||
|
path: 'team/:id',
|
||||||
|
component: TeamCmp,
|
||||||
|
children: [
|
||||||
|
{path: 'user/:name', component: UserCmp},
|
||||||
|
{path: 'simple', outlet: 'right', children: [{path: '', component: SimpleCmp}]},
|
||||||
|
]
|
||||||
|
}]);
|
||||||
|
|
||||||
|
|
||||||
|
router.navigateByUrl('/team/22/user/john');
|
||||||
|
advance(fixture);
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('team 22 [ user john, right: ]');
|
||||||
|
|
||||||
|
router.navigateByUrl('/team/22/(user/john//right:simple)');
|
||||||
|
advance(fixture);
|
||||||
|
|
||||||
|
expect(fixture.nativeElement).toHaveText('team 22 [ user john, right: simple ]');
|
||||||
|
})));
|
||||||
|
|
||||||
describe('should use the injector of the lazily-loaded configuration', () => {
|
describe('should use the injector of the lazily-loaded configuration', () => {
|
||||||
class LazyLoadedServiceDefinedInModule {}
|
class LazyLoadedServiceDefinedInModule {}
|
||||||
@ -4102,6 +4168,10 @@ function createRoot(router: Router, type: any): ComponentFixture<any> {
|
|||||||
return f;
|
return f;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Component({selector: 'lazy', template: 'lazy-loaded'})
|
||||||
|
class LazyComponent {
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
imports: [RouterTestingModule, CommonModule],
|
imports: [RouterTestingModule, CommonModule],
|
||||||
|
Loading…
x
Reference in New Issue
Block a user