toiljs 0.0.15 → 0.0.16

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 (217) hide show
  1. package/.babelrc +13 -13
  2. package/.gitattributes +2 -2
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +38 -38
  4. package/.github/ISSUE_TEMPLATE/bug_report.yml +90 -90
  5. package/.github/ISSUE_TEMPLATE/config.yml +8 -8
  6. package/.github/ISSUE_TEMPLATE/feature_request.md +20 -20
  7. package/.github/PULL_REQUEST_TEMPLATE.md +43 -43
  8. package/.github/changelog-config.json +45 -45
  9. package/.github/dependabot.yml +27 -27
  10. package/.github/workflows/ci.yml +191 -191
  11. package/.prettierrc.json +11 -11
  12. package/.vscode/settings.json +9 -9
  13. package/CHANGELOG.md +5 -5
  14. package/LICENSE +187 -187
  15. package/README.md +339 -315
  16. package/as-pect.asconfig.json +34 -34
  17. package/as-pect.config.js +65 -65
  18. package/assets/logo.svg +36 -36
  19. package/build/backend/.tsbuildinfo +1 -1
  20. package/build/cli/.tsbuildinfo +1 -1
  21. package/build/cli/index.js +0 -0
  22. package/build/client/.tsbuildinfo +1 -1
  23. package/build/client/dev/devtools.d.ts +6 -0
  24. package/build/client/dev/devtools.js +442 -0
  25. package/build/client/dev/error-overlay.d.ts +9 -0
  26. package/build/client/dev/error-overlay.js +19 -4
  27. package/build/client/navigation/prefetch.d.ts +1 -0
  28. package/build/client/navigation/prefetch.js +35 -0
  29. package/build/client/routing/Router.js +1 -1
  30. package/build/client/routing/hooks.js +6 -2
  31. package/build/client/routing/loader.d.ts +23 -0
  32. package/build/client/routing/loader.js +53 -7
  33. package/build/client/routing/mount.js +4 -3
  34. package/build/compiler/.tsbuildinfo +1 -1
  35. package/build/compiler/config.d.ts +16 -0
  36. package/build/compiler/config.js +7 -0
  37. package/build/compiler/docs.js +16 -16
  38. package/build/compiler/index.d.ts +2 -2
  39. package/build/compiler/index.js +1 -1
  40. package/build/compiler/plugin.js +156 -0
  41. package/build/compiler/prerender.d.ts +1 -0
  42. package/build/compiler/prerender.js +1 -1
  43. package/build/compiler/seo.d.ts +1 -1
  44. package/build/compiler/seo.js +5 -4
  45. package/build/compiler/ssg.js +32 -1
  46. package/build/io/.tsbuildinfo +1 -1
  47. package/build/logger/.tsbuildinfo +1 -1
  48. package/build/shared/.tsbuildinfo +1 -1
  49. package/eslint.config.js +48 -48
  50. package/examples/basic/client/404.tsx +11 -11
  51. package/examples/basic/client/components/.gitkeep +1 -1
  52. package/examples/basic/client/global-error.tsx +13 -13
  53. package/examples/basic/client/layout.tsx +25 -25
  54. package/examples/basic/client/public/images/.gitkeep +1 -1
  55. package/examples/basic/client/public/images/logo.svg +36 -36
  56. package/examples/basic/client/public/robots.txt +2 -2
  57. package/examples/basic/client/routes/docs/[...slug].tsx +12 -12
  58. package/examples/basic/client/routes/features/error/error.tsx +16 -16
  59. package/examples/basic/client/routes/features/template/b.tsx +14 -14
  60. package/examples/basic/client/routes/files/[[...slug]].tsx +21 -21
  61. package/examples/basic/client/routes/gallery/layout.tsx +13 -13
  62. package/examples/basic/client/routes/io.tsx +24 -24
  63. package/examples/basic/client/routes/loader-demo/loading.tsx +13 -13
  64. package/examples/basic/client/routes/search.tsx +61 -61
  65. package/examples/basic/client/toil.tsx +5 -5
  66. package/package.json +155 -148
  67. package/presets/eslint.js +88 -88
  68. package/presets/no-uint8array-tostring.js +200 -200
  69. package/presets/prettier.json +18 -18
  70. package/presets/tsconfig.json +37 -37
  71. package/src/backend/index.ts +160 -160
  72. package/src/cli/proc.ts +50 -50
  73. package/src/cli/updates.ts +69 -69
  74. package/src/cli/validate.ts +31 -31
  75. package/src/client/channel/channel.ts +146 -146
  76. package/src/client/components/Form.tsx +65 -65
  77. package/src/client/components/Script.tsx +113 -113
  78. package/src/client/components/Slot.tsx +21 -21
  79. package/src/client/dev/devtools.tsx +973 -0
  80. package/src/client/dev/error-overlay.tsx +30 -4
  81. package/src/client/head/head.ts +167 -167
  82. package/src/client/head/metadata.ts +112 -112
  83. package/src/client/index.ts +89 -89
  84. package/src/client/navigation/NavLink.tsx +86 -86
  85. package/src/client/navigation/navigation.ts +235 -235
  86. package/src/client/navigation/prefetch.ts +169 -130
  87. package/src/client/navigation/scroll.ts +53 -53
  88. package/src/client/routing/Router.tsx +8 -2
  89. package/src/client/routing/action.ts +122 -122
  90. package/src/client/routing/error-boundary.tsx +43 -43
  91. package/src/client/routing/hooks.ts +21 -6
  92. package/src/client/routing/loader.ts +325 -235
  93. package/src/client/routing/match.ts +47 -47
  94. package/src/client/routing/mount.tsx +54 -52
  95. package/src/client/routing/params-context.ts +10 -10
  96. package/src/client/routing/slot-context.ts +7 -7
  97. package/src/client/search/search.ts +189 -189
  98. package/src/client/search/use-page-search.ts +73 -73
  99. package/src/client/types.ts +73 -73
  100. package/src/compiler/config.ts +219 -182
  101. package/src/compiler/docs.ts +228 -228
  102. package/src/compiler/generate.ts +394 -394
  103. package/src/compiler/index.ts +64 -57
  104. package/src/compiler/pages.ts +70 -70
  105. package/src/compiler/plugin.ts +170 -2
  106. package/src/compiler/prerender.ts +156 -156
  107. package/src/compiler/seo.ts +397 -390
  108. package/src/compiler/ssg.ts +162 -126
  109. package/src/io/BinaryReader.ts +340 -340
  110. package/src/io/BinaryWriter.ts +385 -385
  111. package/src/io/FastMap.ts +127 -127
  112. package/src/io/index.ts +11 -11
  113. package/src/io/lengths.ts +14 -14
  114. package/src/io/types.ts +18 -18
  115. package/src/logger/index.ts +22 -22
  116. package/src/server/index.ts +10 -10
  117. package/src/server/main.ts +13 -13
  118. package/src/server/tsconfig.json +4 -4
  119. package/src/shared/index.ts +10 -10
  120. package/std/client/index.d.ts +15 -15
  121. package/std/client/package.json +3 -3
  122. package/test/assembly/example.spec.ts +7 -7
  123. package/test/channel.test.ts +21 -21
  124. package/test/dom/Link.test.tsx +47 -47
  125. package/test/dom/NavLink.test.tsx +37 -37
  126. package/test/dom/error-overlay.test.tsx +44 -44
  127. package/test/dom/loader.test.tsx +121 -121
  128. package/test/dom/navigation.test.ts +59 -59
  129. package/test/dom/revalidate.test.tsx +38 -38
  130. package/test/dom/route-head.test.tsx +78 -78
  131. package/test/dom/router-loading.test.tsx +44 -44
  132. package/test/dom/scroll.test.ts +56 -56
  133. package/test/dom/use-metadata.test.tsx +58 -58
  134. package/test/io.test.ts +93 -93
  135. package/test/navlink.test.ts +28 -28
  136. package/test/placeholder.test.ts +9 -9
  137. package/test/routes.test.ts +76 -76
  138. package/test/seo.test.ts +175 -164
  139. package/test/slot-layouts.test.ts +69 -69
  140. package/test/ssg.test.ts +36 -36
  141. package/test/update.test.ts +44 -44
  142. package/test/validate.test.ts +42 -42
  143. package/toil-routes.d.ts +7 -0
  144. package/toilconfig.json +30 -30
  145. package/tsconfig.backend.json +13 -13
  146. package/tsconfig.base.json +35 -35
  147. package/tsconfig.cli.json +13 -13
  148. package/tsconfig.client.json +14 -14
  149. package/tsconfig.compiler.json +13 -13
  150. package/tsconfig.io.json +12 -12
  151. package/tsconfig.json +22 -22
  152. package/tsconfig.logger.json +12 -12
  153. package/tsconfig.server.json +10 -10
  154. package/tsconfig.shared.json +12 -12
  155. package/vitest.config.ts +26 -26
  156. package/.idea/codeStyles/Project.xml +0 -54
  157. package/.idea/codeStyles/codeStyleConfig.xml +0 -5
  158. package/.idea/inspectionProfiles/Project_Default.xml +0 -6
  159. package/.idea/modules.xml +0 -8
  160. package/.idea/prettier.xml +0 -7
  161. package/.idea/toiljs.iml +0 -8
  162. package/.idea/vcs.xml +0 -6
  163. package/.toil/entry.tsx +0 -9
  164. package/.toil/index.html +0 -12
  165. package/.toil/routes.ts +0 -9
  166. package/build/cli/configure.d.ts +0 -16
  167. package/build/cli/configure.js +0 -272
  168. package/build/cli/create.d.ts +0 -16
  169. package/build/cli/create.js +0 -420
  170. package/build/cli/diagnostics.d.ts +0 -55
  171. package/build/cli/diagnostics.js +0 -333
  172. package/build/cli/doctor.d.ts +0 -6
  173. package/build/cli/doctor.js +0 -249
  174. package/build/cli/features.d.ts +0 -25
  175. package/build/cli/features.js +0 -107
  176. package/build/cli/index.d.ts +0 -2
  177. package/build/cli/proc.d.ts +0 -6
  178. package/build/cli/proc.js +0 -31
  179. package/build/cli/ui.d.ts +0 -9
  180. package/build/cli/ui.js +0 -75
  181. package/build/cli/update.d.ts +0 -7
  182. package/build/cli/update.js +0 -117
  183. package/build/cli/updates.d.ts +0 -10
  184. package/build/cli/updates.js +0 -45
  185. package/build/cli/validate.d.ts +0 -4
  186. package/build/cli/validate.js +0 -19
  187. package/build/client/Link.d.ts +0 -8
  188. package/build/client/Link.js +0 -44
  189. package/build/client/NavLink.d.ts +0 -14
  190. package/build/client/NavLink.js +0 -37
  191. package/build/client/Router.d.ts +0 -7
  192. package/build/client/Router.js +0 -55
  193. package/build/client/channel.d.ts +0 -23
  194. package/build/client/channel.js +0 -94
  195. package/build/client/error-boundary.d.ts +0 -16
  196. package/build/client/error-boundary.js +0 -19
  197. package/build/client/head.d.ts +0 -26
  198. package/build/client/head.js +0 -87
  199. package/build/client/hooks.d.ts +0 -17
  200. package/build/client/hooks.js +0 -48
  201. package/build/client/lazy.d.ts +0 -16
  202. package/build/client/lazy.js +0 -53
  203. package/build/client/match.d.ts +0 -2
  204. package/build/client/match.js +0 -32
  205. package/build/client/mount.d.ts +0 -2
  206. package/build/client/mount.js +0 -13
  207. package/build/client/navigation.d.ts +0 -13
  208. package/build/client/navigation.js +0 -97
  209. package/build/client/params-context.d.ts +0 -2
  210. package/build/client/params-context.js +0 -2
  211. package/build/client/prefetch.d.ts +0 -11
  212. package/build/client/prefetch.js +0 -100
  213. package/build/client/runtime.d.ts +0 -31
  214. package/build/client/runtime.js +0 -112
  215. package/build/client/scroll.d.ts +0 -8
  216. package/build/client/scroll.js +0 -36
  217. package/toil-env.d.ts +0 -16
