@rangojs/router 0.0.0-experimental.3 → 0.0.0-experimental.30

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 (297) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +883 -4
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +4655 -747
  5. package/package.json +78 -50
  6. package/skills/cache-guide/SKILL.md +262 -0
  7. package/skills/caching/SKILL.md +54 -25
  8. package/skills/composability/SKILL.md +172 -0
  9. package/skills/debug-manifest/SKILL.md +12 -8
  10. package/skills/document-cache/SKILL.md +23 -21
  11. package/skills/fonts/SKILL.md +167 -0
  12. package/skills/hooks/SKILL.md +390 -63
  13. package/skills/host-router/SKILL.md +218 -0
  14. package/skills/intercept/SKILL.md +133 -10
  15. package/skills/layout/SKILL.md +102 -5
  16. package/skills/links/SKILL.md +239 -0
  17. package/skills/loader/SKILL.md +366 -29
  18. package/skills/middleware/SKILL.md +173 -36
  19. package/skills/mime-routes/SKILL.md +128 -0
  20. package/skills/parallel/SKILL.md +80 -3
  21. package/skills/prerender/SKILL.md +643 -0
  22. package/skills/rango/SKILL.md +86 -16
  23. package/skills/response-routes/SKILL.md +411 -0
  24. package/skills/route/SKILL.md +227 -14
  25. package/skills/router-setup/SKILL.md +225 -32
  26. package/skills/tailwind/SKILL.md +129 -0
  27. package/skills/theme/SKILL.md +12 -11
  28. package/skills/typesafety/SKILL.md +401 -75
  29. package/skills/use-cache/SKILL.md +324 -0
  30. package/src/__internal.ts +10 -4
  31. package/src/bin/rango.ts +321 -0
  32. package/src/browser/action-coordinator.ts +97 -0
  33. package/src/browser/action-response-classifier.ts +99 -0
  34. package/src/browser/event-controller.ts +87 -64
  35. package/src/browser/history-state.ts +80 -0
  36. package/src/browser/intercept-utils.ts +52 -0
  37. package/src/browser/link-interceptor.ts +20 -4
  38. package/src/browser/logging.ts +55 -0
  39. package/src/browser/merge-segment-loaders.ts +20 -12
  40. package/src/browser/navigation-bridge.ts +201 -553
  41. package/src/browser/navigation-client.ts +124 -71
  42. package/src/browser/navigation-store.ts +33 -50
  43. package/src/browser/navigation-transaction.ts +295 -0
  44. package/src/browser/network-error-handler.ts +61 -0
  45. package/src/browser/partial-update.ts +267 -317
  46. package/src/browser/prefetch/cache.ts +146 -0
  47. package/src/browser/prefetch/fetch.ts +135 -0
  48. package/src/browser/prefetch/observer.ts +65 -0
  49. package/src/browser/prefetch/policy.ts +42 -0
  50. package/src/browser/prefetch/queue.ts +88 -0
  51. package/src/browser/rango-state.ts +112 -0
  52. package/src/browser/react/Link.tsx +173 -73
  53. package/src/browser/react/NavigationProvider.tsx +138 -27
  54. package/src/browser/react/context.ts +6 -0
  55. package/src/browser/react/filter-segment-order.ts +11 -0
  56. package/src/browser/react/index.ts +12 -12
  57. package/src/browser/react/location-state-shared.ts +95 -53
  58. package/src/browser/react/location-state.ts +60 -15
  59. package/src/browser/react/mount-context.ts +37 -0
  60. package/src/browser/react/nonce-context.ts +23 -0
  61. package/src/browser/react/shallow-equal.ts +27 -0
  62. package/src/browser/react/use-action.ts +29 -51
  63. package/src/browser/react/use-client-cache.ts +5 -3
  64. package/src/browser/react/use-handle.ts +49 -65
  65. package/src/browser/react/use-href.tsx +20 -188
  66. package/src/browser/react/use-link-status.ts +6 -5
  67. package/src/browser/react/use-mount.ts +31 -0
  68. package/src/browser/react/use-navigation.ts +27 -78
  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 +111 -26
  76. package/src/browser/scroll-restoration.ts +92 -16
  77. package/src/browser/segment-reconciler.ts +216 -0
  78. package/src/browser/segment-structure-assert.ts +83 -0
  79. package/src/browser/server-action-bridge.ts +504 -584
  80. package/src/browser/shallow.ts +6 -1
  81. package/src/browser/types.ts +92 -57
  82. package/src/browser/validate-redirect-origin.ts +29 -0
  83. package/src/build/generate-manifest.ts +438 -0
  84. package/src/build/generate-route-types.ts +36 -0
  85. package/src/build/index.ts +35 -0
  86. package/src/build/route-trie.ts +265 -0
  87. package/src/build/route-types/ast-helpers.ts +25 -0
  88. package/src/build/route-types/ast-route-extraction.ts +98 -0
  89. package/src/build/route-types/codegen.ts +102 -0
  90. package/src/build/route-types/include-resolution.ts +411 -0
  91. package/src/build/route-types/param-extraction.ts +48 -0
  92. package/src/build/route-types/per-module-writer.ts +128 -0
  93. package/src/build/route-types/router-processing.ts +469 -0
  94. package/src/build/route-types/scan-filter.ts +78 -0
  95. package/src/build/runtime-discovery.ts +231 -0
  96. package/src/cache/background-task.ts +34 -0
  97. package/src/cache/cache-key-utils.ts +44 -0
  98. package/src/cache/cache-policy.ts +125 -0
  99. package/src/cache/cache-runtime.ts +338 -0
  100. package/src/cache/cache-scope.ts +120 -303
  101. package/src/cache/cf/cf-cache-store.ts +119 -7
  102. package/src/cache/cf/index.ts +8 -2
  103. package/src/cache/document-cache.ts +101 -72
  104. package/src/cache/handle-capture.ts +81 -0
  105. package/src/cache/handle-snapshot.ts +41 -0
  106. package/src/cache/index.ts +0 -15
  107. package/src/cache/memory-segment-store.ts +191 -13
  108. package/src/cache/profile-registry.ts +73 -0
  109. package/src/cache/read-through-swr.ts +134 -0
  110. package/src/cache/segment-codec.ts +256 -0
  111. package/src/cache/taint.ts +98 -0
  112. package/src/cache/types.ts +72 -122
  113. package/src/client.rsc.tsx +10 -15
  114. package/src/client.tsx +114 -135
  115. package/src/component-utils.ts +4 -4
  116. package/src/components/DefaultDocument.tsx +5 -1
  117. package/src/context-var.ts +86 -0
  118. package/src/debug.ts +17 -7
  119. package/src/errors.ts +108 -2
  120. package/src/handle.ts +34 -19
  121. package/src/handles/MetaTags.tsx +73 -20
  122. package/src/handles/meta.ts +30 -13
  123. package/src/host/cookie-handler.ts +165 -0
  124. package/src/host/errors.ts +97 -0
  125. package/src/host/index.ts +53 -0
  126. package/src/host/pattern-matcher.ts +214 -0
  127. package/src/host/router.ts +352 -0
  128. package/src/host/testing.ts +79 -0
  129. package/src/host/types.ts +146 -0
  130. package/src/host/utils.ts +25 -0
  131. package/src/href-client.ts +135 -49
  132. package/src/index.rsc.ts +182 -17
  133. package/src/index.ts +238 -24
  134. package/src/internal-debug.ts +11 -0
  135. package/src/loader.rsc.ts +27 -142
  136. package/src/loader.ts +27 -10
  137. package/src/network-error-thrower.tsx +3 -1
  138. package/src/outlet-provider.tsx +45 -0
  139. package/src/prerender/param-hash.ts +37 -0
  140. package/src/prerender/store.ts +185 -0
  141. package/src/prerender.ts +463 -0
  142. package/src/reverse.ts +330 -0
  143. package/src/root-error-boundary.tsx +41 -29
  144. package/src/route-content-wrapper.tsx +9 -11
  145. package/src/route-definition/dsl-helpers.ts +934 -0
  146. package/src/route-definition/helper-factories.ts +200 -0
  147. package/src/route-definition/helpers-types.ts +430 -0
  148. package/src/route-definition/index.ts +52 -0
  149. package/src/route-definition/redirect.ts +93 -0
  150. package/src/route-definition.ts +1 -1388
  151. package/src/route-map-builder.ts +241 -112
  152. package/src/route-name.ts +53 -0
  153. package/src/route-types.ts +70 -9
  154. package/src/router/content-negotiation.ts +116 -0
  155. package/src/router/debug-manifest.ts +72 -0
  156. package/src/router/error-handling.ts +9 -9
  157. package/src/router/find-match.ts +158 -0
  158. package/src/router/handler-context.ts +371 -81
  159. package/src/router/intercept-resolution.ts +395 -0
  160. package/src/router/lazy-includes.ts +234 -0
  161. package/src/router/loader-resolution.ts +215 -122
  162. package/src/router/logging.ts +248 -0
  163. package/src/router/manifest.ts +155 -32
  164. package/src/router/match-api.ts +620 -0
  165. package/src/router/match-context.ts +5 -3
  166. package/src/router/match-handlers.ts +440 -0
  167. package/src/router/match-middleware/background-revalidation.ts +80 -93
  168. package/src/router/match-middleware/cache-lookup.ts +382 -9
  169. package/src/router/match-middleware/cache-store.ts +51 -22
  170. package/src/router/match-middleware/intercept-resolution.ts +55 -17
  171. package/src/router/match-middleware/segment-resolution.ts +24 -6
  172. package/src/router/match-pipelines.ts +10 -45
  173. package/src/router/match-result.ts +34 -29
  174. package/src/router/metrics.ts +235 -15
  175. package/src/router/middleware-cookies.ts +55 -0
  176. package/src/router/middleware-types.ts +222 -0
  177. package/src/router/middleware.ts +324 -367
  178. package/src/router/pattern-matching.ts +321 -30
  179. package/src/router/prerender-match.ts +400 -0
  180. package/src/router/preview-match.ts +170 -0
  181. package/src/router/revalidation.ts +137 -38
  182. package/src/router/router-context.ts +36 -21
  183. package/src/router/router-interfaces.ts +452 -0
  184. package/src/router/router-options.ts +592 -0
  185. package/src/router/router-registry.ts +24 -0
  186. package/src/router/segment-resolution/fresh.ts +570 -0
  187. package/src/router/segment-resolution/helpers.ts +263 -0
  188. package/src/router/segment-resolution/loader-cache.ts +198 -0
  189. package/src/router/segment-resolution/revalidation.ts +1241 -0
  190. package/src/router/segment-resolution/static-store.ts +67 -0
  191. package/src/router/segment-resolution.ts +21 -0
  192. package/src/router/segment-wrappers.ts +289 -0
  193. package/src/router/telemetry-otel.ts +299 -0
  194. package/src/router/telemetry.ts +300 -0
  195. package/src/router/timeout.ts +148 -0
  196. package/src/router/trie-matching.ts +239 -0
  197. package/src/router/types.ts +77 -3
  198. package/src/router.ts +688 -3656
  199. package/src/rsc/handler-context.ts +45 -0
  200. package/src/rsc/handler.ts +786 -760
  201. package/src/rsc/helpers.ts +140 -6
  202. package/src/rsc/index.ts +5 -25
  203. package/src/rsc/loader-fetch.ts +209 -0
  204. package/src/rsc/manifest-init.ts +86 -0
  205. package/src/rsc/nonce.ts +14 -0
  206. package/src/rsc/origin-guard.ts +141 -0
  207. package/src/rsc/progressive-enhancement.ts +379 -0
  208. package/src/rsc/response-error.ts +37 -0
  209. package/src/rsc/response-route-handler.ts +347 -0
  210. package/src/rsc/rsc-rendering.ts +235 -0
  211. package/src/rsc/runtime-warnings.ts +42 -0
  212. package/src/rsc/server-action.ts +348 -0
  213. package/src/rsc/ssr-setup.ts +128 -0
  214. package/src/rsc/types.ts +40 -14
  215. package/src/search-params.ts +230 -0
  216. package/src/segment-system.tsx +57 -61
  217. package/src/server/context.ts +202 -51
  218. package/src/server/cookie-store.ts +190 -0
  219. package/src/server/fetchable-loader-store.ts +37 -0
  220. package/src/server/handle-store.ts +94 -15
  221. package/src/server/loader-registry.ts +15 -56
  222. package/src/server/request-context.ts +422 -70
  223. package/src/server.ts +36 -120
  224. package/src/ssr/index.tsx +157 -26
  225. package/src/static-handler.ts +114 -0
  226. package/src/theme/ThemeProvider.tsx +21 -15
  227. package/src/theme/ThemeScript.tsx +5 -5
  228. package/src/theme/constants.ts +5 -2
  229. package/src/theme/index.ts +4 -14
  230. package/src/theme/theme-context.ts +4 -30
  231. package/src/theme/theme-script.ts +21 -18
  232. package/src/types/boundaries.ts +158 -0
  233. package/src/types/cache-types.ts +198 -0
  234. package/src/types/error-types.ts +192 -0
  235. package/src/types/global-namespace.ts +100 -0
  236. package/src/types/handler-context.ts +687 -0
  237. package/src/types/index.ts +88 -0
  238. package/src/types/loader-types.ts +183 -0
  239. package/src/types/route-config.ts +170 -0
  240. package/src/types/route-entry.ts +102 -0
  241. package/src/types/segments.ts +148 -0
  242. package/src/types.ts +1 -1577
  243. package/src/urls/include-helper.ts +197 -0
  244. package/src/urls/index.ts +53 -0
  245. package/src/urls/path-helper-types.ts +339 -0
  246. package/src/urls/path-helper.ts +329 -0
  247. package/src/urls/pattern-types.ts +95 -0
  248. package/src/urls/response-types.ts +106 -0
  249. package/src/urls/type-extraction.ts +372 -0
  250. package/src/urls/urls-function.ts +98 -0
  251. package/src/urls.ts +1 -726
  252. package/src/use-loader.tsx +85 -77
  253. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  254. package/src/vite/discovery/discover-routers.ts +344 -0
  255. package/src/vite/discovery/prerender-collection.ts +385 -0
  256. package/src/vite/discovery/route-types-writer.ts +258 -0
  257. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  258. package/src/vite/discovery/state.ts +110 -0
  259. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  260. package/src/vite/index.ts +11 -782
  261. package/src/vite/plugin-types.ts +131 -0
  262. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  263. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  264. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  265. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
  266. package/src/vite/plugins/expose-id-utils.ts +287 -0
  267. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  268. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  269. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  270. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  271. package/src/vite/plugins/expose-ids/types.ts +45 -0
  272. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  273. package/src/vite/plugins/refresh-cmd.ts +65 -0
  274. package/src/vite/plugins/use-cache-transform.ts +323 -0
  275. package/src/vite/plugins/version-injector.ts +83 -0
  276. package/src/vite/plugins/version-plugin.ts +254 -0
  277. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +29 -15
  278. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  279. package/src/vite/rango.ts +510 -0
  280. package/src/vite/router-discovery.ts +785 -0
  281. package/src/vite/utils/ast-handler-extract.ts +517 -0
  282. package/src/vite/utils/banner.ts +36 -0
  283. package/src/vite/utils/bundle-analysis.ts +137 -0
  284. package/src/vite/utils/manifest-utils.ts +70 -0
  285. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  286. package/src/vite/utils/prerender-utils.ts +189 -0
  287. package/src/vite/utils/shared-utils.ts +169 -0
  288. package/CLAUDE.md +0 -3
  289. package/src/browser/lru-cache.ts +0 -69
  290. package/src/browser/request-controller.ts +0 -164
  291. package/src/cache/memory-store.ts +0 -253
  292. package/src/href-context.ts +0 -33
  293. package/src/href.ts +0 -255
  294. package/src/vite/expose-handle-id.ts +0 -209
  295. package/src/vite/expose-loader-id.ts +0 -357
  296. package/src/vite/expose-location-state-id.ts +0 -177
  297. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -0,0 +1,230 @@
