@rangojs/router 0.0.0-experimental.49 → 0.0.0-experimental.4fba1c4c

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/README.md +196 -43
  2. package/dist/bin/rango.js +269 -96
  3. package/dist/testing/vitest.js +48 -0
  4. package/dist/vite/index.js +2659 -883
  5. package/dist/vite/index.js.bak +5448 -0
  6. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  7. package/package.json +57 -11
  8. package/skills/breadcrumbs/SKILL.md +3 -1
  9. package/skills/bundle-analysis/SKILL.md +159 -0
  10. package/skills/cache-guide/SKILL.md +243 -21
  11. package/skills/caching/SKILL.md +118 -2
  12. package/skills/composability/SKILL.md +27 -2
  13. package/skills/document-cache/SKILL.md +78 -55
  14. package/skills/handler-use/SKILL.md +364 -0
  15. package/skills/hooks/SKILL.md +229 -20
  16. package/skills/host-router/SKILL.md +45 -20
  17. package/skills/i18n/SKILL.md +276 -0
  18. package/skills/intercept/SKILL.md +46 -4
  19. package/skills/layout/SKILL.md +28 -7
  20. package/skills/links/SKILL.md +249 -17
  21. package/skills/loader/SKILL.md +273 -53
  22. package/skills/middleware/SKILL.md +49 -12
  23. package/skills/migrate-nextjs/SKILL.md +562 -0
  24. package/skills/migrate-react-router/SKILL.md +769 -0
  25. package/skills/mime-routes/SKILL.md +27 -0
  26. package/skills/observability/SKILL.md +137 -0
  27. package/skills/parallel/SKILL.md +71 -6
  28. package/skills/prerender/SKILL.md +123 -100
  29. package/skills/rango/SKILL.md +242 -22
  30. package/skills/react-compiler/SKILL.md +168 -0
  31. package/skills/response-routes/SKILL.md +66 -9
  32. package/skills/route/SKILL.md +88 -4
  33. package/skills/router-setup/SKILL.md +90 -5
  34. package/skills/server-actions/SKILL.md +751 -0
  35. package/skills/streams-and-websockets/SKILL.md +283 -0
  36. package/skills/testing/SKILL.md +734 -0
  37. package/skills/typesafety/SKILL.md +329 -27
  38. package/skills/use-cache/SKILL.md +34 -5
  39. package/skills/view-transitions/SKILL.md +294 -0
  40. package/src/__augment-tests__/augment.ts +81 -0
  41. package/src/__augment-tests__/augmented.check.ts +117 -0
  42. package/src/__internal.ts +1 -1
  43. package/src/browser/action-coordinator.ts +53 -36
  44. package/src/browser/app-shell.ts +52 -0
  45. package/src/browser/app-version.ts +14 -0
  46. package/src/browser/event-controller.ts +86 -70
  47. package/src/browser/history-state.ts +21 -0
  48. package/src/browser/index.ts +3 -3
  49. package/src/browser/navigation-bridge.ts +101 -13
  50. package/src/browser/navigation-client.ts +125 -53
  51. package/src/browser/navigation-store.ts +75 -17
  52. package/src/browser/navigation-transaction.ts +10 -28
  53. package/src/browser/partial-update.ts +90 -30
  54. package/src/browser/prefetch/cache.ts +129 -21
  55. package/src/browser/prefetch/fetch.ts +156 -18
  56. package/src/browser/prefetch/queue.ts +92 -29
  57. package/src/browser/prefetch/resource-ready.ts +77 -0
  58. package/src/browser/rango-state.ts +53 -13
  59. package/src/browser/react/Link.tsx +72 -8
  60. package/src/browser/react/NavigationProvider.tsx +83 -33
  61. package/src/browser/react/context.ts +7 -2
  62. package/src/browser/react/filter-segment-order.ts +51 -7
  63. package/src/browser/react/index.ts +3 -0
  64. package/src/browser/react/location-state-shared.ts +175 -4
  65. package/src/browser/react/location-state.ts +39 -13
  66. package/src/browser/react/use-handle.ts +23 -64
  67. package/src/browser/react/use-navigation.ts +22 -2
  68. package/src/browser/react/use-params.ts +20 -8
  69. package/src/browser/react/use-reverse.ts +106 -0
  70. package/src/browser/react/use-router.ts +43 -10
  71. package/src/browser/react/use-segments.ts +11 -8
  72. package/src/browser/response-adapter.ts +25 -0
  73. package/src/browser/rsc-router.tsx +87 -22
  74. package/src/browser/scroll-restoration.ts +29 -19
  75. package/src/browser/segment-reconciler.ts +36 -14
  76. package/src/browser/segment-structure-assert.ts +2 -2
  77. package/src/browser/server-action-bridge.ts +31 -36
  78. package/src/browser/types.ts +48 -5
  79. package/src/build/collect-fallback-refs.ts +107 -0
  80. package/src/build/generate-manifest.ts +65 -40
  81. package/src/build/generate-route-types.ts +5 -0
  82. package/src/build/index.ts +2 -0
  83. package/src/build/route-trie.ts +52 -25
  84. package/src/build/route-types/codegen.ts +4 -4
  85. package/src/build/route-types/include-resolution.ts +9 -2
  86. package/src/build/route-types/per-module-writer.ts +7 -4
  87. package/src/build/route-types/router-processing.ts +266 -86
  88. package/src/build/route-types/scan-filter.ts +9 -2
  89. package/src/build/route-types/source-scan.ts +118 -0
  90. package/src/build/runtime-discovery.ts +9 -20
  91. package/src/cache/cache-scope.ts +74 -47
  92. package/src/cache/cf/cf-cache-store.ts +54 -13
  93. package/src/cache/taint.ts +55 -0
  94. package/src/client.rsc.tsx +3 -0
  95. package/src/client.tsx +94 -238
  96. package/src/context-var.ts +72 -2
  97. package/src/decode-loader-results.ts +36 -0
  98. package/src/errors.ts +30 -1
  99. package/src/handle.ts +65 -12
  100. package/src/host/index.ts +2 -2
  101. package/src/host/router.ts +129 -57
  102. package/src/host/types.ts +31 -2
  103. package/src/host/utils.ts +1 -1
  104. package/src/href-client.ts +140 -20
  105. package/src/index.rsc.ts +12 -5
  106. package/src/index.ts +61 -11
  107. package/src/loader-store.ts +500 -0
  108. package/src/loader.rsc.ts +21 -6
  109. package/src/loader.ts +3 -10
  110. package/src/missing-id-error.ts +68 -0
  111. package/src/outlet-context.ts +1 -1
  112. package/src/prerender/store.ts +5 -4
  113. package/src/prerender.ts +141 -80
  114. package/src/response-utils.ts +37 -0
  115. package/src/reverse.ts +65 -15
  116. package/src/route-content-wrapper.tsx +6 -28
  117. package/src/route-definition/dsl-helpers.ts +411 -261
  118. package/src/route-definition/helper-factories.ts +29 -139
  119. package/src/route-definition/helpers-types.ts +110 -34
  120. package/src/route-definition/index.ts +3 -0
  121. package/src/route-definition/redirect.ts +9 -1
  122. package/src/route-definition/resolve-handler-use.ts +155 -0
  123. package/src/route-definition/use-item-types.ts +32 -0
  124. package/src/route-types.ts +37 -41
  125. package/src/router/basename.ts +14 -0
  126. package/src/router/content-negotiation.ts +113 -1
  127. package/src/router/error-handling.ts +1 -1
  128. package/src/router/handler-context.ts +77 -38
  129. package/src/router/intercept-resolution.ts +13 -22
  130. package/src/router/lazy-includes.ts +8 -8
  131. package/src/router/loader-resolution.ts +174 -22
  132. package/src/router/manifest.ts +22 -13
  133. package/src/router/match-api.ts +128 -192
  134. package/src/router/match-handlers.ts +63 -20
  135. package/src/router/match-middleware/cache-lookup.ts +70 -97
  136. package/src/router/match-middleware/cache-store.ts +8 -2
  137. package/src/router/match-middleware/segment-resolution.ts +53 -0
  138. package/src/router/match-result.ts +103 -4
  139. package/src/router/metrics.ts +1 -1
  140. package/src/router/middleware-types.ts +21 -34
  141. package/src/router/middleware.ts +101 -89
  142. package/src/router/navigation-snapshot.ts +182 -0
  143. package/src/router/pattern-matching.ts +101 -17
  144. package/src/router/prerender-match.ts +110 -10
  145. package/src/router/preview-match.ts +32 -102
  146. package/src/router/request-classification.ts +286 -0
  147. package/src/router/revalidation.ts +58 -2
  148. package/src/router/route-snapshot.ts +245 -0
  149. package/src/router/router-interfaces.ts +77 -28
  150. package/src/router/router-options.ts +76 -11
  151. package/src/router/router-registry.ts +2 -5
  152. package/src/router/segment-resolution/fresh.ts +105 -13
  153. package/src/router/segment-resolution/helpers.ts +29 -24
  154. package/src/router/segment-resolution/revalidation.ts +236 -112
  155. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  156. package/src/router/substitute-pattern-params.ts +56 -0
  157. package/src/router/telemetry.ts +99 -0
  158. package/src/router/trie-matching.ts +18 -13
  159. package/src/router/types.ts +9 -0
  160. package/src/router/url-params.ts +49 -0
  161. package/src/router.ts +86 -22
  162. package/src/rsc/handler-context.ts +2 -2
  163. package/src/rsc/handler.ts +440 -381
  164. package/src/rsc/helpers.ts +91 -43
  165. package/src/rsc/index.ts +1 -1
  166. package/src/rsc/loader-fetch.ts +23 -3
  167. package/src/rsc/manifest-init.ts +5 -1
  168. package/src/rsc/origin-guard.ts +28 -10
  169. package/src/rsc/progressive-enhancement.ts +18 -2
  170. package/src/rsc/response-route-handler.ts +46 -53
  171. package/src/rsc/rsc-rendering.ts +41 -48
  172. package/src/rsc/runtime-warnings.ts +9 -10
  173. package/src/rsc/server-action.ts +25 -37
  174. package/src/rsc/ssr-setup.ts +18 -2
  175. package/src/rsc/types.ts +17 -3
  176. package/src/search-params.ts +4 -4
  177. package/src/segment-content-promise.ts +67 -0
  178. package/src/segment-loader-promise.ts +122 -0
  179. package/src/segment-system.tsx +132 -116
  180. package/src/serialize.ts +243 -0
  181. package/src/server/context.ts +190 -51
  182. package/src/server/cookie-store.ts +28 -4
  183. package/src/server/handle-store.ts +19 -0
  184. package/src/server/loader-registry.ts +9 -8
  185. package/src/server/request-context.ts +195 -57
  186. package/src/ssr/index.tsx +8 -1
  187. package/src/static-handler.ts +19 -7
  188. package/src/testing/cache-status.ts +166 -0
  189. package/src/testing/collect-handle.ts +63 -0
  190. package/src/testing/dispatch.ts +440 -0
  191. package/src/testing/dom.entry.ts +22 -0
  192. package/src/testing/e2e/fixture.ts +154 -0
  193. package/src/testing/e2e/index.ts +149 -0
  194. package/src/testing/e2e/matchers.ts +51 -0
  195. package/src/testing/e2e/page-helpers.ts +272 -0
  196. package/src/testing/e2e/parity.ts +306 -0
  197. package/src/testing/e2e/server.ts +183 -0
  198. package/src/testing/flight-matchers.ts +104 -0
  199. package/src/testing/flight-runtime.d.ts +21 -0
  200. package/src/testing/flight.entry.ts +22 -0
  201. package/src/testing/flight.ts +182 -0
  202. package/src/testing/generated-routes.ts +223 -0
  203. package/src/testing/index.ts +106 -0
  204. package/src/testing/internal/context.ts +304 -0
  205. package/src/testing/render-route.tsx +565 -0
  206. package/src/testing/run-loader.ts +341 -0
  207. package/src/testing/run-middleware.ts +179 -0
  208. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  209. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  210. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  211. package/src/testing/vitest-stubs/version.ts +5 -0
  212. package/src/testing/vitest.ts +183 -0
  213. package/src/types/cache-types.ts +4 -4
  214. package/src/types/global-namespace.ts +39 -26
  215. package/src/types/handler-context.ts +103 -67
  216. package/src/types/index.ts +1 -0
  217. package/src/types/loader-types.ts +41 -15
  218. package/src/types/request-scope.ts +126 -0
  219. package/src/types/route-entry.ts +12 -1
  220. package/src/types/segments.ts +36 -2
  221. package/src/urls/include-helper.ts +34 -67
  222. package/src/urls/index.ts +0 -3
  223. package/src/urls/path-helper-types.ts +50 -9
  224. package/src/urls/path-helper.ts +63 -63
  225. package/src/urls/pattern-types.ts +48 -19
  226. package/src/urls/response-types.ts +25 -22
  227. package/src/urls/type-extraction.ts +26 -116
  228. package/src/urls/urls-function.ts +1 -5
  229. package/src/use-loader.tsx +487 -44
  230. package/src/vite/debug.ts +185 -0
  231. package/src/vite/discovery/bundle-postprocess.ts +34 -37
  232. package/src/vite/discovery/discover-routers.ts +105 -51
  233. package/src/vite/discovery/discovery-errors.ts +194 -0
  234. package/src/vite/discovery/gate-state.ts +171 -0
  235. package/src/vite/discovery/prerender-collection.ts +188 -93
  236. package/src/vite/discovery/route-types-writer.ts +40 -84
  237. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  238. package/src/vite/discovery/state.ts +46 -4
  239. package/src/vite/discovery/virtual-module-codegen.ts +13 -23
  240. package/src/vite/index.ts +6 -0
  241. package/src/vite/plugin-types.ts +126 -4
  242. package/src/vite/plugins/cjs-to-esm.ts +8 -7
  243. package/src/vite/plugins/client-ref-dedup.ts +16 -0
  244. package/src/vite/plugins/client-ref-hashing.ts +28 -5
  245. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  246. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  247. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  248. package/src/vite/plugins/expose-action-id.ts +54 -30
  249. package/src/vite/plugins/expose-id-utils.ts +24 -8
  250. package/src/vite/plugins/expose-ids/export-analysis.ts +100 -20
  251. package/src/vite/plugins/expose-ids/handler-transform.ts +12 -35
  252. package/src/vite/plugins/expose-ids/loader-transform.ts +3 -5
  253. package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
  254. package/src/vite/plugins/expose-internal-ids.ts +544 -317
  255. package/src/vite/plugins/performance-tracks.ts +92 -0
  256. package/src/vite/plugins/refresh-cmd.ts +88 -26
  257. package/src/vite/plugins/use-cache-transform.ts +65 -50
  258. package/src/vite/plugins/version-injector.ts +39 -23
  259. package/src/vite/plugins/version-plugin.ts +59 -2
  260. package/src/vite/plugins/virtual-entries.ts +2 -2
  261. package/src/vite/rango.ts +130 -26
  262. package/src/vite/router-discovery.ts +920 -129
  263. package/src/vite/utils/ast-handler-extract.ts +15 -15
  264. package/src/vite/utils/banner.ts +1 -1
  265. package/src/vite/utils/bundle-analysis.ts +4 -2
  266. package/src/vite/utils/client-chunks.ts +190 -0
  267. package/src/vite/utils/forward-user-plugins.ts +193 -0
  268. package/src/vite/utils/manifest-utils.ts +21 -5
  269. package/src/vite/utils/package-resolution.ts +41 -1
  270. package/src/vite/utils/prerender-utils.ts +38 -5
  271. package/src/vite/utils/shared-utils.ts +109 -27
  272. package/src/browser/action-response-classifier.ts +0 -99
