@timber-js/app 0.2.0-alpha.9 → 0.2.0-alpha.90

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 (610) hide show
  1. package/dist/_chunks/actions-DLnUaR65.js +421 -0
  2. package/dist/_chunks/actions-DLnUaR65.js.map +1 -0
  3. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
  4. package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
  5. package/dist/_chunks/chunk-BYIpzuS7.js +39 -0
  6. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  7. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  8. package/dist/_chunks/define-C77ScO0m.js +106 -0
  9. package/dist/_chunks/define-C77ScO0m.js.map +1 -0
  10. package/dist/_chunks/define-Itxvcd7F.js +199 -0
  11. package/dist/_chunks/define-Itxvcd7F.js.map +1 -0
  12. package/dist/_chunks/define-cookie-BowvzoP0.js +94 -0
  13. package/dist/_chunks/define-cookie-BowvzoP0.js.map +1 -0
  14. package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
  15. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
  16. package/dist/_chunks/format-CYBGxKtc.js +14 -0
  17. package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
  18. package/dist/_chunks/{interception-BOoWmLUA.js → interception-DRlhJWbu.js} +219 -97
  19. package/dist/_chunks/interception-DRlhJWbu.js.map +1 -0
  20. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
  21. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
  22. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  23. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  24. package/dist/_chunks/request-context-CK5tZqIP.js +478 -0
  25. package/dist/_chunks/request-context-CK5tZqIP.js.map +1 -0
  26. package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
  27. package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
  28. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  29. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  30. package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
  31. package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
  32. package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
  33. package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
  34. package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
  35. package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
  36. package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
  37. package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
  38. package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
  39. package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
  40. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-BiV5GJgm.js} +7 -4
  41. package/dist/_chunks/use-query-states-BiV5GJgm.js.map +1 -0
  42. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  43. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  44. package/dist/adapters/cloudflare-dev.js +73 -0
  45. package/dist/adapters/cloudflare-dev.js.map +1 -0
  46. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  47. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  48. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  49. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  50. package/dist/adapters/cloudflare.d.ts +148 -12
  51. package/dist/adapters/cloudflare.d.ts.map +1 -1
  52. package/dist/adapters/cloudflare.js +135 -11
  53. package/dist/adapters/cloudflare.js.map +1 -1
  54. package/dist/adapters/compress-module.d.ts.map +1 -1
  55. package/dist/adapters/nitro.d.ts +17 -1
  56. package/dist/adapters/nitro.d.ts.map +1 -1
  57. package/dist/adapters/nitro.js +56 -13
  58. package/dist/adapters/nitro.js.map +1 -1
  59. package/dist/cache/cache-api.d.ts +24 -0
  60. package/dist/cache/cache-api.d.ts.map +1 -0
  61. package/dist/cache/handler-store.d.ts +31 -0
  62. package/dist/cache/handler-store.d.ts.map +1 -0
  63. package/dist/cache/index.d.ts +23 -7
  64. package/dist/cache/index.d.ts.map +1 -1
  65. package/dist/cache/index.js +142 -80
  66. package/dist/cache/index.js.map +1 -1
  67. package/dist/cache/singleflight.d.ts +18 -1
  68. package/dist/cache/singleflight.d.ts.map +1 -1
  69. package/dist/cache/sizeof.d.ts +22 -0
  70. package/dist/cache/sizeof.d.ts.map +1 -0
  71. package/dist/cache/timber-cache.d.ts +1 -1
  72. package/dist/cache/timber-cache.d.ts.map +1 -1
  73. package/dist/cli.d.ts +6 -1
  74. package/dist/cli.d.ts.map +1 -1
  75. package/dist/cli.js +8 -3
  76. package/dist/cli.js.map +1 -1
  77. package/dist/client/browser-dev.d.ts +27 -1
  78. package/dist/client/browser-dev.d.ts.map +1 -1
  79. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  80. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  81. package/dist/client/browser-entry/hmr.d.ts +21 -0
  82. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  83. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  84. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  85. package/dist/client/browser-entry/index.d.ts +30 -0
  86. package/dist/client/browser-entry/index.d.ts.map +1 -0
  87. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  88. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  89. package/dist/client/browser-entry/router-init.d.ts +23 -0
  90. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  91. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  92. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  93. package/dist/client/browser-entry/scroll.d.ts +19 -0
  94. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  95. package/dist/client/error-boundary.d.ts +12 -5
  96. package/dist/client/error-boundary.d.ts.map +1 -1
  97. package/dist/client/error-boundary.js +10 -4
  98. package/dist/client/error-boundary.js.map +1 -1
  99. package/dist/client/error-reconstituter.d.ts +54 -0
  100. package/dist/client/error-reconstituter.d.ts.map +1 -0
  101. package/dist/client/form.d.ts +6 -3
  102. package/dist/client/form.d.ts.map +1 -1
  103. package/dist/client/history.d.ts +19 -4
  104. package/dist/client/history.d.ts.map +1 -1
  105. package/dist/client/index.d.ts +9 -21
  106. package/dist/client/index.d.ts.map +1 -1
  107. package/dist/client/index.js +229 -1018
  108. package/dist/client/index.js.map +1 -1
  109. package/dist/client/internal.d.ts +18 -0
  110. package/dist/client/internal.d.ts.map +1 -0
  111. package/dist/client/internal.js +890 -0
  112. package/dist/client/internal.js.map +1 -0
  113. package/dist/client/link-pending-store.d.ts +63 -0
  114. package/dist/client/link-pending-store.d.ts.map +1 -0
  115. package/dist/client/link.d.ts +62 -55
  116. package/dist/client/link.d.ts.map +1 -1
  117. package/dist/client/nav-link-store.d.ts +36 -0
  118. package/dist/client/nav-link-store.d.ts.map +1 -0
  119. package/dist/client/navigation-api-types.d.ts +90 -0
  120. package/dist/client/navigation-api-types.d.ts.map +1 -0
  121. package/dist/client/navigation-api.d.ts +115 -0
  122. package/dist/client/navigation-api.d.ts.map +1 -0
  123. package/dist/client/navigation-context.d.ts +13 -2
  124. package/dist/client/navigation-context.d.ts.map +1 -1
  125. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  126. package/dist/client/navigation-root.d.ts.map +1 -0
  127. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  128. package/dist/client/router-ref.d.ts +1 -1
  129. package/dist/client/router.d.ts +70 -4
  130. package/dist/client/router.d.ts.map +1 -1
  131. package/dist/client/rsc-fetch.d.ts +38 -3
  132. package/dist/client/rsc-fetch.d.ts.map +1 -1
  133. package/dist/client/segment-cache.d.ts +1 -1
  134. package/dist/client/segment-cache.d.ts.map +1 -1
  135. package/dist/client/segment-outlet.d.ts +63 -0
  136. package/dist/client/segment-outlet.d.ts.map +1 -0
  137. package/dist/client/ssr-data.d.ts +13 -4
  138. package/dist/client/ssr-data.d.ts.map +1 -1
  139. package/dist/client/stale-reload.d.ts +15 -0
  140. package/dist/client/stale-reload.d.ts.map +1 -1
  141. package/dist/client/top-loader.d.ts +5 -5
  142. package/dist/client/top-loader.d.ts.map +1 -1
  143. package/dist/client/use-link-status.d.ts +5 -5
  144. package/dist/client/use-link-status.d.ts.map +1 -1
  145. package/dist/client/use-params.d.ts +6 -4
  146. package/dist/client/use-params.d.ts.map +1 -1
  147. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  148. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  149. package/dist/client/use-query-states.d.ts +1 -1
  150. package/dist/client/use-query-states.d.ts.map +1 -1
  151. package/dist/client/use-router.d.ts +1 -1
  152. package/dist/codec.d.ts +33 -0
  153. package/dist/codec.d.ts.map +1 -0
  154. package/dist/codec.js +2 -0
  155. package/dist/config-types.d.ts +266 -0
  156. package/dist/config-types.d.ts.map +1 -0
  157. package/dist/config-validation.d.ts +51 -0
  158. package/dist/config-validation.d.ts.map +1 -0
  159. package/dist/content/index.d.ts +1 -10
  160. package/dist/content/index.d.ts.map +1 -1
  161. package/dist/content/index.js +0 -2
  162. package/dist/cookies/define-cookie.d.ts +35 -14
  163. package/dist/cookies/define-cookie.d.ts.map +1 -1
  164. package/dist/cookies/index.js +1 -83
  165. package/dist/fonts/bundle.d.ts +48 -0
  166. package/dist/fonts/bundle.d.ts.map +1 -0
  167. package/dist/fonts/css.d.ts +1 -0
  168. package/dist/fonts/css.d.ts.map +1 -1
  169. package/dist/fonts/dev-middleware.d.ts +22 -0
  170. package/dist/fonts/dev-middleware.d.ts.map +1 -0
  171. package/dist/fonts/pipeline.d.ts +138 -0
  172. package/dist/fonts/pipeline.d.ts.map +1 -0
  173. package/dist/fonts/transform.d.ts +72 -0
  174. package/dist/fonts/transform.d.ts.map +1 -0
  175. package/dist/fonts/types.d.ts +45 -1
  176. package/dist/fonts/types.d.ts.map +1 -1
  177. package/dist/fonts/virtual-modules.d.ts +59 -0
  178. package/dist/fonts/virtual-modules.d.ts.map +1 -0
  179. package/dist/index.d.ts +45 -190
  180. package/dist/index.d.ts.map +1 -1
  181. package/dist/index.js +4294 -2453
  182. package/dist/index.js.map +1 -1
  183. package/dist/plugin-context.d.ts +107 -0
  184. package/dist/plugin-context.d.ts.map +1 -0
  185. package/dist/plugins/adapter-build.d.ts +1 -1
  186. package/dist/plugins/adapter-build.d.ts.map +1 -1
  187. package/dist/plugins/build-manifest.d.ts +2 -2
  188. package/dist/plugins/build-manifest.d.ts.map +1 -1
  189. package/dist/plugins/build-report.d.ts +3 -3
  190. package/dist/plugins/build-report.d.ts.map +1 -1
  191. package/dist/plugins/client-chunks.d.ts +32 -0
  192. package/dist/plugins/client-chunks.d.ts.map +1 -0
  193. package/dist/plugins/content.d.ts +1 -1
  194. package/dist/plugins/content.d.ts.map +1 -1
  195. package/dist/plugins/dev-404-page.d.ts +56 -0
  196. package/dist/plugins/dev-404-page.d.ts.map +1 -0
  197. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  198. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  199. package/dist/plugins/dev-error-overlay.d.ts +49 -9
  200. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  201. package/dist/plugins/dev-error-page.d.ts +58 -0
  202. package/dist/plugins/dev-error-page.d.ts.map +1 -0
  203. package/dist/plugins/dev-logs.d.ts +1 -1
  204. package/dist/plugins/dev-logs.d.ts.map +1 -1
  205. package/dist/plugins/dev-server.d.ts +1 -1
  206. package/dist/plugins/dev-server.d.ts.map +1 -1
  207. package/dist/plugins/dev-terminal-error.d.ts +28 -0
  208. package/dist/plugins/dev-terminal-error.d.ts.map +1 -0
  209. package/dist/plugins/entries.d.ts +1 -1
  210. package/dist/plugins/entries.d.ts.map +1 -1
  211. package/dist/plugins/fonts.d.ts +17 -73
  212. package/dist/plugins/fonts.d.ts.map +1 -1
  213. package/dist/plugins/mdx.d.ts +1 -1
  214. package/dist/plugins/mdx.d.ts.map +1 -1
  215. package/dist/plugins/routing.d.ts +1 -1
  216. package/dist/plugins/routing.d.ts.map +1 -1
  217. package/dist/plugins/server-bundle.d.ts.map +1 -1
  218. package/dist/plugins/shims.d.ts +6 -5
  219. package/dist/plugins/shims.d.ts.map +1 -1
  220. package/dist/plugins/static-build.d.ts +4 -4
  221. package/dist/plugins/static-build.d.ts.map +1 -1
  222. package/dist/routing/codegen.d.ts +2 -2
  223. package/dist/routing/codegen.d.ts.map +1 -1
  224. package/dist/routing/convention-lint.d.ts +41 -0
  225. package/dist/routing/convention-lint.d.ts.map +1 -0
  226. package/dist/routing/index.d.ts +2 -0
  227. package/dist/routing/index.d.ts.map +1 -1
  228. package/dist/routing/index.js +3 -2
  229. package/dist/routing/scanner.d.ts.map +1 -1
  230. package/dist/routing/segment-classify.d.ts +46 -0
  231. package/dist/routing/segment-classify.d.ts.map +1 -0
  232. package/dist/routing/status-file-lint.d.ts +2 -1
  233. package/dist/routing/status-file-lint.d.ts.map +1 -1
  234. package/dist/routing/types.d.ts +16 -4
  235. package/dist/routing/types.d.ts.map +1 -1
  236. package/dist/rsc-runtime/rsc.d.ts +1 -1
  237. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  238. package/dist/rsc-runtime/ssr.d.ts +12 -0
  239. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  240. package/dist/schema-bridge.d.ts +76 -0
  241. package/dist/schema-bridge.d.ts.map +1 -0
  242. package/dist/search-params/define.d.ts +139 -0
  243. package/dist/search-params/define.d.ts.map +1 -0
  244. package/dist/search-params/index.d.ts +4 -7
  245. package/dist/search-params/index.d.ts.map +1 -1
  246. package/dist/search-params/index.js +32 -441
  247. package/dist/search-params/index.js.map +1 -1
  248. package/dist/search-params/registry.d.ts +2 -2
  249. package/dist/search-params/registry.d.ts.map +1 -1
  250. package/dist/search-params/wrappers.d.ts +53 -0
  251. package/dist/search-params/wrappers.d.ts.map +1 -0
  252. package/dist/segment-params/define.d.ts +78 -0
  253. package/dist/segment-params/define.d.ts.map +1 -0
  254. package/dist/segment-params/index.d.ts +3 -0
  255. package/dist/segment-params/index.d.ts.map +1 -0
  256. package/dist/segment-params/index.js +2 -0
  257. package/dist/server/access-gate.d.ts +4 -0
  258. package/dist/server/access-gate.d.ts.map +1 -1
  259. package/dist/server/action-client.d.ts +41 -6
  260. package/dist/server/action-client.d.ts.map +1 -1
  261. package/dist/server/action-encryption.d.ts +76 -0
  262. package/dist/server/action-encryption.d.ts.map +1 -0
  263. package/dist/server/action-handler.d.ts +7 -0
  264. package/dist/server/action-handler.d.ts.map +1 -1
  265. package/dist/server/actions.d.ts +3 -6
  266. package/dist/server/actions.d.ts.map +1 -1
  267. package/dist/server/als-registry.d.ts +32 -4
  268. package/dist/server/als-registry.d.ts.map +1 -1
  269. package/dist/server/build-manifest.d.ts +2 -2
  270. package/dist/server/build-manifest.d.ts.map +1 -1
  271. package/dist/server/debug.d.ts +1 -1
  272. package/dist/server/default-logger.d.ts +22 -0
  273. package/dist/server/default-logger.d.ts.map +1 -0
  274. package/dist/server/deny-page-resolver.d.ts +52 -0
  275. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  276. package/dist/server/deny-renderer.d.ts.map +1 -1
  277. package/dist/server/dev-holding-server.d.ts +52 -0
  278. package/dist/server/dev-holding-server.d.ts.map +1 -0
  279. package/dist/server/dev-source-map.d.ts +22 -0
  280. package/dist/server/dev-source-map.d.ts.map +1 -0
  281. package/dist/server/dev-warnings.d.ts +1 -21
  282. package/dist/server/dev-warnings.d.ts.map +1 -1
  283. package/dist/server/early-hints.d.ts +13 -5
  284. package/dist/server/early-hints.d.ts.map +1 -1
  285. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  286. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  287. package/dist/server/fallback-error.d.ts +12 -7
  288. package/dist/server/fallback-error.d.ts.map +1 -1
  289. package/dist/server/flight-injection-state.d.ts +66 -0
  290. package/dist/server/flight-injection-state.d.ts.map +1 -0
  291. package/dist/server/flight-scripts.d.ts +42 -0
  292. package/dist/server/flight-scripts.d.ts.map +1 -0
  293. package/dist/server/flush.d.ts.map +1 -1
  294. package/dist/server/form-data.d.ts +29 -0
  295. package/dist/server/form-data.d.ts.map +1 -1
  296. package/dist/server/html-injectors.d.ts +51 -11
  297. package/dist/server/html-injectors.d.ts.map +1 -1
  298. package/dist/server/index.d.ts +5 -43
  299. package/dist/server/index.d.ts.map +1 -1
  300. package/dist/server/index.js +195 -2800
  301. package/dist/server/index.js.map +1 -1
  302. package/dist/server/internal.d.ts +46 -0
  303. package/dist/server/internal.d.ts.map +1 -0
  304. package/dist/server/internal.js +2900 -0
  305. package/dist/server/internal.js.map +1 -0
  306. package/dist/server/logger.d.ts +25 -7
  307. package/dist/server/logger.d.ts.map +1 -1
  308. package/dist/server/middleware-runner.d.ts +19 -4
  309. package/dist/server/middleware-runner.d.ts.map +1 -1
  310. package/dist/server/node-stream-transforms.d.ts +113 -0
  311. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  312. package/dist/server/page-deny-boundary.d.ts +31 -0
  313. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  314. package/dist/server/pipeline-interception.d.ts +1 -1
  315. package/dist/server/pipeline-interception.d.ts.map +1 -1
  316. package/dist/server/pipeline-metadata.d.ts +6 -0
  317. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  318. package/dist/server/pipeline.d.ts +52 -10
  319. package/dist/server/pipeline.d.ts.map +1 -1
  320. package/dist/server/primitives.d.ts +69 -18
  321. package/dist/server/primitives.d.ts.map +1 -1
  322. package/dist/server/render-timeout.d.ts +51 -0
  323. package/dist/server/render-timeout.d.ts.map +1 -0
  324. package/dist/server/request-context.d.ts +112 -43
  325. package/dist/server/request-context.d.ts.map +1 -1
  326. package/dist/server/route-element-builder.d.ts +27 -1
  327. package/dist/server/route-element-builder.d.ts.map +1 -1
  328. package/dist/server/route-handler.d.ts.map +1 -1
  329. package/dist/server/route-matcher.d.ts +16 -2
  330. package/dist/server/route-matcher.d.ts.map +1 -1
  331. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  332. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  333. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  334. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  335. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  336. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  337. package/dist/server/rsc-entry/index.d.ts +20 -3
  338. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  339. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  340. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  341. package/dist/server/rsc-entry/rsc-stream.d.ts +14 -1
  342. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  343. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  344. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  345. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  346. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  347. package/dist/server/safe-load.d.ts +46 -0
  348. package/dist/server/safe-load.d.ts.map +1 -0
  349. package/dist/server/sensitive-fields.d.ts +74 -0
  350. package/dist/server/sensitive-fields.d.ts.map +1 -0
  351. package/dist/server/sitemap-generator.d.ts +129 -0
  352. package/dist/server/sitemap-generator.d.ts.map +1 -0
  353. package/dist/server/sitemap-handler.d.ts +22 -0
  354. package/dist/server/sitemap-handler.d.ts.map +1 -0
  355. package/dist/server/slot-resolver.d.ts +1 -1
  356. package/dist/server/slot-resolver.d.ts.map +1 -1
  357. package/dist/server/ssr-entry.d.ts +23 -0
  358. package/dist/server/ssr-entry.d.ts.map +1 -1
  359. package/dist/server/ssr-render.d.ts +39 -21
  360. package/dist/server/ssr-render.d.ts.map +1 -1
  361. package/dist/server/ssr-wrappers.d.ts +50 -0
  362. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  363. package/dist/server/status-code-resolver.d.ts +1 -1
  364. package/dist/server/status-code-resolver.d.ts.map +1 -1
  365. package/dist/server/stream-utils.d.ts +36 -0
  366. package/dist/server/stream-utils.d.ts.map +1 -0
  367. package/dist/server/tracing.d.ts +4 -4
  368. package/dist/server/tracing.d.ts.map +1 -1
  369. package/dist/server/tree-builder.d.ts +22 -19
  370. package/dist/server/tree-builder.d.ts.map +1 -1
  371. package/dist/server/types.d.ts +1 -4
  372. package/dist/server/types.d.ts.map +1 -1
  373. package/dist/server/version-skew.d.ts +61 -0
  374. package/dist/server/version-skew.d.ts.map +1 -0
  375. package/dist/shared/merge-search-params.d.ts +22 -0
  376. package/dist/shared/merge-search-params.d.ts.map +1 -0
  377. package/dist/shims/font-google.d.ts +1 -1
  378. package/dist/shims/font-google.d.ts.map +1 -1
  379. package/dist/shims/font-google.js +42 -0
  380. package/dist/shims/font-google.js.map +1 -0
  381. package/dist/shims/font-local.d.ts +26 -0
  382. package/dist/shims/font-local.d.ts.map +1 -0
  383. package/dist/shims/font-local.js +20 -0
  384. package/dist/shims/font-local.js.map +1 -0
  385. package/dist/shims/headers.d.ts +2 -1
  386. package/dist/shims/headers.d.ts.map +1 -1
  387. package/dist/shims/navigation-client.d.ts +1 -1
  388. package/dist/shims/navigation-client.d.ts.map +1 -1
  389. package/dist/shims/navigation.d.ts +3 -2
  390. package/dist/shims/navigation.d.ts.map +1 -1
  391. package/dist/utils/directive-parser.d.ts +5 -2
  392. package/dist/utils/directive-parser.d.ts.map +1 -1
  393. package/dist/utils/state-machine.d.ts +80 -0
  394. package/dist/utils/state-machine.d.ts.map +1 -0
  395. package/package.json +51 -16
  396. package/src/adapters/cloudflare-dev.ts +177 -0
  397. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  398. package/src/adapters/cloudflare.ts +342 -28
  399. package/src/adapters/compress-module.ts +24 -4
  400. package/src/adapters/nitro.ts +52 -8
  401. package/src/adapters/wrangler.d.ts +7 -0
  402. package/src/cache/cache-api.ts +38 -0
  403. package/src/cache/handler-store.ts +68 -0
  404. package/src/cache/index.ts +81 -18
  405. package/src/cache/singleflight.ts +62 -4
  406. package/src/cache/sizeof.ts +31 -0
  407. package/src/cache/timber-cache.ts +24 -20
  408. package/src/cli.ts +16 -6
  409. package/src/client/browser-dev.ts +128 -1
  410. package/src/client/browser-entry/action-dispatch.ts +116 -0
  411. package/src/client/browser-entry/hmr.ts +81 -0
  412. package/src/client/browser-entry/hydrate.ts +145 -0
  413. package/src/client/browser-entry/index.ts +143 -0
  414. package/src/client/browser-entry/post-hydration.ts +119 -0
  415. package/src/client/browser-entry/router-init.ts +193 -0
  416. package/src/client/browser-entry/rsc-stream.ts +157 -0
  417. package/src/client/browser-entry/scroll.ts +27 -0
  418. package/src/client/error-boundary.tsx +48 -16
  419. package/src/client/error-reconstituter.tsx +65 -0
  420. package/src/client/form.tsx +14 -7
  421. package/src/client/history.ts +26 -4
  422. package/src/client/index.ts +65 -38
  423. package/src/client/internal.ts +57 -0
  424. package/src/client/link-pending-store.ts +111 -0
  425. package/src/client/link.tsx +342 -113
  426. package/src/client/nav-link-store.ts +47 -0
  427. package/src/client/navigation-api-types.ts +112 -0
  428. package/src/client/navigation-api.ts +332 -0
  429. package/src/client/navigation-context.ts +31 -6
  430. package/src/client/navigation-root.tsx +342 -0
  431. package/src/client/nuqs-adapter.tsx +16 -3
  432. package/src/client/router-ref.ts +1 -1
  433. package/src/client/router.ts +299 -72
  434. package/src/client/rsc-fetch.ts +97 -8
  435. package/src/client/segment-cache.ts +1 -1
  436. package/src/client/segment-outlet.tsx +86 -0
  437. package/src/client/ssr-data.ts +13 -5
  438. package/src/client/stale-reload.ts +72 -3
  439. package/src/client/top-loader.tsx +18 -6
  440. package/src/client/use-link-status.ts +7 -7
  441. package/src/client/use-params.ts +7 -5
  442. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  443. package/src/client/use-query-states.ts +9 -3
  444. package/src/client/use-router.ts +1 -1
  445. package/src/codec.ts +49 -0
  446. package/src/config-types.ts +264 -0
  447. package/src/config-validation.ts +303 -0
  448. package/src/content/index.ts +5 -13
  449. package/src/cookies/define-cookie.ts +78 -25
  450. package/src/cookies/index.ts +8 -0
  451. package/src/fonts/bundle.ts +142 -0
  452. package/src/fonts/css.ts +2 -1
  453. package/src/fonts/dev-middleware.ts +74 -0
  454. package/src/fonts/pipeline.ts +275 -0
  455. package/src/fonts/transform.ts +353 -0
  456. package/src/fonts/types.ts +50 -1
  457. package/src/fonts/virtual-modules.ts +159 -0
  458. package/src/index.ts +314 -355
  459. package/src/plugin-context.ts +240 -0
  460. package/src/plugins/adapter-build.ts +9 -3
  461. package/src/plugins/build-manifest.ts +13 -2
  462. package/src/plugins/build-report.ts +3 -3
  463. package/src/plugins/client-chunks.ts +65 -0
  464. package/src/plugins/content.ts +1 -1
  465. package/src/plugins/dev-404-page.ts +418 -0
  466. package/src/plugins/dev-browser-logs.ts +288 -0
  467. package/src/plugins/dev-error-overlay.ts +286 -42
  468. package/src/plugins/dev-error-page.ts +536 -0
  469. package/src/plugins/dev-logs.ts +1 -1
  470. package/src/plugins/dev-server.ts +146 -19
  471. package/src/plugins/dev-terminal-error.ts +217 -0
  472. package/src/plugins/entries.ts +111 -10
  473. package/src/plugins/fonts.ts +133 -638
  474. package/src/plugins/mdx.ts +1 -1
  475. package/src/plugins/routing.ts +213 -31
  476. package/src/plugins/server-action-exports.ts +1 -1
  477. package/src/plugins/server-bundle.ts +32 -1
  478. package/src/plugins/shims.ts +136 -35
  479. package/src/plugins/static-build.ts +17 -11
  480. package/src/routing/codegen.ts +273 -105
  481. package/src/routing/convention-lint.ts +356 -0
  482. package/src/routing/index.ts +2 -0
  483. package/src/routing/scanner.ts +93 -23
  484. package/src/routing/segment-classify.ts +89 -0
  485. package/src/routing/status-file-lint.ts +3 -2
  486. package/src/routing/types.ts +17 -4
  487. package/src/rsc-runtime/rsc.ts +2 -0
  488. package/src/rsc-runtime/ssr.ts +50 -0
  489. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  490. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  491. package/src/search-params/define.ts +482 -0
  492. package/src/search-params/index.ts +14 -20
  493. package/src/search-params/registry.ts +2 -2
  494. package/src/search-params/wrappers.ts +85 -0
  495. package/src/segment-params/define.ts +279 -0
  496. package/src/segment-params/index.ts +9 -0
  497. package/src/server/access-gate.tsx +70 -29
  498. package/src/server/action-client.ts +88 -15
  499. package/src/server/action-encryption.ts +144 -0
  500. package/src/server/action-handler.ts +53 -6
  501. package/src/server/actions.ts +10 -9
  502. package/src/server/als-registry.ts +34 -6
  503. package/src/server/build-manifest.ts +10 -4
  504. package/src/server/compress.ts +25 -7
  505. package/src/server/debug.ts +1 -1
  506. package/src/server/default-logger.ts +99 -0
  507. package/src/server/deny-page-resolver.ts +154 -0
  508. package/src/server/deny-renderer.ts +24 -38
  509. package/src/server/dev-holding-server.ts +185 -0
  510. package/src/server/dev-source-map.ts +31 -0
  511. package/src/server/dev-warnings.ts +4 -49
  512. package/src/server/early-hints.ts +36 -15
  513. package/src/server/error-boundary-wrapper.ts +74 -22
  514. package/src/server/fallback-error.ts +74 -102
  515. package/src/server/flight-injection-state.ts +113 -0
  516. package/src/server/flight-scripts.ts +62 -0
  517. package/src/server/flush.ts +2 -1
  518. package/src/server/form-data.ts +76 -0
  519. package/src/server/html-injectors.ts +280 -120
  520. package/src/server/index.ts +25 -177
  521. package/src/server/internal.ts +169 -0
  522. package/src/server/logger.ts +44 -36
  523. package/src/server/middleware-runner.ts +31 -4
  524. package/src/server/node-stream-transforms.ts +509 -0
  525. package/src/server/page-deny-boundary.tsx +56 -0
  526. package/src/server/pipeline-interception.ts +17 -16
  527. package/src/server/pipeline-metadata.ts +13 -0
  528. package/src/server/pipeline.ts +261 -66
  529. package/src/server/primitives.ts +111 -28
  530. package/src/server/render-timeout.ts +108 -0
  531. package/src/server/request-context.ts +293 -132
  532. package/src/server/route-element-builder.ts +283 -191
  533. package/src/server/route-handler.ts +24 -4
  534. package/src/server/route-matcher.ts +31 -20
  535. package/src/server/rsc-entry/api-handler.ts +15 -16
  536. package/src/server/rsc-entry/error-renderer.ts +305 -89
  537. package/src/server/rsc-entry/helpers.ts +134 -5
  538. package/src/server/rsc-entry/index.ts +304 -111
  539. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  540. package/src/server/rsc-entry/rsc-stream.ts +81 -13
  541. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  542. package/src/server/rsc-entry/ssr-renderer.ts +171 -38
  543. package/src/server/safe-load.ts +60 -0
  544. package/src/server/sensitive-fields.ts +230 -0
  545. package/src/server/sitemap-generator.ts +338 -0
  546. package/src/server/sitemap-handler.ts +126 -0
  547. package/src/server/slot-resolver.ts +244 -229
  548. package/src/server/ssr-entry.ts +215 -32
  549. package/src/server/ssr-render.ts +289 -67
  550. package/src/server/ssr-wrappers.tsx +139 -0
  551. package/src/server/status-code-resolver.ts +1 -1
  552. package/src/server/stream-utils.ts +213 -0
  553. package/src/server/tracing.ts +20 -9
  554. package/src/server/tree-builder.ts +92 -58
  555. package/src/server/types.ts +3 -6
  556. package/src/server/version-skew.ts +104 -0
  557. package/src/shared/merge-search-params.ts +55 -0
  558. package/src/shims/font-google.ts +1 -1
  559. package/src/shims/font-local.ts +34 -0
  560. package/src/shims/headers.ts +5 -1
  561. package/src/shims/navigation-client.ts +1 -1
  562. package/src/shims/navigation.ts +7 -2
  563. package/src/utils/directive-parser.ts +5 -2
  564. package/src/utils/state-machine.ts +111 -0
  565. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  566. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  567. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  568. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  569. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  570. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  571. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  572. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  573. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  574. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  575. package/dist/cache/register-cached-function.d.ts +0 -17
  576. package/dist/cache/register-cached-function.d.ts.map +0 -1
  577. package/dist/client/browser-entry.d.ts +0 -21
  578. package/dist/client/browser-entry.d.ts.map +0 -1
  579. package/dist/client/link-status-provider.d.ts +0 -11
  580. package/dist/client/link-status-provider.d.ts.map +0 -1
  581. package/dist/client/transition-root.d.ts.map +0 -1
  582. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  583. package/dist/cookies/index.js.map +0 -1
  584. package/dist/plugins/cache-transform.d.ts +0 -36
  585. package/dist/plugins/cache-transform.d.ts.map +0 -1
  586. package/dist/plugins/dynamic-transform.d.ts +0 -72
  587. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  588. package/dist/search-params/analyze.d.ts +0 -54
  589. package/dist/search-params/analyze.d.ts.map +0 -1
  590. package/dist/search-params/builtin-codecs.d.ts +0 -105
  591. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  592. package/dist/search-params/codecs.d.ts +0 -53
  593. package/dist/search-params/codecs.d.ts.map +0 -1
  594. package/dist/search-params/create.d.ts +0 -106
  595. package/dist/search-params/create.d.ts.map +0 -1
  596. package/dist/server/prerender.d.ts +0 -77
  597. package/dist/server/prerender.d.ts.map +0 -1
  598. package/dist/server/response-cache.d.ts +0 -54
  599. package/dist/server/response-cache.d.ts.map +0 -1
  600. package/src/cache/register-cached-function.ts +0 -103
  601. package/src/client/browser-entry.ts +0 -678
  602. package/src/client/link-status-provider.tsx +0 -30
  603. package/src/client/transition-root.tsx +0 -166
  604. package/src/plugins/cache-transform.ts +0 -199
  605. package/src/plugins/dynamic-transform.ts +0 -161
  606. package/src/search-params/analyze.ts +0 -192
  607. package/src/search-params/builtin-codecs.ts +0 -228
  608. package/src/search-params/create.ts +0 -321
  609. package/src/server/prerender.ts +0 -139
  610. package/src/server/response-cache.ts +0 -410
