Back to Posts

Flutter Camera Flash: How to Control the Flashlight

9 min read
<div style="text-align: center;"> <img src="" alt="Flutter Camera Flash Example" width="300" /> </div>

This comprehensive guide will walk you through implementing camera flash control in Flutter applications. Learn how to handle permissions, control the flashlight, and create a user-friendly interface for flash management.

Basic Flash Control

1. Using the camera Package

import 'package:camera/camera.dart';

class FlashControl extends StatefulWidget {
  @override
  _FlashControlState createState() => _FlashControlState();
}

class _FlashControlState extends State<FlashControl> {
  CameraController? _controller;
  FlashMode _flashMode = FlashMode.off;

  @override
  void initState() {
    super.initState();
    _initializeCamera();
  }

  Future<void> _initializeCamera() async {
    final cameras = await availableCameras();
    _controller = CameraController(
      cameras[0],
      ResolutionPreset.medium,
    );
    await _controller!.initialize();
    setState(() {});
  }

  Future<void> _toggleFlash() async {
    if (_controller == null) return;

    setState(() {
      _flashMode = _flashMode == FlashMode.off ? FlashMode.torch : FlashMode.off;
    });

    await _controller!.setFlashMode(_flashMode);
  }

  @override
  Widget build(BuildContext context) {
    if (_controller == null || !_controller!.value.isInitialized) {
      return Center(child: CircularProgressIndicator());
    }

    return Scaffold(
      body: CameraPreview(_controller!),
      floatingActionButton: FloatingActionButton(
        onPressed: _toggleFlash,
        child: Icon(_flashMode == FlashMode.off ? Icons.flash_off : Icons.flash_on),
      ),
    );
  }
}

2. Using the torch_compatibility Package

import 'package:torch_compatibility/torch_compatibility.dart';

class TorchControl extends StatefulWidget {
  @override
  _TorchControlState createState() => _TorchControlState();
}

class _TorchControlState extends State<TorchControl> {
  bool _isTorchOn = false;

  Future<void> _toggleTorch() async {
    try {
      if (_isTorchOn) {
        await TorchCompatibility.turnOff();
      } else {
        await TorchCompatibility.turnOn();
      }
      setState(() {
        _isTorchOn = !_isTorchOn;
      });
    } catch (e) {
      print('Error toggling torch: $e');
    }
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Icon(
              _isTorchOn ? Icons.flash_on : Icons.flash_off,
              size: 100,
              color: _isTorchOn ? Colors.yellow : Colors.grey,
            ),
            SizedBox(height: 20),
            ElevatedButton(
              onPressed: _toggleTorch,
              child: Text(_isTorchOn ? 'Turn Off Flash' : 'Turn On Flash'),
            ),
          ],
        ),
      ),
    );
  }
}

Permission Handling

1. Android Permissions

Add to android/app/src/main/AndroidManifest.xml:

<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.FLASHLIGHT" />

2. iOS Permissions

Add to ios/Runner/Info.plist:

<key>NSCameraUsageDescription</key>
<string>This app needs camera access to control the flash</string>

3. Permission Request Implementation

import 'package:permission_handler/permission_handler.dart';

class PermissionHandler {
  static Future<bool> requestCameraPermission() async {
    final status = await Permission.camera.request();
    return status.isGranted;
  }

  static Future<bool> checkCameraPermission() async {
    final status = await Permission.camera.status;
    return status.isGranted;
  }
}

Advanced Flash Features

1. Flash Modes

enum FlashMode {
  off,
  auto,
  always,
  torch,
}

class FlashModeControl extends StatefulWidget {
  @override
  _FlashModeControlState createState() => _FlashModeControlState();
}

class _FlashModeControlState extends State<FlashModeControl> {
  FlashMode _currentMode = FlashMode.off;
  CameraController? _controller;

  Future<void> _setFlashMode(FlashMode mode) async {
    if (_controller == null) return;

    setState(() {
      _currentMode = mode;
    });

    switch (mode) {
      case FlashMode.off:
        await _controller!.setFlashMode(FlashMode.off);
        break;
      case FlashMode.auto:
        await _controller!.setFlashMode(FlashMode.auto);
        break;
      case FlashMode.always:
        await _controller!.setFlashMode(FlashMode.always);
        break;
      case FlashMode.torch:
        await _controller!.setFlashMode(FlashMode.torch);
        break;
    }
  }

