atomic-design-atoms

Master the creation of atomic components - the fundamental, indivisible building blocks of your design system. Atoms are the smallest functional units that cannot be broken down further without losing meaning.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "atomic-design-atoms" with this command: npx skills add thebushidocollective/han/thebushidocollective-han-atomic-design-atoms

Atomic Design: Atoms

Master the creation of atomic components - the fundamental, indivisible building blocks of your design system. Atoms are the smallest functional units that cannot be broken down further without losing meaning.

What Are Atoms?

Atoms are the basic UI elements that serve as the foundation for everything else in your design system. They are:

  • Indivisible: Cannot be broken down into smaller functional units

  • Reusable: Used throughout the application in various contexts

  • Stateless: Typically controlled by parent components

  • Styled: Implement design tokens for consistent appearance

  • Accessible: Built with a11y in mind from the start

Common Atom Types

Interactive Atoms

  • Buttons

  • Links

  • Inputs (text, checkbox, radio, select)

  • Toggles/Switches

  • Sliders

Display Atoms

  • Typography (headings, paragraphs, labels)

  • Icons

  • Images/Avatars

  • Badges/Tags

  • Dividers

  • Spinners/Loaders

Form Atoms

  • Input fields

  • Textareas

  • Checkboxes

  • Radio buttons

  • Select dropdowns

  • Labels

Button Atom Example

Basic Implementation

// atoms/Button/Button.tsx import React from 'react'; import type { ButtonHTMLAttributes } from 'react'; import styles from './Button.module.css';

export type ButtonVariant = 'primary' | 'secondary' | 'tertiary' | 'danger'; export type ButtonSize = 'sm' | 'md' | 'lg';

export interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> { /** Visual style variant / variant?: ButtonVariant; /* Size of the button / size?: ButtonSize; /* Full width button / fullWidth?: boolean; /* Loading state / isLoading?: boolean; /* Left icon / leftIcon?: React.ReactNode; /* Right icon */ rightIcon?: React.ReactNode; }

