UNPKG

36.5 kBJavaScriptView Raw
1import { Inject, InjectionToken, NgModule, isDevMode, } from '@angular/core';
2import { NavigationCancel, NavigationError, NavigationEnd, RoutesRecognized, NavigationStart, } from '@angular/router';
3import { isNgrxMockEnvironment, select, ACTIVE_RUNTIME_CHECKS, } from '@ngrx/store';
4import { withLatestFrom } from 'rxjs/operators';
5import { ROUTER_CANCEL, ROUTER_ERROR, ROUTER_NAVIGATED, ROUTER_NAVIGATION, ROUTER_REQUEST, } from './actions';
6import { RouterStateSerializer, } from './serializers/base';
7import { FullRouterStateSerializer, } from './serializers/full_serializer';
8import { MinimalRouterStateSerializer } from './serializers/minimal_serializer';
9import * as i0 from "@angular/core";
10import * as i1 from "@ngrx/store";
11import * as i2 from "@angular/router";
12import * as i3 from "./serializers/base";
13export var NavigationActionTiming;
14(function (NavigationActionTiming) {
15 NavigationActionTiming[NavigationActionTiming["PreActivation"] = 1] = "PreActivation";
16 NavigationActionTiming[NavigationActionTiming["PostActivation"] = 2] = "PostActivation";
17})(NavigationActionTiming || (NavigationActionTiming = {}));
18export const _ROUTER_CONFIG = new InjectionToken('@ngrx/router-store Internal Configuration');
19export const ROUTER_CONFIG = new InjectionToken('@ngrx/router-store Configuration');
20export const DEFAULT_ROUTER_FEATURENAME = 'router';
21export function _createRouterConfig(config) {
22 return {
23 stateKey: DEFAULT_ROUTER_FEATURENAME,
24 serializer: MinimalRouterStateSerializer,
25 navigationActionTiming: NavigationActionTiming.PreActivation,
26 ...config,
27 };
28}
29var 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 */
77export 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 });
249i0.ɵɵ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 */
262function isSameUrl(first, second) {
263 return stripTrailingSlash(first) === stripTrailingSlash(second);
264}
265function 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