@@ -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
+ useOptimistic,
23
+ useEffect,
24
+ useRef,
25
+ type AnchorHTMLAttributes,
26
+ type ReactNode,
27
+ type MouseEvent as ReactMouseEvent,
28
+ } from 'react';
29
+ import type { SearchParamsDefinition } from '../search-params/define.js';
30
+ import { getSearchParamsDefinition } from '../search-params/registry.js';
31
+ import type { LinkFunction } from './index.js';
32
+ import { classifyUrlSegment, type UrlSegment } from '../routing/segment-classify.js';
33
+ import { LinkStatusContext } from './use-link-status.js';
24
34
  import { getRouterOrNull } from './router-ref.js';
35
+ import { getSsrData } from './ssr-data.js';
36
+ import { mergePreservedSearchParams } from '../shared/merge-search-params.js';
37
+ import {
38
+ setLinkForCurrentNavigation,
39
+ unmountLinkForCurrentNavigation,
40
+ LINK_IDLE,
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
 
@@ -33,8 +69,12 @@ export type OnNavigateHandler = (e: OnNavigateEvent) => void;
33
69
 
34
70
  /**
35
71
  * Base props shared by all Link variants.
72
+ *
73
+ * Exported so the public `LinkFunction` interface (declared in
74
+ * `./index.ts`, where module augmentation can merge into it) can
75
+ * compose this without duplication.
36
76
  */
37
- interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'href'> {
77
+ export interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'href'> {
38
78
  /** Prefetch the RSC payload on hover */
39
79
  prefetch?: boolean;
40
80
  /**
@@ -42,6 +82,20 @@ interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'h
42
82
  * Set to false for tabbed interfaces where content changes within a fixed layout.
43
83
  */
44
84
  scroll?: boolean;
85
+ /**
86
+ * Preserve search params from the current URL across navigation.
87
+ *
88
+ * - `true` — preserve ALL current search params (target params take precedence)
89
+ * - `string[]` — preserve only the named params (e.g. `['private', 'token']`)
90
+ *
91
+ * Useful for route-group gating where a search param (e.g. `?private=access`)
92
+ * must persist across internal navigations. The target href's own search params
93
+ * always take precedence over preserved ones.
94
+ *
95
+ * During SSR, reads search params from the request context. On the client,
96
+ * reads from the current URL and updates reactively when the URL changes.
97
+ */
98
+ preserveSearchParams?: true | string[];
45
99
  /**
46
100
  * Called before client-side navigation commits. Call `e.preventDefault()`
47
101
  * to cancel the default navigation — the caller is then responsible for
@@ -54,48 +108,67 @@ interface LinkBaseProps extends Omit<AnchorHTMLAttributes<HTMLAnchorElement>, 'h
54
108
  children?: ReactNode;
55
109
  }
56
110
 
111
+ // ─── Typed Link Props ────────────────────────────────────────────
112
+
57
113
  /**
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.
114
+ * Widen server-side string params to string | number for Link convenience.
115
+ * Exported for use by codegen-generated overloads.
61
116
  */
62
- export interface LinkPropsWithHref extends LinkBaseProps {
63
- 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
- */
69
- searchParams?: {
70
- definition: SearchParamsDefinition<Record<string, unknown>>;
71
- values: Record<string, unknown>;
72
- };
73
- }
117
+ export type LinkSegmentParams<T> = {
118
+ [K in keyof T]: [string] extends [T[K]] ? string | number : T[K];
119
+ };
120
+
121
+ // ─── External Href Types ─────────────────────────────────────────
122
+ //
123
+ // `ExternalHref` and the public `LinkFunction` interface live in
124
+ // `./index.ts` rather than this file. They MUST be originally declared
125
+ // in the same module that the codegen augments (`@timber-js/app/client`)
126
+ // so that codegen-generated per-route call signatures merge with the
127
+ // same interface that types the `Link` constant. Re-exporting an
128
+ // interface via `export type {}` does NOT participate in module
129
+ // augmentation merging — only originally-declared interfaces do.
130
+ // See TIM-624.
131
+
132
+ // ─── searchParams prop shapes ────────────────────────────────────
133
+ //
134
+ // Per-route Link overloads (emitted by codegen) pass the flat values shape:
135
+ // searchParams={{ page: 2, q: 'boots' }}
136
+ // The framework looks up the route's SearchParamsDefinition from the
137
+ // search-params registry at runtime (see TIM-830).
138
+ //
139
+ // The catch-all overload in client/index.ts (external/computed hrefs)
140
+ // additionally accepts the legacy wrapped shape:
141
+ // searchParams={{ definition: def, values: { page: 2 } }}
142
+ // because there is no way to look up a definition from a computed string.
143
+ //
144
+ // `resolveHref` discriminates at runtime by presence of a `definition` key.
145
+ type WrappedSearchParamsProp = {
146
+ definition: SearchParamsDefinition<Record<string, unknown>>;
147
+ values: Record<string, unknown>;
148
+ };
149
+ type FlatSearchParamsProp = Record<string, unknown>;
150
+ type LinkSearchParamsProp = WrappedSearchParamsProp | FlatSearchParamsProp;
74
151
 
75
152
  /**
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 }}>
153
+ * Runtime-only loose props used internally by the Link implementation.
154
+ * Not exposed to callers the public API uses LinkFunction.
79
155
  */
80
- export interface LinkPropsWithParams extends LinkBaseProps {
81
- /** Route pattern with dynamic segments (e.g. "/products/[id]") */
156
+ interface LinkRuntimeProps extends LinkBaseProps {
82
157
  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
- */
92
- searchParams?: {
93
- definition: SearchParamsDefinition<Record<string, unknown>>;
94
- values: Record<string, unknown>;
95
- };
158
+ segmentParams?: Record<string, string | number | string[]>;
159
+ searchParams?: LinkSearchParamsProp;
96
160
  }
97
161
 
98
- export type LinkProps = LinkPropsWithHref | LinkPropsWithParams;
162
+ // Legacy exports for backward compat (used by buildLinkProps, tests, etc.)
163
+ export type LinkPropsWithHref = LinkBaseProps & {
164
+ href: string;
165
+ segmentParams?: never;
166
+ searchParams?: LinkSearchParamsProp;
167
+ };
168
+ export type LinkPropsWithParams = LinkRuntimeProps & {
169
+ segmentParams: Record<string, string | number | string[]>;
170
+ };
171
+ export type LinkProps = LinkRuntimeProps;
99
172
 
100
173
  // ─── Dangerous URL Scheme Detection ──────────────────────────────
101
174
 
@@ -141,57 +214,91 @@ export function isInternalHref(href: string): boolean {
141
214
  * - [...param] → catch-all (joined with /)
142
215
  * - [[...param]] → optional catch-all (omitted if undefined/empty)
143
216
  */
217
+ /**
218
+ * Parse a route pattern's path portion into classified segments.
219
+ * Exported for testing. Uses the shared character-based classifier.
220
+ */
221
+ export function parseSegments(pattern: string): UrlSegment[] {
222
+ return pattern.split('/').filter(Boolean).map(classifyUrlSegment);
223
+ }
224
+
225
+ /**
226
+ * Resolve a single classified segment into its string representation.
227
+ * Returns null for optional catch-all with no value (filtered out before join).
228
+ */
229
+ function resolveSegment(
230
+ seg: UrlSegment,
231
+ params: Record<string, string | number | string[]>,
232
+ pattern: string
233
+ ): string | null {
234
+ switch (seg.kind) {
235
+ case 'static':
236
+ return seg.value;
237
+
238
+ case 'optional-catch-all': {
239
+ const value = params[seg.name];
240
+ if (value === undefined || (Array.isArray(value) && value.length === 0)) {
241
+ return null;
242
+ }
243
+ const segments = Array.isArray(value) ? value : [value];
244
+ return segments.map(encodeURIComponent).join('/');
245
+ }
246
+
247
+ case 'catch-all': {
248
+ const value = params[seg.name];
249
+ if (value === undefined) {
250
+ throw new Error(
251
+ `<Link> missing required catch-all param "${seg.name}" for pattern "${pattern}".`
252
+ );
253
+ }
254
+ const segments = Array.isArray(value) ? value : [value];
255
+ if (segments.length === 0) {
256
+ throw new Error(
257
+ `<Link> catch-all param "${seg.name}" must have at least one segment for pattern "${pattern}".`
258
+ );
259
+ }
260
+ return segments.map(encodeURIComponent).join('/');
261
+ }
262
+
263
+ case 'dynamic': {
264
+ const value = params[seg.name];
265
+ if (value === undefined) {
266
+ throw new Error(`<Link> missing required param "${seg.name}" for pattern "${pattern}".`);
267
+ }
268
+ if (Array.isArray(value)) {
269
+ throw new Error(
270
+ `<Link> param "${seg.name}" expected a string but received an array for pattern "${pattern}".`
271
+ );
272
+ }
273
+ return encodeURIComponent(String(value));
274
+ }
275
+ }
276
+ }
277
+
278
+ /**
279
+ * Split a URL pattern into the path portion and any trailing ?query/#hash suffix.
280
+ * Uses URL parsing for correctness rather than manual index arithmetic.
281
+ */
282
+ function splitPatternSuffix(pattern: string): [path: string, suffix: string] {
283
+ if (!pattern.includes('?') && !pattern.includes('#')) {
284
+ return [pattern, ''];
285
+ }
286
+ const url = new URL(pattern, 'http://x');
287
+ const suffix = url.search + url.hash;
288
+ const path = pattern.slice(0, pattern.length - suffix.length);
289
+ return [path, suffix];
290
+ }
291
+
144
292
  export function interpolateParams(
145
293
  pattern: string,
146
294
  params: Record<string, string | number | string[]>
147
295
  ): 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
- }
161
-
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
- }
296
+ const [pathPart, suffix] = splitPatternSuffix(pattern);
177
297
 
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
- );
298
+ const resolved = parseSegments(pathPart)
299
+ .map((seg) => resolveSegment(seg, params, pattern))
300
+ .filter((s): s is string => s !== null);
301
+ return ('/' + resolved.join('/') || '/') + suffix;
195
302
  }
196
303
 
197
304
  // ─── Resolve Href ───────────────────────────────────────────────
@@ -204,13 +311,19 @@ export function interpolateParams(
204
311
  * - searchParams serialization via SearchParamsDefinition
205
312
  * - Validation that searchParams and inline query strings are exclusive
206
313
  */
314
+ /**
315
+ * Runtime discriminator: treat `searchParams` as the legacy wrapped shape
316
+ * only when it literally has a `definition` key. Everything else is the
317
+ * flat `Partial<T>` values shape (TIM-830).
318
+ */
319
+ function isWrappedSearchParamsProp(sp: LinkSearchParamsProp): sp is WrappedSearchParamsProp {
320
+ return 'definition' in sp;
321
+ }
322
+
207
323
  export function resolveHref(
208
324
  href: string,
209
325
  params?: Record<string, string | number | string[]>,
210
- searchParams?: {
211
- definition: SearchParamsDefinition<Record<string, unknown>>;
212
- values: Record<string, unknown>;
213
- }
326
+ searchParams?: LinkSearchParamsProp
214
327
  ): string {
215
328
  let resolvedPath = href;
216
329
 
@@ -229,7 +342,36 @@ export function resolveHref(
229
342
  );
230
343
  }
231
344
 
232
- const qs = searchParams.definition.serialize(searchParams.values);
345
+ let definition: SearchParamsDefinition<Record<string, unknown>> | undefined;
346
+ let values: Record<string, unknown>;
347
+
348
+ if (isWrappedSearchParamsProp(searchParams)) {
349
+ // Legacy wrapped shape — used by the catch-all overload for
350
+ // computed/external hrefs where no route lookup is possible.
351
+ definition = searchParams.definition;
352
+ values = searchParams.values;
353
+ } else {
354
+ // Flat shape (TIM-830): look up the definition from the runtime
355
+ // registry using the un-interpolated href pattern (e.g. '/products/[id]').
356
+ // The search-params registry is populated eagerly at startup by the
357
+ // virtual:timber-search-params-registry module generated by the
358
+ // timber-routing Vite plugin.
359
+ definition = getSearchParamsDefinition(href) as
360
+ | SearchParamsDefinition<Record<string, unknown>>
361
+ | undefined;
362
+ values = searchParams;
363
+ if (!definition) {
364
+ throw new Error(
365
+ `<Link> received a flat searchParams object for href "${href}", but no ` +
366
+ `SearchParamsDefinition is registered for that route. ` +
367
+ `Either the route does not export \`searchParams\` from its \`params.ts\`/\`page.tsx\`, ` +
368
+ `or the search-params registry module was not loaded. ` +
369
+ `For external or computed hrefs, pass the legacy \`{ definition, values }\` shape instead.`
370
+ );
371
+ }
372
+ }
373
+
374
+ const qs = definition.serialize(values);
233
375
  if (qs) {
234
376
  resolvedPath = `${resolvedPath}?${qs}`;
235
377
  }
@@ -251,10 +393,7 @@ interface LinkOutputProps {
251
393
  export function buildLinkProps(
252
394
  props: Pick<LinkPropsWithHref, 'href'> & {
253
395
  params?: Record<string, string | number | string[]>;
254
- searchParams?: {
255
- definition: SearchParamsDefinition<Record<string, unknown>>;
256
- values: Record<string, unknown>;
257
- };
396
+ searchParams?: LinkSearchParamsProp;
258
397
  }
259
398
  ): LinkOutputProps {
260
399
  const resolvedHref = resolveHref(props.href, props.params, props.searchParams);
@@ -302,23 +441,76 @@ function shouldInterceptClick(
302
441
  * navigation via the router. No global event delegation — each Link owns
303
442
  * its own click handling.
304
443
  *
305
- * Supports typed routes via codegen overloads. At runtime:
306
- * - `params` prop interpolates dynamic segments in the href pattern
444
+ * Supports typed routes via the Routes interface (populated by codegen).
445
+ * At runtime:
446
+ * - `segmentParams` prop interpolates dynamic segments in the href pattern
307
447
  * - `searchParams` prop serializes query parameters via a SearchParamsDefinition
448
+ *
449
+ * Typed via the LinkFunction callable interface. The base call signature
450
+ * forbids segmentParams; per-route signatures are added by codegen via
451
+ * interface merging. See TIM-624.
308
452
  */
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 });
453
+ // Cast to LinkFunction — the callable interface provides the public type,
454
+ // but the implementation destructures LinkRuntimeProps internally.
455
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
456
+ export const Link: LinkFunction = function LinkImpl(props: any) {
457
+ const {
458
+ href,
459
+ prefetch,
460
+ scroll,
461
+ segmentParams,
462
+ searchParams,
463
+ preserveSearchParams,
464
+ onNavigate,
465
+ onClick: userOnClick,
466
+ onMouseEnter: userOnMouseEnter,
467
+ children,
468
+ ...rest
469
+ } = props as LinkRuntimeProps;
470
+ const { href: baseHref } = buildLinkProps({ href, params: segmentParams, searchParams });
471
+
472
+ // ─── Per-link pending state (useOptimistic) ────────────────────────
473
+ // Each Link has its own pending state. Only the clicked link's
474
+ // setter is invoked during navigation — zero other links re-render.
475
+ //
476
+ // Link click stores the instance; NavigationRoot activates inside startTransition.
477
+ // useOptimistic auto-reverts to LINK_IDLE when the navigation
478
+ // startTransition — batched with the new tree commit.
479
+ //
480
+ // See design/19-client-navigation.md §"Per-Link Pending State"
481
+ const [linkStatus, setIsPending] = useOptimistic(LINK_IDLE);
482
+
483
+ // Build the link instance ref for the pending store.
484
+ // The ref is stable across renders — we update the setter on each
485
+ // render to keep it current.
486
+ const linkInstanceRef = useRef<LinkPendingInstance | null>(null);
487
+ if (!linkInstanceRef.current) {
488
+ linkInstanceRef.current = { setIsPending };
489
+ } else {
490
+ linkInstanceRef.current.setIsPending = setIsPending;
491
+ }
492
+
493
+ // Clean up if this link unmounts while it's the current navigation link.
494
+ // Prevents calling setOptimistic on an unmounted component.
495
+ useEffect(() => {
496
+ const instance = linkInstanceRef.current;
497
+ return () => {
498
+ if (instance) {
499
+ unmountLinkForCurrentNavigation(instance);
500
+ }
501
+ };
502
+ }, []);
503
+
504
+ // Preserve search params from the current URL when requested.
505
+ // useSearchParams() works during both SSR (reads from request context)
506
+ // and on the client (reads from window.location, reactive to URL changes).
507
+ // We read current search params directly to avoid unconditional hook calls.
508
+ // On the client, window.location.search is always current; during SSR,
509
+ // getSsrData() provides the request's search params.
510
+ const resolvedHref = preserveSearchParams
511
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
512
+ : baseHref;
513
+
322
514
  const internal = isInternalHref(resolvedHref);
323
515
 
324
516
  // ─── Click handler ───────────────────────────────────────────
@@ -349,9 +541,42 @@ export function Link({
349
541
  const router = getRouterOrNull();
350
542
  if (!router) return; // SSR or pre-hydration — fall through to browser nav
351
543
 
352
- event.preventDefault();
353
544
  const shouldScroll = scroll !== false;
354
- void router.navigate(resolvedHref, { scroll: shouldScroll });
545
+
546
+ // Register this link in the pending store. The actual
547
+ // setIsPending(LINK_PENDING) call happens inside NavigationRoot's
548
+ // async startTransition via activateLinkPending().
549
+
550
+ setLinkForCurrentNavigation(linkInstanceRef.current);
551
+
552
+ // When Navigation API is active, let the <a> click propagate
553
+ // naturally — do NOT call preventDefault(). The navigate event
554
+ // handler intercepts it and runs the RSC pipeline. This is a
555
+ // user-initiated navigation, so Chrome shows the native loading
556
+ // indicator (tab spinner). Metadata (scroll, link instance) is
557
+ // passed via nav-link-store so the handler can configure the nav.
558
+ //
559
+ // Without Navigation API (fallback), preventDefault and drive
560
+ // navigation through the router as before.
561
+ if (hasNavigationApi()) {
562
+ setNavLinkMetadata({
563
+ scroll: shouldScroll,
564
+ linkInstance: linkInstanceRef.current,
565
+ });
566
+ // Don't preventDefault — let the <a> click fire the navigate event
567
+ return;
568
+ }
569
+
570
+ // History API fallback — prevent default and navigate via router
571
+ event.preventDefault();
572
+
573
+ // Re-merge preserved search params at click time to pick up any
574
+ // URL changes since render (e.g. from other navigations or pushState).
575
+ const navHref = preserveSearchParams
576
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
577
+ : resolvedHref;
578
+
579
+ void router.navigate(navHref, { scroll: shouldScroll });
355
580
  }
356
581
  : userOnClick; // External links — just pass through user's onClick
357
582
 
@@ -362,14 +587,18 @@ export function Link({
362
587
  userOnMouseEnter?.(event);
363
588
  const router = getRouterOrNull();
364
589
  if (router) {
365
- router.prefetch(resolvedHref);
590
+ // Re-merge preserved search params at hover time for fresh prefetch URL
591
+ const prefetchHref = preserveSearchParams
592
+ ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams)
593
+ : resolvedHref;
594
+ router.prefetch(prefetchHref);
366
595
  }
367
596
  }
368
597
  : userOnMouseEnter;
369
598
 
370
599
  return (
371
600
  <a {...rest} href={resolvedHref} onClick={handleClick} onMouseEnter={handleMouseEnter}>
372
- <LinkStatusProvider href={resolvedHref}>{children}</LinkStatusProvider>
601
+ <LinkStatusContext.Provider value={linkStatus}>{children}</LinkStatusContext.Provider>
373
602
  </a>
374
603
  );
375
- }
604
+ };
@@ -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
+ }