navigation-stack 0.3.0 → 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 -60
  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 -53
  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 -59
  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,62 @@
1
+ // Adds a 100x100 scrollable container on the website.
2
+ //
3
+ // * When the current location is "/", it renders a large amount of content
4
+ // (20000x20000 to be specific) inside the container, making it scrollable.
5
+ //
6
+ // * At any other location, it doesn't render anything in the container.
7
+ //
8
+ export default function withScrollableContainerAtIndexPage(app) {
9
+ const container = document.createElement('div');
10
+ container.style.height = '100px';
11
+ container.style.width = '100px';
12
+ container.style.overflow = 'hidden';
13
+ document.body.appendChild(container);
14
+
15
+ let element;
16
+ let unregister;
17
+
18
+ // This will only be called once, so no need to guard.
19
+ function listen(listener) {
20
+ function shouldUpdatePageScrollPositionForLocation(
21
+ location,
22
+ prevLocation,
23
+ ) {
24
+ // Disable the automatic scroll restoration after a SHIFT, to check the
25
+ // scroll-on-register behavior.
26
+ if (prevLocation && location.operation === 'SHIFT') {
27
+ return false;
28
+ }
29
+
30
+ return true;
31
+ }
32
+
33
+ const unlisten = app.listen((location) => {
34
+ listener(location);
35
+
36
+ if (location.pathname === '/') {
37
+ element = document.createElement('div');
38
+ element.style.height = '20000px';
39
+ element.style.width = '20000px';
40
+ container.appendChild(element);
41
+
42
+ unregister = app.registerScrollableContainer('container', container, {
43
+ shouldUpdatePageScrollPositionForLocation,
44
+ });
45
+ } else {
46
+ container.removeChild(element);
47
+ unregister();
48
+ }
49
+ });
50
+
51
+ return () => {
52
+ unlisten();
53
+ document.body.removeChild(container);
54
+ };
55
+ }
56
+
57
+ return {
58
+ ...app,
59
+ container,
60
+ listen,
61
+ };
62
+ }
@@ -0,0 +1,348 @@
1
+ import parseInputLocation from '../../src/parseInputLocation';
2
+ import InMemorySession from '../../src/session/InMemorySession';
3
+
4
+ describe('InMemorySession', () => {
5
+ it('should parse the initial location', () => {
6
+ const session = new InMemorySession();
7
+
8
+ let location;
9
+ session.subscribe((newLocation) => {
10
+ location = newLocation;
11
+ });
12
+
13
+ session.start(parseInputLocation('/initial?bar=baz#qux'));
14
+
15
+ expect(location).to.deep.include({
16
+ operation: 'INIT',
17
+ pathname: '/initial',
18
+ search: '?bar=baz',
19
+ query: {
20
+ bar: 'baz',
21
+ },
22
+ hash: '#qux',
23
+ index: 0,
24
+ delta: 0,
25
+ });
26
+
27
+ session.stop();
28
+ });
29
+
30
+ it('should support basic navigation', () => {
31
+ const session = new InMemorySession();
32
+
33
+ let location;
34
+ session.subscribe((newLocation) => {
35
+ location = newLocation;
36
+ });
37
+
38
+ const listener = sinon.spy();
39
+ session.subscribe(listener);
40
+
41
+ session.start(parseInputLocation('/initial'));
42
+
43
+ expect(listener).to.have.been.calledOnce();
44
+ expect(listener.firstCall.args[0]).to.deep.include({
45
+ operation: 'INIT',
46
+ pathname: '/initial',
47
+ index: 0,
48
+ delta: 0,
49
+ });
50
+ listener.resetHistory();
51
+
52
+ session.navigate('PUSH', {
53
+ pathname: '/new',
54
+ });
55
+
56
+ const newLocation = location;
57
+
58
+ expect(newLocation).to.deep.include({
59
+ operation: 'PUSH',
60
+ pathname: '/new',
61
+ index: 1,
62
+ delta: 1,
63
+ });
64
+ expect(newLocation.key).not.to.be.empty();
65
+
66
+ expect(listener).to.have.been.calledOnce();
67
+ expect(listener.firstCall.args[0]).to.deep.include({
68
+ operation: 'PUSH',
69
+ pathname: '/new',
70
+ index: 1,
71
+ delta: 1,
72
+ });
73
+ listener.resetHistory();
74
+
75
+ session.navigate('PUSH', { pathname: '/new-2' });
76
+
77
+ expect(location).to.include({
78
+ operation: 'PUSH',
79
+ pathname: '/new-2',
80
+ index: 2,
81
+ delta: 1,
82
+ });
83
+
84
+ expect(listener).to.have.been.calledOnce();
85
+ expect(listener.firstCall.args[0]).to.deep.include({
86
+ operation: 'PUSH',
87
+ pathname: '/new-2',
88
+ index: 2,
89
+ delta: 1,
90
+ });
91
+ listener.resetHistory();
92
+
93
+ session.navigate('REPLACE', { pathname: '/new-3' });
94
+
95
+ expect(location).to.include({
96
+ operation: 'REPLACE',
97
+ pathname: '/new-3',
98
+ index: 2,
99
+ delta: 0,
100
+ });
101
+
102
+ expect(listener).to.have.been.calledOnce();
103
+ expect(listener.firstCall.args[0]).to.deep.include({
104
+ operation: 'REPLACE',
105
+ pathname: '/new-3',
106
+ index: 2,
107
+ delta: 0,
108
+ });
109
+ listener.resetHistory();
110
+
111
+ session.shift(-1);
112
+
113
+ expect(listener).to.have.been.calledOnce();
114
+ expect(listener.firstCall.args[0]).to.deep.include({
115
+ operation: 'SHIFT',
116
+ pathname: '/new',
117
+ key: newLocation.key,
118
+ index: 1,
119
+ delta: -1,
120
+ });
121
+ listener.resetHistory();
122
+
123
+ session.stop();
124
+ });
125
+
126
+ it('should support subscribing and unsubscribing', () => {
127
+ const session = new InMemorySession();
128
+ session.start(parseInputLocation('/initial'));
129
+ session.navigate('PUSH', { pathname: '/new' });
130
+ session.navigate('PUSH', { pathname: '/new-2' });
131
+
132
+ const listener = sinon.spy();
133
+ const unsubscribe = session.subscribe(listener);
134
+
135
+ session.shift(-1);
136
+
137
+ expect(listener).to.have.been.calledOnce();
138
+ expect(listener.firstCall.args[0]).to.include({
139
+ operation: 'SHIFT',
140
+ pathname: '/new',
141
+ });
142
+ listener.resetHistory();
143
+
144
+ unsubscribe();
145
+
146
+ session.shift(-1);
147
+
148
+ expect(listener).not.to.have.been.called();
149
+
150
+ session.stop();
151
+ });
152
+
153
+ it('should respect stack bounds', () => {
154
+ const session = new InMemorySession();
155
+ session.start(parseInputLocation('/initial'));
156
+ session.navigate('PUSH', { pathname: '/new' });
157
+ session.navigate('PUSH', { pathname: '/new-2' });
158
+
159
+ const listener = sinon.spy();
160
+ session.subscribe(listener);
161
+
162
+ expect(() => {
163
+ session.shift(-390);
164
+ }).to.throw('out of navigation history bounds');
165
+
166
+ expect(listener).to.not.have.been.called();
167
+
168
+ session.shift(-2);
169
+
170
+ expect(listener).to.have.been.calledOnce();
171
+ expect(listener.firstCall.args[0]).to.include({
172
+ operation: 'SHIFT',
173
+ pathname: '/initial',
174
+ delta: -2,
175
+ });
176
+ listener.resetHistory();
177
+
178
+ expect(() => {
179
+ session.shift(-1);
180
+ }).to.throw('out of navigation history bounds');
181
+
182
+ expect(listener).not.to.have.been.called();
183
+
184
+ expect(() => {
185
+ session.shift(+22);
186
+ }).to.throw('out of navigation history bounds');
187
+
188
+ session.shift(+2);
189
+
190
+ expect(listener).to.have.been.calledOnce();
191
+ expect(listener.firstCall.args[0]).to.include({
192
+ operation: 'SHIFT',
193
+ pathname: '/new-2',
194
+ delta: 2,
195
+ });
196
+ listener.resetHistory();
197
+
198
+ expect(() => {
199
+ session.shift(+1);
200
+ }).to.throw('out of navigation history bounds');
201
+
202
+ expect(listener).not.to.have.been.called();
203
+
204
+ session.stop();
205
+ });
206
+
207
+ it('should not reset forward entries on replace', () => {
208
+ const session = new InMemorySession();
209
+ session.start(parseInputLocation('/initial'));
210
+ session.navigate('PUSH', { pathname: '/new' });
211
+ session.navigate('PUSH', { pathname: '/new-2' });
212
+ session.shift(-2);
213
+ session.navigate('REPLACE', { pathname: '/new-3' });
214
+
215
+ const listener = sinon.spy();
216
+ session.subscribe(listener);
217
+
218
+ session.shift(+1);
219
+
220
+ expect(listener).to.have.been.calledOnce();
221
+ expect(listener.firstCall.args[0]).to.include({
222
+ operation: 'SHIFT',
223
+ pathname: '/new',
224
+ delta: 1,
225
+ });
226
+
227
+ session.stop();
228
+ });
229
+
230
+ it('should reset forward entries on push', () => {
231
+ const session = new InMemorySession();
232
+
233
+ session.start(parseInputLocation('/initial'));
234
+ session.navigate('PUSH', { pathname: '/new' });
235
+ session.navigate('PUSH', { pathname: '/new-2' });
236
+ session.shift(-2);
237
+ session.navigate('PUSH', { pathname: '/new-3' });
238
+
239
+ const listener = sinon.spy();
240
+ session.subscribe(listener);
241
+
242
+ expect(() => {
243
+ session.shift(+1);
244
+ }).to.throw('out of navigation history bounds');
245
+
246
+ expect(listener).not.to.have.been.called();
247
+
248
+ session.stop();
249
+ });
250
+
251
+ it('should remove all subscriptions on stop', () => {
252
+ const session = new InMemorySession();
253
+
254
+ session.start(parseInputLocation('/initial'));
255
+
256
+ // eslint-disable-next-line no-underscore-dangle
257
+ expect(session._subscription._listeners.length).to.equal(1);
258
+
259
+ const listener = sinon.spy();
260
+ session.subscribe(listener);
261
+
262
+ // eslint-disable-next-line no-underscore-dangle
263
+ expect(session._subscription._listeners.length).to.equal(2);
264
+
265
+ session.stop();
266
+
267
+ // eslint-disable-next-line no-underscore-dangle
268
+ expect(session._subscription._listeners.length).to.equal(0);
269
+ });
270
+
271
+ // describe('persistence', () => {
272
+ // // beforeEach(() => {
273
+ // // window.sessionStorage.clear();
274
+ // // });
275
+ //
276
+ // it('should support persistence', () => {
277
+ // // function save(key, data) {
278
+ // // window.sessionStorage.setItem(key, data);
279
+ // // }
280
+ // //
281
+ // // function load(key) {
282
+ // // return window.sessionStorage.getItem(key);
283
+ // // }
284
+ //
285
+ // const storage = {};
286
+ //
287
+ // const save = (key, data) => {
288
+ // storage[key] = data;
289
+ // };
290
+ //
291
+ // const load = (key) => {
292
+ // return storage[key];
293
+ // };
294
+ //
295
+ // const session1 = new InMemorySession({ save, load });
296
+ // expect(session1.start(parseInputLocation('/initial))).to.include({
297
+ // pathname: '/initial',
298
+ // });
299
+ //
300
+ // session1._navigation.navigate('PUSH', { pathname: '/new' });
301
+ // session1._navigation.navigate('PUSH', { pathname: '/new-2' });
302
+ // session1._navigation.shift(-1);
303
+ //
304
+ // const session2 = new InMemorySession({ save, load });
305
+ // expect(parseInputLocation(session2.start('/initial'))).to.include({
306
+ // pathname: '/new',
307
+ // });
308
+ //
309
+ // session2._navigation.shift(+1);
310
+ // expect(parseInputLocation(session2.start())).to.include({
311
+ // pathname: '/new-2',
312
+ // });
313
+ // session1.stop();
314
+ // session2.stop();
315
+ // });
316
+ //
317
+ // it('should ignore broken session storage entry', () => {
318
+ // // function save(key, data) {
319
+ // // window.sessionStorage.setItem(key, data);
320
+ // // }
321
+ // //
322
+ // // function load(key) {
323
+ // // return window.sessionStorage.getItem(key);
324
+ // // }
325
+ //
326
+ // const storage = {};
327
+ //
328
+ // const save = (key, data) => {
329
+ // storage[key] = data;
330
+ // };
331
+ //
332
+ // const load = (key) => {
333
+ // return storage[key];
334
+ // };
335
+ //
336
+ // save(
337
+ // // `stack` should have sufficient items so that the `index` wouldn't be out of bounds.
338
+ // JSON.stringify({ stack: [], index: 2 }),
339
+ // );
340
+ //
341
+ // const session = new InMemorySession({ save, load });
342
+ // expect(session.start(parseInputLocation('/initial'))).to.include({
343
+ // pathname: '/initial',
344
+ // });
345
+ // session.stop();
346
+ // });
347
+ // });
348
+ });
@@ -1,11 +1,19 @@
1
- import ServerSession from '../../src/session/ServerSession';
1
+ import parseInputLocation from '../../src/parseInputLocation';
2
+ import ServerSideRenderSession from '../../src/session/ServerSideRenderSession';
2
3
 
