| Crates.io | crabcamera |
| lib.rs | crabcamera |
| version | 0.1.0 |
| created_at | 2025-09-07 06:14:59.827827+00 |
| updated_at | 2025-09-07 06:14:59.827827+00 |
| description | Advanced cross-platform camera integration for Tauri applications |
| homepage | |
| repository | https://github.com/Michael-A-Kuykendall/crabcamera |
| max_upload_size | |
| id | 1827861 |
| size | 296,567 |
CrabCamera will be free forever. No asterisks. No "free for now." No pivot to paid.
CrabCamera is the first production-ready desktop camera plugin for Tauri applications. It provides unified camera access across Windows, macOS, and Linux with professional controls and zero-config setup. It's designed to be the invisible infrastructure that makes desktop camera apps just work.
| Feature | CrabCamera | Web APIs | Other Plugins |
|---|---|---|---|
| Desktop Native | Windows/macOS/Linux 🏆 | Limited browser | Mobile-only |
| Hardware Access | Direct camera control 🏆 | Browser restricted | Basic access |
| Professional Controls | Auto-focus, exposure 🏆 | Limited | Basic |
| Cross-Platform | Unified API 🏆 | Platform dependent | Single platform |
| Production Ready | 63 comprehensive tests 🏆 | No guarantees | Proof-of-concept |
| Memory Safety | Zero unsafe code 🏆 | N/A | Manual management |
BONUS: Professional camera controls with platform-optimized settings for maximum image quality.
[dependencies]
crabcamera = "0.1"
tauri = { version = "2.0", features = ["protocol-asset"] }
// src-tauri/src/main.rs
use crabcamera;
fn main() {
tauri::Builder::default()
.plugin(crabcamera::init())
.run(tauri::generate_context!())
.expect("error while running tauri application");
}
// tauri.conf.json
{
"plugins": {
"crabcamera": {}
}
}
import { invoke } from '@tauri-apps/api/tauri';
// Initialize camera system
await invoke('initialize_camera_system');
// Get available cameras
const cameras = await invoke('get_available_cameras');
console.log('Available cameras:', cameras);
// Get recommended format for high quality
const format = await invoke('get_recommended_format');
// Capture a photo
const photo = await invoke('capture_single_photo', {
deviceId: cameras[0].id,
format: format
});
// Initialize the camera system
initialize_camera_system() -> Result<String>
// Get all available cameras with capabilities
get_available_cameras() -> Result<Vec<CameraDeviceInfo>>
// Get platform-specific information
get_platform_info() -> Result<PlatformInfo>
// Test camera system functionality
test_camera_system() -> Result<SystemTestResult>
// Check if specific camera is available
check_camera_availability(device_id: String) -> Result<bool>
// Get supported formats for a camera
get_camera_formats(device_id: String) -> Result<Vec<CameraFormat>>
// Get recommended settings for quality photography
get_recommended_format() -> Result<CameraFormat>
get_optimal_settings() -> Result<CameraInitParams>
// Single photo capture
capture_single_photo(device_id: String, format: CameraFormat) -> Result<CameraFrame>
// Photo sequence for burst mode
capture_photo_sequence(params: SequenceParams) -> Result<Vec<CameraFrame>>
// Real-time streaming
start_camera_preview(device_id: String) -> Result<()>
stop_camera_preview() -> Result<()>
// Save frames to disk
save_frame_to_disk(frame: CameraFrame, path: String) -> Result<()>
// Handle camera permissions properly
request_camera_permission() -> Result<bool>
check_camera_permission_status() -> Result<PermissionStatus>
I built CrabCamera because desktop applications deserve native camera access without the limitations of web APIs or mobile-only plugins.
This is my commitment: CrabCamera stays MIT licensed, forever. If you want to support development, sponsor it. If you don't, just build something incredible with it.
CrabCamera saves developers weeks of cross-platform camera integration. If it's useful, consider sponsoring for $5/month — less than a coffee, infinitely more valuable than web API limitations.
| Metric | CrabCamera | Web APIs | Mobile Plugins |
|---|---|---|---|
| Desktop Support | Full native | Browser dependent | None |
| Camera Access | Direct hardware | getUserMedia limited | N/A |
| Image Quality | Professional controls | Basic settings | Basic |
| Cross-Platform | Windows/macOS/Linux | Browser variation | iOS/Android only |
| Performance | Native speed | Browser overhead | N/A |
| Reliability | 63 tests passing | No guarantees | Varies |
pub struct CameraDeviceInfo {
pub id: String,
pub name: String,
pub description: String,
pub is_available: bool,
pub supports_formats: Vec<CameraFormat>,
}
pub struct CameraFormat {
pub width: u32,
pub height: u32,
pub fps: f32,
pub format_type: String, // "RGB8", "JPEG", etc.
}
pub struct CameraFrame {
pub data: Vec<u8>,
pub width: u32,
pub height: u32,
pub format: String,
pub timestamp: DateTime<Utc>,
}
pub enum Platform {
Windows,
MacOS,
Linux,
Unknown,
}
// Automatic platform detection
let platform = Platform::current();
See our amazing sponsors who make CrabCamera possible! 🙏
Sponsorship Tiers:
Companies: Need invoicing? Email michaelallenkuykendall@gmail.com
✅ Ready for production:
✅ Use cases in production:
// Simple photo booth with camera selection
const cameras = await invoke('get_available_cameras');
const selectedCamera = cameras[0];
const format = await invoke('get_recommended_format');
// Take photo when user clicks
document.getElementById('capture').onclick = async () => {
const photo = await invoke('capture_single_photo', {
deviceId: selectedCamera.id,
format: format
});
// Display photo in UI
displayPhoto(photo);
};
// Monitor multiple cameras
const cameras = await invoke('get_available_cameras');
for (const camera of cameras) {
await invoke('start_camera_preview', { deviceId: camera.id });
// Set up streaming handlers for each camera
setupCameraStream(camera);
}
MIT License - forever and always.
Philosophy: Desktop applications deserve native camera access. CrabCamera is camera infrastructure.
Forever maintainer: Michael A. Kuykendall
Promise: This will never become a paid product
Mission: Making desktop camera development effortless
"Native performance. Cross-platform compatibility. Zero hassle."