Type safe routing for Next.js
nextjs-routes
makes Next.js's next/link
and next/router
routes type safe with zero runtime overhead. nextjs-routes
scans your pages
and/or app
directory and generates route types based on your application's routes.
nextjs-routes
drops into your existing Next.js application with minimal configuration. You won't have to change any code, unless it finds some broken links!
π¦ Zero config
π¨ Types only -- zero runtime
π No more broken links
πͺ Route autocompletion
π Supports all Next.js route types: static, dynamic, catch all and optional catch all
-
Add this package to your project:
npm install nextjs-routes
oryarn add nextjs-routes
-
Update your
next.config.js
:+ const withRoutes = require("nextjs-routes/config")(); /** @type {import('next').NextConfig} */ const nextConfig = { reactStrictMode: true, }; - module.exports = nextConfig; + module.exports = withRoutes(nextConfig);
File watching will only run in Next.js' development server (eg
npx next dev
). -
Start your next server:
npx next dev
oryarn next dev
That's it! A nextjs-routes.d.ts
file will be generated the first time you start your server. Check this file into version control. next/link
and next/router
type definitions have been augmented to verify your application's routes. No more broken links, and you get route autocompletion π.
Whenever your routes change, your nextjs-routes.d.ts
file will automatically update.
Link
's href
prop is now typed based on your application routes:
import Link from "next/link";
<Link
href={{
pathname: "/foos/[foo]",
query: { foo: "bar" },
}}
>
Bar
</Link>;
If the route doesn't require any parameters, you can also use a path string:
<Link href="/foo">Foo</Link>
useRouter
's returned router instance types for push
, replace
and query
are now typed based on your application routes.
Identical to Link
, push
and replace
now expect a UrlObject or path string:
import { useRouter } from "next/router";
const router = useRouter();
router.push({ pathname: "/foos/[foo]", query: { foo: "test" } });
import { useRouter } from "next/router";
const router = useRouter();
router.replace({ pathname: "/" });
import { useRouter } from "next/router";
// query is typed as a union of all query parameters defined by your application's routes
const { query } = useRouter();
By default, query
will be typed as the union of all possible query parameters defined by your application routes. If you'd like to narrow the type to fewer routes or a single page, you can supply a type argument:
import { useRouter } from "next/router";
// query is now typed as `{ foo: string }`
const { query } = useRouter<"/foos/[foo]">();
If you want to use the generated Route
type in your code, you can import it from nextjs-routes
:
import type { Route } from "nextjs-routes";
If you want to use the generated Query
for a given Route
, you can import it from nextjs-routes
:
import type { RoutedQuery } from "nextjs-routes";
This is useful as the context type parameter inside getStaticProps
and getServerSideProps
:
import type { GetStaticProps } from "next";
import type { RoutedQuery } from "nextjs-routes";
export const getStaticProps: GetStaticProps<{}, RoutedQuery<"/foos/[foo]">> = (
context
) => {
// context.params will include `foo` as a string
const { foo } = context.params;
};
nextjs-routes
generates types for the pathname
and query
for every page in your pages
and/or app
directory. The generated types are written to nextjs-routes.d.ts
which is automatically referenced by your Next project's tsconfig.json
. nextjs-routes.d.ts
redefines the types for next/link
and next/router
and applies the generated route types.
There are some cases where you may want to generate a type safe path from a Route
object, such as when fetch
ing from an API route or serving redirects from getServerSideProps
. These accept strings
instead of the Route
object that Link
and useRouter
accept. Because these do not perform the same string interpolation for dynamic routes, runtime code is required instead of a type only solution.
For these cases, you can use route
from nextjs-routes
:
import { route } from "nextjs-routes";
fetch(route({ pathname: "/api/foos/[foo]", query: { foo: "foobar" } }));
import { route } from "nextjs-routes";
export const getServerSideProps: GetServerSideProps = async (context) => {
return {
redirect: {
destination: route({ pathname: "/foos/[foo]", query: { foo: "foobar" } }),
permanent: false,
},
};
};
nextjs-routes
refines Link
and useRouter
based on your Nextjs i18n configuration.
The following next.config.js
:
module.exports = withRoutes({
i18n: {
defaultLocale: "de-DE",
locales: ["de-DE", "en-FR", "en-US"],
},
});
Will type Link
and useRouter
's locale
as 'de-DE' | 'en-FR' | 'en-US'
. All other i18n properties (defaultLocale
, domainLocales
and locales
) are also typed.
If you want to use the generated Locale
type, you can import it from nextjs-routes
:
import { Locale } from "nextjs-routes";
You can pass the following options to withRoutes
in your next.config.js
:
const withRoutes = require("nextjs-routes/config")({
outDir: "types",
});
outDir
: The file path indicating the output directory where the generated route types should be written to (e.g.: "types"). The default is to create the file in the same folder as yournext.config.js
file.
PR's and issues welcomed! For more guidance check out CONTRIBUTING.md
Are you interested in bringing a nextjs-routes
like experience to another framework? Open an issue and let's collaborate.
See the project's MIT License.