npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@stackloop/ui

v2.4.2

Published

A modern, animated React component library built with Framer Motion, Lucide Icons, and Tailwind CSS

Readme

Installation

NPM

npm install --save @stackloop/ui

Peer Dependencies

Ensure react and react-dom (>=18) are installed.

Next.js Setup

1. Import Theme CSS

Import the library CSS in your root layout or _app.tsx:

// app/layout.tsx (App Router)
import '@stackloop/ui/theme.css'

export default function RootLayout({ children }) {
  return (
    <html lang="en">
      <body>{children}</body>
    </html>
  )
}
// pages/_app.tsx (Pages Router)
import '@stackloop/ui/theme.css'

export default function App({ Component, pageProps }) {
  return <Component {...pageProps} />
}

2. Tailwind Configuration

If using Tailwind CSS v4 with @theme, the library's theme variables are already configured. If you need to customize colors:

/* In your global CSS or theme.css */
@theme {
  /* Override library colors */
  --color-primary: #your-color;
  --color-primary-dark: #your-darker-color;
  --color-border: #your-border-color;
  --color-border-dark: #your-darker-border;
  --color-secondary: #your-secondary-bg;
  --color-background: #your-bg-color;
  --color-foreground: #your-text-color;
  
  /* Semantic colors */
  --color-success: #10b981;
  --color-warning: #f59e0b;
  --color-error: #ef4444;
  --color-info: #3b82f6;
}

Usage with React (Vite/CRA)

// In your main entry file (main.tsx or index.tsx)
import '@stackloop/ui/theme.css'

Importing Components

Import components from the package root:

import { Button, Modal, Input } from '@stackloop/ui'

All components are client-side components with 'use client' directive, making them compatible with Next.js App Router.

Theme Customization

The library uses a simplified color system with semantic variables:

Color Variables

| Variable | Default | Description | |----------|---------|-------------| | --color-primary | #525252 | Primary brand color | | --color-primary-dark | #404040 | Darker primary variant | | --color-border | #e5e5e5 | Default border color | | --color-border-dark | #d4d4d4 | Darker border variant | | --color-secondary | #fafafa | Secondary background | | --color-background | #ffffff | Main background | | --color-foreground | #171717 | Primary text color | | --color-success | #10b981 | Success state | | --color-warning | #f59e0b | Warning state | | --color-error | #ef4444 | Error state | | --color-info | #3b82f6 | Info state |

Customizing Colors

Create a custom theme file or extend the existing one:

/* styles/custom-theme.css */
@import '@stackloop/ui/theme.css';

@theme {
  /* Brand colors */
  --color-primary: #3b82f6;
  --color-primary-dark: #2563eb;
  
  /* Borders */
  --color-border: #e2e8f0;
  --color-border-dark: #cbd5e1;
  
  /* Backgrounds */
  --color-secondary: #f8fafc;
  --color-background: #ffffff;
  --color-foreground: #0f172a;
}

Dark Mode Support

You can add dark mode variants:

@theme {
  /* Light mode (default) */
  --color-background: #ffffff;
  --color-foreground: #171717;
  
  /* Dark mode */
  @media (prefers-color-scheme: dark) {
    --color-background: #0a0a0a;
    --color-foreground: #fafafa;
    --color-primary: #60a5fa;
    --color-border: #27272a;
  }
}

Components Reference

  • For each component below you'll find a short description, props (type, default, notes) and a minimal usage example.

Checkbox:

  • Description: Accessible checkbox with optional label and description.

  • Props:

    • label: string — optional.
    • description: string — optional.
    • onChange: (checked: boolean) => void — optional.
    • className: string — optional.
    • Inherits standard input props (e.g. disabled, defaultChecked, checked).
  • Usage:

    import { Checkbox } from '@stackloop/ui'
    
    <Checkbox label="Accept terms" onChange={(v) => console.log(v)} />

