@timber-js/app 0.2.0-alpha.7 → 0.2.0-alpha.71

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 (500) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-BJARkOcu.js} +1 -1
  3. package/dist/_chunks/als-registry-BJARkOcu.js.map +1 -0
  4. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  5. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  6. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  7. package/dist/_chunks/define-CGuYoRHU.js +199 -0
  8. package/dist/_chunks/define-CGuYoRHU.js.map +1 -0
  9. package/dist/_chunks/define-Dz1bqwaS.js +106 -0
  10. package/dist/_chunks/define-Dz1bqwaS.js.map +1 -0
  11. package/dist/_chunks/define-cookie-B5mewxwM.js +93 -0
  12. package/dist/_chunks/define-cookie-B5mewxwM.js.map +1 -0
  13. package/dist/_chunks/error-boundary-D9hzsveV.js +216 -0
  14. package/dist/_chunks/error-boundary-D9hzsveV.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → format-Rn922VH2.js} +3 -20
  16. package/dist/_chunks/format-Rn922VH2.js.map +1 -0
  17. package/dist/_chunks/{tracing-Cwn7697K.js → handler-store-BVePM7hp.js} +68 -3
  18. package/dist/_chunks/handler-store-BVePM7hp.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-CEdHHviP.js} +171 -97
  20. package/dist/_chunks/interception-CEdHHviP.js.map +1 -0
  21. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  22. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  23. package/dist/_chunks/{request-context-DIkVh_jG.js → request-context-CywiO4jV.js} +181 -69
  24. package/dist/_chunks/request-context-CywiO4jV.js.map +1 -0
  25. package/dist/_chunks/schema-bridge-C4SwjCQD.js +86 -0
  26. package/dist/_chunks/schema-bridge-C4SwjCQD.js.map +1 -0
  27. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  28. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  29. package/dist/_chunks/segment-context-hzuJ048X.js +72 -0
  30. package/dist/_chunks/segment-context-hzuJ048X.js.map +1 -0
  31. package/dist/_chunks/stale-reload-BLUC_Pl_.js +64 -0
  32. package/dist/_chunks/stale-reload-BLUC_Pl_.js.map +1 -0
  33. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-DAhgj8Gx.js} +1 -1
  34. package/dist/_chunks/use-query-states-DAhgj8Gx.js.map +1 -0
  35. package/dist/_chunks/wrappers-LZbghvn0.js +63 -0
  36. package/dist/_chunks/wrappers-LZbghvn0.js.map +1 -0
  37. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  38. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  39. package/dist/adapters/cloudflare-dev.js +73 -0
  40. package/dist/adapters/cloudflare-dev.js.map +1 -0
  41. package/dist/adapters/cloudflare.d.ts +148 -12
  42. package/dist/adapters/cloudflare.d.ts.map +1 -1
  43. package/dist/adapters/cloudflare.js +135 -11
  44. package/dist/adapters/cloudflare.js.map +1 -1
  45. package/dist/adapters/compress-module.d.ts.map +1 -1
  46. package/dist/adapters/nitro.d.ts +17 -1
  47. package/dist/adapters/nitro.d.ts.map +1 -1
  48. package/dist/adapters/nitro.js +56 -13
  49. package/dist/adapters/nitro.js.map +1 -1
  50. package/dist/cache/cache-api.d.ts +24 -0
  51. package/dist/cache/cache-api.d.ts.map +1 -0
  52. package/dist/cache/fast-hash.d.ts +22 -0
  53. package/dist/cache/fast-hash.d.ts.map +1 -0
  54. package/dist/cache/handler-store.d.ts +31 -0
  55. package/dist/cache/handler-store.d.ts.map +1 -0
  56. package/dist/cache/index.d.ts +7 -5
  57. package/dist/cache/index.d.ts.map +1 -1
  58. package/dist/cache/index.js +111 -73
  59. package/dist/cache/index.js.map +1 -1
  60. package/dist/cache/singleflight.d.ts +18 -1
  61. package/dist/cache/singleflight.d.ts.map +1 -1
  62. package/dist/cache/timber-cache.d.ts +1 -1
  63. package/dist/cache/timber-cache.d.ts.map +1 -1
  64. package/dist/client/error-boundary.d.ts +12 -5
  65. package/dist/client/error-boundary.d.ts.map +1 -1
  66. package/dist/client/error-boundary.js +1 -125
  67. package/dist/client/error-reconstituter.d.ts +54 -0
  68. package/dist/client/error-reconstituter.d.ts.map +1 -0
  69. package/dist/client/form.d.ts +2 -2
  70. package/dist/client/form.d.ts.map +1 -1
  71. package/dist/client/history.d.ts +19 -4
  72. package/dist/client/history.d.ts.map +1 -1
  73. package/dist/client/index.d.ts +6 -5
  74. package/dist/client/index.d.ts.map +1 -1
  75. package/dist/client/index.js +537 -166
  76. package/dist/client/index.js.map +1 -1
  77. package/dist/client/link-pending-store.d.ts +78 -0
  78. package/dist/client/link-pending-store.d.ts.map +1 -0
  79. package/dist/client/link.d.ts +90 -32
  80. package/dist/client/link.d.ts.map +1 -1
  81. package/dist/client/nav-link-store.d.ts +36 -0
  82. package/dist/client/nav-link-store.d.ts.map +1 -0
  83. package/dist/client/navigation-api-types.d.ts +90 -0
  84. package/dist/client/navigation-api-types.d.ts.map +1 -0
  85. package/dist/client/navigation-api.d.ts +115 -0
  86. package/dist/client/navigation-api.d.ts.map +1 -0
  87. package/dist/client/navigation-context.d.ts +13 -2
  88. package/dist/client/navigation-context.d.ts.map +1 -1
  89. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  90. package/dist/client/navigation-root.d.ts.map +1 -0
  91. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  92. package/dist/client/router.d.ts +70 -4
  93. package/dist/client/router.d.ts.map +1 -1
  94. package/dist/client/rsc-fetch.d.ts +38 -3
  95. package/dist/client/rsc-fetch.d.ts.map +1 -1
  96. package/dist/client/segment-cache.d.ts +1 -1
  97. package/dist/client/segment-cache.d.ts.map +1 -1
  98. package/dist/client/segment-context.d.ts +1 -1
  99. package/dist/client/segment-context.d.ts.map +1 -1
  100. package/dist/client/segment-merger.d.ts.map +1 -1
  101. package/dist/client/segment-outlet.d.ts +63 -0
  102. package/dist/client/segment-outlet.d.ts.map +1 -0
  103. package/dist/client/ssr-data.d.ts +13 -4
  104. package/dist/client/ssr-data.d.ts.map +1 -1
  105. package/dist/client/stale-reload.d.ts +15 -0
  106. package/dist/client/stale-reload.d.ts.map +1 -1
  107. package/dist/client/top-loader.d.ts +3 -3
  108. package/dist/client/top-loader.d.ts.map +1 -1
  109. package/dist/client/use-params.d.ts +6 -4
  110. package/dist/client/use-params.d.ts.map +1 -1
  111. package/dist/client/use-query-states.d.ts +1 -1
  112. package/dist/client/use-query-states.d.ts.map +1 -1
  113. package/dist/codec.d.ts +23 -0
  114. package/dist/codec.d.ts.map +1 -0
  115. package/dist/codec.js +2 -0
  116. package/dist/cookies/define-cookie.d.ts +35 -14
  117. package/dist/cookies/define-cookie.d.ts.map +1 -1
  118. package/dist/cookies/index.d.ts +2 -0
  119. package/dist/cookies/index.d.ts.map +1 -1
  120. package/dist/cookies/index.js +3 -84
  121. package/dist/fonts/css.d.ts +1 -0
  122. package/dist/fonts/css.d.ts.map +1 -1
  123. package/dist/index.d.ts +154 -38
  124. package/dist/index.d.ts.map +1 -1
  125. package/dist/index.js +12092 -11916
  126. package/dist/index.js.map +1 -1
  127. package/dist/plugins/adapter-build.d.ts +1 -1
  128. package/dist/plugins/adapter-build.d.ts.map +1 -1
  129. package/dist/plugins/build-manifest.d.ts +2 -2
  130. package/dist/plugins/build-manifest.d.ts.map +1 -1
  131. package/dist/plugins/build-report.d.ts +3 -3
  132. package/dist/plugins/build-report.d.ts.map +1 -1
  133. package/dist/plugins/client-chunks.d.ts +32 -0
  134. package/dist/plugins/client-chunks.d.ts.map +1 -0
  135. package/dist/plugins/content.d.ts +1 -1
  136. package/dist/plugins/content.d.ts.map +1 -1
  137. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  138. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  139. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  140. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  141. package/dist/plugins/dev-logs.d.ts +1 -1
  142. package/dist/plugins/dev-logs.d.ts.map +1 -1
  143. package/dist/plugins/dev-server.d.ts +1 -1
  144. package/dist/plugins/dev-server.d.ts.map +1 -1
  145. package/dist/plugins/entries.d.ts +1 -1
  146. package/dist/plugins/entries.d.ts.map +1 -1
  147. package/dist/plugins/fonts.d.ts +19 -5
  148. package/dist/plugins/fonts.d.ts.map +1 -1
  149. package/dist/plugins/mdx.d.ts +1 -1
  150. package/dist/plugins/mdx.d.ts.map +1 -1
  151. package/dist/plugins/routing.d.ts +1 -1
  152. package/dist/plugins/routing.d.ts.map +1 -1
  153. package/dist/plugins/server-bundle.d.ts.map +1 -1
  154. package/dist/plugins/shims.d.ts +6 -5
  155. package/dist/plugins/shims.d.ts.map +1 -1
  156. package/dist/plugins/static-build.d.ts +1 -1
  157. package/dist/plugins/static-build.d.ts.map +1 -1
  158. package/dist/routing/codegen.d.ts +2 -2
  159. package/dist/routing/codegen.d.ts.map +1 -1
  160. package/dist/routing/index.d.ts +2 -0
  161. package/dist/routing/index.d.ts.map +1 -1
  162. package/dist/routing/index.js +3 -2
  163. package/dist/routing/scanner.d.ts.map +1 -1
  164. package/dist/routing/segment-classify.d.ts +46 -0
  165. package/dist/routing/segment-classify.d.ts.map +1 -0
  166. package/dist/routing/status-file-lint.d.ts +2 -1
  167. package/dist/routing/status-file-lint.d.ts.map +1 -1
  168. package/dist/routing/types.d.ts +16 -4
  169. package/dist/routing/types.d.ts.map +1 -1
  170. package/dist/rsc-runtime/rsc.d.ts +1 -1
  171. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  172. package/dist/rsc-runtime/ssr.d.ts +12 -0
  173. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  174. package/dist/schema-bridge.d.ts +76 -0
  175. package/dist/schema-bridge.d.ts.map +1 -0
  176. package/dist/search-params/define.d.ts +139 -0
  177. package/dist/search-params/define.d.ts.map +1 -0
  178. package/dist/search-params/index.d.ts +4 -6
  179. package/dist/search-params/index.d.ts.map +1 -1
  180. package/dist/search-params/index.js +4 -474
  181. package/dist/search-params/registry.d.ts +1 -1
  182. package/dist/search-params/wrappers.d.ts +53 -0
  183. package/dist/search-params/wrappers.d.ts.map +1 -0
  184. package/dist/segment-params/define.d.ts +78 -0
  185. package/dist/segment-params/define.d.ts.map +1 -0
  186. package/dist/segment-params/index.d.ts +7 -0
  187. package/dist/segment-params/index.d.ts.map +1 -0
  188. package/dist/segment-params/index.js +4 -0
  189. package/dist/server/access-gate.d.ts +4 -0
  190. package/dist/server/access-gate.d.ts.map +1 -1
  191. package/dist/server/action-client.d.ts +12 -1
  192. package/dist/server/action-client.d.ts.map +1 -1
  193. package/dist/server/action-encryption.d.ts +76 -0
  194. package/dist/server/action-encryption.d.ts.map +1 -0
  195. package/dist/server/action-handler.d.ts.map +1 -1
  196. package/dist/server/actions.d.ts +3 -6
  197. package/dist/server/actions.d.ts.map +1 -1
  198. package/dist/server/als-registry.d.ts +32 -4
  199. package/dist/server/als-registry.d.ts.map +1 -1
  200. package/dist/server/build-manifest.d.ts +2 -2
  201. package/dist/server/build-manifest.d.ts.map +1 -1
  202. package/dist/server/debug.d.ts +1 -1
  203. package/dist/server/default-logger.d.ts +22 -0
  204. package/dist/server/default-logger.d.ts.map +1 -0
  205. package/dist/server/deny-page-resolver.d.ts +52 -0
  206. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  207. package/dist/server/deny-renderer.d.ts.map +1 -1
  208. package/dist/server/dev-warnings.d.ts +0 -14
  209. package/dist/server/dev-warnings.d.ts.map +1 -1
  210. package/dist/server/early-hints.d.ts +13 -5
  211. package/dist/server/early-hints.d.ts.map +1 -1
  212. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  213. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  214. package/dist/server/fallback-error.d.ts +4 -3
  215. package/dist/server/fallback-error.d.ts.map +1 -1
  216. package/dist/server/flight-injection-state.d.ts +66 -0
  217. package/dist/server/flight-injection-state.d.ts.map +1 -0
  218. package/dist/server/flight-scripts.d.ts +42 -0
  219. package/dist/server/flight-scripts.d.ts.map +1 -0
  220. package/dist/server/flush.d.ts.map +1 -1
  221. package/dist/server/form-data.d.ts +29 -0
  222. package/dist/server/form-data.d.ts.map +1 -1
  223. package/dist/server/html-injectors.d.ts +51 -11
  224. package/dist/server/html-injectors.d.ts.map +1 -1
  225. package/dist/server/index.d.ts +5 -3
  226. package/dist/server/index.d.ts.map +1 -1
  227. package/dist/server/index.js +2176 -1663
  228. package/dist/server/index.js.map +1 -1
  229. package/dist/server/logger.d.ts +25 -7
  230. package/dist/server/logger.d.ts.map +1 -1
  231. package/dist/server/middleware-runner.d.ts +19 -4
  232. package/dist/server/middleware-runner.d.ts.map +1 -1
  233. package/dist/server/node-stream-transforms.d.ts +113 -0
  234. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  235. package/dist/server/page-deny-boundary.d.ts +31 -0
  236. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  237. package/dist/server/pipeline-interception.d.ts +1 -1
  238. package/dist/server/pipeline-interception.d.ts.map +1 -1
  239. package/dist/server/pipeline-metadata.d.ts +6 -0
  240. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  241. package/dist/server/pipeline.d.ts +32 -10
  242. package/dist/server/pipeline.d.ts.map +1 -1
  243. package/dist/server/primitives.d.ts +30 -3
  244. package/dist/server/primitives.d.ts.map +1 -1
  245. package/dist/server/render-timeout.d.ts +51 -0
  246. package/dist/server/render-timeout.d.ts.map +1 -0
  247. package/dist/server/request-context.d.ts +76 -37
  248. package/dist/server/request-context.d.ts.map +1 -1
  249. package/dist/server/route-element-builder.d.ts +27 -1
  250. package/dist/server/route-element-builder.d.ts.map +1 -1
  251. package/dist/server/route-handler.d.ts.map +1 -1
  252. package/dist/server/route-matcher.d.ts +9 -2
  253. package/dist/server/route-matcher.d.ts.map +1 -1
  254. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  255. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  256. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  257. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  258. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  259. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  260. package/dist/server/rsc-entry/index.d.ts +8 -3
  261. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  262. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  263. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  264. package/dist/server/rsc-entry/rsc-stream.d.ts +10 -1
  265. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  266. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  267. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  268. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  269. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  270. package/dist/server/safe-load.d.ts +46 -0
  271. package/dist/server/safe-load.d.ts.map +1 -0
  272. package/dist/server/sitemap-generator.d.ts +129 -0
  273. package/dist/server/sitemap-generator.d.ts.map +1 -0
  274. package/dist/server/sitemap-handler.d.ts +22 -0
  275. package/dist/server/sitemap-handler.d.ts.map +1 -0
  276. package/dist/server/slot-resolver.d.ts +1 -1
  277. package/dist/server/slot-resolver.d.ts.map +1 -1
  278. package/dist/server/ssr-entry.d.ts +22 -0
  279. package/dist/server/ssr-entry.d.ts.map +1 -1
  280. package/dist/server/ssr-render.d.ts +39 -21
  281. package/dist/server/ssr-render.d.ts.map +1 -1
  282. package/dist/server/ssr-wrappers.d.ts +50 -0
  283. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  284. package/dist/server/status-code-resolver.d.ts +1 -1
  285. package/dist/server/status-code-resolver.d.ts.map +1 -1
  286. package/dist/server/stream-utils.d.ts +36 -0
  287. package/dist/server/stream-utils.d.ts.map +1 -0
  288. package/dist/server/tracing.d.ts +10 -0
  289. package/dist/server/tracing.d.ts.map +1 -1
  290. package/dist/server/tree-builder.d.ts +22 -19
  291. package/dist/server/tree-builder.d.ts.map +1 -1
  292. package/dist/server/types.d.ts +1 -4
  293. package/dist/server/types.d.ts.map +1 -1
  294. package/dist/server/version-skew.d.ts +61 -0
  295. package/dist/server/version-skew.d.ts.map +1 -0
  296. package/dist/server/waituntil-bridge.d.ts.map +1 -1
  297. package/dist/shared/merge-search-params.d.ts +22 -0
  298. package/dist/shared/merge-search-params.d.ts.map +1 -0
  299. package/dist/shims/font-google.d.ts +1 -1
  300. package/dist/shims/font-google.d.ts.map +1 -1
  301. package/dist/shims/font-google.js +42 -0
  302. package/dist/shims/font-google.js.map +1 -0
  303. package/dist/shims/font-local.d.ts +26 -0
  304. package/dist/shims/font-local.d.ts.map +1 -0
  305. package/dist/shims/font-local.js +20 -0
  306. package/dist/shims/font-local.js.map +1 -0
  307. package/dist/shims/navigation-client.d.ts +1 -1
  308. package/dist/shims/navigation-client.d.ts.map +1 -1
  309. package/dist/shims/navigation.d.ts +1 -1
  310. package/dist/shims/navigation.d.ts.map +1 -1
  311. package/dist/utils/directive-parser.d.ts +5 -2
  312. package/dist/utils/directive-parser.d.ts.map +1 -1
  313. package/dist/utils/state-machine.d.ts +80 -0
  314. package/dist/utils/state-machine.d.ts.map +1 -0
  315. package/package.json +37 -17
  316. package/src/adapters/cloudflare-dev.ts +177 -0
  317. package/src/adapters/cloudflare.ts +342 -28
  318. package/src/adapters/compress-module.ts +24 -4
  319. package/src/adapters/nitro.ts +58 -9
  320. package/src/adapters/wrangler.d.ts +7 -0
  321. package/src/cache/cache-api.ts +38 -0
  322. package/src/cache/fast-hash.ts +34 -0
  323. package/src/cache/handler-store.ts +68 -0
  324. package/src/cache/index.ts +9 -5
  325. package/src/cache/singleflight.ts +62 -4
  326. package/src/cache/timber-cache.ts +40 -29
  327. package/src/cli.ts +0 -0
  328. package/src/client/browser-entry.ts +314 -142
  329. package/src/client/error-boundary.tsx +48 -16
  330. package/src/client/error-reconstituter.tsx +65 -0
  331. package/src/client/form.tsx +2 -2
  332. package/src/client/history.ts +26 -4
  333. package/src/client/index.ts +13 -4
  334. package/src/client/link-pending-store.ts +136 -0
  335. package/src/client/link.tsx +346 -105
  336. package/src/client/nav-link-store.ts +47 -0
  337. package/src/client/navigation-api-types.ts +112 -0
  338. package/src/client/navigation-api.ts +332 -0
  339. package/src/client/navigation-context.ts +27 -6
  340. package/src/client/navigation-root.tsx +346 -0
  341. package/src/client/nuqs-adapter.tsx +16 -3
  342. package/src/client/router.ts +302 -77
  343. package/src/client/rsc-fetch.ts +93 -5
  344. package/src/client/segment-cache.ts +1 -1
  345. package/src/client/segment-context.ts +6 -1
  346. package/src/client/segment-merger.ts +2 -8
  347. package/src/client/segment-outlet.tsx +86 -0
  348. package/src/client/ssr-data.ts +13 -5
  349. package/src/client/stale-reload.ts +73 -6
  350. package/src/client/top-loader.tsx +22 -13
  351. package/src/client/use-navigation-pending.ts +1 -1
  352. package/src/client/use-params.ts +7 -5
  353. package/src/client/use-query-states.ts +2 -2
  354. package/src/codec.ts +34 -0
  355. package/src/cookies/define-cookie.ts +72 -21
  356. package/src/cookies/index.ts +7 -0
  357. package/src/fonts/css.ts +2 -1
  358. package/src/index.ts +328 -92
  359. package/src/plugins/adapter-build.ts +8 -2
  360. package/src/plugins/build-manifest.ts +13 -2
  361. package/src/plugins/build-report.ts +3 -3
  362. package/src/plugins/client-chunks.ts +65 -0
  363. package/src/plugins/content.ts +1 -1
  364. package/src/plugins/dev-browser-logs.ts +288 -0
  365. package/src/plugins/dev-error-overlay.ts +70 -1
  366. package/src/plugins/dev-logs.ts +1 -1
  367. package/src/plugins/dev-server.ts +55 -9
  368. package/src/plugins/entries.ts +70 -9
  369. package/src/plugins/fonts.ts +167 -61
  370. package/src/plugins/mdx.ts +1 -1
  371. package/src/plugins/routing.ts +57 -17
  372. package/src/plugins/server-action-exports.ts +1 -1
  373. package/src/plugins/server-bundle.ts +32 -1
  374. package/src/plugins/shims.ts +76 -33
  375. package/src/plugins/static-build.ts +10 -6
  376. package/src/routing/codegen.ts +165 -105
  377. package/src/routing/index.ts +2 -0
  378. package/src/routing/scanner.ts +93 -23
  379. package/src/routing/segment-classify.ts +89 -0
  380. package/src/routing/status-file-lint.ts +3 -2
  381. package/src/routing/types.ts +17 -4
  382. package/src/rsc-runtime/rsc.ts +2 -0
  383. package/src/rsc-runtime/ssr.ts +50 -0
  384. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  385. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  386. package/src/search-params/define.ts +482 -0
  387. package/src/search-params/index.ts +13 -19
  388. package/src/search-params/registry.ts +1 -1
  389. package/src/search-params/wrappers.ts +85 -0
  390. package/src/segment-params/define.ts +279 -0
  391. package/src/segment-params/index.ts +28 -0
  392. package/src/server/access-gate.tsx +70 -29
  393. package/src/server/action-client.ts +28 -3
  394. package/src/server/action-encryption.ts +144 -0
  395. package/src/server/action-handler.ts +20 -3
  396. package/src/server/actions.ts +10 -9
  397. package/src/server/als-registry.ts +32 -4
  398. package/src/server/build-manifest.ts +10 -4
  399. package/src/server/compress.ts +25 -7
  400. package/src/server/debug.ts +1 -1
  401. package/src/server/default-logger.ts +99 -0
  402. package/src/server/deny-page-resolver.ts +154 -0
  403. package/src/server/deny-renderer.ts +24 -38
  404. package/src/server/dev-warnings.ts +2 -28
  405. package/src/server/early-hints.ts +36 -15
  406. package/src/server/error-boundary-wrapper.ts +74 -22
  407. package/src/server/fallback-error.ts +31 -15
  408. package/src/server/flight-injection-state.ts +113 -0
  409. package/src/server/flight-scripts.ts +62 -0
  410. package/src/server/flush.ts +2 -1
  411. package/src/server/form-data.ts +76 -0
  412. package/src/server/html-injectors.ts +277 -117
  413. package/src/server/index.ts +9 -5
  414. package/src/server/logger.ts +44 -36
  415. package/src/server/middleware-runner.ts +31 -4
  416. package/src/server/node-stream-transforms.ts +509 -0
  417. package/src/server/page-deny-boundary.tsx +56 -0
  418. package/src/server/pipeline-interception.ts +17 -16
  419. package/src/server/pipeline-metadata.ts +13 -0
  420. package/src/server/pipeline.ts +195 -51
  421. package/src/server/primitives.ts +47 -5
  422. package/src/server/render-timeout.ts +108 -0
  423. package/src/server/request-context.ts +240 -117
  424. package/src/server/route-element-builder.ts +284 -197
  425. package/src/server/route-handler.ts +24 -4
  426. package/src/server/route-matcher.ts +24 -20
  427. package/src/server/rsc-entry/api-handler.ts +15 -16
  428. package/src/server/rsc-entry/error-renderer.ts +300 -89
  429. package/src/server/rsc-entry/helpers.ts +134 -5
  430. package/src/server/rsc-entry/index.ts +202 -113
  431. package/src/server/rsc-entry/rsc-payload.ts +100 -21
  432. package/src/server/rsc-entry/rsc-stream.ts +74 -18
  433. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  434. package/src/server/rsc-entry/ssr-renderer.ts +173 -40
  435. package/src/server/safe-load.ts +60 -0
  436. package/src/server/sitemap-generator.ts +338 -0
  437. package/src/server/sitemap-handler.ts +126 -0
  438. package/src/server/slot-resolver.ts +243 -228
  439. package/src/server/ssr-entry.ts +211 -32
  440. package/src/server/ssr-render.ts +289 -67
  441. package/src/server/ssr-wrappers.tsx +139 -0
  442. package/src/server/status-code-resolver.ts +1 -1
  443. package/src/server/stream-utils.ts +213 -0
  444. package/src/server/tracing.ts +37 -3
  445. package/src/server/tree-builder.ts +92 -58
  446. package/src/server/types.ts +3 -6
  447. package/src/server/version-skew.ts +104 -0
  448. package/src/server/waituntil-bridge.ts +4 -1
  449. package/src/shared/merge-search-params.ts +55 -0
  450. package/src/shims/font-google.ts +1 -1
  451. package/src/shims/font-local.ts +34 -0
  452. package/src/shims/navigation-client.ts +1 -1
  453. package/src/shims/navigation.ts +2 -1
  454. package/src/utils/directive-parser.ts +5 -2
  455. package/src/utils/state-machine.ts +111 -0
  456. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  457. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  458. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  459. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  460. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  461. package/dist/_chunks/ssr-data-MjmprTmO.js +0 -88
  462. package/dist/_chunks/ssr-data-MjmprTmO.js.map +0 -1
  463. package/dist/_chunks/tracing-Cwn7697K.js.map +0 -1
  464. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  465. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  466. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  467. package/dist/cache/register-cached-function.d.ts +0 -17
  468. package/dist/cache/register-cached-function.d.ts.map +0 -1
  469. package/dist/client/error-boundary.js.map +0 -1
  470. package/dist/client/link-status-provider.d.ts +0 -11
  471. package/dist/client/link-status-provider.d.ts.map +0 -1
  472. package/dist/client/transition-root.d.ts.map +0 -1
  473. package/dist/cookies/index.js.map +0 -1
  474. package/dist/plugins/cache-transform.d.ts +0 -36
  475. package/dist/plugins/cache-transform.d.ts.map +0 -1
  476. package/dist/plugins/dynamic-transform.d.ts +0 -72
  477. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  478. package/dist/search-params/analyze.d.ts +0 -54
  479. package/dist/search-params/analyze.d.ts.map +0 -1
  480. package/dist/search-params/builtin-codecs.d.ts +0 -105
  481. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  482. package/dist/search-params/codecs.d.ts +0 -53
  483. package/dist/search-params/codecs.d.ts.map +0 -1
  484. package/dist/search-params/create.d.ts +0 -106
  485. package/dist/search-params/create.d.ts.map +0 -1
  486. package/dist/search-params/index.js.map +0 -1
  487. package/dist/server/prerender.d.ts +0 -77
  488. package/dist/server/prerender.d.ts.map +0 -1
  489. package/dist/server/response-cache.d.ts +0 -53
  490. package/dist/server/response-cache.d.ts.map +0 -1
  491. package/src/cache/register-cached-function.ts +0 -99
  492. package/src/client/link-status-provider.tsx +0 -30
  493. package/src/client/transition-root.tsx +0 -160
  494. package/src/plugins/cache-transform.ts +0 -199
  495. package/src/plugins/dynamic-transform.ts +0 -161
  496. package/src/search-params/analyze.ts +0 -192
  497. package/src/search-params/builtin-codecs.ts +0 -228
  498. package/src/search-params/create.ts +0 -321
  499. package/src/server/prerender.ts +0 -139
  500. package/src/server/response-cache.ts +0 -277
