aidlab_sdk 1.7.4 copy "aidlab_sdk: ^1.7.4" to clipboard
aidlab_sdk: ^1.7.4 copied to clipboard

Aidlab Flutter SDK. For more information please visit https://www.aidlab.com/developer

example/lib/main.dart

import 'package:aidlab_sdk/aidlab_sdk.dart';
import 'package:flutter/material.dart';
import 'package:permission_handler/permission_handler.dart';
import 'line_chart.dart';
import 'dart:io';

AidlabManager? aidlabManager;

// In-memory connection state and device names (autopair removed)
class AutoPairService {
  static final Map<String, String> _deviceNames = {};
  static final Set<String> _connectedDevices = {};
  static final Set<String> _connectingDevices = {};

  static String getDeviceName(String address) {
    return _deviceNames[address] ?? address;
  }

  static void setDeviceName(String address, String name) {
    _deviceNames[address] = name;
  }

  static bool isConnected(String address) {
    return _connectedDevices.contains(address);
  }

  static bool isConnecting(String address) {
    return _connectingDevices.contains(address);
  }

  static void setConnected(String address, bool connected) {
    if (connected) {
      _connectedDevices.add(address);
      _connectingDevices.remove(address);
    } else {
      _connectedDevices.remove(address);
      _connectingDevices.remove(address);
    }
  }

  static void setConnecting(String address, bool connecting) {
    if (connecting) {
      _connectingDevices.add(address);
    } else {
      _connectingDevices.remove(address);
    }
  }

  static bool canConnect(String address) {
    return !isConnected(address) && !isConnecting(address);
  }
}

void main() {
  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Aidlab Flutter SDK Example',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: const DeviceListScreen(),
    );
  }
}

class DeviceListScreen extends StatefulWidget {
  const DeviceListScreen({super.key});

  @override
  State<DeviceListScreen> createState() => _DeviceListScreenState();
}

