navigation-stack 0.3.1 → 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (253) hide show
  1. package/README.md +603 -163
  2. package/data-storage/package.json +6 -0
  3. package/karma.conf.cjs +21 -4
  4. package/lib/cjs/NavigationStack.js +73 -0
  5. package/lib/cjs/data-storage/DataStorage.js +71 -0
  6. package/lib/cjs/data-storage/LocationDataStorage.js +29 -0
  7. package/lib/cjs/data-storage/index.js +9 -0
  8. package/lib/cjs/environment/InMemoryEnvironment.js +15 -0
  9. package/lib/cjs/environment/WebBrowserEnvironment.js +15 -0
  10. package/lib/cjs/environment/data-storage/InMemoryDataStorage.js +27 -0
  11. package/lib/cjs/environment/data-storage/WebBrowserDataStorage.js +21 -0
  12. package/lib/cjs/environment/scroll-position/InMemoryScrollPosition.js +44 -0
  13. package/lib/cjs/environment/scroll-position/WebBrowserScrollPosition.js +60 -0
  14. package/lib/cjs/getLocationFromInternalLocation.js +14 -0
  15. package/lib/cjs/index.js +20 -16
  16. package/lib/cjs/navigationBlockers.js +25 -23
  17. package/lib/cjs/{normalizeInputLocation.js → parseInputLocation.js} +25 -9
  18. package/lib/cjs/{ActionTypes.js → redux/ActionTypes.js} +1 -1
  19. package/lib/cjs/redux/ActionTypesInternal.js +8 -0
  20. package/lib/cjs/{Actions.js → redux/Actions.js} +5 -4
  21. package/lib/cjs/redux/createMiddlewares.js +60 -0
  22. package/lib/cjs/redux/index.js +13 -0
  23. package/lib/cjs/redux/internalLocationReducer.js +14 -0
  24. package/lib/cjs/redux/middleware/createAddInputLocationBasePathMiddleware.js +32 -0
  25. package/lib/cjs/redux/middleware/createNonProgrammaticNavigationBlockerMiddleware.js +113 -0
  26. package/lib/cjs/redux/middleware/createProgrammaticNavigationBlockerMiddleware.js +94 -0
  27. package/lib/cjs/redux/middleware/createRemoveOutputLocationBasePathMiddleware.js +30 -0
  28. package/lib/cjs/redux/middleware/createUpdateInternalLocationMiddleware.js +73 -0
  29. package/lib/cjs/{middleware/navigationActionMiddleware.js → redux/middleware/navigationOperationMiddleware.js} +11 -8
  30. package/lib/cjs/{middleware/normalizeInputLocationMiddleware.js → redux/middleware/parseInputLocationMiddleware.js} +6 -4
  31. package/lib/cjs/redux/middleware/updateLocationMiddleware.js +34 -0
  32. package/lib/cjs/scroll-position/PageScrollPositionSetter.js +97 -0
  33. package/lib/cjs/scroll-position/ScrollPositionAutoSaver.js +130 -0
  34. package/lib/cjs/scroll-position/ScrollPositionRestoration.js +383 -0
  35. package/lib/cjs/scroll-position/ScrollPositionSaver.js +81 -0
  36. package/lib/cjs/scroll-position/ScrollPositionSetter.js +16 -0
  37. package/lib/cjs/scroll-position/constants.js +5 -0
  38. package/lib/cjs/scroll-position/index.js +7 -0
  39. package/lib/cjs/scroll-position/scheduleNextTick.js +11 -0
  40. package/lib/cjs/session/InMemorySession.js +22 -0
  41. package/lib/cjs/session/ServerSideRenderSession.js +17 -0
  42. package/lib/cjs/session/Session.js +196 -0
  43. package/lib/cjs/session/WebBrowserSession.js +20 -0
  44. package/lib/cjs/session/key/createSessionKey.js +23 -0
  45. package/lib/cjs/session/lifecycle/InMemorySessionLifecycle.js +19 -0
  46. package/lib/cjs/session/lifecycle/WebBrowserSessionLifecycle.js +128 -0
  47. package/lib/cjs/session/lifecycle/page-lifecycle/PageLifecycle.js +269 -0
  48. package/lib/cjs/session/lifecycle/page-lifecycle/PageLifecycleInstance.js +8 -0
  49. package/lib/cjs/session/lifecycle/page-lifecycle/supportsConstructableEventTarget.js +33 -0
  50. package/lib/cjs/session/navigation/InMemoryNavigation.js +104 -0
  51. package/lib/cjs/session/navigation/ServerSideNavigation.js +61 -0
  52. package/lib/cjs/session/navigation/WebBrowserNavigation.js +221 -0
  53. package/lib/cjs/session/navigation/error/NavigationOutOfBoundsError.js +12 -0
  54. package/lib/cjs/session/navigation/error/ServerSideNavigationError.js +21 -0
  55. package/lib/cjs/session/navigation/operation/operations.js +11 -0
  56. package/lib/cjs/session/subscription/Subscription.js +81 -0
  57. package/lib/data-storage/index.d.ts +35 -0
  58. package/lib/esm/NavigationStack.js +66 -0
  59. package/lib/esm/data-storage/DataStorage.js +65 -0
  60. package/lib/esm/data-storage/LocationDataStorage.js +22 -0
  61. package/lib/esm/data-storage/index.js +2 -0
  62. package/lib/esm/environment/InMemoryEnvironment.js +8 -0
  63. package/lib/esm/environment/WebBrowserEnvironment.js +8 -0
  64. package/lib/esm/environment/data-storage/InMemoryDataStorage.js +21 -0
  65. package/lib/esm/environment/data-storage/WebBrowserDataStorage.js +15 -0
  66. package/lib/esm/environment/scroll-position/InMemoryScrollPosition.js +38 -0
  67. package/lib/esm/environment/scroll-position/WebBrowserScrollPosition.js +54 -0
  68. package/lib/esm/getLocationFromInternalLocation.js +9 -0
  69. package/lib/esm/index.js +10 -8
  70. package/lib/esm/navigationBlockers.js +25 -23
  71. package/lib/esm/{normalizeInputLocation.js → parseInputLocation.js} +24 -8
  72. package/lib/esm/{ActionTypes.js → redux/ActionTypes.js} +1 -1
  73. package/lib/esm/redux/ActionTypesInternal.js +3 -0
  74. package/lib/esm/{Actions.js → redux/Actions.js} +5 -4
  75. package/lib/esm/redux/createMiddlewares.js +54 -0
  76. package/lib/esm/redux/index.js +4 -0
  77. package/lib/esm/redux/internalLocationReducer.js +8 -0
  78. package/lib/esm/redux/middleware/createAddInputLocationBasePathMiddleware.js +27 -0
  79. package/lib/esm/redux/middleware/createNonProgrammaticNavigationBlockerMiddleware.js +108 -0
  80. package/lib/esm/redux/middleware/createProgrammaticNavigationBlockerMiddleware.js +88 -0
  81. package/lib/esm/redux/middleware/createRemoveOutputLocationBasePathMiddleware.js +25 -0
  82. package/lib/esm/redux/middleware/createUpdateInternalLocationMiddleware.js +68 -0
  83. package/lib/esm/{middleware/navigationActionMiddleware.js → redux/middleware/navigationOperationMiddleware.js} +10 -7
  84. package/lib/esm/{middleware/normalizeInputLocationMiddleware.js → redux/middleware/parseInputLocationMiddleware.js} +5 -3
  85. package/lib/esm/redux/middleware/updateLocationMiddleware.js +28 -0
  86. package/lib/esm/scroll-position/PageScrollPositionSetter.js +91 -0
  87. package/lib/esm/scroll-position/ScrollPositionAutoSaver.js +123 -0
  88. package/lib/esm/scroll-position/ScrollPositionRestoration.js +376 -0
  89. package/lib/esm/scroll-position/ScrollPositionSaver.js +74 -0
  90. package/lib/esm/scroll-position/ScrollPositionSetter.js +10 -0
  91. package/lib/esm/scroll-position/constants.js +1 -0
  92. package/lib/esm/scroll-position/index.js +1 -0
  93. package/lib/esm/scroll-position/scheduleNextTick.js +6 -0
  94. package/lib/esm/session/InMemorySession.js +15 -0
  95. package/lib/esm/session/ServerSideRenderSession.js +11 -0
  96. package/lib/esm/session/Session.js +189 -0
  97. package/lib/esm/session/WebBrowserSession.js +13 -0
  98. package/lib/esm/session/key/createSessionKey.js +18 -0
  99. package/lib/esm/session/lifecycle/InMemorySessionLifecycle.js +13 -0
  100. package/lib/esm/session/lifecycle/WebBrowserSessionLifecycle.js +120 -0
  101. package/lib/esm/session/lifecycle/page-lifecycle/PageLifecycle.js +263 -0
  102. package/lib/esm/session/lifecycle/page-lifecycle/PageLifecycleInstance.js +2 -0
  103. package/lib/esm/session/lifecycle/page-lifecycle/supportsConstructableEventTarget.js +30 -0
  104. package/lib/esm/session/navigation/InMemoryNavigation.js +97 -0
  105. package/lib/esm/session/navigation/ServerSideNavigation.js +54 -0
  106. package/lib/esm/session/navigation/WebBrowserNavigation.js +213 -0
  107. package/lib/esm/session/navigation/error/NavigationOutOfBoundsError.js +6 -0
  108. package/lib/esm/session/navigation/error/ServerSideNavigationError.js +14 -0
  109. package/lib/esm/session/navigation/operation/operations.js +6 -0
  110. package/lib/esm/session/subscription/Subscription.js +75 -0
  111. package/lib/index.d.ts +178 -157
  112. package/lib/redux/index.d.ts +90 -0
  113. package/lib/scroll-position/index.d.ts +107 -0
  114. package/package.json +9 -5
  115. package/redux/package.json +6 -0
  116. package/scroll-position/package.json +6 -0
  117. package/src/NavigationStack.js +84 -0
  118. package/src/data-storage/DataStorage.js +69 -0
  119. package/src/data-storage/LocationDataStorage.js +23 -0
  120. package/src/data-storage/index.js +2 -0
  121. package/src/environment/InMemoryEnvironment.js +9 -0
  122. package/src/environment/WebBrowserEnvironment.js +9 -0
  123. package/src/environment/data-storage/InMemoryDataStorage.js +23 -0
  124. package/src/environment/data-storage/WebBrowserDataStorage.js +17 -0
  125. package/src/environment/scroll-position/InMemoryScrollPosition.js +45 -0
  126. package/src/environment/scroll-position/WebBrowserScrollPosition.js +72 -0
  127. package/src/getLocationFromInternalLocation.js +7 -0
  128. package/src/index.js +10 -8
  129. package/src/navigationBlockers.js +28 -27
  130. package/src/{normalizeInputLocation.js → parseInputLocation.js} +23 -8
  131. package/src/{ActionTypes.js → redux/ActionTypes.js} +1 -1
  132. package/src/redux/ActionTypesInternal.js +3 -0
  133. package/src/{Actions.js → redux/Actions.js} +4 -3
  134. package/src/redux/createMiddlewares.js +65 -0
  135. package/src/redux/index.js +4 -0
  136. package/src/redux/internalLocationReducer.js +9 -0
  137. package/src/redux/middleware/createAddInputLocationBasePathMiddleware.js +27 -0
  138. package/src/redux/middleware/createNonProgrammaticNavigationBlockerMiddleware.js +119 -0
  139. package/src/redux/middleware/createProgrammaticNavigationBlockerMiddleware.js +94 -0
  140. package/src/redux/middleware/createRemoveOutputLocationBasePathMiddleware.js +26 -0
  141. package/src/redux/middleware/createUpdateInternalLocationMiddleware.js +72 -0
  142. package/src/{middleware/navigationActionMiddleware.js → redux/middleware/navigationOperationMiddleware.js} +10 -3
  143. package/src/{middleware/normalizeInputLocationMiddleware.js → redux/middleware/parseInputLocationMiddleware.js} +5 -3
  144. package/src/redux/middleware/updateLocationMiddleware.js +28 -0
  145. package/src/scroll-position/PageScrollPositionSetter.js +110 -0
  146. package/src/scroll-position/ScrollPositionAutoSaver.js +151 -0
  147. package/src/scroll-position/ScrollPositionRestoration.js +506 -0
  148. package/src/scroll-position/ScrollPositionSaver.js +100 -0
  149. package/src/scroll-position/ScrollPositionSetter.js +16 -0
  150. package/src/scroll-position/constants.js +1 -0
  151. package/src/scroll-position/index.js +1 -0
  152. package/src/scroll-position/scheduleNextTick.js +6 -0
  153. package/src/session/InMemorySession.js +13 -0
  154. package/src/session/ServerSideRenderSession.js +9 -0
  155. package/src/session/Session.js +216 -0
  156. package/src/session/WebBrowserSession.js +13 -0
  157. package/src/session/key/createSessionKey.js +18 -0
  158. package/src/session/lifecycle/InMemorySessionLifecycle.js +13 -0
  159. package/src/session/lifecycle/WebBrowserSessionLifecycle.js +126 -0
  160. package/src/session/lifecycle/page-lifecycle/PageLifecycle.js +291 -0
  161. package/src/session/lifecycle/page-lifecycle/PageLifecycleInstance.js +3 -0
  162. package/src/session/lifecycle/page-lifecycle/supportsConstructableEventTarget.js +32 -0
  163. package/src/session/navigation/InMemoryNavigation.js +78 -0
  164. package/src/session/navigation/ServerSideNavigation.js +43 -0
  165. package/src/session/navigation/WebBrowserNavigation.js +224 -0
  166. package/src/session/navigation/error/NavigationOutOfBoundsError.js +7 -0
  167. package/src/session/navigation/error/ServerSideNavigationError.js +18 -0
  168. package/src/session/navigation/operation/operations.js +6 -0
  169. package/src/session/subscription/Subscription.js +76 -0
  170. package/test/NavigationStack.test.js +296 -0
  171. package/test/{LocationDataStorage.test.js → data-storage/LocationDataStorage.test.js} +3 -3
  172. package/test/data-storage/index.test.js +8 -0
  173. package/test/index.js +12 -0
  174. package/test/index.test.js +8 -7
  175. package/test/{helpers.js → middlewareTestUtil.js} +9 -12
  176. package/test/{normalizeInputLocation.test.js → parseInputLocationMiddleware.test.js} +9 -9
  177. package/test/{Action.test.js → redux/Action.test.js} +7 -6
  178. package/test/{ActionTypes.test.js → redux/ActionTypes.test.js} +2 -2
  179. package/test/redux/createMiddlewares.test.js +96 -0
  180. package/test/redux/index.test.js +10 -0
  181. package/test/{locationReducer.test.js → redux/locationReducer.test.js} +4 -7
  182. package/test/redux/middleware/createAddInputLocationBasePathMiddleware.test.js +40 -0
  183. package/test/redux/middleware/createNonProgrammaticNavigationBlockerMiddleware.test.js +264 -0
  184. package/test/redux/middleware/createProgrammaticNavigationBlockerMiddleware.test.js +312 -0
  185. package/test/redux/middleware/createRemoveOutputLocationBasePathMiddleware.test.js +51 -0
  186. package/test/{middleware/navigationActionMiddleware.test.js → redux/middleware/navigationOperationMiddleware.test.js} +16 -12
  187. package/test/{middleware/normalizeInputLocationMiddleware.test.js → redux/middleware/parseInputLocationMiddleware.test.js} +4 -4
  188. package/test/scroll-position/ScrollPositionRestoration.test.js +418 -0
  189. package/test/scroll-position/addScrollableContainer.js +36 -0
  190. package/test/scroll-position/addScrollableContainerWithHyperlink.js +50 -0
  191. package/test/scroll-position/createApp.js +112 -0
  192. package/test/scroll-position/delay.js +9 -0
  193. package/test/scroll-position/mockPageLifecycle.js +17 -0
  194. package/test/scroll-position/runApp.js +24 -0
  195. package/test/scroll-position/withScrollableContainerAtIndexPage.js +62 -0
  196. package/test/session/InMemorySession.test.js +348 -0
  197. package/test/session/ServerSession.test.js +17 -9
  198. package/test/session/WebBrowserSession.test.js +265 -0
  199. package/test/testUtil.js +3 -0
  200. package/types/data-storage/index.d.ts +35 -0
  201. package/types/index.d.ts +178 -157
  202. package/types/redux/index.d.ts +90 -0
  203. package/types/scroll-position/index.d.ts +107 -0
  204. package/types/tsconfig.json +1 -1
  205. package/lib/cjs/LocationDataStorage.js +0 -61
  206. package/lib/cjs/addBeforeLocationChangeListener.js +0 -7
  207. package/lib/cjs/beforeLocationChangeListeners.js +0 -51
  208. package/lib/cjs/createMiddlewares.js +0 -47
  209. package/lib/cjs/middleware/createBasePathMiddleware.js +0 -24
  210. package/lib/cjs/middleware/createBeforeLocationChangeListenerMiddleware.js +0 -39
  211. package/lib/cjs/middleware/createLocationMiddleware.js +0 -56
  212. package/lib/cjs/middleware/createNavigationBlockerMiddleware.js +0 -161
  213. package/lib/cjs/middleware/createTransformLocationMiddleware.js +0 -38
  214. package/lib/cjs/onlyAllowedOnClientSide.js +0 -10
  215. package/lib/cjs/session/BrowserSession.js +0 -235
  216. package/lib/cjs/session/MemorySession.js +0 -223
  217. package/lib/cjs/session/ServerSession.js +0 -65
  218. package/lib/esm/LocationDataStorage.js +0 -54
  219. package/lib/esm/addBeforeLocationChangeListener.js +0 -2
  220. package/lib/esm/beforeLocationChangeListeners.js +0 -44
  221. package/lib/esm/createMiddlewares.js +0 -41
  222. package/lib/esm/middleware/createBasePathMiddleware.js +0 -19
  223. package/lib/esm/middleware/createBeforeLocationChangeListenerMiddleware.js +0 -34
  224. package/lib/esm/middleware/createLocationMiddleware.js +0 -50
  225. package/lib/esm/middleware/createNavigationBlockerMiddleware.js +0 -156
  226. package/lib/esm/middleware/createTransformLocationMiddleware.js +0 -33
  227. package/lib/esm/onlyAllowedOnClientSide.js +0 -5
  228. package/lib/esm/session/BrowserSession.js +0 -229
  229. package/lib/esm/session/MemorySession.js +0 -217
  230. package/lib/esm/session/ServerSession.js +0 -58
  231. package/src/LocationDataStorage.js +0 -60
  232. package/src/addBeforeLocationChangeListener.js +0 -2
  233. package/src/beforeLocationChangeListeners.js +0 -54
  234. package/src/createMiddlewares.js +0 -45
  235. package/src/middleware/createBasePathMiddleware.js +0 -20
  236. package/src/middleware/createBeforeLocationChangeListenerMiddleware.js +0 -40
  237. package/src/middleware/createLocationMiddleware.js +0 -55
  238. package/src/middleware/createNavigationBlockerMiddleware.js +0 -168
  239. package/src/middleware/createTransformLocationMiddleware.js +0 -29
  240. package/src/onlyAllowedOnClientSide.js +0 -5
  241. package/src/session/BrowserSession.js +0 -235
  242. package/src/session/MemorySession.js +0 -219
  243. package/src/session/ServerSession.js +0 -67
  244. package/test/createMiddlewares.test.js +0 -62
  245. package/test/middleware/createBasePathMiddleware.test.js +0 -67
  246. package/test/middleware/createBeforeLocationChangeListenerMiddleware.test.js +0 -141
  247. package/test/middleware/createNavigationBlockerMiddleware.test.js +0 -471
  248. package/test/middleware/createTransformLocationMiddleware.test.js +0 -44
  249. package/test/session/BrowserSession.test.js +0 -182
  250. package/test/session/MemorySession.test.js +0 -244
  251. /package/lib/cjs/{locationReducer.js → redux/locationReducer.js} +0 -0
  252. /package/lib/esm/{locationReducer.js → redux/locationReducer.js} +0 -0
  253. /package/src/{locationReducer.js → redux/locationReducer.js} +0 -0
