Textarea

Installation

Add the following Soul components

The textarea component uses the field-error and label 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

form/textarea/index.tsx

import { clsx } from 'clsx';import * as React from 'react';import { FieldError } from '@/vibes/soul/form/field-error';import { Label } from '@/vibes/soul/form/label';/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css *  :root { *    --textarea-light-background: hsl(var(--background)); *    --textarea-light-text: hsl(var(--foreground)); *    --textarea-light-placeholder: hsl(var(--contrast-gray-500)); *    --textarea-light-border: hsl(var(--border-contrast-100)); *    --textarea-light-border-focus: hsl(var(--foreground)); *    --textarea-light-border-error: hsl(var(--error)); *    --textarea-dark-background: hsl(var(--foreground)); *    --textarea-dark-text: hsl(var(--background)); *    --textarea-dark-placeholder: hsl(var(--contrast-gray-100)); *    --textarea-dark-border: hsl(var(--border-contrast-500)); *    --textarea-dark-border-focus: hsl(var(--background)); *    --textarea-dark-border-error: hsl(var(--error)); *  } * ``` */export const Textarea = React.forwardRef<  React.ComponentRef<'textarea'>,  React.ComponentPropsWithoutRef<'textarea'> & {    prepend?: React.ReactNode;    label?: string;    errors?: string[];    colorScheme?: 'light' | 'dark';  }>(({ label, className, required, errors, colorScheme = 'light', ...rest }, ref) => {  const id = React.useId();  return (    <div className={clsx('space-y-2', className)}>      {label != null && label !== '' && (        <Label colorScheme={colorScheme} htmlFor={id}>          {label}        </Label>      )}      <textarea        {...rest}        className={clsx(          'w-full rounded-lg border p-3 transition-colors duration-200 placeholder:font-normal focus:outline-none disabled:cursor-not-allowed disabled:opacity-50',          {            light:              'bg-[var(--textarea-light-background,hsl(var(--background)))] text-[var(--textarea-light-text,hsl(var(--foreground)))] placeholder-[var(--textarea-light-placeholder,hsl(var(--contrast-gray-500)))] focus:border-[var(--textarea-light-border-focus,hsl(var(--foreground)))]',            dark: 'bg-[var(--textarea-dark-background,hsl(var(--foreground)))] text-[var(--textarea-dark-text,hsl(var(--background)))] placeholder-[var(--textarea-dark-placeholder,hsl(var(--contrast-gray-100)))] focus:border-[var(--textarea-dark-border-focus,hsl(var(--background)))]',          }[colorScheme],          {            light:              errors && errors.length > 0                ? 'border-[var(--textarea-light-border-error,hsl(var(--error)))]'                : 'border-[var(--textarea-light-border,hsl(var(--border-contrast-100)))]',            dark:              errors && errors.length > 0                ? 'border-[var(--textarea-dark-border-error,hsl(var(--error)))]'                : 'border-[var(--textarea-dark-border,hsl(var(--border-contrast-500)))]',          }[colorScheme],        )}        id={id}        ref={ref}      />      {errors?.map((error) => <FieldError key={error}>{error}</FieldError>)}    </div>  );});Textarea.displayName = 'Textarea';

Usage

import { Textarea } from '@/vibes/soul/form/textarea';function Usage() {  return (      <Textarea />  );}

API Reference

TextareaProps

PropTypeDefault
className
string
prepend
React.ReactNode
label
string
errors
string[]
colorScheme
'light' | 'dark'
'light'

CSS Variables

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

:root {  --textarea-light-background: hsl(var(--background));  --textarea-light-text: hsl(var(--foreground));  --textarea-light-placeholder: hsl(var(--contrast-gray-500));  --textarea-light-border: hsl(var(--border-contrast-100));  --textarea-light-border-focus: hsl(var(--foreground));  --textarea-light-border-error: hsl(var(--error));  --textarea-dark-background: hsl(var(--foreground));  --textarea-dark-text: hsl(var(--background));  --textarea-dark-placeholder: hsl(var(--contrast-gray-100));  --textarea-dark-border: hsl(var(--border-contrast-500));  --textarea-dark-border-focus: hsl(var(--background));  --textarea-dark-border-error: hsl(var(--error));}