Button:

  • Description: Animated button with variants, sizes, icons and loading state.

  • Props:

    • variant: 'primary' | 'secondary' | 'outline' | 'ghost' | 'danger' — default: 'primary'.
    • size: 'sm' | 'md' | 'lg' — default: 'md'.
    • loading: boolean — default: false.
    • icon: ReactNode — optional.
    • className: string — optional.
    • Inherits standard button props.
  • Usage:

    import { Button } from '@stackloop/ui'
    
    <Button variant="outline" size="lg" onClick={() => {}}>Save</Button>

Input:

  • Description: Text input with label, error and optional icons. Includes automatic password visibility toggle for password inputs.

  • Props:

    • label: string — optional.
    • error: string — optional.
    • hint: string — optional.
    • leftIcon / rightIcon: ReactNode — optional.
    • className: string — optional.
    • Inherits input HTML attributes.
  • Features:

    • Password Toggle: When type="password", automatically displays an Eye/EyeOff icon to toggle password visibility. This overrides any custom rightIcon.
    • Icon Support: Display icons on left or right side of input
    • Validation: Built-in error and hint display
    • Accessibility: Proper labels and ARIA attributes
  • Usage:

    import { Input } from '@stackloop/ui'
    
    <Input label="Email" placeholder="[email protected]" />
      
    // Password with automatic toggle
    <Input label="Password" type="password" placeholder="Enter password" />
      
    // With icons
    <Input 
      label="Search" 
      leftIcon={<Search />} 
      placeholder="Search..." 
    />

Modal:

  • Description: Centered modal with backdrop, title and Escape-to-close handling.

  • Props:

    • isOpen: boolean — required.
    • onClose: () => void — required.
    • children: ReactNode — required.
    • title: string — optional.
    • size: 'sm'|'md'|'lg'|'xl'|'full' — default: 'md'.
    • className: string — optional.
  • Subcomponents: ModalContent, ModalFooter.

  • Usage:

    import { Modal, ModalContent, ModalFooter } from '@stackloop/ui'
    
    <Modal isOpen={open} onClose={() => setOpen(false)} title="My Modal">
      <ModalContent>Body</ModalContent>
      <ModalFooter>
        <button onClick={() => setOpen(false)}>Close</button>
      </ModalFooter>
    </Modal>

