@rangojs/router 0.0.0-experimental.10 → 0.0.0-experimental.100

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 (329) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +1037 -4
  3. package/dist/bin/rango.js +1619 -157
  4. package/dist/vite/index.js +5762 -2301
  5. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  6. package/package.json +71 -63
  7. package/skills/breadcrumbs/SKILL.md +252 -0
  8. package/skills/cache-guide/SKILL.md +294 -0
  9. package/skills/caching/SKILL.md +93 -23
  10. package/skills/composability/SKILL.md +172 -0
  11. package/skills/debug-manifest/SKILL.md +12 -8
  12. package/skills/document-cache/SKILL.md +18 -16
  13. package/skills/fonts/SKILL.md +6 -4
  14. package/skills/handler-use/SKILL.md +364 -0
  15. package/skills/hooks/SKILL.md +367 -71
  16. package/skills/host-router/SKILL.md +218 -0
  17. package/skills/i18n/SKILL.md +276 -0
  18. package/skills/intercept/SKILL.md +176 -8
  19. package/skills/layout/SKILL.md +124 -3
  20. package/skills/links/SKILL.md +304 -25
  21. package/skills/loader/SKILL.md +474 -47
  22. package/skills/middleware/SKILL.md +207 -37
  23. package/skills/migrate-nextjs/SKILL.md +562 -0
  24. package/skills/migrate-react-router/SKILL.md +769 -0
  25. package/skills/mime-routes/SKILL.md +15 -11
  26. package/skills/parallel/SKILL.md +272 -1
  27. package/skills/prerender/SKILL.md +467 -65
  28. package/skills/rango/SKILL.md +89 -21
  29. package/skills/response-routes/SKILL.md +152 -91
  30. package/skills/route/SKILL.md +305 -14
  31. package/skills/router-setup/SKILL.md +210 -32
  32. package/skills/server-actions/SKILL.md +739 -0
  33. package/skills/streams-and-websockets/SKILL.md +283 -0
  34. package/skills/theme/SKILL.md +9 -8
  35. package/skills/typesafety/SKILL.md +333 -86
  36. package/skills/use-cache/SKILL.md +324 -0
  37. package/skills/view-transitions/SKILL.md +212 -0
  38. package/src/__internal.ts +102 -4
  39. package/src/bin/rango.ts +312 -15
  40. package/src/browser/action-coordinator.ts +97 -0
  41. package/src/browser/action-response-classifier.ts +99 -0
  42. package/src/browser/app-shell.ts +52 -0
  43. package/src/browser/app-version.ts +14 -0
  44. package/src/browser/event-controller.ts +136 -68
  45. package/src/browser/history-state.ts +80 -0
  46. package/src/browser/intercept-utils.ts +52 -0
  47. package/src/browser/link-interceptor.ts +24 -4
  48. package/src/browser/logging.ts +55 -0
  49. package/src/browser/merge-segment-loaders.ts +20 -12
  50. package/src/browser/navigation-bridge.ts +374 -561
  51. package/src/browser/navigation-client.ts +228 -70
  52. package/src/browser/navigation-store.ts +97 -55
  53. package/src/browser/navigation-transaction.ts +297 -0
  54. package/src/browser/network-error-handler.ts +61 -0
  55. package/src/browser/partial-update.ts +376 -315
  56. package/src/browser/prefetch/cache.ts +314 -0
  57. package/src/browser/prefetch/fetch.ts +282 -0
  58. package/src/browser/prefetch/observer.ts +65 -0
  59. package/src/browser/prefetch/policy.ts +48 -0
  60. package/src/browser/prefetch/queue.ts +191 -0
  61. package/src/browser/prefetch/resource-ready.ts +77 -0
  62. package/src/browser/rango-state.ts +152 -0
  63. package/src/browser/react/Link.tsx +255 -71
  64. package/src/browser/react/NavigationProvider.tsx +152 -24
  65. package/src/browser/react/context.ts +11 -0
  66. package/src/browser/react/filter-segment-order.ts +55 -0
  67. package/src/browser/react/index.ts +15 -12
  68. package/src/browser/react/location-state-shared.ts +95 -53
  69. package/src/browser/react/location-state.ts +60 -15
  70. package/src/browser/react/mount-context.ts +6 -1
  71. package/src/browser/react/nonce-context.ts +23 -0
  72. package/src/browser/react/shallow-equal.ts +27 -0
  73. package/src/browser/react/use-action.ts +29 -51
  74. package/src/browser/react/use-client-cache.ts +5 -3
  75. package/src/browser/react/use-handle.ts +30 -120
  76. package/src/browser/react/use-link-status.ts +6 -5
  77. package/src/browser/react/use-navigation.ts +44 -65
  78. package/src/browser/react/use-params.ts +78 -0
  79. package/src/browser/react/use-pathname.ts +47 -0
  80. package/src/browser/react/use-reverse.ts +99 -0
  81. package/src/browser/react/use-router.ts +83 -0
  82. package/src/browser/react/use-search-params.ts +56 -0
  83. package/src/browser/react/use-segments.ts +85 -99
  84. package/src/browser/response-adapter.ts +73 -0
  85. package/src/browser/rsc-router.tsx +246 -64
  86. package/src/browser/scroll-restoration.ts +127 -52
  87. package/src/browser/segment-reconciler.ts +243 -0
  88. package/src/browser/segment-structure-assert.ts +16 -0
  89. package/src/browser/server-action-bridge.ts +510 -603
  90. package/src/browser/shallow.ts +6 -1
  91. package/src/browser/types.ts +158 -48
  92. package/src/browser/validate-redirect-origin.ts +29 -0
  93. package/src/build/generate-manifest.ts +84 -23
  94. package/src/build/generate-route-types.ts +39 -828
  95. package/src/build/index.ts +4 -5
  96. package/src/build/route-trie.ts +85 -32
  97. package/src/build/route-types/ast-helpers.ts +25 -0
  98. package/src/build/route-types/ast-route-extraction.ts +98 -0
  99. package/src/build/route-types/codegen.ts +102 -0
  100. package/src/build/route-types/include-resolution.ts +418 -0
  101. package/src/build/route-types/param-extraction.ts +48 -0
  102. package/src/build/route-types/per-module-writer.ts +128 -0
  103. package/src/build/route-types/router-processing.ts +618 -0
  104. package/src/build/route-types/scan-filter.ts +85 -0
  105. package/src/build/runtime-discovery.ts +231 -0
  106. package/src/cache/background-task.ts +34 -0
  107. package/src/cache/cache-key-utils.ts +44 -0
  108. package/src/cache/cache-policy.ts +125 -0
  109. package/src/cache/cache-runtime.ts +342 -0
  110. package/src/cache/cache-scope.ts +167 -307
  111. package/src/cache/cf/cf-cache-store.ts +573 -21
  112. package/src/cache/cf/index.ts +13 -3
  113. package/src/cache/document-cache.ts +116 -77
  114. package/src/cache/handle-capture.ts +81 -0
  115. package/src/cache/handle-snapshot.ts +41 -0
  116. package/src/cache/index.ts +1 -15
  117. package/src/cache/memory-segment-store.ts +191 -13
  118. package/src/cache/profile-registry.ts +73 -0
  119. package/src/cache/read-through-swr.ts +134 -0
  120. package/src/cache/segment-codec.ts +256 -0
  121. package/src/cache/taint.ts +153 -0
  122. package/src/cache/types.ts +72 -122
  123. package/src/client.rsc.tsx +6 -1
  124. package/src/client.tsx +118 -302
  125. package/src/component-utils.ts +4 -4
  126. package/src/components/DefaultDocument.tsx +5 -1
  127. package/src/context-var.ts +156 -0
  128. package/src/debug.ts +19 -9
  129. package/src/errors.ts +77 -7
  130. package/src/handle.ts +55 -10
  131. package/src/handles/MetaTags.tsx +73 -20
  132. package/src/handles/breadcrumbs.ts +66 -0
  133. package/src/handles/index.ts +1 -0
  134. package/src/handles/meta.ts +30 -13
  135. package/src/host/cookie-handler.ts +21 -15
  136. package/src/host/errors.ts +8 -8
  137. package/src/host/index.ts +4 -7
  138. package/src/host/pattern-matcher.ts +27 -27
  139. package/src/host/router.ts +61 -39
  140. package/src/host/testing.ts +8 -8
  141. package/src/host/types.ts +15 -7
  142. package/src/host/utils.ts +1 -1
  143. package/src/href-client.ts +65 -45
  144. package/src/index.rsc.ts +138 -21
  145. package/src/index.ts +206 -51
  146. package/src/internal-debug.ts +11 -0
  147. package/src/loader.rsc.ts +25 -143
  148. package/src/loader.ts +27 -10
  149. package/src/network-error-thrower.tsx +3 -1
  150. package/src/outlet-context.ts +1 -1
  151. package/src/outlet-provider.tsx +45 -0
  152. package/src/prerender/param-hash.ts +4 -2
  153. package/src/prerender/store.ts +159 -13
  154. package/src/prerender.ts +397 -29
  155. package/src/response-utils.ts +28 -0
  156. package/src/reverse.ts +231 -121
  157. package/src/root-error-boundary.tsx +41 -29
  158. package/src/route-content-wrapper.tsx +7 -4
  159. package/src/route-definition/dsl-helpers.ts +1134 -0
  160. package/src/route-definition/helper-factories.ts +200 -0
  161. package/src/route-definition/helpers-types.ts +483 -0
  162. package/src/route-definition/index.ts +55 -0
  163. package/src/route-definition/redirect.ts +101 -0
  164. package/src/route-definition/resolve-handler-use.ts +155 -0
  165. package/src/route-definition.ts +1 -1431
  166. package/src/route-map-builder.ts +162 -123
  167. package/src/route-name.ts +53 -0
  168. package/src/route-types.ts +66 -9
  169. package/src/router/content-negotiation.ts +215 -0
  170. package/src/router/debug-manifest.ts +72 -0
  171. package/src/router/error-handling.ts +9 -9
  172. package/src/router/find-match.ts +160 -0
  173. package/src/router/handler-context.ts +418 -86
  174. package/src/router/intercept-resolution.ts +35 -20
  175. package/src/router/lazy-includes.ts +237 -0
  176. package/src/router/loader-resolution.ts +359 -128
  177. package/src/router/logging.ts +251 -0
  178. package/src/router/manifest.ts +98 -32
  179. package/src/router/match-api.ts +196 -261
  180. package/src/router/match-context.ts +4 -2
  181. package/src/router/match-handlers.ts +441 -0
  182. package/src/router/match-middleware/background-revalidation.ts +108 -93
  183. package/src/router/match-middleware/cache-lookup.ts +415 -86
  184. package/src/router/match-middleware/cache-store.ts +91 -29
  185. package/src/router/match-middleware/intercept-resolution.ts +48 -21
  186. package/src/router/match-middleware/segment-resolution.ts +73 -9
  187. package/src/router/match-pipelines.ts +10 -45
  188. package/src/router/match-result.ts +154 -35
  189. package/src/router/metrics.ts +240 -15
  190. package/src/router/middleware-cookies.ts +55 -0
  191. package/src/router/middleware-types.ts +209 -0
  192. package/src/router/middleware.ts +373 -371
  193. package/src/router/navigation-snapshot.ts +182 -0
  194. package/src/router/pattern-matching.ts +292 -52
  195. package/src/router/prerender-match.ts +502 -0
  196. package/src/router/preview-match.ts +98 -0
  197. package/src/router/request-classification.ts +310 -0
  198. package/src/router/revalidation.ts +152 -39
  199. package/src/router/route-snapshot.ts +245 -0
  200. package/src/router/router-context.ts +41 -21
  201. package/src/router/router-interfaces.ts +484 -0
  202. package/src/router/router-options.ts +618 -0
  203. package/src/router/router-registry.ts +24 -0
  204. package/src/router/segment-resolution/fresh.ts +756 -0
  205. package/src/router/segment-resolution/helpers.ts +268 -0
  206. package/src/router/segment-resolution/loader-cache.ts +199 -0
  207. package/src/router/segment-resolution/revalidation.ts +1407 -0
  208. package/src/router/segment-resolution/static-store.ts +67 -0
  209. package/src/router/segment-resolution.ts +21 -1315
  210. package/src/router/segment-wrappers.ts +291 -0
  211. package/src/router/substitute-pattern-params.ts +56 -0
  212. package/src/router/telemetry-otel.ts +299 -0
  213. package/src/router/telemetry.ts +300 -0
  214. package/src/router/timeout.ts +148 -0
  215. package/src/router/trie-matching.ts +111 -39
  216. package/src/router/types.ts +17 -9
  217. package/src/router/url-params.ts +49 -0
  218. package/src/router.ts +642 -2011
  219. package/src/rsc/handler-context.ts +45 -0
  220. package/src/rsc/handler.ts +864 -1114
  221. package/src/rsc/helpers.ts +181 -19
  222. package/src/rsc/index.ts +0 -20
  223. package/src/rsc/loader-fetch.ts +229 -0
  224. package/src/rsc/manifest-init.ts +90 -0
  225. package/src/rsc/nonce.ts +14 -0
  226. package/src/rsc/origin-guard.ts +141 -0
  227. package/src/rsc/progressive-enhancement.ts +395 -0
  228. package/src/rsc/response-error.ts +37 -0
  229. package/src/rsc/response-route-handler.ts +360 -0
  230. package/src/rsc/rsc-rendering.ts +256 -0
  231. package/src/rsc/runtime-warnings.ts +42 -0
  232. package/src/rsc/server-action.ts +360 -0
  233. package/src/rsc/ssr-setup.ts +128 -0
  234. package/src/rsc/types.ts +52 -11
  235. package/src/search-params.ts +230 -0
  236. package/src/segment-content-promise.ts +67 -0
  237. package/src/segment-loader-promise.ts +122 -0
  238. package/src/segment-system.tsx +187 -38
  239. package/src/server/context.ts +333 -59
  240. package/src/server/cookie-store.ts +190 -0
  241. package/src/server/fetchable-loader-store.ts +37 -0
  242. package/src/server/handle-store.ts +113 -15
  243. package/src/server/loader-registry.ts +24 -64
  244. package/src/server/request-context.ts +603 -109
  245. package/src/server.ts +35 -155
  246. package/src/ssr/index.tsx +107 -30
  247. package/src/static-handler.ts +126 -0
  248. package/src/theme/ThemeProvider.tsx +21 -15
  249. package/src/theme/ThemeScript.tsx +5 -5
  250. package/src/theme/constants.ts +5 -2
  251. package/src/theme/index.ts +4 -14
  252. package/src/theme/theme-context.ts +4 -30
  253. package/src/theme/theme-script.ts +21 -18
  254. package/src/types/boundaries.ts +158 -0
  255. package/src/types/cache-types.ts +198 -0
  256. package/src/types/error-types.ts +192 -0
  257. package/src/types/global-namespace.ts +100 -0
  258. package/src/types/handler-context.ts +764 -0
  259. package/src/types/index.ts +88 -0
  260. package/src/types/loader-types.ts +209 -0
  261. package/src/types/request-scope.ts +126 -0
  262. package/src/types/route-config.ts +170 -0
  263. package/src/types/route-entry.ts +120 -0
  264. package/src/types/segments.ts +167 -0
  265. package/src/types.ts +1 -1757
  266. package/src/urls/include-helper.ts +207 -0
  267. package/src/urls/index.ts +53 -0
  268. package/src/urls/path-helper-types.ts +372 -0
  269. package/src/urls/path-helper.ts +364 -0
  270. package/src/urls/pattern-types.ts +107 -0
  271. package/src/urls/response-types.ts +108 -0
  272. package/src/urls/type-extraction.ts +372 -0
  273. package/src/urls/urls-function.ts +98 -0
  274. package/src/urls.ts +1 -1282
  275. package/src/use-loader.tsx +161 -81
  276. package/src/vite/debug.ts +184 -0
  277. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  278. package/src/vite/discovery/discover-routers.ts +376 -0
  279. package/src/vite/discovery/gate-state.ts +171 -0
  280. package/src/vite/discovery/prerender-collection.ts +486 -0
  281. package/src/vite/discovery/route-types-writer.ts +258 -0
  282. package/src/vite/discovery/self-gen-tracking.ts +73 -0
  283. package/src/vite/discovery/state.ts +117 -0
  284. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  285. package/src/vite/index.ts +15 -2063
  286. package/src/vite/plugin-types.ts +103 -0
  287. package/src/vite/plugins/cjs-to-esm.ts +98 -0
  288. package/src/vite/plugins/client-ref-dedup.ts +131 -0
  289. package/src/vite/plugins/client-ref-hashing.ts +117 -0
  290. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  291. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  292. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  293. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +107 -64
  294. package/src/vite/plugins/expose-id-utils.ts +299 -0
  295. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  296. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  297. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  298. package/src/vite/plugins/expose-ids/router-transform.ts +127 -0
  299. package/src/vite/plugins/expose-ids/types.ts +45 -0
  300. package/src/vite/plugins/expose-internal-ids.ts +816 -0
  301. package/src/vite/plugins/performance-tracks.ts +96 -0
  302. package/src/vite/plugins/refresh-cmd.ts +127 -0
  303. package/src/vite/plugins/use-cache-transform.ts +336 -0
  304. package/src/vite/plugins/version-injector.ts +109 -0
  305. package/src/vite/plugins/version-plugin.ts +266 -0
  306. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  307. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  308. package/src/vite/rango.ts +497 -0
  309. package/src/vite/router-discovery.ts +1423 -0
  310. package/src/vite/utils/ast-handler-extract.ts +517 -0
  311. package/src/vite/utils/banner.ts +36 -0
  312. package/src/vite/utils/bundle-analysis.ts +137 -0
  313. package/src/vite/utils/manifest-utils.ts +70 -0
  314. package/src/vite/utils/package-resolution.ts +161 -0
  315. package/src/vite/utils/prerender-utils.ts +222 -0
  316. package/src/vite/utils/shared-utils.ts +170 -0
  317. package/CLAUDE.md +0 -43
  318. package/src/browser/lru-cache.ts +0 -69
  319. package/src/browser/request-controller.ts +0 -164
  320. package/src/cache/memory-store.ts +0 -253
  321. package/src/href-context.ts +0 -33
  322. package/src/router.gen.ts +0 -6
  323. package/src/urls.gen.ts +0 -8
  324. package/src/vite/expose-handle-id.ts +0 -209
  325. package/src/vite/expose-loader-id.ts +0 -426
  326. package/src/vite/expose-location-state-id.ts +0 -177
  327. package/src/vite/expose-prerender-handler-id.ts +0 -429
  328. package/src/vite/package-resolution.ts +0 -125
  329. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -9,36 +9,10 @@ import {
9
9
  useRef,
10
10
  } from "react";
