@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430

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 (300) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +884 -4
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +4474 -867
  5. package/package.json +60 -51
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +50 -21
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +12 -8
  11. package/skills/document-cache/SKILL.md +18 -16
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +334 -72
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +131 -8
  16. package/skills/layout/SKILL.md +100 -3
  17. package/skills/links/SKILL.md +89 -30
  18. package/skills/loader/SKILL.md +388 -38
  19. package/skills/middleware/SKILL.md +171 -34
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +78 -1
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +85 -16
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +226 -14
  26. package/skills/router-setup/SKILL.md +123 -30
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +9 -8
  29. package/skills/typesafety/SKILL.md +318 -89
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +102 -4
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/event-controller.ts +87 -64
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/intercept-utils.ts +52 -0
  38. package/src/browser/link-interceptor.ts +24 -4
  39. package/src/browser/logging.ts +55 -0
  40. package/src/browser/merge-segment-loaders.ts +20 -12
  41. package/src/browser/navigation-bridge.ts +285 -553
  42. package/src/browser/navigation-client.ts +124 -71
  43. package/src/browser/navigation-store.ts +33 -50
  44. package/src/browser/navigation-transaction.ts +295 -0
  45. package/src/browser/network-error-handler.ts +61 -0
  46. package/src/browser/partial-update.ts +258 -308
  47. package/src/browser/prefetch/cache.ts +146 -0
  48. package/src/browser/prefetch/fetch.ts +135 -0
  49. package/src/browser/prefetch/observer.ts +65 -0
  50. package/src/browser/prefetch/policy.ts +42 -0
  51. package/src/browser/prefetch/queue.ts +88 -0
  52. package/src/browser/rango-state.ts +112 -0
  53. package/src/browser/react/Link.tsx +185 -73
  54. package/src/browser/react/NavigationProvider.tsx +51 -11
  55. package/src/browser/react/context.ts +6 -0
  56. package/src/browser/react/filter-segment-order.ts +11 -0
  57. package/src/browser/react/index.ts +12 -12
  58. package/src/browser/react/location-state-shared.ts +95 -53
  59. package/src/browser/react/location-state.ts +60 -15
  60. package/src/browser/react/mount-context.ts +6 -1
  61. package/src/browser/react/nonce-context.ts +23 -0
  62. package/src/browser/react/shallow-equal.ts +27 -0
  63. package/src/browser/react/use-action.ts +29 -51
  64. package/src/browser/react/use-client-cache.ts +5 -3
  65. package/src/browser/react/use-handle.ts +32 -79
  66. package/src/browser/react/use-href.tsx +2 -2
  67. package/src/browser/react/use-link-status.ts +6 -5
  68. package/src/browser/react/use-navigation.ts +22 -63
  69. package/src/browser/react/use-params.ts +65 -0
  70. package/src/browser/react/use-pathname.ts +47 -0
  71. package/src/browser/react/use-router.ts +63 -0
  72. package/src/browser/react/use-search-params.ts +56 -0
  73. package/src/browser/react/use-segments.ts +80 -97
  74. package/src/browser/response-adapter.ts +73 -0
  75. package/src/browser/rsc-router.tsx +107 -26
  76. package/src/browser/scroll-restoration.ts +92 -16
  77. package/src/browser/segment-reconciler.ts +216 -0
  78. package/src/browser/segment-structure-assert.ts +16 -0
  79. package/src/browser/server-action-bridge.ts +504 -599
  80. package/src/browser/shallow.ts +6 -1
  81. package/src/browser/types.ts +109 -47
  82. package/src/browser/validate-redirect-origin.ts +29 -0
  83. package/src/build/generate-manifest.ts +235 -24
  84. package/src/build/generate-route-types.ts +36 -0
  85. package/src/build/index.ts +13 -0
  86. package/src/build/route-trie.ts +265 -0
  87. package/src/build/route-types/ast-helpers.ts +25 -0
  88. package/src/build/route-types/ast-route-extraction.ts +98 -0
  89. package/src/build/route-types/codegen.ts +102 -0
  90. package/src/build/route-types/include-resolution.ts +411 -0
  91. package/src/build/route-types/param-extraction.ts +48 -0
  92. package/src/build/route-types/per-module-writer.ts +128 -0
  93. package/src/build/route-types/router-processing.ts +469 -0
  94. package/src/build/route-types/scan-filter.ts +78 -0
  95. package/src/build/runtime-discovery.ts +231 -0
  96. package/src/cache/background-task.ts +34 -0
  97. package/src/cache/cache-key-utils.ts +44 -0
  98. package/src/cache/cache-policy.ts +125 -0
  99. package/src/cache/cache-runtime.ts +338 -0
  100. package/src/cache/cache-scope.ts +120 -303
  101. package/src/cache/cf/cf-cache-store.ts +119 -7
  102. package/src/cache/cf/index.ts +8 -2
  103. package/src/cache/document-cache.ts +101 -72
  104. package/src/cache/handle-capture.ts +81 -0
  105. package/src/cache/handle-snapshot.ts +41 -0
  106. package/src/cache/index.ts +0 -15
  107. package/src/cache/memory-segment-store.ts +191 -13
  108. package/src/cache/profile-registry.ts +73 -0
  109. package/src/cache/read-through-swr.ts +134 -0
  110. package/src/cache/segment-codec.ts +256 -0
  111. package/src/cache/taint.ts +98 -0
  112. package/src/cache/types.ts +72 -122
  113. package/src/client.rsc.tsx +3 -1
  114. package/src/client.tsx +106 -126
  115. package/src/component-utils.ts +4 -4
  116. package/src/components/DefaultDocument.tsx +5 -1
  117. package/src/context-var.ts +86 -0
  118. package/src/debug.ts +17 -7
  119. package/src/errors.ts +108 -2
  120. package/src/handle.ts +15 -29
  121. package/src/handles/MetaTags.tsx +73 -20
  122. package/src/handles/breadcrumbs.ts +66 -0
  123. package/src/handles/index.ts +1 -0
  124. package/src/handles/meta.ts +30 -13
  125. package/src/host/cookie-handler.ts +21 -15
  126. package/src/host/errors.ts +8 -8
  127. package/src/host/index.ts +4 -7
  128. package/src/host/pattern-matcher.ts +27 -27
  129. package/src/host/router.ts +61 -39
  130. package/src/host/testing.ts +8 -8
  131. package/src/host/types.ts +15 -7
  132. package/src/host/utils.ts +1 -1
  133. package/src/href-client.ts +119 -29
  134. package/src/index.rsc.ts +153 -19
  135. package/src/index.ts +211 -30
  136. package/src/internal-debug.ts +11 -0
  137. package/src/loader.rsc.ts +26 -157
  138. package/src/loader.ts +27 -10
  139. package/src/network-error-thrower.tsx +3 -1
  140. package/src/outlet-provider.tsx +45 -0
  141. package/src/prerender/param-hash.ts +37 -0
  142. package/src/prerender/store.ts +185 -0
  143. package/src/prerender.ts +463 -0
  144. package/src/reverse.ts +330 -0
  145. package/src/root-error-boundary.tsx +41 -29
  146. package/src/route-content-wrapper.tsx +7 -4
  147. package/src/route-definition/dsl-helpers.ts +934 -0
  148. package/src/route-definition/helper-factories.ts +200 -0
  149. package/src/route-definition/helpers-types.ts +430 -0
  150. package/src/route-definition/index.ts +52 -0
  151. package/src/route-definition/redirect.ts +93 -0
  152. package/src/route-definition.ts +1 -1428
  153. package/src/route-map-builder.ts +211 -123
  154. package/src/route-name.ts +53 -0
  155. package/src/route-types.ts +59 -8
  156. package/src/router/content-negotiation.ts +116 -0
  157. package/src/router/debug-manifest.ts +72 -0
  158. package/src/router/error-handling.ts +9 -9
  159. package/src/router/find-match.ts +158 -0
  160. package/src/router/handler-context.ts +374 -81
  161. package/src/router/intercept-resolution.ts +395 -0
  162. package/src/router/lazy-includes.ts +234 -0
  163. package/src/router/loader-resolution.ts +215 -122
  164. package/src/router/logging.ts +248 -0
  165. package/src/router/manifest.ts +148 -35
  166. package/src/router/match-api.ts +620 -0
  167. package/src/router/match-context.ts +5 -3
  168. package/src/router/match-handlers.ts +440 -0
  169. package/src/router/match-middleware/background-revalidation.ts +80 -93
  170. package/src/router/match-middleware/cache-lookup.ts +382 -9
  171. package/src/router/match-middleware/cache-store.ts +51 -22
  172. package/src/router/match-middleware/intercept-resolution.ts +55 -17
  173. package/src/router/match-middleware/segment-resolution.ts +24 -6
  174. package/src/router/match-pipelines.ts +10 -45
  175. package/src/router/match-result.ts +34 -28
  176. package/src/router/metrics.ts +235 -15
  177. package/src/router/middleware-cookies.ts +55 -0
  178. package/src/router/middleware-types.ts +222 -0
  179. package/src/router/middleware.ts +324 -367
  180. package/src/router/pattern-matching.ts +211 -43
  181. package/src/router/prerender-match.ts +402 -0
  182. package/src/router/preview-match.ts +170 -0
  183. package/src/router/revalidation.ts +137 -38
  184. package/src/router/router-context.ts +36 -21
  185. package/src/router/router-interfaces.ts +452 -0
  186. package/src/router/router-options.ts +592 -0
  187. package/src/router/router-registry.ts +24 -0
  188. package/src/router/segment-resolution/fresh.ts +570 -0
  189. package/src/router/segment-resolution/helpers.ts +263 -0
  190. package/src/router/segment-resolution/loader-cache.ts +198 -0
  191. package/src/router/segment-resolution/revalidation.ts +1241 -0
  192. package/src/router/segment-resolution/static-store.ts +67 -0
  193. package/src/router/segment-resolution.ts +21 -0
  194. package/src/router/segment-wrappers.ts +289 -0
  195. package/src/router/telemetry-otel.ts +299 -0
  196. package/src/router/telemetry.ts +300 -0
  197. package/src/router/timeout.ts +148 -0
  198. package/src/router/trie-matching.ts +239 -0
  199. package/src/router/types.ts +77 -3
  200. package/src/router.ts +692 -4257
  201. package/src/rsc/handler-context.ts +45 -0
  202. package/src/rsc/handler.ts +764 -754
  203. package/src/rsc/helpers.ts +140 -6
  204. package/src/rsc/index.ts +0 -20
  205. package/src/rsc/loader-fetch.ts +209 -0
  206. package/src/rsc/manifest-init.ts +86 -0
  207. package/src/rsc/nonce.ts +14 -0
  208. package/src/rsc/origin-guard.ts +141 -0
  209. package/src/rsc/progressive-enhancement.ts +379 -0
  210. package/src/rsc/response-error.ts +37 -0
  211. package/src/rsc/response-route-handler.ts +347 -0
  212. package/src/rsc/rsc-rendering.ts +235 -0
  213. package/src/rsc/runtime-warnings.ts +42 -0
  214. package/src/rsc/server-action.ts +348 -0
  215. package/src/rsc/ssr-setup.ts +128 -0
  216. package/src/rsc/types.ts +38 -11
  217. package/src/search-params.ts +230 -0
  218. package/src/segment-system.tsx +25 -13
  219. package/src/server/context.ts +182 -51
  220. package/src/server/cookie-store.ts +190 -0
  221. package/src/server/fetchable-loader-store.ts +37 -0
  222. package/src/server/handle-store.ts +94 -15
  223. package/src/server/loader-registry.ts +15 -56
  224. package/src/server/request-context.ts +430 -70
  225. package/src/server.ts +35 -130
  226. package/src/ssr/index.tsx +100 -31
  227. package/src/static-handler.ts +114 -0
  228. package/src/theme/ThemeProvider.tsx +21 -15
  229. package/src/theme/ThemeScript.tsx +5 -5
  230. package/src/theme/constants.ts +5 -2
  231. package/src/theme/index.ts +4 -14
  232. package/src/theme/theme-context.ts +4 -30
  233. package/src/theme/theme-script.ts +21 -18
  234. package/src/types/boundaries.ts +158 -0
  235. package/src/types/cache-types.ts +198 -0
  236. package/src/types/error-types.ts +192 -0
  237. package/src/types/global-namespace.ts +100 -0
  238. package/src/types/handler-context.ts +687 -0
  239. package/src/types/index.ts +88 -0
  240. package/src/types/loader-types.ts +183 -0
  241. package/src/types/route-config.ts +170 -0
  242. package/src/types/route-entry.ts +102 -0
  243. package/src/types/segments.ts +148 -0
  244. package/src/types.ts +1 -1623
  245. package/src/urls/include-helper.ts +197 -0
  246. package/src/urls/index.ts +53 -0
  247. package/src/urls/path-helper-types.ts +339 -0
  248. package/src/urls/path-helper.ts +329 -0
  249. package/src/urls/pattern-types.ts +95 -0
  250. package/src/urls/response-types.ts +106 -0
  251. package/src/urls/type-extraction.ts +372 -0
  252. package/src/urls/urls-function.ts +98 -0
  253. package/src/urls.ts +1 -802
  254. package/src/use-loader.tsx +85 -77
  255. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  256. package/src/vite/discovery/discover-routers.ts +344 -0
  257. package/src/vite/discovery/prerender-collection.ts +385 -0
  258. package/src/vite/discovery/route-types-writer.ts +258 -0
  259. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  260. package/src/vite/discovery/state.ts +110 -0
  261. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  262. package/src/vite/index.ts +11 -1133
  263. package/src/vite/plugin-types.ts +131 -0
  264. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  265. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  266. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  267. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
  268. package/src/vite/plugins/expose-id-utils.ts +287 -0
  269. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  270. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  271. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  272. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  273. package/src/vite/plugins/expose-ids/types.ts +45 -0
  274. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  275. package/src/vite/plugins/refresh-cmd.ts +65 -0
  276. package/src/vite/plugins/use-cache-transform.ts +323 -0
  277. package/src/vite/plugins/version-injector.ts +83 -0
  278. package/src/vite/plugins/version-plugin.ts +254 -0
  279. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  280. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  281. package/src/vite/rango.ts +510 -0
  282. package/src/vite/router-discovery.ts +785 -0
  283. package/src/vite/utils/ast-handler-extract.ts +517 -0
  284. package/src/vite/utils/banner.ts +36 -0
  285. package/src/vite/utils/bundle-analysis.ts +137 -0
  286. package/src/vite/utils/manifest-utils.ts +70 -0
  287. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  288. package/src/vite/utils/prerender-utils.ts +189 -0
  289. package/src/vite/utils/shared-utils.ts +169 -0
  290. package/CLAUDE.md +0 -43
  291. package/src/browser/lru-cache.ts +0 -69
  292. package/src/browser/request-controller.ts +0 -164
  293. package/src/cache/memory-store.ts +0 -253
  294. package/src/href-context.ts +0 -33
  295. package/src/href.ts +0 -255
  296. package/src/server/route-manifest-cache.ts +0 -173
  297. package/src/vite/expose-handle-id.ts +0 -209
  298. package/src/vite/expose-loader-id.ts +0 -426
  299. package/src/vite/expose-location-state-id.ts +0 -177
  300. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -1,442 +1,50 @@
