Setting Up TypeScript For Sanity In NextJS

Jan 2, 2025

2 min read

You can use Sanity TypeGen to generate TypeScript types for schema types and GROQ query results in your Next.js application. If you've used sanity init with the embedded Studio option, it's already available.

[!TIP] Sanity TypeGen generates types for queries that are assigned to a variable and use either the groq template literal or defineQuery function.

If your Sanity Studio schema types exist in a separate project or repository, you can configure Sanity TypeGen to write types to your Next.js project.

Create a sanity-typegen.json file at your project's root to configure Sanity TypeGen:

// sanity-typegen.json 
{ 
"path": "./src/**/*.{ts,tsx,js,jsx}", 
"schema": "./src/sanity/extract.json", 
"generates": "./src/sanity/types.ts"
 }

Note: This configuration places the generated types and schema extraction within the /src/sanity directory instead of the default root location. This aligns with the schema extraction path specified in the package.json scripts below.

Run this command to extract your Sanity Studio schema to a JSON file each time your schema types change:

npx sanity@latest schema extract

Run this command to generate TypeScript types for your Sanity Studio schema and GROQ queries and each time your schema types or GROQ queries change:

npx sanity@latest typegen generate

Update your Next.js project's package.json to execute both commands with npm run typegen:

"scripts": { 
"predev": "npm run typegen", 
"dev": "next", 
"prebuild": "npm run typegen", 
"build": "next build", 
"start": "next start", 
"lint": "next lint", 
"typegen": "sanity schema extract --path=src/sanity/extract.json && sanity typegen generate" 
},

Using query result types

Sanity TypeGen creates TypeScript types for your GROQ query results. You can use these types explicitly as generics:

import {client} from '@/sanity/lib/client' 
import {POSTS_QUERY} from '@/sanity/lib/queries' 
import {POSTS_QUERYResult} from '@/sanity/types'
const posts = await client.fetch<POSTS_QUERYResult>(POSTS_QUERY)

However, automatic type inference offers a simpler approach. When GROQ queries are wrapped in defineQuery, types are inferred automatically:

import {client} from '@/sanity/lib/client' 
import {POSTS_QUERY} from '@/sanity/lib/queries'
const posts = await client.fetch(POSTS_QUERY)