Table:

  • Description: Generic sortable table component with loading skeleton, row interactions, and responsive design. Supports client-side sorting and custom cell rendering.

  • Props:

    • data: T[] — required. Array of data items to display in the table.
    • columns: Column<T>[] — required. Array of column definitions (see Column interface below).
    • loading: boolean — optional. Shows animated skeleton loading state when true.
    • onRowClick: (item: T) => void — optional. Callback fired when a row is clicked. Adds hover effect and pointer cursor to rows.
    • keyExtractor: (item: T) => string — required. Function to extract unique key from each data item for React reconciliation.
    • className: string — optional. Additional CSS classes for the table wrapper.
  • Column Interface:

    interface Column<T> {
      key: string;                         // Unique column identifier and default accessor key
      header: string;                      // Column header text displayed in table header
      sortable?: boolean;                  // Enable sorting for this column (default: false)
      render?: (item: T) => React.ReactNode; // Custom render function - can return any valid React element
      width?: string;                      // CSS width value (e.g., '100px', '20%', 'auto')
      truncate?: boolean;                  // Enable text truncation with ellipsis for long content (default: false)
    }
  • Custom Rendering with render:

    The render function accepts the current row item and can return any React node, including:

    • JSX elements (buttons, badges, icons)
    • Formatted strings or numbers
    • Complex components with conditional logic
    • Nested elements with multiple components

    Examples:

    // Render Badge components
    {
      key: 'status',
      header: 'Status',
      render: (item) => (
        <Badge variant={item.status === 'active' ? 'success' : 'danger'}>
          {item.status}
        </Badge>
      )
    }
    
    // Render action buttons
    {
      key: 'actions',
      header: 'Actions',
      render: (item) => (
        <div className="flex gap-2">
          <Button size="sm" onClick={() => handleEdit(item)}>Edit</Button>
          <Button size="sm" variant="danger" onClick={() => handleDelete(item)}>Delete</Button>
        </div>
      )
    }
    
    // Render images with fallback
    {
      key: 'avatar',
      header: 'Avatar',
      render: (user) => (
        <img 
          src={user.avatar || '/default-avatar.png'} 
          alt={user.name}
          className="w-10 h-10 rounded-full"
        />
      )
    }
    
    // Render formatted dates
    {
      key: 'createdAt',
      header: 'Created',
      render: (item) => new Date(item.createdAt).toLocaleDateString('en-US', {
        year: 'numeric',
        month: 'short',
        day: 'numeric'
      })
    }
    
    // Render icons with conditional colors
    {
      key: 'verified',
      header: 'Verified',
      render: (user) => user.verified ? (
        <Check className="w-5 h-5 text-success" />
      ) : (
        <X className="w-5 h-5 text-error" />
      )
    }
    
    // Render multiple values combined
    {
      key: 'fullName',
      header: 'User',
      render: (user) => (
        <div>
          <div className="font-semibold">{user.firstName} {user.lastName}</div>
          <div className="text-sm text-primary/60">{user.email}</div>
        </div>
      )
    }
    
    // Enable text truncation for long content
    {
      key: 'description',
      header: 'Description',
      truncate: true,      // Adds ellipsis when text exceeds cell width
      width: '300px'       // Set explicit width to control truncation point
    }
  • Text Truncation:

    • Set truncate: true on any column to automatically truncate long text with ellipsis (...) when content exceeds the cell width.
    • Hovering over truncated cells shows a browser tooltip with the full text.
    • Must specify width property (e.g., '200px', '30%', '20rem') to define when truncation occurs.
    • Without width, the cell will expand to fit content and truncation won't activate.
    • Only applies to default cell rendering; custom render functions handle their own truncation.

    Example:

    const columns = [
      { key: 'title', header: 'Title', sortable: true },
      { 
        key: 'description', 
        header: 'Description', 
        truncate: true,        // Enable ellipsis
        width: '250px'         // Required: defines max width before truncation
      },
      { 
        key: 'email', 
        header: 'Email', 
        truncate: true,
        width: '200px'
      },
      { key: 'author', header: 'Author' }
    ]
  • Sorting Behavior:

    • Click sortable column headers to toggle between ascending → descending → no sort.
    • Sort icons: ChevronUp (ascending), ChevronDown (descending), ChevronsUpDown (sortable but not active).
    • Sorting is client-side using JavaScript's localeCompare for strings and numeric comparison for numbers.
    • Only one column can be sorted at a time.
  • Loading State:

    • When loading={true}, displays skeleton rows with animated gradient shimmer.
    • Shows 5 skeleton rows by default, preserving table structure and column widths.
    • Loading state prevents interactions and hides real data.
  • Styling & Customization:

    • Responsive: Horizontal scroll on mobile, full table view on desktop.
    • Hover states: Rows have hover background when onRowClick is provided.
    • Colors: Uses semantic color tokens (border, border-dark, background, foreground-color, primary).
    • Animations: Powered by Framer Motion for smooth row entry and sorting transitions.
  • Complete Usage Example:

    import { Table, Badge, Button } from '@stackloop/ui'
    import { Check, X, Edit, Trash2 } from 'lucide-react'
    
    const columns = [
      { 
        key: 'id', 
        header: 'ID', 
        width: '80px',
        sortable: true 
      },
      { 
        key: 'name', 
        header: 'Name', 
        sortable: true 
      },
      { 
        key: 'status', 
        header: 'Status',
        render: (user) => (
          <Badge variant={user.status === 'active' ? 'success' : 'default'}>
            {user.status}
          </Badge>
        )
      },
      {
        key: 'verified',
        header: 'Verified',
        render: (user) => user.verified ? 
          <Check className="w-5 h-5 text-success" /> : 
          <X className="w-5 h-5 text-error" />
      },
      {
        key: 'actions',
        header: 'Actions',
        render: (user) => (
          <div className="flex gap-2">
            <Button size="sm" icon={<Edit />} onClick={() => handleEdit(user)}>
              Edit
            </Button>
            <Button size="sm" variant="danger" icon={<Trash2 />}>
              Delete
            </Button>
          </div>
        )
      }
    ]
    
    <Table 
      data={users} 
      columns={columns} 
      keyExtractor={(user) => String(user.id)}
      onRowClick={(user) => navigate(`/users/${user.id}`)}
      loading={isLoading}
    />

