ACF UI
Components

Camera

A camera component for capturing photos with front/back camera toggle, mirror modes, and mask overlays.

Camera not active

Installation

CLI

npx shadcn@latest add "https://acfui.com/r/camera"
pnpm dlx shadcn@latest add "https://acfui.com/r/camera"
yarn dlx shadcn@latest add "https://acfui.com/r/camera"
bun x shadcn@latest add "https://acfui.com/r/camera"

Manual

Install the following dependencies:

npm install class-variance-authority lucide-react
pnpm add class-variance-authority lucide-react
yarn add class-variance-authority lucide-react
bun add class-variance-authority lucide-react

Copy and paste the following code into your project.

Usage

import { Camera } from "@/components/ui/camera"
<Camera 
  onCapture={(imageData) => console.log('Photo captured:', imageData)}
  onError={(error) => console.error('Camera error:', error)}
/>

Examples

Output Format Adapters

The Camera component supports different output formats to suit various use cases.

Base64 String Output (Default)

Perfect for immediate display, database storage, or simple image processing.

<Camera
  outputFormat="base64"
  captureFormat="image/jpeg"
  captureQuality={0.8}
  onCapture={(imageData) => {
    // imageData is a base64 string
    console.log('Base64 data:', imageData)
    // Example: "..."
    
    // Direct usage in img src
    setProfileImage(imageData)
  }}
/>

File Object Output

Ideal for file uploads, FormData submissions, or when you need file metadata.

<Camera
  outputFormat="file"
  captureFormat="image/png"
  captureQuality={0.9}
  onCapture={(imageData) => {
    if (imageData instanceof File) {
      console.log('File object:', imageData)
      // Example file properties:
      // imageData.name: "photo-1703847234567.png"
      // imageData.size: 245760 (bytes)
      // imageData.type: "image/png"
      // imageData.lastModified: 1703847234567
      
      // Direct upload to server
      uploadFile(imageData)
    }
  }}
/>

Complete Example with Both Formats

import { Camera } from "@/components/ui/camera"
import { useState } from "react"

export function CameraExample() {
  const [capturedData, setCapturedData] = useState<{
    type: "base64" | "file"
    data: string | File
    preview?: string
  } | null>(null)

  const onCaptureBase64 = (imageData: string | File) => {
    if (typeof imageData === "string") {
      setCapturedData({
        type: "base64",
        data: imageData,
        preview: imageData
      })
    }
  }

  const onCaptureFile = (imageData: string | File) => {
    if (imageData instanceof File) {
      // Convert File to base64 for preview
      const reader = new FileReader()
      reader.onload = (e) => {
        setCapturedData({
          type: "file",
          data: imageData,
          preview: e.target?.result as string
        })
      }
      reader.readAsDataURL(imageData)
    }
  }

  const uploadFile = async (file: File) => {
    const formData = new FormData()
    formData.append('image', file)

    try {
      const response = await fetch('/api/upload', {
        method: 'POST',
        body: formData
      })
      
      if (response.ok) {
        console.log('File uploaded successfully')
      }
    } catch (error) {
      console.error('Upload failed:', error)
    }
  }

  return (
    <div className="grid grid-cols-1 md:grid-cols-2 gap-6">
      {/* Base64 Output */}
      <div>
        <h3>Base64 Output</h3>
        <Camera
          size="sm"
          outputFormat="base64"
          onCapture={onCaptureBase64}
        />
      </div>

      {/* File Output */}
      <div>
        <h3>File Output</h3>
        <Camera
          size="sm"
          outputFormat="file"
          onCapture={onCaptureFile}
        />
      </div>

      {/* Display Results */}
      {capturedData && (
        <div className="col-span-full">
          <h4>Captured {capturedData.type.toUpperCase()}</h4>
          {capturedData.preview && (
            <img src={capturedData.preview} alt="Captured" />
          )}
          <pre>{JSON.stringify({
            type: capturedData.type,
            ...(capturedData.data instanceof File ? {
              name: capturedData.data.name,
              size: capturedData.data.size,
              type: capturedData.data.type
            } : {
              length: capturedData.data.length
            })
          }, null, 2)}</pre>
        </div>
      )}
    </div>
  )
}

