AI News Hub Logo

AI News Hub

How to Build a React Blog with Next.js and Cosmic

DEV Community
Tony Spiro

Building a blog with React and Next.js is one of the most common use cases for modern web development. But where you store and manage your content matters just as much as the frontend you choose. In this tutorial, you'll learn how to build a fast, production-ready React blog using the Next.js App Router and Cosmic as your headless CMS. Want to skip ahead? Grab the free Simple React Blog template and deploy in minutes. Managing blog content directly in your codebase works fine for a handful of posts, but it breaks down fast. Markdown files scattered across a repo, no visual editing interface, and every content update requiring a developer and a deployment are all pain points that accumulate quickly. A headless CMS solves this by separating your content layer from your presentation layer. You manage posts, authors, and categories in a structured dashboard. Your Next.js app fetches that content via the REST API at build time or on demand. Non-technical editors can publish without touching code, and developers stay focused on the frontend. Cosmic gives you a clean API, a TypeScript SDK, a generous free tier, and zero infrastructure to manage. Before you start, you'll need: Node.js 18 or later A free Cosmic account Basic familiarity with React and TypeScript After signing up for a free Cosmic account, create a new bucket. You can start from the Simple React Blog template to get a pre-configured bucket with sample content and a ready-to-deploy Next.js app, or follow along manually. In your Cosmic dashboard, navigate to Object Types and create a new type called blog-posts. Add the following metafields: Title - Text (built-in) Slug - Text (built-in) Cover Image - File (image) Excerpt - Textarea Content - Markdown Published Date - Date Author - Text Once saved, create a few sample posts so you have content to query during development. Go to Settings > API Keys in your bucket. You'll need: Bucket Slug (e.g. my-blog-production) Read Key (for public content fetching) Save these as environment variables. npx create-next-app@latest my-blog --typescript --app cd my-blog Install the Cosmic SDK: npm install @cosmicjs/sdk Create a .env.local file: COSMIC_BUCKET_SLUG=your-bucket-slug COSMIC_READ_KEY=your-read-key Create a shared Cosmic client at lib/cosmic.ts: import { createBucketClient } from '@cosmicjs/sdk' export const cosmic = createBucketClient({ bucketSlug: process.env.COSMIC_BUCKET_SLUG as string, readKey: process.env.COSMIC_READ_KEY as string, }) // types/post.ts export interface Post { id: string slug: string title: string metadata: { content: string excerpt: string cover_image: { imgix_url: string } published_date: string author: string } } const { objects: posts } = await cosmic.objects .find({ type: 'blog-posts' }) .props(['id', 'slug', 'title', 'metadata']) .depth(1) const { object: post } = await cosmic.objects .findOne({ type: 'blog-posts', slug: params.slug }) .props(['id', 'slug', 'title', 'metadata']) .depth(1) app/blog/page.tsx) import { cosmic } from '@/lib/cosmic' import Link from 'next/link' export default async function BlogPage() { const { objects: posts } = await cosmic.objects .find({ type: 'blog-posts' }) .props(['id', 'slug', 'title', 'metadata']) return ( Blog {posts.map((post) => ( {post.title} {post.metadata.excerpt} ))} ) } app/blog/[slug]/page.tsx) import { cosmic } from '@/lib/cosmic' import { marked } from 'marked' export async function generateStaticParams() { const { objects: posts } = await cosmic.objects .find({ type: 'blog-posts' }) .props(['slug']) return posts.map((post) => ({ slug: post.slug })) } export default async function PostPage({ params }: { params: { slug: string } }) { const { object: post } = await cosmic.objects .findOne({ type: 'blog-posts', slug: params.slug }) .props(['title', 'metadata']) const html = marked(post.metadata.content) return ( {post.title} ) } Cosmic stores all media through Imgix, a powerful image CDN. Append transformation parameters directly to any media URL: https://imgix.cosmicjs.com/your-image.jpg?w=800&auto=format&q=80&fit=crop Common params: w= sets width auto=format serves WebP to supported browsers q=80 reduces file size fit=crop crops to exact dimensions Push your project to a GitHub repository Go to vercel.com and import the repo Add your environment variables under Settings > Environment Variables: COSMIC_BUCKET_SLUG COSMIC_READ_KEY Click Deploy Vercel will detect Next.js automatically, build your app, and give you a live URL in under a minute. // app/api/revalidate/route.ts import { revalidatePath } from 'next/cache' import { NextRequest } from 'next/server' export async function POST(request: NextRequest) { revalidatePath('/blog') return Response.json({ revalidated: true }) } Then configure a Cosmic webhook to call this endpoint whenever a post is published. A Next.js App Router project with TypeScript A Cosmic bucket with a structured content model Server-side data fetching via the @cosmicjs/sdk Static pre-rendering with generateStaticParams Imgix-powered image optimization Live on Vercel with optional on-demand revalidation Your editors can log into the Cosmic dashboard and publish new posts without touching your codebase. Your readers get a fast, statically rendered experience. And your SEO improves because every post is pre-rendered HTML at build time. Don't want to build from scratch? The Simple React Blog template gives you everything in this tutorial, already wired up and ready to deploy with one click. Get the free React blog template Start building free Have questions or want to talk through your project? Book a quick intro call with the Cosmic team.