@@ -108,6 +108,33 @@ path.text("/api/data", () => "plain text version", { name: "dataText" }),
108
108
  Without an RSC primary, there is no `text/html` candidate — the Accept header
109
109
  picks among the response-type candidates directly.
110
110
 
111
+ ## Type Safety For Negotiated Paths
112
+
113
+ `router.named-routes.gen.ts` validates route names, params, search, `href()`, and
114
+ the `Rango.Path` type, but it does not carry response payload metadata. For MIME or
115
+ response payload types, use one of these surfaces:
116
+
117
+ - `RouteResponse<typeof patterns, "routeName">` for a specific response variant
118
+ by route name. This is the clearest option when several MIME variants share
119
+ one URL pattern.
120
+ - `Rango.PathResponse<"/products/:id">` (ambient, no import) for global lookup by URL pattern or concrete path after the app
121
+ registers `typeof router.routeMap`:
122
+
123
+ ```typescript
124
+ // router.tsx
125
+ export const router = createRouter({ document: Document }).routes(urlpatterns);
126
+
127
+ declare global {
128
+ namespace Rango {
129
+ interface RegisteredRoutes extends typeof router.routeMap {}
130
+ }
131
+ }
132
+ ```
133
+
134
+ `RegisteredRoutes` is what exposes the richer routeMap entries containing
135
+ response payload metadata. Without it, URL-pattern response lookup has paths but
136
+ no payloads, so response types resolve to `ResponseEnvelope<never>`.
137
+
111
138
  ## How It Works