3
- describe('ServerSession', () => {
4
+ describe('ServerSideRenderSession', () => {
4
5
  it('should parse the initial location', () => {
5
- const session = new ServerSession('/foo?bar=baz#qux');
6
+ const session = new ServerSideRenderSession();
6
7
 
7
- expect(session.navigation.init()).to.deep.include({
8
- action: 'INIT',
8
+ let location;
9
+ session.subscribe((newLocation) => {
10
+ location = newLocation;
11
+ });
12
+
13
+ session.start(parseInputLocation('/foo?bar=baz#qux'));
14
+
15
+ expect(location).to.deep.include({
16
+ operation: 'INIT',
9
17
  pathname: '/foo',
10
18
  search: '?bar=baz',
11
19
  query: {
@@ -15,9 +23,9 @@ describe('ServerSession', () => {
15
23
  });
16
24
  });
17
25
 
18
- it('should have dummy support for subscriptions', () => {
19
- const session = new ServerSession('/foo?bar=baz#qux');
20
- const unsubscribe = session.navigation.subscribe();
21
- expect(unsubscribe).to.not.throw();
26
+ it('should support subscriptions', () => {
27
+ const session = new ServerSideRenderSession();
28
+ // eslint-disable-next-line no-unused-vars
29
+ expect(() => session.subscribe((location) => {})).to.not.throw();
22
30
  });
23
31
  });
@@ -0,0 +1,265 @@
1
+ import delay from 'delay';
2
+
3
+ import parseInputLocation from '../../src/parseInputLocation';
4
+ import WebBrowserSession from '../../src/session/WebBrowserSession';
5
+
6
+ describe('WebBrowserSession.stop()', () => {
7
+ const sandbox = sinon.createSandbox();
8
+
9
+ afterEach(() => {
10
+ // Even if a test errors, the spies should be removed, otherwise it'll say:
11
+ // "Attempted to wrap addEventListener which is already wrapped".
12
+ sandbox.restore();
13
+ });
14
+
15
+ it('should add and remove "popstate" event listener', () => {
16
+ sandbox.spy(window, 'addEventListener');
17
+ sandbox.spy(window, 'removeEventListener');
18
+
19
+ const session = new WebBrowserSession();
20
+
21
+ session.subscribe(() => {});
22
+
23
+ const listener = sinon.spy();
24
+
25
+ // This subscription won't be "unsubscribed" by the code.
26
+ // It is expected to be "unsubscribed" automatically on `session.stop()`
27
+ // and not hold off the clearing of `popstate` listener.
28
+ session.subscribe(listener);
29
+
30
+ session.start(window.location);
31
+ session.stop();
32
+
33
+ expect(window.addEventListener)
34
+ .to.have.been.calledOnce()
35
+ .and.to.have.been.called.with('popstate');
36
+
37
+ expect(window.removeEventListener)
38
+ .to.have.been.calledOnce()
39
+ .and.to.have.been.called.with('popstate');
40
+ });
41
+ });
42
+
43
+ describe('WebBrowserSession', () => {
44
+ let session;
45
+
46
+ afterEach(() => {
47
+ // Even if a test errors, the session should still be stopped
48
+ // in order to remove the "popstate" listener so that it doesn't
49
+ // interfere with other tests.
50
+ session.stop();
51
+ });
52
+
53
+ it('should parse the initial location', () => {
54
+ window.history.replaceState(null, null, '/initial?bar=baz#qux');
55
+ session = new WebBrowserSession();
56
+
57
+ let location;
58
+ session.subscribe((newLocation) => {
59
+ location = newLocation;
60
+ });
61
+
62
+ session.start(parseInputLocation(window.location));
63
+
64
+ expect(location).to.deep.include({
65
+ operation: 'INIT',
66
+ pathname: '/initial',
67
+ search: '?bar=baz',
68
+ query: {
69
+ bar: 'baz',
70
+ },
71
+ hash: '#qux',
72
+ index: 0,
73
+ delta: 0,
74
+ });
75
+ });
76
+
77
+ it('should require initialization', () => {
78
+ session = new WebBrowserSession();
79
+
80
+ expect(() =>
81
+ session.navigate('PUSH', {
82
+ pathname: '/new',
83
+ search: '?search',
84
+ hash: '#hash',
85
+ }),
86
+ ).to.throw('Not started');
87
+ });
88
+
89
+ it('should support basic navigation', async () => {
90
+ window.history.replaceState(null, null, '/initial');
91
+
92
+ session = new WebBrowserSession();
93
+
94
+ let location;
95
+ session.subscribe((newLocation) => {
96
+ location = newLocation;
97
+ });
98
+
99
+ const listener = sinon.spy();
100
+ session.subscribe(listener);
101
+
102
+ session.start(parseInputLocation(window.location));
103
+
104
+ expect(listener).to.have.been.calledOnce();
105
+ expect(listener.firstCall.args[0]).to.deep.include({
106
+ operation: 'INIT',
107
+ pathname: '/initial',
108
+ index: 0,
109
+ delta: 0,
110
+ });
111
+ listener.resetHistory();
112
+
113
+ session.navigate('PUSH', {
114
+ pathname: '/new',
115
+ search: '?search',
116
+ hash: '#hash',
117
+ });
118
+
119
+ const newLocation = location;
120
+
121
+ expect(window.location).to.include({
122
+ pathname: '/new',
123
+ search: '?search',
124
+ hash: '#hash',
125
+ });
126
+ expect(newLocation).to.deep.include({
127
+ operation: 'PUSH',
128
+ pathname: '/new',
129
+ search: '?search',
130
+ hash: '#hash',
131
+ index: 1,
132
+ delta: 1,
133
+ });
134
+ expect(newLocation.key).not.to.be.empty();
135
+
136
+ expect(listener).to.have.been.calledOnce();
137
+ expect(listener.firstCall.args[0]).to.deep.include({
138
+ operation: 'PUSH',
139
+ pathname: '/new',
140
+ index: 1,
141
+ delta: 1,
142
+ });
143
+ listener.resetHistory();
144
+
145
+ session.navigate('PUSH', {
146
+ pathname: '/new-2',
147
+ search: '',
148
+ hash: '',
149
+ });
150
+
151
+ expect(location).to.include({
152
+ operation: 'PUSH',
153
+ pathname: '/new-2',
154
+ index: 2,
155
+ delta: 1,
156
+ });
157
+
158
+ expect(listener).to.have.been.calledOnce();
159
+ expect(listener.firstCall.args[0]).to.deep.include({
160
+ operation: 'PUSH',
161
+ pathname: '/new-2',
162
+ index: 2,
163
+ delta: 1,
164
+ });
165
+ listener.resetHistory();
166
+
167
+ expect(window.location.pathname).to.equal('/new-2');
168
+
169
+ session.navigate('REPLACE', {
170
+ pathname: '/new-3',
171
+ search: '',
172
+ hash: '',
173
+ });
174
+
175
+ expect(location).to.include({
176
+ operation: 'REPLACE',
177
+ pathname: '/new-3',
178
+ index: 2,
179
+ delta: 0,
180
+ });
181
+ await delay(20);
182
+
183
+ expect(window.location.pathname).to.equal('/new-3');
184
+
185
+ expect(listener).to.have.been.calledOnce();
186
+ expect(listener.firstCall.args[0]).to.deep.include({
187
+ operation: 'REPLACE',
188
+ pathname: '/new-3',
189
+ index: 2,
190
+ delta: 0,
191
+ });
192
+ listener.resetHistory();
193
+
194
+ session.shift(-1);
195
+ await delay(20);
196
+
197
+ expect(window.location).to.include({
198
+ pathname: '/new',
199
+ search: '?search',
200
+ hash: '#hash',
201
+ });
202
+
203
+ expect(listener).to.have.been.calledOnce();
204
+ expect(listener.firstCall.args[0]).to.deep.include({
205
+ operation: 'SHIFT',
206
+ pathname: '/new',
207
+ search: '?search',
208
+ hash: '#hash',
209
+ key: newLocation.key,
210
+ index: 1,
211
+ delta: -1,
212
+ });
213
+ listener.resetHistory();
214
+
215
+ window.history.back();
216
+ await delay(20);
217
+
218
+ expect(window.location.pathname).to.equal('/initial');
219
+
220
+ expect(listener).to.have.been.calledOnce();
221
+ expect(listener.firstCall.args[0]).to.deep.include({
222
+ operation: 'SHIFT',
223
+ pathname: '/initial',
224
+ index: 0,
225
+ delta: -1,
226
+ });
227
+ listener.resetHistory();
228
+ });
229
+
230
+ it('should support subscribing and unsubscribing', async () => {
231
+ window.history.replaceState(null, null, '/');
232
+ session = new WebBrowserSession();
233
+ session.start(parseInputLocation(window.location));
234
+ session.navigate('PUSH', {
235
+ pathname: '/new',
236
+ search: '',
237
+ hash: '',
238
+ });
239
+ session.navigate('PUSH', {
240
+ pathname: '/new-2',
241
+ search: '',
242
+ hash: '',
243
+ });
244
+
245
+ const listener = sinon.spy();
246
+ const unsubscribe = session.subscribe(listener);
247
+
248
+ session.shift(-1);
249
+ await delay(20);
250
+
251
+ expect(listener).to.have.been.calledOnce();
252
+ expect(listener.firstCall.args[0]).to.include({
253
+ operation: 'SHIFT',
254
+ pathname: '/new',
255
+ });
256
+ listener.resetHistory();
257
+
258
+ unsubscribe();
259
+
260
+ session.shift(-1);
261
+ await delay(20);
262
+
263
+ expect(listener).not.to.have.been.called();
264
+ });
265
+ });