@@ -18,10 +18,46 @@
18
18
  // - params and fully-resolved string href are mutually exclusive
19
19
  // - searchParams and inline query string are mutually exclusive
20
20
 
21
- import type { AnchorHTMLAttributes, ReactNode, MouseEvent as ReactMouseEvent } from 'react';
22
- import type { SearchParamsDefinition } from '#/search-params/create.js';
23
- import { LinkStatusProvider } from './link-status-provider.js';
21
+ import {
22
+ useState,
23
+ useEffect,
24
+ useRef,
25
+ type AnchorHTMLAttributes,
26
+ type JSX,
27
+ type ReactNode,
28
+ type MouseEvent as ReactMouseEvent,
29
+ } from 'react';
30
+ import type { SearchParamsDefinition } from '../search-params/define.js';
31
+ import { classifyUrlSegment, type UrlSegment } from '../routing/segment-classify.js';
32
+ import { LinkStatusContext } from './use-link-status.js';
24
33
  import { getRouterOrNull } from './router-ref.js';
34
+ import { getSsrData } from './ssr-data.js';
35
+ import { mergePreservedSearchParams } from '../shared/merge-search-params.js';
36
+ import {
37
+ setLinkForCurrentNavigation,
38
+ unmountLinkForCurrentNavigation,
39
+ IDLE_LINK_STATUS,
40
+ PENDING_LINK_STATUS,
41
+ type LinkPendingInstance,
42
+ } from './link-pending-store.js';
43
+ import { setNavLinkMetadata } from './nav-link-store.js';
44
+ import { hasNavigationApi } from './navigation-api.js';
45
+
46
+ // ─── Current Search Params ────────────────────────────────────────
47
+
48
+ /**
49
+ * Read the current URL's search string without requiring a React hook.
50
+ * On the client, reads window.location.search. During SSR, reads from
51
+ * the request context (getSsrData). Returns empty string if unavailable.
52
+ */
53
+ function getCurrentSearch(): string {
54
+ if (typeof window !== 'undefined') return window.location.search;
55
+ const data = getSsrData();
56
+ if (!data) return '';
57
+ const sp = new URLSearchParams(data.searchParams);
58
+ const str = sp.toString();
59
+ return str ? `?${str}` : '';
60
+ }
25
61
 