11
11
  import { NavigationStoreContext } from "./context.js";
12
- import type { PublicNavigationState, NavigateOptions } from "../types.js";
12
+ import { shallowEqual } from "./shallow-equal.js";
13
+ import type { PublicNavigationState } from "../types.js";
13
14
  import type { DerivedNavigationState } from "../event-controller.js";
14
15
 
15
- /**
16
- * Shallow equality check for selector results
17
- */
18
- function shallowEqual<T>(a: T, b: T): boolean {
19
- if (Object.is(a, b)) return true;
20
- if (
21
- typeof a !== "object" ||
22
- a === null ||
23
- typeof b !== "object" ||
24
- b === null
25
- ) {
26
- return false;
27
- }
28
- const keysA = Object.keys(a);
29
- const keysB = Object.keys(b);
30
- if (keysA.length !== keysB.length) return false;
31
- for (const key of keysA) {
32
- if (
33
- !Object.hasOwn(b, key) ||
34
- !Object.is((a as any)[key], (b as any)[key])
35
- ) {
36
- return false;
37
- }
38
- }
39
- return true;
40
- }
41
-
42
16
  /**
43
17
  * Convert derived state to public version (strips inflightActions)
44
18
  */
@@ -47,45 +21,29 @@ function toPublicState(state: DerivedNavigationState): PublicNavigationState {
47
21
  return publicState;
48
22
  }
