@rangojs/router 0.0.0-experimental.002d056c

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 (305) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +899 -0
  3. package/dist/bin/rango.js +1606 -0
  4. package/dist/vite/index.js +5153 -0
  5. package/package.json +177 -0
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +253 -0
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +112 -0
  11. package/skills/document-cache/SKILL.md +182 -0
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +704 -0
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +313 -0
  16. package/skills/layout/SKILL.md +310 -0
  17. package/skills/links/SKILL.md +239 -0
  18. package/skills/loader/SKILL.md +596 -0
  19. package/skills/middleware/SKILL.md +339 -0
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +305 -0
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +118 -0
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +385 -0
  26. package/skills/router-setup/SKILL.md +439 -0
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +79 -0
  29. package/skills/typesafety/SKILL.md +623 -0
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +273 -0
  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 +899 -0
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/index.ts +18 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +141 -0
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +134 -0
  42. package/src/browser/navigation-bridge.ts +638 -0
  43. package/src/browser/navigation-client.ts +261 -0
  44. package/src/browser/navigation-store.ts +806 -0
  45. package/src/browser/navigation-transaction.ts +297 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +582 -0
  48. package/src/browser/prefetch/cache.ts +206 -0
  49. package/src/browser/prefetch/fetch.ts +145 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +48 -0
  52. package/src/browser/prefetch/queue.ts +128 -0
  53. package/src/browser/rango-state.ts +112 -0
  54. package/src/browser/react/Link.tsx +368 -0
  55. package/src/browser/react/NavigationProvider.tsx +413 -0
  56. package/src/browser/react/ScrollRestoration.tsx +94 -0
  57. package/src/browser/react/context.ts +59 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +52 -0
  60. package/src/browser/react/location-state-shared.ts +162 -0
  61. package/src/browser/react/location-state.ts +107 -0
  62. package/src/browser/react/mount-context.ts +37 -0
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +218 -0
  66. package/src/browser/react/use-client-cache.ts +58 -0
  67. package/src/browser/react/use-handle.ts +162 -0
  68. package/src/browser/react/use-href.tsx +40 -0
  69. package/src/browser/react/use-link-status.ts +135 -0
  70. package/src/browser/react/use-mount.ts +31 -0
  71. package/src/browser/react/use-navigation.ts +99 -0
  72. package/src/browser/react/use-params.ts +65 -0
  73. package/src/browser/react/use-pathname.ts +47 -0
  74. package/src/browser/react/use-router.ts +63 -0
  75. package/src/browser/react/use-search-params.ts +56 -0
  76. package/src/browser/react/use-segments.ts +171 -0
  77. package/src/browser/response-adapter.ts +73 -0
  78. package/src/browser/rsc-router.tsx +464 -0
  79. package/src/browser/scroll-restoration.ts +397 -0
  80. package/src/browser/segment-reconciler.ts +216 -0
  81. package/src/browser/segment-structure-assert.ts +83 -0
  82. package/src/browser/server-action-bridge.ts +667 -0
  83. package/src/browser/shallow.ts +40 -0
  84. package/src/browser/types.ts +547 -0
  85. package/src/browser/validate-redirect-origin.ts +29 -0
  86. package/src/build/generate-manifest.ts +438 -0
  87. package/src/build/generate-route-types.ts +36 -0
  88. package/src/build/index.ts +35 -0
  89. package/src/build/route-trie.ts +265 -0
  90. package/src/build/route-types/ast-helpers.ts +25 -0
  91. package/src/build/route-types/ast-route-extraction.ts +98 -0
  92. package/src/build/route-types/codegen.ts +102 -0
  93. package/src/build/route-types/include-resolution.ts +411 -0
  94. package/src/build/route-types/param-extraction.ts +48 -0
  95. package/src/build/route-types/per-module-writer.ts +128 -0
  96. package/src/build/route-types/router-processing.ts +479 -0
  97. package/src/build/route-types/scan-filter.ts +78 -0
  98. package/src/build/runtime-discovery.ts +231 -0
  99. package/src/cache/background-task.ts +34 -0
  100. package/src/cache/cache-key-utils.ts +44 -0
  101. package/src/cache/cache-policy.ts +125 -0
  102. package/src/cache/cache-runtime.ts +338 -0
  103. package/src/cache/cache-scope.ts +382 -0
  104. package/src/cache/cf/cf-cache-store.ts +982 -0
  105. package/src/cache/cf/index.ts +29 -0
  106. package/src/cache/document-cache.ts +369 -0
  107. package/src/cache/handle-capture.ts +81 -0
  108. package/src/cache/handle-snapshot.ts +41 -0
  109. package/src/cache/index.ts +44 -0
  110. package/src/cache/memory-segment-store.ts +328 -0
  111. package/src/cache/profile-registry.ts +73 -0
  112. package/src/cache/read-through-swr.ts +134 -0
  113. package/src/cache/segment-codec.ts +256 -0
  114. package/src/cache/taint.ts +98 -0
  115. package/src/cache/types.ts +342 -0
  116. package/src/client.rsc.tsx +85 -0
  117. package/src/client.tsx +601 -0
  118. package/src/component-utils.ts +76 -0
  119. package/src/components/DefaultDocument.tsx +27 -0
  120. package/src/context-var.ts +86 -0
  121. package/src/debug.ts +243 -0
  122. package/src/default-error-boundary.tsx +88 -0
  123. package/src/deps/browser.ts +8 -0
  124. package/src/deps/html-stream-client.ts +2 -0
  125. package/src/deps/html-stream-server.ts +2 -0
  126. package/src/deps/rsc.ts +10 -0
  127. package/src/deps/ssr.ts +2 -0
  128. package/src/errors.ts +365 -0
  129. package/src/handle.ts +135 -0
  130. package/src/handles/MetaTags.tsx +246 -0
  131. package/src/handles/breadcrumbs.ts +66 -0
  132. package/src/handles/index.ts +7 -0
  133. package/src/handles/meta.ts +264 -0
  134. package/src/host/cookie-handler.ts +165 -0
  135. package/src/host/errors.ts +97 -0
  136. package/src/host/index.ts +53 -0
  137. package/src/host/pattern-matcher.ts +214 -0
  138. package/src/host/router.ts +352 -0
  139. package/src/host/testing.ts +79 -0
  140. package/src/host/types.ts +146 -0
  141. package/src/host/utils.ts +25 -0
  142. package/src/href-client.ts +222 -0
  143. package/src/index.rsc.ts +233 -0
  144. package/src/index.ts +277 -0
  145. package/src/internal-debug.ts +11 -0
  146. package/src/loader.rsc.ts +89 -0
  147. package/src/loader.ts +64 -0
  148. package/src/network-error-thrower.tsx +23 -0
  149. package/src/outlet-context.ts +15 -0
  150. package/src/outlet-provider.tsx +45 -0
  151. package/src/prerender/param-hash.ts +37 -0
  152. package/src/prerender/store.ts +185 -0
  153. package/src/prerender.ts +463 -0
  154. package/src/reverse.ts +330 -0
  155. package/src/root-error-boundary.tsx +289 -0
  156. package/src/route-content-wrapper.tsx +196 -0
  157. package/src/route-definition/dsl-helpers.ts +934 -0
  158. package/src/route-definition/helper-factories.ts +200 -0
  159. package/src/route-definition/helpers-types.ts +430 -0
  160. package/src/route-definition/index.ts +52 -0
  161. package/src/route-definition/redirect.ts +93 -0
  162. package/src/route-definition.ts +1 -0
  163. package/src/route-map-builder.ts +281 -0
  164. package/src/route-name.ts +53 -0
  165. package/src/route-types.ts +259 -0
  166. package/src/router/content-negotiation.ts +116 -0
  167. package/src/router/debug-manifest.ts +72 -0
  168. package/src/router/error-handling.ts +287 -0
  169. package/src/router/find-match.ts +160 -0
  170. package/src/router/handler-context.ts +451 -0
  171. package/src/router/intercept-resolution.ts +397 -0
  172. package/src/router/lazy-includes.ts +236 -0
  173. package/src/router/loader-resolution.ts +420 -0
  174. package/src/router/logging.ts +251 -0
  175. package/src/router/manifest.ts +269 -0
  176. package/src/router/match-api.ts +620 -0
  177. package/src/router/match-context.ts +266 -0
  178. package/src/router/match-handlers.ts +440 -0
  179. package/src/router/match-middleware/background-revalidation.ts +223 -0
  180. package/src/router/match-middleware/cache-lookup.ts +634 -0
  181. package/src/router/match-middleware/cache-store.ts +295 -0
  182. package/src/router/match-middleware/index.ts +81 -0
  183. package/src/router/match-middleware/intercept-resolution.ts +306 -0
  184. package/src/router/match-middleware/segment-resolution.ts +193 -0
  185. package/src/router/match-pipelines.ts +179 -0
  186. package/src/router/match-result.ts +219 -0
  187. package/src/router/metrics.ts +282 -0
  188. package/src/router/middleware-cookies.ts +55 -0
  189. package/src/router/middleware-types.ts +222 -0
  190. package/src/router/middleware.ts +749 -0
  191. package/src/router/pattern-matching.ts +563 -0
  192. package/src/router/prerender-match.ts +402 -0
  193. package/src/router/preview-match.ts +170 -0
  194. package/src/router/revalidation.ts +289 -0
  195. package/src/router/router-context.ts +320 -0
  196. package/src/router/router-interfaces.ts +452 -0
  197. package/src/router/router-options.ts +592 -0
  198. package/src/router/router-registry.ts +24 -0
  199. package/src/router/segment-resolution/fresh.ts +570 -0
  200. package/src/router/segment-resolution/helpers.ts +263 -0
  201. package/src/router/segment-resolution/loader-cache.ts +198 -0
  202. package/src/router/segment-resolution/revalidation.ts +1242 -0
  203. package/src/router/segment-resolution/static-store.ts +67 -0
  204. package/src/router/segment-resolution.ts +21 -0
  205. package/src/router/segment-wrappers.ts +291 -0
  206. package/src/router/telemetry-otel.ts +299 -0
  207. package/src/router/telemetry.ts +300 -0
  208. package/src/router/timeout.ts +148 -0
  209. package/src/router/trie-matching.ts +239 -0
  210. package/src/router/types.ts +170 -0
  211. package/src/router.ts +1006 -0
  212. package/src/rsc/handler-context.ts +45 -0
  213. package/src/rsc/handler.ts +1089 -0
  214. package/src/rsc/helpers.ts +198 -0
  215. package/src/rsc/index.ts +36 -0
  216. package/src/rsc/loader-fetch.ts +209 -0
  217. package/src/rsc/manifest-init.ts +86 -0
  218. package/src/rsc/nonce.ts +32 -0
  219. package/src/rsc/origin-guard.ts +141 -0
  220. package/src/rsc/progressive-enhancement.ts +379 -0
  221. package/src/rsc/response-error.ts +37 -0
  222. package/src/rsc/response-route-handler.ts +347 -0
  223. package/src/rsc/rsc-rendering.ts +237 -0
  224. package/src/rsc/runtime-warnings.ts +42 -0
  225. package/src/rsc/server-action.ts +348 -0
  226. package/src/rsc/ssr-setup.ts +128 -0
  227. package/src/rsc/types.ts +263 -0
  228. package/src/search-params.ts +230 -0
  229. package/src/segment-system.tsx +454 -0
  230. package/src/server/context.ts +591 -0
  231. package/src/server/cookie-store.ts +190 -0
  232. package/src/server/fetchable-loader-store.ts +37 -0
  233. package/src/server/handle-store.ts +308 -0
  234. package/src/server/loader-registry.ts +133 -0
  235. package/src/server/request-context.ts +920 -0
  236. package/src/server/root-layout.tsx +10 -0
  237. package/src/server/tsconfig.json +14 -0
  238. package/src/server.ts +51 -0
  239. package/src/ssr/index.tsx +365 -0
  240. package/src/static-handler.ts +114 -0
  241. package/src/theme/ThemeProvider.tsx +297 -0
  242. package/src/theme/ThemeScript.tsx +61 -0
  243. package/src/theme/constants.ts +62 -0
  244. package/src/theme/index.ts +48 -0
  245. package/src/theme/theme-context.ts +44 -0
  246. package/src/theme/theme-script.ts +155 -0
  247. package/src/theme/types.ts +182 -0
  248. package/src/theme/use-theme.ts +44 -0
  249. package/src/types/boundaries.ts +158 -0
  250. package/src/types/cache-types.ts +198 -0
  251. package/src/types/error-types.ts +192 -0
  252. package/src/types/global-namespace.ts +100 -0
  253. package/src/types/handler-context.ts +687 -0
  254. package/src/types/index.ts +88 -0
  255. package/src/types/loader-types.ts +183 -0
  256. package/src/types/route-config.ts +170 -0
  257. package/src/types/route-entry.ts +109 -0
  258. package/src/types/segments.ts +148 -0
  259. package/src/types.ts +1 -0
  260. package/src/urls/include-helper.ts +197 -0
  261. package/src/urls/index.ts +53 -0
  262. package/src/urls/path-helper-types.ts +339 -0
  263. package/src/urls/path-helper.ts +329 -0
  264. package/src/urls/pattern-types.ts +95 -0
  265. package/src/urls/response-types.ts +106 -0
  266. package/src/urls/type-extraction.ts +372 -0
  267. package/src/urls/urls-function.ts +98 -0
  268. package/src/urls.ts +1 -0
  269. package/src/use-loader.tsx +354 -0
  270. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  271. package/src/vite/discovery/discover-routers.ts +344 -0
  272. package/src/vite/discovery/prerender-collection.ts +385 -0
  273. package/src/vite/discovery/route-types-writer.ts +258 -0
  274. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  275. package/src/vite/discovery/state.ts +108 -0
  276. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  277. package/src/vite/index.ts +16 -0
  278. package/src/vite/plugin-types.ts +48 -0
  279. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  280. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  281. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  282. package/src/vite/plugins/expose-action-id.ts +363 -0
  283. package/src/vite/plugins/expose-id-utils.ts +287 -0
  284. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  285. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  286. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  287. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  288. package/src/vite/plugins/expose-ids/types.ts +45 -0
  289. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  290. package/src/vite/plugins/refresh-cmd.ts +65 -0
  291. package/src/vite/plugins/use-cache-transform.ts +323 -0
  292. package/src/vite/plugins/version-injector.ts +83 -0
  293. package/src/vite/plugins/version-plugin.ts +266 -0
  294. package/src/vite/plugins/version.d.ts +12 -0
  295. package/src/vite/plugins/virtual-entries.ts +123 -0
  296. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  297. package/src/vite/rango.ts +445 -0
  298. package/src/vite/router-discovery.ts +777 -0
  299. package/src/vite/utils/ast-handler-extract.ts +517 -0
  300. package/src/vite/utils/banner.ts +36 -0
  301. package/src/vite/utils/bundle-analysis.ts +137 -0
  302. package/src/vite/utils/manifest-utils.ts +70 -0
  303. package/src/vite/utils/package-resolution.ts +121 -0
  304. package/src/vite/utils/prerender-utils.ts +189 -0
  305. package/src/vite/utils/shared-utils.ts +169 -0
