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