49
23
 
50
-
51
- /**
52
- * Navigation methods returned by useNavigation
53
- */
54
- export interface NavigationMethods {
55
- navigate: (url: string, options?: NavigateOptions) => Promise<void>;
56
- refresh: () => Promise<void>;
57
- }
58
-
59
24
  /**
60
- * Full value returned when no selector is provided
61
- */
62
- export type NavigationValue = PublicNavigationState & NavigationMethods;
63
-
64
- /**
65
- * Hook to access navigation state with optional selector for performance
25
+ * Hook to access reactive navigation state with optional selector for performance.
66
26
  *
67
- * Uses the event controller for reactive state management.
68
- * State is derived from source of truth (currentNavigation, inflightActions).
27
+ * Returns state only. For actions (push, replace, refresh, prefetch),
28
+ * use useRouter() instead.
69
29
  *
70
30
  * @example
71
31
  * ```tsx
72
- * const state = useNavigation(nav => nav.state);
32
+ * const { state, location } = useNavigation();
73
33
  * const isLoading = useNavigation(nav => nav.state === 'loading');
74
34
  * ```
75
35
  */
76
- export function useNavigation(): NavigationValue;
36
+ export function useNavigation(): PublicNavigationState;
77
37
  export function useNavigation<T>(
78
38
  selector: (state: PublicNavigationState) => T,
79
39
  ): T;