112
139
 
113
140
  1. **Build time**: `buildRouteTrie()` calls `mergeLeaves()` when multiple routes share a pattern.
@@ -0,0 +1,137 @@
1
+ ---
2
+ name: observability
3
+ description: Debug Rango request performance with debugPerformance, Server-Timing, structured telemetry, and tracing
4
+ argument-hint:
5
+ ---
6
+
7
+ # Observability
8
+
9
+ Use this when you need to understand request latency, cache decisions,
10
+ revalidation behavior, loader overlap, or production traces.
11
+
12
+ Rango exposes two complementary observability surfaces:
13
+
14
+ 1. **Performance timeline** (`debugPerformance`) — per-request waterfall for
15
+ local or targeted debugging. It prints to the console and emits
16
+ `Server-Timing`.
17
+ 2. **Structured telemetry** (`telemetry`) — lifecycle events sent to a pluggable
18
+ sink for production monitoring, OpenTelemetry, or custom metrics.
19
+
20
+ The essentials are below. The exported `TelemetryEvent` union type
21
+ (`import type { TelemetryEvent } from "@rangojs/router"`) is the full event
22
+ contract — every event kind and its fields are typed there.
23
+
24
+ ## Performance timeline
25
+
26
+ Enable globally while debugging:
27
+
28
+ ```typescript
29
+ import { createRouter } from "@rangojs/router";
30
+
31
+ const router = createRouter({
32
+ document: Document,
33
+ urls: urlpatterns,
34
+ debugPerformance: true,
35
+ });
36
+ ```
37
+
38
+ Or enable for selected requests from middleware:
39
+
40
+ ```typescript
41
+ middleware(async (ctx, next) => {
42
+ if (ctx.url.searchParams.has("debug")) {
43
+ ctx.debugPerformance();
44
+ }
45
+ await next();
46
+ });
47
+ ```
48
+
49
+ Call `ctx.debugPerformance()` before `await next()`. The request then prints a
50
+ shared-axis waterfall and adds a `Server-Timing` header.
51
+
52
+ Read the timeline as intervals:
53
+
54
+ - `handler:total` is the whole router request.
55
+ - `render:total` / `ssr-render-html` show the render pass.
56
+ - `loader:*` rows should overlap render work. If a loader starts only after the
57
+ render bar, it is serialized latency.
58
+ - Cache, route matching, middleware pre/post, RSC serialization, and SSR phases
59
+ appear as separate spans, so the slow phase is visible without guessing.
60
+
61
+ ## Structured telemetry
62
+
63
+ Use telemetry when you want durable production events rather than a one-request
64
+ debug waterfall.
65
+
66
+ ```typescript
67
+ import { createRouter, createConsoleSink } from "@rangojs/router";
68
+
69
+ const router = createRouter({
70
+ document: Document,
71
+ urls: urlpatterns,
72
+ telemetry: createConsoleSink(),
73
+ });
74
+ ```
75
+
76
+ For OpenTelemetry:
77
+
78
+ ```typescript
79
+ import { createRouter, createOTelSink } from "@rangojs/router";
80
+ import { trace } from "@opentelemetry/api";
81
+
82
+ const router = createRouter({
83
+ document: Document,
84
+ urls: urlpatterns,
85
+ telemetry: createOTelSink(trace.getTracer("my-app")),
86
+ });
87
+ ```
88
+
89
+ Custom sinks implement `emit(event)`:
90
+
91
+ ```typescript
92
+ import { createRouter } from "@rangojs/router";
93
+
94
+ const router = createRouter({
95
+ document: Document,
96
+ urls: urlpatterns,
97
+ telemetry: {
98
+ emit(event) {
99
+ myMetrics.record(event);
100
+ },
101
+ },
102
+ });
103
+ ```
104
+
105
+ Events include `request.start/end/error`, `loader.start/end/error`,
106
+ `handler.error`, `cache.decision`, and `revalidation.decision`.
107
+
108
+ ## Debugging revalidation and stale data
109
+
110
+ When stale UI or unexpected partial renders are the question, use all three
111
+ layers together:
112
+
113
+ ```typescript
114
+ import { createConsoleSink, createRouter } from "@rangojs/router";
115
+
116
+ const router = createRouter({
117
+ document: Document,
118
+ urls: urlpatterns,
119
+ debugPerformance: true,
120
+ telemetry: createConsoleSink(),
121
+ });
122
+ ```
123
+
124
+ Then inspect:
125
+
126
+ - `revalidation.decision` telemetry to see which segment re-ran or skipped.
127
+ - cache spans / `cache.decision` events to see hit, miss, stale, and background
128
+ revalidation behavior.
129
+ - loader spans to confirm live loaders overlap the render rather than blocking
130
+ first paint.
131
+ - the `Server-Timing` header to compare local logs with browser-network timing.
132
+
133
+ ## Zero-overhead defaults
134
+
135
+ `debugPerformance` is off by default, and `telemetry` emits nothing unless a sink
136
+ is configured. Per-request `ctx.debugPerformance()` lets you turn on the
137
+ waterfall only for the route, user, or query param you are investigating.
@@ -8,9 +8,6 @@ argument-hint: [@slot-name]
8
8
 