Privacy & Camera Permissions

Enhanced camera permission handling with custom modal and cross-browser compatibility.

Camera: Checking...
Privacy & Camera Access
We respect your privacy. Camera access is used only for photo capture and is not recorded or stored.
Local ProcessingNo RecordingPrivacy First

Permission Flow

The camera component includes an intelligent permission flow:

  1. Permission Check: Automatically detects if camera access is already granted
  2. Custom Modal: Shows user-friendly explanation before browser prompt (only when needed)
  3. Native Prompt: Triggers browser's permission dialog after user confirms
  4. Direct Access: If already granted, skips modal and starts camera immediately

Cross-Browser Support

  • Chrome/Edge: Uses Permissions API for reliable permission detection
  • Safari: Falls back to getUserMedia test for compatibility
  • Firefox: Full Permissions API support with enhanced error handling
  • Mobile Browsers: Optimized for touch interfaces and mobile permissions

Enhanced UX Features

  • Smart Modal: Only shows when permission needed (not for returning users)
  • Clear Instructions: Step-by-step guidance for granting permissions
  • Recovery Flow: Easy retry mechanism for denied permissions
  • Privacy Badges: Visual indicators of data protection policies
  • Permission Status: Real-time display of current access level

Different Sizes

Choose from predefined size variants to fit your layout needs.

Default Size

Default Size

Standard camera size (320x240px) - perfect for most use cases

Camera not active

Small Size

Small Size

Compact camera size (256x192px) - ideal for thumbnails and tight spaces

Camera not active

Large Size

Large Size

Large camera size (384x288px) - great for detailed photo capture

Camera not active

Extra Large Size

Extra Large Size

Extra large camera size (448x320px) - perfect for high-quality captures

Camera not active

Aspect Ratios

Choose from different aspect ratios to fit your design requirements.

4:3 Traditional

4:3 Aspect Ratio

Traditional camera aspect ratio - ideal for standard photography

Camera not active

16:9 Widescreen

16:9 Aspect Ratio

Widescreen format - perfect for cinematic and landscape shots

Camera not active

1:1 Square

1:1 Aspect Ratio

Square format - ideal for profile photos and social media

Camera not active

Mirror Modes

Mirror mode allows you to flip the camera preview and/or the captured image.

No Mirror

No Mirror

Default behavior - no mirroring in preview or capture

Camera not active

Mirror All

Mirror All

Mirror both preview and captured image - useful for selfies

Camera not active

Mirror Preview Only

Mirror Preview Only

Mirror preview but capture normal image - best user experience

Camera not active

Mask Overlays

Add overlay masks to guide photo composition for specific use cases.

Square Mask

Square Mask

Perfect for profile photos and square avatars

Camera not active

Round Mask

Round Mask

Ideal for circular profile pictures and user avatars

Camera not active

Card Mask

Card Mask

Great for ID cards, licenses, and document photography

Camera not active

Manual Control

Camera not active

Custom Capture Handler

Camera not active

Privacy & Security

Enhanced Camera Permission Flow

The camera component features an intelligent, privacy-first permission system:

Smart Permission Detection

// Automatic permission checking on component mount
useEffect(() => {
  checkCameraPermission() // Multi-method browser detection
}, [])

Custom Modal Before Native Prompt

  • When needed: Shows custom explanation modal before browser prompt
  • When not needed: Direct camera access for returning users with permissions
  • Cross-browser: Works consistently across Chrome, Safari, Firefox, Edge

Permission States Handled

  • checking: Detecting current permission status
  • prompt: Shows custom modal → triggers native browser prompt
  • granted: Direct camera access, no modal needed
  • denied: Clear recovery instructions with retry mechanism

Camera Access Permission Flow

