This open-source starter template is built with Next.js 14, Park UI (Ark UI + Panda CSS), Contentlayer, and Sveltia CMS. It's designed to be a simple and customizable way to launch a modern blog, with support for MDX and multiple categories like Code Blog, Inspiration, Podcasts, Tools, and Resources.
- ๐ ๏ธ Next.js 14: Fast, modern React framework for production-ready web applications.
- ๐จ Park UI: Headless UI framework for rapid UI development.
- ๐ผ Panda CSS: Zero-runtime CSS design system styling engine.
- ๐ MDX Support: Write your blog posts in Markdown with JSX components.
- ๐๏ธ Contentlayer: Simple content management with files as data.
- ๐ Sveltia CMS: Easily manage your posts through a friendly CMS interface.
- ๐ท๏ธ Categories: Pre-configured sections for Code Blog, Inspiration, Podcasts, and Tools.
- ๐ผ๏ธ Optimized Images: Use the
<ExportedImage />component from thenext-image-export-optimizerpackage for optimized image handling in static exports, replacing the default Next.js<Image />component for better control over image quality, formats (like WEBP), and cache settings. - ๐ Dark/Light Mode: Automatically adapts to the user's operating system settings.
- Next.js 14
- Park UI
- Panda CSS
- MDX
- Contentlayer
- Sveltia CMS
Follow these steps to get the project up and running:
-
Install dependencies:
pnpm install
-
Run the development server:
pnpm run dev
Open http://localhost:3000 in your browser to see the app running.
You can create different types of content (like blog posts or podcasts) in separate folders under content/. Each folder corresponds to a category in your blog.
Content is written in MDX format and managed using Contentlayer. To create a new post, add an .mdx file in the appropriate folder inside content/ for each category:
content/
โโโ blog/
โโโ inspiration/
โโโ podcasts/
โโโ resources/
โโโ tools/
Each post supports frontmatter fields like title, description, date, and featured.
Example frontmatter for a blog post:
---
templateKey: blog
title: >
Custom Scrollbar with CSS (WebKit)
date: 2021-06-19T19:28:37.629Z
featured: true
description: >
Learn how to customize the scrollbar with CSS for WebKit browsers, providing a visually appealing design for scrollable elements on your website.
tags:
- web-development
- css
---
Your markdown content here...Example inspiration post, with a video:
---
templateKey: inspiration
title: >
2Advanced Studios โ Flash website in 2003
date: 2003-03-03T15:04:10.000Z
featured: true
description: >
An old flash website (v4) of 2advanced.com, that has inspired me quite a lot in my teenage years.
tags:
- animation
- flash
- website
image: /media/2advanced-flash-website-v4-2003.jpg
---
<Video src="/media/2advanced-flash-website-v4-2003.mp4" />
## ยฉ 2003 Advanced Studios
Currently, the 2Advanced Studios are closed for real, since some years already.The local development server will automatically start a Sveltia CMS development server alongside your Next.js development server when you run:
pnpm run devYou can then access the CMS at http://localhost:3000/admin/index.html.
Note
If you'd like to test the CMS locally, set local_backend: true in public/admin/config.yml. Don't forget to restart the server after making changes.
Customize the project to suit your needs by editing the following files:
config.js: Your Bleg Starter configuration.panda.config.js: Panda CSS configuration.next.config.js: Next.js custom settings.next-seo.config.js: SEO configuration for Next.js.contentlayer.config.js: Contentlayer configuration for MDX files.public/admin/config.yml: Decap CMS configuration.
- The project uses the
next-image-export-optimizerpackage to enhance image handling in static exports. - Custom settings for image optimization:
- Image Folder: Images are stored in the
public/mediafolder. - Export Settings: Optimized images are exported to the
out/folder. - Quality: Image quality is set to 75%.
- WEBP Format: By default, the images are converted to WEBP for improved performance.
- Blurred Placeholder: Blurry placeholders are enabled for a smoother loading experience. To disable this, set
nextImageExportOptimizer_generateAndUseBlurImagestofalsein your.next.config.jsfile, and passplaceholder="empty"to all<ExportedImage>components.
- Image Folder: Images are stored in the
Note
Replace Next.js <Image /> components with <ExportedImage /> to leverage these optimizations.
Example usage:
import ExportedImage from 'next-image-export-optimizer'
<ExportedImage
src="/media/example.jpg"
alt="Example Image"
width={800}
height={600}
placeholder="blur"
/>This project uses Contentlayer to automatically generate TypeScript types for your content. The configuration is managed in the contentlayer.config.ts file located at the root of the project. Each document type (e.g., Blog, Inspiration, Podcasts, Tools, Pages) has its own structure and generated types, ensuring type safety when working with content in your components.
Below is an example of the TypeScript types generated for the Blog document:
import { defineDocumentType, makeSource } from 'contentlayer2/source-files'
const Blog = defineDocumentType(() => ({
name: 'Blog',
filePathPattern: `blog/*.md`,
contentType: 'markdown',
fields: {
title: { type: 'string', required: true },
date: { type: 'date', required: false },
description: { type: 'string', required: false },
tags: { type: 'json', required: false },
templateKey: { type: 'string', required: true },
featured: { type: 'boolean', required: false },
},
computedFields: {
slug: {
type: 'string',
resolve: (doc) => doc._raw.sourceFileName.replace(/\.md/, ''),
},
},
}))
export default makeSource({
contentDirPath: 'content',
documentTypes: [Page, Blog, Inspiration, Podcasts, Tools, Resources],
disableImportAliasWarning: true,
})Note
The slug field is automatically computed from the filename, removing the .md extension.
The types are generated into the ./.contentlayer directory and can be used throughout the application. Simply import the types from there:
import { allBlogs, Blog } from '../../.contentlayer/generated')
import { pick } from '@contentlayer2/client'When working with content in your components, you can use the generated types to ensure type safety. For example, when mapping over blog posts:
let blogs = allBlogs.map((blog) => pick(blog, ['title', 'date', 'slug', 'description', 'templateKey'])In this example, allBlogs is an array of Blog types, and we're using the pick function to select specific fields from each blog post.
When working with content from allBlogs, you can assert the type to ensure it's a Blog type:
const blog = allBlogs.find((b) => b.slug === params.slug) as BlogThis way, you can access the fields of the Blog type without any issues.
When passing content to components, you can use the Blog type to ensure the correct structure:
<BlogPostCard key={post.slug} post={post as Blog} />With these types in place, you can benefit from strong typing and auto-completion when working with your content.
Deploy your own instance of this blog starter project using one of the following providers:
Deploy your own instance of this blog starter project using one of the following providers:
Note
If you encounter errors related to sharp during deployment, please try removing the package-lock.json file, as this can sometimes resolve issues with Sharp's dependencies.
If you're deploying your site using GitHub as backend and using Sveltia CMS for content management.
Tip
For more information on setting up GitHub backend with Sveltia CMS, see: https://github.com/sveltia/sveltia-cms.
This project is licensed under the MIT License. See the LICENSE file for details.
Contributions are welcome! Here are some ways you can help improve the project:
- Bug Fixes: If you encounter any issues, please submit an issue or pull request to address it.
- Feature Additions: Have an idea for a new feature? Open an issue to discuss it or submit a pull request with your implementation.
- Documentation Improvements: Help us enhance the README by suggesting edits or additional information.
- Fork the repository
- Create a new branch (
git checkout -b feature/your-feature-name) - Make your changes
- Push to your branch (
git push origin feature/your-feature-name) - Create a pull request
Thank you for considering contributing to the project!
Built with โค๏ธ in ๐ฎ๐ช, ๐ต๐น and ๐ง๐ช using Next.js (an amazing open-source React framework), Panda CSS (for rapid UI development), Contentlayer (for managing content), and Sveltia CMS (for a user-friendly content management experience).
A big thank you to the communities behind these projects for their hard work and dedication!