1
1
  import type {
2
2
  NavigationBridge,
3
3
  NavigationBridgeConfig,
4
- NavigateOptions,
5
- NavigationStore,
4
+ NavigateOptionsInternal,
6
5
  ResolvedSegment,
7
6
  } from "./types.js";
7
+ import * as React from "react";
8
+ import { startTransition } from "react";
8
9
  import {
9
- isLocationStateEntry,
10
- resolveLocationStateEntries,
11
- } from "./react/location-state-shared.js";
12
-
13
- /**
14
- * Check if state is from typed LocationStateEntry[] (has __rsc_ls_ keys)
15
- */
16
- function isTypedLocationState(
17
- state: unknown
18
- ): state is Record<string, unknown> {
19
- if (state === null || typeof state !== "object") return false;
20
- return Object.keys(state).some((key) => key.startsWith("__rsc_ls_"));
21
- }
10
+ createNavigationTransaction,
11
+ resolveNavigationState,
12
+ } from "./navigation-transaction.js";
13
+ import { buildHistoryState } from "./history-state.js";
14
+ import {
15
+ handleNavigationStart,
16
+ handleNavigationEnd,
17
+ ensureHistoryKey,
18
+ } from "./scroll-restoration.js";
22
19
 
23
- /**
24
- * Resolve navigation state - handles both LocationStateEntry[] and legacy formats
25
- */
26
- function resolveNavigationState(state: unknown): unknown {
27
- // Check if it's an array of LocationStateEntry
28
- if (
29
- Array.isArray(state) &&
30
- state.length > 0 &&
31
- isLocationStateEntry(state[0])
32
- ) {
33
- return resolveLocationStateEntries(state);
34
- }
35
- // Return as-is for legacy formats
36
- return state;
37
- }
20
+ // addTransitionType is only available in React experimental
21
+ const addTransitionType: ((type: string) => void) | undefined =
22
+ "addTransitionType" in React ? (React as any).addTransitionType : undefined;
38
23
 
