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