Dropdown:

  • Description: General-purpose select component with optional search, clear, and icons. Use this for general UI selections outside of forms. For form-specific needs with validation, use the Select component instead.

  • Props:

    • options: { value: string; label: string; icon?: ReactNode }[] — required.
    • value: string — optional.
    • onChange: (value: string) => void — required.
    • placeholder: string — default: 'Select an option'.
    • label, error, searchable (default false), clearable (default true), disabled, className.
  • Usage:

    import { Dropdown } from '@stackloop/ui'
    
    <Dropdown options={[{value:'a',label:'A'}]} value={val} onChange={setVal} searchable />

Select:

  • Description: Form-specific select component with label, error, hint, and validation support. Specifically designed for use in forms with proper semantics, accessibility, and validation integration. Use this component when building forms. For general UI selections (navigation, filters, etc.), use the Dropdown component instead. Based on Dropdown but includes form-specific features like required prop, hint text, and better integration with form libraries (React Hook Form, Formik, etc.).

  • Props:

    • options: { value: string; label: string; icon?: ReactNode; disabled?: boolean }[] — required. Array of selectable options with optional icons and disabled state.
    • value: string — optional. Currently selected value.
    • onChange: (value: string) => void — required. Callback fired when selection changes.
    • placeholder: string — default: 'Select an option'. Shown when no value is selected.
    • label: string — optional. Label displayed above the select.
    • error: string — optional. Error message displayed below select with error styling.
    • hint: string — optional. Helper text displayed below select when no error is present.
    • searchable: boolean — default: false. Enables search input to filter options.
    • clearable: boolean — default: true. Shows clear button when value is selected.
    • required: boolean — optional. Displays asterisk (*) next to label and sets aria-required.
    • disabled: boolean — optional. Disables the select.
    • className: string — optional. Additional CSS classes for the wrapper.
  • Features:

    • Form Integration: Works seamlessly with form libraries (React Hook Form, Formik, etc.)
    • Validation Support: Built-in error and hint display with animated transitions
    • Accessibility: Full ARIA attributes, keyboard navigation, and screen reader support
    • Search: Optional searchable mode with real-time filtering
    • Icons: Support for icons in options for better visual recognition
    • Disabled Options: Individual options can be disabled while keeping select active
    • Clearable: Optional clear button to reset selection
    • Touch-Friendly: Optimized for mobile with proper touch targets
  • Usage:

    import { Select } from '@stackloop/ui'
    import { User, Settings, Bell } from 'lucide-react'
    
    // Basic usage
    <Select 
      label="Country"
      options={[
        { value: 'us', label: 'United States' },
        { value: 'uk', label: 'United Kingdom' },
        { value: 'ca', label: 'Canada' }
      ]} 
      value={country} 
      onChange={setCountry}
      required
    />
    
    // With icons and search
    <Select 
      label="Navigation"
      options={[
        { value: 'profile', label: 'Profile', icon: <User /> },
        { value: 'settings', label: 'Settings', icon: <Settings /> },
        { value: 'notifications', label: 'Notifications', icon: <Bell />, disabled: true }
      ]}
      value={selected}
      onChange={setSelected}
      searchable
      placeholder="Choose a page"
    />
    
    // With error and hint
    <Select 
      label="Payment Method"
      options={paymentMethods}
      value={payment}
      onChange={setPayment}
      error={errors.payment}
      hint="Select your preferred payment method"
      required
    />
    
    // With React Hook Form
    import { useForm, Controller } from 'react-hook-form'
    
    const { control, formState: { errors } } = useForm()
    
    <Controller
      name="category"
      control={control}
      rules={{ required: 'Category is required' }}
      render={({ field }) => (
        <Select
          label="Category"
          options={categories}
          value={field.value}
          onChange={field.onChange}
          error={errors.category?.message}
          required
        />
      )}
    />

