Swatch Radio Group

Installation

Add the following Soul components

The swatch-radio-group component uses the field-error and label components. Make sure you have added them to your project.

Install the following dependencies

npm install clsx lucide-react @radix-ui/react-radio-group

Copy and paste the following code into your project

form/swatch-radio-group/index.tsx

import * as RadioGroupPrimitive from '@radix-ui/react-radio-group';import { clsx } from 'clsx';import { X } from 'lucide-react';import Image from 'next/image';import * as React from 'react';import { FieldError } from '@/vibes/soul/form/field-error';import { Label } from '@/vibes/soul/form/label';type SwatchOption =  | {      type: 'color';      value: string;      label: string;      color: string;      disabled?: boolean;    }  | {      type: 'image';      value: string;      label: string;      image: { src: string; alt: string };      disabled?: boolean;    };/** * This component supports various CSS variables for theming. Here's a comprehensive list, along * with their default values: * * ```css *  :root { *    --swatch-radio-group-focus: hsl(var(--primary)); *    --swatch-radio-group-light-icon: hsl(var(--foreground)); *    --swatch-radio-group-light-unchecked-border: transparent; *    --swatch-radio-group-light-unchecked-border-hover: hsl(var(--border-contrast-200)); *    --swatch-radio-group-light-disabled-border: transparent; *    --swatch-radio-group-light-border-error: hsl(var(--error)); *    --swatch-radio-group-light-checked-border: hsl(var(--foreground)); *    --swatch-radio-group-light-option-border: hsl(var(--foreground) / 10%); *    --swatch-radio-group-dark-icon: hsl(var(--background)); *    --swatch-radio-group-dark-unchecked-border: transparent; *    --swatch-radio-group-dark-unchecked-border-hover: hsl(var(--border-contrast-400)); *    --swatch-radio-group-dark-disabled-border: transparent; *    --swatch-radio-group-dark-border-error: hsl(var(--error)); *    --swatch-radio-group-dark-checked-border: hsl(var(--background)); *    --swatch-radio-group-dark-option-border: hsl(var(--background) / 10%); *  } * ``` */export const SwatchRadioGroup = React.forwardRef<  React.ComponentRef<typeof RadioGroupPrimitive.Root>,  React.ComponentPropsWithoutRef<typeof RadioGroupPrimitive.Root> & {    label?: string;    options: SwatchOption[];    errors?: string[];    colorScheme?: 'light' | 'dark';    onOptionMouseEnter?: (value: string) => void;  }>(  (    { label, options, errors, className, colorScheme = 'light', onOptionMouseEnter, ...rest },    ref,  ) => {    const id = React.useId();    return (      <div className={clsx('space-y-2', className)}>        {label !== undefined && label !== '' && (          <Label colorScheme={colorScheme} id={id}>            {label}          </Label>        )}        <RadioGroupPrimitive.Root          {...rest}          aria-labelledby={id}          className="flex flex-wrap gap-1"          ref={ref}        >          {options.map((option) => (            <RadioGroupPrimitive.Item              aria-label={option.label}              className={clsx(                'group relative box-content h-8 w-8 rounded-full border p-0.5 transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-[var(--swatch-radio-group-focus,hsl(var(--primary)))] data-[disabled]:pointer-events-none [&:disabled>.disabled-icon]:grid',                {                  light:                    'hover:border-[var(--swatch-radio-group-light-unchecked-border-hover,hsl(var(--border-contrast-200)))] data-[state=checked]:border-[var(--swatch-radio-group-light-checked-border,hsl(var(--foreground)))]',                  dark: 'hover:border-[var(--swatch-radio-group-dark-unchecked-border-hover,hsl(var(--border-contrast-400)))] data-[state=checked]:border-[var(--swatch-radio-group-dark-checked-border,hsl(var(--background)))]',                }[colorScheme],                {                  light:                    errors && errors.length > 0                      ? 'border-[var(--swatch-radio-group-light-border-error,hsl(var(--error)))] disabled:border-[var(--swatch-radio-group-light-disabled-border,transparent)]'                      : 'border-[var(--swatch-radio-group-light-unchecked-border,transparent)]',                  dark:                    errors && errors.length > 0                      ? 'border-[var(--swatch-radio-group-dark-border-error,hsl(var(--error)))] disabled:border-[var(--swatch-radio-group-dark-disabled-border,transparent)]'                      : 'border-[var(--swatch-radio-group-dark-unchecked-border,transparent)]',                }[colorScheme],              )}              disabled={option.disabled}              key={option.value}              onMouseEnter={() => {                onOptionMouseEnter?.(option.value);              }}              value={option.value}            >              {option.type === 'color' ? (                <span                  className={clsx(                    'block size-full rounded-full border group-disabled:opacity-20',                    {                      light:                        'border-[var(--swatch-radio-group-light-option-border,hsl(var(--foreground)/10%))]',                      dark: 'border-[var(--swatch-radio-group-dark-option-border,hsl(var(--background)/10%))]',                    }[colorScheme],                  )}                  style={{ backgroundColor: option.color }}                />              ) : (                <span                  className={clsx(                    'relative block size-full overflow-hidden rounded-full border',                    {                      light:                        'border-[var(--swatch-radio-group-light-option-border,hsl(var(--foreground)/10%))]',                      dark: 'border-[var(--swatch-radio-group-dark-option-border,hsl(var(--background)/10%))]',                    }[colorScheme],                  )}                >                  <Image alt={option.image.alt} height={40} src={option.image.src} width={40} />                </span>              )}              <div                className={clsx(                  'disabled-icon absolute inset-0 hidden place-content-center',                  {                    light: 'text-[var(--swatch-radio-group-light-icon,hsl(var(--foreground)))]',                    dark: 'text-[var(--swatch-radio-group-dark-icon,hsl(var(--background)))]',                  }[colorScheme],                )}              >                <X size={16} strokeWidth={1.5} />              </div>            </RadioGroupPrimitive.Item>          ))}        </RadioGroupPrimitive.Root>        {errors?.map((error) => <FieldError key={error}>{error}</FieldError>)}      </div>    );  },);SwatchRadioGroup.displayName = 'SwatchRadioGroup';

