Blog Post Card

Blog Post Card is used to display a blog post in a card preview.

  • Full keyboard navigation
  • Hover animation
  • When no image is provided, the card will display a typographic background with the title.

Installation

Add the following Soul components

The blog-post-card component uses the skeleton component. Make sure you have added it to your project.

Install the following dependencies

npm install clsx

Copy and paste the following code into your project

primitives/blog-post-card/index.tsx

import { clsx } from 'clsx';import Image from 'next/image';import Link from 'next/link';import * as Skeleton from '@/vibes/soul/primitives/skeleton';export interface BlogPost {  title: string;  author?: string | null;  content: string;  date: string;  image?: {    src: string;    alt: string;  } | null;  href: string;}export interface BlogPostCardProps extends BlogPost {  className?: string;}/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css * :root { *   --blog-post-card-focus: hsl(var(--primary)); *   --blog-post-card-image-background: hsl(var(--contrast-100)); *   --blog-post-card-empty-text: hsl(var(--foreground)/15%); *   --blog-post-card-title-text: hsl(var(--foreground)); *   --blog-post-card-content-text: hsl(var(--contrast-400)); *   --blog-post-card-author-date-text: hsl(var(--foreground)); *   --blog-post-card-font-family: var(--font-family-body); *   --blog-post-card-summary-text: hsl(var(--contrast-400)); *   --blog-post-card-author-date-text: hsl(var(--foreground)); * } * ``` */export function BlogPostCard({  author,  content,  date,  href,  image,  title,  className,}: BlogPostCardProps) {  return (    <article      className={clsx(        'group relative w-full max-w-md font-[family-name:var(--blog-post-card-font-family,var(--font-family-body))] @container',        className,      )}    >      <div className="relative aspect-[4/3] w-full overflow-hidden rounded-2xl bg-[var(--blog-post-card-image-background,hsl(var(--contrast-100)))]">        {image != null ? (          <Image            alt={image.alt}            className="object-cover transition-transform duration-500 ease-out group-hover:scale-110"            fill            sizes="(min-width: 80rem) 25vw, (min-width: 56rem) 33vw, (min-width: 28rem) 50vw, 100vw"            src={image.src}          />        ) : (          <div className="p-4 text-5xl font-bold leading-none tracking-tighter text-[var(--blog-post-card-empty-text,hsl(var(--foreground)/15%))]">            {title}          </div>        )}      </div>      <h5 className="mt-4 text-lg font-medium leading-snug text-[var(--blog-post-card-title-text,hsl(var(--foreground)))]">        {title}      </h5>      <p className="mt-1.5 line-clamp-3 text-sm font-normal text-[var(--blog-post-card-content-text,hsl(var(--contrast-400)))]">        {content}      </p>      <div className="mt-3 text-sm text-[var(--blog-post-card-author-date-text,hsl(var(--foreground)))]">        <time dateTime={date}>          {new Date(date).toLocaleDateString('en-US', {            year: 'numeric',            month: 'long',            day: 'numeric',          })}        </time>        {author != null && (          <>            <span className="after:mx-2 after:content-['•']" />            <span>{author}</span>          </>        )}      </div>      <Link        className={clsx(          'absolute inset-0 rounded-b-lg rounded-t-2xl focus:outline-none focus-visible:ring-2 focus-visible:ring-[var(--blog-post-card-focus,hsl(var(--primary)))] focus-visible:ring-offset-4',        )}        href={href}      >        <span className="sr-only">View article</span>      </Link>    </article>  );}export function BlogPostCardSkeleton({  aspectRatio = '4:3',  className,}: {  aspectRatio?: '5:6' | '3:4' | '4:3' | '1:1';  className?: string;}) {  return (    <div className={clsx('w-full max-w-md', className)}>      <Skeleton.Box        className={clsx(          'mb-4 w-full rounded-2xl',          {            '5:6': 'aspect-[5/6]',            '3:4': 'aspect-[3/4]',            '4:3': 'aspect-[4/3]',            '1:1': 'aspect-square',          }[aspectRatio],        )}      />      <Skeleton.Text characterCount={25} className="mt-4 rounded text-lg" />      <div className="mt-1.5">        <Skeleton.Text characterCount="full" className="rounded text-sm" />        <Skeleton.Text characterCount="full" className="rounded text-sm" />        <Skeleton.Text characterCount={15} className="rounded text-sm" />      </div>      <Skeleton.Text characterCount={10} className="mt-3 rounded text-sm" />    </div>  );}

Usage

import { BlogPostCard } from '@/vibes/soul/primitives/blog-post-card';function Usage() {  return (    <BlogPostCard      title: 'Vestibulum eleifend placerat ligula and even more text for a long title'      author="Ryan Smith"      content="'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Quisque eget velit hendrerit erat imperdiet tincidunt. Lorem ipsum dolor sit amet, consectetur adipiscing elit. This is some more text to test out here."      date="2022-01-01"      href="#"      image: {{        src: 'https://rstr.in/monogram/vibes/mrlTNE1TJfB',        alt: 'Vestibulum eleifend placerat ligula',      }},      href="/blog/how-to-create-a-blog-post-card"    />  );}

API Reference

BlogPostCardProps

PropTypeDefault
className
string
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-card-focus: hsl(var(--primary));  --blog-post-card-image-background: hsl(var(--contrast-100));  --blog-post-card-empty-text: hsl(var(--foreground) / 15%);  --blog-post-card-title-text: hsl(var(--foreground));  --blog-post-card-content-text: hsl(var(--contrast-400));  --blog-post-card-author-date-text: hsl(var(--foreground));  --blog-post-card-font-family: var(--font-family-body);  --blog-post-card-summary-text: hsl(var(--contrast-400));  --blog-post-card-author-date-text: hsl(var(--foreground));}