1 | import { Inject, InjectionToken, NgModule, isDevMode, } from '@angular/core';
|
2 | import { NavigationCancel, NavigationError, NavigationEnd, RoutesRecognized, NavigationStart, } from '@angular/router';
|
3 | import { isNgrxMockEnvironment, select, ACTIVE_RUNTIME_CHECKS, } from '@ngrx/store';
|
4 | import { withLatestFrom } from 'rxjs/operators';
|
5 | import { ROUTER_CANCEL, ROUTER_ERROR, ROUTER_NAVIGATED, ROUTER_NAVIGATION, ROUTER_REQUEST, } from './actions';
|
6 | import { RouterStateSerializer, } from './serializers/base';
|
7 | import { FullRouterStateSerializer, } from './serializers/full_serializer';
|
8 | import { MinimalRouterStateSerializer } from './serializers/minimal_serializer';
|
9 | import * as i0 from "@angular/core";
|
10 | import * as i1 from "@ngrx/store";
|
11 | import * as i2 from "@angular/router";
|
12 | import * as i3 from "./serializers/base";
|
13 | export var NavigationActionTiming;
|
14 | (function (NavigationActionTiming) {
|
15 | NavigationActionTiming[NavigationActionTiming["PreActivation"] = 1] = "PreActivation";
|
16 | NavigationActionTiming[NavigationActionTiming["PostActivation"] = 2] = "PostActivation";
|
17 | })(NavigationActionTiming || (NavigationActionTiming = {}));
|
18 | export const _ROUTER_CONFIG = new InjectionToken('@ngrx/router-store Internal Configuration');
|
19 | export const ROUTER_CONFIG = new InjectionToken('@ngrx/router-store Configuration');
|
20 | export const DEFAULT_ROUTER_FEATURENAME = 'router';
|
21 | export function _createRouterConfig(config) {
|
22 | return {
|
23 | stateKey: DEFAULT_ROUTER_FEATURENAME,
|
24 | serializer: MinimalRouterStateSerializer,
|
25 | navigationActionTiming: NavigationActionTiming.PreActivation,
|
26 | ...config,
|
27 | };
|
28 | }
|
29 | var RouterTrigger;
|
30 | (function (RouterTrigger) {
|
31 | RouterTrigger[RouterTrigger["NONE"] = 1] = "NONE";
|
32 | RouterTrigger[RouterTrigger["ROUTER"] = 2] = "ROUTER";
|
33 | RouterTrigger[RouterTrigger["STORE"] = 3] = "STORE";
|
34 | })(RouterTrigger || (RouterTrigger = {}));
|
35 | /**
|
36 | * Connects RouterModule with StoreModule.
|
37 | *
|
38 | * During the navigation, before any guards or resolvers run, the router will dispatch
|
39 | * a ROUTER_NAVIGATION action, which has the following signature:
|
40 | *
|
41 | * ```
|
42 | * export type RouterNavigationPayload = {
|
43 | * routerState: SerializedRouterStateSnapshot,
|
44 | * event: RoutesRecognized
|
45 | * }
|
46 | * ```
|
47 | *
|
48 | * Either a reducer or an effect can be invoked in response to this action.
|
49 | * If the invoked reducer throws, the navigation will be canceled.
|
50 | *
|
51 | * If navigation gets canceled because of a guard, a ROUTER_CANCEL action will be
|
52 | * dispatched. If navigation results in an error, a ROUTER_ERROR action will be dispatched.
|
53 | *
|
54 | * Both ROUTER_CANCEL and ROUTER_ERROR contain the store state before the navigation
|
55 | * which can be used to restore the consistency of the store.
|
56 | *
|
57 | * Usage:
|
58 | *
|
59 | * ```typescript
|
60 | * @NgModule({
|
61 | * declarations: [AppCmp, SimpleCmp],
|
62 | * imports: [
|
63 | * BrowserModule,
|
64 | * StoreModule.forRoot(mapOfReducers),
|
65 | * RouterModule.forRoot([
|
66 | * { path: '', component: SimpleCmp },
|
67 | * { path: 'next', component: SimpleCmp }
|
68 | * ]),
|
69 | * StoreRouterConnectingModule.forRoot()
|
70 | * ],
|
71 | * bootstrap: [AppCmp]
|
72 | * })
|
73 | * export class AppModule {
|
74 | * }
|
75 | * ```
|
76 | */
|
77 | export class StoreRouterConnectingModule {
|
78 | constructor(store, router, serializer, errorHandler, config, activeRuntimeChecks) {
|
79 | this.store = store;
|
80 | this.router = router;
|
81 | this.serializer = serializer;
|
82 | this.errorHandler = errorHandler;
|
83 | this.config = config;
|
84 | this.activeRuntimeChecks = activeRuntimeChecks;
|
85 | this.lastEvent = null;
|
86 | this.routerState = null;
|
87 | this.trigger = RouterTrigger.NONE;
|
88 | this.stateKey = this.config.stateKey;
|
89 | if (!isNgrxMockEnvironment() &&
|
90 | isDevMode() &&
|
91 | (activeRuntimeChecks?.strictActionSerializability ||
|
92 | activeRuntimeChecks?.strictStateSerializability) &&
|
93 | this.serializer instanceof FullRouterStateSerializer) {
|
94 | console.warn('@ngrx/router-store: The serializability runtime checks cannot be enabled ' +
|
95 | 'with the FullRouterStateSerializer. The FullRouterStateSerializer ' +
|
96 | 'has an unserializable router state and actions that are not serializable. ' +
|
97 | 'To use the serializability runtime checks either use ' +
|
98 | 'the MinimalRouterStateSerializer or implement a custom router state serializer.');
|
99 | }
|
100 | this.setUpStoreStateListener();
|
101 | this.setUpRouterEventsListener();
|
102 | }
|
103 | static forRoot(config = {}) {
|
104 | return {
|
105 | ngModule: StoreRouterConnectingModule,
|
106 | providers: [
|
107 | { provide: _ROUTER_CONFIG, useValue: config },
|
108 | {
|
109 | provide: ROUTER_CONFIG,
|
110 | useFactory: _createRouterConfig,
|
111 | deps: [_ROUTER_CONFIG],
|
112 | },
|
113 | {
|
114 | provide: RouterStateSerializer,
|
115 | useClass: config.serializer
|
116 | ? config.serializer
|
117 | : config.routerState === 0 /* Full */
|
118 | ? FullRouterStateSerializer
|
119 | : MinimalRouterStateSerializer,
|
120 | },
|
121 | ],
|
122 | };
|
123 | }
|
124 | setUpStoreStateListener() {
|
125 | this.store
|
126 | .pipe(select(this.stateKey), withLatestFrom(this.store))
|
127 | .subscribe(([routerStoreState, storeState]) => {
|
128 | this.navigateIfNeeded(routerStoreState, storeState);
|
129 | });
|
130 | }
|
131 | navigateIfNeeded(routerStoreState, storeState) {
|
132 | if (!routerStoreState || !routerStoreState.state) {
|
133 | return;
|
134 | }
|
135 | if (this.trigger === RouterTrigger.ROUTER) {
|
136 | return;
|
137 | }
|
138 | if (this.lastEvent instanceof NavigationStart) {
|
139 | return;
|
140 | }
|
141 | const url = routerStoreState.state.url;
|
142 | if (!isSameUrl(this.router.url, url)) {
|
143 | this.storeState = storeState;
|
144 | this.trigger = RouterTrigger.STORE;
|
145 | this.router.navigateByUrl(url).catch((error) => {
|
146 | this.errorHandler.handleError(error);
|
147 | });
|
148 | }
|
149 | }
|
150 | setUpRouterEventsListener() {
|
151 | const dispatchNavLate = this.config.navigationActionTiming ===
|
152 | NavigationActionTiming.PostActivation;
|
153 | let routesRecognized;
|
154 | this.router.events
|
155 | .pipe(withLatestFrom(this.store))
|
156 | .subscribe(([event, storeState]) => {
|
157 | this.lastEvent = event;
|
158 | if (event instanceof NavigationStart) {
|
159 | this.routerState = this.serializer.serialize(this.router.routerState.snapshot);
|
160 | if (this.trigger !== RouterTrigger.STORE) {
|
161 | this.storeState = storeState;
|
162 | this.dispatchRouterRequest(event);
|
163 | }
|
164 | }
|
165 | else if (event instanceof RoutesRecognized) {
|
166 | routesRecognized = event;
|
167 | if (!dispatchNavLate && this.trigger !== RouterTrigger.STORE) {
|
168 | this.dispatchRouterNavigation(event);
|
169 | }
|
170 | }
|
171 | else if (event instanceof NavigationCancel) {
|
172 | this.dispatchRouterCancel(event);
|
173 | this.reset();
|
174 | }
|
175 | else if (event instanceof NavigationError) {
|
176 | this.dispatchRouterError(event);
|
177 | this.reset();
|
178 | }
|
179 | else if (event instanceof NavigationEnd) {
|
180 | if (this.trigger !== RouterTrigger.STORE) {
|
181 | if (dispatchNavLate) {
|
182 | this.dispatchRouterNavigation(routesRecognized);
|
183 | }
|
184 | this.dispatchRouterNavigated(event);
|
185 | }
|
186 | this.reset();
|
187 | }
|
188 | });
|
189 | }
|
190 | dispatchRouterRequest(event) {
|
191 | this.dispatchRouterAction(ROUTER_REQUEST, { event });
|
192 | }
|
193 | dispatchRouterNavigation(lastRoutesRecognized) {
|
194 | const nextRouterState = this.serializer.serialize(lastRoutesRecognized.state);
|
195 | this.dispatchRouterAction(ROUTER_NAVIGATION, {
|
196 | routerState: nextRouterState,
|
197 | event: new RoutesRecognized(lastRoutesRecognized.id, lastRoutesRecognized.url, lastRoutesRecognized.urlAfterRedirects, nextRouterState),
|
198 | });
|
199 | }
|
200 | dispatchRouterCancel(event) {
|
201 | this.dispatchRouterAction(ROUTER_CANCEL, {
|
202 | storeState: this.storeState,
|
203 | event,
|
204 | });
|
205 | }
|
206 | dispatchRouterError(event) {
|
207 | this.dispatchRouterAction(ROUTER_ERROR, {
|
208 | storeState: this.storeState,
|
209 | event: new NavigationError(event.id, event.url, `${event}`),
|
210 | });
|
211 | }
|
212 | dispatchRouterNavigated(event) {
|
213 | const routerState = this.serializer.serialize(this.router.routerState.snapshot);
|
214 | this.dispatchRouterAction(ROUTER_NAVIGATED, { event, routerState });
|
215 | }
|
216 | dispatchRouterAction(type, payload) {
|
217 | this.trigger = RouterTrigger.ROUTER;
|
218 | try {
|
219 | this.store.dispatch({
|
220 | type,
|
221 | payload: {
|
222 | routerState: this.routerState,
|
223 | ...payload,
|
224 | event: this.config.routerState === 0 /* Full */
|
225 | ? payload.event
|
226 | : {
|
227 | id: payload.event.id,
|
228 | url: payload.event.url,
|
229 | // safe, as it will just be `undefined` for non-NavigationEnd router events
|
230 | urlAfterRedirects: payload.event
|
231 | .urlAfterRedirects,
|
232 | },
|
233 | },
|
234 | });
|
235 | }
|
236 | finally {
|
237 | this.trigger = RouterTrigger.NONE;
|
238 | }
|
239 | }
|
240 | reset() {
|
241 | this.trigger = RouterTrigger.NONE;
|
242 | this.storeState = null;
|
243 | this.routerState = null;
|
244 | }
|
245 | }
|
246 | /** @nocollapse */ StoreRouterConnectingModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.0-rc.1", ngImport: i0, type: StoreRouterConnectingModule, deps: [{ token: i1.Store }, { token: i2.Router }, { token: i3.RouterStateSerializer }, { token: i0.ErrorHandler }, { token: ROUTER_CONFIG }, { token: ACTIVE_RUNTIME_CHECKS }], target: i0.ɵɵFactoryTarget.NgModule });
|
247 | /** @nocollapse */ StoreRouterConnectingModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.0.0-rc.1", ngImport: i0, type: StoreRouterConnectingModule });
|
248 | /** @nocollapse */ StoreRouterConnectingModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.0.0-rc.1", ngImport: i0, type: StoreRouterConnectingModule });
|
249 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.0-rc.1", ngImport: i0, type: StoreRouterConnectingModule, decorators: [{
|
250 | type: NgModule,
|
251 | args: [{}]
|
252 | }], ctorParameters: function () { return [{ type: i1.Store }, { type: i2.Router }, { type: i3.RouterStateSerializer }, { type: i0.ErrorHandler }, { type: undefined, decorators: [{
|
253 | type: Inject,
|
254 | args: [ROUTER_CONFIG]
|
255 | }] }, { type: undefined, decorators: [{
|
256 | type: Inject,
|
257 | args: [ACTIVE_RUNTIME_CHECKS]
|
258 | }] }]; } });
|
259 | /**
|
260 | * Check if the URLs are matching. Accounts for the possibility of trailing "/" in url.
|
261 | */
|
262 | function isSameUrl(first, second) {
|
263 | return stripTrailingSlash(first) === stripTrailingSlash(second);
|
264 | }
|
265 | function stripTrailingSlash(text) {
|
266 | if (text?.length > 0 && text[text.length - 1] === '/') {
|
267 | return text.substring(0, text.length - 1);
|
268 | }
|
269 | return text;
|
270 | }
|
271 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"router_store_module.js","sourceRoot":"","sources":["../../../../../modules/router-store/src/router_store_module.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Inject,\n  InjectionToken,\n  ModuleWithProviders,\n  NgModule,\n  ErrorHandler,\n  isDevMode,\n} from '@angular/core';\nimport {\n  NavigationCancel,\n  NavigationError,\n  NavigationEnd,\n  Router,\n  RoutesRecognized,\n  NavigationStart,\n  Event,\n  RouterEvent,\n} from '@angular/router';\nimport {\n  isNgrxMockEnvironment,\n  RuntimeChecks,\n  select,\n  Selector,\n  Store,\n  ACTIVE_RUNTIME_CHECKS,\n} from '@ngrx/store';\nimport { withLatestFrom } from 'rxjs/operators';\n\nimport {\n  ROUTER_CANCEL,\n  ROUTER_ERROR,\n  ROUTER_NAVIGATED,\n  ROUTER_NAVIGATION,\n  ROUTER_REQUEST,\n} from './actions';\nimport { RouterReducerState } from './reducer';\nimport {\n  RouterStateSerializer,\n  BaseRouterStoreState,\n} from './serializers/base';\nimport {\n  FullRouterStateSerializer,\n  SerializedRouterStateSnapshot,\n} from './serializers/full_serializer';\nimport { MinimalRouterStateSerializer } from './serializers/minimal_serializer';\n\nexport type StateKeyOrSelector<\n  T extends BaseRouterStoreState = SerializedRouterStateSnapshot\n> = string | Selector<any, RouterReducerState<T>>;\n\n/**\n * Minimal = Serializes the router event with MinimalRouterStateSerializer\n * Full = Serializes the router event with FullRouterStateSerializer\n */\nexport const enum RouterState {\n  Full,\n  Minimal,\n}\n\nexport interface StoreRouterConfig<\n  T extends BaseRouterStoreState = SerializedRouterStateSnapshot\n> {\n  stateKey?: StateKeyOrSelector<T>;\n  serializer?: new (...args: any[]) => RouterStateSerializer;\n  /**\n   * By default, ROUTER_NAVIGATION is dispatched before guards and resolvers run.\n   * Therefore, the action could run too soon, for example\n   * there may be a navigation cancel due to a guard saying the navigation is not allowed.\n   * To run ROUTER_NAVIGATION after guards and resolvers,\n   * set this property to NavigationActionTiming.PostActivation.\n   */\n  navigationActionTiming?: NavigationActionTiming;\n  /**\n   * Decides which router serializer should be used, if there is none provided, and the metadata on the dispatched @ngrx/router-store action payload.\n   * Set to `Minimal` to use the `MinimalRouterStateSerializer` and to set a minimal router event with the navigation id and url as payload.\n   * Set to `Full` to use the `FullRouterStateSerializer` and to set the angular router events as payload.\n   */\n  routerState?: RouterState;\n}\n\ninterface StoreRouterActionPayload {\n  event: RouterEvent;\n  routerState?: SerializedRouterStateSnapshot;\n  storeState?: any;\n}\n\nexport enum NavigationActionTiming {\n  PreActivation = 1,\n  PostActivation = 2,\n}\n\nexport const _ROUTER_CONFIG = new InjectionToken(\n  '@ngrx/router-store Internal Configuration'\n);\nexport const ROUTER_CONFIG = new InjectionToken(\n  '@ngrx/router-store Configuration'\n);\nexport const DEFAULT_ROUTER_FEATURENAME = 'router';\n\nexport function _createRouterConfig(\n  config: StoreRouterConfig\n): StoreRouterConfig {\n  return {\n    stateKey: DEFAULT_ROUTER_FEATURENAME,\n    serializer: MinimalRouterStateSerializer,\n    navigationActionTiming: NavigationActionTiming.PreActivation,\n    ...config,\n  };\n}\n\nenum RouterTrigger {\n  NONE = 1,\n  ROUTER = 2,\n  STORE = 3,\n}\n\n/**\n * Connects RouterModule with StoreModule.\n *\n * During the navigation, before any guards or resolvers run, the router will dispatch\n * a ROUTER_NAVIGATION action, which has the following signature:\n *\n * ```\n * export type RouterNavigationPayload = {\n *   routerState: SerializedRouterStateSnapshot,\n *   event: RoutesRecognized\n * }\n * ```\n *\n * Either a reducer or an effect can be invoked in response to this action.\n * If the invoked reducer throws, the navigation will be canceled.\n *\n * If navigation gets canceled because of a guard, a ROUTER_CANCEL action will be\n * dispatched. If navigation results in an error, a ROUTER_ERROR action will be dispatched.\n *\n * Both ROUTER_CANCEL and ROUTER_ERROR contain the store state before the navigation\n * which can be used to restore the consistency of the store.\n *\n * Usage:\n *\n * ```typescript\n * @NgModule({\n *   declarations: [AppCmp, SimpleCmp],\n *   imports: [\n *     BrowserModule,\n *     StoreModule.forRoot(mapOfReducers),\n *     RouterModule.forRoot([\n *       { path: '', component: SimpleCmp },\n *       { path: 'next', component: SimpleCmp }\n *     ]),\n *     StoreRouterConnectingModule.forRoot()\n *   ],\n *   bootstrap: [AppCmp]\n * })\n * export class AppModule {\n * }\n * ```\n */\n@NgModule({})\nexport class StoreRouterConnectingModule {\n  private lastEvent: Event | null = null;\n  private routerState: SerializedRouterStateSnapshot | null = null;\n  private storeState: any;\n  private trigger = RouterTrigger.NONE;\n  private readonly stateKey: StateKeyOrSelector;\n\n  static forRoot<\n    T extends BaseRouterStoreState = SerializedRouterStateSnapshot\n  >(\n    config: StoreRouterConfig<T> = {}\n  ): ModuleWithProviders<StoreRouterConnectingModule> {\n    return {\n      ngModule: StoreRouterConnectingModule,\n      providers: [\n        { provide: _ROUTER_CONFIG, useValue: config },\n        {\n          provide: ROUTER_CONFIG,\n          useFactory: _createRouterConfig,\n          deps: [_ROUTER_CONFIG],\n        },\n        {\n          provide: RouterStateSerializer,\n          useClass: config.serializer\n            ? config.serializer\n            : config.routerState === RouterState.Full\n            ? FullRouterStateSerializer\n            : MinimalRouterStateSerializer,\n        },\n      ],\n    };\n  }\n\n  constructor(\n    private store: Store<any>,\n    private router: Router,\n    private serializer: RouterStateSerializer<SerializedRouterStateSnapshot>,\n    private errorHandler: ErrorHandler,\n    @Inject(ROUTER_CONFIG) private readonly config: StoreRouterConfig,\n    @Inject(ACTIVE_RUNTIME_CHECKS)\n    private readonly activeRuntimeChecks: RuntimeChecks\n  ) {\n    this.stateKey = this.config.stateKey as StateKeyOrSelector;\n\n    if (\n      !isNgrxMockEnvironment() &&\n      isDevMode() &&\n      (activeRuntimeChecks?.strictActionSerializability ||\n        activeRuntimeChecks?.strictStateSerializability) &&\n      this.serializer instanceof FullRouterStateSerializer\n    ) {\n      console.warn(\n        '@ngrx/router-store: The serializability runtime checks cannot be enabled ' +\n          'with the FullRouterStateSerializer. The FullRouterStateSerializer ' +\n          'has an unserializable router state and actions that are not serializable. ' +\n          'To use the serializability runtime checks either use ' +\n          'the MinimalRouterStateSerializer or implement a custom router state serializer.'\n      );\n    }\n\n    this.setUpStoreStateListener();\n    this.setUpRouterEventsListener();\n  }\n\n  private setUpStoreStateListener(): void {\n    this.store\n      .pipe(select(this.stateKey as any), withLatestFrom(this.store))\n      .subscribe(([routerStoreState, storeState]) => {\n        this.navigateIfNeeded(routerStoreState, storeState);\n      });\n  }\n\n  private navigateIfNeeded(\n    routerStoreState: RouterReducerState,\n    storeState: any\n  ): void {\n    if (!routerStoreState || !routerStoreState.state) {\n      return;\n    }\n    if (this.trigger === RouterTrigger.ROUTER) {\n      return;\n    }\n    if (this.lastEvent instanceof NavigationStart) {\n      return;\n    }\n\n    const url = routerStoreState.state.url;\n    if (!isSameUrl(this.router.url, url)) {\n      this.storeState = storeState;\n      this.trigger = RouterTrigger.STORE;\n      this.router.navigateByUrl(url).catch((error) => {\n        this.errorHandler.handleError(error);\n      });\n    }\n  }\n\n  private setUpRouterEventsListener(): void {\n    const dispatchNavLate =\n      this.config.navigationActionTiming ===\n      NavigationActionTiming.PostActivation;\n    let routesRecognized: RoutesRecognized;\n\n    this.router.events\n      .pipe(withLatestFrom(this.store))\n      .subscribe(([event, storeState]) => {\n        this.lastEvent = event;\n\n        if (event instanceof NavigationStart) {\n          this.routerState = this.serializer.serialize(\n            this.router.routerState.snapshot\n          );\n          if (this.trigger !== RouterTrigger.STORE) {\n            this.storeState = storeState;\n            this.dispatchRouterRequest(event);\n          }\n        } else if (event instanceof RoutesRecognized) {\n          routesRecognized = event;\n\n          if (!dispatchNavLate && this.trigger !== RouterTrigger.STORE) {\n            this.dispatchRouterNavigation(event);\n          }\n        } else if (event instanceof NavigationCancel) {\n          this.dispatchRouterCancel(event);\n          this.reset();\n        } else if (event instanceof NavigationError) {\n          this.dispatchRouterError(event);\n          this.reset();\n        } else if (event instanceof NavigationEnd) {\n          if (this.trigger !== RouterTrigger.STORE) {\n            if (dispatchNavLate) {\n              this.dispatchRouterNavigation(routesRecognized);\n            }\n            this.dispatchRouterNavigated(event);\n          }\n          this.reset();\n        }\n      });\n  }\n\n  private dispatchRouterRequest(event: NavigationStart): void {\n    this.dispatchRouterAction(ROUTER_REQUEST, { event });\n  }\n\n  private dispatchRouterNavigation(\n    lastRoutesRecognized: RoutesRecognized\n  ): void {\n    const nextRouterState = this.serializer.serialize(\n      lastRoutesRecognized.state\n    );\n    this.dispatchRouterAction(ROUTER_NAVIGATION, {\n      routerState: nextRouterState,\n      event: new RoutesRecognized(\n        lastRoutesRecognized.id,\n        lastRoutesRecognized.url,\n        lastRoutesRecognized.urlAfterRedirects,\n        nextRouterState\n      ),\n    });\n  }\n\n  private dispatchRouterCancel(event: NavigationCancel): void {\n    this.dispatchRouterAction(ROUTER_CANCEL, {\n      storeState: this.storeState,\n      event,\n    });\n  }\n\n  private dispatchRouterError(event: NavigationError): void {\n    this.dispatchRouterAction(ROUTER_ERROR, {\n      storeState: this.storeState,\n      event: new NavigationError(event.id, event.url, `${event}`),\n    });\n  }\n\n  private dispatchRouterNavigated(event: NavigationEnd): void {\n    const routerState = this.serializer.serialize(\n      this.router.routerState.snapshot\n    );\n    this.dispatchRouterAction(ROUTER_NAVIGATED, { event, routerState });\n  }\n\n  private dispatchRouterAction(\n    type: string,\n    payload: StoreRouterActionPayload\n  ): void {\n    this.trigger = RouterTrigger.ROUTER;\n    try {\n      this.store.dispatch({\n        type,\n        payload: {\n          routerState: this.routerState,\n          ...payload,\n          event:\n            this.config.routerState === RouterState.Full\n              ? payload.event\n              : {\n                  id: payload.event.id,\n                  url: payload.event.url,\n                  // safe, as it will just be `undefined` for non-NavigationEnd router events\n                  urlAfterRedirects: (payload.event as NavigationEnd)\n                    .urlAfterRedirects,\n                },\n        },\n      });\n    } finally {\n      this.trigger = RouterTrigger.NONE;\n    }\n  }\n\n  private reset() {\n    this.trigger = RouterTrigger.NONE;\n    this.storeState = null;\n    this.routerState = null;\n  }\n}\n\n/**\n * Check if the URLs are matching. Accounts for the possibility of trailing \"/\" in url.\n */\nfunction isSameUrl(first: string, second: string): boolean {\n  return stripTrailingSlash(first) === stripTrailingSlash(second);\n}\n\nfunction stripTrailingSlash(text: string): string {\n  if (text?.length > 0 && text[text.length - 1] === '/') {\n    return text.substring(0, text.length - 1);\n  }\n  return text;\n}\n"]} |
\ | No newline at end of file |