@rangojs/router 0.0.0-experimental.5 → 0.0.0-experimental.50

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 (301) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +884 -4
  3. package/dist/bin/rango.js +1606 -0
  4. package/dist/vite/index.js +4567 -769
  5. package/package.json +77 -58
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +85 -23
  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 +204 -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 +92 -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 +282 -557
  42. package/src/browser/navigation-client.ts +157 -71
  43. package/src/browser/navigation-store.ts +33 -50
  44. package/src/browser/navigation-transaction.ts +297 -0
  45. package/src/browser/network-error-handler.ts +61 -0
  46. package/src/browser/partial-update.ts +303 -310
  47. package/src/browser/prefetch/cache.ts +206 -0
  48. package/src/browser/prefetch/fetch.ts +144 -0
  49. package/src/browser/prefetch/observer.ts +65 -0
  50. package/src/browser/prefetch/policy.ts +48 -0
  51. package/src/browser/prefetch/queue.ts +128 -0
  52. package/src/browser/rango-state.ts +112 -0
  53. package/src/browser/react/Link.tsx +193 -73
  54. package/src/browser/react/NavigationProvider.tsx +160 -13
  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 +24 -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 +188 -55
  76. package/src/browser/scroll-restoration.ts +117 -44
  77. package/src/browser/segment-reconciler.ts +221 -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 +118 -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 +479 -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 +342 -0
  100. package/src/cache/cache-scope.ts +167 -309
  101. package/src/cache/cf/cf-cache-store.ts +571 -17
  102. package/src/cache/cf/index.ts +13 -3
  103. package/src/cache/document-cache.ts +116 -77
  104. package/src/cache/handle-capture.ts +81 -0
  105. package/src/cache/handle-snapshot.ts +41 -0
  106. package/src/cache/index.ts +1 -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 +19 -9
  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 +165 -0
  126. package/src/host/errors.ts +97 -0
  127. package/src/host/index.ts +53 -0
  128. package/src/host/pattern-matcher.ts +214 -0
  129. package/src/host/router.ts +352 -0
  130. package/src/host/testing.ts +79 -0
  131. package/src/host/types.ts +146 -0
  132. package/src/host/utils.ts +25 -0
  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 -147
  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 +959 -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 +217 -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 +160 -0
  160. package/src/router/handler-context.ts +374 -81
  161. package/src/router/intercept-resolution.ts +397 -0
  162. package/src/router/lazy-includes.ts +237 -0
  163. package/src/router/loader-resolution.ts +215 -122
  164. package/src/router/logging.ts +251 -0
  165. package/src/router/manifest.ts +154 -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 +108 -93
  170. package/src/router/match-middleware/cache-lookup.ts +440 -10
  171. package/src/router/match-middleware/cache-store.ts +98 -26
  172. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  173. package/src/router/match-middleware/segment-resolution.ts +27 -6
  174. package/src/router/match-pipelines.ts +10 -45
  175. package/src/router/match-result.ts +55 -33
  176. package/src/router/metrics.ts +240 -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 +327 -369
  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 +41 -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 +677 -0
  189. package/src/router/segment-resolution/helpers.ts +263 -0
  190. package/src/router/segment-resolution/loader-cache.ts +199 -0
  191. package/src/router/segment-resolution/revalidation.ts +1296 -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 +291 -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 +665 -4182
  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 +237 -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 +172 -21
  219. package/src/server/context.ts +266 -58
  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 +439 -73
  225. package/src/server.ts +35 -128
  226. package/src/ssr/index.tsx +101 -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 +773 -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 +109 -0
  243. package/src/types/segments.ts +150 -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 +108 -0
  261. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  262. package/src/vite/index.ts +11 -782
  263. package/src/vite/plugin-types.ts +48 -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 -53
  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 +266 -0
  279. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +27 -16
  280. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  281. package/src/vite/rango.ts +445 -0
  282. package/src/vite/router-discovery.ts +777 -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/warmup/connection-warmup.tsx +0 -94
  301. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -1,69 +1,71 @@
1
1
  "use client";
2
2
 