1
+ /**
2
+ * Search parameter schema types and runtime utilities.
3
+ *
4
+ * Provides a lightweight schema system for typed query parameters.
5
+ * When a route defines a `search` schema, ctx.search provides a typed
6
+ * object with parsed values. ctx.searchParams always remains a standard
7
+ * URLSearchParams instance.
8
+ */
9
+
10
+ // ============================================================================
11
+ // Schema Types
12
+ // ============================================================================
13
+
14
+ /** Supported scalar types for search params (append ? for optional). */
15
+ export type SearchSchemaValue =
16
+ | "string"
17
+ | "number"
18
+ | "boolean"
19
+ | "string?"
20
+ | "number?"
21
+ | "boolean?";
22
+
23
+ /** A search schema maps param names to their type descriptors. */
24
+ export type SearchSchema = Record<string, SearchSchemaValue>;
25
+
26
+ // ============================================================================
27
+ // Type-Level Schema Resolution
28
+ // ============================================================================
29
+
30
+ /** Strip trailing `?` from a schema value to get the base type. */
31
+ type BaseType<T extends string> = T extends `${infer B}?` ? B : T;
32
+
33
+ /** Map a base type string to its TypeScript type. */
34
+ type ResolveBaseType<T extends string> = T extends "string"
35
+ ? string
36
+ : T extends "number"
37
+ ? number
38
+ : T extends "boolean"
39
+ ? boolean
40
+ : never;
41
+
42
+ /** Keys whose schema value does NOT end with `?`. */
43
+ type RequiredKeys<T extends SearchSchema> = {
44
+ [K in keyof T]: T[K] extends `${string}?` ? never : K;
45
+ }[keyof T];
46
+
47
+ /** Keys whose schema value ends with `?`. */
48
+ type OptionalKeys<T extends SearchSchema> = {
49
+ [K in keyof T]: T[K] extends `${string}?` ? K : never;
50
+ }[keyof T];
51
+
52
+ /** Flatten an intersection type into a single object type. */
53
+ type Simplify<T> = { [K in keyof T]: T[K] };
54
+
55
+ /**
56
+ * Resolve a SearchSchema to its typed object.
57
+ *
58
+ * Both required and optional params resolve to `T | undefined` at the handler
59
+ * level. The required/optional distinction is a consumer-facing contract
60
+ * (e.g., for href() and reverse() autocomplete) — it tells callers which
61
+ * params the route expects, but the handler must still check for undefined
62
+ * since the framework cannot trust the client to send all required params.
63
+ *
64
+ * @example
65
+ * type S = { q: "string"; page: "number?"; sort: "string?" };
66
+ * type R = ResolveSearchSchema<S>;
67
+ * // { q: string | undefined; page?: number; sort?: string }
68
+ */
69
+ export type ResolveSearchSchema<T extends SearchSchema> = Simplify<
70
+ {
71
+ [K in RequiredKeys<T> & string]:
72
+ | ResolveBaseType<BaseType<T[K]>>
73
+ | undefined;
74
+ } & {
75
+ [K in OptionalKeys<T> & string]?: ResolveBaseType<BaseType<T[K]>>;
76
+ }
77
+ >;
78
+
79
+ // ============================================================================
80
+ // Route-Level Type Extraction
81
+ // ============================================================================
82
+
83
+ /** Resolve the global route map from RegisteredRoutes or GeneratedRouteMap. */
84
+ type GlobalRouteMap = keyof RSCRouter.RegisteredRoutes extends never
85
+ ? keyof RSCRouter.GeneratedRouteMap extends never
86
+ ? Record<string, string>
87
+ : RSCRouter.GeneratedRouteMap
88
+ : RSCRouter.RegisteredRoutes;
89
+
90
+ /**
91
+ * Extract the resolved search params type for a named route.
92
+ * Looks up the search schema from the route map and resolves it.
93
+ *
94
+ * @example
95
+ * ```typescript
96
+ * // Given: path("/search", handler, { name: "search", search: { q: "string", page: "number?" } })
97
+ * type Params = RouteSearchParams<"search">;
98
+ * // { q: string; page?: number }
99
+ * ```
100
+ */
101
+ export type RouteSearchParams<TName extends string, TRouteMap = never> = [
102
+ TRouteMap,
103
+ ] extends [never]
104
+ ? ExtractAndResolveSearch<GlobalRouteMap, TName>
105
+ : ExtractAndResolveSearch<TRouteMap, TName>;
106
+
107
+ type ExtractAndResolveSearch<TRouteMap, TName> = TName extends keyof TRouteMap
108
+ ? TRouteMap[TName] extends { readonly search: infer S extends SearchSchema }
109
+ ? ResolveSearchSchema<S>
110
+ : {}
111
+ : {};
112
+
113
+ /**
114
+ * Extract the route params type for a named route.
115
+ * Looks up the path pattern from the route map and extracts params.
116
+ *
117
+ * @example
118
+ * ```typescript
119
+ * // Given: path("/blog/:slug", handler, { name: "blogPost" })
120
+ * type Params = RouteParams<"blogPost">;
121
+ * // { slug: string }
122
+ * ```
123
+ */
124
+ export type RouteParams<TName extends string, TRouteMap = never> = [
125
+ TRouteMap,
126
+ ] extends [never]
127
+ ? ExtractRouteParamsFromMap<GlobalRouteMap, TName>
128
+ : ExtractRouteParamsFromMap<TRouteMap, TName>;
129
+
130
+ type ExtractRouteParamsFromMap<TRouteMap, TName> = TName extends keyof TRouteMap
131
+ ? TRouteMap[TName] extends string
132
+ ? ExtractParamsFromPattern<TRouteMap[TName]>
133
+ : TRouteMap[TName] extends { readonly path: infer P extends string }
134
+ ? ExtractParamsFromPattern<P>
135
+ : {}
136
+ : {};
137
+
138
+ /** Parse "a|b|c" into "a" | "b" | "c" */
139
+ type ParseConstraint<T extends string> =
140
+ T extends `${infer First}|${infer Rest}` ? First | ParseConstraint<Rest> : T;
141
+
142
+ /** Minimal inline param extraction (avoids importing from types.ts to prevent circular deps). */
143
+ type ExtractParamsFromPattern<T extends string> =
144
+ T extends `${string}:${infer Param}/${infer Rest}`
145
+ ? Param extends `${infer Name}(${infer C})?`
146
+ ? {
147
+ [K in Name]?: ParseConstraint<C>;
148
+ } & ExtractParamsFromPattern<`/${Rest}`>
149
+ : Param extends `${infer Name}(${infer C})`
150
+ ? {
151
+ [K in Name]: ParseConstraint<C>;
152
+ } & ExtractParamsFromPattern<`/${Rest}`>
153
+ : Param extends `${infer Name}?`
154
+ ? { [K in Name]?: string } & ExtractParamsFromPattern<`/${Rest}`>
155
+ : { [K in Param]: string } & ExtractParamsFromPattern<`/${Rest}`>
156
+ : T extends `${string}:${infer Param}`
157
+ ? Param extends `${infer Name}(${infer C})?`
158
+ ? { [K in Name]?: ParseConstraint<C> }
159
+ : Param extends `${infer Name}(${infer C})`
160
+ ? { [K in Name]: ParseConstraint<C> }
161
+ : Param extends `${infer Name}?`
162
+ ? { [K in Name]?: string }
163
+ : { [K in Param]: string }
164
+ : {};
165
+
166
+ // ============================================================================
167
+ // Runtime Parser
168
+ // ============================================================================
169
+
170
+ /**
171
+ * Parse URLSearchParams into a typed object using the given schema.
172
+ *
173
+ * - `"string"` / `"string?"` - kept as-is
174
+ * - `"number"` / `"number?"` - coerced via `Number()`; NaN treated as missing
175
+ * - `"boolean"` / `"boolean?"` - `"true"` / `"1"` -> true, `"false"` / `"0"` / `""` -> false
176
+ *
177
+ * Missing params (both required and optional) are omitted from the result
178
+ * (undefined). The required/optional distinction is a consumer-facing contract
179
+ * only — the handler must check for undefined.
180
+ */
181
+ export function parseSearchParams<T extends SearchSchema>(
182
+ searchParams: URLSearchParams,
183
+ schema: T,
184
+ ): ResolveSearchSchema<T> {
185
+ const result: Record<string, unknown> = {};
186
+
187
+ for (const [key, descriptor] of Object.entries(schema)) {
188
+ const isOptional = descriptor.endsWith("?");
189
+ const baseType = isOptional ? descriptor.slice(0, -1) : descriptor;
190
+ const raw = searchParams.get(key);
191
+
192
+ if (raw === null) {
193
+ // Missing params are omitted (undefined) regardless of required/optional
194
+ continue;
195
+ }
196
+
197
+ if (baseType === "string") {
198
+ result[key] = raw;
199
+ } else if (baseType === "number") {
200
+ const num = Number(raw);
201
+ if (!Number.isNaN(num)) {
202
+ result[key] = num;
203
+ }
204
+ // NaN treated as missing (undefined)
205
+ } else if (baseType === "boolean") {
206
+ result[key] = raw === "true" || raw === "1";
207
+ }
208
+ }
209
+
210
+ return result as ResolveSearchSchema<T>;
211
+ }
212
+
213
+ // ============================================================================
214
+ // Runtime Serializer
215
+ // ============================================================================
216
+
217
+ /**
218
+ * Serialize a typed search params object to a query string (without leading `?`).
219
+ * Skips `undefined` and `null` values.
220
+ */
221
+ export function serializeSearchParams(params: Record<string, unknown>): string {
222
+ const parts: string[] = [];
223
+ for (const [key, value] of Object.entries(params)) {
224
+ if (value === undefined || value === null) continue;
225
+ parts.push(
226
+ `${encodeURIComponent(key)}=${encodeURIComponent(String(value))}`,
227
+ );
228
+ }
229
+ return parts.join("&");
230
+ }
@@ -1,6 +1,7 @@
1
+ import * as React from "react";
1
2
  import { createElement, type ReactNode, type ComponentType } from "react";
