Blog Post Content
Installation
Add the following Soul components
The blog-post-content component uses the button-link, skeleton, breadcrumbs, streamable and section-layout components. Make sure you have added them to your project.
Install the following dependencies
npm install clsx
Copy and paste the following code into your project
sections/blog-post-content/index.tsx
import { clsx } from 'clsx';import Image from 'next/image';import { Stream, Streamable } from '@/vibes/soul/lib/streamable';import { ButtonLink } from '@/vibes/soul/primitives/button-link';import * as Skeleton from '@/vibes/soul/primitives/skeleton';import { type Breadcrumb, Breadcrumbs, BreadcrumbsSkeleton,} from '@/vibes/soul/sections/breadcrumbs';import { SectionLayout } from '@/vibes/soul/sections/section-layout';interface Tag { label: string; link: { href: string; target?: string; };}interface Image { src: string; alt: string;}export interface BlogPost { title: string; author?: string | null; date: string; tags?: Tag[] | null; content: string; image?: Image | null;}export interface BlogPostContentProps { blogPost: Streamable<BlogPost>; breadcrumbs?: Streamable<Breadcrumb[]>; className?: string;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { * --blog-post-content-font-family: var(--font-family-body); * --blog-post-content-title-font-family: var(--font-family-body); * --blog-post-content-title: hsl(var(--foreground)); * --blog-post-content-image-background: hsl(var(--contrast-100)); * } * ``` */export function BlogPostContent({ blogPost: streamableBlogPost, className = '', breadcrumbs,}: BlogPostContentProps) { return ( <SectionLayout className={clsx('group/blog-post-content', className)}> <div className="mx-auto max-w-screen-2xl px-4 py-10 @xl:px-6 @xl:py-14 @4xl:px-8 @4xl:py-20"> <Stream fallback={<BlogPostContentSkeleton />} value={streamableBlogPost}> {(blogPost) => { const { title, author, date, tags, content, image } = blogPost; return ( <> <header className="mx-auto w-full max-w-4xl pb-8 font-[family-name:var(--blog-post-content-info-font-family,var(--font-family-body))] @2xl:pb-12 @4xl:pb-16"> {breadcrumbs && <Breadcrumbs breadcrumbs={breadcrumbs} />} <h1 className="mb-4 mt-8 font-[family-name:var(--blog-post-content-title-font-family,var(--font-family-heading))] text-4xl font-medium leading-none text-[var(--blog-post-content-title,hsl(var(--foreground)))] @xl:text-5xl @4xl:text-6xl"> {title} </h1> <p> {date}{' '} {author !== null && ( <> <span className="px-1">•</span> {author} </> )} </p> {tags && tags.length > 0 && ( <div className="-ml-1 mt-4 flex flex-wrap gap-1.5 @xl:mt-6"> {tags.map((tag, index) => ( <ButtonLink href={tag.link.href} key={index} size="small" variant="tertiary" > {tag.label} </ButtonLink> ))} </div> )} </header> {image && ( <Image alt={image.alt} className="mb-8 aspect-video w-full rounded-2xl bg-[var(--blog-post-content-image-background,hsl(var(--contrast-100)))] object-cover @2xl:mb-12 @4xl:mb-16" height={780} src={image.src} width={1280} /> )} <article className="prose mx-auto w-full max-w-4xl space-y-4" dangerouslySetInnerHTML={{ __html: content }} /> </> ); }} </Stream> </div> </SectionLayout> );}export function BlogPostContentSkeleton({ className }: Pick<BlogPostContentProps, 'className'>) { return ( <Skeleton.Root className={clsx('group-has-[[data-pending]]/blog-post-content:animate-pulse', className)} pending > <div className="mx-auto w-full max-w-4xl pb-8 @2xl:pb-12 @4xl:pb-16"> <BreadcrumbsSkeleton /> <div className="mb-4 mt-8"> <Skeleton.Text characterCount={60} className="rounded-lg text-4xl @xl:text-5xl @4xl:text-6xl" /> </div> <div> <Skeleton.Box className="h-6 w-1/4 rounded-lg" /> </div> <div className="mt-4 flex w-2/6 min-w-[250px] flex-wrap gap-3 @xl:mt-6"> <Skeleton.Box className="h-10 w-16 rounded-full" /> <Skeleton.Box className="h-10 w-14 rounded-full" /> <Skeleton.Box className="h-10 w-20 rounded-full" /> </div> </div> <Skeleton.Box className="mb-8 aspect-video w-full rounded-2xl @2xl:mb-12 @4xl:mb-16" /> <div className="mx-auto w-full max-w-4xl space-y-8 pb-8 text-xl @2xl:pb-12 @4xl:pb-16"> <Skeleton.Text characterCount={60} className="rounded-lg" /> <div className="space-y-4 text-lg"> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount="full" className="rounded-lg" /> <Skeleton.Text characterCount={50} className="rounded-lg" /> </div> </div> </Skeleton.Root> );}
Usage
import { BlogPost, BlogPostContent } from '@/vibes/soul/sections/blog-post-content';function Usage() { return <BlogPostContent blogPost={blogPost} />;}const blogPost = { author: 'Sam Smith', content: 'My Blog Post', date: 'June 30, 2024', image: { src: 'https://rstr.in/monogram/vibes/RNZYqBoUs7C', alt: 'A plant with large leaves and a short stem.', }, title: "Top 5 Plants to Purify Your Home's Air",};
API Reference
BlogPostContentProps
Prop | Type | Default |
---|---|---|
className | string | |
blogPost* | ||
breadcrumbs | ||
className | string |
BlogPost
Prop | Type | Default |
---|---|---|
title* | string | |
author | string | |
date* | string | |
tags | Tag[] | null | |
content* | string | |
image | { src: string, alt: string } | null |
Breadcrumb
Prop | Type | Default |
---|---|---|
label* | string | |
href* | string |
Tag
Prop | Type | Default |
---|---|---|
label* | string | |
link* | { href: string, target string } |
CSS Variables
This component supports various CSS variables for theming. Here's a comprehensive list.
:root { --blog-post-content-font-family: var(--font-family-body); --blog-post-content-title-font-family: var(--font-family-body); --blog-post-content-title: hsl(var(--foreground)); --blog-post-content-image-background: hsl(var(--contrast-100));}