-
-
Notifications
You must be signed in to change notification settings - Fork 10.5k
/
Copy pathgenerate.ts
90 lines (74 loc) · 3.28 KB
/
generate.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
import ts from "dedent";
import * as Path from "pathe";
import { type RouteManifestEntry } from "../config/routes";
import { type Context } from "./context";
import { getTypesPath } from "./paths";
import * as Params from "./params";
import * as Route from "./route";
function getTypescriptSafePath(path: string) {
// In typescript, we want to support "moduleResolution": "nodenext" as well as not having "allowImportingTsExtensions": true,
// so we normalize all JS like files to `.js`, but allow other extensions such as `.mdx` and others that might be used as routes.
return path.replace(/\.(js|ts)x?$/, ".js");
}
export function generate(ctx: Context, route: RouteManifestEntry): string {
const lineage = Route.lineage(ctx.config.routes, route);
const fullpath = Route.fullpath(lineage);
const typesPath = getTypesPath(ctx, route);
const parents = lineage.slice(0, -1);
const parentTypeImports = parents
.map((parent, i) => {
const rel = Path.relative(
Path.dirname(typesPath),
getTypesPath(ctx, parent)
);
const indent = i === 0 ? "" : " ".repeat(2);
let source = getTypescriptSafePath(rel);
if (!source.startsWith("../")) source = "./" + source;
return `${indent}import type { Info as Parent${i} } from "${source}"`;
})
.join("\n");
return ts`
// React Router generated types for route:
// ${route.file}
import type * as T from "react-router/route-module"
${parentTypeImports}
type Module = typeof import("../${getTypescriptSafePath(Path.basename(route.file))}")
export type Info = {
parents: [${parents.map((_, i) => `Parent${i}`).join(", ")}],
id: "${route.id}"
file: "${route.file}"
path: "${route.path}"
params: {${formatParamProperties(
fullpath
)}} & { [key: string]: string | undefined }
module: Module
loaderData: T.CreateLoaderData<Module>
actionData: T.CreateActionData<Module>
}
export namespace Route {
export type LinkDescriptors = T.LinkDescriptors
export type LinksFunction = () => LinkDescriptors
export type MetaArgs = T.CreateMetaArgs<Info>
export type MetaDescriptors = T.MetaDescriptors
export type MetaFunction = (args: MetaArgs) => MetaDescriptors
export type HeadersArgs = T.HeadersArgs
export type HeadersFunction = (args: HeadersArgs) => Headers | HeadersInit
export type unstable_MiddlewareFunction = T.CreateServerMiddlewareFunction<Info>
export type unstable_ClientMiddlewareFunction = T.CreateClientMiddlewareFunction<Info>
export type LoaderArgs = T.CreateServerLoaderArgs<Info>
export type ClientLoaderArgs = T.CreateClientLoaderArgs<Info>
export type ActionArgs = T.CreateServerActionArgs<Info>
export type ClientActionArgs = T.CreateClientActionArgs<Info>
export type HydrateFallbackProps = T.CreateHydrateFallbackProps<Info>
export type ComponentProps = T.CreateComponentProps<Info>
export type ErrorBoundaryProps = T.CreateErrorBoundaryProps<Info>
}
`;
}
function formatParamProperties(fullpath: string) {
const params = Params.parse(fullpath);
const properties = Object.entries(params).map(([name, isRequired]) =>
isRequired ? `"${name}": string` : `"${name}"?: string`
);
return properties.join("; ");
}