3
- import React, { forwardRef, useCallback, useContext, useRef, type ForwardRefExoticComponent, type RefAttributes } from "react";
3
+ import React, {
4
+ forwardRef,
5
+ useCallback,
6
+ useContext,
7
+ useEffect,
8
+ useRef,
9
+ type ForwardRefExoticComponent,
10
+ type RefAttributes,
11
+ } from "react";
4
12
  import { NavigationStoreContext } from "./context.js";
5
13
  import { LinkContext } from "./use-link-status.js";
6
14
  import type { NavigateOptions } from "../types.js";
15
+ import { isHashOnlyNavigation } from "../link-interceptor.js";
7
16
  import {
8
- type LocationStateEntry,
9
17
  isLocationStateEntry,
18
+ type LocationStateEntry,
10
19
  resolveLocationStateEntries,
11
20
  } from "./location-state.js";
12
21
 
13
22
  /**
14
- * State value or getter function for just-in-time state resolution (legacy)
23
+ * State prop type for Link component.
24
+ * - LocationStateEntry[]: Type-safe state entries via createLocationState()
25
+ * - StateOrGetter: Plain state object or click-time getter function
26
+ * - Record<string, unknown>: Plain state object passed to history.pushState
15
27
  */
16
28
  export type StateOrGetter<T = unknown> = T | (() => T);
17
29
 
18
- /**
19
- * State prop type for Link component
20
- * - LocationStateEntry[]: Type-safe state entries (always lazy)
21
- * - StateOrGetter: Legacy format for backwards compatibility
22
- */
23
- export type LinkState = LocationStateEntry[] | StateOrGetter;
24
-
25
- // Track prefetched URLs to avoid duplicate <link> elements
26
- const prefetchedUrls = new Set<string>();
30
+ export type LinkState =
31
+ | LocationStateEntry[]
32
+ | StateOrGetter<Record<string, unknown>>;
27
33
 
28
- /**
29
- * Inject a <link rel="prefetch"> element into the document head
30
- * for the given URL with RSC partial request parameters.
31
- */
32
- function prefetchUrl(url: string, segmentIds: string[]): void {
33
- if (prefetchedUrls.has(url)) return;
34
- prefetchedUrls.add(url);
35
-
36
- // Build RSC partial URL with segment IDs
37
- const targetUrl = new URL(url, window.location.origin);
38
- targetUrl.searchParams.set("_rsc_partial", "true");
39
- if (segmentIds.length > 0) {
40
- targetUrl.searchParams.set("_rsc_segments", segmentIds.join(","));
41
- }
34
+ import { prefetchDirect, prefetchQueued } from "../prefetch/fetch.js";
35
+ import {
36
+ observeForPrefetch,
37
+ unobserveForPrefetch,
38
+ } from "../prefetch/observer.js";
42
39
 
43
- // Inject <link rel="prefetch"> into head
44
- const link = document.createElement("link");
45
- link.rel = "prefetch";
46
- link.href = targetUrl.toString();
47
- link.as = "fetch";
48
- document.head.appendChild(link);
49
- }
40
+ // Touch device detection for adaptive strategy.
41
+ // Checked once at module load (Link.tsx is "use client", runs only in browser).
42
+ const isTouchDevice =
43
+ typeof window !== "undefined" && window.matchMedia("(hover: none)").matches;
50
44
 
51
45
  /**
52
46
  * Prefetch strategy for the Link component
53
- * - "hover": Prefetch on mouse enter (uses native <link rel="prefetch">)
54
- * - "viewport": Prefetch when link enters viewport (not yet implemented)
55
- * - "hybrid": Hover on desktop, viewport on mobile (not yet implemented)
47
+ * - "hover": Prefetch on mouse enter (direct, no queue)
48
+ * - "viewport": Prefetch when link enters viewport (queued, waits for idle)
49
+ * - "render": Prefetch on component mount regardless of visibility (queued, waits for idle)
50
+ * - "adaptive": Hover on pointer devices, viewport on touch devices
56
51
  * - "none": No prefetching (default)
57
52
  */