BottomSheet:

  • Description: Mobile bottom sheet with header and optional close button.

  • Props:

    • isOpen: boolean — required.
    • onClose: () => void — required.
    • title: string — optional.
    • children: ReactNode — required.
    • showCloseButton: boolean — default: true.
    • className: string — optional.
  • Usage:

    import { BottomSheet } from '@stackloop/ui'
    
    <BottomSheet isOpen={open} onClose={() => setOpen(false)} title="Actions">...</BottomSheet>

DatePicker:

  • Description: Date picker with month navigation and min/max options.

  • Props:

    • value: Date — optional.
    • onChange: (date: Date) => void — required.
    • label, placeholder (default 'Select date'), error, disabled, minDate, maxDate, className.
  • Usage:

    import { DatePicker } from '@stackloop/ui'
    
    <DatePicker value={date} onChange={setDate} />

DualSlider:

  • Description: Two linked sliders showing relative portions.

  • Props:

    • value1, value2: number — required.
    • onChange: (v1:number, v2:number)=>void — required.
    • label1, label2: string — required.
    • min (default 0), max (default 100), step (default 1), unit (default '%'), disabled, className.
  • Usage:

    import { DualSlider } from '@stackloop/ui'
    
    <DualSlider value1={30} value2={70} onChange={(a,b)=>{}} label1="A" label2="B" />

Pagination:

  • Description: Pagination with previous/next and numeric buttons.

  • Props:

    • currentPage: number — required.
    • totalPages: number — required.
    • onPageChange: (page:number)=>void — required.
    • totalItems, itemsPerPage, className — optional.
  • Usage:

    import { Pagination } from '@stackloop/ui'
    
    <Pagination currentPage={1} totalPages={10} onPageChange={setPage} />

Badge:

  • Description: Inline badge with color variants and optional dot.

  • Props:

    • children: ReactNode — required.
    • variant: 'default'|'primary'|'success'|'warning'|'danger'|'info' — default: 'default'.
    • size: 'sm'|'md'|'lg' — default: 'md'.
    • dot: boolean — default: false.
    • className: string — optional.
  • Usage:

    import { Badge } from '@stackloop/ui'
    
    <Badge variant="primary">New</Badge>

Spinner:

  • Description: Animated loading spinner with size variants and optional label.

  • Props:

    • size: 'sm' | 'md' | 'lg' | 'xl' — default: 'md'. Controls spinner dimensions (sm=16px, md=32px, lg=48px, xl=64px).
    • variant: 'primary' | 'secondary' | 'white' — default: 'primary'. Color variant of the spinner.
    • label: string — optional. Text displayed below spinner.
    • className: string — optional. Additional CSS classes for the wrapper.
  • Usage:

    import { Spinner } from '@stackloop/ui'
    
    // Basic spinner
    <Spinner />
    
    // With label
    <Spinner label="Loading..." />
    
    // Custom size and variant
    <Spinner size="lg" variant="white" />
    
    // In a button
    <Button disabled>
      <Spinner size="sm" variant="white" label="Processing..." />
    </Button>

