Icon

Installation

Add the following Soul components

The icon component uses the skeleton component. Make sure you have added it to your project.

Install the following dependencies

npm install clsx lucide-react

Copy and paste the following code into your project

primitives/icon/index.tsx

import { clsx } from 'clsx';import dynamicIconImports from 'lucide-react/dynamicIconImports';import { lazy, Suspense, useMemo } from 'react';import * as Skeleton from '@/vibes/soul/primitives/skeleton';export type IconName = keyof typeof dynamicIconImports;// eslint-disable-next-line @typescript-eslint/consistent-type-assertionsexport const IconNames = Object.keys(dynamicIconImports) as IconName[];export interface IconProps {  className?: string;  size?: number;  name: IconName;  strokeWidth?: number;}export function Icon({ className, name, size = 24, strokeWidth = 1.5 }: IconProps) {  const LucideIcon = useMemo(() => lazy(dynamicIconImports[name]), [name]);  return (    <Suspense fallback={<IconSkeleton className={className} size={size} />}>      <LucideIcon className={className} name={name} size={size} strokeWidth={strokeWidth} />    </Suspense>  );}export function IconSkeleton({ className, size = 24 }: Pick<IconProps, 'className' | 'size'>) {  return (    <Skeleton.Icon      className={clsx('overflow-hidden rounded-full', className)}      icon={        <div          className="bg-[var(--skeleton,hsl(var(--contrast-300)))] opacity-25"          style={{ width: size, height: size }}        />      }    />  );}

Usage

import { Icon } from '@/vibes/soul/primitives/icon';function Usage() {  return (      <Icon className="text-contrast-500" name="user" size={32} />  );}

API Reference

IconProps

PropTypeDefault
className
string
size
number
24
name*
IconName
strokeWidth
number
1.5

Refer to the Lucide docs for a full list of available icons.