39
- /**
40
- * Build history state object from user state
41
- * - Typed state: spread directly into history.state
42
- * - Legacy state: store in history.state.state
43
- */
44
- function buildHistoryState(
45
- userState: unknown,
46
- routerState?: { intercept?: boolean; sourceUrl?: string }
47
- ): Record<string, unknown> | null {
48
- const result: Record<string, unknown> = {};
49
-
50
- // Add router internal state
51
- if (routerState?.intercept) {
52
- result.intercept = true;
53
- if (routerState.sourceUrl) {
54
- result.sourceUrl = routerState.sourceUrl;
55
- }
56
- }
57
-
58
- // Add user state
59
- if (userState !== undefined) {
60
- if (isTypedLocationState(userState)) {
61
- // Typed state: spread directly
62
- Object.assign(result, userState);
63
- } else {
64
- // Legacy state: store in .state
65
- result.state = userState;
66
- }
67
- }
68
-
69
- return Object.keys(result).length > 0 ? result : null;
70
- }
71
24
  import { setupLinkInterception } from "./link-interceptor.js";
72
25
  import { createPartialUpdater } from "./partial-update.js";
73
26
  import { generateHistoryKey } from "./navigation-store.js";
27
+ import type { EventController } from "./event-controller.js";
28
+ import { isInterceptOnlyCache } from "./intercept-utils.js";
74
29
  import {
75
- handleNavigationStart,
76
- handleNavigationEnd,
77
- ensureHistoryKey,
78
- } from "./scroll-restoration.js";
79
- import type { EventController, NavigationHandle } from "./event-controller.js";
80
- import { NetworkError, isNetworkError } from "../errors.js";
81
- import { NetworkErrorThrower } from "../network-error-thrower.js";
82
- import { createElement, startTransition } from "react";
30
+ toNetworkError,
31
+ emitNetworkError,
32
+ isBackgroundSuppressible,
33
+ } from "./network-error-handler.js";
34
+ import { debugLog } from "./logging.js";
35
+ import { ServerRedirect } from "../errors.js";
36
+ import { validateRedirectOrigin } from "./validate-redirect-origin.js";
83
37
 
84
38
  // Polyfill Symbol.dispose for Safari and older browsers
85
39
  if (typeof Symbol.dispose === "undefined") {
86
40
  (Symbol as any).dispose = Symbol("Symbol.dispose");
87
41
  }
88
42
 