@@ -0,0 +1,312 @@
1
+ import delay from 'delay';
2
+ import pDefer from 'p-defer';
3
+ import { applyMiddleware, createStore } from 'redux';
4
+
5
+ import addNavigationBlockerOriginal from '../../../src/addNavigationBlocker';
6
+ import Actions from '../../../src/redux/Actions';
7
+ import createMiddlewares from '../../../src/redux/createMiddlewares';
8
+ import internalLocationReducer from '../../../src/redux/internalLocationReducer';
9
+ import InMemorySession from '../../../src/session/InMemorySession';
10
+ import { shouldWarn } from '../../testUtil';
11
+
12
+ describe('createProgrammaticNavigationBlockerMiddleware', () => {
13
+ // const sandbox = sinon.createSandbox();
14
+
15
+ let session;
16
+ let store;
17
+
18
+ function addNavigationBlocker(blocker) {
19
+ return addNavigationBlockerOriginal(session, blocker);
20
+ }
21
+
22
+ beforeEach(() => {
23
+ session = new InMemorySession();
24
+
25
+ store = createStore(
26
+ internalLocationReducer,
27
+ applyMiddleware(
28
+ ...createMiddlewares(session, {
29
+ _internalLocationReducer: true,
30
+ }),
31
+ ),
32
+ );
33
+ store.dispatch(Actions.init('/initial'));
34
+
35
+ sinon.spy(session.lifecycle, 'addTerminationBlocker');
36
+ });
37
+
38
+ afterEach(() => {
39
+ store.dispatch(Actions.stop());
40
+
41
+ // sandbox.restore();
42
+ });
43
+
44
+ describe('PUSH navigation', () => {
45
+ it('should block navigation when blocker returns `true`', () => {
46
+ const blocker = sinon.stub().returns(true);
47
+ addNavigationBlocker(blocker);
48
+
49
+ store.dispatch(Actions.push('/new'));
50
+ expect(store.getState().pathname).to.equal('/initial');
51
+
52
+ expect(blocker).to.have.been.calledOnce();
53
+
54
+ expect(blocker.firstCall.args[0]).to.include({
55
+ // operation: 'PUSH',
56
+ pathname: '/new',
57
+ });
58
+ });
59
+
60
+ it('should allow navigation when blocker returns `undefined`', () => {
61
+ addNavigationBlocker(() => undefined);
62
+
63
+ store.dispatch(Actions.push('/new'));
64
+ expect(store.getState().pathname).to.equal('/new');
65
+ });
66
+
67
+ it("should fall through when first blocker doesn't return `true`", () => {
68
+ const blocker1 = sinon.stub().returns(undefined);
69
+ const blocker2 = sinon.stub().returns(true);
70
+
71
+ addNavigationBlocker(blocker1);
72
+ addNavigationBlocker(blocker2);
73
+
74
+ store.dispatch(Actions.push('/new'));
75
+ expect(store.getState().pathname).to.equal('/initial');
76
+
77
+ expect(blocker1).to.have.been.calledOnce();
78
+ expect(blocker2).to.have.been.calledOnce();
79
+ });
80
+
81
+ it('should not fall through when first blocker returns `true`', () => {
82
+ const blocker1 = sinon.stub().returns(true);
83
+ const blocker2 = sinon.stub().returns(undefined);
84
+
85
+ addNavigationBlocker(blocker1);
86
+ addNavigationBlocker(blocker2);
87
+
88
+ store.dispatch(Actions.push('/new'));
89
+ expect(store.getState().pathname).to.equal('/initial');
90
+
91
+ expect(blocker1).to.have.been.calledOnce();
92
+ expect(blocker2).not.to.have.been.called();
93
+ });
94
+
95
+ it('should warn on and ignore blockers that throw', () => {
96
+ shouldWarn(
97
+ 'Ignoring navigation blocker `syncBlocker` that failed with `Error: Navigation blocker error example`.',
98
+ );
99
+
100
+ const syncBlocker = () => {
101
+ throw new Error('Navigation blocker error example');
102
+ };
103
+
104
+ addNavigationBlocker(syncBlocker);
105
+
106
+ store.dispatch(Actions.push('/new'));
107
+ expect(store.getState().pathname).to.equal('/new');
108
+ });
109
+
110
+ // it('should show a confirmation dialog and allow navigation on string', () => {
111
+ // sandbox.stub(window, 'confirm').returns(true);
112
+ //
113
+ // addNavigationBlocker(({ pathname }) => pathname);
114
+ //
115
+ // store.dispatch(Actions.push('/new'));
116
+ // expect(store.getState().pathname).to.equal('/new');
117
+ //
118
+ // expect(window.confirm)
119
+ // .to.have.been.calledOnce()
120
+ // .and.to.have.been.called.with('/new');
121
+ // });
122
+
123
+ // it('should show a confirmation dialog and block navigation on string', () => {
124
+ // sandbox.stub(window, 'confirm').returns(false);
125
+ //
126
+ // addNavigationBlocker(({ pathname }) => pathname);
127
+ //
128
+ // store.dispatch(Actions.push('/new'));
129
+ // expect(store.getState().pathname).to.equal('/initial');
130
+ //
131
+ // expect(window.confirm)
132
+ // .to.have.been.calledOnce()
133
+ // .and.to.have.been.called.with('/new');
134
+ // });
135
+
136
+ it('should allow navigation when blocker returns `undefined` (async)', async () => {
137
+ const navigationBlockerDeferred = pDefer();
138
+ addNavigationBlocker(() => navigationBlockerDeferred.promise);
139
+
140
+ store.dispatch(Actions.push('/new'));
141
+ expect(store.getState().pathname).to.equal('/initial');
142
+
143
+ navigationBlockerDeferred.resolve(undefined);
144
+ await delay(10);
145
+
146
+ expect(store.getState().pathname).to.equal('/new');
147
+ });
148
+
149
+ it('should block navigation when blocker returns `true` (async)', async () => {
150
+ const navigationBlockerDeferred = pDefer();
151
+ addNavigationBlocker(() => navigationBlockerDeferred.promise);
152
+
153
+ store.dispatch(Actions.push('/new'));
154
+ expect(store.getState().pathname).to.equal('/initial');
155
+
156
+ navigationBlockerDeferred.resolve(true);
157
+ await delay(10);
158
+
159
+ expect(store.getState().pathname).to.equal('/initial');
160
+ });
161
+
162
+ it('should allow chaining async blockers', async () => {
163
+ const navigationBlockerDeferred1 = pDefer();
164
+ const navigationBlockerDeferred2 = pDefer();
165
+
166
+ addNavigationBlocker(() => navigationBlockerDeferred1.promise);
167
+ addNavigationBlocker(() => navigationBlockerDeferred2.promise);
168
+
169
+ store.dispatch(Actions.push('/new'));
170
+ expect(store.getState().pathname).to.equal('/initial');
171
+
172
+ navigationBlockerDeferred1.resolve(undefined);
173
+ await delay(10);
174
+
175
+ expect(store.getState().pathname).to.equal('/initial');
176
+
177
+ navigationBlockerDeferred2.resolve(undefined);
178
+ await delay(10);
179
+
180
+ expect(store.getState().pathname).to.equal('/new');
181
+ });
182
+
183
+ it('should warn on and ignore async blockers that throw an error', async () => {
184
+ shouldWarn(
185
+ 'Ignoring navigation blocker `asyncBlocker` that failed with `Error: Navigation blocker error example`.',
186
+ );
187
+
188
+ // eslint-disable-next-line require-await
189
+ const asyncBlocker = async () => {
190
+ throw new Error('Navigation blocker error example');
191
+ };
192
+
193
+ addNavigationBlocker(asyncBlocker);
194
+
195
+ store.dispatch(Actions.push('/new'));
196
+ expect(store.getState().pathname).to.equal('/initial');
197
+
198
+ await delay(10);
199
+
200
+ expect(store.getState().pathname).to.equal('/new');
201
+ });
202
+
203
+ it('should allow removing blockers', () => {
204
+ const removeNavigationBlocker = addNavigationBlocker(() => true);
205
+
206
+ store.dispatch(Actions.push('/new'));
207
+ expect(store.getState().pathname).to.equal('/initial');
208
+
209
+ removeNavigationBlocker();
210
+
211
+ store.dispatch(Actions.push('/new'));
212
+ expect(store.getState().pathname).to.equal('/new');
213
+ });
214
+ });
215
+ });
216
+
217
+ describe('addTerminationBlocker', () => {
218
+ // const sandbox = sinon.createSandbox();
219
+
220
+ let session;
221
+ let store;
222
+
223
+ function addNavigationBlocker(blocker) {
224
+ return addNavigationBlockerOriginal(session, blocker);
225
+ }
226
+
227
+ beforeEach(() => {
228
+ session = new InMemorySession();
229
+
230
+ store = createStore(
231
+ internalLocationReducer,
232
+ applyMiddleware(...createMiddlewares(session)),
233
+ );
234
+ store.dispatch(Actions.init('/initial'));
235
+
236
+ sinon.spy(session.lifecycle, 'addTerminationBlocker');
237
+
238
+ // sandbox.stub(window, 'addEventListener');
239
+ // sandbox.stub(window, 'removeEventListener');
240
+ });
241
+
242
+ afterEach(() => {
243
+ if (store) {
244
+ store.dispatch(Actions.stop());
245
+ }
246
+
247
+ // sandbox.restore();
248
+ });
249
+
250
+ it('should add/remove event blocker', () => {
251
+ expect(session.lifecycle.addTerminationBlocker).not.to.have.been.called();
252
+ // expect(window.addEventListener).not.to.have.been.called();
253
+
254
+ const removeNavigationBlocker1 = addNavigationBlocker(() => null, {
255
+ beforeUnload: true,
256
+ });
257
+ expect(session.lifecycle.addTerminationBlocker).to.have.been.calledOnce();
258
+ // expect(window.addEventListener)
259
+ // .to.have.been.calledOnce()
260
+ // .and.to.have.been.called.with('beforeunload');
261
+
262
+ const removeNavigationBlocker2 = addNavigationBlocker(() => null, {
263
+ beforeUnload: true,
264
+ });
265
+ expect(session.lifecycle.addTerminationBlocker).to.have.been.calledOnce();
266
+ // expect(window.addEventListener)
267
+ // .to.have.been.calledOnce()
268
+ // .and.to.have.been.called.with('beforeunload');
269
+
270
+ const removeTerminationBlocker = sinon.stub();
271
+ // eslint-disable-next-line no-underscore-dangle
272
+ session._removeTerminationBlocker = removeTerminationBlocker;
273
+
274
+ removeNavigationBlocker1();
275
+ // expect(window.removeEventListener).not.to.have.been.called();
276
+ expect(removeTerminationBlocker).not.to.have.been.called();
277
+
278
+ removeNavigationBlocker2();
279
+ // expect(window.removeEventListener)
280
+ // .to.have.been.calledOnce()
281
+ // .and.to.have.been.called.with('beforeunload');
282
+ expect(removeTerminationBlocker).to.have.been.calledOnce();
283
+ });
284
+
285
+ it('should remove event blocker on stop', () => {
286
+ addNavigationBlocker(() => null);
287
+
288
+ const removeTerminationBlocker = sinon.stub();
289
+ // eslint-disable-next-line no-underscore-dangle
290
+ session._removeTerminationBlocker = removeTerminationBlocker;
291
+
292
+ // expect(window.removeEventListener).not.to.have.been.called();
293
+ expect(removeTerminationBlocker).not.to.have.been.called();
294
+
295
+ store.dispatch(Actions.stop());
296
+ // Prevent `store.dispatch(Actions.stop())` in `afterEach()`.
297
+ store = undefined;
298
+
299
+ // expect(window.removeEventListener)
300
+ // .to.have.been.calledOnce()
301
+ // .and.to.have.been.called.with('beforeunload');
302
+ expect(removeTerminationBlocker).to.have.been.calledOnce();
303
+ });
304
+
305
+ // it('should not add a global "before destroy" listener when no `beforeTermination` blocker has been added', () => {
306
+ // const removeNavigationBlocker = addNavigationBlocker(() => null);
307
+ // expect(window.addEventListener).not.to.have.been.called();
308
+ //
309
+ // removeNavigationBlocker();
310
+ // expect(window.removeEventListener).not.to.have.been.called();
311
+ // });
312
+ });
@@ -0,0 +1,51 @@
1
+ import createRemoveOutputLocationBasePathMiddleware from '../../../src/redux/middleware/createRemoveOutputLocationBasePathMiddleware';
2
+ import { transformOutputLocationUsingMiddleware } from '../../middlewareTestUtil';
3
+
4
+ describe('createRemoveOutputLocationBasePathMiddleware', () => {
5
+ [
6
+ ['/base', 'generic `basePath`'],
7
+ ['/base/', '`basePath` with a trailing slash'],
8
+ ].forEach(([basePath, title]) => {
9
+ describe(title, () => {
10
+ const removeOutputLocationBasePathMiddleware =
11
+ createRemoveOutputLocationBasePathMiddleware(basePath);
12
+
13
+ it('should strip `basePath` from `location.pathname` on subscription locations', () => {
14
+ expect(
15
+ transformOutputLocationUsingMiddleware(
16
+ removeOutputLocationBasePathMiddleware,
17
+ { pathname: '/base/path' },
18
+ ),
19
+ ).to.eql({
20
+ pathname: '/path',
21
+ });
22
+ });
23
+
24
+ it('should handle unrecognized paths on subscription locations', () => {
25
+ expect(
26
+ transformOutputLocationUsingMiddleware(
27
+ removeOutputLocationBasePathMiddleware,
28
+ { pathname: '/path' },
29
+ ),
30
+ ).to.eql({
31
+ pathname: '/path',
32
+ });
33
+ });
34
+ });
35
+ });
36
+
37
+ describe('No `basePath` specified', () => {
38
+ const removeOutputLocationBasePathMiddleware =
39
+ createRemoveOutputLocationBasePathMiddleware('/');
40
+
41
+ it('should not modify `location.pathname` of subscription locations', () => {
42
+ const location = { pathname: '/path' };
43
+ expect(
44
+ transformOutputLocationUsingMiddleware(
45
+ removeOutputLocationBasePathMiddleware,
46
+ location,
47
+ ),
48
+ ).to.equal(location);
49
+ });
50
+ });
51
+ });
@@ -1,12 +1,12 @@
1
- import ActionTypes from '../../src/ActionTypes';
2
- import navigationActionMiddleware from '../../src/middleware/navigationActionMiddleware';
1
+ import ActionTypes from '../../../src/redux/ActionTypes';
2
+ import navigationOperationMiddleware from '../../../src/redux/middleware/navigationOperationMiddleware';
3
3
 