26
62
  // ─── Types ───────────────────────────────────────────────────────
27
63
 
@@ -42,6 +78,20 @@ interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'h
42
78
  * Set to false for tabbed interfaces where content changes within a fixed layout.
43
79
  */
44
80
  scroll?: boolean;
81
+ /**
82
+ * Preserve search params from the current URL across navigation.
83
+ *
84
+ * - `true` — preserve ALL current search params (target params take precedence)
85
+ * - `string[]` — preserve only the named params (e.g. `['private', 'token']`)
86
+ *
87
+ * Useful for route-group gating where a search param (e.g. `?private=access`)
88
+ * must persist across internal navigations. The target href's own search params
89
+ * always take precedence over preserved ones.
90
+ *
91
+ * During SSR, reads search params from the request context. On the client,
92
+ * reads from the current URL and updates reactively when the URL changes.
93
+ */
94
+ preserveSearchParams?: true | string[];
45
95
  /**
46
96
  * Called before client-side navigation commits. Call `e.preventDefault()`
47
97
  * to cancel the default navigation — the caller is then responsible for
@@ -54,48 +104,107 @@ interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'h
54
104
  children?: ReactNode;
55
105
  }
56
106
 
107
+ // ─── Typed Link Props ────────────────────────────────────────────
108
+
109
+ /**
110
+ * Widen server-side string params to string | number for Link convenience.
111
+ * Exported for use by codegen-generated overloads.
112
+ */
113
+ export type LinkSegmentParams<T> = {
114
+ [K in keyof T]: [string] extends [T[K]] ? string | number : T[K];
115
+ };
116
+
117
+ // ─── External Href Types ─────────────────────────────────────────
118
+
119
+ /**
120
+ * Href types accepted by the catch-all (non-route) call signature.
121
+ *
122
+ * - External protocols: https://, http://, mailto:, tel:, ftp://
123
+ * - Hash-only and query-only links: #section, ?param=value
124
+ * - Computed `string` variables (non-literal)
125
+ *
126
+ * Internal path literals like "/typo-route" do NOT match — they must
127
+ * be a known route (from codegen) or stored in a `string` variable.
128
+ * This catches wrong hrefs at the type level. See TIM-624.
129
+ */
130
+ type ExternalHref =
131
+ | `http://${string}`
132
+ | `https://${string}`
133
+ | `mailto:${string}`
134
+ | `tel:${string}`
135
+ | `ftp://${string}`
136
+ | `//${string}`
137
+ | `#${string}`
138
+ | `?${string}`;
139
+
140
+ /**
141
+ * Callable interface for the Link component.
142
+ *
143
+ * Two kinds of call signatures:
144
+ * 1. Per-route (added by codegen via interface merging): DIRECT types
145
+ * for segmentParams — preserves TypeScript excess property checking.
146
+ * 2. Catch-all (below): accepts external hrefs and computed `string`
147
+ * variables. Does NOT accept internal path literals — those must
148
+ * match a known route from the codegen.
149
+ *
150
+ * See TIM-624.
151
+ */
152
+ export interface LinkFunction {
153
+ // External links (literal protocol hrefs)
154
+ (
155
+ props: LinkBaseProps & {
156
+ href: ExternalHref;
157
+ segmentParams?: never;
158
+ searchParams?: {
159
+ definition: SearchParamsDefinition<Record<string, unknown>>;
160
+ values: Record<string, unknown>;
161
+ };
162
+ }
163
+ ): JSX.Element;
164
+ // Computed/variable href (non-literal string) — e.g. href={myVar}
165
+ // `string extends H` is true only when H is the wide `string` type,
166
+ // not a specific literal. Template literal hrefs like `/blog/${slug}`
167
+ // are handled by resolved-pattern signatures in the codegen.
168
+ <H extends string>(
169
+ props: string extends H
170
+ ? LinkBaseProps & {
171
+ href: H;
172
+ segmentParams?: Record<string, string | number | string[]>;
173
+ searchParams?: {
174
+ definition: SearchParamsDefinition<Record<string, unknown>>;
175
+ values: Record<string, unknown>;
176
+ };
177
+ }
178
+ : never
179
+ ): JSX.Element;
180
+ }
181
+
57
182
  /**
58
- * Link with a fully-resolved string href.
59
- * When using a string href with params already interpolated,
60
- * the params prop is not available.
183
+ * Runtime-only loose props used internally by the Link implementation.
184
+ * Not exposed to callers the public API uses LinkFunction.
61
185
  */