89
- /**
90
- * Check if a segment is an intercept segment
91
- * Intercept segments have namespace starting with "intercept:" or ID containing .@
92
- */
93
- function isInterceptSegment(s: ResolvedSegment): boolean {
94
- return (
95
- s.namespace?.startsWith("intercept:") ||
96
- (s.type === "parallel" && s.id.includes(".@"))
97
- );
98
- }
99
-
100
- /**
101
- * Check if cached segments are intercept-only (no main route segments)
102
- * Intercept responses shouldn't be used for optimistic rendering since
103
- * whether interception happens depends on the current page context
104
- */
105
- function isInterceptOnlyCache(segments: ResolvedSegment[]): boolean {
106
- return segments.some(isInterceptSegment);
107
- }
108
-
109
- /**
110
- * Options for committing a navigation transaction
111
- */
112
- interface CommitOptions {
113
- url: string;
114
- segmentIds: string[];
115
- segments: ResolvedSegment[];
116
- replace?: boolean;
117
- scroll?: boolean;
118
- /** User-provided state to store in history.state */
119
- state?: unknown;
120
- /** If true, only update store without changing URL/history (for server actions) */
121
- storeOnly?: boolean;
122
- /** If true, this is an intercept route - store in history.state for popstate handling */
123
- intercept?: boolean;
124
- /** Source URL where the intercept was triggered from (stored in history.state) */
125
- interceptSourceUrl?: string;
126
- /** If true, only update cache without touching store or history (for background stale revalidation) */
127
- cacheOnly?: boolean;
128
- }
129
-
130
- /**
131
- * Options that can override the pre-configured commit settings
132
- */
133
- interface BoundCommitOverrides {
134
- /** Override scroll behavior (e.g., disable for intercepts) */
135
- scroll?: boolean;
136
- /** Override replace behavior (e.g., force replace for intercepts) */
137
- replace?: boolean;
138
- /** Override user-provided state */
139
- state?: unknown;
140
- /** Mark this as an intercept route */
141
- intercept?: boolean;
142
- /** Source URL where intercept was triggered from */
143
- interceptSourceUrl?: string;
144
- /** If true, only update cache (for stale revalidation) */
145
- cacheOnly?: boolean;
146
- }
147
-
148
- /**
149
- * Token for tracking an active stream - call end() when stream completes
150
- */
151
- export interface StreamingToken {
152
- end(): void;
153
- }
154
-
155
- /**
156
- * Bound transaction with pre-configured commit options (without segmentIds/segments)
157
- */
158
- export interface BoundTransaction {
159
- readonly currentUrl: string;
160
- /** Start streaming and get a token to end it when the stream completes */
161
- startStreaming(): StreamingToken;
162
- commit(
163
- segmentIds: string[],
164
- segments: ResolvedSegment[],
165
- overrides?: BoundCommitOverrides
166
- ): void;
167
- }
168
-
169
- /**
170
- * Navigation transaction for managing state during navigation
171
- * Uses the event controller handle for lifecycle management
172
- */
173
- interface NavigationTransaction extends Disposable {
174
- /** Optimistically commit from cache - instant render before revalidation */
175
- optimisticCommit(options: CommitOptions): void;
176
- /** Final commit with server data (or reconciliation after optimistic) */
177
- commit(options: CommitOptions): void;
178
- with(
179
- options: Omit<CommitOptions, "segmentIds" | "segments">
180
- ): BoundTransaction;
181
- /** The navigation handle from the event controller */
182
- handle: NavigationHandle;
43
+ /** Get IDs of non-loader segments (layouts, routes, parallels). */
44
+ function getNonLoaderSegmentIds(segments: ResolvedSegment[]): string[] {
45
+ return segments.filter((s) => s.type !== "loader").map((s) => s.id);
183
46
  }
184
47
 
185
- /**
186
- * Creates a navigation transaction that coordinates with the event controller.
187
- * Handles loading state transitions and cleanup on completion/abort.
188
- *
189
- * Supports optimistic navigation: render from cache immediately,
190
- * then revalidate in background and reconcile if data changed.
191
- */
192
- function createNavigationTransaction(
193
- store: NavigationStore,
194
- eventController: EventController,
195
- url: string,
196
- options?: NavigateOptions & { skipLoadingState?: boolean }
197
- ): NavigationTransaction {
198
- let committed = false;
199
- let optimisticallyCommitted = false;
200
- let earlyStatePushed = false;
201
- const currentUrl = window.location.href;
202
-
203
- // Start navigation in event controller (this sets loading state)
204
- const handle = eventController.startNavigation(url, options);
205
-
206
- // If state is provided, push it to history immediately so loading UI can access it
207
- // This enables "optimistic state" - showing product names in skeletons etc.
208
- if (options?.state !== undefined && !options?.replace) {
209
- const earlyHistoryState = buildHistoryState(options.state);
210
- window.history.pushState(earlyHistoryState, "", url);
211
- earlyStatePushed = true;
212
- }
213
-
214
- /**
215
- * Optimistically commit from cache - renders immediately before revalidation
216
- * Sets optimisticallyCommitted flag so final commit() knows to reconcile
217
- */
218
- function optimisticCommit(opts: CommitOptions): void {
219
- optimisticallyCommitted = true;
220
-
221
- const { url, segmentIds, segments, replace, scroll } = opts;
222
- const parsedUrl = new URL(url, window.location.origin);
223
-
224
- // Save current scroll position before navigating
225
- handleNavigationStart();
226
-
227
- // Update segment state
228
- store.setSegmentIds(segmentIds);
229
- store.setCurrentUrl(url);
230
- store.setPath(parsedUrl.pathname);
231
-
232
- // Generate history key from URL
233
- const historyKey = generateHistoryKey(url);
234
- store.setHistoryKey(historyKey);
235
-
236
- // Cache segments with current handleData (will be overwritten by fresh data on final commit)
237
- const currentHandleData = eventController.getHandleState().data;
238
- store.cacheSegmentsForHistory(historyKey, segments, currentHandleData);
239
-
240
- // Build history state with user state if provided
241
- const historyState = buildHistoryState(opts.state);
242
-
243
- // Update browser URL
244
- // Use replaceState if we already pushed early (for optimistic state access)
245
- if (replace || earlyStatePushed) {
246
- window.history.replaceState(historyState, "", url);
247
- } else {
248
- window.history.pushState(historyState, "", url);
249
- }
250
-
251
- // Ensure new history entry has a scroll restoration key
252
- ensureHistoryKey();
253
-
254
- // Complete the navigation in event controller (sets idle state)
255
- handle.complete(parsedUrl);
256
-
257
- // Handle scroll after navigation
258
- handleNavigationEnd({ scroll });
259
-
260
- console.log(
261
- "[Browser] Optimistic commit from cache, historyKey:",
262
- historyKey
263
- );
264
- }
265
-
266
- /**
267
- * Commit the navigation - updates store and URL atomically
268
- * If optimisticCommit was called, this becomes a reconciliation
269
- */
270
- function commit(opts: CommitOptions): void {
271
- committed = true;
272
-
273
- // If optimistic commit already done, adjust options for reconciliation
274
- const isReconciliation = optimisticallyCommitted;
275
- const {
276
- url,
277
- segmentIds,
278
- segments,
279
- storeOnly,
280
- intercept,
281
- interceptSourceUrl,
282
- cacheOnly,
283
- } = opts;
284
- // For reconciliation: always replace (URL already pushed), no scroll
285
- const replace = isReconciliation ? true : opts.replace;
286
- const scroll = isReconciliation ? false : opts.scroll;
287
-
288
- const parsedUrl = new URL(url, window.location.origin);
289
-
290
- // Generate history key from URL (with intercept suffix for separate caching)
291
- const historyKey = generateHistoryKey(url, { intercept });
292
-
293
- // For cache-only commits (stale revalidation), only update cache and return
294
- // Don't touch store state or history - user may have navigated elsewhere
295
- if (cacheOnly) {
296
- const currentHandleData = eventController.getHandleState().data;
297
- store.cacheSegmentsForHistory(historyKey, segments, currentHandleData);
298
- console.log("[Browser] Cache-only commit, historyKey:", historyKey);
299
- return;
300
- }
301
-
302
- // Save current scroll position before navigating (only for non-reconciliation)
303
- if (!isReconciliation) {
304
- handleNavigationStart();
305
- }
306
-
307
- // Update segment state atomically
308
- store.setSegmentIds(segmentIds);
309
- store.setCurrentUrl(url);
310
- store.setPath(parsedUrl.pathname);
311
-
312
- store.setHistoryKey(historyKey);
313
-
314
- // Cache segments with current handleData for this history entry (fresh data overwrites optimistic)
315
- const currentHandleData = eventController.getHandleState().data;
316
- store.cacheSegmentsForHistory(historyKey, segments, currentHandleData);
317
-
318
- // For server actions, skip URL/history updates but still complete navigation
319
- if (storeOnly) {
320
- console.log("[Browser] Store updated (action)");
321
- // Complete navigation to clear loading state
322
- handle.complete(parsedUrl);
323
- return;
324
- }
325
-
326
- // Build history state - include user state and intercept info for popstate handling
327
- const historyState = buildHistoryState(opts.state, {
328
- intercept,
329
- sourceUrl: interceptSourceUrl,
330
- });
331
-
332
- // Update browser URL (skip if reconciliation - already done in optimisticCommit)
333
- if (!isReconciliation) {
334
- // Use replaceState if we already pushed early (for optimistic state access) or replace requested
335
- if (replace || earlyStatePushed) {
336
- window.history.replaceState(historyState, "", url);
337
- } else {
338
- window.history.pushState(historyState, "", url);
339
- }
340
- // Ensure new history entry has a scroll restoration key
341
- ensureHistoryKey();
342
- }
343
-
344
- // Complete the navigation in event controller (sets idle state, updates location)
345
- handle.complete(parsedUrl);
346
-
347
- // Handle scroll after navigation (skip if reconciliation)
348
- if (!isReconciliation) {
349
- handleNavigationEnd({ scroll });
350
- }
351
-
352
- if (isReconciliation) {
353
- console.log("[Browser] Reconciliation commit, historyKey:", historyKey);
354
- } else {
355
- console.log(
356
- "[Browser] Navigation committed, historyKey:",
357
- historyKey,
358
- intercept ? "(intercept)" : ""
359
- );
360
- }
361
- }
362
-
363
- return {
364
- handle,
365
- optimisticCommit,
366
- commit,
367
-
368
- /**
369
- * Create a bound transaction with pre-configured URL options
370
- * segmentIds and segments provided at commit time (after they're resolved)
371
- */
372
- with(
373
- opts: Omit<CommitOptions, "segmentIds" | "segments">
374
- ): BoundTransaction {
375
- return {
376
- get currentUrl() {
377
- return currentUrl;
378
- },
379
- startStreaming() {
380
- return handle.startStreaming();
381
- },
382
- commit: (
383
- segmentIds: string[],
384
- segments: ResolvedSegment[],
385
- overrides?: BoundCommitOverrides
386
- ) => {
387
- // Allow overrides to disable scroll (e.g., for intercepts)
388
- const finalScroll =
389
- overrides?.scroll !== undefined ? overrides.scroll : opts.scroll;
390
- // Allow overrides to force replace (e.g., for intercepts)
391
- const finalReplace =
392
- overrides?.replace !== undefined ? overrides.replace : opts.replace;
393
- // Intercept info: overrides take precedence, fallback to opts
394
- const intercept =
395
- overrides?.intercept !== undefined
396
- ? overrides.intercept
397
- : opts.intercept;
398
- const interceptSourceUrl =
399
- overrides?.interceptSourceUrl !== undefined
400
- ? overrides.interceptSourceUrl
401
- : opts.interceptSourceUrl;
402
- // Cache-only mode: overrides take precedence, fallback to opts
403
- const cacheOnly =
404
- overrides?.cacheOnly !== undefined
405
- ? overrides.cacheOnly
406
- : opts.cacheOnly;
407
- // User state: overrides take precedence, fallback to opts
408
- const state =
409
- overrides?.state !== undefined ? overrides.state : opts.state;
410
- commit({
411
- ...opts,
412
- segmentIds,
413
- segments,
414
- scroll: finalScroll,
415
- replace: finalReplace,
416
- state,
417
- intercept,
418
- interceptSourceUrl,
419
- cacheOnly,
420
- });
421
- },
422
- };
423
- },
424
-
425
- [Symbol.dispose]() {
426
- // If aborted, another navigation took over - don't touch state
427
- if (handle.signal.aborted) return;
428
-
429
- // If not committed (and not optimistically committed), the handle's dispose
430
- // will reset state to idle via the event controller
431
- if (!committed && !optimisticallyCommitted) {
432
- handle[Symbol.dispose]();
433
- // The NavigationHandle's [Symbol.dispose] handles this
434
- }
435
- },
436
- };
437
- }
438
-
439
- // Export for use by server-action-bridge
440
48
  export { createNavigationTransaction };
441
49
 
442
50
  /**
@@ -462,9 +70,10 @@ export interface NavigationBridgeConfigWithController extends NavigationBridgeCo
462
70
  * @returns NavigationBridge instance
463
71
  */
464
72
  export function createNavigationBridge(
465
- config: NavigationBridgeConfigWithController
73
+ config: NavigationBridgeConfigWithController,
466
74
  ): NavigationBridge {
467
- const { store, client, eventController, onUpdate, renderSegments, version } = config;
75
+ const { store, client, eventController, onUpdate, renderSegments, version } =
76
+ config;
468
77
 
469
78
  // Create shared partial updater
470
79
  const fetchPartialUpdate = createPartialUpdater({
@@ -478,19 +87,121 @@ export function createNavigationBridge(
478
87
  return {
479
88
  /**
480
89
  * Navigate to a URL
481
- * Uses optimistic rendering from cache when available (SWR pattern)
90
+ * Uses cached segments for SWR revalidation when available
482
91
  */
483
- async navigate(url: string, options?: NavigateOptions): Promise<void> {
92
+ async navigate(
93
+ url: string,
94
+ options?: NavigateOptionsInternal,
95
+ ): Promise<void> {
484
96
  // Resolve LocationStateEntry[] to flat object if needed
485
97
  const resolvedState =
486
98
  options?.state !== undefined
487
99
  ? resolveNavigationState(options.state)
488
100
  : undefined;
489
101
 
102
+ // Cross-origin URLs are not handled by SPA navigation.
103
+ // Fall back to a full browser navigation for http/https only.
104
+ let targetUrl: URL;
105
+ try {
106
+ targetUrl = new URL(url, window.location.origin);
107
+ } catch {
108
+ console.warn(`[rango] navigate() ignored: malformed URL "${url}"`);
109
+ return;
110
+ }
111
+ if (targetUrl.origin !== window.location.origin) {
112
+ if (targetUrl.protocol !== "http:" && targetUrl.protocol !== "https:") {
113
+ console.error(
114
+ `[rango] navigate() blocked: unsupported scheme "${targetUrl.protocol}"`,
115
+ );
116
+ return;
117
+ }
118
+ window.location.href = targetUrl.href;
119
+ return;
120
+ }
121
+
122
+ // Shallow navigation: skip RSC fetch when revalidate is false
123
+ // and the pathname hasn't changed (search param / hash only change).
124
+ if (
125
+ options?.revalidate === false &&
126
+ targetUrl.pathname === new URL(window.location.href).pathname
127
+ ) {
128
+ // Preserve intercept context from the current history entry so that
129
+ // popstate uses the correct cache key (:intercept suffix) and restores
130
+ // the right full-page vs modal semantics.
131
+ const currentHistoryState = window.history.state;
132
+ const isIntercept = currentHistoryState?.intercept === true;
133
+ const interceptSourceUrl = isIntercept
134
+ ? currentHistoryState?.sourceUrl
135
+ : undefined;
136
+
137
+ const historyKey = generateHistoryKey(url, { intercept: isIntercept });
138
+
139
+ // Copy current segments to the new history key so back/forward restores instantly
140
+ const currentKey = store.getHistoryKey();
141
+ const currentCache = store.getCachedSegments(currentKey);
142
+ if (currentCache?.segments) {
143
+ const currentHandleData = eventController.getHandleState().data;
144
+ store.cacheSegmentsForHistory(
145
+ historyKey,
146
+ currentCache.segments,
147
+ currentHandleData,
148
+ );
149
+ }
150
+
151
+ // Save current scroll position before changing URL
152
+ handleNavigationStart();
153
+
154
+ // Snapshot old state before pushState/replaceState overwrites it
155
+ const oldState = window.history.state;
156
+
157
+ // Update browser URL (carry intercept context into history state)
158
+ const historyState = buildHistoryState(
159
+ resolvedState,
160
+ {
161
+ intercept: isIntercept || undefined,
162
+ sourceUrl: interceptSourceUrl,
163
+ },
164
+ {},
165
+ );
166
+ if (options.replace) {
167
+ window.history.replaceState(historyState, "", url);
168
+ } else {
169
+ window.history.pushState(historyState, "", url);
170
+ }
171
+
172
+ // Ensure new history entry has a scroll restoration key
173
+ ensureHistoryKey();
174
+
175
+ // Notify useLocationState() hooks when state changes
176
+ const hasOldState =
177
+ oldState &&
178
+ typeof oldState === "object" &&
179
+ ("state" in oldState ||
180
+ Object.keys(oldState).some((k) => k.startsWith("__rsc_ls_")));
181
+ const hasNewState =
182
+ historyState &&
183
+ ("state" in historyState ||
184
+ Object.keys(historyState).some((k) => k.startsWith("__rsc_ls_")));
185
+ if (hasOldState || hasNewState) {
186
+ window.dispatchEvent(new Event("__rsc_locationstate"));
187
+ }
188
+
189
+ // Update store history key so future navigations reference the right cache
190
+ store.setHistoryKey(historyKey);
191
+ store.setCurrentUrl(url);
192
+
193
+ // Notify hooks — location updates, state stays idle
194
+ eventController.setLocation(targetUrl);
195
+
196
+ // Handle post-navigation scroll
197
+ handleNavigationEnd({ scroll: options.scroll });
198
+ return;
199
+ }
200
+
490
201
  // Only abort pending requests when navigating to a different route
491
202
  // Same-route navigation (e.g., /todos -> /todos) should not cancel in-flight actions
492
203
  const currentPath = new URL(window.location.href).pathname;
493
- const targetPath = new URL(url, window.location.origin).pathname;
204
+ const targetPath = targetUrl.pathname;
494
205
  if (currentPath !== targetPath) {
495
206
  eventController.abortNavigation();
496
207
  }
@@ -503,7 +214,9 @@ export function createNavigationBridge(
503
214
  const isLeavingIntercept = isCurrentlyIntercept && isSamePathNavigation;
504
215
 
505
216
  if (isLeavingIntercept) {
506
- console.log(`[Browser] Leaving intercept - same URL navigation from intercept`);
217
+ debugLog(
218
+ "[Browser] Leaving intercept - same URL navigation from intercept",
219
+ );
507
220
  // Clear intercept source URL to ensure server doesn't treat this as intercept
508
221
  store.setInterceptSourceUrl(null);
509
222
  }
@@ -517,7 +230,7 @@ export function createNavigationBridge(
517
230
  store.cacheSegmentsForHistory(
518
231
  sourceHistoryKey,
519
232
  sourceCached.segments,
520
- currentHandleData
233
+ currentHandleData,
521
234
  );
522
235
  }
523
236
 
@@ -532,7 +245,7 @@ export function createNavigationBridge(
532
245
  const cachedHandleData = cached?.handleData;
533
246
  if (cachedSegments && sourceCached?.segments) {
534
247
  const sourceSegmentMap = new Map(
535
- sourceCached.segments.map((s) => [s.id, s])
248
+ sourceCached.segments.map((s) => [s.id, s]),
536
249
  );
537
250
  cachedSegments = cachedSegments.map((targetSeg) => {
538
251
  const sourceSeg = sourceSegmentMap.get(targetSeg.id);
@@ -547,18 +260,20 @@ export function createNavigationBridge(
547
260
  const interceptHistoryKey = generateHistoryKey(url, { intercept: true });
548
261
  const hasInterceptCache = store.hasHistoryCache(interceptHistoryKey);
549
262
 
550
- // Skip optimistic rendering for:
263
+ // Skip cached SWR for:
551
264
  // 1. intercept caches - interception depends on source page context
552
265
  // 2. routes that CAN be intercepted - we don't know if this navigation will intercept
553
266
  // 3. when leaving intercept - we need fresh non-intercept segments from server
267
+ // 4. redirect-with-state - force re-render so hooks read fresh state
554
268
  const hasUsableCache =
555
269
  cachedSegments &&
556
270
  cachedSegments.length > 0 &&
557
271
  !isInterceptOnlyCache(cachedSegments) &&
558
272
  !hasInterceptCache &&
559
- !isLeavingIntercept;
273
+ !isLeavingIntercept &&
274
+ !options?._skipCache;
560
275
 
561
- using tx = createNavigationTransaction(store, eventController, url, {
276
+ const tx = createNavigationTransaction(store, eventController, url, {
562
277
  ...options,
563
278
  state: resolvedState,
564
279
  skipLoadingState: hasUsableCache,
@@ -568,7 +283,11 @@ export function createNavigationBridge(
568
283
  try {
569
284
  await fetchPartialUpdate(
570
285
  url,
571
- hasUsableCache ? cachedSegments!.map((s) => s.id) : undefined,
286
+ hasUsableCache
287
+ ? getNonLoaderSegmentIds(cachedSegments!)
288
+ : options?._skipCache
289
+ ? [] // Action redirect: send no segments so server renders everything fresh
290
+ : undefined,
572
291
  false,
573
292
  tx.handle.signal,
574
293
  tx.with({
@@ -577,55 +296,56 @@ export function createNavigationBridge(
577
296
  scroll: options?.scroll,
578
297
  state: resolvedState,
579
298
  }),
580
- // Pass cached segments (merged with current page's fresh segments for shared IDs)
581
- // so the segment map is consistent with what we tell the server we have.
582
- // Server decides what needs revalidation based on route matching and custom functions.
583
- // No need for staleRevalidation flag - we're sending the freshest segments we have.
584
- // Also pass cached handle data for restoring breadcrumbs when server returns empty diff.
585
- // When leaving intercept, pass the flag so fetchPartialUpdate knows to filter segments.
586
299
  hasUsableCache
587
- ? { targetCacheSegments: cachedSegments, targetCacheHandleData: cachedHandleData }
300
+ ? {
301
+ type: "navigate" as const,
302
+ targetCacheSegments: cachedSegments,
303
+ targetCacheHandleData: cachedHandleData,
304
+ }
588
305
  : isLeavingIntercept
589
- ? { leavingIntercept: true }
590
- : undefined
306
+ ? { type: "leave-intercept" as const }
307
+ : undefined,
591
308
  );
592
309
  } catch (error) {
593
- // Ignore AbortError - navigation was cancelled by a newer navigation
310
+ // Server-side redirect with location state: the current transaction's
311
+ // cleanup resets loading state. Re-navigate to the redirect
312
+ // target carrying the server-set state into history.pushState.
313
+ if (error instanceof ServerRedirect) {
314
+ const redirectUrl = validateRedirectOrigin(
315
+ error.url,
316
+ window.location.origin,
317
+ );
318
+ if (!redirectUrl) {
319
+ return;
320
+ }
321
+ return this.navigate(redirectUrl, {
322
+ state: error.state,
323
+ replace: options?.replace,
324
+ _skipCache: true,
325
+ } as NavigateOptionsInternal);
326
+ }
327
+
594
328
  if (error instanceof DOMException && error.name === "AbortError") {
595
- console.log("[Browser] Navigation aborted by newer navigation");
329
+ debugLog("[Browser] Navigation aborted by newer navigation");
596
330
  return;
597
331
  }
598
332
 
599
- // Handle network errors by triggering root error boundary
600
- if (error instanceof NetworkError || isNetworkError(error)) {
601
- const networkError =
602
- error instanceof NetworkError
603
- ? error
604
- : new NetworkError(
605
- "Unable to connect to server. Please check your connection.",
606
- { cause: error, url, operation: "navigation" }
607
- );
608
-
333
+ const networkError = toNetworkError(error, {
334
+ url,
335
+ operation: "navigation",
336
+ });
337
+ if (networkError) {
609
338
  console.error(
610
339
  "[Browser] Network error during navigation:",
611
- networkError
340
+ networkError,
612
341
  );
613
-
614
- // Emit update with NetworkErrorThrower to trigger root error boundary
615
- startTransition(() => {
616
- onUpdate({
617
- root: createElement(NetworkErrorThrower, { error: networkError }),
618
- metadata: {
619
- pathname: url,
620
- segments: [],
621
- isError: true,
622
- },
623
- });
624
- });
342
+ emitNetworkError(onUpdate, networkError, url);
625
343
  return;
626
344
  }
627
345
 
628
346
  throw error;
347
+ } finally {
348
+ tx[Symbol.dispose]();
629
349
  }
630
350
  },
631
351
 
@@ -635,11 +355,11 @@ export function createNavigationBridge(
635
355
  async refresh(): Promise<void> {
636
356
  eventController.abortNavigation();
637
357
 
638
- using tx = createNavigationTransaction(
358
+ const tx = createNavigationTransaction(
639
359
  store,
640
360
  eventController,
641
361
  window.location.href,
642
- { replace: true }
362
+ { replace: true },
643
363
  );
644
364
 
645
365
  try {
@@ -649,41 +369,24 @@ export function createNavigationBridge(
649
369
  [],
650
370
  false,
651
371
  tx.handle.signal,
652
- tx.with({ url: window.location.href, replace: true, scroll: false })
372
+ tx.with({ url: window.location.href, replace: true, scroll: false }),
653
373
  );
654
374
  } catch (error) {
655
- // Handle network errors by triggering root error boundary
656
- if (error instanceof NetworkError || isNetworkError(error)) {
657
- const networkError =
658
- error instanceof NetworkError
659
- ? error
660
- : new NetworkError(
661
- "Unable to connect to server. Please check your connection.",
662
- {
663
- cause: error,
664
- url: window.location.href,
665
- operation: "revalidation",
666
- }
667
- );
668
-
375
+ const networkError = toNetworkError(error, {
376
+ url: window.location.href,
377
+ operation: "revalidation",
378
+ });
379
+ if (networkError) {
669
380
  console.error(
670
381
  "[Browser] Network error during refresh:",
671
- networkError
382
+ networkError,
672
383
  );
673
-
674
- startTransition(() => {
675
- onUpdate({
676
- root: createElement(NetworkErrorThrower, { error: networkError }),
677
- metadata: {
678
- pathname: window.location.href,
679
- segments: [],
680
- isError: true,
681
- },
682
- });
683
- });
384
+ emitNetworkError(onUpdate, networkError, window.location.href);
684
385
  return;
685
386
  }
686
387
  throw error;
388
+ } finally {
389
+ tx[Symbol.dispose]();
687
390
  }
688
391
  },
689
392
 
@@ -707,8 +410,8 @@ export function createNavigationBridge(
707
410
  const currentInterceptSource = store.getInterceptSourceUrl();
708
411
  const newInterceptSource = interceptSourceUrl ?? null;
709
412
  if (currentInterceptSource !== newInterceptSource) {
710
- console.log(
711
- `[Browser] Intercept context changing (${currentInterceptSource} -> ${newInterceptSource}), aborting in-flight actions`
413
+ debugLog(
414
+ `[Browser] Intercept context changing (${currentInterceptSource} -> ${newInterceptSource}), aborting in-flight actions`,
712
415
  );
713
416
  eventController.abortAllActions();
714
417
  }
@@ -716,11 +419,11 @@ export function createNavigationBridge(
716
419
  // Compute history key from URL (with intercept suffix if applicable)
717
420
  const historyKey = generateHistoryKey(url, { intercept: isIntercept });
718
421
 
719
- console.log(
422
+ debugLog(
720
423
  "[Browser] Popstate -",
721
424
  isIntercept ? "intercept" : "normal",
722
425
  "key:",
723
- historyKey
426
+ historyKey,
724
427
  );
725
428
 
726
429
  // Update location in event controller
@@ -751,10 +454,19 @@ export function createNavigationBridge(
751
454
 
752
455
  // Render from cache - force await to skip loading fallbacks
753
456
  try {
754
- const root = renderSegments(cachedSegments, {
457
+ const root = await renderSegments(cachedSegments, {
755
458
  forceAwait: true,
756
459
  });
757
- onUpdate({
460
+ // Merge params from cached segments for useParams restoration.
461
+ // Set params on event controller before onUpdate so both location
462
+ // and params are current when the debounced notify() fires.
463
+ const cachedParams: Record<string, string> = {};
464
+ for (const s of cachedSegments) {
465
+ if (s.params) Object.assign(cachedParams, s.params);
466
+ }
467
+ eventController.setParams(cachedParams);
468
+
469
+ const popstateUpdate = {
758
470
  root,
759
471
  metadata: {
760
472
  pathname: new URL(url).pathname,
@@ -763,23 +475,35 @@ export function createNavigationBridge(
763
475
  matched: cachedSegments.map((s) => s.id),
764
476
  diff: [],
765
477
  cachedHandleData,
478
+ params: cachedParams,
766
479
  },
767
- });
480
+ };
481
+ const hasTransition = cachedSegments.some((s) => s.transition);
482
+ if (hasTransition) {
483
+ startTransition(() => {
484
+ if (addTransitionType) {
485
+ addTransitionType("navigation-back");
486
+ }
487
+ onUpdate(popstateUpdate);
488
+ });
489
+ } else {
490
+ onUpdate(popstateUpdate);
491
+ }
768
492
 
769
493
  // Restore scroll position for back/forward navigation
770
494
  handleNavigationEnd({ restore: true, isStreaming });
771
495
 
772
496
  // SWR: If stale, trigger background revalidation
773
497
  if (isStale) {
774
- console.log("[Browser] Cache is stale, background revalidating...");
498
+ debugLog("[Browser] Cache is stale, background revalidating...");
775
499
  // Background revalidation - don't await, just fire and forget
776
- const segmentIds = cachedSegments.map((s) => s.id);
500
+ const segmentIds = getNonLoaderSegmentIds(cachedSegments);
777
501
 
778
- using tx = createNavigationTransaction(
502
+ const tx = createNavigationTransaction(
779
503
  store,
780
504
  eventController,
781
505
  url,
782
- { skipLoadingState: true, replace: true }
506
+ { skipLoadingState: true, replace: true },
783
507
  );
784
508
 
785
509
  fetchPartialUpdate(
@@ -795,41 +519,33 @@ export function createNavigationBridge(
795
519
  interceptSourceUrl,
796
520
  cacheOnly: true,
797
521
  }),
798
- { staleRevalidation: true, interceptSourceUrl }
799
- ).catch((error) => {
800
- if (
801
- error instanceof DOMException &&
802
- error.name === "AbortError"
803
- ) {
804
- console.log("[Browser] Background revalidation aborted");
805
- return;
806
- }
807
- // For background revalidation, network errors are logged but don't trigger error boundary
808
- // since the user is already seeing cached content
809
- if (error instanceof NetworkError || isNetworkError(error)) {
810
- console.warn(
811
- "[Browser] Background revalidation network error (cached content preserved):",
812
- error.message
522
+ { type: "stale-revalidation", interceptSourceUrl },
523
+ )
524
+ .catch((error) => {
525
+ if (isBackgroundSuppressible(error)) return;
526
+ console.error(
527
+ "[Browser] Background revalidation failed:",
528
+ error,
813
529
  );
814
- return;
815
- }
816
- console.error("[Browser] Background revalidation failed:", error);
817
- });
530
+ })
531
+ .finally(() => {
532
+ tx[Symbol.dispose]();
533
+ });
818
534
  }
819
535
  return;
820
536
  } catch (error) {
821
537
  console.warn(
822
538
  "[Browser] Failed to render from cache, fetching:",
823
- error
539
+ error,
824
540
  );
825
541
  // Fall through to fetch
826
542
  }
827
543
  } else {
828
- console.log("[Browser] History cache miss for key:", historyKey);
544
+ debugLog("[Browser] History cache miss for key:", historyKey);
829
545
  }
830
546
 
831
547
  // Fetch if not cached
832
- using tx = createNavigationTransaction(store, eventController, url, {
548
+ const tx = createNavigationTransaction(store, eventController, url, {
833
549
  replace: true,
834
550
  });
835
551
 
@@ -839,45 +555,39 @@ export function createNavigationBridge(
839
555
  undefined,
840
556
  false,
841
557
  tx.handle.signal,
842
- tx.with({ url, replace: true, scroll: false })
558
+ tx.with({
559
+ url,
560
+ replace: true,
561
+ scroll: false,
562
+ intercept: isIntercept,
563
+ interceptSourceUrl,
564
+ }),
565
+ isIntercept ? { type: "navigate", interceptSourceUrl } : undefined,
843
566
  );
844
567
  // Restore scroll position after fetch completes
845
568
  handleNavigationEnd({ restore: true, isStreaming });
846
569
  } catch (error) {
847
570
  if (error instanceof DOMException && error.name === "AbortError") {
848
- console.log("[Browser] Popstate navigation aborted");
571
+ debugLog("[Browser] Popstate navigation aborted");
849
572
  return;
850
573
  }
851
574
 
852
- // Handle network errors by triggering root error boundary
853
- if (error instanceof NetworkError || isNetworkError(error)) {
854
- const networkError =
855
- error instanceof NetworkError
856
- ? error
857
- : new NetworkError(
858
- "Unable to connect to server. Please check your connection.",
859
- { cause: error, url, operation: "navigation" }
860
- );
861
-
575
+ const networkError = toNetworkError(error, {
576
+ url,
577
+ operation: "navigation",
578
+ });
579
+ if (networkError) {
862
580
  console.error(
863
- "[Browser] Network error during popstate navigation:",
864
- networkError
581
+ "[Browser] Network error during popstate:",
582
+ networkError,
865
583
  );
866
-
867
- startTransition(() => {
868
- onUpdate({
869
- root: createElement(NetworkErrorThrower, { error: networkError }),
870
- metadata: {
871
- pathname: url,
872
- segments: [],
873
- isError: true,
874
- },
875
- });
876
- });
584
+ emitNetworkError(onUpdate, networkError, url);
877
585
  return;
878
586
  }
879
587
 
880
588
  throw error;
589
+ } finally {
590
+ tx[Symbol.dispose]();
881
591
  }
882
592
  },
883
593
 
@@ -894,17 +604,39 @@ export function createNavigationBridge(
894
604
  this.handlePopstate();
895
605
  };
896
606
 
607
+ // When the browser restores a page from bfcache (back-forward cache),
608
+ // any in-flight navigation state is stale. This happens when:
609
+ // 1. A navigation triggers X-RSC-Reload (e.g., response route hit via SPA)
610
+ // 2. window.location.href does a hard navigation
611
+ // 3. The user presses back and the browser restores from bfcache
612
+ // At that point, currentNavigation is still set from step 1, so
613
+ // getState() returns "loading" and the progress bar shows.
614
+ // Abort the stale navigation to reset state to idle.
615
+ const handlePageShow = (event: PageTransitionEvent) => {
616
+ if (event.persisted) {
617
+ debugLog(
618
+ "[Browser] Page restored from bfcache, resetting navigation state",
619
+ );
620
+ eventController.abortNavigation();
621
+ // pagehide flips scrollRestoration to "auto" for bfcache compat;
622
+ // restore "manual" so the router controls scroll on SPA navigations.
623
+ window.history.scrollRestoration = "manual";
624
+ }
625
+ };
626
+
897
627
  // Register cross-tab refresh callback with the store
898
628
  store.setCrossTabRefreshCallback(() => {
899
629
  this.refresh();
900
630
  });
901
631
 
902
632
  window.addEventListener("popstate", handlePopstate);
903
- console.log("[Browser] Navigation bridge ready");
633
+ window.addEventListener("pageshow", handlePageShow);
634
+ debugLog("[Browser] Navigation bridge ready");
904
635
 
905
636
  return () => {
906
637
  cleanupLinks();
907
638
  window.removeEventListener("popstate", handlePopstate);
639
+ window.removeEventListener("pageshow", handlePageShow);
908
640
  };
909
641
  },
910
642
  };