Usage

'use client';import { SwatchRadioGroup } from '@/vibes/soul/form/swatch-radio-group';function Usage() {  return (      <SwatchRadioGroup        options={[          { type: 'color', label: 'Option 1', value: 'option-1', color: 'red' },          { type: 'color', label: 'Option 2', value: 'option-2', color: 'green' },          { type: 'color', label: 'Option 3', value: 'option-3', color: 'blue' },        ]}      />  );}

API Reference

SwatchRadioGroupProps

PropTypeDefault
className
string
label
string
options
SwatchOption[]
errors
string[]
colorScheme
'light' | 'dark'
'light'
onOptionMouseEnter
(value: string) => void

SwatchOption

type SwatchOption =  | {      type: 'color';      value: string;      label: string;      color: string;      disabled?: boolean;    }  | {      type: 'image';      value: string;      label: string;      image: { src: string; alt: string };      disabled?: boolean;    };

CSS Variables

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

:root {  --swatch-radio-group-focus: hsl(var(--primary));  --swatch-radio-group-light-icon: hsl(var(--foreground));  --swatch-radio-group-light-unchecked-border: transparent;  --swatch-radio-group-light-unchecked-border-hover: hsl(var(--border-contrast-200));  --swatch-radio-group-light-disabled-border: transparent;  --swatch-radio-group-light-border-error: hsl(var(--error));  --swatch-radio-group-light-checked-border: hsl(var(--foreground));  --swatch-radio-group-light-option-border: hsl(var(--foreground) / 10%);  --swatch-radio-group-dark-icon: hsl(var(--background));  --swatch-radio-group-dark-unchecked-border: transparent;  --swatch-radio-group-dark-unchecked-border-hover: hsl(var(--border-contrast-400));  --swatch-radio-group-dark-disabled-border: transparent;  --swatch-radio-group-dark-border-error: hsl(var(--error));  --swatch-radio-group-dark-checked-border: hsl(var(--background));  --swatch-radio-group-dark-option-border: hsl(var(--background) / 10%);}