80
40
  export function useNavigation<T>(
81
41
  selector?: (state: PublicNavigationState) => T,
82
- ): T | NavigationValue {
42
+ ): T | PublicNavigationState {
83
43
  const ctx = useContext(NavigationStoreContext);
84
44
 
85
45
  if (!ctx) {
86
- throw new Error(
87
- "useNavigation must be used within NavigationStoreContext.Provider"
88
- );
46
+ throw new Error("useNavigation must be used within NavigationProvider");
89
47
  }
90
48
 
91
49
  // Base state for useOptimistic
@@ -95,16 +53,32 @@ export function useNavigation<T>(
95
53
  });
96
54
  const prevState = useRef(baseValue);
97
55
 
56
+ // Tracks whether the most recent setOptimisticValue call pinned the value
57
+ // to a non-idle state. Used to decide whether to emit a release update when
58
+ // returning to idle, so the optimistic store doesn't stay pinned if a
59
+ // parent transition (e.g. <Link> click) is still pending.
60
+ const optimisticPinnedRef = useRef(false);
61
+
98
62
  // useOptimistic allows immediate updates during transitions/actions
99
63
  const [value, setOptimisticValue] = useOptimistic(baseValue);
100
64
 
65
+ // Store selector in a ref so the subscription callback always uses the
66
+ // latest selector without re-subscribing on every render (inline functions
67
+ // have a new identity each render). This is event-driven by design: the
68
+ // value updates when the store emits, not when the selector changes.
69
+ // Between events there is nothing new to select from.
70
+ const selectorRef = useRef(selector);
71
+ selectorRef.current = selector;
72
+
101
73
  // Subscribe to event controller state changes (only runs on client)