@@ -1,73 +1,73 @@
1
- /**
2
- * React binding for the page-metadata {@link searchPages search}. Gives a route component reactive,
3
- * memoized search results plus a `goTo` helper that navigates straight to a matched page, a drop-in
4
- * for a site-wide "jump to page" / command-palette style search box.
5
- */
6
- import { useMemo } from 'react';
7
-
8
- import { navigate, type NavigateOptions } from '../navigation/navigation.js';
9
- import type { Href } from '../types.js';
10
- import {
11
- getPages,
12
- type PageMeta,
13
- pagePath,
14
- type PageSearchOptions,
15
- type PageSearchResult,
16
- searchPages,
17
- } from './search.js';
18
-
19
- /** What {@link usePageSearch} returns. */
20
- export interface PageSearch {
21
- /** Ranked matches for the current query (best first); empty when the query is blank. */
22
- readonly results: readonly PageSearchResult[];
23
- /** The full registered page index (handy for rendering an "all pages" listing). */
24
- readonly pages: readonly PageMeta[];
25
- /**
26
- * Navigates to a result / page / raw path. A dynamic (`:param`) page can't be navigated to
27
- * as-is, so passing one (or its result) is a no-op unless you pass a concrete path string with
28
- * the params already filled in. A stable reference, safe to destructure.
29
- */
30
- readonly goTo: (
31
- target: string | PageMeta | PageSearchResult,
32
- options?: NavigateOptions,
33
- ) => void;
34
- }
35
-
36
- /** Whether a path can be navigated to directly (no unfilled dynamic segments). */
37
- function isNavigable(path: string): boolean {
38
- return !/[:*]/.test(path);
39
- }
40
-
41
- /**
42
- * Searches the project's pages by `query` and returns ranked {@link PageSearchResult}s, recomputed
43
- * only when the query or options change. Use the returned `goTo` to redirect to a match:
44
- *
45
- * ```tsx
46
- * const { results, goTo } = usePageSearch(query);
47
- * return results.map((r) => (
48
- * <button key={r.page.path} onClick={() => { goTo(r); }}>{r.page.metadata.title ?? r.page.path}</button>
49
- * ));
50
- * ```
51
- */
52
- export function usePageSearch(query: string, options: PageSearchOptions = {}): PageSearch {
53
- const { limit, includeDynamic, fields } = options;
54
- const fieldsKey = fields?.join(',');
55
- const results = useMemo(
56
- () => searchPages(query, { limit, includeDynamic, fields }),
57
- // `fields` is compared by content (fieldsKey) so a fresh array literal each render is fine.
58
- [query, limit, includeDynamic, fieldsKey],
59
- );
60
-
61
- return useMemo<PageSearch>(
62
- () => ({
63
- results,
64
- pages: getPages(),
65
- goTo(target, navOptions) {
66
- const path = pagePath(target);
67
- if (typeof target !== 'string' && !isNavigable(path)) return;
68
- navigate(path as Href, navOptions);
69
- },
70
- }),
71
- [results],
72
- );
73
- }
1
+ /**
2
+ * React binding for the page-metadata {@link searchPages search}. Gives a route component reactive,
3
+ * memoized search results plus a `goTo` helper that navigates straight to a matched page, a drop-in
4
+ * for a site-wide "jump to page" / command-palette style search box.
5
+ */
6
+ import { useMemo } from 'react';
7
+
8
+ import { navigate, type NavigateOptions } from '../navigation/navigation.js';
9
+ import type { Href } from '../types.js';
10
+ import {
11
+ getPages,
12
+ type PageMeta,
13
+ pagePath,
14
+ type PageSearchOptions,
15
+ type PageSearchResult,
16
+ searchPages,
17
+ } from './search.js';
18
+
19
+ /** What {@link usePageSearch} returns. */
20
+ export interface PageSearch {
21
+ /** Ranked matches for the current query (best first); empty when the query is blank. */
22
+ readonly results: readonly PageSearchResult[];
23
+ /** The full registered page index (handy for rendering an "all pages" listing). */
24
+ readonly pages: readonly PageMeta[];
25
+ /**
26
+ * Navigates to a result / page / raw path. A dynamic (`:param`) page can't be navigated to
27
+ * as-is, so passing one (or its result) is a no-op unless you pass a concrete path string with
28
+ * the params already filled in. A stable reference, safe to destructure.
29
+ */
30
+ readonly goTo: (
31
+ target: string | PageMeta | PageSearchResult,
32
+ options?: NavigateOptions,
33
+ ) => void;
34
+ }
35
+
36
+ /** Whether a path can be navigated to directly (no unfilled dynamic segments). */
37
+ function isNavigable(path: string): boolean {
38
+ return !/[:*]/.test(path);
39
+ }
40
+
41
+ /**
42
+ * Searches the project's pages by `query` and returns ranked {@link PageSearchResult}s, recomputed
43
+ * only when the query or options change. Use the returned `goTo` to redirect to a match:
44
+ *
45
+ * ```tsx
46
+ * const { results, goTo } = usePageSearch(query);
47
+ * return results.map((r) => (
48
+ * <button key={r.page.path} onClick={() => { goTo(r); }}>{r.page.metadata.title ?? r.page.path}</button>
49
+ * ));
50
+ * ```
51
+ */
52
+ export function usePageSearch(query: string, options: PageSearchOptions = {}): PageSearch {
53
+ const { limit, includeDynamic, fields } = options;
54
+ const fieldsKey = fields?.join(',');
55
+ const results = useMemo(
56
+ () => searchPages(query, { limit, includeDynamic, fields }),
57
+ // `fields` is compared by content (fieldsKey) so a fresh array literal each render is fine.
58
+ [query, limit, includeDynamic, fieldsKey],
59
+ );
60
+
61
+ return useMemo<PageSearch>(
62
+ () => ({
63
+ results,
64
+ pages: getPages(),
65
+ goTo(target, navOptions) {
66
+ const path = pagePath(target);
67
+ if (typeof target !== 'string' && !isNavigable(path)) return;
68
+ navigate(path as Href, navOptions);
69
+ },
70
+ }),
71
+ [results],
72
+ );
73
+ }
@@ -1,73 +1,73 @@
1
- /**
2
- * Public router types shared across the client runtime. Kept dependency-free (type-only React
3
- * imports) so any module can import them without pulling in component or DOM code.
4
- */
5
- import type { ComponentType, ReactNode } from 'react';
6
-
7
- /**
8
- * Augmentation point for the project's generated route types. The compiler emits (into
9
- * `toil-routes.d.ts`) `declare module 'toiljs/client' { interface Register { routePath: <union> } }`,
10
- * which narrows {@link RoutePath} from `string` to the project's actual routes.
11
- */
12
- export interface Register {}
13
-
14
- /**
15
- * Union of the project's route paths, static routes as literals, dynamic/catch-all as
16
- * `` `…/${string}` `` templates. Falls back to `string` before the types are generated.
17
- */
18
- export type RoutePath = Register extends { routePath: infer P }
19
- ? P extends string
20
- ? P
21
- : string
22
- : string;
23
-
24
- /**
25
- * An href accepted by `Link` / `NavLink` / `navigate`: a known {@link RoutePath} (optionally with
26
- * `?query` or `#hash`), or an absolute/protocol URL (`https:`, `mailto:`, …). When routes haven't
27
- * been generated yet, this is just `string`.
28
- */
29
- export type Href =
30
- | RoutePath
31
- | `${RoutePath}?${string}`
32
- | `${RoutePath}#${string}`
33
- | `${string}:${string}`;
34
-
35
- /** Lazy loader for a layout component (wraps children). */
36
- export type LayoutComponentLoader = () => Promise<{
37
- default: ComponentType<{ children?: ReactNode }>;
38
- }>;
39
-
40
- /** Props passed to an `error.tsx` / `global-error.tsx` component. */
41
- export interface RouteErrorProps {
42
- readonly error: Error;
43
- readonly reset: () => void;
44
- }
45
-
46
- /** Lazy loader for an error component (`error.tsx` / `global-error.tsx`), or `null` if none. */
47
- export type ErrorComponentLoader =
48
- | (() => Promise<{ default: ComponentType<RouteErrorProps> }>)
49
- | null;
50
-
51
- /**
52
- * A route entry produced by the compiler: a URL pattern, a lazy loader for its page component, and
53
- * the chain of nested layout loaders (shallowest → deepest, from nested `layout.tsx` files) that wrap it.
54
- */
55
- export interface RouteDef {
56
- readonly pattern: string;
57
- readonly load: () => Promise<{ default: ComponentType }>;
58
- readonly layouts?: readonly LayoutComponentLoader[];
59
- /** `template.tsx` chain (root → nested), like layouts, but re-mounted on each navigation. */
60
- readonly templates?: readonly LayoutComponentLoader[];
61
- /** Nearest `loading.tsx`, shown as the Suspense fallback while this route loads. */
62
- readonly loading?: () => Promise<{ default: ComponentType }>;
63
- /** Nearest `error.tsx`, rendered by an error boundary around this route. */
64
- readonly errorComponent?: () => Promise<{ default: ComponentType<RouteErrorProps> }>;
65
- /** Intercepting route (`(.)`/`(..)`/`(...)`), matched in its slot only on soft navigation. */
66
- readonly intercept?: boolean;
67
- }
68
-
69
- /** Optional root layout loader (wraps every page). `null` when the project defines no layout. */
70
- export type LayoutLoader = LayoutComponentLoader | null;
71
-
72
- /** Optional custom not-found (404) page loader, rendered when no route matches. */
73
- export type NotFoundLoader = (() => Promise<{ default: ComponentType }>) | null;
1
+ /**
2
+ * Public router types shared across the client runtime. Kept dependency-free (type-only React
3
+ * imports) so any module can import them without pulling in component or DOM code.
4
+ */
5
+ import type { ComponentType, ReactNode } from 'react';
6
+
7
+ /**
8
+ * Augmentation point for the project's generated route types. The compiler emits (into
9
+ * `toil-routes.d.ts`) `declare module 'toiljs/client' { interface Register { routePath: <union> } }`,
10
+ * which narrows {@link RoutePath} from `string` to the project's actual routes.
11
+ */
12
+ export interface Register {}
13
+
14
+ /**
15
+ * Union of the project's route paths, static routes as literals, dynamic/catch-all as
16
+ * `` `…/${string}` `` templates. Falls back to `string` before the types are generated.
17
+ */
18
+ export type RoutePath = Register extends { routePath: infer P }
19
+ ? P extends string
20
+ ? P
21
+ : string
22
+ : string;
23
+
24
+ /**
25
+ * An href accepted by `Link` / `NavLink` / `navigate`: a known {@link RoutePath} (optionally with
26
+ * `?query` or `#hash`), or an absolute/protocol URL (`https:`, `mailto:`, …). When routes haven't
27
+ * been generated yet, this is just `string`.
28
+ */
29
+ export type Href =
30
+ | RoutePath
31
+ | `${RoutePath}?${string}`
32
+ | `${RoutePath}#${string}`
33
+ | `${string}:${string}`;
34
+
35
+ /** Lazy loader for a layout component (wraps children). */
36
+ export type LayoutComponentLoader = () => Promise<{
37
+ default: ComponentType<{ children?: ReactNode }>;
38
+ }>;
39
+
40
+ /** Props passed to an `error.tsx` / `global-error.tsx` component. */
41
+ export interface RouteErrorProps {
42
+ readonly error: Error;
43
+ readonly reset: () => void;
44
+ }
45
+
46
+ /** Lazy loader for an error component (`error.tsx` / `global-error.tsx`), or `null` if none. */
47
+ export type ErrorComponentLoader =
48
+ | (() => Promise<{ default: ComponentType<RouteErrorProps> }>)
49
+ | null;
50
+
51
+ /**
52
+ * A route entry produced by the compiler: a URL pattern, a lazy loader for its page component, and
53
+ * the chain of nested layout loaders (shallowest → deepest, from nested `layout.tsx` files) that wrap it.
54
+ */
55
+ export interface RouteDef {
56
+ readonly pattern: string;
57
+ readonly load: () => Promise<{ default: ComponentType }>;
58
+ readonly layouts?: readonly LayoutComponentLoader[];
59
+ /** `template.tsx` chain (root → nested), like layouts, but re-mounted on each navigation. */
60
+ readonly templates?: readonly LayoutComponentLoader[];
61
+ /** Nearest `loading.tsx`, shown as the Suspense fallback while this route loads. */
62
+ readonly loading?: () => Promise<{ default: ComponentType }>;
63
+ /** Nearest `error.tsx`, rendered by an error boundary around this route. */
64
+ readonly errorComponent?: () => Promise<{ default: ComponentType<RouteErrorProps> }>;
65
+ /** Intercepting route (`(.)`/`(..)`/`(...)`), matched in its slot only on soft navigation. */
66
+ readonly intercept?: boolean;
67
+ }
68
+
69
+ /** Optional root layout loader (wraps every page). `null` when the project defines no layout. */
70
+ export type LayoutLoader = LayoutComponentLoader | null;
71
+
72
+ /** Optional custom not-found (404) page loader, rendered when no route matches. */
73
+ export type NotFoundLoader = (() => Promise<{ default: ComponentType }>) | null;