@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430
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 +1601 -0
- package/dist/vite/index.js +4474 -867
- package/package.json +60 -51
- 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 +167 -0
- package/skills/hooks/SKILL.md +334 -72
- 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 +89 -30
- package/skills/loader/SKILL.md +388 -38
- package/skills/middleware/SKILL.md +171 -34
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +78 -1
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +85 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +226 -14
- package/skills/router-setup/SKILL.md +123 -30
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +318 -89
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +321 -0
- 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 +124 -71
- 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 +258 -308
- package/src/browser/prefetch/cache.ts +146 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +42 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +185 -73
- 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 +32 -79
- package/src/browser/react/use-href.tsx +2 -2
- 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 +107 -26
- 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 +109 -47
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +265 -0
- 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 -303
- 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 +106 -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 +108 -2
- package/src/handle.ts +15 -29
- 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 +119 -29
- package/src/index.rsc.ts +153 -19
- package/src/index.ts +211 -30
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +26 -157
- 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 +37 -0
- package/src/prerender/store.ts +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- 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 -1428
- package/src/route-map-builder.ts +211 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +59 -8
- 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 +395 -0
- 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 +148 -35
- package/src/router/match-api.ts +620 -0
- package/src/router/match-context.ts +5 -3
- 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 +382 -9
- package/src/router/match-middleware/cache-store.ts +51 -22
- package/src/router/match-middleware/intercept-resolution.ts +55 -17
- package/src/router/match-middleware/segment-resolution.ts +24 -6
- 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 +211 -43
- 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 +1241 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- 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 +239 -0
- package/src/router/types.ts +77 -3
- package/src/router.ts +692 -4257
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +764 -754
- 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 +235 -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 +182 -51
- 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 -130
- 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 -1623
- 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 -802
- 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 -1133
- 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/href.ts +0 -255
- package/src/server/route-manifest-cache.ts +0 -173
- 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/{version.d.ts → plugins/version.d.ts} +0 -0
package/src/server.ts
CHANGED
|
@@ -1,146 +1,51 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
2
|
+
* @rangojs/router/server — Internal subpath
|
|
3
3
|
*
|
|
4
|
-
*
|
|
5
|
-
*
|
|
4
|
+
* This module is NOT user-facing. Import from "@rangojs/router" instead.
|
|
5
|
+
*
|
|
6
|
+
* Exports here are consumed by the Vite plugin (discovery, manifest injection,
|
|
7
|
+
* virtual modules) and the RSC handler internals. They are not part of the
|
|
8
|
+
* public API and may change without notice.
|
|
6
9
|
*/
|
|
7
10
|
|
|
8
|
-
//
|
|
9
|
-
export {
|
|
10
|
-
createLoader,
|
|
11
|
-
redirect,
|
|
12
|
-
type RouteHelpers,
|
|
13
|
-
type RouteHandlers,
|
|
14
|
-
} from "./route-definition.js";
|
|
11
|
+
// Router registry (used by Vite plugin for build-time discovery)
|
|
12
|
+
export { RSC_ROUTER_BRAND, RouterRegistry } from "./router.js";
|
|
15
13
|
|
|
16
|
-
//
|
|
14
|
+
// Host router registry (used by Vite plugin for host-router lazy discovery)
|
|
17
15
|
export {
|
|
18
|
-
|
|
19
|
-
type
|
|
20
|
-
|
|
21
|
-
type UrlPatterns,
|
|
22
|
-
type IncludeOptions,
|
|
23
|
-
} from "./urls.js";
|
|
24
|
-
|
|
25
|
-
// Re-export IncludeItem from route-types
|
|
26
|
-
export type { IncludeItem } from "./route-types.js";
|
|
16
|
+
HostRouterRegistry,
|
|
17
|
+
type HostRouterRegistryEntry,
|
|
18
|
+
} from "./host/router.js";
|
|
27
19
|
|
|
28
|
-
//
|
|
20
|
+
// Route map builder (Vite plugin injects these via virtual modules)
|
|
29
21
|
export {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
22
|
+
registerRouteMap,
|
|
23
|
+
setCachedManifest,
|
|
24
|
+
clearCachedManifest,
|
|
25
|
+
clearAllRouterData,
|
|
26
|
+
getGlobalRouteMap,
|
|
27
|
+
getRouterManifest,
|
|
28
|
+
setPrecomputedEntries,
|
|
29
|
+
setRouteTrie,
|
|
30
|
+
setManifestReadyPromise,
|
|
31
|
+
setRouterManifest,
|
|
32
|
+
setRouterTrie,
|
|
33
|
+
setRouterPrecomputedEntries,
|
|
34
|
+
registerRouterManifestLoader,
|
|
35
|
+
ensureRouterManifest,
|
|
36
|
+
} from "./route-map-builder.js";
|
|
37
37
|
|
|
38
|
-
//
|
|
38
|
+
// Loader registry (Vite plugin registers lazy loader imports)
|
|
39
39
|
export {
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
type PrefixRoutePatterns,
|
|
44
|
-
type ParamsFor,
|
|
45
|
-
type SanitizePrefix,
|
|
46
|
-
type MergeRoutes,
|
|
47
|
-
} from "./href.js";
|
|
48
|
-
|
|
49
|
-
// Segment system (server-only)
|
|
50
|
-
export { renderSegments } from "./segment-system.js";
|
|
51
|
-
|
|
52
|
-
// Performance tracking (server-only)
|
|
53
|
-
export { track } from "./server/context.js";
|
|
54
|
-
|
|
55
|
-
// Handle API (works in both server and client contexts)
|
|
56
|
-
export { createHandle, isHandle, type Handle } from "./handle.js";
|
|
57
|
-
|
|
58
|
-
// Built-in handles
|
|
59
|
-
export { Meta } from "./handles/meta.js";
|
|
60
|
-
|
|
61
|
-
// Loader registry (for GET-based loader fetching)
|
|
62
|
-
export { registerLoaderById, setLoaderImports } from "./server/loader-registry.js";
|
|
63
|
-
|
|
64
|
-
// Route map builder (for build-time manifest registration)
|
|
65
|
-
export { registerRouteMap, setCachedManifest } from "./route-map-builder.js";
|
|
40
|
+
registerLoaderById,
|
|
41
|
+
setLoaderImports,
|
|
42
|
+
} from "./server/loader-registry.js";
|
|
66
43
|
|
|
67
|
-
// Request context (
|
|
44
|
+
// Request context creation (used by RSC handler, not user-facing)
|
|
68
45
|
export {
|
|
69
|
-
getRequestContext,
|
|
70
|
-
requireRequestContext,
|
|
71
46
|
createRequestContext,
|
|
72
|
-
type RequestContext,
|
|
73
47
|
type CreateRequestContextOptions,
|
|
74
48
|
} from "./server/request-context.js";
|
|
75
49
|
|
|
76
|
-
//
|
|
77
|
-
export
|
|
78
|
-
|
|
79
|
-
// Middleware context types (Middleware type is exported from types.ts)
|
|
80
|
-
export type {
|
|
81
|
-
MiddlewareContext,
|
|
82
|
-
CookieOptions,
|
|
83
|
-
} from "./router/middleware.js";
|
|
84
|
-
|
|
85
|
-
// Error classes and utilities
|
|
86
|
-
export {
|
|
87
|
-
RouteNotFoundError,
|
|
88
|
-
DataNotFoundError,
|
|
89
|
-
notFound,
|
|
90
|
-
MiddlewareError,
|
|
91
|
-
HandlerError,
|
|
92
|
-
BuildError,
|
|
93
|
-
InvalidHandlerError,
|
|
94
|
-
sanitizeError,
|
|
95
|
-
} from "./errors.js";
|
|
96
|
-
|
|
97
|
-
// Component utilities
|
|
98
|
-
export {
|
|
99
|
-
isClientComponent,
|
|
100
|
-
assertClientComponent,
|
|
101
|
-
} from "./component-utils.js";
|
|
102
|
-
|
|
103
|
-
// Debug utilities for route matching (development only)
|
|
104
|
-
export {
|
|
105
|
-
enableMatchDebug,
|
|
106
|
-
getMatchDebugStats,
|
|
107
|
-
} from "./router/pattern-matching.js";
|
|
108
|
-
|
|
109
|
-
// Types (re-exported for convenience - user-facing only)
|
|
110
|
-
export type {
|
|
111
|
-
// Configuration types
|
|
112
|
-
RouterEnv,
|
|
113
|
-
DefaultEnv,
|
|
114
|
-
RouteDefinition,
|
|
115
|
-
RouteConfig,
|
|
116
|
-
RouteDefinitionOptions,
|
|
117
|
-
TrailingSlashMode,
|
|
118
|
-
// Handler types
|
|
119
|
-
Handler, // Supports params object, path pattern, or route name
|
|
120
|
-
HandlerContext,
|
|
121
|
-
ExtractParams,
|
|
122
|
-
GenericParams,
|
|
123
|
-
// Middleware types (also exported from router/middleware.js above)
|
|
124
|
-
Middleware, // Supports env type and optional route name for params
|
|
125
|
-
// Revalidation types
|
|
126
|
-
RevalidateParams,
|
|
127
|
-
Revalidate,
|
|
128
|
-
RouteKeys,
|
|
129
|
-
// Loader types
|
|
130
|
-
LoaderDefinition,
|
|
131
|
-
LoaderFn,
|
|
132
|
-
LoaderContext,
|
|
133
|
-
// Error boundary types
|
|
134
|
-
ErrorInfo,
|
|
135
|
-
ErrorBoundaryFallbackProps,
|
|
136
|
-
ErrorBoundaryHandler,
|
|
137
|
-
ClientErrorBoundaryFallbackProps,
|
|
138
|
-
// NotFound boundary types
|
|
139
|
-
NotFoundInfo,
|
|
140
|
-
NotFoundBoundaryFallbackProps,
|
|
141
|
-
NotFoundBoundaryHandler,
|
|
142
|
-
// Error handling callback types
|
|
143
|
-
ErrorPhase,
|
|
144
|
-
OnErrorContext,
|
|
145
|
-
OnErrorCallback,
|
|
146
|
-
} from "./types.js";
|
|
50
|
+
// Component utilities (used internally for server/client boundary checks)
|
|
51
|
+
export { isClientComponent, assertClientComponent } from "./component-utils.js";
|
package/src/ssr/index.tsx
CHANGED
|
@@ -1,14 +1,18 @@
|
|
|
1
1
|
import React from "react";
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
import { initThemeConfigSync } from "../theme/theme-context.js";
|
|
2
|
+
import { renderSegments } from "../segment-system.js";
|
|
3
|
+
import { filterSegmentOrder } from "../browser/react/filter-segment-order.js";
|
|
5
4
|
import { ThemeProvider } from "../theme/ThemeProvider.js";
|
|
5
|
+
import { NonceContext } from "../browser/react/nonce-context.js";
|
|
6
6
|
import { NavigationStoreContext } from "../browser/react/context.js";
|
|
7
7
|
import type { NavigationStoreContextValue } from "../browser/react/context.js";
|
|
8
8
|
import type { HandleData } from "../browser/types.js";
|
|
9
9
|
import type { ErrorPhase } from "../types.js";
|
|
10
|
+
import type { ResolvedSegment } from "../types.js";
|
|
10
11
|
import type { ResolvedThemeConfig, Theme } from "../theme/types.js";
|
|
11
|
-
import type {
|
|
12
|
+
import type {
|
|
13
|
+
EventController,
|
|
14
|
+
DerivedNavigationState,
|
|
15
|
+
} from "../browser/event-controller.js";
|
|
12
16
|
|
|
13
17
|
/**
|
|
14
18
|
* Options for injectRSCPayload
|
|
@@ -29,6 +33,13 @@ interface RenderToReadableStreamOptions {
|
|
|
29
33
|
formState?: unknown;
|
|
30
34
|
}
|
|
31
35
|
|
|
36
|
+
/**
|
|
37
|
+
* ReadableStream with the allReady promise added by react-dom/server.edge.
|
|
38
|
+
*/
|
|
39
|
+
interface ReactDOMReadableStream extends ReadableStream<Uint8Array> {
|
|
40
|
+
allReady: Promise<void>;
|
|
41
|
+
}
|
|
42
|
+
|
|
32
43
|
/**
|
|
33
44
|
* Options for the renderHTML function
|
|
34
45
|
*/
|
|
@@ -45,6 +56,14 @@ export interface SSRRenderOptions {
|
|
|
45
56
|
* Nonce for Content Security Policy (CSP)
|
|
46
57
|
*/
|
|
47
58
|
nonce?: string;
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* SSR stream mode.
|
|
62
|
+
*
|
|
63
|
+
* - `"stream"` (default) — start flushing HTML immediately.
|
|
64
|
+
* - `"allReady"` — await `stream.allReady` before returning.
|
|
65
|
+
*/
|
|
66
|
+
streamMode?: import("../router/router-options.js").SSRStreamMode;
|
|
48
67
|
}
|
|
49
68
|
|
|
50
69
|
/**
|
|
@@ -54,22 +73,24 @@ export interface SSRDependencies<TEnv = unknown> {
|
|
|
54
73
|
/**
|
|
55
74
|
* createFromReadableStream from @vitejs/plugin-rsc/ssr
|
|
56
75
|
*/
|
|
57
|
-
createFromReadableStream: <T>(
|
|
76
|
+
createFromReadableStream: <T>(
|
|
77
|
+
stream: ReadableStream<Uint8Array>,
|
|
78
|
+
) => Promise<T>;
|
|
58
79
|
|
|
59
80
|
/**
|
|
60
81
|
* renderToReadableStream from react-dom/server.edge
|
|
61
82
|
*/
|
|
62
83
|
renderToReadableStream: (
|
|
63
84
|
element: React.ReactNode,
|
|
64
|
-
options?: RenderToReadableStreamOptions
|
|
65
|
-
) => Promise<
|
|
85
|
+
options?: RenderToReadableStreamOptions,
|
|
86
|
+
) => Promise<ReactDOMReadableStream>;
|
|
66
87
|
|
|
67
88
|
/**
|
|
68
89
|
* injectRSCPayload from rsc-html-stream/server
|
|
69
90
|
*/
|
|
70
91
|
injectRSCPayload: (
|
|
71
92
|
rscStream: ReadableStream<Uint8Array>,
|
|
72
|
-
options?: InjectRSCPayloadOptions
|
|
93
|
+
options?: InjectRSCPayloadOptions,
|
|
73
94
|
) => TransformStream<Uint8Array, Uint8Array>;
|
|
74
95
|
|
|
75
96
|
/**
|
|
@@ -101,13 +122,16 @@ export interface SSRDependencies<TEnv = unknown> {
|
|
|
101
122
|
* RSC payload type (minimal interface for SSR)
|
|
102
123
|
*/
|
|
103
124
|
interface RscPayload {
|
|
104
|
-
root: React.ReactNode;
|
|
105
125
|
metadata?: {
|
|
126
|
+
segments?: ResolvedSegment[];
|
|
127
|
+
rootLayout?: React.ComponentType<{ children: React.ReactNode }>;
|
|
106
128
|
handles?: AsyncGenerator<HandleData, void, unknown>;
|
|
107
129
|
matched?: string[];
|
|
108
130
|
pathname?: string;
|
|
131
|
+
params?: Record<string, string>;
|
|
109
132
|
themeConfig?: ResolvedThemeConfig | null;
|
|
110
133
|
initialTheme?: Theme;
|
|
134
|
+
version?: string;
|
|
111
135
|
};
|
|
112
136
|
}
|
|
113
137
|
|
|
@@ -116,7 +140,7 @@ interface RscPayload {
|
|
|
116
140
|
* Used for SSR where we need to await all handle data before rendering.
|
|
117
141
|
*/
|
|
118
142
|
async function consumeAsyncGenerator(
|
|
119
|
-
generator: AsyncGenerator<HandleData, void, unknown
|
|
143
|
+
generator: AsyncGenerator<HandleData, void, unknown>,
|
|
120
144
|
): Promise<HandleData> {
|
|
121
145
|
let lastData: HandleData = {};
|
|
122
146
|
for await (const data of generator) {
|
|
@@ -129,8 +153,18 @@ async function consumeAsyncGenerator(
|
|
|
129
153
|
* Create a minimal event controller for SSR.
|
|
130
154
|
* This provides the correct pathname so useNavigation returns the right value during SSR.
|
|
131
155
|
*/
|
|
132
|
-
function createSsrEventController(
|
|
133
|
-
|
|
156
|
+
function createSsrEventController(opts: {
|
|
157
|
+
pathname: string;
|
|
158
|
+
params?: Record<string, string>;
|
|
159
|
+
handleData?: HandleData;
|
|
160
|
+
matched?: string[];
|
|
161
|
+
}): EventController {
|
|
162
|
+
const location = new URL(opts.pathname, "http://localhost");
|
|
163
|
+
let params = opts.params ?? {};
|
|
164
|
+
const handleState = {
|
|
165
|
+
data: opts.handleData ?? {},
|
|
166
|
+
segmentOrder: filterSegmentOrder(opts.matched ?? []),
|
|
167
|
+
};
|
|
134
168
|
const state: DerivedNavigationState = {
|
|
135
169
|
state: "idle",
|
|
136
170
|
isStreaming: false,
|
|
@@ -141,6 +175,7 @@ function createSsrEventController(pathname: string): EventController {
|
|
|
141
175
|
|
|
142
176
|
return {
|
|
143
177
|
getState: () => state,
|
|
178
|
+
getLocation: () => location,
|
|
144
179
|
subscribe: () => () => {},
|
|
145
180
|
getActionState: () => ({
|
|
146
181
|
state: "idle",
|
|
@@ -152,7 +187,11 @@ function createSsrEventController(pathname: string): EventController {
|
|
|
152
187
|
subscribeToAction: () => () => {},
|
|
153
188
|
subscribeToHandles: () => () => {},
|
|
154
189
|
setHandleData: () => {},
|
|
155
|
-
getHandleState: () =>
|
|
190
|
+
getHandleState: () => handleState,
|
|
191
|
+
setParams: (nextParams) => {
|
|
192
|
+
params = nextParams;
|
|
193
|
+
},
|
|
194
|
+
getParams: () => params,
|
|
156
195
|
setLocation: () => {},
|
|
157
196
|
startNavigation: () => {
|
|
158
197
|
throw new Error("Navigation not supported during SSR");
|
|
@@ -164,6 +203,7 @@ function createSsrEventController(pathname: string): EventController {
|
|
|
164
203
|
abortAllActions: () => {},
|
|
165
204
|
getCurrentNavigation: () => null,
|
|
166
205
|
getInflightActions: () => new Map(),
|
|
206
|
+
hadAnyConcurrentActions: () => false,
|
|
167
207
|
};
|
|
168
208
|
}
|
|
169
209
|
|
|
@@ -203,9 +243,9 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
|
|
|
203
243
|
*/
|
|
204
244
|
return async function renderHTML(
|
|
205
245
|
rscStream: ReadableStream<Uint8Array>,
|
|
206
|
-
options?: SSRRenderOptions
|
|
246
|
+
options?: SSRRenderOptions,
|
|
207
247
|
): Promise<ReadableStream<Uint8Array>> {
|
|
208
|
-
const { nonce, formState } = options ?? {};
|
|
248
|
+
const { nonce, formState, streamMode } = options ?? {};
|
|
209
249
|
|
|
210
250
|
try {
|
|
211
251
|
// Tee the stream:
|
|
@@ -220,47 +260,68 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
|
|
|
220
260
|
function SsrRoot() {
|
|
221
261
|
payload ??= createFromReadableStream<RscPayload>(rscStream1);
|
|
222
262
|
const resolved = React.use(payload);
|
|
223
|
-
|
|
224
|
-
// Initialize segments state before children render (for useSegments hook)
|
|
225
|
-
initSegmentsSync(resolved.metadata?.matched, resolved.metadata?.pathname);
|
|
226
|
-
|
|
227
|
-
// Initialize theme config for MetaTags to render theme script
|
|
228
263
|
const themeConfig = resolved.metadata?.themeConfig ?? null;
|
|
229
|
-
|
|
264
|
+
const pathname = resolved.metadata?.pathname ?? "/";
|
|
230
265
|
|
|
231
|
-
// Await handles
|
|
266
|
+
// Await handles before creating SSR event controller so hooks can
|
|
267
|
+
// read request-local handle data via NavigationStoreContext.
|
|
232
268
|
// The handles property is an async generator that yields on each push
|
|
233
269
|
// Memoize the promise since async generators can only be iterated once
|
|
270
|
+
let handleData: HandleData = {};
|
|
234
271
|
if (resolved.metadata?.handles) {
|
|
235
272
|
handlesPromise ??= consumeAsyncGenerator(resolved.metadata.handles);
|
|
236
|
-
|
|
237
|
-
initHandleDataSync(handleData, resolved.metadata.matched);
|
|
273
|
+
handleData = React.use(handlesPromise);
|
|
238
274
|
}
|
|
239
275
|
|
|
240
|
-
// Create SSR context with
|
|
276
|
+
// Create SSR context with request-local pathname/params/handles.
|
|
241
277
|
ssrContextValue ??= {
|
|
242
278
|
store: null as any,
|
|
243
|
-
eventController: createSsrEventController(
|
|
279
|
+
eventController: createSsrEventController({
|
|
280
|
+
pathname,
|
|
281
|
+
params: resolved.metadata?.params,
|
|
282
|
+
handleData,
|
|
283
|
+
matched: resolved.metadata?.matched,
|
|
284
|
+
}),
|
|
244
285
|
navigate: async () => {},
|
|
245
286
|
refresh: async () => {},
|
|
287
|
+
version: resolved.metadata?.version,
|
|
246
288
|
};
|
|
247
289
|
|
|
248
|
-
// Build content tree
|
|
290
|
+
// Build content tree from segments.
|
|
249
291
|
// Order must match NavigationProvider: NavigationStoreContext > ThemeProvider > content
|
|
250
|
-
|
|
292
|
+
const reconstructedRoot = renderSegments(
|
|
293
|
+
resolved.metadata?.segments ?? [],
|
|
294
|
+
{
|
|
295
|
+
rootLayout: resolved.metadata?.rootLayout,
|
|
296
|
+
},
|
|
297
|
+
);
|
|
298
|
+
let content: React.ReactNode =
|
|
299
|
+
reconstructedRoot instanceof Promise
|
|
300
|
+
? React.use(reconstructedRoot)
|
|
301
|
+
: reconstructedRoot;
|
|
251
302
|
|
|
252
303
|
// Wrap content with ThemeProvider if theme is enabled
|
|
253
304
|
if (themeConfig) {
|
|
254
305
|
content = (
|
|
255
|
-
<ThemeProvider
|
|
306
|
+
<ThemeProvider
|
|
307
|
+
config={themeConfig}
|
|
308
|
+
initialTheme={resolved.metadata?.initialTheme}
|
|
309
|
+
>
|
|
256
310
|
{content}
|
|
257
311
|
</ThemeProvider>
|
|
258
312
|
);
|
|
259
313
|
}
|
|
260
314
|
|
|
315
|
+
// Wrap with NonceContext so client components (e.g. MetaTags) can
|
|
316
|
+
// apply CSP nonces to inline scripts during SSR. Always present to
|
|
317
|
+
// match the browser-side NavigationProvider tree shape for hydration.
|
|
318
|
+
content = (
|
|
319
|
+
<NonceContext.Provider value={nonce}>{content}</NonceContext.Provider>
|
|
320
|
+
);
|
|
321
|
+
|
|
261
322
|
// Wrap with NavigationStoreContext for useNavigation hook
|
|
262
323
|
return (
|
|
263
|
-
<NavigationStoreContext.Provider value={ssrContextValue}>
|
|
324
|
+
<NavigationStoreContext.Provider value={ssrContextValue!}>
|
|
264
325
|
{content}
|
|
265
326
|
</NavigationStoreContext.Provider>
|
|
266
327
|
);
|
|
@@ -278,12 +339,20 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
|
|
|
278
339
|
nonce,
|
|
279
340
|
});
|
|
280
341
|
|
|
342
|
+
// Wait for all Suspense boundaries to resolve when streamMode is "allReady".
|
|
343
|
+
// This buffers the entire HTML before flushing — used for bots that
|
|
344
|
+
// cannot process streamed HTML.
|
|
345
|
+
if (streamMode === "allReady") {
|
|
346
|
+
await htmlStream.allReady;
|
|
347
|
+
}
|
|
348
|
+
|
|
281
349
|
// Inject RSC payload into HTML as <script nonce="...">__FLIGHT_DATA__</script>
|
|
282
350
|
return htmlStream.pipeThrough(injectRSCPayload(rscStream2, { nonce }));
|
|
283
351
|
} catch (error) {
|
|
284
352
|
// Invoke onError callback if provided
|
|
285
353
|
if (onError) {
|
|
286
|
-
const errorObj =
|
|
354
|
+
const errorObj =
|
|
355
|
+
error instanceof Error ? error : new Error(String(error));
|
|
287
356
|
try {
|
|
288
357
|
onError(errorObj, { phase: "rendering" });
|
|
289
358
|
} catch (callbackError) {
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Static handler definition for build-time rendering of individual segments.
|
|
3
|
+
*
|
|
4
|
+
* Static wraps a handler so that in production the segment is
|
|
5
|
+
* rendered once at build time. The handler is then replaced with a static
|
|
6
|
+
* asset import -- no runtime store lookup needed.
|
|
7
|
+
*
|
|
8
|
+
* In dev mode, Static behaves as a normal handler: the wrapped
|
|
9
|
+
* function runs on every request, identical to a regular layout/path handler.
|
|
10
|
+
*
|
|
11
|
+
* The $$id is auto-generated by the Vite exposeInternalIds plugin based on
|
|
12
|
+
* file path and export name. No manual naming required.
|
|
13
|
+
*
|
|
14
|
+
* Key difference from Prerender:
|
|
15
|
+
* - Prerender: route-scoped, produces URLs via getParams, renders subtree per-params
|
|
16
|
+
* - Static: segment-scoped, renders once, no URLs, no params
|
|
17
|
+
*
|
|
18
|
+
* Works on: layout(), parallel(), and path().
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* ```ts
|
|
22
|
+
* export const DocsNav = Static((ctx) => <Nav docs={readDocsSync()} />);
|
|
23
|
+
* export const DocShell = Static((ctx) => <Shell />);
|
|
24
|
+
*
|
|
25
|
+
* urls(({ path, layout }) => [
|
|
26
|
+
* layout(DocsNav, () => [
|
|
27
|
+
* path("/getting-started", DocShell, { name: "doc.gs" }),
|
|
28
|
+
* path("/:slug", Prerender(getParams, DocPageHandler)),
|
|
29
|
+
* ]),
|
|
30
|
+
* ]);
|
|
31
|
+
* ```
|
|
32
|
+
*/
|
|
33
|
+
import type { ReactNode } from "react";
|
|
34
|
+
import type { Handler } from "./types.js";
|
|
35
|
+
import type { PrerenderOptions, StaticBuildContext } from "./prerender.js";
|
|
36
|
+
import { isCachedFunction } from "./cache/taint.js";
|
|
37
|
+
|
|
38
|
+
// -- Types ------------------------------------------------------------------
|
|
39
|
+
|
|
40
|
+
export interface StaticHandlerDefinition<
|
|
41
|
+
TParams extends Record<string, any> = any,
|
|
42
|
+
> {
|
|
43
|
+
readonly __brand: "staticHandler";
|
|
44
|
+
/** Auto-generated unique ID (injected by Vite plugin). */
|
|
45
|
+
$$id: string;
|
|
46
|
+
/** In dev mode, the actual handler function that layout/path/parallel can call. */
|
|
47
|
+
handler: Handler<TParams>;
|
|
48
|
+
/** Static handler options (passthrough support). */
|
|
49
|
+
options?: PrerenderOptions;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// -- Function ---------------------------------------------------------------
|
|
53
|
+
|
|
54
|
+
export function Static<TParams extends Record<string, any> = {}>(
|
|
55
|
+
handler: (ctx: StaticBuildContext) => ReactNode | Promise<ReactNode>,
|
|
56
|
+
options?: PrerenderOptions,
|
|
57
|
+
__injectedId?: string,
|
|
58
|
+
): StaticHandlerDefinition<TParams>;
|
|
59
|
+
|
|
60
|
+
// -- Implementation ---------------------------------------------------------
|
|
61
|
+
|
|
62
|
+
export function Static<TParams extends Record<string, any>>(
|
|
63
|
+
handler: Function,
|
|
64
|
+
optionsOrId?: PrerenderOptions | string,
|
|
65
|
+
maybeId?: string,
|
|
66
|
+
): StaticHandlerDefinition<TParams> {
|
|
67
|
+
if (isCachedFunction(handler)) {
|
|
68
|
+
throw new Error(
|
|
69
|
+
'A "use cache" function cannot be used as a Static() handler. ' +
|
|
70
|
+
"Static handlers are rendered once at build time. Remove the " +
|
|
71
|
+
'"use cache" directive — Static already provides caching.',
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
let options: PrerenderOptions | undefined;
|
|
76
|
+
let id: string;
|
|
77
|
+
|
|
78
|
+
if (typeof optionsOrId === "string") {
|
|
79
|
+
id = optionsOrId;
|
|
80
|
+
} else {
|
|
81
|
+
options = optionsOrId as PrerenderOptions | undefined;
|
|
82
|
+
id = maybeId ?? "";
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
if (!id) {
|
|
86
|
+
throw new Error(
|
|
87
|
+
"[rsc-router] Static: missing $$id. " +
|
|
88
|
+
"Ensure the exposeInternalIds Vite plugin is configured.",
|
|
89
|
+
);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return {
|
|
93
|
+
__brand: "staticHandler" as const,
|
|
94
|
+
$$id: id,
|
|
95
|
+
handler: handler as Handler<TParams>,
|
|
96
|
+
...(options ? { options } : {}),
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// -- Type guard -------------------------------------------------------------
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Type guard to check if a value is a StaticHandlerDefinition.
|
|
104
|
+
*/
|
|
105
|
+
export function isStaticHandler(
|
|
106
|
+
value: unknown,
|
|
107
|
+
): value is StaticHandlerDefinition {
|
|
108
|
+
return (
|
|
109
|
+
typeof value === "object" &&
|
|
110
|
+
value !== null &&
|
|
111
|
+
"__brand" in value &&
|
|
112
|
+
(value as { __brand: unknown }).__brand === "staticHandler"
|
|
113
|
+
);
|
|
114
|
+
}
|
|
@@ -11,7 +11,13 @@
|
|
|
11
11
|
* - Handles SSR hydration by deferring system theme detection
|
|
12
12
|
*/
|
|
13
13
|
|
|
14
|
-
import React, {
|
|
14
|
+
import React, {
|
|
15
|
+
useCallback,
|
|
16
|
+
useEffect,
|
|
17
|
+
useMemo,
|
|
18
|
+
useState,
|
|
19
|
+
useRef,
|
|
20
|
+
} from "react";
|
|
15
21
|
import { ThemeContext } from "./theme-context.js";
|
|
16
22
|
import type {
|
|
17
23
|
ResolvedTheme,
|
|
@@ -44,7 +50,12 @@ function readThemeFromCookie(storageKey: string): string | null {
|
|
|
44
50
|
for (const cookie of cookies) {
|
|
45
51
|
const [name, ...rest] = cookie.trim().split("=");
|
|
46
52
|
if (name === storageKey) {
|
|
47
|
-
|
|
53
|
+
const raw = rest.join("=");
|
|
54
|
+
try {
|
|
55
|
+
return decodeURIComponent(raw);
|
|
56
|
+
} catch {
|
|
57
|
+
return raw;
|
|
58
|
+
}
|
|
48
59
|
}
|
|
49
60
|
}
|
|
50
61
|
return null;
|
|
@@ -90,15 +101,13 @@ function writeThemeToStorage(storageKey: string, theme: Theme): void {
|
|
|
90
101
|
/**
|
|
91
102
|
* Apply theme to HTML element
|
|
92
103
|
*/
|
|
93
|
-
function applyThemeToDocument(
|
|
94
|
-
theme: Theme,
|
|
95
|
-
config: ResolvedThemeConfig
|
|
96
|
-
): void {
|
|
104
|
+
function applyThemeToDocument(theme: Theme, config: ResolvedThemeConfig): void {
|
|
97
105
|
if (typeof document === "undefined") return;
|
|
98
106
|
|
|
99
|
-
const resolved =
|
|
100
|
-
|
|
101
|
-
|
|
107
|
+
const resolved =
|
|
108
|
+
theme === "system" && config.enableSystem
|
|
109
|
+
? getSystemTheme()
|
|
110
|
+
: (theme as ResolvedTheme);
|
|
102
111
|
|
|
103
112
|
const value = config.value[resolved] || resolved;
|
|
104
113
|
const el = document.documentElement;
|
|
@@ -188,7 +197,7 @@ export function ThemeProvider({
|
|
|
188
197
|
writeThemeToStorage(config.storageKey, newTheme);
|
|
189
198
|
applyThemeToDocument(newTheme, config);
|
|
190
199
|
},
|
|
191
|
-
[config]
|
|
200
|
+
[config],
|
|
192
201
|
);
|
|
193
202
|
|
|
194
203
|
// Listen for system preference changes
|
|
@@ -227,10 +236,7 @@ export function ThemeProvider({
|
|
|
227
236
|
if (!newTheme) return;
|
|
228
237
|
|
|
229
238
|
// Validate and apply
|
|
230
|
-
if (
|
|
231
|
-
newTheme === "system" ||
|
|
232
|
-
config.themes.includes(newTheme)
|
|
233
|
-
) {
|
|
239
|
+
if (newTheme === "system" || config.themes.includes(newTheme)) {
|
|
234
240
|
setThemeState(newTheme as Theme);
|
|
235
241
|
applyThemeToDocument(newTheme as Theme, config);
|
|
236
242
|
}
|
|
@@ -280,7 +286,7 @@ export function ThemeProvider({
|
|
|
280
286
|
themes,
|
|
281
287
|
config,
|
|
282
288
|
}),
|
|
283
|
-
[theme, setTheme, resolvedTheme, systemTheme, themes, config, mounted]
|
|
289
|
+
[theme, setTheme, resolvedTheme, systemTheme, themes, config, mounted],
|
|
284
290
|
);
|
|
285
291
|
|
|
286
292
|
return (
|
|
@@ -49,13 +49,13 @@ export interface ThemeScriptProps {
|
|
|
49
49
|
* This renders a synchronous inline script that applies the theme
|
|
50
50
|
* to the HTML element before React hydration, preventing FOUC.
|
|
51
51
|
*/
|
|
52
|
-
export function ThemeScript({
|
|
52
|
+
export function ThemeScript({
|
|
53
|
+
config,
|
|
54
|
+
nonce,
|
|
55
|
+
}: ThemeScriptProps): React.ReactNode {
|
|
53
56
|
const scriptContent = generateThemeScript(config);
|
|
54
57
|
|
|
55
58
|
return (
|
|
56
|
-
<script
|
|
57
|
-
nonce={nonce}
|
|
58
|
-
dangerouslySetInnerHTML={{ __html: scriptContent }}
|
|
59
|
-
/>
|
|
59
|
+
<script nonce={nonce} dangerouslySetInnerHTML={{ __html: scriptContent }} />
|
|
60
60
|
);
|
|
61
61
|
}
|
package/src/theme/constants.ts
CHANGED
|
@@ -33,7 +33,9 @@ export const THEME_COOKIE: {
|
|
|
33
33
|
* Resolve theme config by applying defaults.
|
|
34
34
|
* Accepts `true` to enable with all defaults, or a config object.
|
|
35
35
|
*/
|
|
36
|
-
export function resolveThemeConfig(
|
|
36
|
+
export function resolveThemeConfig(
|
|
37
|
+
config: ThemeConfig | true,
|
|
38
|
+
): ResolvedThemeConfig {
|
|
37
39
|
// Handle `theme: true` shorthand
|
|
38
40
|
if (config === true) {
|
|
39
41
|
config = {};
|
|
@@ -53,7 +55,8 @@ export function resolveThemeConfig(config: ThemeConfig | true): ResolvedThemeCon
|
|
|
53
55
|
attribute: config.attribute ?? THEME_DEFAULTS.attribute,
|
|
54
56
|
storageKey: config.storageKey ?? THEME_DEFAULTS.storageKey,
|
|
55
57
|
enableSystem: config.enableSystem ?? THEME_DEFAULTS.enableSystem,
|
|
56
|
-
enableColorScheme:
|
|
58
|
+
enableColorScheme:
|
|
59
|
+
config.enableColorScheme ?? THEME_DEFAULTS.enableColorScheme,
|
|
57
60
|
value,
|
|
58
61
|
};
|
|
59
62
|
}
|