102
74
  useEffect(() => {
103
75
  // Subscribe to updates from event controller
104
76
  return ctx.eventController.subscribe(() => {
105
77
  const currentState = ctx.eventController.getState();
106
78
  const publicState = toPublicState(currentState);
107
- const nextSelected = selector ? selector(publicState) : publicState;
79
+ const nextSelected = selectorRef.current
80
+ ? selectorRef.current(publicState)
81
+ : publicState;
108
82
 
109
83
  // Check if selected value has changed
110
84
  if (!shallowEqual(nextSelected, prevState.current)) {
@@ -114,27 +88,32 @@ export function useNavigation<T>(
114
88
  const hasInflightActions =
115
89
  ctx.eventController.getInflightActions().size > 0;
116
90
 
117
- if (hasInflightActions || publicState.state !== "idle") {
118
- // Use optimistic update for immediate feedback during transitions
91
+ const shouldPin = hasInflightActions || publicState.state !== "idle";
92
+
93
+ if (shouldPin) {
94
+ // Pin the optimistic store so the loading value shows immediately
95
+ // even if a parent transition (e.g. <Link> click) defers the
96
+ // urgent setBaseValue commit.
119
97
  startTransition(() => {
120
98
  setOptimisticValue(nextSelected);
121
99
  });
100
+ optimisticPinnedRef.current = true;
101
+ } else if (optimisticPinnedRef.current) {
102
+ // Release a previously-pinned optimistic value. Without this,
103
+ // useOptimistic keeps returning the stale loading value while
104
+ // any parent transition is still pending, even after baseValue
105
+ // flipped to idle.
106
+ startTransition(() => {
107
+ setOptimisticValue(nextSelected);
108
+ });
109
+ optimisticPinnedRef.current = false;
122
110
  }
123
111
 
124
112
  // Always update base state so UI reflects current state
125
113
  setBaseValue(nextSelected);
126
114
  }
127
115
  });
128
- }, [selector]);
129
-
130
- // If no selector, include navigation methods
131
- if (!selector) {
132
- return {
133
- ...(value as PublicNavigationState),
134
- navigate: ctx.navigate,
135
- refresh: ctx.refresh,
136
- };
137
- }
116
+ }, []);
138
117
 