2
3
  import { OutletProvider } from "./client.js";
3
- import { HrefProvider } from "./browser/react/use-href.js";
4
+ import { MountContextProvider } from "./browser/react/mount-context.js";
4
5
  import type {
5
6
  ResolvedSegment,
6
7
  LoaderDataResult,
@@ -14,6 +15,11 @@ import {
14
15
  } from "./route-content-wrapper.js";
15
16
  import { RootErrorBoundary } from "./root-error-boundary.js";
16
17
 
18
+ // ViewTransition is only available in React experimental.
19
+ // Access via namespace import to avoid compile-time errors on stable React.
20
+ const ReactViewTransition: any =
21
+ "ViewTransition" in React ? (React as any).ViewTransition : null;
22
+
17
23
  /**
18
24
  * Resolve loader data from raw results, unwrapping LoaderDataResult wrappers
19
25
  */
@@ -84,18 +90,6 @@ export interface RenderSegmentsOptions {
84
90
  * preventing the app shell from unmounting during errors (avoids FOUC).
85
91
  */
86
92
  rootLayout?: ComponentType<RootLayoutProps>;
87
-
88
- /**
89
- * Route map for useHref() during SSR.
90
- * Maps route names to URL patterns.
91
- */
92
- routeMap?: Record<string, string>;
93
-
94
- /**
95
- * Current matched route name for useHref() during SSR.
96
- * Used for local name resolution.
97
- */
98
- routeName?: string;
99
93
  }
100
94
 
101
95
  /**
@@ -146,9 +140,8 @@ export async function renderSegments(
146
140
  interceptSegments,
147
141
  forceAwait,
148
142
  rootLayout: RootLayout,
149
- routeMap,
150
- routeName,
151
143
  } = options || {};
144
+
152
145
  const temporalLazyRefs: Promise<any>[] = [];
153
146
 
154
147
  /**
@@ -221,8 +214,9 @@ export async function renderSegments(
221
214
  if (isAction && component instanceof Promise) {
222
215
  resolvedComponent = await component;
223
216
  }
217
+
224
218
  let nodeContent: ReactNode =
225
- loading || loading === null || resolvedComponent instanceof Promise
219
+ loading !== null && loading !== undefined && loading !== false
226
220
  ? createElement(RouteContentWrapper, {
227
221
  key: `suspense-loading-${id}`,
228
222
  content:
@@ -230,9 +224,22 @@ export async function renderSegments(
230
224
  ? resolvedComponent
231
225
  : Promise.resolve(resolvedComponent),
232
226
  fallback: loading,
227
+ segmentId: id,
233
228
  })
234
229
  : registerLazyRef(resolvedComponent);
235
230
 
231
+ // Wrap with <ViewTransition> if transition config exists (React experimental only).
232
+ // An empty config ({}) creates a bare <ViewTransition> boundary that participates
233
+ // in transitions without adding custom animation classes. Named element-level
234
+ // <ViewTransition> components inside (with name/share props) morph independently
235
+ // from the parent's default cross-fade.
236
+ if (ReactViewTransition && node.segment.transition) {
237
+ nodeContent = createElement(ReactViewTransition, {
238
+ ...node.segment.transition,
239
+ children: nodeContent,
240
+ });
241
+ }
242
+
236
243
  // Common props for OutletProvider
237
244
  const outletContent: ReactNode =
238
245
  node.segment.type === "layout" ? content : null;
@@ -254,7 +261,7 @@ export async function renderSegments(
254
261
  // This ensures cached segments (which may not have loader segments) have the same
255
262
  // tree structure as fresh segments, preventing React remounts
256
263
  // If forceAwait or isAction is set, pre-resolve promises so LoaderBoundary won't suspend
257
- if (loading !== undefined) {
264
+ if (loading !== undefined && loading !== null) {
258
265
  content = createElement(LoaderBoundary, {
259
266
  key: `loader-boundary-${key}`,
260
267
  loaderDataPromise:
@@ -267,11 +274,7 @@ export async function renderSegments(
267
274
  parallel: node.parallel,
268
275
  children: nodeContent,
269
276
  });
270
- continue;
271
- }
272
-
273
- // No loading skeleton defined - use OutletProvider directly
274
- if (loaderEntries.length === 0) {
277
+ } else if (loaderEntries.length === 0) {
275
278
  // No loaders, no loading - simple OutletProvider
276
279
  content = createElement(OutletProvider, {
277
280
  key,
@@ -280,24 +283,34 @@ export async function renderSegments(
280
283
  parallel: node.parallel,
281
284
  children: nodeContent,
282
285
  });
283
- continue;
284
- }
286
+ } else {
287
+ // Has loaders but no loading skeleton - await loaders and render directly
288
+ const resolvedData = await loaderDataPromise;
289
+ const { loaderData, errorFallback } = resolveLoaderData(
290
+ resolvedData,
291
+ loaderIds,
292
+ );
285
293
 
286
- // Has loaders but no loading skeleton - await loaders and render directly
287
- const resolvedData = await loaderDataPromise;
288
- const { loaderData, errorFallback } = resolveLoaderData(
289
- resolvedData,
290
- loaderIds,
291
- );
294
+ content = createElement(OutletProvider, {
295
+ key,
296
+ content: outletContent,
297
+ segment: node.segment,
298
+ parallel: node.parallel,
299
+ loaderData: Object.keys(loaderData).length > 0 ? loaderData : undefined,
300
+ children: errorFallback ?? nodeContent,
301
+ });
302
+ }
292
303
 
293
- content = createElement(OutletProvider, {
294
- key,
295
- content: outletContent,
296
- segment: node.segment,
297
- parallel: node.parallel,
298
- loaderData: Object.keys(loaderData).length > 0 ? loaderData : undefined,
299
- children: errorFallback ?? nodeContent,
300
- });
304
+ // Wrap with MountContextProvider for include() scoped components.
305
+ // Must use MountContextProvider (a proper "use client" export) instead of
306
+ // MountContext.Provider directly, because .Provider is a property on the
307
+ // context object and resolves to undefined through RSC client reference proxies.
308
+ if (node.segment.mountPath) {
309
+ content = createElement(MountContextProvider, {
310
+ value: node.segment.mountPath,
311
+ children: content,
312
+ });
313
+ }
301
314
  }
302
315
 
303
316
  // Always wrap with root error boundary to prevent white screens
@@ -305,7 +318,7 @@ export async function renderSegments(
305
318
  const errorBoundaryWrapped = createElement(RootErrorBoundary, {
306
319
  children: content,
307
320
  });
308
- if (typeof window === "object" || typeof document === "object") {
321
+ if (typeof window === "object") {
309
322
  await Promise.allSettled(temporalLazyRefs);
310
323
  }
311
324
 
@@ -320,16 +333,6 @@ export async function renderSegments(
320
333
  });
321
334
  }
322
335
 
323
- // Wrap with HrefProvider for useHref() to work during SSR
324
- // This provides route info to client components during server rendering
325
- if (routeMap) {
326
- result = createElement(HrefProvider, {
327
- routeMap,
328
- routeName,
329
- children: result,
330
- });
331
- }
332
-
333
336
  return result;
334
337
  }
335
338
 
@@ -423,20 +426,13 @@ function* segmentTreeWalk(
423
426
  }
424
427
  }
425
428
 
426
- // Sort segments by ID to ensure consistent root-to-leaf ordering
427
- // regardless of the order they arrive in the input array (which can differ
428
- // between document requests and actions)
429
- // Shorter IDs come first (closer to root), same length sorted lexicographically
430
- nonParallels.sort((a, b) => {
431
- if (a.id.length !== b.id.length) {
432
- return a.id.length - b.id.length;
433
- }
434
- return a.id.localeCompare(b.id);
435
- });
436
-
437
- // Iterate bottom-to-top using reverse() to process leaf segments first
429
+ // Segments arrive in root-to-leaf order from the server (resolveSegment
430
+ // and resolveSegmentWithRevalidation push segments in this order).
431
+ // All consumers (reconcileSegments, cache) preserve this order.
432
+ // No sorting needed iterate bottom-to-top to process leaf segments first.
438
433
  // This processes route/leaf layouts first, then parent layouts
439
434
  // Note: We reverse the array to iterate from end to start (bottom-to-top)
435
+
440
436
  for (let i = nonParallels.length - 1; i >= 0; i--) {
441
437
  const segment = nonParallels[i];
442
438