[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | import { Compiler, Injectable, Injector, NgModuleFactoryLoader, NgModuleRef } from '@angular/core';
|
---|
| 9 | import { from, of } from 'rxjs';
|
---|
| 10 | import { catchError, concatMap, filter, map, mergeAll, mergeMap } from 'rxjs/operators';
|
---|
| 11 | import { NavigationEnd, RouteConfigLoadEnd, RouteConfigLoadStart } from './events';
|
---|
| 12 | import { Router } from './router';
|
---|
| 13 | import { RouterConfigLoader } from './router_config_loader';
|
---|
| 14 | /**
|
---|
| 15 | * @description
|
---|
| 16 | *
|
---|
| 17 | * Provides a preloading strategy.
|
---|
| 18 | *
|
---|
| 19 | * @publicApi
|
---|
| 20 | */
|
---|
| 21 | export class PreloadingStrategy {
|
---|
| 22 | }
|
---|
| 23 | /**
|
---|
| 24 | * @description
|
---|
| 25 | *
|
---|
| 26 | * Provides a preloading strategy that preloads all modules as quickly as possible.
|
---|
| 27 | *
|
---|
| 28 | * ```
|
---|
| 29 | * RouterModule.forRoot(ROUTES, {preloadingStrategy: PreloadAllModules})
|
---|
| 30 | * ```
|
---|
| 31 | *
|
---|
| 32 | * @publicApi
|
---|
| 33 | */
|
---|
| 34 | export class PreloadAllModules {
|
---|
| 35 | preload(route, fn) {
|
---|
| 36 | return fn().pipe(catchError(() => of(null)));
|
---|
| 37 | }
|
---|
| 38 | }
|
---|
| 39 | /**
|
---|
| 40 | * @description
|
---|
| 41 | *
|
---|
| 42 | * Provides a preloading strategy that does not preload any modules.
|
---|
| 43 | *
|
---|
| 44 | * This strategy is enabled by default.
|
---|
| 45 | *
|
---|
| 46 | * @publicApi
|
---|
| 47 | */
|
---|
| 48 | export class NoPreloading {
|
---|
| 49 | preload(route, fn) {
|
---|
| 50 | return of(null);
|
---|
| 51 | }
|
---|
| 52 | }
|
---|
| 53 | /**
|
---|
| 54 | * The preloader optimistically loads all router configurations to
|
---|
| 55 | * make navigations into lazily-loaded sections of the application faster.
|
---|
| 56 | *
|
---|
| 57 | * The preloader runs in the background. When the router bootstraps, the preloader
|
---|
| 58 | * starts listening to all navigation events. After every such event, the preloader
|
---|
| 59 | * will check if any configurations can be loaded lazily.
|
---|
| 60 | *
|
---|
| 61 | * If a route is protected by `canLoad` guards, the preloaded will not load it.
|
---|
| 62 | *
|
---|
| 63 | * @publicApi
|
---|
| 64 | */
|
---|
| 65 | export class RouterPreloader {
|
---|
| 66 | constructor(router, moduleLoader, compiler, injector, preloadingStrategy) {
|
---|
| 67 | this.router = router;
|
---|
| 68 | this.injector = injector;
|
---|
| 69 | this.preloadingStrategy = preloadingStrategy;
|
---|
| 70 | const onStartLoad = (r) => router.triggerEvent(new RouteConfigLoadStart(r));
|
---|
| 71 | const onEndLoad = (r) => router.triggerEvent(new RouteConfigLoadEnd(r));
|
---|
| 72 | this.loader = new RouterConfigLoader(moduleLoader, compiler, onStartLoad, onEndLoad);
|
---|
| 73 | }
|
---|
| 74 | setUpPreloading() {
|
---|
| 75 | this.subscription =
|
---|
| 76 | this.router.events
|
---|
| 77 | .pipe(filter((e) => e instanceof NavigationEnd), concatMap(() => this.preload()))
|
---|
| 78 | .subscribe(() => { });
|
---|
| 79 | }
|
---|
| 80 | preload() {
|
---|
| 81 | const ngModule = this.injector.get(NgModuleRef);
|
---|
| 82 | return this.processRoutes(ngModule, this.router.config);
|
---|
| 83 | }
|
---|
| 84 | /** @nodoc */
|
---|
| 85 | ngOnDestroy() {
|
---|
| 86 | if (this.subscription) {
|
---|
| 87 | this.subscription.unsubscribe();
|
---|
| 88 | }
|
---|
| 89 | }
|
---|
| 90 | processRoutes(ngModule, routes) {
|
---|
| 91 | const res = [];
|
---|
| 92 | for (const route of routes) {
|
---|
| 93 | // we already have the config loaded, just recurse
|
---|
| 94 | if (route.loadChildren && !route.canLoad && route._loadedConfig) {
|
---|
| 95 | const childConfig = route._loadedConfig;
|
---|
| 96 | res.push(this.processRoutes(childConfig.module, childConfig.routes));
|
---|
| 97 | // no config loaded, fetch the config
|
---|
| 98 | }
|
---|
| 99 | else if (route.loadChildren && !route.canLoad) {
|
---|
| 100 | res.push(this.preloadConfig(ngModule, route));
|
---|
| 101 | // recurse into children
|
---|
| 102 | }
|
---|
| 103 | else if (route.children) {
|
---|
| 104 | res.push(this.processRoutes(ngModule, route.children));
|
---|
| 105 | }
|
---|
| 106 | }
|
---|
| 107 | return from(res).pipe(mergeAll(), map((_) => void 0));
|
---|
| 108 | }
|
---|
| 109 | preloadConfig(ngModule, route) {
|
---|
| 110 | return this.preloadingStrategy.preload(route, () => {
|
---|
| 111 | const loaded$ = route._loadedConfig ? of(route._loadedConfig) :
|
---|
| 112 | this.loader.load(ngModule.injector, route);
|
---|
| 113 | return loaded$.pipe(mergeMap((config) => {
|
---|
| 114 | route._loadedConfig = config;
|
---|
| 115 | return this.processRoutes(config.module, config.routes);
|
---|
| 116 | }));
|
---|
| 117 | });
|
---|
| 118 | }
|
---|
| 119 | }
|
---|
| 120 | RouterPreloader.decorators = [
|
---|
| 121 | { type: Injectable }
|
---|
| 122 | ];
|
---|
| 123 | RouterPreloader.ctorParameters = () => [
|
---|
| 124 | { type: Router },
|
---|
| 125 | { type: NgModuleFactoryLoader },
|
---|
| 126 | { type: Compiler },
|
---|
| 127 | { type: Injector },
|
---|
| 128 | { type: PreloadingStrategy }
|
---|
| 129 | ];
|
---|
| 130 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"router_preloader.js","sourceRoot":"","sources":["../../../../../../packages/router/src/router_preloader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {Compiler, Injectable, Injector, NgModuleFactoryLoader, NgModuleRef, OnDestroy} from '@angular/core';\nimport {from, Observable, of, Subscription} from 'rxjs';\nimport {catchError, concatMap, filter, map, mergeAll, mergeMap} from 'rxjs/operators';\n\nimport {LoadedRouterConfig, Route, Routes} from './config';\nimport {Event, NavigationEnd, RouteConfigLoadEnd, RouteConfigLoadStart} from './events';\nimport {Router} from './router';\nimport {RouterConfigLoader} from './router_config_loader';\n\n\n/**\n * @description\n *\n * Provides a preloading strategy.\n *\n * @publicApi\n */\nexport abstract class PreloadingStrategy {\n  abstract preload(route: Route, fn: () => Observable<any>): Observable<any>;\n}\n\n/**\n * @description\n *\n * Provides a preloading strategy that preloads all modules as quickly as possible.\n *\n * ```\n * RouterModule.forRoot(ROUTES, {preloadingStrategy: PreloadAllModules})\n * ```\n *\n * @publicApi\n */\nexport class PreloadAllModules implements PreloadingStrategy {\n  preload(route: Route, fn: () => Observable<any>): Observable<any> {\n    return fn().pipe(catchError(() => of(null)));\n  }\n}\n\n/**\n * @description\n *\n * Provides a preloading strategy that does not preload any modules.\n *\n * This strategy is enabled by default.\n *\n * @publicApi\n */\nexport class NoPreloading implements PreloadingStrategy {\n  preload(route: Route, fn: () => Observable<any>): Observable<any> {\n    return of(null);\n  }\n}\n\n/**\n * The preloader optimistically loads all router configurations to\n * make navigations into lazily-loaded sections of the application faster.\n *\n * The preloader runs in the background. When the router bootstraps, the preloader\n * starts listening to all navigation events. After every such event, the preloader\n * will check if any configurations can be loaded lazily.\n *\n * If a route is protected by `canLoad` guards, the preloaded will not load it.\n *\n * @publicApi\n */\n@Injectable()\nexport class RouterPreloader implements OnDestroy {\n  private loader: RouterConfigLoader;\n  private subscription?: Subscription;\n\n  constructor(\n      private router: Router, moduleLoader: NgModuleFactoryLoader, compiler: Compiler,\n      private injector: Injector, private preloadingStrategy: PreloadingStrategy) {\n    const onStartLoad = (r: Route) => router.triggerEvent(new RouteConfigLoadStart(r));\n    const onEndLoad = (r: Route) => router.triggerEvent(new RouteConfigLoadEnd(r));\n\n    this.loader = new RouterConfigLoader(moduleLoader, compiler, onStartLoad, onEndLoad);\n  }\n\n  setUpPreloading(): void {\n    this.subscription =\n        this.router.events\n            .pipe(filter((e: Event) => e instanceof NavigationEnd), concatMap(() => this.preload()))\n            .subscribe(() => {});\n  }\n\n  preload(): Observable<any> {\n    const ngModule = this.injector.get(NgModuleRef);\n    return this.processRoutes(ngModule, this.router.config);\n  }\n\n  /** @nodoc */\n  ngOnDestroy(): void {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n    }\n  }\n\n  private processRoutes(ngModule: NgModuleRef<any>, routes: Routes): Observable<void> {\n    const res: Observable<any>[] = [];\n    for (const route of routes) {\n      // we already have the config loaded, just recurse\n      if (route.loadChildren && !route.canLoad && route._loadedConfig) {\n        const childConfig = route._loadedConfig;\n        res.push(this.processRoutes(childConfig.module, childConfig.routes));\n\n        // no config loaded, fetch the config\n      } else if (route.loadChildren && !route.canLoad) {\n        res.push(this.preloadConfig(ngModule, route));\n\n        // recurse into children\n      } else if (route.children) {\n        res.push(this.processRoutes(ngModule, route.children));\n      }\n    }\n    return from(res).pipe(mergeAll(), map((_) => void 0));\n  }\n\n  private preloadConfig(ngModule: NgModuleRef<any>, route: Route): Observable<void> {\n    return this.preloadingStrategy.preload(route, () => {\n      const loaded$ = route._loadedConfig ? of(route._loadedConfig) :\n                                            this.loader.load(ngModule.injector, route);\n      return loaded$.pipe(mergeMap((config: LoadedRouterConfig) => {\n        route._loadedConfig = config;\n        return this.processRoutes(config.module, config.routes);\n      }));\n    });\n  }\n}\n"]} |
---|