
test title
- Blog Page
- Fetches the latest blog posts using
client.fetch
with the Sanity GROQ query. - Displays the blog post's
title
,image
, andpublishedAt
date. - Links each post to its individual slug-based page (
/blog/{slug}
).
- Fetches the latest blog posts using
- Slug Page
- Fetches the details of a d single blog post based on the slug parameter.
- Displays the blog post's
image
,title
,publishedAt
date, andbody
. - Includes SEO tags like
og:title
,og:image
, andmeta
descriptions.
- Sanity
post
Schema- Defines the structure of a blog post, including fields for
title
,slug
,publishedAt
,image
,body
,category
,tags
, and SEO-related fields (seoTitle
,seoDescription
, andseoImage
). - Utilizes Sanity's built-in
image
type withhotspot
for better cropping options. - Supports an array of
body
blocks for rich text content and embedded images.
- Defines the structure of a blog post, including fields for
Key Considerations and Potential Enhancements
1. Dynamic Image Handling with next/image
- You've configured the dynamic URL generation using
urlFor(post.image)
with Sanity's image builder. - Ensure your
next.config.js
has thecdn.sanity.io
hostname underimages.domains
.
javascriptCopyEditimages: {
domains: ['cdn.sanity.io'],
},
2. SEO Enhancements
- On the slug page, you're using
seoTitle
andseoDescription
for SEO metadata. Great practice! - Consider fallback values if
seoTitle
orseoDescription
is empty:
javascriptCopyEdit<title>{post.seoTitle || post.title} | My Blog</title>
<meta name="description" content={post.seoDescription || post.body[0]?.children[0]?.text || post.title} />
3. Error Handling
- Handle cases where no blog posts exist on the main blog page:
javascriptCopyEdit{posts.length === 0 ? (
<p>No blog posts available. Check back soon!</p>
) : (
posts.map((post) => (/* Existing post rendering logic */))
)}
4. Lazy Loading Images
- For better performance, use
loading="lazy"
in yourImage
components:
jsxCopyEdit<Image
src={urlFor(post.image).width(400).height(300).url()}
alt={post.title}
width={400}
height={300}
className="w-full h-48 object-cover rounded-md"
loading="lazy"
/>
5. Sanity body
Rendering
- You're using
PortableText
for renderingbody
content. Ensure you have serializers for custom blocks like images:
javascriptCopyEditimport { PortableText } from '@portabletext/react';
const serializers = {
types: {
image: ({ value }) => (
<Image src={urlFor(value).url()} alt={value.alt || 'Blog Image'} width={800} height={450} className="rounded-lg" />
),
},
};
<PortableText value={post.body} components={serializers} />
6. Pagination for Blog Page
- If you have a large number of blog posts, consider adding pagination.
- Update your
POSTS_QUERY
to accept askip
parameter for paginated queries:
javascriptCopyEditconst POSTS_QUERY = `*[ _type == "post" && defined(slug.current)] | order(publishedAt desc)[$start...$end] { _id, title, slug, publishedAt, image }`;
Testing Checklist
- Verify Sanity Client: Ensure your Sanity client is correctly configured in
@/sanity/client
. - Image URLs: Confirm that
urlFor
is correctly generating URLs forimage
andseoImage
. - SEO Metadata: Double-check SEO tags on the slug page.
- 404 Handling: Verify the fallback for posts or slug pages that don’t exist.