This application supports typesafe routing for NextJS using the declarative-routing
system.
Declarative Routes is a system for typesafe routing in React. It uses a combination of TypeScript and a custom routing system to ensure that your routes are always in sync with your code. You'll never have to worry about broken links or missing routes again.
In NextJS applications, Declarative Routes also handles API routes, so you'll have typesafe input and output from all of your APIs. In addition to fetch
functions that are written for you automatically.
Here are the routes of the application:
Route | Verb | Route Name | Using It |
---|---|---|---|
/api/pokemon/[pokemonId] |
GET | ApiPokemonPokemonId |
getApiPokemonPokemonId(...) |
/api/pokemon |
GET | ApiPokemon |
getApiPokemon(...) |
/ |
- | Home |
<Home.Link> |
/pokemon/[pokemonId] |
- | PokemonPokemonId |
<PokemonPokemonId.Link> |
/search |
- | Search |
<Search.Link> |
To use the routes, you can import them from @/routes
and use them in your code.
For pages, use the Link
component (built on top of next/link
) to link to other pages. For example:
import { ProductDetail } from "@/routes";
return (
<ProductDetail.Link productId={"abc123"}>Product abc123</ProductDetail.Link>
);
This is the equivalent of doing <Link href="/product/abc123">Product abc123</Link>
but with typesafety. And you never have to remember the URL. If the route moves, the typesafe route will be updated automatically.
For APIs, use the exported fetch
wrapping functions. For example:
import { useEffect } from "react";
import { getProductInfo } from "@/routes";
useEffect(() => {
// Parameters are typed to the input of the API
getProductInfo({ productId: "abc123" }).then((data) => {
// Data is typed to the result of the API
console.log(data);
});
}, []);
This is the equivalent of doing fetch('/api/product/abc123')
but with typesafety, and you never have to remember the URL. If the API moves, the typesafe route will be updated automatically.
Tou'll need to run pnpm dr:build
to update the generated files. This will update the types and the @/routes
module to reflect the changes.
The way the system works the .info.ts
files are link to the @/routes/index.ts
file. So changing the Zod schemas for the routes does NOT require a rebuild. You need to run the build command when:
- You change the name of the route in the
.info.ts
file - You change the location of the route (e.g.
/product
to/products
) - You change the parameters of the route (e.g.
/product/[id]
to/product/[productId]
) - You add or remove routes
- You add or remove verbs from API routes (e.g. adding
POST
to an existing route)
You can also run the build command in watch mode using pnpm dr:build:watch
but we don't recommend using that unless you are changing routes a lot. It's a neat party trick to change a route directory name and to watch the links automagically change with hot module reloading, but routes really don't change that much.
Post setup there are some additional tasks that you need to complete to completely typesafe your routes. We've compiled a handy check list so you can keep track of your progress.
-
/api/pokemon/[pokemonId]/route.info.ts
: Add typing forGET
- Convert
GET
fetch calls to/api/pokemon/[pokemonId]
togetApiPokemonPokemonId(...)
calls -
/api/pokemon/route.info.ts
: Add typing forGET
- Convert
GET
fetch calls to/api/pokemon
togetApiPokemon(...)
calls -
/page.info.ts
: Add search typing to if the page supports search paramaters - Convert
Link
components for/
to<Home.Link>
-
/pokemon/[pokemonId]/page.info.ts
: Add search typing to if the page supports search paramaters - Convert
Link
components for/pokemon/[pokemonId]
to<PokemonPokemonId.Link>
- Convert
params
typing in/pokemon/[pokemonId]/page.ts
toz.infer<>
-
/search/page.info.ts
: Add search typing to if the page supports search paramaters - Convert
Link
components for/search
to<Search.Link>
Once you've got that done you can remove this section.
You own this routing system once you install it. And we anticipate as part of that ownership you'll want to customize the routing system. That's why we create a makeRoute.tsx
file in the @/routes
module. This file is a copy of the makeRoute.tsx
file from the declarative-routing
package. You can modify this file to change the behavior of the routing system.
For example, you might want to change the way GET
, POST
, PUT
, and DELETE
are handled. Or you might want to change the way the Link
component works. You can do all of that by modifying the makeRoute.tsx
file.
We do NOT recommend changing the parameters of makeRoute
, makeGetRoute
, makePostRoute
, makePutRoute
, or makeDeleteRoute
functions because that would cause incompatiblity with the build
command of declarative-routing
.
This system is based on the work in Fix Next.JS Routing To Have Full Type-Safety. However the original article had a significantly different interface and didn't cover API routes at all.