@timber-js/app 0.2.0-alpha.8 → 0.2.0-alpha.81

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 (571) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/actions-Dg-ANYHb.js +421 -0
  3. package/dist/_chunks/actions-Dg-ANYHb.js.map +1 -0
  4. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
  5. package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
  6. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  7. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  8. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  9. package/dist/_chunks/define-C77ScO0m.js +106 -0
  10. package/dist/_chunks/define-C77ScO0m.js.map +1 -0
  11. package/dist/_chunks/define-CZqDwhSu.js +199 -0
  12. package/dist/_chunks/define-CZqDwhSu.js.map +1 -0
  13. package/dist/_chunks/define-cookie-C2IkoFGN.js +94 -0
  14. package/dist/_chunks/define-cookie-C2IkoFGN.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
  16. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
  17. package/dist/_chunks/format-CYBGxKtc.js +14 -0
  18. package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-Dpn_UfAD.js} +171 -97
  20. package/dist/_chunks/interception-Dpn_UfAD.js.map +1 -0
  21. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
  22. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
  23. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  24. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  25. package/dist/_chunks/request-context-qMsWgy9C.js +478 -0
  26. package/dist/_chunks/request-context-qMsWgy9C.js.map +1 -0
  27. package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
  28. package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
  29. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  30. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  31. package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
  32. package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
  33. package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
  34. package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
  35. package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
  36. package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
  37. package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
  38. package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
  39. package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
  40. package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
  41. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-Lo_s_pw2.js} +4 -4
  42. package/dist/_chunks/use-query-states-Lo_s_pw2.js.map +1 -0
  43. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  44. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  45. package/dist/adapters/cloudflare-dev.js +73 -0
  46. package/dist/adapters/cloudflare-dev.js.map +1 -0
  47. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  48. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  49. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  50. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  51. package/dist/adapters/cloudflare.d.ts +148 -12
  52. package/dist/adapters/cloudflare.d.ts.map +1 -1
  53. package/dist/adapters/cloudflare.js +135 -11
  54. package/dist/adapters/cloudflare.js.map +1 -1
  55. package/dist/adapters/compress-module.d.ts.map +1 -1
  56. package/dist/adapters/nitro.d.ts +17 -1
  57. package/dist/adapters/nitro.d.ts.map +1 -1
  58. package/dist/adapters/nitro.js +56 -13
  59. package/dist/adapters/nitro.js.map +1 -1
  60. package/dist/cache/cache-api.d.ts +24 -0
  61. package/dist/cache/cache-api.d.ts.map +1 -0
  62. package/dist/cache/handler-store.d.ts +31 -0
  63. package/dist/cache/handler-store.d.ts.map +1 -0
  64. package/dist/cache/index.d.ts +23 -7
  65. package/dist/cache/index.d.ts.map +1 -1
  66. package/dist/cache/index.js +142 -80
  67. package/dist/cache/index.js.map +1 -1
  68. package/dist/cache/singleflight.d.ts +18 -1
  69. package/dist/cache/singleflight.d.ts.map +1 -1
  70. package/dist/cache/sizeof.d.ts +22 -0
  71. package/dist/cache/sizeof.d.ts.map +1 -0
  72. package/dist/cache/timber-cache.d.ts +1 -1
  73. package/dist/cache/timber-cache.d.ts.map +1 -1
  74. package/dist/cli.d.ts +6 -1
  75. package/dist/cli.d.ts.map +1 -1
  76. package/dist/cli.js +8 -3
  77. package/dist/cli.js.map +1 -1
  78. package/dist/client/browser-dev.d.ts +27 -1
  79. package/dist/client/browser-dev.d.ts.map +1 -1
  80. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  81. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  82. package/dist/client/browser-entry/hmr.d.ts +21 -0
  83. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  84. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  85. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  86. package/dist/client/browser-entry/index.d.ts +30 -0
  87. package/dist/client/browser-entry/index.d.ts.map +1 -0
  88. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  89. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  90. package/dist/client/browser-entry/router-init.d.ts +23 -0
  91. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  92. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  93. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  94. package/dist/client/browser-entry/scroll.d.ts +19 -0
  95. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  96. package/dist/client/error-boundary.d.ts +12 -5
  97. package/dist/client/error-boundary.d.ts.map +1 -1
  98. package/dist/client/error-boundary.js +10 -4
  99. package/dist/client/error-boundary.js.map +1 -1
  100. package/dist/client/error-reconstituter.d.ts +54 -0
  101. package/dist/client/error-reconstituter.d.ts.map +1 -0
  102. package/dist/client/form.d.ts +3 -3
  103. package/dist/client/form.d.ts.map +1 -1
  104. package/dist/client/history.d.ts +19 -4
  105. package/dist/client/history.d.ts.map +1 -1
  106. package/dist/client/index.d.ts +7 -21
  107. package/dist/client/index.d.ts.map +1 -1
  108. package/dist/client/index.js +210 -1017
  109. package/dist/client/index.js.map +1 -1
  110. package/dist/client/internal.d.ts +18 -0
  111. package/dist/client/internal.d.ts.map +1 -0
  112. package/dist/client/internal.js +890 -0
  113. package/dist/client/internal.js.map +1 -0
  114. package/dist/client/link-pending-store.d.ts +63 -0
  115. package/dist/client/link-pending-store.d.ts.map +1 -0
  116. package/dist/client/link.d.ts +90 -32
  117. package/dist/client/link.d.ts.map +1 -1
  118. package/dist/client/nav-link-store.d.ts +36 -0
  119. package/dist/client/nav-link-store.d.ts.map +1 -0
  120. package/dist/client/navigation-api-types.d.ts +90 -0
  121. package/dist/client/navigation-api-types.d.ts.map +1 -0
  122. package/dist/client/navigation-api.d.ts +115 -0
  123. package/dist/client/navigation-api.d.ts.map +1 -0
  124. package/dist/client/navigation-context.d.ts +13 -2
  125. package/dist/client/navigation-context.d.ts.map +1 -1
  126. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  127. package/dist/client/navigation-root.d.ts.map +1 -0
  128. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  129. package/dist/client/router-ref.d.ts +1 -1
  130. package/dist/client/router.d.ts +70 -4
  131. package/dist/client/router.d.ts.map +1 -1
  132. package/dist/client/rsc-fetch.d.ts +38 -3
  133. package/dist/client/rsc-fetch.d.ts.map +1 -1
  134. package/dist/client/segment-cache.d.ts +1 -1
  135. package/dist/client/segment-cache.d.ts.map +1 -1
  136. package/dist/client/segment-outlet.d.ts +63 -0
  137. package/dist/client/segment-outlet.d.ts.map +1 -0
  138. package/dist/client/ssr-data.d.ts +13 -4
  139. package/dist/client/ssr-data.d.ts.map +1 -1
  140. package/dist/client/stale-reload.d.ts +15 -0
  141. package/dist/client/stale-reload.d.ts.map +1 -1
  142. package/dist/client/top-loader.d.ts +5 -5
  143. package/dist/client/top-loader.d.ts.map +1 -1
  144. package/dist/client/use-link-status.d.ts +5 -5
  145. package/dist/client/use-link-status.d.ts.map +1 -1
  146. package/dist/client/use-params.d.ts +6 -4
  147. package/dist/client/use-params.d.ts.map +1 -1
  148. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  149. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  150. package/dist/client/use-query-states.d.ts +1 -1
  151. package/dist/client/use-query-states.d.ts.map +1 -1
  152. package/dist/client/use-router.d.ts +1 -1
  153. package/dist/codec.d.ts +33 -0
  154. package/dist/codec.d.ts.map +1 -0
  155. package/dist/codec.js +2 -0
  156. package/dist/config-types.d.ts +227 -0
  157. package/dist/config-types.d.ts.map +1 -0
  158. package/dist/content/index.d.ts +1 -10
  159. package/dist/content/index.d.ts.map +1 -1
  160. package/dist/content/index.js +0 -2
  161. package/dist/cookies/define-cookie.d.ts +35 -14
  162. package/dist/cookies/define-cookie.d.ts.map +1 -1
  163. package/dist/cookies/index.js +1 -83
  164. package/dist/fonts/css.d.ts +1 -0
  165. package/dist/fonts/css.d.ts.map +1 -1
  166. package/dist/index.d.ts +45 -192
  167. package/dist/index.d.ts.map +1 -1
  168. package/dist/index.js +12357 -11925
  169. package/dist/index.js.map +1 -1
  170. package/dist/plugin-context.d.ts +107 -0
  171. package/dist/plugin-context.d.ts.map +1 -0
  172. package/dist/plugins/adapter-build.d.ts +1 -1
  173. package/dist/plugins/adapter-build.d.ts.map +1 -1
  174. package/dist/plugins/build-manifest.d.ts +2 -2
  175. package/dist/plugins/build-manifest.d.ts.map +1 -1
  176. package/dist/plugins/build-report.d.ts +3 -3
  177. package/dist/plugins/build-report.d.ts.map +1 -1
  178. package/dist/plugins/client-chunks.d.ts +32 -0
  179. package/dist/plugins/client-chunks.d.ts.map +1 -0
  180. package/dist/plugins/content.d.ts +1 -1
  181. package/dist/plugins/content.d.ts.map +1 -1
  182. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  183. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  184. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  185. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  186. package/dist/plugins/dev-logs.d.ts +1 -1
  187. package/dist/plugins/dev-logs.d.ts.map +1 -1
  188. package/dist/plugins/dev-server.d.ts +1 -1
  189. package/dist/plugins/dev-server.d.ts.map +1 -1
  190. package/dist/plugins/entries.d.ts +1 -1
  191. package/dist/plugins/entries.d.ts.map +1 -1
  192. package/dist/plugins/fonts.d.ts +19 -5
  193. package/dist/plugins/fonts.d.ts.map +1 -1
  194. package/dist/plugins/mdx.d.ts +1 -1
  195. package/dist/plugins/mdx.d.ts.map +1 -1
  196. package/dist/plugins/routing.d.ts +1 -1
  197. package/dist/plugins/routing.d.ts.map +1 -1
  198. package/dist/plugins/server-bundle.d.ts.map +1 -1
  199. package/dist/plugins/shims.d.ts +6 -5
  200. package/dist/plugins/shims.d.ts.map +1 -1
  201. package/dist/plugins/static-build.d.ts +4 -4
  202. package/dist/plugins/static-build.d.ts.map +1 -1
  203. package/dist/routing/codegen.d.ts +2 -2
  204. package/dist/routing/codegen.d.ts.map +1 -1
  205. package/dist/routing/index.d.ts +2 -0
  206. package/dist/routing/index.d.ts.map +1 -1
  207. package/dist/routing/index.js +3 -2
  208. package/dist/routing/scanner.d.ts.map +1 -1
  209. package/dist/routing/segment-classify.d.ts +46 -0
  210. package/dist/routing/segment-classify.d.ts.map +1 -0
  211. package/dist/routing/status-file-lint.d.ts +2 -1
  212. package/dist/routing/status-file-lint.d.ts.map +1 -1
  213. package/dist/routing/types.d.ts +16 -4
  214. package/dist/routing/types.d.ts.map +1 -1
  215. package/dist/rsc-runtime/rsc.d.ts +1 -1
  216. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  217. package/dist/rsc-runtime/ssr.d.ts +12 -0
  218. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  219. package/dist/schema-bridge.d.ts +76 -0
  220. package/dist/schema-bridge.d.ts.map +1 -0
  221. package/dist/search-params/define.d.ts +139 -0
  222. package/dist/search-params/define.d.ts.map +1 -0
  223. package/dist/search-params/index.d.ts +4 -7
  224. package/dist/search-params/index.d.ts.map +1 -1
  225. package/dist/search-params/index.js +32 -441
  226. package/dist/search-params/index.js.map +1 -1
  227. package/dist/search-params/registry.d.ts +2 -2
  228. package/dist/search-params/registry.d.ts.map +1 -1
  229. package/dist/search-params/wrappers.d.ts +53 -0
  230. package/dist/search-params/wrappers.d.ts.map +1 -0
  231. package/dist/segment-params/define.d.ts +78 -0
  232. package/dist/segment-params/define.d.ts.map +1 -0
  233. package/dist/segment-params/index.d.ts +3 -0
  234. package/dist/segment-params/index.d.ts.map +1 -0
  235. package/dist/segment-params/index.js +2 -0
  236. package/dist/server/access-gate.d.ts +4 -0
  237. package/dist/server/access-gate.d.ts.map +1 -1
  238. package/dist/server/action-client.d.ts +25 -6
  239. package/dist/server/action-client.d.ts.map +1 -1
  240. package/dist/server/action-encryption.d.ts +76 -0
  241. package/dist/server/action-encryption.d.ts.map +1 -0
  242. package/dist/server/action-handler.d.ts.map +1 -1
  243. package/dist/server/actions.d.ts +3 -6
  244. package/dist/server/actions.d.ts.map +1 -1
  245. package/dist/server/als-registry.d.ts +32 -4
  246. package/dist/server/als-registry.d.ts.map +1 -1
  247. package/dist/server/build-manifest.d.ts +2 -2
  248. package/dist/server/build-manifest.d.ts.map +1 -1
  249. package/dist/server/debug.d.ts +1 -1
  250. package/dist/server/default-logger.d.ts +22 -0
  251. package/dist/server/default-logger.d.ts.map +1 -0
  252. package/dist/server/deny-page-resolver.d.ts +52 -0
  253. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  254. package/dist/server/deny-renderer.d.ts.map +1 -1
  255. package/dist/server/dev-holding-server.d.ts +52 -0
  256. package/dist/server/dev-holding-server.d.ts.map +1 -0
  257. package/dist/server/dev-warnings.d.ts +1 -21
  258. package/dist/server/dev-warnings.d.ts.map +1 -1
  259. package/dist/server/early-hints.d.ts +13 -5
  260. package/dist/server/early-hints.d.ts.map +1 -1
  261. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  262. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  263. package/dist/server/fallback-error.d.ts +4 -3
  264. package/dist/server/fallback-error.d.ts.map +1 -1
  265. package/dist/server/flight-injection-state.d.ts +66 -0
  266. package/dist/server/flight-injection-state.d.ts.map +1 -0
  267. package/dist/server/flight-scripts.d.ts +42 -0
  268. package/dist/server/flight-scripts.d.ts.map +1 -0
  269. package/dist/server/flush.d.ts.map +1 -1
  270. package/dist/server/form-data.d.ts +29 -0
  271. package/dist/server/form-data.d.ts.map +1 -1
  272. package/dist/server/html-injectors.d.ts +51 -11
  273. package/dist/server/html-injectors.d.ts.map +1 -1
  274. package/dist/server/index.d.ts +5 -43
  275. package/dist/server/index.d.ts.map +1 -1
  276. package/dist/server/index.js +37 -2798
  277. package/dist/server/index.js.map +1 -1
  278. package/dist/server/internal.d.ts +46 -0
  279. package/dist/server/internal.d.ts.map +1 -0
  280. package/dist/server/internal.js +2883 -0
  281. package/dist/server/internal.js.map +1 -0
  282. package/dist/server/logger.d.ts +25 -7
  283. package/dist/server/logger.d.ts.map +1 -1
  284. package/dist/server/middleware-runner.d.ts +19 -4
  285. package/dist/server/middleware-runner.d.ts.map +1 -1
  286. package/dist/server/node-stream-transforms.d.ts +113 -0
  287. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  288. package/dist/server/page-deny-boundary.d.ts +31 -0
  289. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  290. package/dist/server/pipeline-interception.d.ts +1 -1
  291. package/dist/server/pipeline-interception.d.ts.map +1 -1
  292. package/dist/server/pipeline-metadata.d.ts +6 -0
  293. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  294. package/dist/server/pipeline.d.ts +42 -10
  295. package/dist/server/pipeline.d.ts.map +1 -1
  296. package/dist/server/primitives.d.ts +69 -18
  297. package/dist/server/primitives.d.ts.map +1 -1
  298. package/dist/server/render-timeout.d.ts +51 -0
  299. package/dist/server/render-timeout.d.ts.map +1 -0
  300. package/dist/server/request-context.d.ts +112 -43
  301. package/dist/server/request-context.d.ts.map +1 -1
  302. package/dist/server/route-element-builder.d.ts +27 -1
  303. package/dist/server/route-element-builder.d.ts.map +1 -1
  304. package/dist/server/route-handler.d.ts.map +1 -1
  305. package/dist/server/route-matcher.d.ts +9 -2
  306. package/dist/server/route-matcher.d.ts.map +1 -1
  307. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  308. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  309. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  310. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  311. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  312. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  313. package/dist/server/rsc-entry/index.d.ts +8 -3
  314. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  315. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  316. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  317. package/dist/server/rsc-entry/rsc-stream.d.ts +4 -1
  318. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  319. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  320. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  321. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  322. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  323. package/dist/server/safe-load.d.ts +46 -0
  324. package/dist/server/safe-load.d.ts.map +1 -0
  325. package/dist/server/sitemap-generator.d.ts +129 -0
  326. package/dist/server/sitemap-generator.d.ts.map +1 -0
  327. package/dist/server/sitemap-handler.d.ts +22 -0
  328. package/dist/server/sitemap-handler.d.ts.map +1 -0
  329. package/dist/server/slot-resolver.d.ts +1 -1
  330. package/dist/server/slot-resolver.d.ts.map +1 -1
  331. package/dist/server/ssr-entry.d.ts +22 -0
  332. package/dist/server/ssr-entry.d.ts.map +1 -1
  333. package/dist/server/ssr-render.d.ts +39 -21
  334. package/dist/server/ssr-render.d.ts.map +1 -1
  335. package/dist/server/ssr-wrappers.d.ts +50 -0
  336. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  337. package/dist/server/status-code-resolver.d.ts +1 -1
  338. package/dist/server/status-code-resolver.d.ts.map +1 -1
  339. package/dist/server/stream-utils.d.ts +36 -0
  340. package/dist/server/stream-utils.d.ts.map +1 -0
  341. package/dist/server/tracing.d.ts +4 -4
  342. package/dist/server/tracing.d.ts.map +1 -1
  343. package/dist/server/tree-builder.d.ts +22 -19
  344. package/dist/server/tree-builder.d.ts.map +1 -1
  345. package/dist/server/types.d.ts +1 -4
  346. package/dist/server/types.d.ts.map +1 -1
  347. package/dist/server/version-skew.d.ts +61 -0
  348. package/dist/server/version-skew.d.ts.map +1 -0
  349. package/dist/shared/merge-search-params.d.ts +22 -0
  350. package/dist/shared/merge-search-params.d.ts.map +1 -0
  351. package/dist/shims/font-google.d.ts +1 -1
  352. package/dist/shims/font-google.d.ts.map +1 -1
  353. package/dist/shims/font-google.js +42 -0
  354. package/dist/shims/font-google.js.map +1 -0
  355. package/dist/shims/font-local.d.ts +26 -0
  356. package/dist/shims/font-local.d.ts.map +1 -0
  357. package/dist/shims/font-local.js +20 -0
  358. package/dist/shims/font-local.js.map +1 -0
  359. package/dist/shims/headers.d.ts +2 -1
  360. package/dist/shims/headers.d.ts.map +1 -1
  361. package/dist/shims/navigation-client.d.ts +1 -1
  362. package/dist/shims/navigation-client.d.ts.map +1 -1
  363. package/dist/shims/navigation.d.ts +3 -2
  364. package/dist/shims/navigation.d.ts.map +1 -1
  365. package/dist/utils/directive-parser.d.ts +5 -2
  366. package/dist/utils/directive-parser.d.ts.map +1 -1
  367. package/dist/utils/state-machine.d.ts +80 -0
  368. package/dist/utils/state-machine.d.ts.map +1 -0
  369. package/package.json +56 -22
  370. package/src/adapters/cloudflare-dev.ts +177 -0
  371. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  372. package/src/adapters/cloudflare.ts +342 -28
  373. package/src/adapters/compress-module.ts +24 -4
  374. package/src/adapters/nitro.ts +52 -8
  375. package/src/adapters/wrangler.d.ts +7 -0
  376. package/src/cache/cache-api.ts +38 -0
  377. package/src/cache/handler-store.ts +68 -0
  378. package/src/cache/index.ts +81 -18
  379. package/src/cache/singleflight.ts +62 -4
  380. package/src/cache/sizeof.ts +31 -0
  381. package/src/cache/timber-cache.ts +24 -20
  382. package/src/cli.ts +16 -6
  383. package/src/client/browser-dev.ts +128 -1
  384. package/src/client/browser-entry/action-dispatch.ts +116 -0
  385. package/src/client/browser-entry/hmr.ts +81 -0
  386. package/src/client/browser-entry/hydrate.ts +145 -0
  387. package/src/client/browser-entry/index.ts +138 -0
  388. package/src/client/browser-entry/post-hydration.ts +119 -0
  389. package/src/client/browser-entry/router-init.ts +193 -0
  390. package/src/client/browser-entry/rsc-stream.ts +157 -0
  391. package/src/client/browser-entry/scroll.ts +27 -0
  392. package/src/client/error-boundary.tsx +48 -16
  393. package/src/client/error-reconstituter.tsx +65 -0
  394. package/src/client/form.tsx +9 -7
  395. package/src/client/history.ts +26 -4
  396. package/src/client/index.ts +19 -38
  397. package/src/client/internal.ts +57 -0
  398. package/src/client/link-pending-store.ts +111 -0
  399. package/src/client/link.tsx +329 -97
  400. package/src/client/nav-link-store.ts +47 -0
  401. package/src/client/navigation-api-types.ts +112 -0
  402. package/src/client/navigation-api.ts +332 -0
  403. package/src/client/navigation-context.ts +31 -6
  404. package/src/client/navigation-root.tsx +342 -0
  405. package/src/client/nuqs-adapter.tsx +16 -3
  406. package/src/client/router-ref.ts +1 -1
  407. package/src/client/router.ts +299 -72
  408. package/src/client/rsc-fetch.ts +97 -8
  409. package/src/client/segment-cache.ts +1 -1
  410. package/src/client/segment-outlet.tsx +86 -0
  411. package/src/client/ssr-data.ts +13 -5
  412. package/src/client/stale-reload.ts +72 -3
  413. package/src/client/top-loader.tsx +16 -8
  414. package/src/client/use-link-status.ts +7 -7
  415. package/src/client/use-params.ts +7 -5
  416. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  417. package/src/client/use-query-states.ts +3 -3
  418. package/src/client/use-router.ts +1 -1
  419. package/src/codec.ts +49 -0
  420. package/src/config-types.ts +225 -0
  421. package/src/content/index.ts +5 -13
  422. package/src/cookies/define-cookie.ts +78 -25
  423. package/src/cookies/index.ts +8 -0
  424. package/src/fonts/css.ts +2 -1
  425. package/src/index.ts +295 -354
  426. package/src/plugin-context.ts +240 -0
  427. package/src/plugins/adapter-build.ts +9 -3
  428. package/src/plugins/build-manifest.ts +13 -2
  429. package/src/plugins/build-report.ts +3 -3
  430. package/src/plugins/client-chunks.ts +65 -0
  431. package/src/plugins/content.ts +1 -1
  432. package/src/plugins/dev-browser-logs.ts +288 -0
  433. package/src/plugins/dev-error-overlay.ts +70 -1
  434. package/src/plugins/dev-logs.ts +1 -1
  435. package/src/plugins/dev-server.ts +70 -9
  436. package/src/plugins/entries.ts +71 -10
  437. package/src/plugins/fonts.ts +168 -61
  438. package/src/plugins/mdx.ts +1 -1
  439. package/src/plugins/routing.ts +57 -17
  440. package/src/plugins/server-action-exports.ts +1 -1
  441. package/src/plugins/server-bundle.ts +32 -1
  442. package/src/plugins/shims.ts +135 -35
  443. package/src/plugins/static-build.ts +17 -11
  444. package/src/routing/codegen.ts +165 -105
  445. package/src/routing/index.ts +2 -0
  446. package/src/routing/scanner.ts +93 -23
  447. package/src/routing/segment-classify.ts +89 -0
  448. package/src/routing/status-file-lint.ts +3 -2
  449. package/src/routing/types.ts +17 -4
  450. package/src/rsc-runtime/rsc.ts +2 -0
  451. package/src/rsc-runtime/ssr.ts +50 -0
  452. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  453. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  454. package/src/search-params/define.ts +482 -0
  455. package/src/search-params/index.ts +14 -20
  456. package/src/search-params/registry.ts +2 -2
  457. package/src/search-params/wrappers.ts +85 -0
  458. package/src/segment-params/define.ts +279 -0
  459. package/src/segment-params/index.ts +9 -0
  460. package/src/server/access-gate.tsx +70 -29
  461. package/src/server/action-client.ts +46 -11
  462. package/src/server/action-encryption.ts +144 -0
  463. package/src/server/action-handler.ts +21 -4
  464. package/src/server/actions.ts +10 -9
  465. package/src/server/als-registry.ts +34 -6
  466. package/src/server/build-manifest.ts +10 -4
  467. package/src/server/compress.ts +25 -7
  468. package/src/server/debug.ts +1 -1
  469. package/src/server/default-logger.ts +99 -0
  470. package/src/server/deny-page-resolver.ts +154 -0
  471. package/src/server/deny-renderer.ts +24 -38
  472. package/src/server/dev-holding-server.ts +185 -0
  473. package/src/server/dev-warnings.ts +4 -49
  474. package/src/server/early-hints.ts +36 -15
  475. package/src/server/error-boundary-wrapper.ts +74 -22
  476. package/src/server/fallback-error.ts +31 -15
  477. package/src/server/flight-injection-state.ts +113 -0
  478. package/src/server/flight-scripts.ts +62 -0
  479. package/src/server/flush.ts +2 -1
  480. package/src/server/form-data.ts +76 -0
  481. package/src/server/html-injectors.ts +280 -120
  482. package/src/server/index.ts +25 -177
  483. package/src/server/internal.ts +169 -0
  484. package/src/server/logger.ts +44 -36
  485. package/src/server/middleware-runner.ts +31 -4
  486. package/src/server/node-stream-transforms.ts +509 -0
  487. package/src/server/page-deny-boundary.tsx +56 -0
  488. package/src/server/pipeline-interception.ts +17 -16
  489. package/src/server/pipeline-metadata.ts +13 -0
  490. package/src/server/pipeline.ts +227 -62
  491. package/src/server/primitives.ts +111 -28
  492. package/src/server/render-timeout.ts +108 -0
  493. package/src/server/request-context.ts +293 -132
  494. package/src/server/route-element-builder.ts +283 -191
  495. package/src/server/route-handler.ts +24 -4
  496. package/src/server/route-matcher.ts +24 -20
  497. package/src/server/rsc-entry/api-handler.ts +15 -16
  498. package/src/server/rsc-entry/error-renderer.ts +300 -89
  499. package/src/server/rsc-entry/helpers.ts +134 -5
  500. package/src/server/rsc-entry/index.ts +200 -112
  501. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  502. package/src/server/rsc-entry/rsc-stream.ts +65 -13
  503. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  504. package/src/server/rsc-entry/ssr-renderer.ts +168 -38
  505. package/src/server/safe-load.ts +60 -0
  506. package/src/server/sitemap-generator.ts +338 -0
  507. package/src/server/sitemap-handler.ts +126 -0
  508. package/src/server/slot-resolver.ts +244 -229
  509. package/src/server/ssr-entry.ts +211 -32
  510. package/src/server/ssr-render.ts +289 -67
  511. package/src/server/ssr-wrappers.tsx +139 -0
  512. package/src/server/status-code-resolver.ts +1 -1
  513. package/src/server/stream-utils.ts +213 -0
  514. package/src/server/tracing.ts +20 -9
  515. package/src/server/tree-builder.ts +92 -58
  516. package/src/server/types.ts +3 -6
  517. package/src/server/version-skew.ts +104 -0
  518. package/src/shared/merge-search-params.ts +55 -0
  519. package/src/shims/font-google.ts +1 -1
  520. package/src/shims/font-local.ts +34 -0
  521. package/src/shims/headers.ts +5 -1
  522. package/src/shims/navigation-client.ts +1 -1
  523. package/src/shims/navigation.ts +7 -2
  524. package/src/utils/directive-parser.ts +5 -2
  525. package/src/utils/state-machine.ts +111 -0
  526. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  527. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  528. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  529. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  530. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  531. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  532. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  533. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  534. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  535. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  536. package/dist/cache/register-cached-function.d.ts +0 -17
  537. package/dist/cache/register-cached-function.d.ts.map +0 -1
  538. package/dist/client/browser-entry.d.ts +0 -21
  539. package/dist/client/browser-entry.d.ts.map +0 -1
  540. package/dist/client/link-status-provider.d.ts +0 -11
  541. package/dist/client/link-status-provider.d.ts.map +0 -1
  542. package/dist/client/transition-root.d.ts.map +0 -1
  543. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  544. package/dist/cookies/index.js.map +0 -1
  545. package/dist/plugins/cache-transform.d.ts +0 -36
  546. package/dist/plugins/cache-transform.d.ts.map +0 -1
  547. package/dist/plugins/dynamic-transform.d.ts +0 -72
  548. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  549. package/dist/search-params/analyze.d.ts +0 -54
  550. package/dist/search-params/analyze.d.ts.map +0 -1
  551. package/dist/search-params/builtin-codecs.d.ts +0 -105
  552. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  553. package/dist/search-params/codecs.d.ts +0 -53
  554. package/dist/search-params/codecs.d.ts.map +0 -1
  555. package/dist/search-params/create.d.ts +0 -106
  556. package/dist/search-params/create.d.ts.map +0 -1
  557. package/dist/server/prerender.d.ts +0 -77
  558. package/dist/server/prerender.d.ts.map +0 -1
  559. package/dist/server/response-cache.d.ts +0 -54
  560. package/dist/server/response-cache.d.ts.map +0 -1
  561. package/src/cache/register-cached-function.ts +0 -103
  562. package/src/client/browser-entry.ts +0 -678
  563. package/src/client/link-status-provider.tsx +0 -30
  564. package/src/client/transition-root.tsx +0 -166
  565. package/src/plugins/cache-transform.ts +0 -199
  566. package/src/plugins/dynamic-transform.ts +0 -161
  567. package/src/search-params/analyze.ts +0 -192
  568. package/src/search-params/builtin-codecs.ts +0 -228
  569. package/src/search-params/create.ts +0 -321
  570. package/src/server/prerender.ts +0 -139
  571. package/src/server/response-cache.ts +0 -410
