Blog Post List

Blog Post List is used to display a list of blog post cards.

Installation

Add the following Soul components

The blog-post-list component uses the blog-post-card, skeleton and streamable 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-list/index.tsx

import { clsx } from 'clsx';import { Stream, Streamable } from '@/vibes/soul/lib/streamable';import {  type BlogPost,  BlogPostCard,  BlogPostCardSkeleton,} from '@/vibes/soul/primitives/blog-post-card';import * as Skeleton from '@/vibes/soul/primitives/skeleton';export interface BlogPostListProps {  blogPosts: Streamable<BlogPost[]>;  className?: string;  emptyStateSubtitle?: Streamable<string>;  emptyStateTitle?: Streamable<string>;  placeholderCount?: number;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { *   --blog-post-list-empty-state-title-font-family: var(--font-family-heading); *   --blog-post-list-empty-state-subtitle-font-family: var(--font-family-body); *   --blog-post-list-empty-state-title: hsl(var(--foreground)); *   --blog-post-list-empty-state-subtitle: hsl(var(--contrast-500)); * } * ``` */export function BlogPostList({  blogPosts: streamableBlogPosts,  className = '',  emptyStateTitle = 'No blog posts found',  emptyStateSubtitle = 'Check back later for more content.',  placeholderCount = 6,}: BlogPostListProps) {  return (    <Stream      fallback={<BlogPostListSkeleton className={className} placeholderCount={placeholderCount} />}      value={streamableBlogPosts}    >      {(blogPosts) => {        if (blogPosts.length === 0) {          return (            <BlogPostListEmptyState              className={className}              emptyStateSubtitle={emptyStateSubtitle}              emptyStateTitle={emptyStateTitle}              placeholderCount={placeholderCount}            />          );        }        return (          <div className={clsx('@container', className)}>            <div className="mx-auto grid grid-cols-1 gap-x-5 gap-y-8 @md:grid-cols-2 @xl:gap-y-10 @3xl:grid-cols-3">              {blogPosts.map(({ ...post }) => (                <BlogPostCard key={post.href} {...post} />              ))}            </div>          </div>        );      }}    </Stream>  );}export function BlogPostListSkeleton({  className,  placeholderCount = 6,}: Pick<BlogPostListProps, 'className' | 'placeholderCount'>) {  return (    <Skeleton.Root      className={clsx('group-has-[[data-pending]]/blog-post-list:animate-pulse', className)}      pending    >      <div className="mx-auto grid grid-cols-1 gap-x-5 gap-y-8 @md:grid-cols-2 @xl:gap-y-10 @3xl:grid-cols-3">        {Array.from({ length: placeholderCount }).map((_, index) => (          <BlogPostCardSkeleton key={index} />        ))}      </div>    </Skeleton.Root>  );}export function BlogPostListEmptyState({  className,  placeholderCount = 6,  emptyStateTitle,  emptyStateSubtitle,}: Omit<BlogPostListProps, 'blogPosts'>) {  return (    <div className={clsx('relative w-full @container', className)}>      <div        className={clsx(          'mx-auto grid grid-cols-1 gap-x-5 gap-y-8 [mask-image:linear-gradient(to_bottom,_black_0%,_transparent_90%)] @md:grid-cols-2 @xl:gap-y-10 @3xl:grid-cols-3',        )}      >        {Array.from({ length: placeholderCount }).map((_, index) => (          <BlogPostCardSkeleton key={index} />        ))}      </div>      <div className="absolute inset-0 mx-auto px-3 py-16 pb-3 @4xl:px-10 @4xl:pb-10 @4xl:pt-28">        <div className="mx-auto max-w-xl space-y-2 text-center @4xl:space-y-3">          <h3 className="font-[family-name:var(--blog-post-list-empty-state-title-font-family,var(--font-family-heading))] text-2xl leading-tight text-[var(--blog-post-list-empty-state-title,hsl(var(--foreground)))] @4xl:text-4xl @4xl:leading-none">            {emptyStateTitle}          </h3>          <p className="font-[family-name:var(--blog-post-list-empty-state-subtitle-font-family,var(--font-family-body))] text-sm text-[var(--blog-post-list-empty-state-subtitle,hsl(var(--contrast-500)))] @4xl:text-lg">            {emptyStateSubtitle}          </p>        </div>      </div>    </div>  );}

Usage

import { BlogPostList } from '@/vibes/soul/sections/blog-post-list';function Usage() {  return (    <BlogPostList blogPosts={blogPosts} />  );}const blogPosts = [  {      id: '5',      title: 'A Guide to Low-Light Houseplants',      content:      'Not all plants need bright sunlight to thrive. This guide highlights the best low-light houseplants, perfect for those darker corners of your home or office that need a touch of green.',      image: {          src: 'https://rstr.in/monogram/vibes/T7CeSkvi11I/OrJN5KVj7I1',          alt: 'Low-Light Houseplants',      },      date: '2024-07-20',      href: '#',      author: 'Author Name',  }];

API Reference

BlogPostListProps

PropTypeDefault
className
string
blogPosts*
Streamable<BlogPost[]>
className
string
emptyStateSubtitle
Streamable<string>
emptyStateTitle
Streamable<string>
placeholderCount
number

BlogPost

PropTypeDefault
title*
string
author
string | null
content*
string
date*
string
image
{src: string; alt: string; } | null
href*
string

CSS Variables

This component supports various CSS variables for theming. Here's a comprehensive list.

:root {  --blog-post-list-empty-state-title-font-family: var(--font-family-heading);  --blog-post-list-empty-state-subtitle-font-family: var(--font-family-body);  --blog-post-list-empty-state-title: hsl(var(--foreground));  --blog-post-list-empty-state-subtitle: hsl(var(--contrast-500));}