@rangojs/router 0.0.0-experimental.9 → 0.0.0-experimental.a5f27bd5
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/AGENTS.md +5 -0
- package/README.md +884 -4
- package/dist/bin/rango.js +1531 -155
- package/dist/vite/index.js +4440 -2170
- package/package.json +60 -54
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +50 -21
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +18 -16
- package/skills/fonts/SKILL.md +6 -4
- package/skills/hooks/SKILL.md +333 -71
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +131 -8
- package/skills/layout/SKILL.md +100 -3
- package/skills/links/SKILL.md +74 -15
- package/skills/loader/SKILL.md +388 -38
- package/skills/middleware/SKILL.md +171 -34
- package/skills/mime-routes/SKILL.md +15 -11
- package/skills/parallel/SKILL.md +78 -1
- package/skills/prerender/SKILL.md +405 -45
- package/skills/rango/SKILL.md +85 -21
- package/skills/response-routes/SKILL.md +144 -91
- package/skills/route/SKILL.md +226 -14
- package/skills/router-setup/SKILL.md +123 -30
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +316 -87
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +312 -15
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/event-controller.ts +87 -64
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +285 -553
- package/src/browser/navigation-client.ts +123 -73
- package/src/browser/navigation-store.ts +33 -50
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +261 -309
- package/src/browser/prefetch/cache.ts +154 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +182 -70
- package/src/browser/react/NavigationProvider.tsx +51 -11
- package/src/browser/react/context.ts +6 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +12 -12
- package/src/browser/react/location-state-shared.ts +95 -53
- package/src/browser/react/location-state.ts +60 -15
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +29 -51
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +29 -70
- package/src/browser/react/use-link-status.ts +6 -5
- package/src/browser/react/use-navigation.ts +22 -63
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +80 -97
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +106 -27
- package/src/browser/scroll-restoration.ts +92 -16
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +504 -599
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +107 -47
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +82 -21
- package/src/build/generate-route-types.ts +36 -752
- package/src/build/index.ts +6 -5
- package/src/build/route-trie.ts +39 -13
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +411 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +469 -0
- package/src/build/route-types/scan-filter.ts +78 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +338 -0
- package/src/cache/cache-scope.ts +120 -301
- package/src/cache/cf/cf-cache-store.ts +119 -7
- package/src/cache/cf/index.ts +8 -2
- package/src/cache/document-cache.ts +101 -72
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +0 -15
- package/src/cache/memory-segment-store.ts +191 -13
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +98 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +84 -126
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +86 -0
- package/src/debug.ts +17 -7
- package/src/errors.ts +77 -7
- package/src/handle.ts +15 -10
- package/src/handles/MetaTags.tsx +73 -20
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/handles/meta.ts +30 -13
- package/src/host/cookie-handler.ts +21 -15
- package/src/host/errors.ts +8 -8
- package/src/host/index.ts +4 -7
- package/src/host/pattern-matcher.ts +27 -27
- package/src/host/router.ts +61 -39
- package/src/host/testing.ts +8 -8
- package/src/host/types.ts +15 -7
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +65 -45
- package/src/index.rsc.ts +133 -21
- package/src/index.ts +164 -52
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +25 -143
- package/src/loader.ts +27 -10
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +4 -2
- package/src/prerender/store.ts +158 -13
- package/src/prerender.ts +333 -26
- package/src/reverse.ts +184 -121
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -1431
- package/src/route-map-builder.ts +156 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +48 -9
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +374 -81
- package/src/router/intercept-resolution.ts +24 -16
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +215 -122
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +83 -32
- package/src/router/match-api.ts +118 -119
- package/src/router/match-context.ts +4 -2
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +80 -93
- package/src/router/match-middleware/cache-lookup.ts +336 -84
- package/src/router/match-middleware/cache-store.ts +43 -24
- package/src/router/match-middleware/intercept-resolution.ts +45 -20
- package/src/router/match-middleware/segment-resolution.ts +16 -8
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +34 -28
- package/src/router/metrics.ts +235 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +324 -367
- package/src/router/pattern-matching.ts +197 -41
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/router-context.ts +36 -21
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1239 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -1315
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +96 -29
- package/src/router/types.ts +16 -9
- package/src/router.ts +590 -1983
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +661 -1015
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +237 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +38 -11
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +25 -13
- package/src/server/context.ts +173 -48
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +94 -15
- package/src/server/loader-registry.ts +15 -56
- package/src/server/request-context.ts +430 -70
- package/src/server.ts +35 -155
- package/src/ssr/index.tsx +100 -31
- package/src/static-handler.ts +114 -0
- package/src/theme/ThemeProvider.tsx +21 -15
- package/src/theme/ThemeScript.tsx +5 -5
- package/src/theme/constants.ts +5 -2
- package/src/theme/index.ts +4 -14
- package/src/theme/theme-context.ts +4 -30
- package/src/theme/theme-script.ts +21 -18
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -1757
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -1282
- package/src/use-loader.tsx +85 -77
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +11 -1963
- package/src/vite/plugin-types.ts +131 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +254 -0
- package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +510 -0
- package/src/vite/router-discovery.ts +785 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
- package/src/vite/utils/prerender-utils.ts +189 -0
- package/src/vite/utils/shared-utils.ts +169 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/cache/memory-store.ts +0 -253
- package/src/href-context.ts +0 -33
- package/src/router.gen.ts +0 -6
- package/src/urls.gen.ts +0 -8
- package/src/vite/expose-handle-id.ts +0 -209
- package/src/vite/expose-loader-id.ts +0 -426
- package/src/vite/expose-location-state-id.ts +0 -177
- package/src/vite/expose-prerender-handler-id.ts +0 -429
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
package/src/reverse.ts
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
1
|
import type { ExtractParams } from "./types.js";
|
|
2
|
+
import type { SearchSchema, ResolveSearchSchema } from "./search-params.js";
|
|
3
|
+
import { serializeSearchParams } from "./search-params.js";
|
|
2
4
|
|
|
3
5
|
/**
|
|
4
6
|
* Sanitize prefix string by removing leading slash
|
|
@@ -8,98 +10,55 @@ export type SanitizePrefix<T extends string> = T extends `/${infer P}` ? P : T;
|
|
|
8
10
|
|
|
9
11
|
/**
|
|
10
12
|
* Helper type to merge multiple route definitions into a single accumulated type.
|
|
11
|
-
* Note: When using createRouter, types accumulate automatically through the
|
|
12
|
-
* builder chain, so this type is typically not needed.
|
|
13
13
|
*
|
|
14
14
|
* @example
|
|
15
15
|
* ```typescript
|
|
16
|
-
*
|
|
17
|
-
* type AppRoutes = MergeRoutes<[
|
|
18
|
-
* typeof homeRoutes,
|
|
19
|
-
* PrefixRoutePatterns<typeof blogRoutes, "/blog">,
|
|
20
|
-
* ]>;
|
|
21
|
-
*
|
|
22
|
-
* // Preferred: Let router accumulate types automatically
|
|
23
|
-
* const router = createRouter<AppEnv>()
|
|
24
|
-
* .routes(homeRoutes).map(...)
|
|
25
|
-
* .routes("/blog", blogRoutes).map(...);
|
|
26
|
-
* type AppRoutes = typeof router.routeMap;
|
|
16
|
+
* type AppRoutes = MergeRoutes<[typeof siteRoutes, typeof apiRoutes]>;
|
|
27
17
|
* ```
|
|
28
18
|
*/
|
|
29
19
|
export type MergeRoutes<T extends unknown[]> = T extends [
|
|
30
20
|
infer First,
|
|
31
|
-
...infer Rest
|
|
21
|
+
...infer Rest,
|
|
32
22
|
]
|
|
33
23
|
? First & MergeRoutes<Rest>
|
|
34
24
|
: {};
|
|
35
25
|
|
|
36
|
-
/**
|
|
37
|
-
* Add key prefix to all entries in a route map
|
|
38
|
-
* { "cart": "/cart" } with prefix "shop" -> { "shop.cart": "/shop/cart" }
|
|
39
|
-
*/
|
|
40
|
-
export type PrefixRouteKeys<
|
|
41
|
-
T,
|
|
42
|
-
Prefix extends string
|
|
43
|
-
> = Prefix extends ""
|
|
44
|
-
? T
|
|
45
|
-
: { [K in keyof T as `${Prefix}.${K & string}`]: T[K] };
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Add path prefix to all patterns in a route map
|
|
49
|
-
* { "cart": "/cart" } with prefix "/shop" -> { "cart": "/shop/cart" }
|
|
50
|
-
*/
|
|
51
|
-
export type PrefixRoutePatterns<
|
|
52
|
-
T,
|
|
53
|
-
PathPrefix extends string
|
|
54
|
-
> = {
|
|
55
|
-
[K in keyof T]: PathPrefix extends "" | "/"
|
|
56
|
-
? T[K]
|
|
57
|
-
: T[K] extends "/"
|
|
58
|
-
? PathPrefix
|
|
59
|
-
: T[K] extends string
|
|
60
|
-
? `${PathPrefix}${T[K]}`
|
|
61
|
-
: T[K];
|
|
62
|
-
};
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* Combined: prefix both keys and patterns
|
|
66
|
-
* Used for module augmentation registration
|
|
67
|
-
*
|
|
68
|
-
* @example
|
|
69
|
-
* ```typescript
|
|
70
|
-
* // Given shopRoutes = { "index": "/", "cart": "/cart", "products.detail": "/product/:slug" }
|
|
71
|
-
* // PrefixedRoutes<typeof shopRoutes, "shop"> produces:
|
|
72
|
-
* // { "shop.index": "/shop", "shop.cart": "/shop/cart", "shop.products.detail": "/shop/product/:slug" }
|
|
73
|
-
* ```
|
|
74
|
-
*/
|
|
75
|
-
export type PrefixedRoutes<
|
|
76
|
-
T,
|
|
77
|
-
KeyPrefix extends string,
|
|
78
|
-
PathPrefix extends string = KeyPrefix extends "" ? "" : `/${KeyPrefix}`
|
|
79
|
-
> = PrefixRouteKeys<PrefixRoutePatterns<T, PathPrefix>, KeyPrefix>;
|
|
80
|
-
|
|
81
26
|
/**
|
|
82
27
|
* Helper to safely extract route patterns from a routes object
|
|
83
28
|
* Handles string values, { path, response } objects, and interface types (like RegisteredRoutes)
|
|
84
29
|
*/
|
|
85
|
-
type RoutePatternFor<
|
|
86
|
-
TRoutes
|
|
87
|
-
|
|
88
|
-
|
|
30
|
+
type RoutePatternFor<
|
|
31
|
+
TRoutes,
|
|
32
|
+
TName extends keyof TRoutes,
|
|
33
|
+
> = TRoutes[TName] extends string
|
|
34
|
+
? TRoutes[TName]
|
|
35
|
+
: TRoutes[TName] extends { readonly path: infer P extends string }
|
|
36
|
+
? P
|
|
37
|
+
: string;
|
|
89
38
|
|
|
90
39
|
/**
|
|
91
40
|
* Extract params type for a route
|
|
92
41
|
*/
|
|
93
|
-
export type ParamsFor<
|
|
94
|
-
TRoutes,
|
|
95
|
-
|
|
96
|
-
> = ExtractParams<RoutePatternFor<TRoutes, TName>>;
|
|
42
|
+
export type ParamsFor<TRoutes, TName extends keyof TRoutes> = ExtractParams<
|
|
43
|
+
RoutePatternFor<TRoutes, TName>
|
|
44
|
+
>;
|
|
97
45
|
|
|
98
46
|
/**
|
|
99
47
|
* Check if an object type has any keys
|
|
100
48
|
*/
|
|
101
49
|
type IsEmptyObject<T> = keyof T extends never ? true : false;
|
|
102
50
|
|
|
51
|
+
/**
|
|
52
|
+
* Extract search schema from a route entry.
|
|
53
|
+
* Returns {} if no search schema is defined.
|
|
54
|
+
*/
|
|
55
|
+
type ExtractSearchSchema<
|
|
56
|
+
TRoutes,
|
|
57
|
+
TName extends keyof TRoutes,
|
|
58
|
+
> = TRoutes[TName] extends { readonly search: infer S extends SearchSchema }
|
|
59
|
+
? S
|
|
60
|
+
: {};
|
|
61
|
+
|
|
103
62
|
/**
|
|
104
63
|
* Type-safe reverse function signature (Django-style URL reversal)
|
|
105
64
|
*
|
|
@@ -117,7 +76,11 @@ export type ReverseFunction<TRoutes> = {
|
|
|
117
76
|
* Route without params - validates route name exists
|
|
118
77
|
*/
|
|
119
78
|
<TName extends keyof TRoutes & string>(
|
|
120
|
-
name: IsEmptyObject<
|
|
79
|
+
name: IsEmptyObject<
|
|
80
|
+
ExtractParams<RoutePatternFor<TRoutes, TName>>
|
|
81
|
+
> extends true
|
|
82
|
+
? TName
|
|
83
|
+
: never,
|
|
121
84
|
): string;
|
|
122
85
|
|
|
123
86
|
/**
|
|
@@ -125,69 +88,135 @@ export type ReverseFunction<TRoutes> = {
|
|
|
125
88
|
*/
|
|
126
89
|
<TName extends keyof TRoutes & string>(
|
|
127
90
|
name: TName,
|
|
128
|
-
params: ExtractParams<RoutePatternFor<TRoutes, TName
|
|
91
|
+
params: ExtractParams<RoutePatternFor<TRoutes, TName>>,
|
|
92
|
+
): string;
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* Route with params and search - validates route name, params, and search
|
|
96
|
+
*/
|
|
97
|
+
<TName extends keyof TRoutes & string>(
|
|
98
|
+
name: TName,
|
|
99
|
+
params: ExtractParams<RoutePatternFor<TRoutes, TName>>,
|
|
100
|
+
search: ResolveSearchSchema<ExtractSearchSchema<TRoutes, TName>>,
|
|
101
|
+
): string;
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* Dot-prefixed route without params - strictly local resolution
|
|
105
|
+
*/
|
|
106
|
+
<TName extends keyof TRoutes & string>(
|
|
107
|
+
name: IsEmptyObject<
|
|
108
|
+
ExtractParams<RoutePatternFor<TRoutes, TName>>
|
|
109
|
+
> extends true
|
|
110
|
+
? `.${TName}`
|
|
111
|
+
: never,
|
|
112
|
+
): string;
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Dot-prefixed route with params - strictly local resolution
|
|
116
|
+
*/
|
|
117
|
+
<TName extends keyof TRoutes & string>(
|
|
118
|
+
name: `.${TName}`,
|
|
119
|
+
params: ExtractParams<RoutePatternFor<TRoutes, TName>>,
|
|
120
|
+
): string;
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* Dot-prefixed route with params and search - strictly local resolution
|
|
124
|
+
*/
|
|
125
|
+
<TName extends keyof TRoutes & string>(
|
|
126
|
+
name: `.${TName}`,
|
|
127
|
+
params: ExtractParams<RoutePatternFor<TRoutes, TName>>,
|
|
128
|
+
search: ResolveSearchSchema<ExtractSearchSchema<TRoutes, TName>>,
|
|
129
129
|
): string;
|
|
130
130
|
};
|
|
131
131
|
|
|
132
132
|
/**
|
|
133
|
-
* Type-safe scoped reverse function
|
|
133
|
+
* Type-safe scoped reverse function with separate local and global namespaces.
|
|
134
134
|
*
|
|
135
|
-
*
|
|
136
|
-
*
|
|
137
|
-
* Path-based URLs (`/...`) are an escape hatch with no validation.
|
|
135
|
+
* - `.name` — local resolution within the current include() scope
|
|
136
|
+
* - `name` — global resolution against the named-routes definition
|
|
138
137
|
*
|
|
139
138
|
* @example
|
|
140
139
|
* ```typescript
|
|
141
|
-
*
|
|
142
|
-
* reverse("
|
|
143
|
-
* reverse("
|
|
144
|
-
*
|
|
145
|
-
* //
|
|
146
|
-
* reverse("
|
|
147
|
-
* reverse("/typo/in/path") // ⚠ Won't catch errors
|
|
140
|
+
* reverse(".article", { slug: "hello" }) // ✓ Local route (resolves with mount prefix)
|
|
141
|
+
* reverse(".index") // ✓ Local route (no params)
|
|
142
|
+
* reverse("magazine.index") // ✓ Global route (fully qualified)
|
|
143
|
+
* reverse("blog.post", { slug: "hello" }) // ✓ Global route + params
|
|
144
|
+
* reverse(".typo") // ✗ Compile error (not in local routes)
|
|
145
|
+
* reverse("typo") // ✗ Compile error (not in global routes)
|
|
148
146
|
* ```
|
|
149
147
|
*/
|
|
150
|
-
export type ScopedReverseFunction<
|
|
148
|
+
export type ScopedReverseFunction<
|
|
149
|
+
TLocalRoutes,
|
|
150
|
+
TGlobalRoutes = TLocalRoutes,
|
|
151
|
+
> = {
|
|
151
152
|
/**
|
|
152
|
-
*
|
|
153
|
-
* @recommended Use this for type-safe URL generation
|
|
153
|
+
* Global route without params
|
|
154
154
|
*/
|
|
155
|
-
<TName extends keyof
|
|
156
|
-
name: IsEmptyObject<
|
|
155
|
+
<TName extends keyof TGlobalRoutes & string>(
|
|
156
|
+
name: IsEmptyObject<
|
|
157
|
+
ExtractParams<RoutePatternFor<TGlobalRoutes, TName>>
|
|
158
|
+
> extends true
|
|
159
|
+
? TName
|
|
160
|
+
: never,
|
|
157
161
|
): string;
|
|
158
162
|
|
|
159
163
|
/**
|
|
160
|
-
*
|
|
161
|
-
* @recommended Use this for type-safe URL generation with parameters
|
|
164
|
+
* Global route with params
|
|
162
165
|
*/
|
|
163
|
-
<TName extends keyof
|
|
166
|
+
<TName extends keyof TGlobalRoutes & string>(
|
|
167
|
+
name: TName,
|
|
168
|
+
params: ExtractParams<RoutePatternFor<TGlobalRoutes, TName>>,
|
|
169
|
+
): string;
|
|
170
|
+
|
|
171
|
+
/**
|
|
172
|
+
* Global route with params and search
|
|
173
|
+
*/
|
|
174
|
+
<TName extends keyof TGlobalRoutes & string>(
|
|
164
175
|
name: TName,
|
|
165
|
-
params: ExtractParams<RoutePatternFor<
|
|
176
|
+
params: ExtractParams<RoutePatternFor<TGlobalRoutes, TName>>,
|
|
177
|
+
search: ResolveSearchSchema<ExtractSearchSchema<TGlobalRoutes, TName>>,
|
|
166
178
|
): string;
|
|
167
179
|
|
|
168
180
|
/**
|
|
169
|
-
*
|
|
170
|
-
* Use for cross-module navigation: "shop.cart", "blog.post"
|
|
181
|
+
* Dot-prefixed local route without params
|
|
171
182
|
*/
|
|
172
|
-
|
|
183
|
+
<TName extends keyof TLocalRoutes & string>(
|
|
184
|
+
name: IsEmptyObject<
|
|
185
|
+
ExtractParams<RoutePatternFor<TLocalRoutes, TName>>
|
|
186
|
+
> extends true
|
|
187
|
+
? `.${TName}`
|
|
188
|
+
: never,
|
|
189
|
+
): string;
|
|
173
190
|
|
|
174
191
|
/**
|
|
175
|
-
*
|
|
176
|
-
* Prefer route names for type safety. Only use paths when necessary.
|
|
192
|
+
* Dot-prefixed local route with params
|
|
177
193
|
*/
|
|
178
|
-
|
|
194
|
+
<TName extends keyof TLocalRoutes & string>(
|
|
195
|
+
name: `.${TName}`,
|
|
196
|
+
params: ExtractParams<RoutePatternFor<TLocalRoutes, TName>>,
|
|
197
|
+
): string;
|
|
198
|
+
|
|
199
|
+
/**
|
|
200
|
+
* Dot-prefixed local route with params and search
|
|
201
|
+
*/
|
|
202
|
+
<TName extends keyof TLocalRoutes & string>(
|
|
203
|
+
name: `.${TName}`,
|
|
204
|
+
params: ExtractParams<RoutePatternFor<TLocalRoutes, TName>>,
|
|
205
|
+
search: ResolveSearchSchema<ExtractSearchSchema<TLocalRoutes, TName>>,
|
|
206
|
+
): string;
|
|
179
207
|
};
|
|
180
208
|
|
|
181
209
|
/**
|
|
182
210
|
* Extract local routes type from UrlPatterns
|
|
183
211
|
* Used with scopedReverse() to get the routes type from patterns
|
|
184
212
|
*/
|
|
185
|
-
export type ExtractLocalRoutes<TPatterns> =
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
213
|
+
export type ExtractLocalRoutes<TPatterns> = TPatterns extends {
|
|
214
|
+
readonly _routes?: infer TRoutes;
|
|
215
|
+
}
|
|
216
|
+
? TRoutes
|
|
217
|
+
: TPatterns extends Record<string, string>
|
|
218
|
+
? TPatterns
|
|
219
|
+
: Record<string, string>;
|
|
191
220
|
|
|
192
221
|
/**
|
|
193
222
|
* Extract the response data type for a named route from a UrlPatterns instance.
|
|
@@ -198,24 +227,23 @@ export type { RouteResponse } from "./urls.js";
|
|
|
198
227
|
/**
|
|
199
228
|
* Get a locally-typed reverse function from ctx.reverse for composable modules.
|
|
200
229
|
*
|
|
201
|
-
* This is a type-only cast - ctx.reverse already resolves
|
|
202
|
-
*
|
|
203
|
-
*
|
|
230
|
+
* This is a type-only cast - ctx.reverse already resolves names at runtime.
|
|
231
|
+
* Provides type safety: `.name` validates against local routes,
|
|
232
|
+
* `name` validates against global named-routes.
|
|
204
233
|
*
|
|
205
234
|
* @param reverse - The ctx.reverse function from HandlerContext
|
|
206
|
-
* @returns The same reverse function,
|
|
235
|
+
* @returns The same reverse function, typed with local + global routes
|
|
207
236
|
*
|
|
208
237
|
* @example
|
|
209
238
|
* ```typescript
|
|
210
239
|
* // urls/blog.tsx
|
|
211
240
|
* export const blogPatterns = urls(({ path }) => [
|
|
212
241
|
* path("/", (ctx) => {
|
|
213
|
-
* // Get locally-typed reverse for this module's routes
|
|
214
242
|
* const reverse = scopedReverse<typeof blogPatterns>(ctx.reverse);
|
|
215
243
|
*
|
|
216
|
-
* reverse("index"); // ✓
|
|
217
|
-
* reverse("post", { slug: "x" }); // ✓
|
|
218
|
-
* reverse("shop.cart");
|
|
244
|
+
* reverse(".index"); // ✓ Local route
|
|
245
|
+
* reverse(".post", { slug: "x" }); // ✓ Local with params
|
|
246
|
+
* reverse("shop.cart"); // ✓ Global route
|
|
219
247
|
*
|
|
220
248
|
* return <BlogIndex />;
|
|
221
249
|
* }, { name: "index" }),
|
|
@@ -225,7 +253,7 @@ export type { RouteResponse } from "./urls.js";
|
|
|
225
253
|
* ```
|
|
226
254
|
*/
|
|
227
255
|
export function scopedReverse<TPatterns>(
|
|
228
|
-
reverse: (
|
|
256
|
+
reverse: (...args: any[]) => string,
|
|
229
257
|
): ScopedReverseFunction<ExtractLocalRoutes<TPatterns>> {
|
|
230
258
|
return reverse as ScopedReverseFunction<ExtractLocalRoutes<TPatterns>>;
|
|
231
259
|
}
|
|
@@ -244,24 +272,59 @@ export function scopedReverse<TPatterns>(
|
|
|
244
272
|
* reverse("detail", { slug: "my-product" }); // "/shop/product/my-product"
|
|
245
273
|
* ```
|
|
246
274
|
*/
|
|
275
|
+
type RouteMapEntry = string | { path: string; search?: Record<string, string> };
|
|
276
|
+
|
|
277
|
+
function resolveRoutePattern(
|
|
278
|
+
entry: RouteMapEntry | undefined,
|
|
279
|
+
): string | undefined {
|
|
280
|
+
if (!entry) return undefined;
|
|
281
|
+
return typeof entry === "string" ? entry : entry.path;
|
|
282
|
+
}
|
|
283
|
+
|
|
247
284
|
export function createReverse<TRoutes extends Record<string, string>>(
|
|
248
|
-
routeMap: TRoutes
|
|
285
|
+
routeMap: TRoutes,
|
|
249
286
|
): ReverseFunction<TRoutes & Record<string, string>> {
|
|
250
|
-
return ((
|
|
251
|
-
|
|
287
|
+
return ((
|
|
288
|
+
name: string,
|
|
289
|
+
params?: Record<string, string>,
|
|
290
|
+
search?: Record<string, unknown>,
|
|
291
|
+
) => {
|
|
292
|
+
const pattern = resolveRoutePattern(
|
|
293
|
+
routeMap[name] as unknown as RouteMapEntry,
|
|
294
|
+
);
|
|
252
295
|
if (!pattern) {
|
|
296
|
+
// During build-time discovery, lazy includes haven't resolved yet.
|
|
297
|
+
// Return a placeholder instead of crashing the build.
|
|
298
|
+
if ((globalThis as any).__rscRouterDiscoveryActive) {
|
|
299
|
+
return `/__unresolved_reverse/${name}`;
|
|
300
|
+
}
|
|
253
301
|
throw new Error(`Unknown route: ${name}`);
|
|
254
302
|
}
|
|
255
303
|
|
|
256
|
-
|
|
304
|
+
let result = pattern;
|
|
305
|
+
if (params) {
|
|
306
|
+
// Replace :param placeholders with actual values
|
|
307
|
+
// Strip constraint syntax: :param(a|b) -> use "param" as key
|
|
308
|
+
result = result.replace(
|
|
309
|
+
/:([a-zA-Z_][a-zA-Z0-9_]*)(\([^)]*\))?\??/g,
|
|
310
|
+
(_, key) => {
|
|
311
|
+
const value = params[key];
|
|
312
|
+
if (value === undefined) {
|
|
313
|
+
throw new Error(`Missing param "${key}" for route "${name}"`);
|
|
314
|
+
}
|
|
315
|
+
return encodeURIComponent(value);
|
|
316
|
+
},
|
|
317
|
+
);
|
|
318
|
+
}
|
|
257
319
|
|
|
258
|
-
//
|
|
259
|
-
|
|
260
|
-
const
|
|
261
|
-
if (
|
|
262
|
-
|
|
320
|
+
// Append search params as query string
|
|
321
|
+
if (search) {
|
|
322
|
+
const qs = serializeSearchParams(search);
|
|
323
|
+
if (qs) {
|
|
324
|
+
result += `?${qs}`;
|
|
263
325
|
}
|
|
264
|
-
|
|
265
|
-
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
return result;
|
|
266
329
|
}) as ReverseFunction<TRoutes>;
|
|
267
330
|
}
|
|
@@ -60,7 +60,8 @@ function NetworkErrorFallback({
|
|
|
60
60
|
marginBottom: "1.5rem",
|
|
61
61
|
}}
|
|
62
62
|
>
|
|
63
|
-
{error.message ||
|
|
63
|
+
{error.message ||
|
|
64
|
+
"Unable to connect to the server. Please check your internet connection."}
|
|
64
65
|
</p>
|
|
65
66
|
<div style={{ display: "flex", gap: "1rem", justifyContent: "center" }}>
|
|
66
67
|
<button
|
|
@@ -104,7 +105,12 @@ function NetworkErrorFallback({
|
|
|
104
105
|
* Default fallback UI for root error boundary
|
|
105
106
|
* This is shown when an unhandled error bubbles up to the root
|
|
106
107
|
*/
|
|
107
|
-
function RootErrorFallback({
|
|
108
|
+
function RootErrorFallback({
|
|
109
|
+
error,
|
|
110
|
+
reset,
|
|
111
|
+
}: ClientErrorBoundaryFallbackProps): ReactNode {
|
|
112
|
+
const isDev = process.env.NODE_ENV !== "production";
|
|
113
|
+
|
|
108
114
|
return (
|
|
109
115
|
<div
|
|
110
116
|
style={{
|
|
@@ -131,38 +137,40 @@ function RootErrorFallback({ error, reset }: ClientErrorBoundaryFallbackProps):
|
|
|
131
137
|
>
|
|
132
138
|
An unexpected error occurred while processing your request.
|
|
133
139
|
</p>
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
background: "#fef2f2",
|
|
137
|
-
border: "1px solid #fecaca",
|
|
138
|
-
borderRadius: "0.5rem",
|
|
139
|
-
padding: "1rem",
|
|
140
|
-
marginBottom: "1rem",
|
|
141
|
-
}}
|
|
142
|
-
>
|
|
143
|
-
<p
|
|
140
|
+
{isDev && (
|
|
141
|
+
<div
|
|
144
142
|
style={{
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
143
|
+
background: "#fef2f2",
|
|
144
|
+
border: "1px solid #fecaca",
|
|
145
|
+
borderRadius: "0.5rem",
|
|
146
|
+
padding: "1rem",
|
|
147
|
+
marginBottom: "1rem",
|
|
148
148
|
}}
|
|
149
149
|
>
|
|
150
|
-
|
|
151
|
-
</p>
|
|
152
|
-
{error.stack && (
|
|
153
|
-
<pre
|
|
150
|
+
<p
|
|
154
151
|
style={{
|
|
155
|
-
|
|
156
|
-
color: "#
|
|
157
|
-
|
|
158
|
-
whiteSpace: "pre-wrap",
|
|
159
|
-
wordBreak: "break-word",
|
|
152
|
+
fontWeight: 600,
|
|
153
|
+
color: "#991b1b",
|
|
154
|
+
marginBottom: "0.5rem",
|
|
160
155
|
}}
|
|
161
156
|
>
|
|
162
|
-
{error.
|
|
163
|
-
</
|
|
164
|
-
|
|
165
|
-
|
|
157
|
+
{error.name}: {error.message}
|
|
158
|
+
</p>
|
|
159
|
+
{error.stack && (
|
|
160
|
+
<pre
|
|
161
|
+
style={{
|
|
162
|
+
fontSize: "0.75rem",
|
|
163
|
+
color: "#6b7280",
|
|
164
|
+
overflow: "auto",
|
|
165
|
+
whiteSpace: "pre-wrap",
|
|
166
|
+
wordBreak: "break-word",
|
|
167
|
+
}}
|
|
168
|
+
>
|
|
169
|
+
{error.stack}
|
|
170
|
+
</pre>
|
|
171
|
+
)}
|
|
172
|
+
</div>
|
|
173
|
+
)}
|
|
166
174
|
<div style={{ display: "flex", gap: "1rem" }}>
|
|
167
175
|
<button
|
|
168
176
|
type="button"
|
|
@@ -231,7 +239,11 @@ export class RootErrorBoundary extends Component<
|
|
|
231
239
|
}
|
|
232
240
|
|
|
233
241
|
componentDidCatch(error: Error, errorInfo: React.ErrorInfo): void {
|
|
234
|
-
console.error(
|
|
242
|
+
console.error(
|
|
243
|
+
"[RootErrorBoundary] Unhandled error caught:",
|
|
244
|
+
error,
|
|
245
|
+
errorInfo,
|
|
246
|
+
);
|
|
235
247
|
}
|
|
236
248
|
|
|
237
249
|
componentDidUpdate(prevProps: { children: ReactNode }): void {
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
import type { ReactNode } from "react";
|
|
3
3
|
import { Suspense, use, useId } from "react";
|
|
4
4
|
import { invariant } from "./errors";
|
|
5
|
-
import { OutletProvider } from "./
|
|
5
|
+
import { OutletProvider } from "./outlet-provider.js";
|
|
6
6
|
import type { ResolvedSegment } from "./types.js";
|
|
7
7
|
import { isLoaderDataResult } from "./types.js";
|
|
8
8
|
|
|
@@ -31,7 +31,10 @@ export function RouteContentWrapper({
|
|
|
31
31
|
return content as ReactNode;
|
|
32
32
|
}
|
|
33
33
|
return (
|
|
34
|
-
<Suspense
|
|
34
|
+
<Suspense
|
|
35
|
+
fallback={fallback ?? null}
|
|
36
|
+
key={segmentId ? "route-content-suspense-" + segmentId : undefined}
|
|
37
|
+
>
|
|
35
38
|
<Suspender content={content} key={segmentId} />
|
|
36
39
|
</Suspense>
|
|
37
40
|
);
|
|
@@ -50,11 +53,11 @@ export function RouteContentWrapperCallback<T>({
|
|
|
50
53
|
invariant(children, "RouteContentWrapperCallback requires children");
|
|
51
54
|
invariant(
|
|
52
55
|
typeof children === "function",
|
|
53
|
-
"RouteContentWrapperCallback requires children to be a function"
|
|
56
|
+
"RouteContentWrapperCallback requires children to be a function",
|
|
54
57
|
);
|
|
55
58
|
invariant(
|
|
56
59
|
resolve !== undefined,
|
|
57
|
-
"RouteContentWrapperCallback requires resolve"
|
|
60
|
+
"RouteContentWrapperCallback requires resolve",
|
|
58
61
|
);
|
|
59
62
|
return (
|
|
60
63
|
<Suspense
|