62
- export interface LinkPropsWithHref extends LinkBaseProps {
186
+ interface LinkRuntimeProps extends LinkBaseProps {
63
187
  href: string;
64
- params?: never;
65
- /**
66
- * Typed search params — serialized via the route's SearchParamsDefinition.
67
- * Mutually exclusive with an inline query string in href.
68
- */
188
+ segmentParams?: Record<string, string | number | string[]>;
69
189
  searchParams?: {
70
190
  definition: SearchParamsDefinition<Record<string, unknown>>;
71
191
  values: Record<string, unknown>;
72
192
  };
73
193
  }
74
194
 
75
- /**
76
- * Link with a route pattern + params for interpolation.
77
- * e.g. <Link href="/products/[id]" params={{ id: "123" }}>
78
- * <Link href="/products/[id]" params={{ id: 123 }}>
79
- */
80
- export interface LinkPropsWithParams extends LinkBaseProps {
81
- /** Route pattern with dynamic segments (e.g. "/products/[id]") */
195
+ // Legacy exports for backward compat (used by buildLinkProps, tests, etc.)
196
+ export type LinkPropsWithHref = LinkBaseProps & {
82
197
  href: string;
83
- /**
84
- * Dynamic segment values to interpolate into the href.
85
- * Single dynamic segments accept string | number (numbers are stringified).
86
- * Catch-all segments accept string[].
87
- */
88
- params: Record<string, string | number | string[]>;
89
- /**
90
- * Typed search params — serialized via the route's SearchParamsDefinition.
91
- */
198
+ segmentParams?: never;
92
199
  searchParams?: {
93
200
  definition: SearchParamsDefinition<Record<string, unknown>>;
94
201
  values: Record<string, unknown>;
95
202
  };
96
- }
97
-
98
- export type LinkProps = LinkPropsWithHref | LinkPropsWithParams;
203
+ };
204
+ export type LinkPropsWithParams = LinkRuntimeProps & {
205
+ segmentParams: Record<string, string | number | string[]>;
206
+ };
207
+ export type LinkProps = LinkRuntimeProps;
99
208
 
100
209
  // ─── Dangerous URL Scheme Detection ──────────────────────────────
101
210
 
@@ -141,57 +250,91 @@ export function isInternalHref(href: string): boolean {
141
250
  * - [...param] → catch-all (joined with /)
142
251
  * - [[...param]] → optional catch-all (omitted if undefined/empty)
143
252
  */
253
+ /**
254
+ * Parse a route pattern's path portion into classified segments.
255
+ * Exported for testing. Uses the shared character-based classifier.
256
+ */
257
+ export function parseSegments(pattern: string): UrlSegment[] {
258
+ return pattern.split('/').filter(Boolean).map(classifyUrlSegment);
259
+ }
260
+
261
+ /**
262
+ * Resolve a single classified segment into its string representation.
263
+ * Returns null for optional catch-all with no value (filtered out before join).
264
+ */
265
+ function resolveSegment(
266
+ seg: UrlSegment,
267
+ params: Record<string, string | number | string[]>,
268
+ pattern: string
269
+ ): string | null {
270
+ switch (seg.kind) {
271
+ case 'static':
272
+ return seg.value;
273
+
274
+ case 'optional-catch-all': {
275
+ const value = params[seg.name];
276
+ if (value === undefined || (Array.isArray(value) && value.length === 0)) {
277
+ return null;
278
+ }
279
+ const segments = Array.isArray(value) ? value : [value];
280
+ return segments.map(encodeURIComponent).join('/');
281
+ }
282
+
283
+ case 'catch-all': {
284
+ const value = params[seg.name];
285
+ if (value === undefined) {
286
+ throw new Error(
287
+ `<Link> missing required catch-all param "${seg.name}" for pattern "${pattern}".`
288
+ );
289
+ }
290
+ const segments = Array.isArray(value) ? value : [value];
291
+ if (segments.length === 0) {
292
+ throw new Error(
293
+ `<Link> catch-all param "${seg.name}" must have at least one segment for pattern "${pattern}".`
294
+ );
295
+ }
296
+ return segments.map(encodeURIComponent).join('/');
297
+ }
298
+
299
+ case 'dynamic': {
300
+ const value = params[seg.name];
301
+ if (value === undefined) {
302
+ throw new Error(`<Link> missing required param "${seg.name}" for pattern "${pattern}".`);
303
+ }
304
+ if (Array.isArray(value)) {
305
+ throw new Error(
306
+ `<Link> param "${seg.name}" expected a string but received an array for pattern "${pattern}".`
307
+ );
308
+ }
309
+ return encodeURIComponent(String(value));
310
+ }
311
+ }
312
+ }
313
+
314
+ /**
315
+ * Split a URL pattern into the path portion and any trailing ?query/#hash suffix.
316
+ * Uses URL parsing for correctness rather than manual index arithmetic.
317
+ */
318
+ function splitPatternSuffix(pattern: string): [path: string, suffix: string] {
319
+ if (!pattern.includes('?') && !pattern.includes('#')) {
320
+ return [pattern, ''];
321
+ }
322
+ const url = new URL(pattern, 'http://x');
323
+ const suffix = url.search + url.hash;
324
+ const path = pattern.slice(0, pattern.length - suffix.length);
325
+ return [path, suffix];
326
+ }
327
+
144
328
  export function interpolateParams(
145
329
  pattern: string,
146
330
  params: Record<string, string | number | string[]>
147
331
  ): string {
148
- return (
149
- pattern
150
- .replace(
151
- /\[\[\.\.\.(\w+)\]\]|\[\.\.\.(\w+)\]|\[(\w+)\]/g,
152
- (_match, optionalCatchAll, catchAll, single) => {
153
- if (optionalCatchAll) {
154
- const value = params[optionalCatchAll];
155
- if (value === undefined || (Array.isArray(value) && value.length === 0)) {
156
- return '';
157
- }
158
- const segments = Array.isArray(value) ? value : [value];
159
- return segments.map(encodeURIComponent).join('/');
160
- }
332
+ const [pathPart, suffix] = splitPatternSuffix(pattern);
161
333
 
162
- if (catchAll) {
163
- const value = params[catchAll];
164
- if (value === undefined) {
165
- throw new Error(
166
- `<Link> missing required catch-all param "${catchAll}" for pattern "${pattern}".`
167
- );
168
- }
169
- const segments = Array.isArray(value) ? value : [value];
170
- if (segments.length === 0) {
171
- throw new Error(
172
- `<Link> catch-all param "${catchAll}" must have at least one segment for pattern "${pattern}".`
173
- );
174
- }
175
- return segments.map(encodeURIComponent).join('/');
176
- }
177
-
178
- // single dynamic segment
179
- const value = params[single];
180
- if (value === undefined) {
181
- throw new Error(`<Link> missing required param "${single}" for pattern "${pattern}".`);
182
- }
183
- if (Array.isArray(value)) {
184
- throw new Error(
185
- `<Link> param "${single}" expected a string but received an array for pattern "${pattern}".`
186
- );
187
- }
188
- // Accept numbers — coerce to string for URL interpolation
189
- return encodeURIComponent(String(value));
190
- }
191
- )
192
- // Clean up trailing slash from empty optional catch-all
193
- .replace(/\/+$/, '') || '/'
194
- );
334
+ const resolved = parseSegments(pathPart)
335
+ .map((seg) => resolveSegment(seg, params, pattern))
336
+ .filter((s): s is string => s !== null);
337
+ return ('/' + resolved.join('/') || '/') + suffix;
195
338
  }
196
339
 
197
340
  // ─── Resolve Href ───────────────────────────────────────────────
@@ -302,23 +445,76 @@ function shouldInterceptClick(
302
445
  * navigation via the router. No global event delegation — each Link owns
303
446
  * its own click handling.
304
447
  *
305
- * Supports typed routes via codegen overloads. At runtime:
306
- * - `params` prop interpolates dynamic segments in the href pattern
448
+ * Supports typed routes via the Routes interface (populated by codegen).
449
+ * At runtime:
450
+ * - `segmentParams` prop interpolates dynamic segments in the href pattern
307
451
  * - `searchParams` prop serializes query parameters via a SearchParamsDefinition
452
+ *
453
+ * Typed via the LinkFunction callable interface. The base call signature
454
+ * forbids segmentParams; per-route signatures are added by codegen via
455
+ * interface merging. See TIM-624.
308
456
  */
309
- export function Link({
310
- href,
311
- prefetch,
312
- scroll,
313
- params,
314
- searchParams,
315
- onNavigate,
316
- onClick: userOnClick,
317
- onMouseEnter: userOnMouseEnter,
318
- children,
319
- ...rest
320
- }: LinkProps) {
321
- const { href: resolvedHref } = buildLinkProps({ href, params, searchParams });
457
+ // Cast to LinkFunction — the callable interface provides the public type,
458
+ // but the implementation destructures LinkRuntimeProps internally.
459
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
460
+ export const Link: LinkFunction = function LinkImpl(props: any) {
461
+ const {
462
+ href,
463
+ prefetch,
464
+ scroll,
465
+ segmentParams,
466
+ searchParams,
467
+ preserveSearchParams,
468
+ onNavigate,
469
+ onClick: userOnClick,
470
+ onMouseEnter: userOnMouseEnter,
471
+ children,
472
+ ...rest
473
+ } = props as LinkRuntimeProps;
474
+ const { href: baseHref } = buildLinkProps({ href, params: segmentParams, searchParams });
475
+
476
+ // ─── Per-link pending state (useState) ────────────────────────
477
+ // Each Link has its own pending state. Only the clicked link's
478
+ // setter is invoked during navigation — zero other links re-render.
479
+ //
480
+ // Eager show: click handler calls setLinkStatus(PENDING) directly (urgent).
481
+ // Atomic clear: NavigationRoot calls resetLinkPending(navId) inside
482
+ // startTransition — batched with the new tree commit.
483
+ //
484
+ // See design/19-client-navigation.md §"Per-Link Pending State"
485
+ const [linkStatus, setLinkStatus] = useState(IDLE_LINK_STATUS);
486
+
487
+ // Build the link instance ref for the pending store.
488
+ // The ref is stable across renders — we update the setter on each
489
+ // render to keep it current.
490
+ const linkInstanceRef = useRef<LinkPendingInstance | null>(null);
491
+ if (!linkInstanceRef.current) {
492
+ linkInstanceRef.current = { setLinkStatus };
493
+ } else {
494
+ linkInstanceRef.current.setLinkStatus = setLinkStatus;
495
+ }
496
+
497
+ // Clean up if this link unmounts while it's the current navigation link.
498
+ // Prevents calling setOptimistic on an unmounted component.
499
+ useEffect(() => {
500
+ const instance = linkInstanceRef.current;
501
+ return () => {
502
+ if (instance) {
503
+ unmountLinkForCurrentNavigation(instance);
504
+ }
505
+ };
506
+ }, []);
507
+
508
+ // Preserve search params from the current URL when requested.
509
+ // useSearchParams() works during both SSR (reads from request context)
510
+ // and on the client (reads from window.location, reactive to URL changes).
511
+ // We read current search params directly to avoid unconditional hook calls.
512
+ // On the client, window.location.search is always current; during SSR,
513
+ // getSsrData() provides the request's search params.
514
+ const resolvedHref = preserveSearchParams
515
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
516
+ : baseHref;
517
+
322
518
  const internal = isInternalHref(resolvedHref);
323
519
 
324
520
  // ─── Click handler ───────────────────────────────────────────
@@ -335,7 +531,11 @@ export function Link({
335
531
  // Call onNavigate if provided — allows caller to cancel
336
532
  if (onNavigate) {
337
533
  let prevented = false;
338
- onNavigate({ preventDefault: () => { prevented = true; } });
534
+ onNavigate({
535
+ preventDefault: () => {
536
+ prevented = true;
537
+ },
538
+ });
339
539
  if (prevented) {
340
540
  event.preventDefault();
341
541
  return;
@@ -345,26 +545,67 @@ export function Link({
345
545
  const router = getRouterOrNull();
346
546
  if (!router) return; // SSR or pre-hydration — fall through to browser nav
347
547
 
348
- event.preventDefault();
349
548
  const shouldScroll = scroll !== false;
350
- void router.navigate(resolvedHref, { scroll: shouldScroll });
549
+
550
+ // Eagerly show pending state on this link (urgent update, immediate).
551
+ // Only this Link re-renders — all other Links are unaffected.
552
+ setLinkStatus(PENDING_LINK_STATUS);
553
+
554
+ // Register this link in the pending store so NavigationRoot can
555
+ // reset it to IDLE inside startTransition (atomic with new tree).
556
+ // Also resets any previous pending link to IDLE.
557
+ setLinkForCurrentNavigation(linkInstanceRef.current);
558
+
559
+ // When Navigation API is active, let the <a> click propagate
560
+ // naturally — do NOT call preventDefault(). The navigate event
561
+ // handler intercepts it and runs the RSC pipeline. This is a
562
+ // user-initiated navigation, so Chrome shows the native loading
563
+ // indicator (tab spinner). Metadata (scroll, link instance) is
564
+ // passed via nav-link-store so the handler can configure the nav.
565
+ //
566
+ // Without Navigation API (fallback), preventDefault and drive
567
+ // navigation through the router as before.
568
+ if (hasNavigationApi()) {
569
+ setNavLinkMetadata({
570
+ scroll: shouldScroll,
571
+ linkInstance: linkInstanceRef.current,
572
+ });
573
+ // Don't preventDefault — let the <a> click fire the navigate event
574
+ return;
575
+ }
576
+
577
+ // History API fallback — prevent default and navigate via router
578
+ event.preventDefault();
579
+
580
+ // Re-merge preserved search params at click time to pick up any
581
+ // URL changes since render (e.g. from other navigations or pushState).
582
+ const navHref = preserveSearchParams
583
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
584
+ : resolvedHref;
585
+
586
+ void router.navigate(navHref, { scroll: shouldScroll });
351
587
  }
352
588
  : userOnClick; // External links — just pass through user's onClick
353
589
 
354
590
  // ─── Hover prefetch ──────────────────────────────────────────
355
- const handleMouseEnter = internal && prefetch
356
- ? (event: ReactMouseEvent<HTMLAnchorElement>) => {
357
- userOnMouseEnter?.(event);
358
- const router = getRouterOrNull();
359
- if (router) {
360
- router.prefetch(resolvedHref);
591
+ const handleMouseEnter =
592
+ internal && prefetch
593
+ ? (event: ReactMouseEvent<HTMLAnchorElement>) => {
594
+ userOnMouseEnter?.(event);
595
+ const router = getRouterOrNull();
596
+ if (router) {
597
+ // Re-merge preserved search params at hover time for fresh prefetch URL
598
+ const prefetchHref = preserveSearchParams
599
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
600
+ : resolvedHref;
601
+ router.prefetch(prefetchHref);
602
+ }
361
603
  }
362
- }
363
- : userOnMouseEnter;
604
+ : userOnMouseEnter;
364
605
 
365
606
  return (
366
607
  <a {...rest} href={resolvedHref} onClick={handleClick} onMouseEnter={handleMouseEnter}>
367
- <LinkStatusProvider href={resolvedHref}>{children}</LinkStatusProvider>
608
+ <LinkStatusContext.Provider value={linkStatus}>{children}</LinkStatusContext.Provider>
368
609
  </a>
369
610
  );
370
- }
611
+ };
@@ -0,0 +1,47 @@
1
+ /**
2
+ * Navigation Link Store — passes per-link metadata from Link's onClick
3
+ * to the Navigation API's navigate event handler.
4
+ *
5
+ * When the Navigation API is active, Link does NOT call event.preventDefault()
6
+ * or router.navigate(). Instead it stores metadata (scroll option, link
7
+ * pending instance) here, and lets the <a> click propagate naturally.
8
+ * The navigate event handler reads this metadata to configure the RSC
9
+ * navigation with the correct options.
10
+ *
11
+ * This store is consumed once per navigation — after reading, the metadata
12
+ * is cleared. If no metadata is present (e.g., a plain <a> tag without
13
+ * our Link component), the navigate handler uses default options.
14
+ *
15
+ * See design/19-client-navigation.md §"Navigation API Integration"
16
+ */
17
+
18
+ import type { LinkPendingInstance } from './link-pending-store.js';
19
+
20
+ export interface NavLinkMetadata {
21
+ /** Whether to scroll to top after navigation. Default: true. */
22
+ scroll: boolean;
23
+ /** The Link's pending state instance for per-link status tracking. */
24
+ linkInstance: LinkPendingInstance | null;
25
+ }
26
+
27
+ let pendingMetadata: NavLinkMetadata | null = null;
28
+
29
+ /**
30
+ * Store metadata from Link's onClick for the next navigate event.
31
+ * Called synchronously in the click handler — the navigate event
32
+ * fires synchronously after onClick returns.
33
+ */
34
+ export function setNavLinkMetadata(metadata: NavLinkMetadata): void {
35
+ pendingMetadata = metadata;
36
+ }
37
+
38
+ /**
39
+ * Consume the stored metadata. Returns null if no Link onClick
40
+ * preceded this navigation (e.g., plain <a> tag, programmatic nav).
41
+ * Clears the store after reading.
42
+ */
43
+ export function consumeNavLinkMetadata(): NavLinkMetadata | null {
44
+ const metadata = pendingMetadata;
45
+ pendingMetadata = null;
46
+ return metadata;
47
+ }
@@ -0,0 +1,112 @@
1
+ /**
2
+ * Ambient type declarations for the Navigation API.
3
+ *
4
+ * The Navigation API is not yet in TypeScript's standard lib. These types
5
+ * are used internally via type assertions — we never import Navigation API
6
+ * types unconditionally. Progressive enhancement only: the API is feature-
7
+ * detected at runtime.
8
+ *
9
+ * See https://developer.mozilla.org/en-US/docs/Web/API/Navigation_API
10
+ */
11
+
12
+ // ─── Navigation Entry ────────────────────────────────────────────
13
+
14
+ export interface NavigationHistoryEntry {
15
+ readonly key: string;
16
+ readonly id: string;
17
+ readonly url: string | null;
18
+ readonly index: number;
19
+ readonly sameDocument: boolean;
20
+ getState(): unknown;
21
+ addEventListener(type: string, listener: EventListener): void;
22
+ removeEventListener(type: string, listener: EventListener): void;
23
+ }
24
+
25
+ // ─── Navigation Destination ──────────────────────────────────────
26
+
27
+ export interface NavigationDestination {
28
+ readonly url: string;
29
+ readonly key: string | null;
30
+ readonly id: string | null;
31
+ readonly index: number;
32
+ readonly sameDocument: boolean;
33
+ getState(): unknown;
34
+ }
35
+
36
+ // ─── Navigate Event ──────────────────────────────────────────────
37
+
38
+ export interface NavigateEvent extends Event {
39
+ readonly navigationType: 'push' | 'replace' | 'reload' | 'traverse';
40
+ readonly destination: NavigationDestination;
41
+ readonly canIntercept: boolean;
42
+ readonly userInitiated: boolean;
43
+ readonly hashChange: boolean;
44
+ readonly signal: AbortSignal;
45
+ readonly formData: FormData | null;
46
+ readonly downloadRequest: string | null;
47
+ readonly info: unknown;
48
+ intercept(options?: NavigateInterceptOptions): void;
49
+ scroll(): void;
50
+ }
51
+
52
+ export interface NavigateInterceptOptions {
53
+ handler?: () => Promise<void>;
54
+ focusReset?: 'after-transition' | 'manual';
55
+ scroll?: 'after-transition' | 'manual';
56
+ }
57
+
58
+ // ─── Navigation Transition ───────────────────────────────────────
59
+
60
+ export interface NavigationTransition {
61
+ readonly navigationType: 'push' | 'replace' | 'reload' | 'traverse';
62
+ readonly from: NavigationHistoryEntry;
63
+ readonly finished: Promise<void>;
64
+ }
65
+
66
+ // ─── Navigation Result ───────────────────────────────────────────
67
+
68
+ export interface NavigationResult {
69
+ committed: Promise<NavigationHistoryEntry>;
70
+ finished: Promise<NavigationHistoryEntry>;
71
+ }
72
+
73
+ // ─── Navigation Interface ────────────────────────────────────────
74
+
75
+ export interface NavigationApi {
76
+ readonly currentEntry: NavigationHistoryEntry | null;
77
+ readonly transition: NavigationTransition | null;
78
+ readonly canGoBack: boolean;
79
+ readonly canGoForward: boolean;
80
+ entries(): NavigationHistoryEntry[];
81
+ navigate(url: string, options?: NavigationNavigateOptions): NavigationResult;
82
+ reload(options?: NavigationReloadOptions): NavigationResult;
83
+ traverseTo(key: string, options?: NavigationOptions): NavigationResult;
84
+ back(options?: NavigationOptions): NavigationResult;
85
+ forward(options?: NavigationOptions): NavigationResult;
86
+ updateCurrentEntry(options: NavigationUpdateCurrentEntryOptions): void;
87
+ addEventListener(type: 'navigate', listener: (event: NavigateEvent) => void): void;
88
+ addEventListener(type: 'navigatesuccess', listener: (event: Event) => void): void;
89
+ addEventListener(type: 'navigateerror', listener: (event: Event) => void): void;
90
+ addEventListener(type: 'currententrychange', listener: (event: Event) => void): void;
91
+ addEventListener(type: string, listener: EventListener): void;
92
+ removeEventListener(type: string, listener: EventListener): void;
93
+ }
94
+
95
+ export interface NavigationNavigateOptions {
96
+ state?: unknown;
97
+ history?: 'auto' | 'push' | 'replace';
98
+ info?: unknown;
99
+ }
100
+
101
+ export interface NavigationReloadOptions {
102
+ state?: unknown;
103
+ info?: unknown;
104
+ }
105
+
106
+ export interface NavigationOptions {
107
+ info?: unknown;
108
+ }
109
+
110
+ export interface NavigationUpdateCurrentEntryOptions {
111
+ state: unknown;
112
+ }