@timber-js/app 0.2.0-alpha.9 → 0.2.0-alpha.91
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.
- package/dist/_chunks/actions-DLnUaR65.js +421 -0
- package/dist/_chunks/actions-DLnUaR65.js.map +1 -0
- package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
- package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
- package/dist/_chunks/chunk-BYIpzuS7.js +39 -0
- package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
- package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
- package/dist/_chunks/define-C77ScO0m.js +106 -0
- package/dist/_chunks/define-C77ScO0m.js.map +1 -0
- package/dist/_chunks/define-Itxvcd7F.js +199 -0
- package/dist/_chunks/define-Itxvcd7F.js.map +1 -0
- package/dist/_chunks/define-cookie-BowvzoP0.js +94 -0
- package/dist/_chunks/define-cookie-BowvzoP0.js.map +1 -0
- package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
- package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
- package/dist/_chunks/format-CYBGxKtc.js +14 -0
- package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
- package/dist/_chunks/{interception-BOoWmLUA.js → interception-ErnB33JX.js} +301 -133
- package/dist/_chunks/interception-ErnB33JX.js.map +1 -0
- package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
- package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
- package/dist/_chunks/request-context-CK5tZqIP.js +478 -0
- package/dist/_chunks/request-context-CK5tZqIP.js.map +1 -0
- package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
- package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
- package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
- package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
- package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
- package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
- package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
- package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
- package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
- package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
- package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
- package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
- package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-BiV5GJgm.js} +7 -4
- package/dist/_chunks/use-query-states-BiV5GJgm.js.map +1 -0
- package/dist/adapters/cloudflare-dev.d.ts +109 -0
- package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
- package/dist/adapters/cloudflare-dev.js +73 -0
- package/dist/adapters/cloudflare-dev.js.map +1 -0
- package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
- package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
- package/dist/adapters/cloudflare-kv-cache.js +95 -0
- package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
- package/dist/adapters/cloudflare.d.ts +148 -12
- package/dist/adapters/cloudflare.d.ts.map +1 -1
- package/dist/adapters/cloudflare.js +135 -11
- package/dist/adapters/cloudflare.js.map +1 -1
- package/dist/adapters/compress-module.d.ts.map +1 -1
- package/dist/adapters/nitro.d.ts +17 -1
- package/dist/adapters/nitro.d.ts.map +1 -1
- package/dist/adapters/nitro.js +56 -13
- package/dist/adapters/nitro.js.map +1 -1
- package/dist/cache/cache-api.d.ts +24 -0
- package/dist/cache/cache-api.d.ts.map +1 -0
- package/dist/cache/handler-store.d.ts +31 -0
- package/dist/cache/handler-store.d.ts.map +1 -0
- package/dist/cache/index.d.ts +23 -7
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +142 -80
- package/dist/cache/index.js.map +1 -1
- package/dist/cache/singleflight.d.ts +18 -1
- package/dist/cache/singleflight.d.ts.map +1 -1
- package/dist/cache/sizeof.d.ts +22 -0
- package/dist/cache/sizeof.d.ts.map +1 -0
- package/dist/cache/timber-cache.d.ts +1 -1
- package/dist/cache/timber-cache.d.ts.map +1 -1
- package/dist/cli.d.ts +6 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +8 -3
- package/dist/cli.js.map +1 -1
- package/dist/client/browser-dev.d.ts +27 -1
- package/dist/client/browser-dev.d.ts.map +1 -1
- package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
- package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
- package/dist/client/browser-entry/hmr.d.ts +21 -0
- package/dist/client/browser-entry/hmr.d.ts.map +1 -0
- package/dist/client/browser-entry/hydrate.d.ts +46 -0
- package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
- package/dist/client/browser-entry/index.d.ts +30 -0
- package/dist/client/browser-entry/index.d.ts.map +1 -0
- package/dist/client/browser-entry/post-hydration.d.ts +26 -0
- package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
- package/dist/client/browser-entry/router-init.d.ts +23 -0
- package/dist/client/browser-entry/router-init.d.ts.map +1 -0
- package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
- package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
- package/dist/client/browser-entry/scroll.d.ts +19 -0
- package/dist/client/browser-entry/scroll.d.ts.map +1 -0
- package/dist/client/error-boundary.d.ts +12 -5
- package/dist/client/error-boundary.d.ts.map +1 -1
- package/dist/client/error-boundary.js +10 -4
- package/dist/client/error-boundary.js.map +1 -1
- package/dist/client/error-reconstituter.d.ts +54 -0
- package/dist/client/error-reconstituter.d.ts.map +1 -0
- package/dist/client/form.d.ts +6 -3
- package/dist/client/form.d.ts.map +1 -1
- package/dist/client/history.d.ts +19 -4
- package/dist/client/history.d.ts.map +1 -1
- package/dist/client/index.d.ts +9 -21
- package/dist/client/index.d.ts.map +1 -1
- package/dist/client/index.js +229 -1018
- package/dist/client/index.js.map +1 -1
- package/dist/client/internal.d.ts +18 -0
- package/dist/client/internal.d.ts.map +1 -0
- package/dist/client/internal.js +890 -0
- package/dist/client/internal.js.map +1 -0
- package/dist/client/link-pending-store.d.ts +63 -0
- package/dist/client/link-pending-store.d.ts.map +1 -0
- package/dist/client/link.d.ts +62 -55
- package/dist/client/link.d.ts.map +1 -1
- package/dist/client/nav-link-store.d.ts +36 -0
- package/dist/client/nav-link-store.d.ts.map +1 -0
- package/dist/client/navigation-api-types.d.ts +90 -0
- package/dist/client/navigation-api-types.d.ts.map +1 -0
- package/dist/client/navigation-api.d.ts +115 -0
- package/dist/client/navigation-api.d.ts.map +1 -0
- package/dist/client/navigation-context.d.ts +13 -2
- package/dist/client/navigation-context.d.ts.map +1 -1
- package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
- package/dist/client/navigation-root.d.ts.map +1 -0
- package/dist/client/nuqs-adapter.d.ts.map +1 -1
- package/dist/client/router-ref.d.ts +1 -1
- package/dist/client/router.d.ts +70 -4
- package/dist/client/router.d.ts.map +1 -1
- package/dist/client/rsc-fetch.d.ts +38 -3
- package/dist/client/rsc-fetch.d.ts.map +1 -1
- package/dist/client/segment-cache.d.ts +1 -1
- package/dist/client/segment-cache.d.ts.map +1 -1
- package/dist/client/segment-outlet.d.ts +63 -0
- package/dist/client/segment-outlet.d.ts.map +1 -0
- package/dist/client/ssr-data.d.ts +13 -4
- package/dist/client/ssr-data.d.ts.map +1 -1
- package/dist/client/stale-reload.d.ts +15 -0
- package/dist/client/stale-reload.d.ts.map +1 -1
- package/dist/client/top-loader.d.ts +5 -5
- package/dist/client/top-loader.d.ts.map +1 -1
- package/dist/client/use-link-status.d.ts +5 -5
- package/dist/client/use-link-status.d.ts.map +1 -1
- package/dist/client/use-params.d.ts +6 -4
- package/dist/client/use-params.d.ts.map +1 -1
- package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
- package/dist/client/use-pending-navigation.d.ts.map +1 -0
- package/dist/client/use-query-states.d.ts +1 -1
- package/dist/client/use-query-states.d.ts.map +1 -1
- package/dist/client/use-router.d.ts +1 -1
- package/dist/codec.d.ts +33 -0
- package/dist/codec.d.ts.map +1 -0
- package/dist/codec.js +2 -0
- package/dist/config-types.d.ts +266 -0
- package/dist/config-types.d.ts.map +1 -0
- package/dist/config-validation.d.ts +51 -0
- package/dist/config-validation.d.ts.map +1 -0
- package/dist/content/index.d.ts +1 -10
- package/dist/content/index.d.ts.map +1 -1
- package/dist/content/index.js +0 -2
- package/dist/cookies/define-cookie.d.ts +35 -14
- package/dist/cookies/define-cookie.d.ts.map +1 -1
- package/dist/cookies/index.js +1 -83
- package/dist/fonts/bundle.d.ts +48 -0
- package/dist/fonts/bundle.d.ts.map +1 -0
- package/dist/fonts/css.d.ts +1 -0
- package/dist/fonts/css.d.ts.map +1 -1
- package/dist/fonts/dev-middleware.d.ts +22 -0
- package/dist/fonts/dev-middleware.d.ts.map +1 -0
- package/dist/fonts/pipeline.d.ts +138 -0
- package/dist/fonts/pipeline.d.ts.map +1 -0
- package/dist/fonts/transform.d.ts +72 -0
- package/dist/fonts/transform.d.ts.map +1 -0
- package/dist/fonts/types.d.ts +45 -1
- package/dist/fonts/types.d.ts.map +1 -1
- package/dist/fonts/virtual-modules.d.ts +59 -0
- package/dist/fonts/virtual-modules.d.ts.map +1 -0
- package/dist/index.d.ts +45 -190
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4294 -2453
- package/dist/index.js.map +1 -1
- package/dist/plugin-context.d.ts +107 -0
- package/dist/plugin-context.d.ts.map +1 -0
- package/dist/plugins/adapter-build.d.ts +1 -1
- package/dist/plugins/adapter-build.d.ts.map +1 -1
- package/dist/plugins/build-manifest.d.ts +2 -2
- package/dist/plugins/build-manifest.d.ts.map +1 -1
- package/dist/plugins/build-report.d.ts +3 -3
- package/dist/plugins/build-report.d.ts.map +1 -1
- package/dist/plugins/client-chunks.d.ts +32 -0
- package/dist/plugins/client-chunks.d.ts.map +1 -0
- package/dist/plugins/content.d.ts +1 -1
- package/dist/plugins/content.d.ts.map +1 -1
- package/dist/plugins/dev-404-page.d.ts +56 -0
- package/dist/plugins/dev-404-page.d.ts.map +1 -0
- package/dist/plugins/dev-browser-logs.d.ts +84 -0
- package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
- package/dist/plugins/dev-error-overlay.d.ts +49 -9
- package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
- package/dist/plugins/dev-error-page.d.ts +58 -0
- package/dist/plugins/dev-error-page.d.ts.map +1 -0
- package/dist/plugins/dev-logs.d.ts +1 -1
- package/dist/plugins/dev-logs.d.ts.map +1 -1
- package/dist/plugins/dev-server.d.ts +1 -1
- package/dist/plugins/dev-server.d.ts.map +1 -1
- package/dist/plugins/dev-terminal-error.d.ts +28 -0
- package/dist/plugins/dev-terminal-error.d.ts.map +1 -0
- package/dist/plugins/entries.d.ts +1 -1
- package/dist/plugins/entries.d.ts.map +1 -1
- package/dist/plugins/fonts.d.ts +17 -73
- package/dist/plugins/fonts.d.ts.map +1 -1
- package/dist/plugins/mdx.d.ts +1 -1
- package/dist/plugins/mdx.d.ts.map +1 -1
- package/dist/plugins/routing.d.ts +1 -1
- package/dist/plugins/routing.d.ts.map +1 -1
- package/dist/plugins/server-bundle.d.ts.map +1 -1
- package/dist/plugins/shims.d.ts +6 -5
- package/dist/plugins/shims.d.ts.map +1 -1
- package/dist/plugins/static-build.d.ts +4 -4
- package/dist/plugins/static-build.d.ts.map +1 -1
- package/dist/routing/codegen-shared.d.ts +38 -0
- package/dist/routing/codegen-shared.d.ts.map +1 -0
- package/dist/routing/codegen-types.d.ts +36 -0
- package/dist/routing/codegen-types.d.ts.map +1 -0
- package/dist/routing/codegen.d.ts +2 -2
- package/dist/routing/codegen.d.ts.map +1 -1
- package/dist/routing/convention-lint.d.ts +41 -0
- package/dist/routing/convention-lint.d.ts.map +1 -0
- package/dist/routing/index.d.ts +2 -0
- package/dist/routing/index.d.ts.map +1 -1
- package/dist/routing/index.js +3 -2
- package/dist/routing/link-codegen.d.ts +90 -0
- package/dist/routing/link-codegen.d.ts.map +1 -0
- package/dist/routing/scanner.d.ts.map +1 -1
- package/dist/routing/segment-classify.d.ts +46 -0
- package/dist/routing/segment-classify.d.ts.map +1 -0
- package/dist/routing/status-file-lint.d.ts +2 -1
- package/dist/routing/status-file-lint.d.ts.map +1 -1
- package/dist/routing/types.d.ts +16 -4
- package/dist/routing/types.d.ts.map +1 -1
- package/dist/rsc-runtime/rsc.d.ts +1 -1
- package/dist/rsc-runtime/rsc.d.ts.map +1 -1
- package/dist/rsc-runtime/ssr.d.ts +12 -0
- package/dist/rsc-runtime/ssr.d.ts.map +1 -1
- package/dist/schema-bridge.d.ts +76 -0
- package/dist/schema-bridge.d.ts.map +1 -0
- package/dist/search-params/define.d.ts +139 -0
- package/dist/search-params/define.d.ts.map +1 -0
- package/dist/search-params/index.d.ts +4 -7
- package/dist/search-params/index.d.ts.map +1 -1
- package/dist/search-params/index.js +32 -441
- package/dist/search-params/index.js.map +1 -1
- package/dist/search-params/registry.d.ts +2 -2
- package/dist/search-params/registry.d.ts.map +1 -1
- package/dist/search-params/wrappers.d.ts +53 -0
- package/dist/search-params/wrappers.d.ts.map +1 -0
- package/dist/segment-params/define.d.ts +78 -0
- package/dist/segment-params/define.d.ts.map +1 -0
- package/dist/segment-params/index.d.ts +3 -0
- package/dist/segment-params/index.d.ts.map +1 -0
- package/dist/segment-params/index.js +2 -0
- package/dist/server/access-gate.d.ts +4 -0
- package/dist/server/access-gate.d.ts.map +1 -1
- package/dist/server/action-client.d.ts +41 -6
- package/dist/server/action-client.d.ts.map +1 -1
- package/dist/server/action-encryption.d.ts +76 -0
- package/dist/server/action-encryption.d.ts.map +1 -0
- package/dist/server/action-handler.d.ts +7 -0
- package/dist/server/action-handler.d.ts.map +1 -1
- package/dist/server/actions.d.ts +3 -6
- package/dist/server/actions.d.ts.map +1 -1
- package/dist/server/als-registry.d.ts +32 -4
- package/dist/server/als-registry.d.ts.map +1 -1
- package/dist/server/build-manifest.d.ts +2 -2
- package/dist/server/build-manifest.d.ts.map +1 -1
- package/dist/server/debug.d.ts +1 -1
- package/dist/server/default-logger.d.ts +22 -0
- package/dist/server/default-logger.d.ts.map +1 -0
- package/dist/server/deny-page-resolver.d.ts +52 -0
- package/dist/server/deny-page-resolver.d.ts.map +1 -0
- package/dist/server/deny-renderer.d.ts.map +1 -1
- package/dist/server/dev-holding-server.d.ts +52 -0
- package/dist/server/dev-holding-server.d.ts.map +1 -0
- package/dist/server/dev-source-map.d.ts +22 -0
- package/dist/server/dev-source-map.d.ts.map +1 -0
- package/dist/server/dev-warnings.d.ts +1 -21
- package/dist/server/dev-warnings.d.ts.map +1 -1
- package/dist/server/early-hints.d.ts +13 -5
- package/dist/server/early-hints.d.ts.map +1 -1
- package/dist/server/error-boundary-wrapper.d.ts +7 -1
- package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
- package/dist/server/fallback-error.d.ts +12 -7
- package/dist/server/fallback-error.d.ts.map +1 -1
- package/dist/server/flight-injection-state.d.ts +66 -0
- package/dist/server/flight-injection-state.d.ts.map +1 -0
- package/dist/server/flight-scripts.d.ts +42 -0
- package/dist/server/flight-scripts.d.ts.map +1 -0
- package/dist/server/flush.d.ts.map +1 -1
- package/dist/server/form-data.d.ts +29 -0
- package/dist/server/form-data.d.ts.map +1 -1
- package/dist/server/html-injectors.d.ts +51 -11
- package/dist/server/html-injectors.d.ts.map +1 -1
- package/dist/server/index.d.ts +5 -43
- package/dist/server/index.d.ts.map +1 -1
- package/dist/server/index.js +195 -2800
- package/dist/server/index.js.map +1 -1
- package/dist/server/internal.d.ts +46 -0
- package/dist/server/internal.d.ts.map +1 -0
- package/dist/server/internal.js +2900 -0
- package/dist/server/internal.js.map +1 -0
- package/dist/server/logger.d.ts +25 -7
- package/dist/server/logger.d.ts.map +1 -1
- package/dist/server/middleware-runner.d.ts +19 -4
- package/dist/server/middleware-runner.d.ts.map +1 -1
- package/dist/server/node-stream-transforms.d.ts +113 -0
- package/dist/server/node-stream-transforms.d.ts.map +1 -0
- package/dist/server/page-deny-boundary.d.ts +31 -0
- package/dist/server/page-deny-boundary.d.ts.map +1 -0
- package/dist/server/pipeline-interception.d.ts +1 -1
- package/dist/server/pipeline-interception.d.ts.map +1 -1
- package/dist/server/pipeline-metadata.d.ts +6 -0
- package/dist/server/pipeline-metadata.d.ts.map +1 -1
- package/dist/server/pipeline.d.ts +52 -10
- package/dist/server/pipeline.d.ts.map +1 -1
- package/dist/server/primitives.d.ts +69 -18
- package/dist/server/primitives.d.ts.map +1 -1
- package/dist/server/render-timeout.d.ts +51 -0
- package/dist/server/render-timeout.d.ts.map +1 -0
- package/dist/server/request-context.d.ts +112 -43
- package/dist/server/request-context.d.ts.map +1 -1
- package/dist/server/route-element-builder.d.ts +27 -1
- package/dist/server/route-element-builder.d.ts.map +1 -1
- package/dist/server/route-handler.d.ts.map +1 -1
- package/dist/server/route-matcher.d.ts +16 -2
- package/dist/server/route-matcher.d.ts.map +1 -1
- package/dist/server/rsc-entry/api-handler.d.ts +2 -2
- package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
- package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
- package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
- package/dist/server/rsc-entry/helpers.d.ts +48 -5
- package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
- package/dist/server/rsc-entry/index.d.ts +20 -3
- package/dist/server/rsc-entry/index.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
- package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts +14 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
- package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
- package/dist/server/safe-load.d.ts +46 -0
- package/dist/server/safe-load.d.ts.map +1 -0
- package/dist/server/sensitive-fields.d.ts +74 -0
- package/dist/server/sensitive-fields.d.ts.map +1 -0
- package/dist/server/sitemap-generator.d.ts +129 -0
- package/dist/server/sitemap-generator.d.ts.map +1 -0
- package/dist/server/sitemap-handler.d.ts +22 -0
- package/dist/server/sitemap-handler.d.ts.map +1 -0
- package/dist/server/slot-resolver.d.ts +1 -1
- package/dist/server/slot-resolver.d.ts.map +1 -1
- package/dist/server/ssr-entry.d.ts +23 -0
- package/dist/server/ssr-entry.d.ts.map +1 -1
- package/dist/server/ssr-render.d.ts +39 -21
- package/dist/server/ssr-render.d.ts.map +1 -1
- package/dist/server/ssr-wrappers.d.ts +50 -0
- package/dist/server/ssr-wrappers.d.ts.map +1 -0
- package/dist/server/status-code-resolver.d.ts +1 -1
- package/dist/server/status-code-resolver.d.ts.map +1 -1
- package/dist/server/stream-utils.d.ts +36 -0
- package/dist/server/stream-utils.d.ts.map +1 -0
- package/dist/server/tracing.d.ts +4 -4
- package/dist/server/tracing.d.ts.map +1 -1
- package/dist/server/tree-builder.d.ts +22 -19
- package/dist/server/tree-builder.d.ts.map +1 -1
- package/dist/server/types.d.ts +1 -4
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/version-skew.d.ts +61 -0
- package/dist/server/version-skew.d.ts.map +1 -0
- package/dist/shared/merge-search-params.d.ts +22 -0
- package/dist/shared/merge-search-params.d.ts.map +1 -0
- package/dist/shims/font-google.d.ts +1 -1
- package/dist/shims/font-google.d.ts.map +1 -1
- package/dist/shims/font-google.js +42 -0
- package/dist/shims/font-google.js.map +1 -0
- package/dist/shims/font-local.d.ts +26 -0
- package/dist/shims/font-local.d.ts.map +1 -0
- package/dist/shims/font-local.js +20 -0
- package/dist/shims/font-local.js.map +1 -0
- package/dist/shims/headers.d.ts +2 -1
- package/dist/shims/headers.d.ts.map +1 -1
- package/dist/shims/navigation-client.d.ts +1 -1
- package/dist/shims/navigation-client.d.ts.map +1 -1
- package/dist/shims/navigation.d.ts +3 -2
- package/dist/shims/navigation.d.ts.map +1 -1
- package/dist/utils/directive-parser.d.ts +5 -2
- package/dist/utils/directive-parser.d.ts.map +1 -1
- package/dist/utils/state-machine.d.ts +80 -0
- package/dist/utils/state-machine.d.ts.map +1 -0
- package/package.json +51 -16
- package/src/adapters/cloudflare-dev.ts +177 -0
- package/src/adapters/cloudflare-kv-cache.ts +142 -0
- package/src/adapters/cloudflare.ts +342 -28
- package/src/adapters/compress-module.ts +24 -4
- package/src/adapters/nitro.ts +52 -8
- package/src/adapters/wrangler.d.ts +7 -0
- package/src/cache/cache-api.ts +38 -0
- package/src/cache/handler-store.ts +68 -0
- package/src/cache/index.ts +81 -18
- package/src/cache/singleflight.ts +62 -4
- package/src/cache/sizeof.ts +31 -0
- package/src/cache/timber-cache.ts +24 -20
- package/src/cli.ts +16 -6
- package/src/client/browser-dev.ts +128 -1
- package/src/client/browser-entry/action-dispatch.ts +116 -0
- package/src/client/browser-entry/hmr.ts +81 -0
- package/src/client/browser-entry/hydrate.ts +145 -0
- package/src/client/browser-entry/index.ts +143 -0
- package/src/client/browser-entry/post-hydration.ts +119 -0
- package/src/client/browser-entry/router-init.ts +193 -0
- package/src/client/browser-entry/rsc-stream.ts +157 -0
- package/src/client/browser-entry/scroll.ts +27 -0
- package/src/client/error-boundary.tsx +48 -16
- package/src/client/error-reconstituter.tsx +65 -0
- package/src/client/form.tsx +14 -7
- package/src/client/history.ts +26 -4
- package/src/client/index.ts +65 -38
- package/src/client/internal.ts +57 -0
- package/src/client/link-pending-store.ts +111 -0
- package/src/client/link.tsx +342 -113
- package/src/client/nav-link-store.ts +47 -0
- package/src/client/navigation-api-types.ts +112 -0
- package/src/client/navigation-api.ts +332 -0
- package/src/client/navigation-context.ts +31 -6
- package/src/client/navigation-root.tsx +342 -0
- package/src/client/nuqs-adapter.tsx +16 -3
- package/src/client/router-ref.ts +1 -1
- package/src/client/router.ts +299 -72
- package/src/client/rsc-fetch.ts +97 -8
- package/src/client/segment-cache.ts +1 -1
- package/src/client/segment-outlet.tsx +86 -0
- package/src/client/ssr-data.ts +13 -5
- package/src/client/stale-reload.ts +72 -3
- package/src/client/top-loader.tsx +18 -6
- package/src/client/use-link-status.ts +7 -7
- package/src/client/use-params.ts +7 -5
- package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
- package/src/client/use-query-states.ts +9 -3
- package/src/client/use-router.ts +1 -1
- package/src/codec.ts +49 -0
- package/src/config-types.ts +264 -0
- package/src/config-validation.ts +303 -0
- package/src/content/index.ts +5 -13
- package/src/cookies/define-cookie.ts +78 -25
- package/src/cookies/index.ts +8 -0
- package/src/fonts/bundle.ts +142 -0
- package/src/fonts/css.ts +2 -1
- package/src/fonts/dev-middleware.ts +74 -0
- package/src/fonts/pipeline.ts +275 -0
- package/src/fonts/transform.ts +353 -0
- package/src/fonts/types.ts +50 -1
- package/src/fonts/virtual-modules.ts +159 -0
- package/src/index.ts +314 -355
- package/src/plugin-context.ts +240 -0
- package/src/plugins/adapter-build.ts +9 -3
- package/src/plugins/build-manifest.ts +13 -2
- package/src/plugins/build-report.ts +3 -3
- package/src/plugins/client-chunks.ts +65 -0
- package/src/plugins/content.ts +1 -1
- package/src/plugins/dev-404-page.ts +418 -0
- package/src/plugins/dev-browser-logs.ts +288 -0
- package/src/plugins/dev-error-overlay.ts +286 -42
- package/src/plugins/dev-error-page.ts +536 -0
- package/src/plugins/dev-logs.ts +1 -1
- package/src/plugins/dev-server.ts +146 -19
- package/src/plugins/dev-terminal-error.ts +217 -0
- package/src/plugins/entries.ts +111 -10
- package/src/plugins/fonts.ts +133 -638
- package/src/plugins/mdx.ts +1 -1
- package/src/plugins/routing.ts +213 -31
- package/src/plugins/server-action-exports.ts +1 -1
- package/src/plugins/server-bundle.ts +32 -1
- package/src/plugins/shims.ts +136 -35
- package/src/plugins/static-build.ts +17 -11
- package/src/routing/codegen-shared.ts +74 -0
- package/src/routing/codegen-types.ts +37 -0
- package/src/routing/codegen.ts +112 -173
- package/src/routing/convention-lint.ts +356 -0
- package/src/routing/index.ts +2 -0
- package/src/routing/link-codegen.ts +262 -0
- package/src/routing/scanner.ts +93 -23
- package/src/routing/segment-classify.ts +89 -0
- package/src/routing/status-file-lint.ts +3 -2
- package/src/routing/types.ts +17 -4
- package/src/rsc-runtime/rsc.ts +2 -0
- package/src/rsc-runtime/ssr.ts +50 -0
- package/src/rsc-runtime/vendor-types.d.ts +7 -0
- package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
- package/src/search-params/define.ts +482 -0
- package/src/search-params/index.ts +14 -20
- package/src/search-params/registry.ts +2 -2
- package/src/search-params/wrappers.ts +85 -0
- package/src/segment-params/define.ts +279 -0
- package/src/segment-params/index.ts +9 -0
- package/src/server/access-gate.tsx +70 -29
- package/src/server/action-client.ts +88 -15
- package/src/server/action-encryption.ts +144 -0
- package/src/server/action-handler.ts +53 -6
- package/src/server/actions.ts +10 -9
- package/src/server/als-registry.ts +34 -6
- package/src/server/build-manifest.ts +10 -4
- package/src/server/compress.ts +25 -7
- package/src/server/debug.ts +1 -1
- package/src/server/default-logger.ts +99 -0
- package/src/server/deny-page-resolver.ts +154 -0
- package/src/server/deny-renderer.ts +24 -38
- package/src/server/dev-holding-server.ts +185 -0
- package/src/server/dev-source-map.ts +31 -0
- package/src/server/dev-warnings.ts +4 -49
- package/src/server/early-hints.ts +36 -15
- package/src/server/error-boundary-wrapper.ts +74 -22
- package/src/server/fallback-error.ts +74 -102
- package/src/server/flight-injection-state.ts +113 -0
- package/src/server/flight-scripts.ts +62 -0
- package/src/server/flush.ts +2 -1
- package/src/server/form-data.ts +76 -0
- package/src/server/html-injectors.ts +280 -120
- package/src/server/index.ts +25 -177
- package/src/server/internal.ts +169 -0
- package/src/server/logger.ts +44 -36
- package/src/server/middleware-runner.ts +31 -4
- package/src/server/node-stream-transforms.ts +509 -0
- package/src/server/page-deny-boundary.tsx +56 -0
- package/src/server/pipeline-interception.ts +17 -16
- package/src/server/pipeline-metadata.ts +13 -0
- package/src/server/pipeline.ts +261 -66
- package/src/server/primitives.ts +111 -28
- package/src/server/render-timeout.ts +108 -0
- package/src/server/request-context.ts +293 -132
- package/src/server/route-element-builder.ts +283 -191
- package/src/server/route-handler.ts +24 -4
- package/src/server/route-matcher.ts +31 -20
- package/src/server/rsc-entry/api-handler.ts +15 -16
- package/src/server/rsc-entry/error-renderer.ts +305 -89
- package/src/server/rsc-entry/helpers.ts +134 -5
- package/src/server/rsc-entry/index.ts +304 -111
- package/src/server/rsc-entry/rsc-payload.ts +65 -18
- package/src/server/rsc-entry/rsc-stream.ts +81 -13
- package/src/server/rsc-entry/ssr-bridge.ts +14 -5
- package/src/server/rsc-entry/ssr-renderer.ts +171 -38
- package/src/server/safe-load.ts +60 -0
- package/src/server/sensitive-fields.ts +230 -0
- package/src/server/sitemap-generator.ts +338 -0
- package/src/server/sitemap-handler.ts +126 -0
- package/src/server/slot-resolver.ts +244 -229
- package/src/server/ssr-entry.ts +215 -32
- package/src/server/ssr-render.ts +289 -67
- package/src/server/ssr-wrappers.tsx +139 -0
- package/src/server/status-code-resolver.ts +1 -1
- package/src/server/stream-utils.ts +213 -0
- package/src/server/tracing.ts +20 -9
- package/src/server/tree-builder.ts +92 -58
- package/src/server/types.ts +3 -6
- package/src/server/version-skew.ts +104 -0
- package/src/shared/merge-search-params.ts +55 -0
- package/src/shims/font-google.ts +1 -1
- package/src/shims/font-local.ts +34 -0
- package/src/shims/headers.ts +5 -1
- package/src/shims/navigation-client.ts +1 -1
- package/src/shims/navigation.ts +7 -2
- package/src/utils/directive-parser.ts +5 -2
- package/src/utils/state-machine.ts +111 -0
- package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
- package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
- package/dist/_chunks/format-DviM89f0.js.map +0 -1
- package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
- package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
- package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
- package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
- package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
- package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
- package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
- package/dist/cache/register-cached-function.d.ts +0 -17
- package/dist/cache/register-cached-function.d.ts.map +0 -1
- package/dist/client/browser-entry.d.ts +0 -21
- package/dist/client/browser-entry.d.ts.map +0 -1
- package/dist/client/link-status-provider.d.ts +0 -11
- package/dist/client/link-status-provider.d.ts.map +0 -1
- package/dist/client/transition-root.d.ts.map +0 -1
- package/dist/client/use-navigation-pending.d.ts.map +0 -1
- package/dist/cookies/index.js.map +0 -1
- package/dist/plugins/cache-transform.d.ts +0 -36
- package/dist/plugins/cache-transform.d.ts.map +0 -1
- package/dist/plugins/dynamic-transform.d.ts +0 -72
- package/dist/plugins/dynamic-transform.d.ts.map +0 -1
- package/dist/search-params/analyze.d.ts +0 -54
- package/dist/search-params/analyze.d.ts.map +0 -1
- package/dist/search-params/builtin-codecs.d.ts +0 -105
- package/dist/search-params/builtin-codecs.d.ts.map +0 -1
- package/dist/search-params/codecs.d.ts +0 -53
- package/dist/search-params/codecs.d.ts.map +0 -1
- package/dist/search-params/create.d.ts +0 -106
- package/dist/search-params/create.d.ts.map +0 -1
- package/dist/server/prerender.d.ts +0 -77
- package/dist/server/prerender.d.ts.map +0 -1
- package/dist/server/response-cache.d.ts +0 -54
- package/dist/server/response-cache.d.ts.map +0 -1
- package/src/cache/register-cached-function.ts +0 -103
- package/src/client/browser-entry.ts +0 -678
- package/src/client/link-status-provider.tsx +0 -30
- package/src/client/transition-root.tsx +0 -166
- package/src/plugins/cache-transform.ts +0 -199
- package/src/plugins/dynamic-transform.ts +0 -161
- package/src/search-params/analyze.ts +0 -192
- package/src/search-params/builtin-codecs.ts +0 -228
- package/src/search-params/create.ts +0 -321
- package/src/server/prerender.ts +0 -139
- package/src/server/response-cache.ts +0 -410
package/dist/client/index.js
CHANGED
|
@@ -1,21 +1,25 @@
|
|
|
1
1
|
"use client";
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
6
|
-
import
|
|
2
|
+
import { r as __exportAll } from "../_chunks/chunk-BYIpzuS7.js";
|
|
3
|
+
import { t as classifyUrlSegment } from "../_chunks/segment-classify-BDNn6EzD.js";
|
|
4
|
+
import { n as useQueryStates, r as getSearchParamsDefinition } from "../_chunks/use-query-states-BiV5GJgm.js";
|
|
5
|
+
import { t as mergePreservedSearchParams } from "../_chunks/merge-search-params-Cm_KIWDX.js";
|
|
6
|
+
import { c as cachedSearchParams, i as _setCachedSearch, n as getSsrData, s as cachedSearch } from "../_chunks/ssr-data-DzuI0bIV.js";
|
|
7
|
+
import { n as useSegmentContext } from "../_chunks/segment-context-fHFLF1PE.js";
|
|
8
|
+
import { c as usePendingNavigationUrl, d as unmountLinkForCurrentNavigation, l as LINK_IDLE, n as useSegmentParams, p as getRouterOrNull, s as useNavigationContext, u as setLinkForCurrentNavigation } from "../_chunks/use-params-Br9YSUFV.js";
|
|
9
|
+
import { t as _registerUseCookieModule } from "../_chunks/define-cookie-BowvzoP0.js";
|
|
10
|
+
import { createContext, useActionState as useActionState$1, useContext, useEffect, useOptimistic, useRef, useSyncExternalStore, useTransition } from "react";
|
|
7
11
|
import { jsx } from "react/jsx-runtime";
|
|
8
12
|
//#region src/client/use-link-status.ts
|
|
9
13
|
/**
|
|
10
14
|
* React context provided by <Link>. Holds the pending status
|
|
11
15
|
* for that specific link's navigation.
|
|
12
16
|
*/
|
|
13
|
-
var LinkStatusContext = createContext({
|
|
17
|
+
var LinkStatusContext = createContext({ isPending: false });
|
|
14
18
|
/**
|
|
15
|
-
* Returns `{
|
|
19
|
+
* Returns `{ isPending: true }` while the nearest parent `<Link>` component's
|
|
16
20
|
* navigation is in flight. Must be used inside a `<Link>` component's children.
|
|
17
21
|
*
|
|
18
|
-
* Unlike `
|
|
22
|
+
* Unlike `usePendingNavigation()` which is global, this hook is scoped to
|
|
19
23
|
* the nearest parent `<Link>` — only the link the user clicked shows pending.
|
|
20
24
|
*
|
|
21
25
|
* ```tsx
|
|
@@ -23,8 +27,8 @@ var LinkStatusContext = createContext({ pending: false });
|
|
|
23
27
|
* import { Link, useLinkStatus } from '@timber-js/app/client'
|
|
24
28
|
*
|
|
25
29
|
* function Hint() {
|
|
26
|
-
* const {
|
|
27
|
-
* return <span className={
|
|
30
|
+
* const { isPending } = useLinkStatus()
|
|
31
|
+
* return <span className={isPending ? 'opacity-50' : ''} />
|
|
28
32
|
* }
|
|
29
33
|
*
|
|
30
34
|
* export function NavLink({ href, children }) {
|
|
@@ -39,191 +43,35 @@ var LinkStatusContext = createContext({ pending: false });
|
|
|
39
43
|
function useLinkStatus() {
|
|
40
44
|
return useContext(LinkStatusContext);
|
|
41
45
|
}
|
|
42
|
-
//#endregion
|
|
43
|
-
//#region src/client/navigation-context.ts
|
|
44
|
-
/**
|
|
45
|
-
* NavigationContext — React context for navigation state.
|
|
46
|
-
*
|
|
47
|
-
* Holds the current route params and pathname, updated atomically
|
|
48
|
-
* with the RSC tree on each navigation. This replaces the previous
|
|
49
|
-
* useSyncExternalStore approach for useParams() and usePathname(),
|
|
50
|
-
* which suffered from a timing gap: the new tree could commit before
|
|
51
|
-
* the external store re-renders fired, causing a frame where both
|
|
52
|
-
* old and new active states were visible simultaneously.
|
|
53
|
-
*
|
|
54
|
-
* By wrapping the RSC payload element in NavigationProvider inside
|
|
55
|
-
* renderRoot(), the context value and the element tree are passed to
|
|
56
|
-
* reactRoot.render() in the same call — atomic by construction.
|
|
57
|
-
* All consumers (useParams, usePathname) see the new values in the
|
|
58
|
-
* same render pass as the new tree.
|
|
59
|
-
*
|
|
60
|
-
* During SSR, no NavigationProvider is mounted. Hooks fall back to
|
|
61
|
-
* the ALS-backed getSsrData() for per-request isolation.
|
|
62
|
-
*
|
|
63
|
-
* IMPORTANT: createContext and useContext are NOT available in the RSC
|
|
64
|
-
* environment (React Server Components use a stripped-down React).
|
|
65
|
-
* The context is lazily initialized on first access, and all functions
|
|
66
|
-
* that depend on these APIs are safe to call from any environment —
|
|
67
|
-
* they return null or no-op when the APIs aren't available.
|
|
68
|
-
*
|
|
69
|
-
* SINGLETON GUARANTEE: All shared mutable state uses globalThis via
|
|
70
|
-
* Symbol.for keys. The RSC client bundler can duplicate this module
|
|
71
|
-
* across chunks (browser-entry graph + client-reference graph). With
|
|
72
|
-
* ESM output, each chunk gets its own module scope — module-level
|
|
73
|
-
* variables would create separate singleton instances per chunk.
|
|
74
|
-
* globalThis guarantees a single instance regardless of duplication.
|
|
75
|
-
*
|
|
76
|
-
* This workaround will be removed when Rolldown ships `format: 'app'`
|
|
77
|
-
* (module registry format that deduplicates like webpack/Turbopack).
|
|
78
|
-
* See design/27-chunking-strategy.md.
|
|
79
|
-
*
|
|
80
|
-
* See design/19-client-navigation.md §"NavigationContext"
|
|
81
|
-
*/
|
|
82
|
-
/**
|
|
83
|
-
* The context is created lazily to avoid calling createContext at module
|
|
84
|
-
* level. In the RSC environment, React.createContext doesn't exist —
|
|
85
|
-
* calling it at import time would crash the server.
|
|
86
|
-
*
|
|
87
|
-
* Context instances are stored on globalThis (NOT in module-level
|
|
88
|
-
* variables) because the ESM bundler can duplicate this module across
|
|
89
|
-
* chunks. Module-level variables would create separate instances per
|
|
90
|
-
* chunk — the provider in TransitionRoot (index chunk) would use
|
|
91
|
-
* context A while the consumer in LinkStatusProvider (shared chunk)
|
|
92
|
-
* reads from context B. globalThis guarantees a single instance.
|
|
93
|
-
*
|
|
94
|
-
* See design/27-chunking-strategy.md §"Singleton Safety"
|
|
95
|
-
*/
|
|
96
|
-
var NAV_CTX_KEY = Symbol.for("__timber_nav_ctx");
|
|
97
|
-
var PENDING_CTX_KEY = Symbol.for("__timber_pending_nav_ctx");
|
|
98
|
-
function getOrCreateContext() {
|
|
99
|
-
const existing = globalThis[NAV_CTX_KEY];
|
|
100
|
-
if (existing !== void 0) return existing;
|
|
101
|
-
if (typeof React.createContext === "function") {
|
|
102
|
-
const ctx = React.createContext(null);
|
|
103
|
-
globalThis[NAV_CTX_KEY] = ctx;
|
|
104
|
-
return ctx;
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
/**
|
|
108
|
-
* Read the navigation context. Returns null during SSR (no provider)
|
|
109
|
-
* or in the RSC environment (no context available).
|
|
110
|
-
* Internal — used by useParams() and usePathname().
|
|
111
|
-
*/
|
|
112
|
-
function useNavigationContext() {
|
|
113
|
-
const ctx = getOrCreateContext();
|
|
114
|
-
if (!ctx) return null;
|
|
115
|
-
if (typeof React.useContext !== "function") return null;
|
|
116
|
-
return React.useContext(ctx);
|
|
117
|
-
}
|
|
118
|
-
/**
|
|
119
|
-
* Wraps children with NavigationContext.Provider.
|
|
120
|
-
*
|
|
121
|
-
* Used in browser-entry.ts renderRoot to wrap the RSC payload element
|
|
122
|
-
* so that navigation state updates atomically with the tree render.
|
|
123
|
-
*/
|
|
124
|
-
function NavigationProvider({ value, children }) {
|
|
125
|
-
const ctx = getOrCreateContext();
|
|
126
|
-
if (!ctx) return children;
|
|
127
|
-
return createElement(ctx.Provider, { value }, children);
|
|
128
|
-
}
|
|
129
|
-
/**
|
|
130
|
-
* Navigation state communicated between the router and renderRoot.
|
|
131
|
-
*
|
|
132
|
-
* The router calls setNavigationState() before renderRoot(). The
|
|
133
|
-
* renderRoot callback reads via getNavigationState() to create the
|
|
134
|
-
* NavigationProvider with the correct params/pathname.
|
|
135
|
-
*
|
|
136
|
-
* This is NOT used by hooks directly — hooks read from React context.
|
|
137
|
-
*
|
|
138
|
-
* Stored on globalThis (like the context instances above) because the
|
|
139
|
-
* router lives in one chunk while renderRoot lives in another. Module-
|
|
140
|
-
* level variables would be separate per chunk.
|
|
141
|
-
*/
|
|
142
|
-
var NAV_STATE_KEY = Symbol.for("__timber_nav_state");
|
|
143
|
-
function _getNavStateStore() {
|
|
144
|
-
const g = globalThis;
|
|
145
|
-
if (!g[NAV_STATE_KEY]) g[NAV_STATE_KEY] = { current: {
|
|
146
|
-
params: {},
|
|
147
|
-
pathname: "/"
|
|
148
|
-
} };
|
|
149
|
-
return g[NAV_STATE_KEY];
|
|
150
|
-
}
|
|
151
|
-
function setNavigationState(state) {
|
|
152
|
-
_getNavStateStore().current = state;
|
|
153
|
-
}
|
|
154
|
-
function getNavigationState() {
|
|
155
|
-
return _getNavStateStore().current;
|
|
156
|
-
}
|
|
157
|
-
/**
|
|
158
|
-
* Separate context for the in-flight navigation URL. Provided by
|
|
159
|
-
* TransitionRoot (urgent useState), consumed by LinkStatusProvider
|
|
160
|
-
* and useNavigationPending.
|
|
161
|
-
*
|
|
162
|
-
* Uses globalThis via Symbol.for for the same reason as NavigationContext
|
|
163
|
-
* above — the bundler may duplicate this module across chunks, and module-
|
|
164
|
-
* level variables would create separate context instances.
|
|
165
|
-
*/
|
|
166
|
-
function getOrCreatePendingContext() {
|
|
167
|
-
const existing = globalThis[PENDING_CTX_KEY];
|
|
168
|
-
if (existing !== void 0) return existing;
|
|
169
|
-
if (typeof React.createContext === "function") {
|
|
170
|
-
const ctx = React.createContext(null);
|
|
171
|
-
globalThis[PENDING_CTX_KEY] = ctx;
|
|
172
|
-
return ctx;
|
|
173
|
-
}
|
|
174
|
-
}
|
|
175
46
|
/**
|
|
176
|
-
*
|
|
177
|
-
*
|
|
47
|
+
* Store metadata from Link's onClick for the next navigate event.
|
|
48
|
+
* Called synchronously in the click handler — the navigate event
|
|
49
|
+
* fires synchronously after onClick returns.
|
|
178
50
|
*/
|
|
179
|
-
function
|
|
180
|
-
const ctx = getOrCreatePendingContext();
|
|
181
|
-
if (!ctx) return null;
|
|
182
|
-
if (typeof React.useContext !== "function") return null;
|
|
183
|
-
return React.useContext(ctx);
|
|
184
|
-
}
|
|
51
|
+
function setNavLinkMetadata(metadata) {}
|
|
185
52
|
//#endregion
|
|
186
|
-
//#region src/client/
|
|
187
|
-
var NOT_PENDING = { pending: false };
|
|
188
|
-
var IS_PENDING = { pending: true };
|
|
53
|
+
//#region src/client/navigation-api.ts
|
|
189
54
|
/**
|
|
190
|
-
*
|
|
191
|
-
*
|
|
192
|
-
* just a context provider around children.
|
|
55
|
+
* Returns true if the Navigation API is available in the current environment.
|
|
56
|
+
* Feature-detected at runtime — no polyfill.
|
|
193
57
|
*/
|
|
194
|
-
function
|
|
195
|
-
|
|
196
|
-
return /* @__PURE__ */ jsx(LinkStatusContext.Provider, {
|
|
197
|
-
value: status,
|
|
198
|
-
children
|
|
199
|
-
});
|
|
58
|
+
function hasNavigationApi() {
|
|
59
|
+
return typeof window !== "undefined" && "navigation" in window && window.navigation != null;
|
|
200
60
|
}
|
|
201
61
|
//#endregion
|
|
202
|
-
//#region src/client/
|
|
203
|
-
/**
|
|
204
|
-
* Set the global router instance. Called once during bootstrap.
|
|
205
|
-
*/
|
|
206
|
-
function setGlobalRouter(router) {
|
|
207
|
-
_setGlobalRouter(router);
|
|
208
|
-
}
|
|
209
|
-
/**
|
|
210
|
-
* Get the global router instance. Throws if called before bootstrap.
|
|
211
|
-
* Used by client-side hooks (useNavigationPending, etc.)
|
|
212
|
-
*/
|
|
213
|
-
function getRouter() {
|
|
214
|
-
if (!globalRouter) throw new Error("[timber] Router not initialized. getRouter() was called before bootstrap().");
|
|
215
|
-
return globalRouter;
|
|
216
|
-
}
|
|
62
|
+
//#region src/client/link.tsx
|
|
217
63
|
/**
|
|
218
|
-
*
|
|
219
|
-
*
|
|
220
|
-
*
|
|
64
|
+
* Read the current URL's search string without requiring a React hook.
|
|
65
|
+
* On the client, reads window.location.search. During SSR, reads from
|
|
66
|
+
* the request context (getSsrData). Returns empty string if unavailable.
|
|
221
67
|
*/
|
|
222
|
-
function
|
|
223
|
-
return
|
|
68
|
+
function getCurrentSearch() {
|
|
69
|
+
if (typeof window !== "undefined") return window.location.search;
|
|
70
|
+
const data = getSsrData();
|
|
71
|
+
if (!data) return "";
|
|
72
|
+
const str = new URLSearchParams(data.searchParams).toString();
|
|
73
|
+
return str ? `?${str}` : "";
|
|
224
74
|
}
|
|
225
|
-
//#endregion
|
|
226
|
-
//#region src/client/link.tsx
|
|
227
75
|
/**
|
|
228
76
|
* Reject dangerous URL schemes that could execute script.
|
|
229
77
|
* Security: design/13-security.md § Link scheme injection (test #9)
|
|
@@ -247,25 +95,53 @@ function isInternalHref(href) {
|
|
|
247
95
|
* - [...param] → catch-all (joined with /)
|
|
248
96
|
* - [[...param]] → optional catch-all (omitted if undefined/empty)
|
|
249
97
|
*/
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
98
|
+
/**
|
|
99
|
+
* Parse a route pattern's path portion into classified segments.
|
|
100
|
+
* Exported for testing. Uses the shared character-based classifier.
|
|
101
|
+
*/
|
|
102
|
+
function parseSegments(pattern) {
|
|
103
|
+
return pattern.split("/").filter(Boolean).map(classifyUrlSegment);
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Resolve a single classified segment into its string representation.
|
|
107
|
+
* Returns null for optional catch-all with no value (filtered out before join).
|
|
108
|
+
*/
|
|
109
|
+
function resolveSegment(seg, params, pattern) {
|
|
110
|
+
switch (seg.kind) {
|
|
111
|
+
case "static": return seg.value;
|
|
112
|
+
case "optional-catch-all": {
|
|
113
|
+
const value = params[seg.name];
|
|
114
|
+
if (value === void 0 || Array.isArray(value) && value.length === 0) return null;
|
|
255
115
|
return (Array.isArray(value) ? value : [value]).map(encodeURIComponent).join("/");
|
|
256
116
|
}
|
|
257
|
-
|
|
258
|
-
const value = params[
|
|
259
|
-
if (value === void 0) throw new Error(`<Link> missing required catch-all param "${
|
|
117
|
+
case "catch-all": {
|
|
118
|
+
const value = params[seg.name];
|
|
119
|
+
if (value === void 0) throw new Error(`<Link> missing required catch-all param "${seg.name}" for pattern "${pattern}".`);
|
|
260
120
|
const segments = Array.isArray(value) ? value : [value];
|
|
261
|
-
if (segments.length === 0) throw new Error(`<Link> catch-all param "${
|
|
121
|
+
if (segments.length === 0) throw new Error(`<Link> catch-all param "${seg.name}" must have at least one segment for pattern "${pattern}".`);
|
|
262
122
|
return segments.map(encodeURIComponent).join("/");
|
|
263
123
|
}
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
124
|
+
case "dynamic": {
|
|
125
|
+
const value = params[seg.name];
|
|
126
|
+
if (value === void 0) throw new Error(`<Link> missing required param "${seg.name}" for pattern "${pattern}".`);
|
|
127
|
+
if (Array.isArray(value)) throw new Error(`<Link> param "${seg.name}" expected a string but received an array for pattern "${pattern}".`);
|
|
128
|
+
return encodeURIComponent(String(value));
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Split a URL pattern into the path portion and any trailing ?query/#hash suffix.
|
|
134
|
+
* Uses URL parsing for correctness rather than manual index arithmetic.
|
|
135
|
+
*/
|
|
136
|
+
function splitPatternSuffix(pattern) {
|
|
137
|
+
if (!pattern.includes("?") && !pattern.includes("#")) return [pattern, ""];
|
|
138
|
+
const url = new URL(pattern, "http://x");
|
|
139
|
+
const suffix = url.search + url.hash;
|
|
140
|
+
return [pattern.slice(0, pattern.length - suffix.length), suffix];
|
|
141
|
+
}
|
|
142
|
+
function interpolateParams(pattern, params) {
|
|
143
|
+
const [pathPart, suffix] = splitPatternSuffix(pattern);
|
|
144
|
+
return ("/" + parseSegments(pathPart).map((seg) => resolveSegment(seg, params, pattern)).filter((s) => s !== null).join("/") || "/") + suffix;
|
|
269
145
|
}
|
|
270
146
|
/**
|
|
271
147
|
* Resolve the final href string from Link props.
|
|
@@ -275,12 +151,30 @@ function interpolateParams(pattern, params) {
|
|
|
275
151
|
* - searchParams serialization via SearchParamsDefinition
|
|
276
152
|
* - Validation that searchParams and inline query strings are exclusive
|
|
277
153
|
*/
|
|
154
|
+
/**
|
|
155
|
+
* Runtime discriminator: treat `searchParams` as the legacy wrapped shape
|
|
156
|
+
* only when it literally has a `definition` key. Everything else is the
|
|
157
|
+
* flat `Partial<T>` values shape (TIM-830).
|
|
158
|
+
*/
|
|
159
|
+
function isWrappedSearchParamsProp(sp) {
|
|
160
|
+
return "definition" in sp;
|
|
161
|
+
}
|
|
278
162
|
function resolveHref(href, params, searchParams) {
|
|
279
163
|
let resolvedPath = href;
|
|
280
164
|
if (params) resolvedPath = interpolateParams(href, params);
|
|
281
165
|
if (searchParams) {
|
|
282
166
|
if (resolvedPath.includes("?")) throw new Error("<Link> received both a searchParams prop and a query string in href. These are mutually exclusive — use one or the other.");
|
|
283
|
-
|
|
167
|
+
let definition;
|
|
168
|
+
let values;
|
|
169
|
+
if (isWrappedSearchParamsProp(searchParams)) {
|
|
170
|
+
definition = searchParams.definition;
|
|
171
|
+
values = searchParams.values;
|
|
172
|
+
} else {
|
|
173
|
+
definition = getSearchParamsDefinition(href);
|
|
174
|
+
values = searchParams;
|
|
175
|
+
if (!definition) throw new Error(`<Link> received a flat searchParams object for href "${href}", but no SearchParamsDefinition is registered for that route. Either the route does not export \`searchParams\` from its \`params.ts\`/\`page.tsx\`, or the search-params registry module was not loaded. For external or computed hrefs, pass the legacy \`{ definition, values }\` shape instead.`);
|
|
176
|
+
}
|
|
177
|
+
const qs = definition.serialize(values);
|
|
284
178
|
if (qs) resolvedPath = `${resolvedPath}?${qs}`;
|
|
285
179
|
}
|
|
286
180
|
return resolvedPath;
|
|
@@ -323,16 +217,33 @@ function shouldInterceptClick(event, resolvedHref) {
|
|
|
323
217
|
* navigation via the router. No global event delegation — each Link owns
|
|
324
218
|
* its own click handling.
|
|
325
219
|
*
|
|
326
|
-
* Supports typed routes via
|
|
327
|
-
*
|
|
220
|
+
* Supports typed routes via the Routes interface (populated by codegen).
|
|
221
|
+
* At runtime:
|
|
222
|
+
* - `segmentParams` prop interpolates dynamic segments in the href pattern
|
|
328
223
|
* - `searchParams` prop serializes query parameters via a SearchParamsDefinition
|
|
224
|
+
*
|
|
225
|
+
* Typed via the LinkFunction callable interface. The base call signature
|
|
226
|
+
* forbids segmentParams; per-route signatures are added by codegen via
|
|
227
|
+
* interface merging. See TIM-624.
|
|
329
228
|
*/
|
|
330
|
-
|
|
331
|
-
const { href:
|
|
229
|
+
var Link = function LinkImpl(props) {
|
|
230
|
+
const { href, prefetch, scroll, segmentParams, searchParams, preserveSearchParams, onNavigate, onClick: userOnClick, onMouseEnter: userOnMouseEnter, children, ...rest } = props;
|
|
231
|
+
const { href: baseHref } = buildLinkProps({
|
|
332
232
|
href,
|
|
333
|
-
params,
|
|
233
|
+
params: segmentParams,
|
|
334
234
|
searchParams
|
|
335
235
|
});
|
|
236
|
+
const [linkStatus, setIsPending] = useOptimistic(LINK_IDLE);
|
|
237
|
+
const linkInstanceRef = useRef(null);
|
|
238
|
+
if (!linkInstanceRef.current) linkInstanceRef.current = { setIsPending };
|
|
239
|
+
else linkInstanceRef.current.setIsPending = setIsPending;
|
|
240
|
+
useEffect(() => {
|
|
241
|
+
const instance = linkInstanceRef.current;
|
|
242
|
+
return () => {
|
|
243
|
+
if (instance) unmountLinkForCurrentNavigation(instance);
|
|
244
|
+
};
|
|
245
|
+
}, []);
|
|
246
|
+
const resolvedHref = preserveSearchParams ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams) : baseHref;
|
|
336
247
|
const internal = isInternalHref(resolvedHref);
|
|
337
248
|
const handleClick = internal ? (event) => {
|
|
338
249
|
userOnClick?.(event);
|
|
@@ -349,800 +260,40 @@ function Link({ href, prefetch, scroll, params, searchParams, onNavigate, onClic
|
|
|
349
260
|
}
|
|
350
261
|
const router = getRouterOrNull();
|
|
351
262
|
if (!router) return;
|
|
352
|
-
event.preventDefault();
|
|
353
263
|
const shouldScroll = scroll !== false;
|
|
354
|
-
|
|
264
|
+
setLinkForCurrentNavigation(linkInstanceRef.current);
|
|
265
|
+
if (hasNavigationApi()) {
|
|
266
|
+
setNavLinkMetadata({
|
|
267
|
+
scroll: shouldScroll,
|
|
268
|
+
linkInstance: linkInstanceRef.current
|
|
269
|
+
});
|
|
270
|
+
return;
|
|
271
|
+
}
|
|
272
|
+
event.preventDefault();
|
|
273
|
+
const navHref = preserveSearchParams ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams) : resolvedHref;
|
|
274
|
+
router.navigate(navHref, { scroll: shouldScroll });
|
|
355
275
|
} : userOnClick;
|
|
356
276
|
const handleMouseEnter = internal && prefetch ? (event) => {
|
|
357
277
|
userOnMouseEnter?.(event);
|
|
358
278
|
const router = getRouterOrNull();
|
|
359
|
-
if (router)
|
|
279
|
+
if (router) {
|
|
280
|
+
const prefetchHref = preserveSearchParams ? mergePreservedSearchParams(baseHref, getCurrentSearch(), preserveSearchParams) : resolvedHref;
|
|
281
|
+
router.prefetch(prefetchHref);
|
|
282
|
+
}
|
|
360
283
|
} : userOnMouseEnter;
|
|
361
284
|
return /* @__PURE__ */ jsx("a", {
|
|
362
285
|
...rest,
|
|
363
286
|
href: resolvedHref,
|
|
364
287
|
onClick: handleClick,
|
|
365
288
|
onMouseEnter: handleMouseEnter,
|
|
366
|
-
children: /* @__PURE__ */ jsx(
|
|
367
|
-
|
|
289
|
+
children: /* @__PURE__ */ jsx(LinkStatusContext.Provider, {
|
|
290
|
+
value: linkStatus,
|
|
368
291
|
children
|
|
369
292
|
})
|
|
370
293
|
});
|
|
371
|
-
}
|
|
372
|
-
//#endregion
|
|
373
|
-
//#region src/client/segment-cache.ts
|
|
374
|
-
/**
|
|
375
|
-
* Maintains the client-side segment tree representing currently mounted
|
|
376
|
-
* layouts and pages. Used for navigation reconciliation — the router diffs
|
|
377
|
-
* new routes against this tree to determine which segments to re-fetch.
|
|
378
|
-
*/
|
|
379
|
-
var SegmentCache = class {
|
|
380
|
-
root;
|
|
381
|
-
get(segment) {
|
|
382
|
-
if (segment === "/" || segment === this.root?.segment) return this.root;
|
|
383
|
-
}
|
|
384
|
-
set(segment, node) {
|
|
385
|
-
if (segment === "/" || !this.root) this.root = node;
|
|
386
|
-
}
|
|
387
|
-
clear() {
|
|
388
|
-
this.root = void 0;
|
|
389
|
-
}
|
|
390
|
-
/**
|
|
391
|
-
* Serialize the mounted segment tree for the X-Timber-State-Tree header.
|
|
392
|
-
* Only includes sync segments — async segments are excluded because the
|
|
393
|
-
* server must always re-render them (they may depend on request context).
|
|
394
|
-
*
|
|
395
|
-
* When mergeableFilter is provided, only segments whose paths are in the
|
|
396
|
-
* set are included. This ensures the server only skips segments that the
|
|
397
|
-
* client can actually merge (i.e., segments whose cached element tree
|
|
398
|
-
* contains an inner SegmentProvider the merger can splice into).
|
|
399
|
-
*
|
|
400
|
-
* This is a performance optimization only, NOT a security boundary.
|
|
401
|
-
* The server always runs all access.ts files regardless of the state tree.
|
|
402
|
-
*/
|
|
403
|
-
serializeStateTree(mergeableFilter) {
|
|
404
|
-
const segments = [];
|
|
405
|
-
if (this.root) collectSyncSegments(this.root, segments, mergeableFilter);
|
|
406
|
-
return { segments };
|
|
407
|
-
}
|
|
408
|
-
};
|
|
409
|
-
/** Recursively collect sync segment paths from the tree */
|
|
410
|
-
function collectSyncSegments(node, out, mergeableFilter) {
|
|
411
|
-
if (!node.isAsync && (!mergeableFilter || mergeableFilter.has(node.segment))) out.push(node.segment);
|
|
412
|
-
for (const child of node.children.values()) collectSyncSegments(child, out, mergeableFilter);
|
|
413
|
-
}
|
|
414
|
-
/**
|
|
415
|
-
* Build a SegmentNode tree from flat segment metadata.
|
|
416
|
-
*
|
|
417
|
-
* Takes an ordered list of segment descriptors (root → leaf) from the
|
|
418
|
-
* server's X-Timber-Segments header and constructs the hierarchical
|
|
419
|
-
* tree structure that SegmentCache expects.
|
|
420
|
-
*
|
|
421
|
-
* Each segment is nested as a child of the previous one, forming a
|
|
422
|
-
* linear chain from root to leaf. The leaf segment (page) is excluded
|
|
423
|
-
* from the tree — pages are never cached across navigations.
|
|
424
|
-
*/
|
|
425
|
-
function buildSegmentTree(segments) {
|
|
426
|
-
if (segments.length === 0) return void 0;
|
|
427
|
-
const layouts = segments.length > 1 ? segments.slice(0, -1) : segments;
|
|
428
|
-
let root;
|
|
429
|
-
let parent;
|
|
430
|
-
for (const info of layouts) {
|
|
431
|
-
const node = {
|
|
432
|
-
segment: info.path,
|
|
433
|
-
payload: null,
|
|
434
|
-
isAsync: info.isAsync,
|
|
435
|
-
children: /* @__PURE__ */ new Map()
|
|
436
|
-
};
|
|
437
|
-
if (!root) root = node;
|
|
438
|
-
if (parent) parent.children.set(info.path, node);
|
|
439
|
-
parent = node;
|
|
440
|
-
}
|
|
441
|
-
return root;
|
|
442
|
-
}
|
|
443
|
-
/**
|
|
444
|
-
* Short-lived cache for hover-triggered prefetches. Entries expire after
|
|
445
|
-
* 30 seconds. When a link is clicked, the prefetched payload is consumed
|
|
446
|
-
* (moved to the history stack) and removed from this cache.
|
|
447
|
-
*
|
|
448
|
-
* timber.js does NOT prefetch on viewport intersection — only explicit
|
|
449
|
-
* hover on <Link prefetch> triggers a prefetch.
|
|
450
|
-
*/
|
|
451
|
-
var PrefetchCache = class PrefetchCache {
|
|
452
|
-
static TTL_MS = 3e4;
|
|
453
|
-
entries = /* @__PURE__ */ new Map();
|
|
454
|
-
set(url, result) {
|
|
455
|
-
this.entries.set(url, {
|
|
456
|
-
result,
|
|
457
|
-
expiresAt: Date.now() + PrefetchCache.TTL_MS
|
|
458
|
-
});
|
|
459
|
-
}
|
|
460
|
-
get(url) {
|
|
461
|
-
const entry = this.entries.get(url);
|
|
462
|
-
if (!entry) return void 0;
|
|
463
|
-
if (Date.now() >= entry.expiresAt) {
|
|
464
|
-
this.entries.delete(url);
|
|
465
|
-
return;
|
|
466
|
-
}
|
|
467
|
-
return entry.result;
|
|
468
|
-
}
|
|
469
|
-
/** Get and remove the entry (used when navigation consumes a prefetch) */
|
|
470
|
-
consume(url) {
|
|
471
|
-
const result = this.get(url);
|
|
472
|
-
if (result !== void 0) this.entries.delete(url);
|
|
473
|
-
return result;
|
|
474
|
-
}
|
|
475
|
-
};
|
|
476
|
-
//#endregion
|
|
477
|
-
//#region src/client/history.ts
|
|
478
|
-
/**
|
|
479
|
-
* Session-lived history stack keyed by URL. Enables instant back/forward
|
|
480
|
-
* navigation without a server roundtrip.
|
|
481
|
-
*
|
|
482
|
-
* On forward navigation, the new page's payload is pushed onto the stack.
|
|
483
|
-
* On popstate, the cached payload is replayed instantly.
|
|
484
|
-
*
|
|
485
|
-
* Scroll positions are stored in history.state (browser History API),
|
|
486
|
-
* not in this stack — see design/19-client-navigation.md §Scroll Restoration.
|
|
487
|
-
*
|
|
488
|
-
* Entries persist for the session duration (no expiry) and are cleared
|
|
489
|
-
* when the tab is closed — matching browser back-button behavior.
|
|
490
|
-
*/
|
|
491
|
-
var HistoryStack = class {
|
|
492
|
-
entries = /* @__PURE__ */ new Map();
|
|
493
|
-
push(url, entry) {
|
|
494
|
-
this.entries.set(url, entry);
|
|
495
|
-
}
|
|
496
|
-
get(url) {
|
|
497
|
-
return this.entries.get(url);
|
|
498
|
-
}
|
|
499
|
-
has(url) {
|
|
500
|
-
return this.entries.has(url);
|
|
501
|
-
}
|
|
502
|
-
};
|
|
503
|
-
//#endregion
|
|
504
|
-
//#region src/client/use-params.ts
|
|
505
|
-
/**
|
|
506
|
-
* Set the current route params in the module-level store.
|
|
507
|
-
*
|
|
508
|
-
* Called by the router on each navigation. This updates the fallback
|
|
509
|
-
* snapshot used by tests and by the hook when called outside a React
|
|
510
|
-
* component (no NavigationContext available).
|
|
511
|
-
*
|
|
512
|
-
* On the client, the primary reactivity path is NavigationContext —
|
|
513
|
-
* the router calls setNavigationState() then renderRoot() which wraps
|
|
514
|
-
* the element in NavigationProvider. setCurrentParams is still called
|
|
515
|
-
* for the module-level fallback.
|
|
516
|
-
*
|
|
517
|
-
* During SSR, params are also available via getSsrData().params
|
|
518
|
-
* (ALS-backed).
|
|
519
|
-
*/
|
|
520
|
-
function setCurrentParams(params) {
|
|
521
|
-
_setCurrentParams(params);
|
|
522
|
-
}
|
|
523
|
-
function useParams(_route) {
|
|
524
|
-
try {
|
|
525
|
-
const navContext = useNavigationContext();
|
|
526
|
-
if (navContext !== null) return navContext.params;
|
|
527
|
-
} catch {}
|
|
528
|
-
return getSsrData()?.params ?? currentParams;
|
|
529
|
-
}
|
|
530
|
-
//#endregion
|
|
531
|
-
//#region src/client/segment-merger.ts
|
|
532
|
-
/**
|
|
533
|
-
* Segment Merger — client-side tree merging for partial RSC payloads.
|
|
534
|
-
*
|
|
535
|
-
* When the server skips rendering sync layouts (because the client already
|
|
536
|
-
* has them cached), the RSC payload is missing outer segment wrappers.
|
|
537
|
-
* This module reconstructs the full element tree by splicing the partial
|
|
538
|
-
* payload into cached segment subtrees.
|
|
539
|
-
*
|
|
540
|
-
* The approach:
|
|
541
|
-
* 1. After each full RSC payload render, walk the decoded element tree
|
|
542
|
-
* and cache each segment's subtree (identified by SegmentProvider boundaries)
|
|
543
|
-
* 2. When a partial payload arrives, wrap it with cached segment elements
|
|
544
|
-
* using React.cloneElement to preserve component identity
|
|
545
|
-
*
|
|
546
|
-
* React.cloneElement preserves the element's `type` — React sees the same
|
|
547
|
-
* component at the same tree position and reconciles (preserving state)
|
|
548
|
-
* rather than remounting. This is how layout state survives navigations.
|
|
549
|
-
*
|
|
550
|
-
* Design docs: 19-client-navigation.md §"Navigation Reconciliation"
|
|
551
|
-
* Security: access.ts runs on the server regardless of skipping — this
|
|
552
|
-
* is a performance optimization only. See 13-security.md.
|
|
553
|
-
*/
|
|
554
|
-
/**
|
|
555
|
-
* Cache of React element subtrees per segment path.
|
|
556
|
-
* Updated after each navigation with the full decoded RSC element tree.
|
|
557
|
-
*/
|
|
558
|
-
var SegmentElementCache = class {
|
|
559
|
-
entries = /* @__PURE__ */ new Map();
|
|
560
|
-
get(segmentPath) {
|
|
561
|
-
return this.entries.get(segmentPath);
|
|
562
|
-
}
|
|
563
|
-
set(segmentPath, entry) {
|
|
564
|
-
this.entries.set(segmentPath, entry);
|
|
565
|
-
}
|
|
566
|
-
has(segmentPath) {
|
|
567
|
-
return this.entries.has(segmentPath);
|
|
568
|
-
}
|
|
569
|
-
clear() {
|
|
570
|
-
this.entries.clear();
|
|
571
|
-
}
|
|
572
|
-
get size() {
|
|
573
|
-
return this.entries.size;
|
|
574
|
-
}
|
|
575
|
-
/**
|
|
576
|
-
* Get the set of segment paths that are safe for the server to skip.
|
|
577
|
-
* Only segments with an inner SegmentProvider (hasMergeableChild) are
|
|
578
|
-
* included — the merger can only replace inner SegmentProviders, not
|
|
579
|
-
* pages embedded in layout output. Used to filter the state tree.
|
|
580
|
-
*
|
|
581
|
-
* Returns an empty set if the element cache is empty (no elements
|
|
582
|
-
* cached yet). This is the safe default — an empty set means no
|
|
583
|
-
* segments pass the filter, so the state tree is empty and the server
|
|
584
|
-
* does a full render. The element cache is populated lazily after the
|
|
585
|
-
* first SPA navigation (RSC-decoded elements from hydration are
|
|
586
|
-
* thenables that can't be walked until React resolves them).
|
|
587
|
-
*/
|
|
588
|
-
getMergeablePaths() {
|
|
589
|
-
const paths = /* @__PURE__ */ new Set();
|
|
590
|
-
for (const [, entry] of this.entries) if (entry.hasMergeableChild) paths.add(entry.segmentPath);
|
|
591
|
-
return paths;
|
|
592
|
-
}
|
|
593
294
|
};
|
|
594
|
-
/**
|
|
595
|
-
* Check if a React element is a SegmentProvider by looking for the
|
|
596
|
-
* `segments` prop (an array of path segments). This is the only
|
|
597
|
-
* component that receives this prop shape.
|
|
598
|
-
*/
|
|
599
|
-
function isSegmentProvider(element) {
|
|
600
|
-
if (!isValidElement(element)) return false;
|
|
601
|
-
const props = element.props;
|
|
602
|
-
return Array.isArray(props.segments);
|
|
603
|
-
}
|
|
604
|
-
/**
|
|
605
|
-
* Extract the segment path from a SegmentProvider element.
|
|
606
|
-
*
|
|
607
|
-
* Uses the `segmentId` prop if available (set by the server for route groups
|
|
608
|
-
* to distinguish siblings that share the same urlPath). Falls back to
|
|
609
|
-
* reconstructing from the `segments` array prop.
|
|
610
|
-
*/
|
|
611
|
-
function getSegmentPath(element) {
|
|
612
|
-
const props = element.props;
|
|
613
|
-
if (props.segmentId) return props.segmentId;
|
|
614
|
-
const filtered = props.segments.filter(Boolean);
|
|
615
|
-
return filtered.length === 0 ? "/" : "/" + filtered.join("/");
|
|
616
|
-
}
|
|
617
|
-
/**
|
|
618
|
-
* Walk a React element tree and extract all SegmentProvider boundaries.
|
|
619
|
-
* Returns an ordered list of segment entries from outermost to innermost.
|
|
620
|
-
*
|
|
621
|
-
* This only finds SegmentProviders along the main children path — it does
|
|
622
|
-
* not descend into parallel routes/slots (those are separate subtrees).
|
|
623
|
-
*/
|
|
624
|
-
function extractSegments(element) {
|
|
625
|
-
const segments = [];
|
|
626
|
-
walkForSegments(element, segments);
|
|
627
|
-
for (let i = 0; i < segments.length; i++) segments[i].hasMergeableChild = i < segments.length - 1;
|
|
628
|
-
return segments;
|
|
629
|
-
}
|
|
630
|
-
function walkForSegments(node, out) {
|
|
631
|
-
if (!isValidElement(node)) return;
|
|
632
|
-
const el = node;
|
|
633
|
-
const props = el.props;
|
|
634
|
-
if (isSegmentProvider(node)) {
|
|
635
|
-
out.push({
|
|
636
|
-
segmentPath: getSegmentPath(el),
|
|
637
|
-
element: el,
|
|
638
|
-
hasMergeableChild: false
|
|
639
|
-
});
|
|
640
|
-
walkChildren(props.children, out);
|
|
641
|
-
return;
|
|
642
|
-
}
|
|
643
|
-
walkChildren(props.children, out);
|
|
644
|
-
}
|
|
645
|
-
function walkChildren(children, out) {
|
|
646
|
-
if (children == null) return;
|
|
647
|
-
if (Array.isArray(children)) for (const child of children) walkForSegments(child, out);
|
|
648
|
-
else walkForSegments(children, out);
|
|
649
|
-
}
|
|
650
|
-
/**
|
|
651
|
-
* Cache all segment subtrees from a fully-rendered RSC element tree.
|
|
652
|
-
* Call this after every full RSC payload render (navigate, refresh, hydration).
|
|
653
|
-
*/
|
|
654
|
-
function cacheSegmentElements(element, cache) {
|
|
655
|
-
const segments = extractSegments(element);
|
|
656
|
-
for (const entry of segments) cache.set(entry.segmentPath, entry);
|
|
657
|
-
}
|
|
658
|
-
function findSegmentProviderPath(node, targetPath) {
|
|
659
|
-
const children = node.props.children;
|
|
660
|
-
if (children == null) return null;
|
|
661
|
-
if (Array.isArray(children)) for (let i = 0; i < children.length; i++) {
|
|
662
|
-
const child = children[i];
|
|
663
|
-
if (!isValidElement(child)) continue;
|
|
664
|
-
if (isSegmentProvider(child)) {
|
|
665
|
-
if (!targetPath || getSegmentPath(child) === targetPath) return [{
|
|
666
|
-
element: node,
|
|
667
|
-
childIndex: i
|
|
668
|
-
}];
|
|
669
|
-
}
|
|
670
|
-
const deeper = findSegmentProviderPath(child, targetPath);
|
|
671
|
-
if (deeper) return [{
|
|
672
|
-
element: node,
|
|
673
|
-
childIndex: i
|
|
674
|
-
}, ...deeper];
|
|
675
|
-
}
|
|
676
|
-
else if (isValidElement(children)) {
|
|
677
|
-
if (isSegmentProvider(children)) {
|
|
678
|
-
if (!targetPath || getSegmentPath(children) === targetPath) return [{
|
|
679
|
-
element: node,
|
|
680
|
-
childIndex: -1
|
|
681
|
-
}];
|
|
682
|
-
}
|
|
683
|
-
const deeper = findSegmentProviderPath(children, targetPath);
|
|
684
|
-
if (deeper) return [{
|
|
685
|
-
element: node,
|
|
686
|
-
childIndex: -1
|
|
687
|
-
}, ...deeper];
|
|
688
|
-
}
|
|
689
|
-
return null;
|
|
690
|
-
}
|
|
691
|
-
/**
|
|
692
|
-
* Replace a nested SegmentProvider within a cached element tree with
|
|
693
|
-
* new content. Uses cloneElement along the path to produce a new tree
|
|
694
|
-
* with preserved component identity at every level except the replaced node.
|
|
695
|
-
*
|
|
696
|
-
* @param cachedElement The cached SegmentProvider element for this segment
|
|
697
|
-
* @param newInnerContent The new React element to splice in at the inner segment position
|
|
698
|
-
* @param innerSegmentPath The path of the inner segment to replace (optional — replaces first found)
|
|
699
|
-
* @returns New element tree with the inner segment replaced
|
|
700
|
-
*/
|
|
701
|
-
function replaceInnerSegment(cachedElement, newInnerContent, innerSegmentPath) {
|
|
702
|
-
const path = findSegmentProviderPath(cachedElement, innerSegmentPath);
|
|
703
|
-
if (!path || path.length === 0) return cachedElement;
|
|
704
|
-
let replacement = newInnerContent;
|
|
705
|
-
for (let i = path.length - 1; i >= 0; i--) {
|
|
706
|
-
const { element, childIndex } = path[i];
|
|
707
|
-
if (childIndex === -1) replacement = cloneElement(element, {}, replacement);
|
|
708
|
-
else {
|
|
709
|
-
const newChildren = [...element.props.children];
|
|
710
|
-
newChildren[childIndex] = replacement;
|
|
711
|
-
replacement = cloneElement(element, {}, ...newChildren);
|
|
712
|
-
}
|
|
713
|
-
}
|
|
714
|
-
return replacement;
|
|
715
|
-
}
|
|
716
|
-
/**
|
|
717
|
-
* Merge a partial RSC payload with cached segment elements.
|
|
718
|
-
*
|
|
719
|
-
* When the server skips segments, the partial payload starts from the
|
|
720
|
-
* first non-skipped segment. This function wraps it with cached elements
|
|
721
|
-
* for the skipped segments, producing a full tree that React can
|
|
722
|
-
* reconcile with the mounted tree (preserving layout state).
|
|
723
|
-
*
|
|
724
|
-
* @param partialPayload The RSC payload element (may be partial)
|
|
725
|
-
* @param skippedSegments Ordered list of segment paths that were skipped (outermost first)
|
|
726
|
-
* @param cache The segment element cache
|
|
727
|
-
* @returns The merged full element tree, or the partial payload if merging isn't possible
|
|
728
|
-
*/
|
|
729
|
-
function mergeSegmentTree(partialPayload, skippedSegments, cache) {
|
|
730
|
-
if (!isValidElement(partialPayload)) return partialPayload;
|
|
731
|
-
if (skippedSegments.length === 0) return partialPayload;
|
|
732
|
-
let result = partialPayload;
|
|
733
|
-
for (let i = skippedSegments.length - 1; i >= 0; i--) {
|
|
734
|
-
const segmentPath = skippedSegments[i];
|
|
735
|
-
const cached = cache.get(segmentPath);
|
|
736
|
-
if (!cached) return partialPayload;
|
|
737
|
-
result = replaceInnerSegment(cached.element, result);
|
|
738
|
-
}
|
|
739
|
-
return result;
|
|
740
|
-
}
|
|
741
295
|
//#endregion
|
|
742
|
-
//#region src/client/
|
|
743
|
-
var RSC_CONTENT_TYPE = "text/x-component";
|
|
744
|
-
/**
|
|
745
|
-
* Generate a short random cache-busting ID (5 chars, a-z0-9).
|
|
746
|
-
* Matches the format Next.js uses for _rsc params.
|
|
747
|
-
*/
|
|
748
|
-
function generateCacheBustId() {
|
|
749
|
-
const chars = "abcdefghijklmnopqrstuvwxyz0123456789";
|
|
750
|
-
let id = "";
|
|
751
|
-
for (let i = 0; i < 5; i++) id += chars[Math.random() * 36 | 0];
|
|
752
|
-
return id;
|
|
753
|
-
}
|
|
754
|
-
/**
|
|
755
|
-
* Append a `_rsc=<id>` query parameter to the URL.
|
|
756
|
-
* Follows Next.js's pattern — prevents CDN/browser from serving cached HTML
|
|
757
|
-
* for RSC navigation requests and signals that this is an RSC fetch.
|
|
758
|
-
*/
|
|
759
|
-
function appendRscParam(url) {
|
|
760
|
-
return `${url}${url.includes("?") ? "&" : "?"}_rsc=${generateCacheBustId()}`;
|
|
761
|
-
}
|
|
762
|
-
function buildRscHeaders(stateTree, currentUrl) {
|
|
763
|
-
const headers = { Accept: RSC_CONTENT_TYPE };
|
|
764
|
-
if (stateTree) headers["X-Timber-State-Tree"] = JSON.stringify(stateTree);
|
|
765
|
-
if (currentUrl) headers["X-Timber-URL"] = currentUrl;
|
|
766
|
-
return headers;
|
|
767
|
-
}
|
|
768
|
-
/**
|
|
769
|
-
* Extract head elements from the X-Timber-Head response header.
|
|
770
|
-
* Returns null if the header is missing or malformed.
|
|
771
|
-
*/
|
|
772
|
-
function extractHeadElements(response) {
|
|
773
|
-
const header = response.headers.get("X-Timber-Head");
|
|
774
|
-
if (!header) return null;
|
|
775
|
-
try {
|
|
776
|
-
return JSON.parse(decodeURIComponent(header));
|
|
777
|
-
} catch {
|
|
778
|
-
return null;
|
|
779
|
-
}
|
|
780
|
-
}
|
|
781
|
-
/**
|
|
782
|
-
* Extract segment metadata from the X-Timber-Segments response header.
|
|
783
|
-
* Returns null if the header is missing or malformed.
|
|
784
|
-
*
|
|
785
|
-
* Format: JSON array of {path, isAsync} objects describing the rendered
|
|
786
|
-
* segment chain from root to leaf. Used to populate the client-side
|
|
787
|
-
* segment cache for state tree diffing on subsequent navigations.
|
|
788
|
-
*/
|
|
789
|
-
function extractSegmentInfo(response) {
|
|
790
|
-
const header = response.headers.get("X-Timber-Segments");
|
|
791
|
-
if (!header) return null;
|
|
792
|
-
try {
|
|
793
|
-
return JSON.parse(header);
|
|
794
|
-
} catch {
|
|
795
|
-
return null;
|
|
796
|
-
}
|
|
797
|
-
}
|
|
798
|
-
/**
|
|
799
|
-
* Extract skipped segment paths from the X-Timber-Skipped-Segments header.
|
|
800
|
-
* Returns null if the header is missing or malformed.
|
|
801
|
-
*
|
|
802
|
-
* When the server skips sync layouts the client already has cached,
|
|
803
|
-
* it sends this header listing the skipped segment paths (outermost first).
|
|
804
|
-
* The client uses this to merge the partial payload with cached segments.
|
|
805
|
-
*/
|
|
806
|
-
function extractSkippedSegments(response) {
|
|
807
|
-
const header = response.headers.get("X-Timber-Skipped-Segments");
|
|
808
|
-
if (!header) return null;
|
|
809
|
-
try {
|
|
810
|
-
const parsed = JSON.parse(header);
|
|
811
|
-
return Array.isArray(parsed) ? parsed : null;
|
|
812
|
-
} catch {
|
|
813
|
-
return null;
|
|
814
|
-
}
|
|
815
|
-
}
|
|
816
|
-
/**
|
|
817
|
-
* Extract route params from the X-Timber-Params response header.
|
|
818
|
-
* Returns null if the header is missing or malformed.
|
|
819
|
-
*
|
|
820
|
-
* Used to populate useParams() after client-side navigation.
|
|
821
|
-
*/
|
|
822
|
-
function extractParams(response) {
|
|
823
|
-
const header = response.headers.get("X-Timber-Params");
|
|
824
|
-
if (!header) return null;
|
|
825
|
-
try {
|
|
826
|
-
return JSON.parse(header);
|
|
827
|
-
} catch {
|
|
828
|
-
return null;
|
|
829
|
-
}
|
|
830
|
-
}
|
|
831
|
-
/**
|
|
832
|
-
* Thrown when an RSC payload response contains X-Timber-Redirect header.
|
|
833
|
-
* Caught in navigate() to trigger a soft router navigation to the redirect target.
|
|
834
|
-
*/
|
|
835
|
-
var RedirectError = class extends Error {
|
|
836
|
-
redirectUrl;
|
|
837
|
-
constructor(url) {
|
|
838
|
-
super(`Server redirect to ${url}`);
|
|
839
|
-
this.redirectUrl = url;
|
|
840
|
-
}
|
|
841
|
-
};
|
|
842
|
-
/**
|
|
843
|
-
* Fetch an RSC payload from the server. If a decodeRsc function is provided,
|
|
844
|
-
* the response is decoded into a React element tree via createFromFetch.
|
|
845
|
-
* Otherwise, the raw response text is returned (test mode).
|
|
846
|
-
*
|
|
847
|
-
* Also extracts head elements from the X-Timber-Head response header
|
|
848
|
-
* so the client can update document.title and <meta> tags after navigation.
|
|
849
|
-
*/
|
|
850
|
-
async function fetchRscPayload(url, deps, stateTree, currentUrl) {
|
|
851
|
-
const rscUrl = appendRscParam(url);
|
|
852
|
-
const headers = buildRscHeaders(stateTree, currentUrl);
|
|
853
|
-
if (deps.decodeRsc) {
|
|
854
|
-
const fetchPromise = deps.fetch(rscUrl, {
|
|
855
|
-
headers,
|
|
856
|
-
redirect: "manual"
|
|
857
|
-
});
|
|
858
|
-
let headElements = null;
|
|
859
|
-
let segmentInfo = null;
|
|
860
|
-
let params = null;
|
|
861
|
-
let skippedSegments = null;
|
|
862
|
-
const wrappedPromise = fetchPromise.then((response) => {
|
|
863
|
-
const redirectLocation = response.headers.get("X-Timber-Redirect") || (response.status >= 300 && response.status < 400 ? response.headers.get("Location") : null);
|
|
864
|
-
if (redirectLocation) throw new RedirectError(redirectLocation);
|
|
865
|
-
headElements = extractHeadElements(response);
|
|
866
|
-
segmentInfo = extractSegmentInfo(response);
|
|
867
|
-
params = extractParams(response);
|
|
868
|
-
skippedSegments = extractSkippedSegments(response);
|
|
869
|
-
return response;
|
|
870
|
-
});
|
|
871
|
-
await wrappedPromise;
|
|
872
|
-
return {
|
|
873
|
-
payload: await deps.decodeRsc(wrappedPromise),
|
|
874
|
-
headElements,
|
|
875
|
-
segmentInfo,
|
|
876
|
-
params,
|
|
877
|
-
skippedSegments
|
|
878
|
-
};
|
|
879
|
-
}
|
|
880
|
-
const response = await deps.fetch(rscUrl, {
|
|
881
|
-
headers,
|
|
882
|
-
redirect: "manual"
|
|
883
|
-
});
|
|
884
|
-
if (response.status >= 300 && response.status < 400) {
|
|
885
|
-
const location = response.headers.get("Location");
|
|
886
|
-
if (location) throw new RedirectError(location);
|
|
887
|
-
}
|
|
888
|
-
return {
|
|
889
|
-
payload: await response.text(),
|
|
890
|
-
headElements: extractHeadElements(response),
|
|
891
|
-
segmentInfo: extractSegmentInfo(response),
|
|
892
|
-
params: extractParams(response),
|
|
893
|
-
skippedSegments: extractSkippedSegments(response)
|
|
894
|
-
};
|
|
895
|
-
}
|
|
896
|
-
//#endregion
|
|
897
|
-
//#region src/client/router.ts
|
|
898
|
-
/**
|
|
899
|
-
* Check if an error is an abort error (connection closed / fetch aborted).
|
|
900
|
-
* Browsers throw DOMException with name 'AbortError' when a fetch is aborted.
|
|
901
|
-
*/
|
|
902
|
-
function isAbortError(error) {
|
|
903
|
-
if (error instanceof DOMException && error.name === "AbortError") return true;
|
|
904
|
-
if (error instanceof Error && error.name === "AbortError") return true;
|
|
905
|
-
return false;
|
|
906
|
-
}
|
|
907
|
-
/**
|
|
908
|
-
* Create a router instance. In production, called once at app hydration
|
|
909
|
-
* with real browser APIs. In tests, called with mock dependencies.
|
|
910
|
-
*/
|
|
911
|
-
function createRouter(deps) {
|
|
912
|
-
const segmentCache = new SegmentCache();
|
|
913
|
-
const prefetchCache = new PrefetchCache();
|
|
914
|
-
const historyStack = new HistoryStack();
|
|
915
|
-
const segmentElementCache = new SegmentElementCache();
|
|
916
|
-
let pending = false;
|
|
917
|
-
let pendingUrl = null;
|
|
918
|
-
const pendingListeners = /* @__PURE__ */ new Set();
|
|
919
|
-
function setPending(value, url) {
|
|
920
|
-
const newPendingUrl = value && url ? url : null;
|
|
921
|
-
if (pending === value && pendingUrl === newPendingUrl) return;
|
|
922
|
-
pending = value;
|
|
923
|
-
pendingUrl = newPendingUrl;
|
|
924
|
-
for (const listener of pendingListeners) listener(value);
|
|
925
|
-
}
|
|
926
|
-
/** Update the segment cache from server-provided segment metadata. */
|
|
927
|
-
function updateSegmentCache(segmentInfo) {
|
|
928
|
-
if (!segmentInfo || segmentInfo.length === 0) return;
|
|
929
|
-
const tree = buildSegmentTree(segmentInfo);
|
|
930
|
-
if (tree) segmentCache.set("/", tree);
|
|
931
|
-
}
|
|
932
|
-
/** Render a decoded RSC payload into the DOM if a renderer is available. */
|
|
933
|
-
function renderPayload(payload) {
|
|
934
|
-
if (deps.renderRoot) deps.renderRoot(payload);
|
|
935
|
-
}
|
|
936
|
-
/**
|
|
937
|
-
* Merge a partial RSC payload with cached segment elements if segments
|
|
938
|
-
* were skipped, then cache segments from the (merged) payload.
|
|
939
|
-
* Returns the merged payload ready for rendering.
|
|
940
|
-
*/
|
|
941
|
-
function mergeAndCachePayload(payload, skippedSegments) {
|
|
942
|
-
let merged = payload;
|
|
943
|
-
if (skippedSegments && skippedSegments.length > 0) merged = mergeSegmentTree(payload, skippedSegments, segmentElementCache);
|
|
944
|
-
cacheSegmentElements(merged, segmentElementCache);
|
|
945
|
-
return merged;
|
|
946
|
-
}
|
|
947
|
-
/**
|
|
948
|
-
* Update navigation state (params + pathname) for the next render.
|
|
949
|
-
*
|
|
950
|
-
* Sets both the module-level fallback (for tests and SSR) and the
|
|
951
|
-
* navigation context state (read by renderRoot to wrap the element
|
|
952
|
-
* in NavigationProvider). The context update is atomic with the tree
|
|
953
|
-
* render — both are passed to reactRoot.render() in the same call.
|
|
954
|
-
*/
|
|
955
|
-
function updateNavigationState(params, url) {
|
|
956
|
-
const resolvedParams = params ?? {};
|
|
957
|
-
setCurrentParams(resolvedParams);
|
|
958
|
-
setNavigationState({
|
|
959
|
-
params: resolvedParams,
|
|
960
|
-
pathname: url.startsWith("http") ? new URL(url).pathname : url.split("?")[0] || "/"
|
|
961
|
-
});
|
|
962
|
-
}
|
|
963
|
-
/**
|
|
964
|
-
* Render a payload via navigateTransition (production) or renderRoot (tests).
|
|
965
|
-
* The perform callback should fetch data, update state, and return the payload.
|
|
966
|
-
* In production, the entire callback runs inside a React transition with
|
|
967
|
-
* useOptimistic for the pending URL. In tests, the payload is rendered directly.
|
|
968
|
-
*/
|
|
969
|
-
async function renderViaTransition(url, perform) {
|
|
970
|
-
if (deps.navigateTransition) {
|
|
971
|
-
let headElements = null;
|
|
972
|
-
await deps.navigateTransition(url, async (wrapPayload) => {
|
|
973
|
-
const result = await perform();
|
|
974
|
-
headElements = result.headElements;
|
|
975
|
-
const merged = mergeAndCachePayload(result.payload, result.skippedSegments);
|
|
976
|
-
historyStack.push(url, {
|
|
977
|
-
payload: merged,
|
|
978
|
-
headElements: result.headElements,
|
|
979
|
-
params: result.params
|
|
980
|
-
});
|
|
981
|
-
return wrapPayload(merged);
|
|
982
|
-
});
|
|
983
|
-
return headElements;
|
|
984
|
-
}
|
|
985
|
-
const result = await perform();
|
|
986
|
-
const merged = mergeAndCachePayload(result.payload, result.skippedSegments);
|
|
987
|
-
historyStack.push(url, {
|
|
988
|
-
payload: merged,
|
|
989
|
-
headElements: result.headElements,
|
|
990
|
-
params: result.params
|
|
991
|
-
});
|
|
992
|
-
renderPayload(merged);
|
|
993
|
-
return result.headElements;
|
|
994
|
-
}
|
|
995
|
-
/** Apply head elements (title, meta tags) to the DOM if available. */
|
|
996
|
-
function applyHead(elements) {
|
|
997
|
-
if (elements && deps.applyHead) deps.applyHead(elements);
|
|
998
|
-
}
|
|
999
|
-
/** Run a callback after the next paint (after React commit). */
|
|
1000
|
-
function afterPaint(callback) {
|
|
1001
|
-
if (deps.afterPaint) deps.afterPaint(callback);
|
|
1002
|
-
else callback();
|
|
1003
|
-
}
|
|
1004
|
-
/**
|
|
1005
|
-
* Core navigation logic shared between the transition and fallback paths.
|
|
1006
|
-
* Fetches the RSC payload, updates all state, and returns the result.
|
|
1007
|
-
*/
|
|
1008
|
-
async function performNavigationFetch(url, options) {
|
|
1009
|
-
const prefetched = prefetchCache.consume(url);
|
|
1010
|
-
let result = prefetched ? {
|
|
1011
|
-
payload: prefetched.payload,
|
|
1012
|
-
headElements: prefetched.headElements,
|
|
1013
|
-
segmentInfo: prefetched.segmentInfo ?? null,
|
|
1014
|
-
params: prefetched.params ?? null,
|
|
1015
|
-
skippedSegments: prefetched.skippedSegments ?? null
|
|
1016
|
-
} : void 0;
|
|
1017
|
-
if (result === void 0) {
|
|
1018
|
-
const stateTree = segmentCache.serializeStateTree(segmentElementCache.getMergeablePaths());
|
|
1019
|
-
const rawCurrentUrl = deps.getCurrentUrl();
|
|
1020
|
-
result = await fetchRscPayload(url, deps, stateTree, rawCurrentUrl.startsWith("http") ? new URL(rawCurrentUrl).pathname : new URL(rawCurrentUrl, "http://localhost").pathname);
|
|
1021
|
-
}
|
|
1022
|
-
if (options.replace) deps.replaceState({
|
|
1023
|
-
timber: true,
|
|
1024
|
-
scrollY: 0
|
|
1025
|
-
}, "", url);
|
|
1026
|
-
else deps.pushState({
|
|
1027
|
-
timber: true,
|
|
1028
|
-
scrollY: 0
|
|
1029
|
-
}, "", url);
|
|
1030
|
-
updateSegmentCache(result.segmentInfo);
|
|
1031
|
-
updateNavigationState(result.params, url);
|
|
1032
|
-
return result;
|
|
1033
|
-
}
|
|
1034
|
-
async function navigate(url, options = {}) {
|
|
1035
|
-
const scroll = options.scroll !== false;
|
|
1036
|
-
const replace = options.replace === true;
|
|
1037
|
-
const currentScrollY = deps.getScrollY();
|
|
1038
|
-
deps.replaceState({
|
|
1039
|
-
timber: true,
|
|
1040
|
-
scrollY: currentScrollY
|
|
1041
|
-
}, "", deps.getCurrentUrl());
|
|
1042
|
-
setPending(true, url);
|
|
1043
|
-
try {
|
|
1044
|
-
applyHead(await renderViaTransition(url, () => performNavigationFetch(url, { replace })));
|
|
1045
|
-
window.dispatchEvent(new Event("timber:navigation-end"));
|
|
1046
|
-
afterPaint(() => {
|
|
1047
|
-
if (scroll) deps.scrollTo(0, 0);
|
|
1048
|
-
else deps.scrollTo(0, currentScrollY);
|
|
1049
|
-
window.dispatchEvent(new Event("timber:scroll-restored"));
|
|
1050
|
-
});
|
|
1051
|
-
} catch (error) {
|
|
1052
|
-
if (error instanceof RedirectError) {
|
|
1053
|
-
setPending(false);
|
|
1054
|
-
await navigate(error.redirectUrl, { replace: true });
|
|
1055
|
-
return;
|
|
1056
|
-
}
|
|
1057
|
-
if (isAbortError(error)) return;
|
|
1058
|
-
throw error;
|
|
1059
|
-
} finally {
|
|
1060
|
-
setPending(false);
|
|
1061
|
-
}
|
|
1062
|
-
}
|
|
1063
|
-
async function refresh() {
|
|
1064
|
-
const currentUrl = deps.getCurrentUrl();
|
|
1065
|
-
setPending(true, currentUrl);
|
|
1066
|
-
try {
|
|
1067
|
-
applyHead(await renderViaTransition(currentUrl, async () => {
|
|
1068
|
-
const result = await fetchRscPayload(currentUrl, deps);
|
|
1069
|
-
updateSegmentCache(result.segmentInfo);
|
|
1070
|
-
updateNavigationState(result.params, currentUrl);
|
|
1071
|
-
return result;
|
|
1072
|
-
}));
|
|
1073
|
-
} finally {
|
|
1074
|
-
setPending(false);
|
|
1075
|
-
}
|
|
1076
|
-
}
|
|
1077
|
-
async function handlePopState(url, scrollY = 0) {
|
|
1078
|
-
const entry = historyStack.get(url);
|
|
1079
|
-
if (entry && entry.payload !== null) {
|
|
1080
|
-
updateNavigationState(entry.params, url);
|
|
1081
|
-
renderPayload(entry.payload);
|
|
1082
|
-
applyHead(entry.headElements);
|
|
1083
|
-
afterPaint(() => {
|
|
1084
|
-
deps.scrollTo(0, scrollY);
|
|
1085
|
-
window.dispatchEvent(new Event("timber:scroll-restored"));
|
|
1086
|
-
});
|
|
1087
|
-
} else {
|
|
1088
|
-
setPending(true, url);
|
|
1089
|
-
try {
|
|
1090
|
-
applyHead(await renderViaTransition(url, async () => {
|
|
1091
|
-
const result = await fetchRscPayload(url, deps, segmentCache.serializeStateTree(segmentElementCache.getMergeablePaths()));
|
|
1092
|
-
updateSegmentCache(result.segmentInfo);
|
|
1093
|
-
updateNavigationState(result.params, url);
|
|
1094
|
-
return result;
|
|
1095
|
-
}));
|
|
1096
|
-
afterPaint(() => {
|
|
1097
|
-
deps.scrollTo(0, scrollY);
|
|
1098
|
-
window.dispatchEvent(new Event("timber:scroll-restored"));
|
|
1099
|
-
});
|
|
1100
|
-
} finally {
|
|
1101
|
-
setPending(false);
|
|
1102
|
-
}
|
|
1103
|
-
}
|
|
1104
|
-
}
|
|
1105
|
-
/**
|
|
1106
|
-
* Prefetch an RSC payload for a URL and store it in the prefetch cache.
|
|
1107
|
-
* Called on hover of <Link prefetch> elements.
|
|
1108
|
-
*/
|
|
1109
|
-
function prefetch(url) {
|
|
1110
|
-
if (prefetchCache.get(url) !== void 0) return;
|
|
1111
|
-
if (historyStack.has(url)) return;
|
|
1112
|
-
fetchRscPayload(url, deps, segmentCache.serializeStateTree(segmentElementCache.getMergeablePaths())).then((result) => {
|
|
1113
|
-
prefetchCache.set(url, result);
|
|
1114
|
-
}, () => {});
|
|
1115
|
-
}
|
|
1116
|
-
return {
|
|
1117
|
-
navigate,
|
|
1118
|
-
refresh,
|
|
1119
|
-
handlePopState,
|
|
1120
|
-
isPending: () => pending,
|
|
1121
|
-
getPendingUrl: () => pendingUrl,
|
|
1122
|
-
onPendingChange(listener) {
|
|
1123
|
-
pendingListeners.add(listener);
|
|
1124
|
-
return () => pendingListeners.delete(listener);
|
|
1125
|
-
},
|
|
1126
|
-
prefetch,
|
|
1127
|
-
applyRevalidation(element, headElements) {
|
|
1128
|
-
const currentUrl = deps.getCurrentUrl();
|
|
1129
|
-
const merged = mergeAndCachePayload(element, null);
|
|
1130
|
-
historyStack.push(currentUrl, {
|
|
1131
|
-
payload: merged,
|
|
1132
|
-
headElements
|
|
1133
|
-
});
|
|
1134
|
-
renderPayload(merged);
|
|
1135
|
-
applyHead(headElements);
|
|
1136
|
-
},
|
|
1137
|
-
initSegmentCache: (segments) => updateSegmentCache(segments),
|
|
1138
|
-
cacheElementTree: (element) => cacheSegmentElements(element, segmentElementCache),
|
|
1139
|
-
segmentCache,
|
|
1140
|
-
prefetchCache,
|
|
1141
|
-
historyStack
|
|
1142
|
-
};
|
|
1143
|
-
}
|
|
1144
|
-
//#endregion
|
|
1145
|
-
//#region src/client/use-navigation-pending.ts
|
|
296
|
+
//#region src/client/use-pending-navigation.ts
|
|
1146
297
|
/**
|
|
1147
298
|
* Returns true while an RSC navigation is in flight.
|
|
1148
299
|
*
|
|
@@ -1155,10 +306,10 @@ function createRouter(deps) {
|
|
|
1155
306
|
*
|
|
1156
307
|
* ```tsx
|
|
1157
308
|
* 'use client'
|
|
1158
|
-
* import {
|
|
309
|
+
* import { usePendingNavigation } from '@timber-js/app/client'
|
|
1159
310
|
*
|
|
1160
311
|
* export function NavBar() {
|
|
1161
|
-
* const isPending =
|
|
312
|
+
* const isPending = usePendingNavigation()
|
|
1162
313
|
* return (
|
|
1163
314
|
* <nav className={isPending ? 'opacity-50' : ''}>
|
|
1164
315
|
* <Link href="/dashboard">Dashboard</Link>
|
|
@@ -1167,7 +318,7 @@ function createRouter(deps) {
|
|
|
1167
318
|
* }
|
|
1168
319
|
* ```
|
|
1169
320
|
*/
|
|
1170
|
-
function
|
|
321
|
+
function usePendingNavigation() {
|
|
1171
322
|
return usePendingNavigationUrl() !== null;
|
|
1172
323
|
}
|
|
1173
324
|
//#endregion
|
|
@@ -1194,7 +345,7 @@ function useNavigationPending() {
|
|
|
1194
345
|
*
|
|
1195
346
|
* For loading UI during navigation, use:
|
|
1196
347
|
* - useLinkStatus() — per-link pending indicator (inside <Link>)
|
|
1197
|
-
* -
|
|
348
|
+
* - usePendingNavigation() — global navigation pending state
|
|
1198
349
|
*/
|
|
1199
350
|
/**
|
|
1200
351
|
* Get a router instance for programmatic navigation.
|
|
@@ -1354,36 +505,6 @@ function useSearchParams() {
|
|
|
1354
505
|
return typeof window !== "undefined" ? getSearchParams() : getServerSearchParams();
|
|
1355
506
|
}
|
|
1356
507
|
//#endregion
|
|
1357
|
-
//#region src/client/segment-context.ts
|
|
1358
|
-
/**
|
|
1359
|
-
* Segment Context — provides layout segment position for useSelectedLayoutSegment hooks.
|
|
1360
|
-
*
|
|
1361
|
-
* Each layout in the segment tree is wrapped with a SegmentProvider that stores
|
|
1362
|
-
* the URL segments from root to the current layout level. The hooks read this
|
|
1363
|
-
* context to determine which child segments are active below the calling layout.
|
|
1364
|
-
*
|
|
1365
|
-
* The context value is intentionally minimal: just the segment path array and
|
|
1366
|
-
* parallel route keys. No internal cache details are exposed.
|
|
1367
|
-
*
|
|
1368
|
-
* Design docs: design/19-client-navigation.md, design/14-ecosystem.md
|
|
1369
|
-
*/
|
|
1370
|
-
var SegmentContext = createContext(null);
|
|
1371
|
-
/** Read the segment context. Returns null if no provider is above this component. */
|
|
1372
|
-
function useSegmentContext() {
|
|
1373
|
-
return useContext(SegmentContext);
|
|
1374
|
-
}
|
|
1375
|
-
/**
|
|
1376
|
-
* Wraps each layout to provide segment position context.
|
|
1377
|
-
* Injected by rsc-entry.ts during element tree construction.
|
|
1378
|
-
*/
|
|
1379
|
-
function SegmentProvider({ segments, segmentId: _segmentId, parallelRouteKeys, children }) {
|
|
1380
|
-
const value = useMemo(() => ({
|
|
1381
|
-
segments,
|
|
1382
|
-
parallelRouteKeys
|
|
1383
|
-
}), [segments.join("/"), parallelRouteKeys.join(",")]);
|
|
1384
|
-
return createElement(SegmentContext.Provider, { value }, children);
|
|
1385
|
-
}
|
|
1386
|
-
//#endregion
|
|
1387
508
|
//#region src/client/use-selected-layout-segment.ts
|
|
1388
509
|
/**
|
|
1389
510
|
* useSelectedLayoutSegment / useSelectedLayoutSegments — client-side hooks
|
|
@@ -1589,6 +710,96 @@ function useFormErrors(result) {
|
|
|
1589
710
|
};
|
|
1590
711
|
}
|
|
1591
712
|
//#endregion
|
|
1592
|
-
|
|
713
|
+
//#region src/client/use-cookie.ts
|
|
714
|
+
/**
|
|
715
|
+
* useCookie — reactive client-side cookie hook.
|
|
716
|
+
*
|
|
717
|
+
* Uses useSyncExternalStore for SSR-safe, reactive cookie access.
|
|
718
|
+
* All components reading the same cookie name re-render on change.
|
|
719
|
+
* No cross-tab sync (intentional — see design/29-cookies.md).
|
|
720
|
+
*
|
|
721
|
+
* See design/29-cookies.md §"useCookie(name) Hook"
|
|
722
|
+
*/
|
|
723
|
+
var use_cookie_exports = /* @__PURE__ */ __exportAll({ useCookie: () => useCookie });
|
|
724
|
+
/** Per-name subscriber sets. */
|
|
725
|
+
var listeners = /* @__PURE__ */ new Map();
|
|
726
|
+
/** Parse a cookie name from document.cookie. */
|
|
727
|
+
function getCookieValue(name) {
|
|
728
|
+
if (typeof document === "undefined") return void 0;
|
|
729
|
+
const match = document.cookie.match(new RegExp("(?:^|;\\s*)" + name.replace(/[.*+?^${}()|[\]\\]/g, "\\$&") + "\\s*=\\s*([^;]*)"));
|
|
730
|
+
return match ? decodeURIComponent(match[1]) : void 0;
|
|
731
|
+
}
|
|
732
|
+
/** Serialize options into a cookie string suffix. */
|
|
733
|
+
function serializeOptions(options) {
|
|
734
|
+
if (!options) return "; Path=/; SameSite=Lax";
|
|
735
|
+
const parts = [];
|
|
736
|
+
parts.push(`Path=${options.path ?? "/"}`);
|
|
737
|
+
if (options.domain) parts.push(`Domain=${options.domain}`);
|
|
738
|
+
if (options.maxAge !== void 0) parts.push(`Max-Age=${options.maxAge}`);
|
|
739
|
+
if (options.expires) parts.push(`Expires=${options.expires.toUTCString()}`);
|
|
740
|
+
const sameSite = options.sameSite ?? "lax";
|
|
741
|
+
parts.push(`SameSite=${sameSite.charAt(0).toUpperCase()}${sameSite.slice(1)}`);
|
|
742
|
+
if (options.secure) parts.push("Secure");
|
|
743
|
+
return "; " + parts.join("; ");
|
|
744
|
+
}
|
|
745
|
+
/** Notify all subscribers for a given cookie name. */
|
|
746
|
+
function notify(name) {
|
|
747
|
+
const subs = listeners.get(name);
|
|
748
|
+
if (subs) for (const fn of subs) fn();
|
|
749
|
+
}
|
|
750
|
+
/**
|
|
751
|
+
* Reactive hook for reading/writing a client-side cookie.
|
|
752
|
+
*
|
|
753
|
+
* Returns `[value, setCookie, deleteCookie]`:
|
|
754
|
+
* - `value`: current cookie value (string | undefined)
|
|
755
|
+
* - `setCookie`: sets the cookie and triggers re-renders
|
|
756
|
+
* - `deleteCookie`: deletes the cookie and triggers re-renders
|
|
757
|
+
*
|
|
758
|
+
* @param name - Cookie name.
|
|
759
|
+
* @param defaultOptions - Default options for setCookie calls.
|
|
760
|
+
*/
|
|
761
|
+
function useCookie(name, defaultOptions) {
|
|
762
|
+
const subscribe = (callback) => {
|
|
763
|
+
let subs = listeners.get(name);
|
|
764
|
+
if (!subs) {
|
|
765
|
+
subs = /* @__PURE__ */ new Set();
|
|
766
|
+
listeners.set(name, subs);
|
|
767
|
+
}
|
|
768
|
+
subs.add(callback);
|
|
769
|
+
return () => {
|
|
770
|
+
subs.delete(callback);
|
|
771
|
+
if (subs.size === 0) listeners.delete(name);
|
|
772
|
+
};
|
|
773
|
+
};
|
|
774
|
+
const getSnapshot = () => getCookieValue(name);
|
|
775
|
+
const getServerSnapshot = () => getSsrData()?.cookies.get(name);
|
|
776
|
+
const value = useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot);
|
|
777
|
+
const setCookie = (newValue, options) => {
|
|
778
|
+
const merged = {
|
|
779
|
+
...defaultOptions,
|
|
780
|
+
...options
|
|
781
|
+
};
|
|
782
|
+
document.cookie = `${name}=${encodeURIComponent(newValue)}${serializeOptions(merged)}`;
|
|
783
|
+
notify(name);
|
|
784
|
+
};
|
|
785
|
+
const deleteCookie = () => {
|
|
786
|
+
const path = defaultOptions?.path ?? "/";
|
|
787
|
+
const domain = defaultOptions?.domain;
|
|
788
|
+
let cookieStr = `${name}=; Max-Age=0; Expires=Thu, 01 Jan 1970 00:00:00 GMT; Path=${path}`;
|
|
789
|
+
if (domain) cookieStr += `; Domain=${domain}`;
|
|
790
|
+
document.cookie = cookieStr;
|
|
791
|
+
notify(name);
|
|
792
|
+
};
|
|
793
|
+
return [
|
|
794
|
+
value,
|
|
795
|
+
setCookie,
|
|
796
|
+
deleteCookie
|
|
797
|
+
];
|
|
798
|
+
}
|
|
799
|
+
//#endregion
|
|
800
|
+
//#region src/client/index.ts
|
|
801
|
+
_registerUseCookieModule(use_cookie_exports);
|
|
802
|
+
//#endregion
|
|
803
|
+
export { Link, LinkStatusContext, buildLinkProps, interpolateParams, mergePreservedSearchParams, resolveHref, useActionState, useCookie, useFormAction, useFormErrors, useLinkStatus, usePathname, usePendingNavigation, useQueryStates, useRouter, useSearchParams, useSegmentParams, useSelectedLayoutSegment, useSelectedLayoutSegments, validateLinkHref };
|
|
1593
804
|
|
|
1594
805
|
//# sourceMappingURL=index.js.map
|