The camera component includes comprehensive permission handling with privacy-first approach:

  • Clear Permission Requests: Custom modal explains camera usage before native prompt
  • Privacy Information: Transparent communication about data handling
  • Error Recovery: Graceful handling of permission denied states with retry options
  • Local Processing: All camera processing happens client-side only

Privacy Features

  • 🔒 Local Processing: No data sent to servers
  • 🚫 No Recording: Only captures individual photos
  • 👁️ Transparent Usage: Clear privacy information displayed
  • Instant Revocation: Users can disable camera access anytime
  • 🛡️ Secure by Default: HTTPS required in production
  • 🎯 Smart UX: Modal only shows when permission actually needed
// Enhanced privacy-focused camera usage
<Camera
  onStart={() => console.log('Camera started with user consent')}
  onError={(error) => {
    // Handle permission denied gracefully with recovery options
    if (error.includes('Permission denied')) {
      showPermissionRecoveryFlow()
    }
  }}
  autoStart={false} // Requires explicit user action
/>

API Reference

Camera

PropTypeDefaultDescription
size"sm" | "md" | "lg" | "xl" | "full""md"Size variant of the camera component
aspect"4:3" | "16:9" | "1:1""4:3"Aspect ratio of the camera
onCapture(imageData: string | File) => void-Callback when a photo is captured
onError(error: string) => void-Callback when an error occurs
onStart() => void-Callback when camera starts
autoStartbooleantrueWhether to automatically start the camera
showControlsbooleantrueWhether to show camera controls
allowDownloadbooleantrueWhether to allow downloading captured photos
silentbooleanfalseWhether to disable shutter sound
mirrorboolean | "preview"falseMirror mode: true (all), "preview" (preview only), false (none)
mask"none" | "square" | "round" | "card""none"Overlay mask for guided composition
outputFormat"base64" | "file""base64"Output format: base64 string or File object
captureFormat"image/jpeg" | "image/png" | "image/webp""image/jpeg"Image format for capture
captureQualitynumber0.9Image quality (0.0 to 1.0)

Output Format Comparison

FormatTypeUse CasesProsCons
base64stringDisplay, database storage, simple processingImmediate usage, no conversion neededLarge string size, not ideal for uploads
fileFileFile uploads, FormData, server submissionsProper file handling, metadata includedRequires conversion for display

Example Values

Base64 Output

// Example base64 string (truncated)
""

// Length: typically 50,000-200,000 characters

File Output

// Example File object properties
{
  name: "photo-1703847234567.png",
  size: 245760,
  type: "image/png", 
  lastModified: 1703847234567,
  // Methods: stream(), text(), arrayBuffer(), etc.
}

Size Variants

SizeDimensionsUse Case
sm256×192pxThumbnails, compact spaces
md320×240pxDefault, most use cases
lg384×288pxDetailed captures
xl448×320pxHigh-quality photos
full100% containerFull-screen camera

Browser Support

  • Chrome: ✅ Full support
  • Firefox: ✅ Full support
  • Safari: ✅ Full support (iOS 11+)
  • Edge: ✅ Full support

Note: Camera access requires HTTPS in production environments and user permission.

Accessibility

  • Proper ARIA labels for camera controls
  • Keyboard navigation support
  • Screen reader announcements for state changes
  • High contrast mode compatibility

Keyboard Interactions

KeyDescription
SpaceEnterWhen focus is on the capture button, takes a photo.
TabMoves focus between camera controls (capture, toggle, sound, stop).
EscapeStops the camera when active.

Examples in Production

Profile Photo Capture

<Camera
  mask="square"
  aspect="1:1"
  mirror="preview"
  onCapture={(imageData) => {
    // Upload to your backend
    uploadProfilePhoto(imageData)
  }}
/>

ID Card Scanning

<Camera
  mask="card"
  aspect="16:9"
  captureQuality={1.0}
  onCapture={(imageData) => {
    // Process ID card
    processIDCard(imageData)
  }}
/>

Avatar Creation

<Camera
  mask="round"
  aspect="1:1"
  size="sm"
  onCapture={(imageData) => {
    // Create avatar
    createAvatar(imageData)
  }}
/>