139
- return value as T;
118
+ return value as T | PublicNavigationState;
140
119
  }
@@ -0,0 +1,78 @@
1
+ "use client";
2
+
3
+ import { useContext, useState, useEffect, useRef } from "react";
4
+ import { NavigationStoreContext } from "./context.js";
5
+ import { shallowEqual } from "./shallow-equal.js";
6
+
7
+ /**
8
+ * Hook to access the current route params.
9
+ *
10
+ * Returns the merged route params from the matched route.
11
+ * Updates when navigation completes, not during pending navigation.
12
+ *
13
+ * @example
14
+ * ```tsx
15
+ * // Route: /products/:productId
16
+ * const params = useParams();
17
+ * // { productId: "123" }
18
+ *
19
+ * // Annotate the expected shape via a generic
20
+ * const { productId } = useParams<{ productId: string }>();
21
+ *
22
+ * // With selector
23
+ * const productId = useParams(p => p.productId);
24
+ * ```
25
+ */
26
+ // `T extends object` (not `Record<string, string | undefined>`) so that
27
+ // interface shapes pass the constraint — interfaces lack an implicit
28
+ // index signature and would otherwise be rejected. The generic is a
29
+ // shape annotation, not a runtime check; the body always returns the
30
+ // underlying params map unchanged. The default and selector input use
31
+ // `string | undefined` because absent optional params are omitted from
32
+ // the params record at runtime — the type must reflect that so callers
33
+ // don't write `p.locale.length` and crash when the segment is absent.
34
+ export function useParams<
35
+ T extends object = Record<string, string | undefined>,
36
+ >(): Readonly<T>;
37
+ export function useParams<T>(
38
+ selector: (params: Record<string, string | undefined>) => T,
39
+ ): T;
40
+ export function useParams<T>(
41
+ selector?: (params: Record<string, string | undefined>) => T,
42
+ ): T | Record<string, string | undefined> {
43
+ const ctx = useContext(NavigationStoreContext);
44
+
45
+ const [value, setValue] = useState<T | Record<string, string>>(() => {
46
+ if (!ctx) {
47
+ return selector ? selector({}) : {};
48
+ }
49
+ const params = ctx.eventController.getParams();
50
+ return selector ? selector(params) : params;
51
+ });
52
+
53
+ const prevValue = useRef(value);
54
+ // Ref keeps the latest selector without re-subscribing. Event-driven by
55
+ // design: value updates on store events, not on selector identity change.
56
+ const selectorRef = useRef(selector);
57
+ selectorRef.current = selector;
58
+
59
+ useEffect(() => {
60
+ if (!ctx) return;
61
+
62
+ const update = () => {
63
+ const params = ctx.eventController.getParams();
64
+ const next = selectorRef.current ? selectorRef.current(params) : params;
65
+
66
+ if (!shallowEqual(next, prevValue.current)) {
67
+ prevValue.current = next;
68
+ setValue(next);
69
+ }
70
+ };
71
+
72
+ update();
73
+
74
+ return ctx.eventController.subscribe(update);
75
+ }, []);
76
+
77
+ return value;
78
+ }
@@ -0,0 +1,47 @@
1
+ "use client";
2
+
3
+ import { useContext, useState, useEffect, useRef } from "react";
4
+ import { NavigationStoreContext } from "./context.js";
5
+
6
+ /**
7
+ * Hook to access the current pathname.
8
+ *
9
+ * Returns the committed pathname string (excludes search params and hash).
10
+ * Updates when navigation completes, not during pending navigation.
11
+ *
12
+ * @example
13
+ * ```tsx
14
+ * const pathname = usePathname();
15
+ * // "/products/123"
16
+ * ```
17
+ */
18
+ export function usePathname(): string {
19
+ const ctx = useContext(NavigationStoreContext);
20
+
21
+ const [pathname, setPathname] = useState<string>(() => {
22
+ if (!ctx) {
23
+ return "/";
24
+ }
25
+ return (ctx.eventController.getState().location as URL).pathname;
26
+ });
27
+
28
+ const prevPathname = useRef(pathname);
29
+
30
+ useEffect(() => {
31
+ if (!ctx) return;
32
+
33
+ const update = () => {
34
+ const next = (ctx.eventController.getState().location as URL).pathname;
35
+ if (next !== prevPathname.current) {
36
+ prevPathname.current = next;
37
+ setPathname(next);
38
+ }
39
+ };
40
+
41
+ update();
42
+
43
+ return ctx.eventController.subscribe(update);
44
+ }, []);
45
+
46
+ return pathname;
47
+ }
@@ -0,0 +1,99 @@
1
+ "use client";
2
+
3
+ import { useCallback } from "react";
4
+ import type { LocalReverseFunction } from "../../reverse.js";
5
+ import { substitutePatternParams } from "../../router/substitute-pattern-params.js";
6
+ import { serializeSearchParams } from "../../search-params.js";
7
+ import { useMount } from "./use-mount.js";
8
+ import { useParams } from "./use-params.js";
9
+
10
+ type RouteEntry = string | { readonly path: string };
11
+ type LocalRouteMap = Readonly<Record<string, RouteEntry>>;
12
+
13
+ function getPattern(entry: RouteEntry | undefined): string | undefined {
14
+ if (entry === undefined) return undefined;
15
+ return typeof entry === "string" ? entry : entry.path;
16
+ }
17
+
18
+ /**
19
+ * Join an include mount prefix with a mount-relative pattern.
20
+ *
21
+ * `pattern === "/"` is the index of the local module — under a non-root
22
+ * mount it must collapse so `/` under `/blog` becomes `/blog`, not
23
+ * `/blog/`. This matches `ctx.reverse(".index")` on the server.
24
+ */
25
+ function joinMount(mount: string, pattern: string): string {
26
+ if (pattern === "/") {
27
+ if (mount === "" || mount === "/") return "/";
28
+ return mount.endsWith("/") ? mount.slice(0, -1) : mount;
29
+ }
30
+ const normalizedMount = mount === "/" ? "" : mount.replace(/\/+$/, "");
31
+ return normalizedMount + pattern;
32
+ }
33
+
34
+ /**
35
+ * Mount-aware reverse function for a locally-imported `routes` map.
36
+ *
37
+ * Resolves dot-prefixed route names against the passed `routes` (typically
38
+ * a generated `routes` from a `urls()` module's `.gen.ts`), prefixes the
39
+ * result with the surrounding `include()` mount path, and substitutes
40
+ * params — auto-filling from the current matched route's params and
41
+ * letting explicit params override.
42
+ *
43
+ * @example
44
+ * ```tsx
45
+ * "use client";
46
+ * import { Link, useReverse } from "@rangojs/router/client";
47
+ * import { routes as blogRoutes } from "../urls/blog.gen.js";
48
+ *
49
+ * function BlogNav() {
50
+ * const reverse = useReverse(blogRoutes);
51
+ * return (
52
+ * <>
53
+ * <Link to={reverse(".index")}>Blog</Link>
54
+ * <Link to={reverse(".post", { postId: "hello" })}>Post</Link>
55
+ * </>
56
+ * );
57
+ * }
58
+ * ```
59
+ */
60
+ export function useReverse<const TRoutes extends LocalRouteMap>(
61
+ routes: TRoutes,
62
+ ): LocalReverseFunction<TRoutes> {
63
+ const mount = useMount();
64
+ const currentParams = useParams();
65
+
66
+ return useCallback(
67
+ ((
68
+ name: string,
69
+ explicitParams?: Record<string, string | undefined>,
70
+ search?: Record<string, unknown>,
71
+ ): string => {
72
+ if (!name.startsWith(".")) {
73
+ throw new Error(`Local route names must start with ".": "${name}"`);
74
+ }
75
+ const lookupName = name.slice(1);
76
+ const entry = (routes as LocalRouteMap)[lookupName];
77
+ const pattern = getPattern(entry);
78
+ if (pattern === undefined) {
79
+ throw new Error(`Unknown local route: "${name}"`);
80
+ }
81
+
82
+ const joined = joinMount(mount, pattern);
83
+
84
+ const mergedParams = explicitParams
85
+ ? { ...currentParams, ...explicitParams }
86
+ : currentParams;
87
+
88
+ const substituted = substitutePatternParams(joined, mergedParams, name);
89
+
90
+ if (search) {
91
+ const qs = serializeSearchParams(search);
92
+ if (qs) return `${substituted}?${qs}`;
93
+ }
94
+
95
+ return substituted;
96
+ }) as LocalReverseFunction<TRoutes>,
97
+ [routes, mount, currentParams],
98
+ );
99
+ }
@@ -0,0 +1,83 @@
1
+ "use client";
2
+
3
+ import { useContext, useMemo } from "react";
4
+ import { NavigationStoreContext } from "./context.js";
5
+ import { prefetchDirect } from "../prefetch/fetch.js";
6
+ import { getAppVersion } from "../app-version.js";
7
+ import type { RouterInstance, RouterNavigateOptions } from "../types.js";
8
+
9
+ /**
10
+ * Hook to access router actions (push, replace, refresh, prefetch, back, forward).
11
+ *
12
+ * Returns a STABLE reference that never changes, so components using
13
+ * useRouter() do not re-render on navigation state changes.
14
+ * For reactive navigation state, use useNavigation() instead.
15
+ *
16
+ * Methods read `basename` from the live context on each call so that
17
+ * cross-app navigation (app-switch) sees the current app's basename
18
+ * rather than the one captured at mount time.
19
+ *
20
+ * @example
21
+ * ```tsx
22
+ * const router = useRouter();
23
+ * router.push("/products");
24
+ * router.replace("/login", { scroll: false });
25
+ * router.prefetch("/dashboard");
26
+ * router.back();
27
+ * ```
28
+ */
29
+ export function useRouter(): RouterInstance {
30
+ const ctx = useContext(NavigationStoreContext);
31
+
32
+ if (!ctx) {
33
+ throw new Error("useRouter must be used within NavigationProvider");
34
+ }
35
+
36
+ // Stable reference: ctx itself is stable, and reads on each method call
37
+ // pick up live basename values from the context (backed by a live ref
38
+ // in NavigationProvider), so app-switch transitions are reflected without
39
+ // recreating this object.
40
+ return useMemo<RouterInstance>(() => {
41
+ /** Prefix a root-relative path with basename if not already prefixed. */
42
+ function withBasename(url: string): string {
43
+ const bn = ctx!.basename;
44
+ if (!bn || !url.startsWith("/") || url.startsWith(bn + "/") || url === bn)
45
+ return url;
46
+ return url === "/" ? bn : bn + url;
47
+ }
48
+
49
+ return {
50
+ push(url: string, options?: RouterNavigateOptions): Promise<void> {
51
+ return ctx.navigate(withBasename(url), { ...options, replace: false });
52
+ },
53
+
54
+ replace(url: string, options?: RouterNavigateOptions): Promise<void> {
55
+ return ctx.navigate(withBasename(url), { ...options, replace: true });
56
+ },
57
+
58
+ refresh(): Promise<void> {
59
+ return ctx.refresh();
60
+ },
61
+
62
+ prefetch(url: string): void {
63
+ const segmentState = ctx.store?.getSegmentState();
64
+ if (segmentState) {
65
+ prefetchDirect(
66
+ withBasename(url),
67
+ segmentState.currentSegmentIds,
68
+ getAppVersion(),
69
+ ctx.store?.getRouterId?.(),
70
+ );
71
+ }
72
+ },
73
+
74
+ back(): void {
75
+ window.history.back();
76
+ },
77
+
78
+ forward(): void {
79
+ window.history.forward();
80
+ },
81
+ };
82
+ }, []);
83
+ }
@@ -0,0 +1,56 @@
1
+ "use client";
2
+
3
+ import { useContext, useState, useEffect, useRef } from "react";
4
+ import { NavigationStoreContext } from "./context.js";
5
+ import type { ReadonlyURLSearchParams } from "../types.js";
6
+
7
+ /**
8
+ * Hook to access the current URL search params.
9
+ *
10
+ * Returns a read-only URLSearchParams object from the committed location.
11
+ * Updates when navigation completes, not during pending navigation.
12
+ *
13
+ * Note: During SSR the search params are not available (the server only sends
14
+ * the pathname). The hook returns empty params during SSR and syncs from
15
+ * the browser URL on mount.
16
+ *
17
+ * @example
18
+ * ```tsx
19
+ * const searchParams = useSearchParams();
20
+ * const query = searchParams.get("q"); // "react"
21
+ * const page = searchParams.get("page"); // "2"
22
+ * ```
23
+ */
24
+ export function useSearchParams(): ReadonlyURLSearchParams {
25
+ const ctx = useContext(NavigationStoreContext);
26
+
27
+ // Always initialize with empty URLSearchParams to match SSR output
28
+ // and avoid hydration mismatch. The useEffect below syncs from
29
+ // the real URL after hydration.
30
+ const [searchParams, setSearchParams] = useState<ReadonlyURLSearchParams>(
31
+ () => new URLSearchParams(),
32
+ );
33
+
34
+ const prevSearch = useRef("");
35
+
36
+ useEffect(() => {
37
+ if (!ctx) return;
38
+
39
+ const update = () => {
40
+ const location = ctx.eventController.getState().location as URL;
41
+ const nextSearch = location.searchParams.toString();
42
+ if (nextSearch !== prevSearch.current) {
43
+ prevSearch.current = nextSearch;
44
+ // Create a snapshot so callers cannot mutate the source URLSearchParams
45
+ setSearchParams(new URLSearchParams(nextSearch));
46
+ }
47
+ };
48
+
49
+ // Sync on mount (picks up search params from browser URL)
50
+ update();
51
+
52
+ return ctx.eventController.subscribe(update);
53
+ }, []);
54
+
55
+ return searchParams;
56
+ }