A Guide for Nested Routes
- Mitchell Christ
As you scale your site, you might consider organizing pages into grouped or nested routes—that is, pages with segmented URLs like "/docs/nested-route/foo/bar".
As of v3.6.0
, SanityPress offers two options to go about nested pages. Each option differs in ease of implementation, and each has it's pros and cons, so read further to find a solution that meets your needs.
Option 1: Slug Slashes
The quick and dirty way to create nested routes is to add forward slashes (/
) to the metadata.slug
value of the desired page document.
Although this is a simple solution, there is no type checking of the parent segments. If the parent page slug changes, make sure to manually update all slug values accordingly.
With this approach, all pages (regardless of being nested or not) will all be listed in the same location in the Studio (under the "Pages" view).
Option 2: New Page Document
This option involves creating a new page.*
document (e.g. page.services
, page.case-studies
, etc).
This approach is more scalable and maintainable, but will require some code additions to both the Sanity Studio backend and Next.js frontend.
By adding a new document, pages under the same type will be listed together in the Studio.
Backend additions
Add a new page.*
document in the studio:
import { defineField, defineType } from 'sanity'
export default defineType({
name: 'page.services',
title: 'Services Page',
type: 'document',
fields: [
defineField({
name: 'title',
type: 'string',
}),
defineField({
name: 'modules',
type: 'array',
of: [
...
],
}),
defineField({
name: 'metadata',
type: 'metadata',
}),
],
preview: {
select: {
title: 'title',
slug: 'metadata.slug.current',
},
prepare: ({ title, slug }) => ({
title,
subtitle: slug && `/services/${slug}`,
}),
},
})
Register the new document:
// documents
import page from './documents/page'
import ServicesPage from './documents/page.services'
...
export const schemaTypes = [
// documents
page,
ServicesPage,
...
]
Add a dedicated view in the Studio structure:
import { group, singleton } from './utils'
import type { StructureResolver } from 'sanity/structure'
import { VscServerProcess } from 'react-icons/vsc'
import { BsDatabaseAdd } from 'react-icons/bs'
const structure: StructureResolver = (S, context) =>
S.list()
.title('Content')
.items([
singleton(S, 'site').icon(VscServerProcess),
S.documentTypeListItem('page').title('Pages'),
S.documentTypeListItem('page.service').title('SerPages'),
S.divider(),
...
])
export default structure
Frontend additions
Duplicate the existing catch-all Next.js page file:
cd src/app
mkdir -p services/[...slug] # parent route is /services
cp \[...slug\]/page.tsx services/[...slug]/page.tsx
Make adjustments to fetch the new page.*
document data:
// Code only partially shown
export async function generateStaticParams() {
const slugs = await client.fetch<string[]>(
groq`*[
_type == 'page.service' &&
defined(metadata.slug.current)
].metadata.slug.current`,
)
return slugs.map((slug) => ({ slug: slug.split('/') }))
}
async function getPage(params: Props['params']) {
return await fetchSanity<Sanity.Page>(
groq`*[
_type == 'page.services' &&
metadata.slug.current == $slug
][0]{
...
}`,
{
params: { slug: params.slug?.join('/') },
tags: ['pages'],
},
)
}
Recap
✅ Slug option PROs:
- No additional setup necessary
- Super simple; slug slashes
❌ Slug option CONs:
- Prone to error if parent routes change
- Manual updates necessary
✅ New document option PROs:
- Changes to parent routes are reflected automatically across all nested pages
❌ New document option CONs:
- Coding involved
- Requires code updates for every new parent route
Other options
Additional options are available for reference, such as this approach proposed by a user.
SanityPress is meant to be a "starter" template, so feel free to customize the code to fit your needs.