class _DeviceListScreenState extends State<DeviceListScreen>
    implements AidlabManagerDelegate {
  final List<Device> _devices = [];
  final Map<String, Device> _discoveredDevices = {};
  bool _isScanning = false;
  bool _permissionsReady = false;

  @override
  void initState() {
    super.initState();
    _ensurePermissions();
    aidlabManager = AidlabManager(this);
  }

  /// Ask for runtime permissions required for BLE scanning.
  /// Android 12+ requires bluetoothScan and bluetoothConnect; Android 6–11 requires Location and enabled location services.
  Future<void> _ensurePermissions() async {
    // iOS: request Bluetooth permission only. Location is not required for CoreBluetooth scanning.
    if (Platform.isIOS) {
      // Request the iOS Bluetooth permission. This triggers the native prompt.
      final bluetoothStatus = await Permission.bluetooth.request();
      setState(() {
        _permissionsReady = bluetoothStatus.isGranted;
      });
      return;
    }

    // Android: request runtime permissions depending on OS version.
    final scanStatus = await Permission.bluetoothScan.request();
    final connectStatus = await Permission.bluetoothConnect.request();
    final locationStatus = await Permission.locationWhenInUse.request();

    final serviceEnabled = await Permission.location.serviceStatus.isEnabled;

    final hasBle12Plus = scanStatus.isGranted && connectStatus.isGranted;
    final hasBlePre12 = locationStatus.isGranted && serviceEnabled;

    if (!serviceEnabled && !hasBle12Plus) {
      // On Android pre-12, users must enable Location Services for BLE scanning.
      await openAppSettings();
    }

    setState(() {
      _permissionsReady = hasBle12Plus || hasBlePre12;
    });
  }

  @override
  void dispose() {
    aidlabManager?.dispose();
    aidlabManager = null;
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Text('Devices'),
      ),
      body: Column(
        children: [
          Expanded(
            child: ListView.builder(
              itemCount: _devices.length,
              itemBuilder: (context, index) {
                final device = _devices[index];
                final deviceName =
                    AutoPairService.getDeviceName(device.address);
                final isConnected = AutoPairService.isConnected(device.address);
                final isConnecting =
                    AutoPairService.isConnecting(device.address);

                return Card(
                  margin:
                      const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
                  color: isConnected
                      ? Colors.green.shade50
                      : isConnecting
                          ? Colors.orange.shade50
                          : null,
                  child: ListTile(
                    title: Text(deviceName),
                    subtitle: Column(
                      crossAxisAlignment: CrossAxisAlignment.start,
                      mainAxisSize: MainAxisSize.min,
                      children: [
                        Text(device.address),
                        if (isConnected)
                          const Text('Connected',
                              style: TextStyle(
                                  color: Colors.green,
                                  fontWeight: FontWeight.bold)),
                        if (isConnecting)
                          const Text('Connecting...',
                              style: TextStyle(
                                  color: Colors.orange,
                                  fontWeight: FontWeight.bold)),
                      ],
                    ),
                    trailing: const Row(
                      mainAxisSize: MainAxisSize.min,
                      children: [],
                    ),
                    onTap: () {
                      if (AutoPairService.canConnect(device.address)) {
                        _connectToDevice(device);
                      } else if (isConnected) {
                        // If already connected, still allow to go to detail screen
                        Navigator.push(
                          context,
                          MaterialPageRoute(
                            builder: (context) =>
                                DeviceDetailScreen(device: device),
                          ),
                        );
                      }
                    },
                  ),
                );
              },
            ),
          ),
          Padding(
            padding:
                const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
            child: _isScanning
                ? ElevatedButton(
                    onPressed: () => stopScan(),
                    style:
                        ElevatedButton.styleFrom(backgroundColor: Colors.red),
                    child: const Text('Stop scan'),
                  )
                : ElevatedButton(
                    onPressed: () async {
                      // iOS: allow starting scan even if permission isn't yet granted;
                      // the system prompt appears upon first CoreBluetooth use.
                      if (Platform.isIOS) {
                        if (!_permissionsReady) {
                          await Permission.bluetooth.request();
                        }
                        startScan();
                        return;
                      }

                      // Android: gate scanning on runtime permissions.
                      await _ensurePermissions();
                      if (!context.mounted) return;
                      if (_permissionsReady) {
                        startScan();
                      } else {
                        ScaffoldMessenger.of(context).showSnackBar(
                          const SnackBar(
                            content: Text(
                                'Grant Bluetooth permission when prompted'),
                          ),
                        );
                      }
                    },
                    child: const Text('Start scan'),
                  ),
          ),
        ],
      ),
    );
  }

  void startScan() {
    aidlabManager?.scan(ScanMode.lowPower);
    setState(() {
      _isScanning = true;
    });
  }

  void stopScan() {
    aidlabManager?.stopScan();
    setState(() {
      _isScanning = false;
      _devices.clear();
    });
  }

  @override
  void didDiscover(Device device, int rssi) {
    // Store discovered devices
    _discoveredDevices[device.address] = device;

    // Save device name if not already saved
    if (!AutoPairService._deviceNames.containsKey(device.address)) {
      AutoPairService.setDeviceName(
          device.address, device.name ?? device.address);
    }

    setState(() {
      if (!_devices.any((d) => d.address == device.address)) {
        _devices.add(device);
      } else {
        _devices.removeWhere((d) => d.address == device.address);
        _devices.add(device);
      }
    });

    // Autopair removed: manual connect only by tapping on device
  }

  @override
  void onBluetoothStarted() {
    ScaffoldMessenger.of(context)
        .showSnackBar(const SnackBar(content: Text("Bluetooth started")));
  }

  @override
  void onDeviceScanStarted() {}

  @override
  void onDeviceScanStopped() {}

  @override
  void onScanFailed(int errorCode) {
    /// Show toast
    ScaffoldMessenger.of(context)
        .showSnackBar(SnackBar(content: Text("Scan failed: $errorCode")));
  }

  // Autopair monitoring removed

  void _connectToDevice(Device device) {
    // Prevent multiple simultaneous connections to same device
    if (AutoPairService.isConnecting(device.address) ||
        AutoPairService.isConnected(device.address)) {
      return;
    }

    // Open detail screen which handles connection lifecycle
    if (_isScanning) {
      stopScan();
    }
    Navigator.push(
      context,
      MaterialPageRoute(
        builder: (context) => DeviceDetailScreen(device: device),
      ),
    ).then((_) {
      // Clean up states when returning from detail screen
      AutoPairService.setConnecting(device.address, false);
      AutoPairService.setConnected(device.address, false);
    });
  }
}

class DeviceDetailScreen extends StatefulWidget {
  final Device device;

  const DeviceDetailScreen({super.key, required this.device});

  @override
  State<DeviceDetailScreen> createState() => _DeviceDetailScreenState();
}