@@ -1,678 +0,0 @@
1
- /**
2
- * Browser Entry — Client-side hydration and navigation bootstrap.
3
- *
4
- * This is a real TypeScript file, not codegen. It initializes the
5
- * client navigation runtime: segment router, prefetch cache, and
6
- * history stack.
7
- *
8
- * Hydration works by:
9
- * 1. Decoding the RSC payload embedded in the initial HTML response
10
- * via createFromReadableStream from @vitejs/plugin-rsc/browser
11
- * 2. Hydrating the decoded React tree via hydrateRoot
12
- * 3. Setting up client-side navigation for subsequent page transitions
13
- *
14
- * After hydration, the browser entry:
15
- * - Link click handling is per-component (Link's onClick), not global delegation
16
- * - Listens for popstate events for back/forward navigation
17
- *
18
- * Design docs: 18-build-system.md §"Entry Files", 19-client-navigation.md
19
- */
20
-
21
- // @ts-expect-error — virtual module provided by timber-entries plugin
22
- import config from 'virtual:timber-config';
23
-
24
- import { createElement } from 'react';
25
- import { hydrateRoot, createRoot, type Root } from 'react-dom/client';
26
- import {
27
- createFromReadableStream,
28
- createFromFetch,
29
- setServerCallback,
30
- encodeReply,
31
- } from '#/rsc-runtime/browser.js';
32
- // Shared-state modules MUST be imported from @timber-js/app/client (the public
33
- // barrel) so they resolve to the same module instances as user code. In Vite dev,
34
- // user code imports @timber-js/app/client from dist/ via package.json exports.
35
- // If we used relative imports (./router-ref.js), Vite would load separate src/
36
- // copies with separate module-level state — e.g., globalRouter set here but
37
- // read as null from the dist/ copy used by useRouter().
38
- import { createRouter, setGlobalRouter, getRouter, setCurrentParams } from '@timber-js/app/client';
39
- import type { RouterDeps, RouterInstance } from '@timber-js/app/client';
40
-
41
- // Internal-only modules (no shared mutable state with user code) use relative
42
- // imports — they don't need singleton behavior across module graphs.
43
- import { applyHeadElements } from './head.js';
44
- import { TimberNuqsAdapter } from './nuqs-adapter.js';
45
- import { isPageUnloading } from './unload-guard.js';
46
- import {
47
- NavigationProvider,
48
- getNavigationState,
49
- setNavigationState,
50
- } from './navigation-context.js';
51
- import { setupServerLogReplay, setupClientErrorForwarding } from './browser-dev.js';
52
- // browser-links.ts removed — Link components own their click/hover handlers directly.
53
- // See LOCAL-340.
54
- import { TransitionRoot, transitionRender, navigateTransition } from './transition-root.js';
55
- import {
56
- isStaleClientReference,
57
- triggerStaleReload,
58
- clearStaleReloadFlag,
59
- } from './stale-reload.js';
60
-
61
- // ─── Server Action Dispatch ──────────────────────────────────────
62
-
63
- /**
64
- * Register the callServer callback for server action dispatch.
65
- *
66
- * When React encounters a server reference (from `'use server'` modules),
67
- * it calls `callServer(id, args)` to dispatch the action to the server.
68
- * The RSC plugin delegates to `globalThis.__viteRscCallServer` which is
69
- * set by `setServerCallback`.
70
- *
71
- * The callback:
72
- * 1. Serializes args via `encodeReply` (RSC wire format)
73
- * 2. POSTs to the current URL with `Accept: text/x-component`
74
- * 3. Decodes the RSC response stream
75
- *
76
- * See design/08-forms-and-actions.md §"Client-Side Form Mechanics"
77
- */
78
- setServerCallback(async (id: string, args: unknown[]) => {
79
- const body = await encodeReply(args);
80
-
81
- // Track the X-Timber-Revalidation header from the response.
82
- // We intercept the fetch promise to read headers before createFromFetch
83
- // consumes the body stream.
84
- let hasRevalidation = false;
85
- let hasRedirect = false;
86
- let headElementsJson: string | null = null;
87
-
88
- const response = fetch(window.location.href, {
89
- method: 'POST',
90
- headers: {
91
- 'Accept': 'text/x-component',
92
- 'x-rsc-action': id,
93
- },
94
- body,
95
- }).then((res) => {
96
- hasRevalidation = res.headers.get('X-Timber-Revalidation') === '1';
97
- hasRedirect = res.headers.get('X-Timber-Redirect') != null;
98
- headElementsJson = res.headers.get('X-Timber-Head');
99
- return res;
100
- });
101
-
102
- let decoded: unknown;
103
- try {
104
- decoded = await createFromFetch(response);
105
- } catch (error) {
106
- if (isStaleClientReference(error)) {
107
- triggerStaleReload();
108
- // Return a never-resolving promise to prevent further processing
109
- return new Promise(() => {});
110
- }
111
- throw error;
112
- }
113
-
114
- // Handle redirect — server encoded the redirect location in the RSC stream
115
- // instead of returning HTTP 302. Perform a client-side SPA navigation.
116
- if (hasRedirect) {
117
- const wrapper = decoded as { _redirect: string; _status: number };
118
- try {
119
- const router = getRouter();
120
- void router.navigate(wrapper._redirect);
121
- } catch {
122
- // Router not yet initialized — fall back to full navigation
123
- window.location.href = wrapper._redirect;
124
- }
125
- return undefined;
126
- }
127
-
128
- if (hasRevalidation) {
129
- // Piggybacked response: wrapper object { _action, _tree }
130
- // Apply the revalidated tree directly — no separate router.refresh() needed.
131
- const wrapper = decoded as { _action: unknown; _tree: unknown };
132
- try {
133
- const router = getRouter();
134
- const headElements = headElementsJson ? JSON.parse(headElementsJson) : null;
135
- router.applyRevalidation(wrapper._tree, headElements);
136
- } catch {
137
- // Router not yet initialized — fall through
138
- }
139
- return wrapper._action;
140
- }
141
-
142
- // No piggybacked revalidation — refresh to pick up any mutations.
143
- // This covers actions that don't call revalidatePath().
144
- try {
145
- const router = getRouter();
146
- void router.refresh();
147
- } catch {
148
- // Router not yet initialized (rare edge case during bootstrap)
149
- }
150
-
151
- return decoded;
152
- });
153
-
154
- // ─── Bootstrap ───────────────────────────────────────────────────
155
-
156
- /**
157
- * Bootstrap the client-side runtime.
158
- *
159
- * Hydrates the server-rendered HTML with React, then initializes
160
- * client-side navigation for SPA transitions.
161
- */
162
- /** Read scroll position from window or scroll containers. */
163
- function getScrollY(): number {
164
- if (window.scrollY || document.documentElement.scrollTop || document.body.scrollTop) {
165
- return window.scrollY || document.documentElement.scrollTop || document.body.scrollTop;
166
- }
167
- for (const el of document.querySelectorAll('[data-timber-scroll-restoration]')) {
168
- if ((el as HTMLElement).scrollTop > 0) return (el as HTMLElement).scrollTop;
169
- }
170
- // Auto-detect: if window isn't scrolled, check for overflow containers.
171
- // Common pattern: layouts use a scrollable div (overflow-y: auto/scroll)
172
- // inside a fixed-height parent (h-screen). In this case window.scrollY is
173
- // always 0 and the real scroll position lives on the overflow container.
174
- const container = findOverflowContainer();
175
- if (container && container.scrollTop > 0) return container.scrollTop;
176
- return 0;
177
- }
178
-
179
- /**
180
- * Find the primary overflow scroll container in the document.
181
- *
182
- * Walks direct children of body and their immediate children looking for
183
- * an element with overflow-y: auto|scroll that is actually scrollable
184
- * (scrollHeight > clientHeight). Returns the first match, or null.
185
- *
186
- * This heuristic covers the common layout patterns:
187
- * <body> → <root-layout> → <div class="overflow-y-auto">
188
- * <body> → <root-layout> → <main> → <nested-layout overflow-y-auto>
189
- *
190
- * We limit depth to 3 to avoid expensive full-tree traversals while still
191
- * reaching nested layout scroll containers (e.g., parallel route layouts
192
- * inside a root layout's <main> element).
193
- *
194
- * DIVERGENCE FROM NEXT.JS: Next.js's ScrollAndFocusHandler scrolls only
195
- * document.documentElement.scrollTop — it does NOT handle overflow containers.
196
- * Layouts using h-screen + overflow-y-auto have the same scroll bug in Next.js.
197
- * This heuristic is a deliberate improvement. The tradeoff is fragility: depth-3
198
- * traversal may miss deeply nested containers or match the wrong element.
199
- * See design/19-client-navigation.md §"Overflow Scroll Containers".
200
- */
201
- function findOverflowContainer(): HTMLElement | null {
202
- const candidates: HTMLElement[] = [];
203
- // Check body's descendants up to depth 3. Depth 3 covers the common case:
204
- // <body> → <root-layout-div> → <main> → <overflow-container>
205
- // React context providers (SegmentProvider, NavigationProvider) don't add
206
- // DOM elements, so depth 3 from body reaches nested layout scroll containers.
207
- for (const child of document.body.children) {
208
- candidates.push(child as HTMLElement);
209
- for (const grandchild of child.children) {
210
- candidates.push(grandchild as HTMLElement);
211
- for (const greatGrandchild of grandchild.children) {
212
- candidates.push(greatGrandchild as HTMLElement);
213
- }
214
- }
215
- }
216
- for (const el of candidates) {
217
- if (!(el instanceof HTMLElement)) continue;
218
- const style = getComputedStyle(el);
219
- const overflowY = style.overflowY;
220
- if ((overflowY === 'auto' || overflowY === 'scroll') && el.scrollHeight > el.clientHeight) {
221
- return el;
222
- }
223
- }
224
- return null;
225
- }
226
-
227
- function bootstrap(runtimeConfig: typeof config): void {
228
- const _config = runtimeConfig;
229
-
230
- // Take manual control of scroll restoration. React's render() on the
231
- // document root resets scroll during DOM reconciliation, so the browser's
232
- // native scroll restoration (scrollRestoration = 'auto') doesn't work —
233
- // the browser restores scroll, then React's commit resets it to 0.
234
- // We save/restore scroll positions explicitly in the history stack.
235
- window.history.scrollRestoration = 'manual';
236
-
237
- // Hydrate the React tree from the RSC payload.
238
- //
239
- // The RSC payload is embedded in the HTML as progressive inline script
240
- // tags that call self.__timber_f.push([type, data]) as RSC chunks arrive.
241
- // Typed tuples: [0] = bootstrap signal, [1, string] = Flight data chunk.
242
- //
243
- // We set up a ReadableStream fed by those push() calls so
244
- // createFromReadableStream can decode the Flight protocol progressively.
245
- //
246
- // For the initial page load, the RSC payload is inlined in the HTML.
247
- // For subsequent navigations, it's fetched from the server.
248
- type FlightSegment = [isBootstrap: 0] | [isData: 1, data: string];
249
-
250
- const timberChunks = (self as unknown as Record<string, FlightSegment[]>).__timber_f;
251
-
252
- let _reactRoot: Root | null = null;
253
- let initialElement: unknown = null;
254
- // Declared here so it's accessible after the if/else hydration block.
255
- // Assigned inside initRouter() which is called in both branches.
256
- let router!: RouterInstance;
257
-
258
- if (timberChunks) {
259
- const encoder = new TextEncoder();
260
-
261
- // Buffer to hold string data until the stream writer is ready.
262
- // Scripts that execute before hydration starts push data here.
263
- let dataBuffer: string[] | undefined = [];
264
- let streamWriter: ReadableStreamDefaultController<Uint8Array> | null = null;
265
- let streamFlushed = false;
266
-
267
- /** Process a typed tuple from __timber_f. */
268
- function handleSegment(seg: FlightSegment): void {
269
- if (seg[0] === 0) {
270
- // Bootstrap signal — initialize buffer (already done above)
271
- if (!dataBuffer) dataBuffer = [];
272
- } else if (seg[0] === 1) {
273
- // Flight data chunk
274
- if (streamWriter) {
275
- streamWriter.enqueue(encoder.encode(seg[1]));
276
- } else if (dataBuffer) {
277
- dataBuffer.push(seg[1]);
278
- }
279
- }
280
- }
281
-
282
- // Process any chunks that arrived before this script executed.
283
- for (const seg of timberChunks) {
284
- handleSegment(seg);
285
- }
286
- // Clear the array to release memory.
287
- timberChunks.length = 0;
288
-
289
- // Patch push() so subsequent script tags feed data in real time.
290
- (timberChunks as unknown as { push: (seg: FlightSegment) => void }).push = handleSegment;
291
-
292
- const rscPayload = new ReadableStream<Uint8Array>({
293
- start(controller) {
294
- streamWriter = controller;
295
- // Flush buffered data into the stream.
296
- if (dataBuffer) {
297
- for (const data of dataBuffer) {
298
- controller.enqueue(encoder.encode(data));
299
- }
300
- dataBuffer = undefined;
301
- }
302
- // If DOM already loaded (non-streaming or fast page), close now.
303
- if (streamFlushed) {
304
- controller.close();
305
- }
306
- },
307
- });
308
-
309
- // Close the stream when the document finishes loading.
310
- // DOMContentLoaded fires after the HTML parser has processed all
311
- // inline scripts (including streamed Suspense replacements and
312
- // RSC data), so all push() calls have completed by this point.
313
- //
314
- // If the page is unloading (user refreshed or navigated away),
315
- // do NOT close the stream. When the connection drops mid-stream,
316
- // DOMContentLoaded fires because the parser finishes. Closing an
317
- // incomplete RSC stream causes React's Flight client to throw
318
- // "Connection closed." — a jarring error on a page being replaced.
319
- // Leaving the stream open is harmless: the page is being torn down.
320
- function onDOMContentLoaded(): void {
321
- if (isPageUnloading()) return;
322
- if (streamWriter && !streamFlushed) {
323
- streamWriter.close();
324
- streamFlushed = true;
325
- dataBuffer = undefined;
326
- }
327
- streamFlushed = true;
328
- }
329
-
330
- if (document.readyState === 'loading') {
331
- document.addEventListener('DOMContentLoaded', onDOMContentLoaded, false);
332
- } else {
333
- // DOM already parsed — close after a microtask to ensure
334
- // any pending push() calls from inline scripts have executed.
335
- setTimeout(onDOMContentLoaded);
336
- }
337
-
338
- const element = createFromReadableStream(rscPayload);
339
- initialElement = element;
340
-
341
- // ── Initialize the navigation router BEFORE hydration ──────────────
342
- // hydrateRoot() synchronously executes component render functions.
343
- // Components that call useRouter() during render need the global
344
- // router to be available, otherwise they get a stale no-op reference.
345
- // The router must be initialized before hydration so useRouter() works.
346
- // renderRoot uses transitionRender (no direct reactRoot dependency).
347
- initRouter();
348
-
349
- // ── Initialize navigation state BEFORE hydration ───────────────────
350
- // Read server-embedded params and set navigation state so that
351
- // useParams() and usePathname() return correct values during hydration.
352
- // This must happen before hydrateRoot so the NavigationProvider
353
- // wrapping the element has the right values on the initial render.
354
- const earlyParams = (self as unknown as Record<string, unknown>).__timber_params;
355
- if (earlyParams && typeof earlyParams === 'object') {
356
- setCurrentParams(earlyParams as Record<string, string | string[]>);
357
- setNavigationState({
358
- params: earlyParams as Record<string, string | string[]>,
359
- pathname: window.location.pathname,
360
- });
361
- delete (self as unknown as Record<string, unknown>).__timber_params;
362
- } else {
363
- setNavigationState({
364
- params: {},
365
- pathname: window.location.pathname,
366
- });
367
- }
368
-
369
- // Hydrate on document — the root layout renders the full <html> tree,
370
- // so React owns the entire document from the root.
371
- // Wrap with NavigationProvider (for atomic useParams/usePathname),
372
- // TimberNuqsAdapter (for nuqs context), and TransitionRoot (for
373
- // transition-based rendering during client navigation).
374
- //
375
- // TransitionRoot holds the element in React state and updates via
376
- // startTransition, so React keeps old UI visible while new Suspense
377
- // boundaries resolve during navigation. See design/05-streaming.md.
378
- const navState = getNavigationState();
379
- const withNav = createElement(
380
- NavigationProvider,
381
- { value: navState },
382
- element as React.ReactNode
383
- );
384
- const wrapped = createElement(TimberNuqsAdapter, null, withNav);
385
- const rootElement = createElement(TransitionRoot, {
386
- initial: wrapped,
387
- topLoaderConfig: _config.topLoader,
388
- });
389
- _reactRoot = hydrateRoot(document, rootElement, {
390
- // Suppress recoverable hydration errors from deny/error signals
391
- // inside Suspense boundaries. The server already handled these
392
- // (wrapStreamWithErrorHandling closes the stream cleanly after
393
- // the shell is flushed). React replays the error during hydration
394
- // but the server HTML is already correct — no recovery needed.
395
- onRecoverableError(error: unknown) {
396
- // Suppress errors during page unload (refresh/navigate away).
397
- // The aborted stream causes incomplete HTML which React flags
398
- // as a recoverable error — but the page is being replaced.
399
- if (isPageUnloading()) return;
400
- // Only log in dev — in production these are expected for
401
- // deny() inside Suspense and streaming error boundaries.
402
- if (process.env.NODE_ENV === 'development') {
403
- console.debug('[timber] Hydration recoverable error:', error);
404
- }
405
- },
406
- });
407
- } else {
408
- // No RSC payload available (plugin hasn't inlined it yet) — create a
409
- // non-hydrated root so client navigation can still render RSC payloads.
410
- // The initial SSR HTML remains as-is; the first client navigation will
411
- // replace it with a React-managed tree.
412
- initRouter();
413
- _reactRoot = createRoot(document);
414
- }
415
-
416
- // ── Router initialization (hoisted above hydrateRoot) ────────────────
417
- // Extracted into a function so both the hydration and createRoot paths
418
- // can call it. Must run before hydrateRoot so useRouter() works during
419
- // the initial render. renderRoot uses transitionRender which is set
420
- // by the TransitionRoot component during hydration.
421
- function initRouter(): void {
422
- const deps: RouterDeps = {
423
- fetch: (url, init) => window.fetch(url, init),
424
- pushState: (data, unused, url) => window.history.pushState(data, unused, url),
425
- replaceState: (data, unused, url) => window.history.replaceState(data, unused, url),
426
- scrollTo: (x, y) => {
427
- window.scrollTo(x, y);
428
- document.documentElement.scrollTop = y;
429
- document.body.scrollTop = y;
430
- // Also scroll any element explicitly marked as a scroll container.
431
- for (const el of document.querySelectorAll('[data-timber-scroll-restoration]')) {
432
- (el as HTMLElement).scrollTop = y;
433
- }
434
- // Auto-detect overflow containers for layouts that scroll inside
435
- // a fixed-height wrapper (e.g., h-screen + overflow-y-auto).
436
- // In these layouts, window.scrollY is always 0 and the real scroll
437
- // lives on the overflow container. Without this, forward navigation
438
- // between pages that share a layout with parallel route slots won't
439
- // scroll to top — the router's window.scrollTo(0,0) is a no-op.
440
- const container = findOverflowContainer();
441
- if (container) {
442
- container.scrollTop = y;
443
- }
444
- },
445
- getCurrentUrl: () => window.location.pathname + window.location.search,
446
- getScrollY,
447
-
448
- // Decode RSC Flight stream using createFromFetch.
449
- // createFromFetch takes a Promise<Response> and progressively
450
- // parses the RSC stream as chunks arrive.
451
- //
452
- // Wrapped with stale client reference detection: if the server
453
- // has been redeployed with new bundles, the RSC payload may
454
- // reference module IDs that don't exist in the old client bundle.
455
- // We catch "Could not find the module" errors and trigger a full
456
- // page reload so the browser fetches the new bundle.
457
- decodeRsc: async (fetchPromise: Promise<Response>) => {
458
- try {
459
- return await createFromFetch(fetchPromise);
460
- } catch (error) {
461
- if (isStaleClientReference(error)) {
462
- triggerStaleReload();
463
- // Return a never-resolving promise to prevent further processing
464
- // while the page is reloading.
465
- return new Promise(() => {});
466
- }
467
- throw error;
468
- }
469
- },
470
-
471
- // Render decoded RSC tree via TransitionRoot's state-based mechanism.
472
- // Used for non-navigation renders (popstate cached replay, applyRevalidation).
473
- // Wraps with NavigationProvider + TimberNuqsAdapter.
474
- //
475
- // For navigation renders (navigate, refresh, popstate-with-fetch),
476
- // navigateTransition is used instead — it wraps the entire navigation
477
- // in a React transition with useOptimistic for the pending URL.
478
- renderRoot: (element: unknown) => {
479
- const navState = getNavigationState();
480
- const withNav = createElement(
481
- NavigationProvider,
482
- { value: navState },
483
- element as React.ReactNode
484
- );
485
- const wrapped = createElement(TimberNuqsAdapter, null, withNav);
486
- transitionRender(wrapped);
487
- },
488
-
489
- // Run a navigation inside a React transition with optimistic pending URL.
490
- // The entire fetch + state update runs inside startTransition. useOptimistic
491
- // shows the pending URL immediately and reverts to null when the transition
492
- // commits (atomic with the new tree + params).
493
- //
494
- // The perform callback receives a wrapPayload function that wraps the
495
- // decoded RSC payload with NavigationProvider + NuqsAdapter — this must
496
- // happen inside the transition so the NavigationProvider reads the
497
- // UPDATED navigation state (set by the router inside perform).
498
- navigateTransition: (pendingUrl: string, perform) => {
499
- return navigateTransition(pendingUrl, async () => {
500
- const payload = await perform((rawPayload: unknown) => {
501
- const navState = getNavigationState();
502
- const withNav = createElement(
503
- NavigationProvider,
504
- { value: navState },
505
- rawPayload as React.ReactNode
506
- );
507
- return createElement(TimberNuqsAdapter, null, withNav);
508
- });
509
- return payload as React.ReactNode;
510
- });
511
- },
512
-
513
- // Schedule a callback after the next paint so scroll operations
514
- // happen after React commits the new content to the DOM.
515
- // Double-rAF ensures the browser has painted the new frame.
516
- afterPaint: (callback: () => void) => {
517
- requestAnimationFrame(() => {
518
- requestAnimationFrame(callback);
519
- });
520
- },
521
-
522
- // Apply resolved head elements (title, meta tags) to the DOM after
523
- // SPA navigation. See design/16-metadata.md.
524
- applyHead: applyHeadElements,
525
- };
526
-
527
- router = createRouter(deps);
528
- setGlobalRouter(router);
529
- }
530
-
531
- // Store the initial page in the history stack so back-button works
532
- // after the first navigation. We store the decoded RSC element so
533
- // back navigation can replay it instantly without a server fetch.
534
- router.historyStack.push(window.location.pathname + window.location.search, {
535
- payload: initialElement,
536
- headElements: null, // SSR already set the correct head
537
- });
538
-
539
- // Initialize history.state with scrollY for the initial entry.
540
- // This ensures back navigation to the initial page restores scroll correctly.
541
- window.history.replaceState({ timber: true, scrollY: 0 }, '');
542
-
543
- // Populate the segment cache from server-embedded segment metadata.
544
- // This enables state tree diffing from the very first client navigation.
545
- // See design/19-client-navigation.md §"X-Timber-State-Tree Header"
546
- const timberSegments = (self as unknown as Record<string, unknown>).__timber_segments;
547
- if (Array.isArray(timberSegments)) {
548
- router.initSegmentCache(timberSegments);
549
- delete (self as unknown as Record<string, unknown>).__timber_segments;
550
- }
551
-
552
- // NOTE: We do NOT cache segment elements from the initial RSC payload here.
553
- // The decoded element from createFromReadableStream is a thenable/lazy
554
- // element that React resolves during render — the segment walker can't
555
- // traverse it. The element cache is populated lazily after the first SPA
556
- // navigation (via mergeAndCachePayload in renderViaTransition), when
557
- // the decoded payload is a fully resolved React element tree.
558
-
559
- // Note: __timber_params is read before hydrateRoot (see above) so that
560
- // NavigationProvider has correct values during hydration. If the hydration
561
- // path was skipped (no RSC payload), populate the fallback here.
562
- const lateTimberParams = (self as unknown as Record<string, unknown>).__timber_params;
563
- if (lateTimberParams && typeof lateTimberParams === 'object') {
564
- setCurrentParams(lateTimberParams as Record<string, string | string[]>);
565
- setNavigationState({
566
- params: lateTimberParams as Record<string, string | string[]>,
567
- pathname: window.location.pathname,
568
- });
569
- delete (self as unknown as Record<string, unknown>).__timber_params;
570
- }
571
-
572
- // Register popstate handler for back/forward navigation.
573
- // Use pathname+search (not full href) to match the URL format used by
574
- // navigate() — Link hrefs are relative paths like "/scroll-test/page-a".
575
- // Read scrollY from history.state — the browser maintains per-entry state
576
- // so duplicate URLs in history each have their own scroll position.
577
- window.addEventListener('popstate', () => {
578
- const state = window.history.state;
579
- const scrollY = state && typeof state.scrollY === 'number' ? state.scrollY : 0;
580
- void router.handlePopState(window.location.pathname + window.location.search, scrollY);
581
- });
582
-
583
- // Keep history.state.scrollY up to date as the user scrolls.
584
- // This ensures that when the user presses back/forward, the departing
585
- // page's scroll position is already saved in its history entry.
586
- // Debounced to avoid excessive replaceState calls during smooth scrolling.
587
- let scrollTimer: ReturnType<typeof setTimeout>;
588
- function saveScrollPosition(): void {
589
- clearTimeout(scrollTimer);
590
- scrollTimer = setTimeout(() => {
591
- const state = window.history.state;
592
- if (state && typeof state === 'object') {
593
- // Use getScrollY to capture scroll from overflow containers too.
594
- window.history.replaceState({ ...state, scrollY: getScrollY() }, '');
595
- }
596
- }, 100);
597
- }
598
- window.addEventListener('scroll', saveScrollPosition, { passive: true });
599
-
600
- // Link click and hover prefetch are handled per-component by Link's
601
- // onClick and onMouseEnter handlers. No global delegation needed.
602
- // See LOCAL-340.
603
-
604
- // Dev-only: Listen for RSC module invalidation events from @vitejs/plugin-rsc.
605
- // When a server component is edited, the RSC plugin sends an "rsc:update"
606
- // event. We trigger a router.refresh() to re-fetch the RSC payload with
607
- // the updated server code. This avoids a full page reload while still
608
- // picking up server-side changes.
609
- // See design/21-dev-server.md §"HMR Wiring"
610
- // Vite injects import.meta.hot in dev mode. Cast to access it without
611
- // requiring vite/client types in the package tsconfig.
612
- const hot = (
613
- import.meta as unknown as {
614
- hot?: {
615
- on(event: string, cb: (...args: unknown[]) => void): void;
616
- send(event: string, data: unknown): void;
617
- };
618
- }
619
- ).hot;
620
- if (hot) {
621
- hot.on('rsc:update', () => {
622
- void router.refresh();
623
- });
624
-
625
- // Listen for dev warnings forwarded from the server via WebSocket.
626
- // See dev-warnings.ts — emitOnce() sends these via server.hot.send().
627
- hot.on('timber:dev-warning', (data: unknown) => {
628
- const warning = data as { level: string; message: string };
629
- if (warning.level === 'error') {
630
- console.error(warning.message);
631
- } else {
632
- console.warn(warning.message);
633
- }
634
- });
635
-
636
- // Listen for server console logs forwarded via WebSocket.
637
- // Replays them in the browser console with a [SERVER] prefix
638
- // so developers can see server output without switching to the terminal.
639
- // See plugins/dev-logs.ts.
640
- setupServerLogReplay(hot);
641
-
642
- // Forward uncaught client errors to the server for the dev overlay.
643
- // The server source-maps the stack and sends it back via Vite's
644
- // error overlay protocol. See dev-server.ts §client error listener.
645
- setupClientErrorForwarding(hot);
646
- }
647
- }
648
-
649
- bootstrap(config);
650
-
651
- // Clear the stale reload flag on successful bootstrap. If the page
652
- // loaded and bootstrapped without hitting a stale reference error,
653
- // the loop guard should reset so the next stale error gets a fresh
654
- // reload attempt.
655
- clearStaleReloadFlag();
656
-
657
- // Global error handler for stale client reference errors during hydration.
658
- // The initial RSC payload is decoded lazily by React via createFromReadableStream.
659
- // If the payload references a module ID from a newer deployment, the error
660
- // surfaces as an unhandled rejection during React's render/hydration cycle.
661
- // This handler catches those errors and triggers a full page reload.
662
- window.addEventListener('unhandledrejection', (event) => {
663
- if (isStaleClientReference(event.reason)) {
664
- event.preventDefault();
665
- triggerStaleReload();
666
- }
667
- });
668
-
669
- // Signal that the client runtime has been initialized.
670
- // Used by E2E tests to wait for hydration before interacting.
671
- // We append a <meta name="timber-ready"> tag rather than setting a
672
- // data attribute on <html>. Since React owns the entire document
673
- // via hydrateRoot(document, ...), mutating <html> attributes causes
674
- // hydration mismatch warnings. Dynamically-added <meta> tags don't
675
- // conflict because React doesn't reconcile them.
676
- const readyMeta = document.createElement('meta');
677
- readyMeta.name = 'timber-ready';
678
- document.head.appendChild(readyMeta);