  @override
  Widget build(BuildContext context) {
    return Row(
      mainAxisAlignment: MainAxisAlignment.spaceEvenly,
      children: [
        _buildFlashButton(FlashMode.off, Icons.flash_off),
        _buildFlashButton(FlashMode.auto, Icons.flash_auto),
        _buildFlashButton(FlashMode.always, Icons.flash_on),
        _buildFlashButton(FlashMode.torch, Icons.highlight),
      ],
    );
  }

  Widget _buildFlashButton(FlashMode mode, IconData icon) {
    return IconButton(
      icon: Icon(icon),
      color: _currentMode == mode ? Colors.blue : Colors.grey,
      onPressed: () => _setFlashMode(mode),
    );
  }
}

2. Flash Animation

class FlashAnimation extends StatefulWidget {
  @override
  _FlashAnimationState createState() => _FlashAnimationState();
}

class _FlashAnimationState extends State<FlashAnimation> with SingleTickerProviderStateMixin {
  late AnimationController _controller;
  late Animation<double> _animation;
  bool _isFlashing = false;

  @override
  void initState() {
    super.initState();
    _controller = AnimationController(
      duration: Duration(milliseconds: 500),
      vsync: this,
    );
    _animation = Tween<double>(begin: 0, end: 1).animate(_controller);
  }

  Future<void> _triggerFlash() async {
    if (_isFlashing) return;

    setState(() {
      _isFlashing = true;
    });

    await _controller.forward();
    await Future.delayed(Duration(milliseconds: 100));
    await _controller.reverse();

    setState(() {
      _isFlashing = false;
    });
  }

  @override
  Widget build(BuildContext context) {
    return AnimatedBuilder(
      animation: _animation,
      builder: (context, child) {
        return Container(
          color: Colors.white.withOpacity(_animation.value),
          child: child,
        );
      },
      child: Center(
        child: ElevatedButton(
          onPressed: _triggerFlash,
          child: Text('Trigger Flash'),
        ),
      ),
    );
  }
}

Best Practices

1. Error Handling

  • Check device compatibility
  • Handle permission denials gracefully
  • Implement proper error messages
  • Provide fallback options
  • Log errors for debugging

2. User Experience

  • Provide clear feedback
  • Use appropriate icons
  • Implement smooth transitions
  • Consider accessibility
  • Test on different devices

3. Performance

  • Initialize camera only when needed
  • Release resources properly
  • Handle lifecycle events
  • Optimize battery usage
  • Consider background operation

Common Issues and Solutions

1. Flash Not Working

class FlashTroubleshooter {
  static Future<bool> checkFlashAvailability() async {
    try {
      return await TorchCompatibility.hasTorch();
    } catch (e) {
      print('Error checking flash availability: $e');
      return false;
    }
  }

  static Future<void> troubleshootFlash() async {
    if (!await checkFlashAvailability()) {
      print('Flash not available on this device');
      return;
    }

    try {
      await TorchCompatibility.turnOn();
      await Future.delayed(Duration(seconds: 1));
      await TorchCompatibility.turnOff();
    } catch (e) {
      print('Error testing flash: $e');
    }
  }
}

2. Permission Handling

class PermissionManager {
  static Future<void> handlePermissionDenial(BuildContext context) async {
    final shouldShowRationale = await Permission.camera.shouldShowRequestRationale;
    
    if (shouldShowRationale) {
      showDialog(
        context: context,
        builder: (context) => AlertDialog(
          title: Text('Camera Permission Required'),
          content: Text('This app needs camera access to control the flash.'),
          actions: [
            TextButton(
              onPressed: () => Navigator.pop(context),
              child: Text('Cancel'),
            ),
            TextButton(
              onPressed: () async {
                Navigator.pop(context);
                await openAppSettings();
              },
              child: Text('Open Settings'),
            ),
          ],
        ),
      );
    }
  }
}

Conclusion

Implementing camera flash control in Flutter requires attention to detail and proper handling of permissions and device capabilities. Remember to:

  • Handle permissions properly
  • Check device compatibility
  • Implement error handling
  • Consider user experience
  • Test thoroughly

Happy coding!