export const Button = React.forwardRef<HTMLButtonElement, ButtonProps>( ( { variant = 'primary', size = 'md', fullWidth = false, isLoading = false, leftIcon, rightIcon, disabled, children, className, ...props }, ref ) => { const classNames = [ styles.button, styles[variant], styles[size], fullWidth && styles.fullWidth, isLoading && styles.loading, className, ] .filter(Boolean) .join(' ');

return (
  &#x3C;button
    ref={ref}
    className={classNames}
    disabled={disabled || isLoading}
    aria-busy={isLoading}
    {...props}
  >
    {isLoading ? (
      &#x3C;span className={styles.spinner} aria-hidden="true" />
    ) : (
      &#x3C;>
        {leftIcon &#x26;&#x26; &#x3C;span className={styles.leftIcon}>{leftIcon}&#x3C;/span>}
        &#x3C;span className={styles.content}>{children}&#x3C;/span>
        {rightIcon &#x26;&#x26; &#x3C;span className={styles.rightIcon}>{rightIcon}&#x3C;/span>}
      &#x3C;/>
    )}
  &#x3C;/button>
);

} );

Button.displayName = 'Button';

Button Styles

/* atoms/Button/Button.module.css */ .button { display: inline-flex; align-items: center; justify-content: center; gap: 8px; border: none; border-radius: 6px; font-weight: 500; cursor: pointer; transition: all 150ms ease-in-out; text-decoration: none; }

.button:focus-visible { outline: 2px solid var(--color-focus); outline-offset: 2px; }

.button:disabled { opacity: 0.5; cursor: not-allowed; }

/* Variants */ .primary { background-color: var(--color-primary-500); color: var(--color-white); }

.primary:hover:not(:disabled) { background-color: var(--color-primary-600); }

.secondary { background-color: transparent; color: var(--color-primary-500); border: 1px solid var(--color-primary-500); }

.secondary:hover:not(:disabled) { background-color: var(--color-primary-50); }

.tertiary { background-color: transparent; color: var(--color-primary-500); }

.tertiary:hover:not(:disabled) { background-color: var(--color-primary-50); }

.danger { background-color: var(--color-danger-500); color: var(--color-white); }

.danger:hover:not(:disabled) { background-color: var(--color-danger-600); }

/* Sizes */ .sm { padding: 6px 12px; font-size: 14px; min-height: 32px; }

.md { padding: 8px 16px; font-size: 16px; min-height: 40px; }

.lg { padding: 12px 24px; font-size: 18px; min-height: 48px; }

/* Modifiers */ .fullWidth { width: 100%; }

.loading { position: relative; color: transparent; }

.spinner { position: absolute; width: 16px; height: 16px; border: 2px solid currentColor; border-right-color: transparent; border-radius: 50%; animation: spin 0.75s linear infinite; }

@keyframes spin { to { transform: rotate(360deg); } }

/* Icon spacing */ .leftIcon, .rightIcon { display: flex; align-items: center; }

Input Atom Example

// atoms/Input/Input.tsx import React from 'react'; import type { InputHTMLAttributes } from 'react'; import styles from './Input.module.css';

export type InputSize = 'sm' | 'md' | 'lg';

export interface InputProps extends Omit<InputHTMLAttributes<HTMLInputElement>, 'size'> { /** Size variant / size?: InputSize; /* Error state / hasError?: boolean; /* Left addon element / leftAddon?: React.ReactNode; /* Right addon element */ rightAddon?: React.ReactNode; }

export const Input = React.forwardRef<HTMLInputElement, InputProps>( ( { size = 'md', hasError = false, leftAddon, rightAddon, disabled, className, ...props }, ref ) => { const wrapperClasses = [ styles.wrapper, styles[size], hasError && styles.error, disabled && styles.disabled, className, ] .filter(Boolean) .join(' ');

return (
  &#x3C;div className={wrapperClasses}>
    {leftAddon &#x26;&#x26; &#x3C;span className={styles.leftAddon}>{leftAddon}&#x3C;/span>}
    &#x3C;input
      ref={ref}
      className={styles.input}
      disabled={disabled}
      aria-invalid={hasError}
      {...props}
    />
    {rightAddon &#x26;&#x26; &#x3C;span className={styles.rightAddon}>{rightAddon}&#x3C;/span>}
  &#x3C;/div>
);

} );

Input.displayName = 'Input';

/* atoms/Input/Input.module.css */ .wrapper { display: flex; align-items: center; border: 1px solid var(--color-neutral-300); border-radius: 6px; background-color: var(--color-white); transition: border-color 150ms, box-shadow 150ms; }

.wrapper:focus-within { border-color: var(--color-primary-500); box-shadow: 0 0 0 3px var(--color-primary-100); }

.input { flex: 1; border: none; background: transparent; outline: none; width: 100%; }

.input::placeholder { color: var(--color-neutral-400); }

/* Error state */ .error { border-color: var(--color-danger-500); }

.error:focus-within { border-color: var(--color-danger-500); box-shadow: 0 0 0 3px var(--color-danger-100); }

/* Disabled state */ .disabled { background-color: var(--color-neutral-100); cursor: not-allowed; }

.disabled .input { cursor: not-allowed; }

/* Sizes */ .sm { min-height: 32px; }

.sm .input { padding: 6px 12px; font-size: 14px; }

.md { min-height: 40px; }

.md .input { padding: 8px 12px; font-size: 16px; }

.lg { min-height: 48px; }

.lg .input { padding: 12px 16px; font-size: 18px; }

/* Addons */ .leftAddon, .rightAddon { display: flex; align-items: center; padding: 0 12px; color: var(--color-neutral-500); }

Label Atom Example

// atoms/Label/Label.tsx import React from 'react'; import type { LabelHTMLAttributes } from 'react'; import styles from './Label.module.css';

export interface LabelProps extends LabelHTMLAttributes<HTMLLabelElement> { /** Indicates required field / required?: boolean; /* Disabled state styling */ disabled?: boolean; }

export const Label = React.forwardRef<HTMLLabelElement, LabelProps>( ({ required = false, disabled = false, children, className, ...props }, ref) => { const classNames = [ styles.label, disabled && styles.disabled, className, ] .filter(Boolean) .join(' ');

return (
  &#x3C;label ref={ref} className={classNames} {...props}>
    {children}
    {required &#x26;&#x26; (
      &#x3C;span className={styles.required} aria-hidden="true">
        *
      &#x3C;/span>
    )}
  &#x3C;/label>
);

} );

Label.displayName = 'Label';

Icon Atom Example

// atoms/Icon/Icon.tsx import React from 'react';

export type IconSize = 'xs' | 'sm' | 'md' | 'lg' | 'xl';

const sizeMap: Record<IconSize, number> = { xs: 12, sm: 16, md: 20, lg: 24, xl: 32, };

export interface IconProps extends React.SVGAttributes<SVGElement> { /** Icon name/identifier / name: string; /* Icon size / size?: IconSize; /* Custom color / color?: string; /* Accessible label */ label?: string; }

export const Icon: React.FC<IconProps> = ({ name, size = 'md', color = 'currentColor', label, className, ...props }) => { const pixelSize = sizeMap[size];

return ( <svg className={className} width={pixelSize} height={pixelSize} fill={color} aria-label={label} aria-hidden={!label} role={label ? 'img' : 'presentation'} {...props} > <use href={/icons.svg#${name}} /> </svg> ); };

Icon.displayName = 'Icon';

Avatar Atom Example

// atoms/Avatar/Avatar.tsx import React from 'react'; import styles from './Avatar.module.css';

export type AvatarSize = 'xs' | 'sm' | 'md' | 'lg' | 'xl';

export interface AvatarProps { /** Image source URL / src?: string; /* Alt text for image / alt: string; /* Fallback initials / initials?: string; /* Size variant / size?: AvatarSize; /* Additional class name */ className?: string; }

export const Avatar: React.FC<AvatarProps> = ({ src, alt, initials, size = 'md', className, }) => { const [imageError, setImageError] = React.useState(false);

const classNames = [styles.avatar, styles[size], className] .filter(Boolean) .join(' ');

const showImage = src && !imageError; const showInitials = !showImage && initials;

return ( <div className={classNames} role="img" aria-label={alt}> {showImage && ( <img src={src} alt={alt} className={styles.image} onError={() => setImageError(true)} /> )} {showInitials && ( <span className={styles.initials} aria-hidden="true"> {initials} </span> )} {!showImage && !showInitials && ( <span className={styles.placeholder} aria-hidden="true"> ? </span> )} </div> ); };

Avatar.displayName = 'Avatar';

Badge Atom Example

// atoms/Badge/Badge.tsx import React from 'react'; import styles from './Badge.module.css';

export type BadgeVariant = | 'default' | 'primary' | 'success' | 'warning' | 'danger' | 'info';

export type BadgeSize = 'sm' | 'md';

export interface BadgeProps { /** Visual variant / variant?: BadgeVariant; /* Size variant / size?: BadgeSize; /* Badge content / children: React.ReactNode; /* Additional class name */ className?: string; }

export const Badge: React.FC<BadgeProps> = ({ variant = 'default', size = 'md', children, className, }) => { const classNames = [styles.badge, styles[variant], styles[size], className] .filter(Boolean) .join(' ');

return <span className={classNames}>{children}</span>; };

Badge.displayName = 'Badge';

Checkbox Atom Example

// atoms/Checkbox/Checkbox.tsx import React from 'react'; import type { InputHTMLAttributes } from 'react'; import styles from './Checkbox.module.css';

export interface CheckboxProps extends Omit<InputHTMLAttributes<HTMLInputElement>, 'type'> { /** Indeterminate state / indeterminate?: boolean; /* Label text */ label?: string; }

export const Checkbox = React.forwardRef<HTMLInputElement, CheckboxProps>( ({ indeterminate = false, label, disabled, className, ...props }, ref) => { const inputRef = React.useRef<HTMLInputElement>(null);

React.useImperativeHandle(ref, () => inputRef.current!);

React.useEffect(() => {
  if (inputRef.current) {
    inputRef.current.indeterminate = indeterminate;
  }
}, [indeterminate]);

const wrapperClasses = [
  styles.wrapper,
  disabled &#x26;&#x26; styles.disabled,
  className,
]
  .filter(Boolean)
  .join(' ');

const checkbox = (
  &#x3C;span className={styles.checkbox}>
    &#x3C;input
      ref={inputRef}
      type="checkbox"
      className={styles.input}
      disabled={disabled}
      {...props}
    />
    &#x3C;span className={styles.control} aria-hidden="true">
      &#x3C;svg className={styles.check} viewBox="0 0 12 10">
        &#x3C;polyline points="1.5 6 4.5 9 10.5 1" />
      &#x3C;/svg>
      &#x3C;svg className={styles.indeterminate} viewBox="0 0 12 2">
        &#x3C;line x1="1" y1="1" x2="11" y2="1" />
      &#x3C;/svg>
    &#x3C;/span>
  &#x3C;/span>
);

if (label) {
  return (
    &#x3C;label className={wrapperClasses}>
      {checkbox}
      &#x3C;span className={styles.label}>{label}&#x3C;/span>
    &#x3C;/label>
  );
}

return checkbox;

} );

Checkbox.displayName = 'Checkbox';

Typography Atoms

// atoms/Typography/Text.tsx import React from 'react'; import styles from './Typography.module.css';

export type TextSize = 'xs' | 'sm' | 'md' | 'lg' | 'xl'; export type TextWeight = 'normal' | 'medium' | 'semibold' | 'bold'; export type TextColor = 'default' | 'muted' | 'primary' | 'success' | 'danger';

export interface TextProps { as?: 'p' | 'span' | 'div'; size?: TextSize; weight?: TextWeight; color?: TextColor; truncate?: boolean; children: React.ReactNode; className?: string; }

export const Text: React.FC<TextProps> = ({ as: Component = 'p', size = 'md', weight = 'normal', color = 'default', truncate = false, children, className, }) => { const classNames = [ styles.text, styles[size-${size}], styles[weight-${weight}], styles[color-${color}], truncate && styles.truncate, className, ] .filter(Boolean) .join(' ');

return <Component className={classNames}>{children}</Component>; };

// atoms/Typography/Heading.tsx export type HeadingLevel = 1 | 2 | 3 | 4 | 5 | 6;

export interface HeadingProps { level: HeadingLevel; as?: h${HeadingLevel}; children: React.ReactNode; className?: string; }

export const Heading: React.FC<HeadingProps> = ({ level, as, children, className, }) => { const Component = as || (h${level} as const); const classNames = [styles.heading, styles[h${level}], className] .filter(Boolean) .join(' ');

return <Component className={classNames}>{children}</Component>; };

Best Practices

  1. Use forwardRef for DOM Access

// GOOD: Allows parent to access DOM node export const Input = React.forwardRef<HTMLInputElement, InputProps>( (props, ref) => <input ref={ref} {...props} /> );

// BAD: No way for parent to access DOM export const Input = (props: InputProps) => <input {...props} />;

  1. Extend Native HTML Attributes

// GOOD: Supports all native button attributes interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> { variant?: 'primary' | 'secondary'; }

// BAD: Missing native attributes interface ButtonProps { onClick?: () => void; disabled?: boolean; }

  1. Provide Sensible Defaults

// GOOD: Works out of the box export const Button = ({ variant = 'primary', size = 'md', type = 'button', // Prevent accidental form submissions ...props }) => { ... };

// BAD: Requires explicit props export const Button = ({ variant, size, ...props }) => { ... };

  1. Keep Atoms Presentation-Only

// GOOD: No business logic const Button = ({ onClick, children }) => ( <button onClick={onClick}>{children}</button> );

// BAD: Atom has API call const SubmitButton = () => { const handleClick = async () => { await api.submit(); // Business logic in atom! }; return <button onClick={handleClick}>Submit</button>; };

Anti-Patterns to Avoid

  1. Atoms with Internal State

// BAD: Atom manages its own state const Input = () => { const [value, setValue] = useState(''); return <input value={value} onChange={(e) => setValue(e.target.value)} />; };

// GOOD: Controlled by parent const Input = ({ value, onChange }) => ( <input value={value} onChange={onChange} /> );

  1. Atoms with Complex Logic

// BAD: Complex validation in atom const EmailInput = ({ value, onChange }) => { const isValid = /^[^\s@]+@[^\s@]+.[^\s@]+$/.test(value); return <input value={value} className={isValid ? '' : 'error'} />; };

// GOOD: Validation handled by parent/molecule const Input = ({ value, onChange, hasError }) => ( <input value={value} className={hasError ? 'error' : ''} /> );

  1. Hardcoded Styles

// BAD: Hardcoded colors const Button = () => ( <button style={{ backgroundColor: '#2196f3' }}>Click</button> );

// GOOD: Uses design tokens const Button = () => ( <button style={{ backgroundColor: 'var(--color-primary-500)' }}> Click </button> );

When to Use This Skill

  • Creating new basic UI components

  • Refactoring existing components to atoms

  • Building a design system foundation

  • Ensuring consistency across components

  • Improving component reusability

Related Skills

  • atomic-design-fundamentals

  • Core methodology overview

  • atomic-design-molecules

  • Composing atoms into molecules

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

android-jetpack-compose

No summary provided by upstream source.

Repository SourceNeeds Review
General

fastapi-async-patterns

No summary provided by upstream source.

Repository SourceNeeds Review
General

storybook-story-writing

No summary provided by upstream source.

Repository SourceNeeds Review
General

atomic-design-fundamentals

No summary provided by upstream source.

Repository SourceNeeds Review