9
9
  Parallel routes render multiple components simultaneously in named slots.
10
10
 
11
- Canonical semantics reference:
12
- [docs/execution-model.md](../../docs/internal/execution-model.md)
13
-
14
11
  ## Basic Parallel Routes
15
12
 
16
13
  ```typescript
@@ -206,6 +203,67 @@ parallel(
206
203
  )
207
204
  ```
208
205
 
206
+ ## Composable Slots via `handler.use`
207
+
208
+ Slot handlers can carry their own loader, loading, error/notFound boundaries, revalidation, and transition defaults via `.use`. The mount site then declares **just the slot names** — no per-call data wiring.
209
+
210
+ ```typescript
211
+ const CartSummary: Handler = async (ctx) => {
212
+ const cart = await ctx.use(CartLoader);
213
+ return <CartSummaryView cart={cart} />;
214
+ };
215
+ CartSummary.use = () => [
216
+ loader(CartLoader),
217
+ loading(<CartSkeleton />),
218
+ revalidate(revalidateCartData),
219
+ ];
220
+
221
+ // Same slot, no copy-pasted plumbing across layouts.
222
+ layout(<DashboardLayout />, () => [
223
+ parallel({ "@cart": CartSummary }),
224
+ path("/dashboard", DashboardIndex, { name: "dashboard.index" }),
225
+ ]);
226
+
227
+ layout(<AccountLayout />, () => [
228
+ parallel({ "@cart": CartSummary }),
229
+ path("/account", AccountIndex, { name: "account.index" }),
230
+ ]);
231
+ ```
232
+
233
+ A slot's `loading()` (whether from `handler.use` or explicit) makes that slot an independent streaming unit, exactly as in the **Streaming Behavior** section above.
234
+
235
+ The `parallel` mount site has the narrowest allow-list for `handler.use` items — slots cannot bring their own middleware or layout, only `revalidate`, `loader`, `loading`, `errorBoundary`, `notFoundBoundary`, and `transition`. See [skills/handler-use](../handler-use/SKILL.md) for the full table and merge rules.
236
+
237
+ `transition` is allowed in the slot allow-list, but slot-level rendering does **not** currently apply a `<ViewTransition>` wrapper — only the layout/route wraps take effect at render time. For a modal-only morph today, use an element-level React `<ViewTransition>` inside the slot's component. The reverse direction is the useful guarantee: a layout-level `transition()` fires when the layout's default outlet content changes but **not** when a `<ParallelOutlet />` mounts new content (modal opens are not subtree updates of the layout VT). See [skills/view-transitions](../view-transitions/SKILL.md) for the wrap rules and the intercept caveat.
238
+
239
+ ### Two scopes for explicit `use`: shared (broadcast) and slot-local
240
+
241
+ `parallel({...slots}, () => [...use])` runs the shared `use()` callback **once per slot** ([dsl-helpers.ts](../../src/route-definition/dsl-helpers.ts)) — items in that callback land on every slot's entry. That's the right behavior for the items the parallel allow-list permits and that accumulate (`loader`, `revalidate`, `errorBoundary`, `notFoundBoundary`, `transition`). (Slots cannot bring `middleware` or `layout` — see the allowed-types note above.)
242
+
243
+ For single-assignment items like `loading()`, broadcasting overwrites every slot's `handler.use` default. Pass a **slot descriptor** `{ handler, use }` instead — items in the descriptor's `use` apply only to that slot:
244
+
245
+ ```typescript
246
+ // @cart gets a custom skeleton; @notifs keeps its handler.use default.
247
+ parallel({
248
+ "@cart": {
249
+ handler: Cart,
250
+ use: () => [loading(<CustomCartSkeleton />)],
251
+ },
252
+ "@notifs": Notifs,
253
+ });
254
+
255
+ // Opt one slot out of streaming while siblings still stream the broadcast.
256
+ parallel(
257
+ {
258
+ "@cart": { handler: Cart, use: () => [loading(false)] },
259
+ "@notifs": Notifs,
260
+ },
261
+ () => [loading(<BroadcastSkeleton />)],
262
+ );
263
+ ```
264
+
265
+ Per-slot merge order is **handler.use → shared use → slot-local use**. Slot-local is the narrowest scope, so it wins for last-write-wins items. See [skills/handler-use § `loading()` is a single-assignment item — scope it correctly](../handler-use/SKILL.md#loading-is-a-single-assignment-item--scope-it-correctly) for the full reasoning.
266
+
209
267
  ## Slot Override Semantics
210
268
 
211
269
  When multiple `parallel()` calls define the same slot name, **the last
@@ -279,7 +337,7 @@ parallel(
279
337
  () => [
280
338
  loader(CartLoader),
281
339
  // Revalidate when cart actions occur
282
- revalidate(({ actionId }) => actionId?.includes("Cart") ?? false),
340
+ revalidate(({ actionId }) => actionId?.includes("Cart") || undefined),
283
341
  ]
284
342
  )
285
343
  ```
@@ -288,6 +346,13 @@ Revalidating only the parallel does not re-run outer handlers/layouts.
288
346
  If the slot reads `ctx.get()` data established above it, opt the outer
289
347
  segment into revalidation as well.
290
348
 
349
+ A `revalidate()` callback may return a hard `boolean`, a soft
350
+ `{ defaultShouldRevalidate }` object, or nothing (`void` / `null` /
351
+ `undefined`) to defer to the next revalidator. See
352
+ [loader/SKILL.md#revalidate-return-shapes](../loader/SKILL.md#revalidate-return-shapes)
353
+ for the full contract — it's the same across `loader()`, `path()`,
354
+ `layout()`, `parallel()`, and `intercept()`.
355
+
291
356
  ### Revalidation Contracts for Parallel Dependencies
292
357
 
293
358
  Prefer named revalidation contracts shared by both the upstream producer and
@@ -296,7 +361,7 @@ the parallel consumer:
296
361
  ```typescript
297
362
  // revalidation-contracts.ts
298
363
  export const revalidateCartData = ({ actionId }) =>
299
- actionId?.includes("src/actions/cart.ts#") ?? false;
364
+ actionId?.includes("src/actions/cart.ts#") || undefined;
300
365
 
301
366
  layout(CartLayout, () => [
302
367
  revalidate(revalidateCartData), // producer reruns
@@ -414,7 +479,7 @@ export const shopPatterns = urls(({
414
479
  () => [
415
480
  loader(CartLoader),
416
481
  loading(<CartSkeleton />),
417
- revalidate(({ actionId }) => actionId?.includes("Cart") ?? false),
482
+ revalidate(({ actionId }) => actionId?.includes("Cart") || undefined),
418
483
  ]
419
484
  ),
420
485