class _DeviceDetailScreenState extends State<DeviceDetailScreen>
    implements DeviceDelegate {
  /// Device information
  String _firmwareRevision = "Unknown";
  String _hardwareRevision = "Unknown";
  String _serialNumber = "Unknown";

  String _heartRate = "Unknown";
  String _skinTemperature = "Unknown";
  String _respirationRate = "Unknown";
  String _rr = "Unknown";
  String _activity = "Unknown";
  int _steps = 0;
  String _soundVolume = "Unknown";
  String _batteryLevel = "Unknown";
  String _signalQuality = "Unknown";
  String _bodyPosition = "Unknown";
  String _wearState = "Unknown";
  String _syncState = "Unknown";
  String _exercise = "Unknown";

  bool _isConnected = false;

  final List<double> _ecg = [];
  final List<double> _respiration = [];

  // Ensures UI leaves "Connecting..." as soon as we have proof of live data.
  void _ensureConnectedUi() {
    if (!_isConnected) {
      AutoPairService.setConnected(widget.device.address, true);
      AutoPairService.setConnecting(widget.device.address, false);
      setState(() {
        _isConnected = true;
      });
    }
  }

  /// Connect to device when the screen is created
  @override
  void initState() {
    super.initState();

    // Always try to connect - let the device handle if already connected
    AutoPairService.setConnecting(widget.device.address, true);

    try {
      widget.device.connect(this);
    } catch (e) {
      AutoPairService.setConnecting(widget.device.address, false);
      setState(() {
        _isConnected = false;
      });
    }
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(_isConnected ? 'Connected Device' : 'Connecting...'),
        backgroundColor: _isConnected ? Colors.green : Colors.orange,
      ),
      body: Column(
        children: [
          Expanded(
            child: SingleChildScrollView(
              child: Column(
                children: [
                  ListTile(
                    title: const Text('Device Address'),
                    trailing: Text(widget.device.address),
                  ),
                  ListTile(
                    title: const Text('Device Name'),
                    trailing: Text(widget.device.name ?? "Unknown"),
                  ),
                  ListTile(
                    title: const Text('Firmware revision'),
                    trailing: Text(_firmwareRevision),
                  ),
                  ListTile(
                    title: const Text('Hardware revision'),
                    trailing: Text(_hardwareRevision),
                  ),
                  ListTile(
                    title: const Text('Serial number'),
                    trailing: Text(_serialNumber),
                  ),
                  ListTile(
                    title: const Text('Battery Level'),
                    trailing: Text(_batteryLevel),
                  ),
                  ListTile(
                    title: const Text('Signal Quality'),
                    trailing: Text(_signalQuality),
                  ),
                  ListTile(
                    title: const Text('Heart Rate'),
                    trailing: Text(_heartRate),
                  ),
                  ListTile(
                    title: const Text('Skin Temperature'),
                    trailing: Text(_skinTemperature),
                  ),
                  ListTile(
                    title: const Text('Respiration Rate'),
                    trailing: Text(_respirationRate),
                  ),
                  ListTile(
                    title: const Text('RR'),
                    trailing: Text(_rr),
                  ),
                  ListTile(
                    title: const Text('Activity'),
                    trailing: Text(_activity),
                  ),
                  ListTile(
                    title: const Text('Steps'),
                    trailing: Text(_steps.toString()),
                  ),
                  ListTile(
                    title: const Text('Sound Volume'),
                    trailing: Text(_soundVolume),
                  ),
                  ListTile(
                    title: const Text('Body Position'),
                    trailing: Text(_bodyPosition),
                  ),
                  ListTile(
                    title: const Text('Wear State'),
                    trailing: Text(_wearState),
                  ),
                  ListTile(
                    title: const Text('Sync State'),
                    trailing: Text(_syncState),
                  ),
                  ListTile(
                    title: const Text('Exercise'),
                    trailing: Text(_exercise),
                  ),
                  Padding(
                    padding: const EdgeInsets.all(8.0),
                    child: SizedBox(
                        width: MediaQuery.of(context).size.width,
                        height: 100,
                        child: LineChart(_ecg)),
                  ),
                  Padding(
                    padding: const EdgeInsets.all(8.0),
                    child: SizedBox(
                        width: MediaQuery.of(context).size.width,
                        height: 100,
                        child: LineChart(_respiration)),
                  ),
                ],
              ),
            ),
          ),
          // Always visible disconnect button at the bottom
          Container(
            width: double.infinity,
            padding: const EdgeInsets.all(16),
            color: Colors.red,
            child: ElevatedButton(
              onPressed: disconnect,
              style: ElevatedButton.styleFrom(
                backgroundColor: Colors.white,
                foregroundColor: Colors.red,
                padding: const EdgeInsets.symmetric(vertical: 12),
              ),
              child: Text(
                _isConnected ? 'DISCONNECT' : 'DISCONNECT & GO BACK',
                style:
                    const TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
              ),
            ),
          ),
        ],
      ),
    );
  }

  @override
  void dispose() {
    super.dispose();
  }

  void disconnect() {
    widget.device.disconnect();

    // Immediately update local state
    AutoPairService.setConnected(widget.device.address, false);
    setState(() {
      _isConnected = false;
    });
  }

  @override
  void didConnect(Device device) {
    /// Update device information and connection state
    AutoPairService.setConnected(device.address, true);
    // Clear connecting state once we are connected to avoid stale UI status
    AutoPairService.setConnecting(device.address, false);

    setState(() {
      _isConnected = true;
      _firmwareRevision = device.firmwareRevision ?? "Unknown";
      _hardwareRevision = device.hardwareRevision ?? "Unknown";
      _serialNumber = device.serialNumber ?? "Unknown";
    });
    List<DataType> dataTypes = [
      DataType.ECG,
      DataType.RESPIRATION,
      DataType.SKIN_TEMPERATURE,
      DataType.ACTIVITY,
      DataType.STEPS,
      DataType.HEART_RATE,
      DataType.RR,
      // DataType.SOUND_VOLUME,
      DataType.RESPIRATION_RATE,
      DataType.BODY_POSITION,
      DataType.PRESSURE,
    ];

    device.collect(dataTypes, []);
  }

  @override
  void didDetectUserEvent(Device device, int timestamp) {}

  @override
  void didDisconnect(Device? device, DisconnectReason reason) {
    // Update connection state
    if (device != null) {
      AutoPairService.setConnected(device.address, false);
      // Ensure connecting flag is cleared on any disconnect
      AutoPairService.setConnecting(device.address, false);
    }

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

    // Show disconnect reason
    ScaffoldMessenger.of(context).showSnackBar(
      SnackBar(
          content: Text('Disconnected: ${reason.toString().split('.').last}')),
    );

    // Automatically go back to scanning screen
    Navigator.pop(context);
  }

  @override
  void didDetectExercise(Device device, Exercise exercise) {
    setState(() {
      _exercise = exercise.toString().split('.').last;
    });
  }

  @override
  void didReceiveAccelerometer(
      Device device, int timestamp, double ax, double ay, double az) {}

  @override
  void didReceiveActivity(Device device, int timestamp, ActivityType activity) {
    setState(() {
      _activity = activity.toString().split('.').last;
    });
  }

  @override
  void didReceiveBatteryLevel(Device device, int stateOfCharge) {
    setState(() {
      _batteryLevel = "$stateOfCharge%";
    });
  }

  @override
  void didReceiveBodyPosition(
      Device device, int timestamp, BodyPosition bodyPosition) {
    setState(() {
      _bodyPosition = bodyPosition.toString().split('.').last;
    });
  }

  @override
  void didReceiveCommand(Device device) {}

  @override
  void didReceiveECG(Device device, int timestamp, double value) {
    _ensureConnectedUi();
    setState(() {
      _ecg.add(value);
      while (_ecg.length > 500) {
        _ecg.removeAt(0);
      }
    });
  }

  @override
  void didReceiveError(String error) {
    // If connection error, clear connecting state
    if (error.toLowerCase().contains('connect') ||
        error.toLowerCase().contains('failed') ||
        error.toLowerCase().contains('timeout')) {
      AutoPairService.setConnecting(widget.device.address, false);
      setState(() {
        _isConnected = false;
      });
    }

    /// Show toast
    ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(error)));
  }

  @override
  void didReceiveGyroscope(
      Device device, int timestamp, double qx, double qy, double qz) {}

  @override
  void didReceiveHeartRate(Device device, int timestamp, int heartRate) {
    _ensureConnectedUi();
    setState(() {
      _heartRate = "$heartRate";
    });
  }

  @override
  void didReceiveMagnetometer(
      Device device, int timestamp, double mx, double my, double mz) {}

  @override
  void didReceiveMessage(Device device, String process, String message) {}

  @override
  void didReceiveOrientation(
      Device device, int timestamp, double roll, double pitch, double yaw) {}

  @override
  void didReceiveQuaternion(Device device, int timestamp, double qw, double qx,
      double qy, double qz) {}

  @override
  void didReceiveRespiration(Device device, int timestamp, double value) {
    _ensureConnectedUi();
    setState(() {
      _respiration.add(value);
      while (_respiration.length > 500) {
        _respiration.removeAt(0);
      }
    });
  }

  @override
  void didReceiveRespirationRate(Device device, int timestamp, int value) {
    _ensureConnectedUi();
    setState(() {
      _respirationRate = "$value";
    });
  }

  @override
  void didReceiveRr(Device device, int timestamp, int rr) {
    setState(() {
      _rr = "$rr";
    });
  }

  @override
  void didReceiveSignalQuality(Device device, int timestamp, int value) {
    _ensureConnectedUi();
    setState(() {
      _signalQuality = "$value";
    });
  }

  @override
  void didReceiveSkinTemperature(Device device, int timestamp, double value) {
    _ensureConnectedUi();
    setState(() {
      _skinTemperature = "${value.toStringAsFixed(1)} °C";
    });
  }

  @override
  void didReceiveSoundFeatures(
      Device device, int timestamp, List<double?> values) {}

  @override
  void didReceiveSoundVolume(Device device, int timestamp, int value) {
    setState(() {
      _soundVolume = "$value dB";
    });
  }

  @override
  void didReceiveSteps(Device device, int timestamp, int steps) {
    setState(() {
      _steps += steps;
    });
  }

  @override
  void didReceivePressure(Device device, int timestamp, int value) {}

  @override
  void didReceivePressureWearState(Device device, WearState wearState) {}

  @override
  void didReceiveUnsynchronizedSize(
      Device device, int unsynchronizedSize, double syncBytesPerSecond) {}

  @override
  void syncStateDidChange(Device device, SyncState state) {
    setState(() {
      _syncState = state.toString().split('.').last;
    });
  }

  @override
  void wearStateDidChange(Device device, WearState wearState) {
    setState(() {
      _wearState = wearState.toString().split('.').last;
    });
  }

  @override
  void didReceivePastAccelerometer(
      Device device, int timestamp, double ax, double ay, double az) {}

  @override
  void didReceivePastActivity(
      Device device, int timestamp, ActivityType activity) {}

  @override
  void didReceivePastBodyPosition(
      Device device, int timestamp, BodyPosition bodyPosition) {}

  @override
  void didReceivePastECG(Device device, int timestamp, double value) {}

  @override
  void didReceivePastGyroscope(
      Device device, int timestamp, double qx, double qy, double qz) {}

  @override
  void didReceivePastHeartRate(Device device, int timestamp, int heartRate) {}

  @override
  void didReceivePastMagnetometer(
      Device device, int timestamp, double mx, double my, double mz) {}

  @override
  void didReceivePastOrientation(
      Device device, int timestamp, double roll, double pitch, double yaw) {}

  @override
  void didReceivePastPressure(Device device, int timestamp, int values) {}

  @override
  void didReceivePastQuaternion(Device device, int timestamp, double qw,
      double qx, double qy, double qz) {}

  @override
  void didReceivePastRespiration(Device device, int timestamp, double value) {}

  @override
  void didReceivePastRespirationRate(Device device, int timestamp, int value) {}

  @override
  void didReceivePastRr(Device device, int timestamp, int rr) {}

  @override
  void didReceivePastSignalQuality(Device device, int timestamp, int value) {}

  @override
  void didReceivePastSkinTemperature(
      Device device, int timestamp, double value) {}

  @override
  void didReceivePastSoundFeatures(
      Device device, int timestamp, List<double?> values) {}

  @override
  void didReceivePastSoundVolume(
      Device device, int timestamp, int soundVolume) {}

  @override
  void didReceivePastSteps(Device device, int timestamp, int value) {}

  @override
  void didReceivePastUserEvent(Device device, int timestamp) {}
}
4
likes
150
points
58
downloads

Publisher

verified publisheraidlab.com

Weekly Downloads

Aidlab Flutter SDK. For more information please visit https://www.aidlab.com/developer

Homepage

Documentation

API reference

License

MIT (license)

Dependencies

flutter

More

Packages that depend on aidlab_sdk

Packages that implement aidlab_sdk