58
- export type PrefetchStrategy = "hover" | "viewport" | "hybrid" | "none";
53
+ export type PrefetchStrategy =
54
+ | "hover"
55
+ | "viewport"
56
+ | "render"
57
+ | "adaptive"
58
+ | "none";
59
59
 
60
60
  /**
61
61
  * Link component props
62
62
  */
63
- export interface LinkProps
64
- extends Omit<React.AnchorHTMLAttributes<HTMLAnchorElement>, "href"> {
63
+ export interface LinkProps extends Omit<
64
+ React.AnchorHTMLAttributes<HTMLAnchorElement>,
65
+ "href"
66
+ > {
65
67
  /**
66
- * The URL to navigate to (typically from router.href())
68
+ * The URL to navigate to (typically from router.reverse())
67
69
  */
68
70
  to: string;
69
71
  /**
@@ -78,6 +80,16 @@ export interface LinkProps
78
80
  * Force full document navigation instead of SPA
79
81
  */
80
82
  reloadDocument?: boolean;
83
+ /**
84
+ * Whether to revalidate server data on navigation.
85
+ * Set to `false` to skip the RSC server fetch and only update the URL.
86
+ *
87
+ * Only takes effect when the pathname stays the same (search param / hash changes).
88
+ * If the pathname changes, this option is ignored and a full navigation occurs.
89
+ *
90
+ * @default true
91
+ */
92
+ revalidate?: boolean;
81
93
  /**
82
94
  * Prefetch strategy for the link destination
83
95
  * @default "none"
@@ -90,16 +102,29 @@ export interface LinkProps
90
102
  * @example
91
103
  * ```tsx
92
104
  * // Type-safe state with createLocationState (recommended)
93
- * const ProductState = createLocationState((p: Product) => ({ name: p.name }));
94
- * <Link to="/product" state={[ProductState(product)]}>View</Link>
105
+ * const ProductState = createLocationState<{ name: string; price: number }>();
106
+ * <Link to="/product" state={[ProductState({ name: product.name, price: product.price })]}>
107
+ * View
108
+ * </Link>
109
+ *
110
+ * // Type-safe just-in-time state (getter called at click time, not render time).
111
+ * // Must be in a client component -- getter can't cross the RSC boundary.
112
+ * <Link
113
+ * to="/product"
114
+ * state={[ProductState(() => ({ name: product.name, price: product.price }))]}
115
+ * >
116
+ * View
117
+ * </Link>
95
118
  *
96
119
  * // Multiple typed states
97
- * <Link to="/checkout" state={[ProductState(p), CartState(c)]}>Checkout</Link>
120
+ * <Link to="/checkout" state={[ProductState({ name: p.name, price: p.price }), CartState(c)]}>
121
+ * Checkout
122
+ * </Link>
98
123
  *
99
- * // Legacy: static state
124
+ * // Plain static state
100
125
  * <Link to="/product" state={{ from: "list" }}>View</Link>
101
126
  *
102
- * // Legacy: dynamic state (called at click time)
127
+ * // Plain just-in-time state (called at click time, requires client component)
103
128
  * <Link to="/product" state={() => ({ scrollY: window.scrollY })}>View</Link>
104
129
  * ```
105
130
  */
@@ -134,9 +159,9 @@ function isExternalUrl(href: string): boolean {
134
159
  /**
135
160
  * Type-safe Link component for SPA navigation
136
161
  *
137
- * Works with router.href() for type-safe URLs:
162
+ * Works with router.reverse() for type-safe URLs:
138
163
  * ```tsx
139
- * <Link to={router.href("shop.products.detail", { slug: "my-product" })}>
164
+ * <Link to={router.reverse("shop.products.detail", { slug: "my-product" })}>
140
165
  * View Product
141
166
  * </Link>
142
167
  * ```
@@ -147,23 +172,45 @@ function isExternalUrl(href: string): boolean {
147
172
  * <Link to="https://example.com">External</Link>
148
173
  * ```
149
174
  */
150
- export const Link: ForwardRefExoticComponent<LinkProps & RefAttributes<HTMLAnchorElement>> = forwardRef<HTMLAnchorElement, LinkProps>(function Link(
175
+ export const Link: ForwardRefExoticComponent<
176
+ LinkProps & RefAttributes<HTMLAnchorElement>
177
+ > = forwardRef<HTMLAnchorElement, LinkProps>(function Link(
151
178
  {
152
179
  to,
153
180
  replace = false,
154
181
  scroll = true,
155
182
  reloadDocument = false,
183
+ revalidate,
156
184
  prefetch = "none",
157
185
  state,
158
186
  children,
159
187
  onClick,
160
188
  ...props
161
189
  },
162
- ref
190
+ ref,
163
191
  ) {
164
192
  const ctx = useContext(NavigationStoreContext);
165
193
  const isExternal = isExternalUrl(to);
166
194
 
195
+ // Resolve adaptive: viewport on touch devices, hover on pointer devices
196
+ const resolvedStrategy =
197
+ prefetch === "adaptive" ? (isTouchDevice ? "viewport" : "hover") : prefetch;
198
+
199
+ // Internal ref for viewport observation; merge with forwarded ref
200
+ const internalRef = useRef<HTMLAnchorElement | null>(null);
201
+ const setRef = useCallback(
202
+ (node: HTMLAnchorElement | null) => {
203
+ internalRef.current = node;
204
+ if (typeof ref === "function") {
205
+ ref(node);
206
+ } else if (ref) {
207
+ (ref as React.MutableRefObject<HTMLAnchorElement | null>).current =
208
+ node;
209
+ }
210
+ },
211
+ [ref],
212
+ );
213
+
167
214
  // Use ref to always get the latest state/getter without adding to useCallback deps
168
215
  // This enables just-in-time state resolution without causing re-renders
169
216
  const stateRef = useRef(state);
@@ -194,43 +241,117 @@ export const Link: ForwardRefExoticComponent<LinkProps & RefAttributes<HTMLAncho
194
241
  const target = (e.currentTarget as HTMLAnchorElement).target;
195
242
  if (target && target !== "_self") return;
196
243
 
244
+ // Hash-only navigation: let the browser handle anchor scrolling natively.
245
+ if (isHashOnlyNavigation(e.currentTarget as HTMLAnchorElement)) {
246
+ return;
247
+ }
248
+
249
+ // No navigation context (outside provider): fall back to native navigation.
250
+ if (!ctx?.navigate) {
251
+ return;
252
+ }
253
+
197
254
  // Prevent default and use SPA navigation
198
255
  e.preventDefault();
199
256
  // Stop propagation to prevent link-interceptor from also handling this
200
257
  e.stopPropagation();
201
258
 
202
- if (ctx?.navigate) {
203
- // Resolve state just-in-time based on format
204
- let resolvedState: unknown;
205
- const currentState = stateRef.current;
206
-
207
- if (Array.isArray(currentState) && currentState.length > 0 && isLocationStateEntry(currentState[0])) {
208
- // Type-safe LocationStateEntry[] - resolve each entry into keyed object
209
- resolvedState = resolveLocationStateEntries(currentState as LocationStateEntry[]);
210
- } else if (typeof currentState === "function") {
211
- // Legacy getter function
212
- resolvedState = currentState();
213
- } else {
214
- // Legacy static value
215
- resolvedState = currentState;
216
- }
259
+ const currentState = stateRef.current;
260
+ let resolvedState: unknown;
217
261
 
218
- ctx.navigate(to, { replace, scroll, state: resolvedState });
262
+ if (
263
+ Array.isArray(currentState) &&
264
+ currentState.length > 0 &&
265
+ isLocationStateEntry(currentState[0])
266
+ ) {
267
+ resolvedState = resolveLocationStateEntries(
268
+ currentState as LocationStateEntry[],
269
+ );
270
+ } else if (typeof currentState === "function") {
271
+ resolvedState = currentState();
272
+ } else if (currentState != null) {
273
+ resolvedState = currentState;
219
274
  }
275
+
276
+ ctx.navigate(to, { replace, scroll, state: resolvedState, revalidate });
220
277
  },
221
- [to, isExternal, reloadDocument, replace, scroll, ctx, onClick]
278
+ [to, isExternal, reloadDocument, replace, scroll, revalidate, ctx, onClick],
222
279
  );
223
280
 
224
281
  const handleMouseEnter = useCallback(() => {
225
- if (prefetch === "hover" && !isExternal && ctx?.store) {
282
+ if (
283
+ (resolvedStrategy === "hover" || resolvedStrategy === "viewport") &&
284
+ !isExternal &&
285
+ ctx?.store
286
+ ) {
287
+ // For "hover", this is the primary prefetch trigger.
288
+ // For "viewport", this upgrades/prioritizes a potentially queued
289
+ // prefetch — prefetchDirect bypasses the queue, and hasPrefetch
290
+ // deduplicates if the viewport prefetch already completed.
291
+ const segmentState = ctx.store.getSegmentState();
292
+ prefetchDirect(to, segmentState.currentSegmentIds, ctx.version);
293
+ }
294
+ }, [resolvedStrategy, to, isExternal, ctx]);
295
+
296
+ // Viewport/render prefetch: waits for idle before starting,
297
+ // uses concurrency-limited queue to avoid flooding.
298
+ useEffect(() => {
299
+ if (isExternal || !ctx?.store) return;
300
+ const isViewport = resolvedStrategy === "viewport";
301
+ const isRender = resolvedStrategy === "render";
302
+ if (!isViewport && !isRender) return;
303
+
304
+ let cancelled = false;
305
+ let unsubIdle: (() => void) | undefined;
306
+ let observedElement: Element | null = null;
307
+
308
+ const triggerPrefetch = () => {
309
+ if (cancelled) return;
226
310
  const segmentState = ctx.store.getSegmentState();
227
- prefetchUrl(to, segmentState.currentSegmentIds);
311
+ prefetchQueued(to, segmentState.currentSegmentIds, ctx.version);
312
+ };
313
+
314
+ // Schedule prefetch only when the app is idle (no navigation/streaming).
315
+ // This avoids competing with hydration and active navigation fetches.
316
+ const scheduleWhenIdle = (callback: () => void) => {
317
+ const state = ctx.eventController.getState();
318
+ if (state.state === "idle" && !state.isStreaming) {
319
+ callback();
320
+ return;
321
+ }
322
+ const unsub = ctx.eventController.subscribe(() => {
323
+ const s = ctx.eventController.getState();
324
+ if (s.state === "idle" && !s.isStreaming) {
325
+ unsub();
326
+ callback();
327
+ }
328
+ });
329
+ unsubIdle = unsub;
330
+ };
331
+
332
+ if (isRender) {
333
+ scheduleWhenIdle(triggerPrefetch);
334
+ } else if (isViewport) {
335
+ const element = internalRef.current;
336
+ if (!element) return;
337
+ observedElement = element;
338
+ observeForPrefetch(element, () => {
339
+ scheduleWhenIdle(triggerPrefetch);
340
+ });
228
341
  }
229
- }, [prefetch, to, isExternal, ctx]);
342
+
343
+ return () => {
344
+ cancelled = true;
345
+ unsubIdle?.();
346
+ if (isViewport && observedElement) {
347
+ unobserveForPrefetch(observedElement);
348
+ }
349
+ };
350
+ }, [resolvedStrategy, to, isExternal, ctx]);
230
351
 
231
352
  return (
232
353
  <a
233
- ref={ref}
354
+ ref={setRef}
234
355
  href={to}
235
356
  onClick={handleClick}
236
357
  onMouseEnter={handleMouseEnter}
@@ -238,11 +359,10 @@ export const Link: ForwardRefExoticComponent<LinkProps & RefAttributes<HTMLAncho
238
359
  data-external={isExternal ? "" : undefined}
239
360
  data-scroll={scroll === false ? "false" : undefined}
240
361
  data-replace={replace ? "true" : undefined}
362
+ data-revalidate={revalidate === false ? "false" : undefined}
241
363
  {...props}
242
364
  >
243
- <LinkContext.Provider value={to}>
244
- {children}
245
- </LinkContext.Provider>
365
+ <LinkContext.Provider value={to}>{children}</LinkContext.Provider>
246
366
  </a>
247
367
  );
248
368
  });
@@ -3,8 +3,10 @@
3
3
  import React, {
4
4
  useState,
5
5
  useEffect,
6
+ useLayoutEffect,
6
7
  useCallback,
7
8
  useMemo,
9
+ useRef,
8
10
  use,
9
11
  type ReactNode,
10
12
  } from "react";
@@ -14,7 +16,7 @@ import {
14
16
  } from "./context.js";
15
17
  import type {
16
18
  NavigationStore,
17
- RscPayload,
19
+ NavigationUpdate,
18
20
  NavigateOptions,
19
21
  NavigationBridge,
20
22
  } from "../types.js";
@@ -22,8 +24,10 @@ import type { EventController } from "../event-controller.js";
22
24
  import { RootErrorBoundary } from "../../root-error-boundary.js";
23
25
  import type { HandleData } from "../types.js";
24
26
  import { ThemeProvider } from "../../theme/ThemeProvider.js";
27
+ import { NonceContext } from "./nonce-context.js";
25
28
  import type { ResolvedThemeConfig, Theme } from "../../theme/types.js";
26
- import { ConnectionWarmup } from "../../warmup/connection-warmup.js";
29
+ import { cancelAllPrefetches } from "../prefetch/queue.js";
30
+ import { handleNavigationEnd } from "../scroll-restoration.js";
27
31
 
28
32
  /**
29
33
  * Process handles from an async generator, updating the event controller
@@ -43,7 +47,7 @@ async function processHandles(
43
47
  matched?: string[];
44
48
  isPartial?: boolean;
45
49
  historyKey: string;
46
- }
50
+ },
47
51
  ): Promise<void> {
48
52
  const { eventController, store, matched, isPartial, historyKey } = opts;
49
53
 
@@ -54,7 +58,7 @@ async function processHandles(
54
58
  // the current route's breadcrumbs (e.g., quick popstate after clicking a link).
55
59
  if (historyKey !== store.getHistoryKey()) {
56
60
  console.log(
57
- "[NavigationProvider] Stopping handle processing - user navigated away"
61
+ "[NavigationProvider] Stopping handle processing - user navigated away",
58
62
  );
59
63
  return;
60
64
  }
@@ -101,9 +105,9 @@ export interface NavigationProviderProps {
101
105
  eventController: EventController;
102
106
 
103
107
  /**
104
- * Initial RSC payload from server
108
+ * Initial rendered tree + metadata from server payload
105
109
  */
106
- initialPayload: RscPayload;
110
+ initialPayload: NavigationUpdate;
107
111
 
108
112
  /**
109
113
  * Navigation bridge for handling navigation
@@ -124,9 +128,15 @@ export interface NavigationProviderProps {
124
128
 
125
129
  /**
126
130
  * Whether connection warmup is enabled.
127
- * When true, renders ConnectionWarmup to keep TLS alive after idle periods.
131
+ * When true, keeps TLS alive by sending HEAD requests after idle periods.
128
132
  */
129
133
  warmupEnabled?: boolean;
134
+
135
+ /**
136
+ * App version from server payload (stable, immutable).
137
+ * Forwarded to prefetch requests for version mismatch detection.
138
+ */
139
+ version?: string;
130
140
  }
131
141
 
132
142
  /**
@@ -158,6 +168,7 @@ export function NavigationProvider({
158
168
  themeConfig,
159
169
  initialTheme,
160
170
  warmupEnabled,
171
+ version,
161
172
  }: NavigationProviderProps): ReactNode {
162
173
  // Track current payload for rendering (this triggers re-renders)
163
174
  const [payload, setPayload] = useState(initialPayload);
@@ -169,7 +180,7 @@ export function NavigationProvider({
169
180
  async (url: string, options?: NavigateOptions): Promise<void> => {
170
181
  await bridge.navigate(url, options);
171
182
  },
172
- []
183
+ [],
173
184
  );
174
185
 
175
186
  /**
@@ -186,18 +197,148 @@ export function NavigationProvider({
186
197
  eventController,
187
198
  navigate,
188
199
  refresh,
200
+ version,
189
201
  }),
190
- []
202
+ [],
191
203
  );
192
204
 
205
+ // Connection warmup: keep TLS alive after idle periods.
206
+ // After 60s of no user interaction, marks connection as "cold".
207
+ // On next interaction or visibility change, sends a HEAD request to warm TLS
208
+ // before the user actually clicks a link.
209
+ useEffect(() => {
210
+ if (!warmupEnabled) return;
211
+
212
+ const IDLE_TIMEOUT = 60_000;
213
+ const DEBOUNCE_DELAY = 150;
214
+
215
+ let idleTimer: ReturnType<typeof setTimeout> | undefined;
216
+ let debounceTimer: ReturnType<typeof setTimeout> | undefined;
217
+ let isCold = false;
218
+ let warmupListenersAttached = false;
219
+
220
+ function sendWarmup() {
221
+ isCold = false;
222
+ fetch("/?_rsc_warmup", { method: "HEAD" }).catch(() => {});
223
+ }
224
+
225
+ function triggerWarmup() {
226
+ if (!isCold) return;
227
+ clearTimeout(debounceTimer);
228
+ debounceTimer = setTimeout(() => {
229
+ sendWarmup();
230
+ detachWarmupListeners();
231
+ resetIdleTimer();
232
+ }, DEBOUNCE_DELAY);
233
+ }
234
+
235
+ function onVisibilityChange() {
236
+ if (document.visibilityState === "visible" && isCold) {
237
+ triggerWarmup();
238
+ }
239
+ }
240
+
241
+ function attachWarmupListeners() {
242
+ if (warmupListenersAttached) return;
243
+ warmupListenersAttached = true;
244
+ document.addEventListener("visibilitychange", onVisibilityChange);
245
+ document.addEventListener("mousemove", triggerWarmup, { once: true });
246
+ document.addEventListener("touchstart", triggerWarmup, { once: true });
247
+ }
248
+
249
+ function detachWarmupListeners() {
250
+ warmupListenersAttached = false;
251
+ document.removeEventListener("visibilitychange", onVisibilityChange);
252
+ document.removeEventListener("mousemove", triggerWarmup);
253
+ document.removeEventListener("touchstart", triggerWarmup);
254
+ }
255
+
256
+ function markCold() {
257
+ isCold = true;
258
+ attachWarmupListeners();
259
+ }
260
+
261
+ function resetIdleTimer() {
262
+ clearTimeout(idleTimer);
263
+ isCold = false;
264
+ idleTimer = setTimeout(markCold, IDLE_TIMEOUT);
265
+ }
266
+
267
+ // Activity events that reset the idle timer
268
+ const activityEvents = [
269
+ "mousemove",
270
+ "keydown",
271
+ "touchstart",
272
+ "scroll",
273
+ ] as const;
274
+ const activityOptions: AddEventListenerOptions = { passive: true };
275
+
276
+ for (const event of activityEvents) {
277
+ document.addEventListener(event, resetIdleTimer, activityOptions);
278
+ }
279
+
280
+ resetIdleTimer();
281
+
282
+ return () => {
283
+ clearTimeout(idleTimer);
284
+ clearTimeout(debounceTimer);
285
+ detachWarmupListeners();
286
+ for (const event of activityEvents) {
287
+ document.removeEventListener(event, resetIdleTimer);
288
+ }
289
+ };
290
+ }, [warmupEnabled]);
291
+
292
+ // Cancel speculative prefetches when navigation starts.
293
+ // Viewport/render prefetches should not compete with navigation fetches.
294
+ useEffect(() => {
295
+ let wasIdle = true;
296
+ const unsub = eventController.subscribe(() => {
297
+ const state = eventController.getState();
298
+ const isIdle = state.state === "idle" && !state.isStreaming;
299
+ if (wasIdle && !isIdle) {
300
+ cancelAllPrefetches();
301
+ }
302
+ wasIdle = isIdle;
303
+ });
304
+ return unsub;
305
+ }, [eventController]);
306
+
307
+ // Pending scroll action to apply after React commits
308
+ const pendingScrollRef = useRef<NavigationUpdate["scroll"]>(undefined);
309
+
310
+ // Apply scroll after React commits the new content to the DOM
311
+ useLayoutEffect(() => {
312
+ const scrollAction = pendingScrollRef.current;
313
+ if (!scrollAction) return;
314
+ pendingScrollRef.current = undefined;
315
+
316
+ if (scrollAction.enabled === false) return;
317
+
318
+ handleNavigationEnd({
319
+ restore: scrollAction.restore,
320
+ scroll: scrollAction.enabled,
321
+ isStreaming: scrollAction.isStreaming,
322
+ });
323
+ });
324
+
193
325
  // Subscribe to UI updates (for re-rendering the tree)
194
326
  useEffect(() => {
195
327
  const unsubscribe = store.onUpdate((update) => {
328
+ // Capture scroll intent — it will be applied in useLayoutEffect
329
+ // after React commits this state update to the DOM.
330
+ // Always assign (even undefined) to clear stale scroll from prior navigations,
331
+ // so server actions or error updates don't accidentally replay old scroll.
332
+ pendingScrollRef.current = update.scroll;
333
+
196
334
  setPayload({
197
335
  root: update.root,
198
336
  metadata: update.metadata,
199
337
  });
200
338
 
339
+ // Update route params
340
+ eventController.setParams(update.metadata.params ?? {});
341
+
201
342
  // Update handle data progressively as it streams in
202
343
  if (update.metadata.handles) {
203
344
  // Capture historyKey now - by the time async processing completes,
@@ -211,7 +352,7 @@ export function NavigationProvider({
211
352
  isPartial: update.metadata.isPartial,
212
353
  historyKey,
213
354
  }).catch((err) =>
214
- console.error("[NavigationProvider] Error consuming handles:", err)
355
+ console.error("[NavigationProvider] Error consuming handles:", err),
215
356
  );
216
357
  } else if (update.metadata.cachedHandleData) {
217
358
  // For back/forward navigation from cache, restore the cached handleData
@@ -219,14 +360,14 @@ export function NavigationProvider({
219
360
  eventController.setHandleData(
220
361
  update.metadata.cachedHandleData,
221
362
  update.metadata.matched,
222
- false // full replace - restore entire cached state
363
+ false, // full replace - restore entire cached state
223
364
  );
224
365
  } else if (update.metadata.matched) {
225
366
  // For cached navigations without handleData, update segmentOrder to clean up stale data
226
367
  eventController.setHandleData(
227
368
  {}, // Empty data - all existing data not in matched will be cleaned up
228
369
  update.metadata.matched,
229
- true // partial update - will clean up segments not in matched
370
+ true, // partial update - will clean up segments not in matched
230
371
  );
231
372
  }
232
373
  });
@@ -257,10 +398,16 @@ export function NavigationProvider({
257
398
  );
258
399
  }
259
400
 
401
+ // Match SSR tree shape: NonceContext.Provider is always present so
402
+ // hydration sees the same component tree. Value is undefined on the
403
+ // client — CSP nonces are a server-side HTML concern.
404
+ content = (
405
+ <NonceContext.Provider value={undefined}>{content}</NonceContext.Provider>
406
+ );
407
+
260
408
  return (
261
409
  <NavigationStoreContext.Provider value={contextValue}>
262
410
  {content}
263
- {warmupEnabled && <ConnectionWarmup />}
264
411
  </NavigationStoreContext.Provider>
265
412
  );
266
413
  }
@@ -41,6 +41,12 @@ export interface NavigationStoreContextValue {
41
41
  * @returns Promise that resolves when refresh is complete
42
42
  */
43
43
  refresh: () => Promise<void>;
44
+
45
+ /**
46
+ * App version from server payload (stable, immutable).
47
+ * Used in prefetch requests for version mismatch detection.
48
+ */
49
+ version: string | undefined;
44
50
  }
45
51
 
46
52
  /**
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Filter segment IDs to only include routes and layouts.
3
+ * Excludes parallels (contain .@) and loaders (contain D followed by digit).
4
+ */
5
+ export function filterSegmentOrder(matched: string[]): string[] {
6
+ return matched.filter((id) => {
7
+ if (id.includes(".@")) return false;
8
+ if (/D\d+\./.test(id)) return false;
9
+ return true;
10
+ });
11
+ }