Toast:

  • Description: Context-based toast notification system with variants, actions, and customizable positioning. Provides non-intrusive feedback messages that auto-dismiss.

  • Components:

    • ToastProvider: Wrapper component that manages toast state and rendering. Must wrap your app or components that use toasts.
    • useToast: Hook to trigger toasts from any component within the provider.
  • ToastProvider Props:

    • children: ReactNode — required. Your app content.
    • position: 'top-left' | 'top-center' | 'top-right' | 'bottom-left' | 'bottom-center' | 'bottom-right' — default: 'top-right'. Where toasts appear on screen.
    • maxToasts: number — default: 5. Maximum number of toasts shown at once.
  • useToast() Returns:

    • addToast(toast): Function to create a new toast notification.
    • removeToast(id): Function to manually dismiss a toast.
    • toasts: Array of currently active toasts.
  • Toast Object:

    • message: string — required. Toast content text.
    • variant: 'success' | 'error' | 'warning' | 'info' | 'default' — optional (default: 'default'). Visual style with corresponding icon.
    • duration: number — optional (default: 5000ms). Auto-dismiss time in milliseconds. Set to 0 for persistent toast.
    • action: { label: string; onClick: () => void } — optional. Action button within the toast.
  • Features:

    • Auto-dismiss: Toasts automatically disappear after duration
    • Manual dismiss: Click X button or call removeToast(id)
    • Variants with icons: Visual feedback with CheckCircle, AlertCircle, AlertTriangle, Info icons
    • Action buttons: Add clickable actions to toasts
    • Animations: Smooth enter/exit animations with Framer Motion
    • Position control: Place toasts anywhere on screen
    • Max limit: Prevents toast overflow
    • Responsive: Adapts to mobile screens
  • Usage:

    import { ToastProvider, useToast } from '@stackloop/ui'
    
    // 1. Wrap your app with ToastProvider
    function App() {
      return (
        <ToastProvider position="top-right" maxToasts={5}>
          <YourApp />
        </ToastProvider>
      )
    }
    
    // 2. Use toasts in any component
    function MyComponent() {
      const { addToast } = useToast()
    
      const handleSuccess = () => {
        addToast({
          message: 'Profile updated successfully!',
          variant: 'success',
          duration: 3000
        })
      }
    
      const handleError = () => {
        addToast({
          message: 'Failed to save changes',
          variant: 'error',
          duration: 5000
        })
      }
    
      const handleWithAction = () => {
        addToast({
          message: 'New message received',
          variant: 'info',
          duration: 0, // Persistent until dismissed
          action: {
            label: 'View',
            onClick: () => navigate('/messages')
          }
        })
      }
    
      const handleWarning = () => {
        addToast({
          message: 'Your session will expire in 5 minutes',
          variant: 'warning'
        })
      }
    
      return (
        <div>
          <Button onClick={handleSuccess}>Save</Button>
          <Button onClick={handleError}>Trigger Error</Button>
          <Button onClick={handleWithAction}>Show Notification</Button>
          <Button onClick={handleWarning}>Show Warning</Button>
        </div>
      )
    }
    
    // 3. Bottom-centered positioning
    <ToastProvider position="bottom-center">
      <App />
    </ToastProvider>

FloatingActionButton (FAB):

  • Description: Floating action button with expanded action list support.

  • Props:

    • icon, label, onClick, actions (array of {label, icon, onClick, variant, labelClassName?, labelStyle?}) — optional.
    • variant: 'primary'|'secondary' — default: 'primary'.
    • position: 'bottom-right'|'bottom-left'|'bottom-center' — default: 'bottom-right'.
    • disabled, className.
  • Action Props:

    • label: string — Action label text.
    • icon: ReactNode — Icon for the action button.
    • onClick: () => void — Action callback.
    • variant: 'primary'|'success'|'danger'|'warning' — Action button color variant.
    • labelClassName: string — Custom CSS classes for the label (e.g., remove background: '!bg-transparent !border-0 text-white').
    • labelStyle: CSSProperties — Inline styles for the label.
  • Usage:

    import { FloatingActionButton as FAB } from '@stackloop/ui'
    
    // Simple FAB
    <FAB label="New" onClick={()=>{}} />
    
    // FAB with actions and custom label styling
    <FAB 
      actions={[
        {
          label: 'Create Post',
          icon: <PlusIcon />,
          onClick: () => {},
          variant: 'primary',
          labelClassName: 'bg-blue-500 text-white border-blue-600'
        },
        {
          label: 'Upload',
          icon: <UploadIcon />,
          onClick: () => {},
          variant: 'success',
          labelClassName: '!bg-transparent !border-0 text-white shadow-none'
        }
      ]}
    />