@@ -0,0 +1,638 @@
1
+ import type {
2
+ NavigationBridge,
3
+ NavigationBridgeConfig,
4
+ NavigateOptionsInternal,
5
+ ResolvedSegment,
6
+ } from "./types.js";
7
+ import * as React from "react";
8
+ import { startTransition } from "react";
9
+ import {
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";
19
+
20
+ // addTransitionType is only available in React experimental
21
+ const addTransitionType: ((type: string) => void) | undefined =
22
+ "addTransitionType" in React ? (React as any).addTransitionType : undefined;
23
+
24
+ import { setupLinkInterception } from "./link-interceptor.js";
25
+ import { createPartialUpdater } from "./partial-update.js";
26
+ import { generateHistoryKey } from "./navigation-store.js";
27
+ import type { EventController } from "./event-controller.js";
28
+ import { isInterceptOnlyCache } from "./intercept-utils.js";
29
+ import {
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";
37
+
38
+ // Polyfill Symbol.dispose for Safari and older browsers
39
+ if (typeof Symbol.dispose === "undefined") {
40
+ (Symbol as any).dispose = Symbol("Symbol.dispose");
41
+ }
42
+
43
+ export { createNavigationTransaction };
44
+
45
+ /**
46
+ * Extended configuration for navigation bridge with event controller
47
+ */
48
+ export interface NavigationBridgeConfigWithController extends NavigationBridgeConfig {
49
+ eventController: EventController;
50
+ /** RSC version from initial payload metadata */
51
+ version?: string;
52
+ }
53
+
54
+ /**
55
+ * Create a navigation bridge for handling client-side navigation
56
+ *
57
+ * The bridge coordinates all navigation operations:
58
+ * - Link click interception
59
+ * - Browser back/forward (popstate)
60
+ * - Programmatic navigation
61
+ *
62
+ * Uses the event controller for reactive state management.
63
+ *
64
+ * @param config - Bridge configuration
65
+ * @returns NavigationBridge instance
66
+ */
67
+ export function createNavigationBridge(
68
+ config: NavigationBridgeConfigWithController,
69
+ ): NavigationBridge {
70
+ const { store, client, eventController, onUpdate, renderSegments, version } =
71
+ config;
72
+
73
+ // Create shared partial updater
74
+ const fetchPartialUpdate = createPartialUpdater({
75
+ store,
76
+ client,
77
+ onUpdate,
78
+ renderSegments,
79
+ version,
80
+ });
81
+
82
+ return {
83
+ /**
84
+ * Navigate to a URL
85
+ * Uses cached segments for SWR revalidation when available
86
+ */
87
+ async navigate(
88
+ url: string,
89
+ options?: NavigateOptionsInternal,
90
+ ): Promise<void> {
91
+ // Resolve LocationStateEntry[] to flat object if needed
92
+ const resolvedState =
93
+ options?.state !== undefined
94
+ ? resolveNavigationState(options.state)
95
+ : undefined;
96
+
97
+ // Cross-origin URLs are not handled by SPA navigation.
98
+ // Fall back to a full browser navigation for http/https only.
99
+ let targetUrl: URL;
100
+ try {
101
+ targetUrl = new URL(url, window.location.origin);
102
+ } catch {
103
+ console.warn(`[rango] navigate() ignored: malformed URL "${url}"`);
104
+ return;
105
+ }
106
+ if (targetUrl.origin !== window.location.origin) {
107
+ if (targetUrl.protocol !== "http:" && targetUrl.protocol !== "https:") {
108
+ console.error(
109
+ `[rango] navigate() blocked: unsupported scheme "${targetUrl.protocol}"`,
110
+ );
111
+ return;
112
+ }
113
+ window.location.href = targetUrl.href;
114
+ return;
115
+ }
116
+
117
+ // Shallow navigation: skip RSC fetch when revalidate is false
118
+ // and the pathname hasn't changed (search param / hash only change).
119
+ if (
120
+ options?.revalidate === false &&
121
+ targetUrl.pathname === new URL(window.location.href).pathname
122
+ ) {
123
+ // Preserve intercept context from the current history entry so that
124
+ // popstate uses the correct cache key (:intercept suffix) and restores
125
+ // the right full-page vs modal semantics.
126
+ const currentHistoryState = window.history.state;
127
+ const isIntercept = currentHistoryState?.intercept === true;
128
+ const interceptSourceUrl = isIntercept
129
+ ? currentHistoryState?.sourceUrl
130
+ : undefined;
131
+
132
+ const historyKey = generateHistoryKey(url, { intercept: isIntercept });
133
+
134
+ // Copy current segments to the new history key so back/forward restores instantly
135
+ const currentKey = store.getHistoryKey();
136
+ const currentCache = store.getCachedSegments(currentKey);
137
+ if (currentCache?.segments) {
138
+ const currentHandleData = eventController.getHandleState().data;
139
+ store.cacheSegmentsForHistory(
140
+ historyKey,
141
+ currentCache.segments,
142
+ currentHandleData,
143
+ );
144
+ }
145
+
146
+ // Save current scroll position before changing URL
147
+ handleNavigationStart();
148
+
149
+ // Snapshot old state before pushState/replaceState overwrites it
150
+ const oldState = window.history.state;
151
+
152
+ // Update browser URL (carry intercept context into history state)
153
+ const historyState = buildHistoryState(
154
+ resolvedState,
155
+ {
156
+ intercept: isIntercept || undefined,
157
+ sourceUrl: interceptSourceUrl,
158
+ },
159
+ {},
160
+ );
161
+ if (options.replace) {
162
+ window.history.replaceState(historyState, "", url);
163
+ } else {
164
+ window.history.pushState(historyState, "", url);
165
+ }
166
+
167
+ // Ensure new history entry has a scroll restoration key
168
+ ensureHistoryKey();
169
+
170
+ // Notify useLocationState() hooks when state changes
171
+ const hasOldState =
172
+ oldState &&
173
+ typeof oldState === "object" &&
174
+ ("state" in oldState ||
175
+ Object.keys(oldState).some((k) => k.startsWith("__rsc_ls_")));
176
+ const hasNewState =
177
+ historyState &&
178
+ ("state" in historyState ||
179
+ Object.keys(historyState).some((k) => k.startsWith("__rsc_ls_")));
180
+ if (hasOldState || hasNewState) {
181
+ window.dispatchEvent(new Event("__rsc_locationstate"));
182
+ }
183
+
184
+ // Update store history key so future navigations reference the right cache
185
+ store.setHistoryKey(historyKey);
186
+ store.setCurrentUrl(url);
187
+
188
+ // Notify hooks — location updates, state stays idle
189
+ eventController.setLocation(targetUrl);
190
+
191
+ // Handle post-navigation scroll
192
+ handleNavigationEnd({ scroll: options.scroll });
193
+ return;
194
+ }
195
+
196
+ // Only abort pending requests when navigating to a different route
197
+ // Same-route navigation (e.g., /todos -> /todos) should not cancel in-flight actions
198
+ const currentPath = new URL(window.location.href).pathname;
199
+ const targetPath = targetUrl.pathname;
200
+ if (currentPath !== targetPath) {
201
+ eventController.abortNavigation();
202
+ }
203
+
204
+ // Check if we're "leaving intercept" - navigating from intercept to same URL without intercept
205
+ // This happens when clicking "View Full Details" in an intercept modal
206
+ const currentHistoryState = window.history.state;
207
+ const isCurrentlyIntercept = currentHistoryState?.intercept === true;
208
+ const isSamePathNavigation = currentPath === targetPath;
209
+ const isLeavingIntercept = isCurrentlyIntercept && isSamePathNavigation;
210
+
211
+ if (isLeavingIntercept) {
212
+ debugLog(
213
+ "[Browser] Leaving intercept - same URL navigation from intercept",
214
+ );
215
+ // Clear intercept source URL to ensure server doesn't treat this as intercept
216
+ store.setInterceptSourceUrl(null);
217
+ }
218
+
219
+ // Before navigating away, update the source page's cache with the latest handleData.
220
+ // This ensures the cache has correct handleData even if handles were streaming.
221
+ const sourceHistoryKey = store.getHistoryKey();
222
+ const sourceCached = store.getCachedSegments(sourceHistoryKey);
223
+ if (sourceCached?.segments && sourceCached.segments.length > 0) {
224
+ const currentHandleData = eventController.getHandleState().data;
225
+ store.cacheSegmentsForHistory(
226
+ sourceHistoryKey,
227
+ sourceCached.segments,
228
+ currentHandleData,
229
+ );
230
+ }
231
+
232
+ // Check if we have cached segments for target URL
233
+ const historyKey = generateHistoryKey(url);
234
+ const cached = store.getCachedSegments(historyKey);
235
+
236
+ // For shared segments (same ID on current and target), use current page's version
237
+ // since it may have fresher data after an action revalidation.
238
+ // This avoids unnecessary server round-trips for shared layout loaders.
239
+ let cachedSegments = cached?.segments;
240
+ const cachedHandleData = cached?.handleData;
241
+ if (cachedSegments && sourceCached?.segments) {
242
+ const sourceSegmentMap = new Map(
243
+ sourceCached.segments.map((s) => [s.id, s]),
244
+ );
245
+ cachedSegments = cachedSegments.map((targetSeg) => {
246
+ const sourceSeg = sourceSegmentMap.get(targetSeg.id);
247
+ // Use source (current page) version for shared segments - it's fresher
248
+ return sourceSeg || targetSeg;
249
+ });
250
+ }
251
+
252
+ // Also check if there's an intercept cache entry for this URL
253
+ // If so, this URL CAN be intercepted, and we shouldn't use the non-intercept cache
254
+ // because the navigation might result in an intercept (depending on source URL)
255
+ const interceptHistoryKey = generateHistoryKey(url, { intercept: true });
256
+ const hasInterceptCache = store.hasHistoryCache(interceptHistoryKey);
257
+
258
+ // Skip cached SWR for:
259
+ // 1. intercept caches - interception depends on source page context
260
+ // 2. routes that CAN be intercepted - we don't know if this navigation will intercept
261
+ // 3. when leaving intercept - we need fresh non-intercept segments from server
262
+ // 4. redirect-with-state - force re-render so hooks read fresh state
263
+ const hasUsableCache =
264
+ cachedSegments &&
265
+ cachedSegments.length > 0 &&
266
+ !isInterceptOnlyCache(cachedSegments) &&
267
+ !hasInterceptCache &&
268
+ !isLeavingIntercept &&
269
+ !options?._skipCache;
270
+
271
+ const tx = createNavigationTransaction(store, eventController, url, {
272
+ ...options,
273
+ state: resolvedState,
274
+ skipLoadingState: hasUsableCache,
275
+ });
276
+
277
+ // REVALIDATE: Fetch fresh data from server
278
+ try {
279
+ await fetchPartialUpdate(
280
+ url,
281
+ hasUsableCache
282
+ ? cachedSegments!.map((s) => s.id)
283
+ : options?._skipCache
284
+ ? [] // Action redirect: send no segments so server renders everything fresh
285
+ : undefined,
286
+ false,
287
+ tx.handle.signal,
288
+ tx.with({
289
+ url,
290
+ replace: options?.replace,
291
+ scroll: options?.scroll,
292
+ state: resolvedState,
293
+ }),
294
+ hasUsableCache
295
+ ? {
296
+ type: "navigate" as const,
297
+ targetCacheSegments: cachedSegments,
298
+ targetCacheHandleData: cachedHandleData,
299
+ }
300
+ : isLeavingIntercept
301
+ ? { type: "leave-intercept" as const }
302
+ : undefined,
303
+ );
304
+ } catch (error) {
305
+ // Server-side redirect with location state: the current transaction's
306
+ // cleanup resets loading state. Re-navigate to the redirect
307
+ // target carrying the server-set state into history.pushState.
308
+ if (error instanceof ServerRedirect) {
309
+ const redirectUrl = validateRedirectOrigin(
310
+ error.url,
311
+ window.location.origin,
312
+ );
313
+ if (!redirectUrl) {
314
+ return;
315
+ }
316
+ return this.navigate(redirectUrl, {
317
+ state: error.state,
318
+ replace: options?.replace,
319
+ _skipCache: true,
320
+ } as NavigateOptionsInternal);
321
+ }
322
+
323
+ if (error instanceof DOMException && error.name === "AbortError") {
324
+ debugLog("[Browser] Navigation aborted by newer navigation");
325
+ return;
326
+ }
327
+
328
+ const networkError = toNetworkError(error, {
329
+ url,
330
+ operation: "navigation",
331
+ });
332
+ if (networkError) {
333
+ console.error(
334
+ "[Browser] Network error during navigation:",
335
+ networkError,
336
+ );
337
+ emitNetworkError(onUpdate, networkError, url);
338
+ return;
339
+ }
340
+
341
+ throw error;
342
+ } finally {
343
+ tx[Symbol.dispose]();
344
+ }
345
+ },
346
+
347
+ /**
348
+ * Refresh current route
349
+ */
350
+ async refresh(): Promise<void> {
351
+ eventController.abortNavigation();
352
+
353
+ const tx = createNavigationTransaction(
354
+ store,
355
+ eventController,
356
+ window.location.href,
357
+ { replace: true },
358
+ );
359
+
360
+ try {
361
+ // Refetch with empty segments to get everything fresh
362
+ await fetchPartialUpdate(
363
+ window.location.href,
364
+ [],
365
+ false,
366
+ tx.handle.signal,
367
+ tx.with({ url: window.location.href, replace: true, scroll: false }),
368
+ );
369
+ } catch (error) {
370
+ const networkError = toNetworkError(error, {
371
+ url: window.location.href,
372
+ operation: "revalidation",
373
+ });
374
+ if (networkError) {
375
+ console.error(
376
+ "[Browser] Network error during refresh:",
377
+ networkError,
378
+ );
379
+ emitNetworkError(onUpdate, networkError, window.location.href);
380
+ return;
381
+ }
382
+ throw error;
383
+ } finally {
384
+ tx[Symbol.dispose]();
385
+ }
386
+ },
387
+
388
+ /**
389
+ * Handle browser back/forward navigation
390
+ * Uses cached segments when available for instant restoration
391
+ */
392
+ async handlePopstate(): Promise<void> {
393
+ // Abort any pending navigation to prevent race conditions
394
+ eventController.abortNavigation();
395
+
396
+ const url = window.location.href;
397
+
398
+ // Check if this history entry is an intercept
399
+ const historyState = window.history.state;
400
+ const isIntercept = historyState?.intercept === true;
401
+ const interceptSourceUrl = historyState?.sourceUrl;
402
+
403
+ // Check if intercept context is changing (same URL, different intercept state)
404
+ // If so, abort in-flight actions - their results would be for wrong context
405
+ const currentInterceptSource = store.getInterceptSourceUrl();
406
+ const newInterceptSource = interceptSourceUrl ?? null;
407
+ if (currentInterceptSource !== newInterceptSource) {
408
+ debugLog(
409
+ `[Browser] Intercept context changing (${currentInterceptSource} -> ${newInterceptSource}), aborting in-flight actions`,
410
+ );
411
+ eventController.abortAllActions();
412
+ }
413
+
414
+ // Compute history key from URL (with intercept suffix if applicable)
415
+ const historyKey = generateHistoryKey(url, { intercept: isIntercept });
416
+
417
+ debugLog(
418
+ "[Browser] Popstate -",
419
+ isIntercept ? "intercept" : "normal",
420
+ "key:",
421
+ historyKey,
422
+ );
423
+
424
+ // Update location in event controller
425
+ eventController.setLocation(new URL(url));
426
+
427
+ // If this is an intercept, restore the intercept context
428
+ if (isIntercept && interceptSourceUrl) {
429
+ store.setInterceptSourceUrl(interceptSourceUrl);
430
+ } else {
431
+ store.setInterceptSourceUrl(null);
432
+ }
433
+
434
+ // Helper to check if streaming is in progress
435
+ const isStreaming = () => eventController.getState().isStreaming;
436
+
437
+ // Check if we can restore from history cache
438
+ const cached = store.getCachedSegments(historyKey);
439
+ const cachedSegments = cached?.segments;
440
+ const cachedHandleData = cached?.handleData;
441
+ const isStale = cached?.stale ?? false;
442
+
443
+ if (cachedSegments && cachedSegments.length > 0) {
444
+ // Update store to point to this history entry
445
+ store.setHistoryKey(historyKey);
446
+ store.setSegmentIds(cachedSegments.map((s) => s.id));
447
+ store.setCurrentUrl(url);
448
+ store.setPath(new URL(url).pathname);
449
+
450
+ // Render from cache - force await to skip loading fallbacks
451
+ try {
452
+ const root = await renderSegments(cachedSegments, {
453
+ forceAwait: true,
454
+ });
455
+ // Merge params from cached segments for useParams restoration.
456
+ // Set params on event controller before onUpdate so both location
457
+ // and params are current when the debounced notify() fires.
458
+ const cachedParams: Record<string, string> = {};
459
+ for (const s of cachedSegments) {
460
+ if (s.params) Object.assign(cachedParams, s.params);
461
+ }
462
+ eventController.setParams(cachedParams);
463
+
464
+ const popstateUpdate = {
465
+ root,
466
+ metadata: {
467
+ pathname: new URL(url).pathname,
468
+ segments: cachedSegments,
469
+ isPartial: true,
470
+ matched: cachedSegments.map((s) => s.id),
471
+ diff: [],
472
+ cachedHandleData,
473
+ params: cachedParams,
474
+ },
475
+ scroll: { restore: true, isStreaming },
476
+ };
477
+ const hasTransition = cachedSegments.some((s) => s.transition);
478
+ if (hasTransition) {
479
+ startTransition(() => {
480
+ if (addTransitionType) {
481
+ addTransitionType("navigation-back");
482
+ }
483
+ onUpdate(popstateUpdate);
484
+ });
485
+ } else {
486
+ onUpdate(popstateUpdate);
487
+ }
488
+
489
+ // SWR: If stale, trigger background revalidation
490
+ if (isStale) {
491
+ debugLog("[Browser] Cache is stale, background revalidating...");
492
+ // Background revalidation - don't await, just fire and forget
493
+ const segmentIds = cachedSegments.map((s) => s.id);
494
+
495
+ const tx = createNavigationTransaction(
496
+ store,
497
+ eventController,
498
+ url,
499
+ { skipLoadingState: true, replace: true },
500
+ );
501
+
502
+ fetchPartialUpdate(
503
+ url,
504
+ segmentIds,
505
+ false,
506
+ tx.handle.signal,
507
+ tx.with({
508
+ url,
509
+ replace: true,
510
+ scroll: false,
511
+ intercept: isIntercept,
512
+ interceptSourceUrl,
513
+ cacheOnly: true,
514
+ }),
515
+ { type: "stale-revalidation", interceptSourceUrl },
516
+ )
517
+ .catch((error) => {
518
+ if (isBackgroundSuppressible(error)) return;
519
+ console.error(
520
+ "[Browser] Background revalidation failed:",
521
+ error,
522
+ );
523
+ })
524
+ .finally(() => {
525
+ tx[Symbol.dispose]();
526
+ });
527
+ }
528
+ return;
529
+ } catch (error) {
530
+ console.warn(
531
+ "[Browser] Failed to render from cache, fetching:",
532
+ error,
533
+ );
534
+ // Fall through to fetch
535
+ }
536
+ } else {
537
+ debugLog("[Browser] History cache miss for key:", historyKey);
538
+ }
539
+
540
+ // Fetch if not cached
541
+ const tx = createNavigationTransaction(store, eventController, url, {
542
+ replace: true,
543
+ });
544
+
545
+ try {
546
+ await fetchPartialUpdate(
547
+ url,
548
+ undefined,
549
+ false,
550
+ tx.handle.signal,
551
+ tx.with({
552
+ url,
553
+ replace: true,
554
+ scroll: false,
555
+ intercept: isIntercept,
556
+ interceptSourceUrl,
557
+ }),
558
+ isIntercept ? { type: "navigate", interceptSourceUrl } : undefined,
559
+ );
560
+ // Restore scroll position after fetch completes
561
+ handleNavigationEnd({ restore: true, isStreaming });
562
+ } catch (error) {
563
+ if (error instanceof DOMException && error.name === "AbortError") {
564
+ debugLog("[Browser] Popstate navigation aborted");
565
+ return;
566
+ }
567
+
568
+ const networkError = toNetworkError(error, {
569
+ url,
570
+ operation: "navigation",
571
+ });
572
+ if (networkError) {
573
+ console.error(
574
+ "[Browser] Network error during popstate:",
575
+ networkError,
576
+ );
577
+ emitNetworkError(onUpdate, networkError, url);
578
+ return;
579
+ }
580
+
581
+ throw error;
582
+ } finally {
583
+ tx[Symbol.dispose]();
584
+ }
585
+ },
586
+
587
+ /**
588
+ * Register link interception
589
+ * @returns Cleanup function
590
+ */
591
+ registerLinkInterception(): () => void {
592
+ const cleanupLinks = setupLinkInterception((url, options) => {
593
+ this.navigate(url, options);
594
+ });
595
+
596
+ const handlePopstate = () => {
597
+ this.handlePopstate();
598
+ };
599
+
600
+ // When the browser restores a page from bfcache (back-forward cache),
601
+ // any in-flight navigation state is stale. This happens when:
602
+ // 1. A navigation triggers X-RSC-Reload (e.g., response route hit via SPA)
603
+ // 2. window.location.href does a hard navigation
604
+ // 3. The user presses back and the browser restores from bfcache
605
+ // At that point, currentNavigation is still set from step 1, so
606
+ // getState() returns "loading" and the progress bar shows.
607
+ // Abort the stale navigation to reset state to idle.
608
+ const handlePageShow = (event: PageTransitionEvent) => {
609
+ if (event.persisted) {
610
+ debugLog(
611
+ "[Browser] Page restored from bfcache, resetting navigation state",
612
+ );
613
+ eventController.abortNavigation();
614
+ // pagehide flips scrollRestoration to "auto" for bfcache compat;
615
+ // restore "manual" so the router controls scroll on SPA navigations.
616
+ window.history.scrollRestoration = "manual";
617
+ }
618
+ };
619
+
620
+ // Register cross-tab refresh callback with the store
621
+ store.setCrossTabRefreshCallback(() => {
622
+ this.refresh();
623
+ });
624
+
625
+ window.addEventListener("popstate", handlePopstate);
626
+ window.addEventListener("pageshow", handlePageShow);
627
+ debugLog("[Browser] Navigation bridge ready");
628
+
629
+ return () => {
630
+ cleanupLinks();
631
+ window.removeEventListener("popstate", handlePopstate);
632
+ window.removeEventListener("pageshow", handlePageShow);
633
+ };
634
+ },
635
+ };
636
+ }
637
+
638
+ export { createNavigationBridge as default };