4
- describe('navigationActionMiddleware', () => {
4
+ describe('navigationOperationMiddleware', () => {
5
5
  let next;
6
6
  let dispatch;
7
7
  beforeEach(() => {
8
8
  next = sinon.spy();
9
- dispatch = navigationActionMiddleware()(next);
9
+ dispatch = navigationOperationMiddleware()(next);
10
10
  });
11
11
 
12
12
  it('should change `type` of PUSH action', () => {
@@ -22,10 +22,12 @@ describe('navigationActionMiddleware', () => {
22
22
  expect(next).to.be.calledWith({
23
23
  type: ActionTypes.NAVIGATE,
24
24
  payload: {
25
- action: 'PUSH',
26
- pathname: '/foo',
27
- search: '?bar=baz',
28
- hash: '#qux',
25
+ operation: 'PUSH',
26
+ location: {
27
+ pathname: '/foo',
28
+ search: '?bar=baz',
29
+ hash: '#qux',
30
+ },
29
31
  },
30
32
  });
31
33
  });
@@ -43,10 +45,12 @@ describe('navigationActionMiddleware', () => {
43
45
  expect(next).to.be.calledWith({
44
46
  type: ActionTypes.NAVIGATE,
45
47
  payload: {
46
- action: 'REPLACE',
47
- pathname: '/foo',
48
- search: '?bar=baz',
49
- hash: '#qux',
48
+ operation: 'REPLACE',
49
+ location: {
50
+ pathname: '/foo',
51
+ search: '?bar=baz',
52
+ hash: '#qux',
53
+ },
50
54
  },
51
55
  });
52
56
  });
@@ -1,12 +1,12 @@
1
- import ActionTypes from '../../src/ActionTypes';
2
- import normalizeInputLocationMiddleware from '../../src/middleware/normalizeInputLocationMiddleware';
1
+ import ActionTypes from '../../../src/redux/ActionTypes';
2
+ import parseInputLocationMiddleware from '../../../src/redux/middleware/parseInputLocationMiddleware';
3
3
 
4
- describe('normalizeInputLocationMiddleware', () => {
4
+ describe('parseInputLocationMiddleware', () => {
5
5
  let next;
6
6
  let dispatch;
7
7
  beforeEach(() => {
8
8
  next = sinon.spy();
9
- dispatch = normalizeInputLocationMiddleware()(next);
9
+ dispatch = parseInputLocationMiddleware()(next);
10
10
  });
11
11
 
12
12
  it('should transform input location of PUSH action (`string` to `object`)', () => {