AudioRecorder:

  • Description: Browser audio recorder component using MediaRecorder.

  • Props:

    • onRecordingComplete: (audioBlob: Blob) => void — required.
    • label: string — default: 'Record Audio'.
    • maxDuration: number — default: 300 seconds.
    • disabled, className.
  • Usage:

    import { AudioRecorder } from '@stackloop/ui'
    
    <AudioRecorder onRecordingComplete={(blob)=>{ /* handle */ }} />

Drawer:

  • Description: Side drawer that slides in from the left or right.

  • Props:

    • isOpen: boolean — required.
    • onClose: () => void — required.
    • children: ReactNode — required.
    • title: string — optional.
    • position: 'left'|'right' — default: 'right'.
    • className: string — optional.
  • Usage:

    import { Drawer } from '@stackloop/ui'
    
    <Drawer isOpen={open} onClose={()=>setOpen(false)} title="Menu">...</Drawer>

Toggle:

  • Description: Switch control with label and description.

  • Props:

    • label: string — optional.
    • description: string — optional.
    • onChange: (checked:boolean)=>void — optional.
    • className: string — optional.
    • Inherits input attributes such as checked, disabled.
  • Usage:

    import { Toggle } from '@stackloop/ui'
    
    <Toggle label="Enable" onChange={(v)=>console.log(v)} />

StatusBadges (OfflineBadge, SyncIndicator):

  • OfflineBadge Props: isOffline: boolean (required), className?: string.

  • SyncIndicator Props: status: 'synced'|'syncing'|'unsynced'|'error' (required), count?: number, className?: string.

  • Usage:

    import { OfflineBadge, SyncIndicator } from '@stackloop/ui'
    
    <OfflineBadge isOffline={true} />
    <SyncIndicator status="syncing" />

Slider:

  • Description: Single-value slider with optional unit and label.

  • Props:

    • value: number — required.
    • onChange: (value:number)=>void — required.
    • min (default 0), max (default 100), step (default 1), label, showValue (default true), unit (default '%'), disabled, className.
  • Usage:

    import { Slider } from '@stackloop/ui'
    
    <Slider value={50} onChange={setValue} />

RadioPills:

  • Description: Radio group styled as pill buttons.

  • Props:

    • options: { value: string; label: string; icon?: ReactNode }[] — required.
    • value: string — optional.
    • onChange: (v:string)=>void — optional.
    • name: string — required.
    • disabled: boolean — optional.
    • className: string — optional.
  • Usage:

    import { RadioPills } from '@stackloop/ui'
    
    <RadioPills name="mode" options={[{value:'a',label:'A'}]} />

Textarea:

  • Description: Multiline input with label, error and helper text.

  • Props:

    • label: string — optional.
    • error: string — optional.
    • helperText: string — optional.
    • className: string — optional.
    • Inherits native textarea attributes.
  • Usage:

    import { Textarea } from '@stackloop/ui'
    
    <Textarea label="Message" helperText="Max 500 chars" />

ThumbnailGrid:

  • Description: Grid to display thumbnails for images, documents, audio with optional remove/view actions.

  • Props:

    • items: { id, name, url, type:'image'|'document'|'audio', size? }[] — required.
    • onRemove: (id:string)=>void — optional.
    • onView: (item)=>void — optional.
    • columns: 2|3|4 — default: 3.
    • className.
  • Usage:

    import { ThumbnailGrid } from '@stackloop/ui'
    
    <ThumbnailGrid items={items} onRemove={(id)=>{}} />

