-
-
Notifications
You must be signed in to change notification settings - Fork 10.5k
/
Copy pathroutes.tsx
128 lines (117 loc) · 3.41 KB
/
routes.tsx
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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
import type { LoaderFunctionArgs } from "react-router-dom";
import {
isRouteErrorResponse,
json,
Link,
Outlet,
useLoaderData,
useRouteError,
} from "react-router-dom";
export function Fallback() {
return <p>Performing initial data "load"</p>;
}
export function Layout() {
return (
<>
<nav>
<Link to="/projects/authorized">Authorized Project</Link>
|
<Link to="/projects/unauthorized">Unauthorized Project</Link>
|
<Link to="/projects/broken">Broken Project</Link>
</nav>
<p>
This example shows the flexibility of{" "}
<code><Route errorElement></code>
</p>
<ul>
<li>
Clicking the "Authorized Project" link will take you to the happy path
where we successfully load and render the details for a project.
</li>
<li>
Clicking the "Unauthorized Project" link will simulate a case where
the user does not have access to the given project, so our loader can
throw a 401 response that is handed in-context by a{" "}
<code><ProjectErrorBoundary></code>.
</li>
<li>
Clicking the "Broken Project" link will return some malformed data
causing a render error. This is beyond what{" "}
<code><ProjectErrorBoundary></code> can handle, so it re-throws
the error and it gets handled by{" "}
<code><RootErrorBoundary></code> instead.
</li>
</ul>
<Outlet />
</>
);
}
export function RootErrorBoundary() {
let error = useRouteError() as Error;
return (
<div>
<h1>Uh oh, something went terribly wrong 😩</h1>
<pre>{error.message || JSON.stringify(error)}</pre>
<button onClick={() => (window.location.href = "/")}>
Click here to reload the app
</button>
</div>
);
}
export function projectLoader({ params }: LoaderFunctionArgs) {
if (params.projectId === "unauthorized") {
throw json({ contactEmail: "administrator@fake.com" }, { status: 401 });
}
if (params.projectId === "broken") {
// Uh oh - in this flow we somehow didn't get our data nested under `project`
// and instead got it at the root - this will cause a render error!
return json({
id: params.projectId,
name: "Break Some Stuff",
owner: "The Joker",
deadline: "June 2022",
cost: "FREE",
});
}
return json({
project: {
id: params.projectId,
name: "Build Some Stuff",
owner: "Joe",
deadline: "June 2022",
cost: "$5,000 USD",
},
});
}
export function Project() {
let { project } = useLoaderData();
return (
<>
<h1>Project Name: {project.name}</h1>
<p>Owner: {project.owner}</p>
<p>Deadline: {project.deadline}</p>
<p>Cost: {project.cost}</p>
</>
);
}
export function ProjectErrorBoundary() {
let error = useRouteError();
// We only care to handle 401's at this level, so if this is not a 401
// ErrorResponse, re-throw to let the RootErrorBoundary handle it
if (!isRouteErrorResponse(error) || error.status !== 401) {
throw error;
}
return (
<>
<h1>You do not have access to this project</h1>
<p>
Please reach out to{" "}
<a href={`mailto:${error.data.contactEmail}`}>
{error.data.contactEmail}
</a>{" "}
to obtain access.
</p>
</>
);
}