Card:

  • Description: Generic card container with variants and content subcomponents.

  • Props:

    • children: ReactNode — required.
    • variant: 'default'|'outlined'|'elevated' — default: 'default'.
    • padding: 'sm'|'md'|'lg'|'none' — default: 'md'.
    • onClick: () => void — optional.
    • hover: boolean — default: false.
    • className: string — optional.
  • Subcomponents: CardHeader, CardTitle, CardDescription, CardContent.

  • Usage:

    import { Card, CardTitle, CardContent } from '@stackloop/ui'
    
    <Card variant="elevated"> <CardTitle>Title</CardTitle> <CardContent>Body</CardContent> </Card>

FileUpload (CameraCapture, FileUploader):

  • CameraCapture Props: onCapture(file:File) => void (required), onRemove?, preview?, label?, disabled?, className?.

  • FileUploader Props: onUpload(files:File[]) => void (required), accept? (default '*/*'), multiple? (default false), label?, disabled?, className?.

  • Usage:

    import { FileUploader, CameraCapture } from '@stackloop/ui'
    
    <FileUploader onUpload={(files)=>{}} multiple accept="image/*" />
    <CameraCapture onCapture={(file)=>{}} />

StepProgress:

  • Description: Horizontal stepper (desktop) and compact dots view (mobile) showing progress.

  • Props:

    • steps: { label: string; description?: string }[] — required.
    • currentStep: number — required.
    • className: optional.
  • Usage:

    import { StepProgress } from '@stackloop/ui'
    
    <StepProgress steps={[{label:'One'},{label:'Two'}]} currentStep={1} />

CreditBar:

  • Description: Thin footer attribution bar for showing developer/company credit. Perfect for "Built with ❤️ by..." displays.

  • Props:

    • text: string — default: 'Built by StackLoop'. The company/developer name to display.
    • showHeart: boolean — default: true. Shows heart emoji in "Built with ❤️ by..." format.
    • href: string — optional. Link URL to make the credit clickable.
    • position: 'fixed' | 'relative' | 'sticky' — default: 'fixed'. Positioning of the bar.
    • backgroundColor: string — default: 'bg-gray-900'. Tailwind background class.
    • textColor: string — default: 'text-gray-400'. Tailwind text color class.
    • className: string — optional. Additional CSS classes.
  • Usage:

    import { CreditBar } from '@stackloop/ui'
    
    // Simple usage with heart
    <CreditBar text="StackLoop" />
    // Displays: "Built with ❤️ by StackLoop"
    
    // With link
    <CreditBar text="StackLoop" href="https://stackloop.com" />
    
    // Without heart
    <CreditBar text="Your Company" showHeart={false} />
    
    // Custom styling
    <CreditBar 
      text="StackLoop"
      backgroundColor="bg-blue-900"
      textColor="text-blue-300"
      position="sticky"
    />

Next.js Best Practices

App Router

All components work seamlessly with Next.js 13+ App Router. They include the 'use client' directive:

// app/page.tsx
import { Button, Card } from '@stackloop/ui'

export default function Page() {
  return (
    <div>
      <Card>
        <Button onClick={() => console.log('clicked')}>Click me</Button>
      </Card>
    </div>
  )
}

Server Components

To use these components with Server Components, import them in client components:

// components/ClientWrapper.tsx
'use client'
import { Button } from '@stackloop/ui'

export function ClientButton() {
  return <Button onClick={() => alert('Hello')}>Click</Button>
}
// app/page.tsx (Server Component)
import { ClientButton } from '@/components/ClientWrapper'

export default function Page() {
  return <ClientButton />
}

Styling Considerations

  • The library uses Tailwind CSS v4 with @theme directive
  • Ensure your Next.js project is configured for Tailwind CSS v4
  • All animations use Framer Motion and are optimized for performance

Browser Support

  • Modern browsers (Chrome, Firefox, Safari, Edge)
  • Mobile browsers (iOS Safari, Chrome Mobile)
  • Audio recording requires MediaRecorder API support

License

MIT


For questions, issues, or contributions, please visit the GitHub repository.