flutter_midi_command 0.4.15 flutter_midi_command: ^0.4.15 copied to clipboard
A Flutter plugin for sending and receiving MIDI messages between Flutter and physical and virtual MIDI devices. Wraps CoreMIDI and android.media.midi in a thin dart/flutter layer.
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_midi_command/flutter_midi_command.dart';
import 'controller.dart';
void main() => runApp(const MyApp());
class MyApp extends StatefulWidget {
const MyApp({Key? key}) : super(key: key);
@override
MyAppState createState() => MyAppState();
}
class MyAppState extends State<MyApp> {
StreamSubscription<String>? _setupSubscription;
StreamSubscription<BluetoothState>? _bluetoothStateSubscription;
final MidiCommand _midiCommand = MidiCommand();
bool _virtualDeviceActivated = false;
bool _iOSNetworkSessionEnabled = false;
bool _didAskForBluetoothPermissions = false;
@override
void initState() {
super.initState();
_setupSubscription = _midiCommand.onMidiSetupChanged?.listen((data) async {
if (kDebugMode) {
print("setup changed $data");
}
setState(() {});
});
_bluetoothStateSubscription =
_midiCommand.onBluetoothStateChanged.listen((data) {
if (kDebugMode) {
print("bluetooth state change $data");
}
setState(() {});
});
_updateNetworkSessionState();
}
@override
void dispose() {
_setupSubscription?.cancel();
_bluetoothStateSubscription?.cancel();
super.dispose();
}
_updateNetworkSessionState() async {
var nse = await _midiCommand.isNetworkSessionEnabled;
if (nse != null) {
setState(() {
_iOSNetworkSessionEnabled = nse;
});
}
}
IconData _deviceIconForType(String type) {
switch (type) {
case "native":
return Icons.devices;
case "network":
return Icons.language;
case "BLE":
return Icons.bluetooth;
default:
return Icons.device_unknown;
}
}
Future<void> _informUserAboutBluetoothPermissions(
BuildContext context) async {
if (_didAskForBluetoothPermissions) {
return;
}
await showDialog<void>(
context: context,
barrierDismissible: false,
builder: (BuildContext context) {
return AlertDialog(
title: const Text(
'Please Grant Bluetooth Permissions to discover BLE MIDI Devices.'),
content: const Text(
'In the next dialog we might ask you for bluetooth permissions.\n'
'Please grant permissions to make bluetooth MIDI possible.'),
actions: <Widget>[
TextButton(
child: const Text('Ok. I got it!'),
onPressed: () {
Navigator.of(context).pop();
},
),
],
);
},
);
_didAskForBluetoothPermissions = true;
return;
}
@override
Widget build(BuildContext context) {
return MaterialApp(
home: Scaffold(
appBar: AppBar(
title: const Text('FlutterMidiCommand Example'),
actions: <Widget>[
Switch(
value: _iOSNetworkSessionEnabled,
onChanged: (newValue) {
_midiCommand.setNetworkSessionEnabled(newValue);
setState(() {
_iOSNetworkSessionEnabled = newValue;
});
}),
Switch(
value: _virtualDeviceActivated,
onChanged: (newValue) {
setState(() {
_virtualDeviceActivated = newValue;
});
if (newValue) {
_midiCommand.addVirtualDevice(name: "Flutter MIDI Command");
} else {
_midiCommand.removeVirtualDevice(
name: "Flutter MIDI Command");
}
}),
Builder(builder: (context) {
return IconButton(
onPressed: () async {
// Ask for bluetooth permissions
await _informUserAboutBluetoothPermissions(context);
// Start bluetooth
if (kDebugMode) {
print("start ble central");
}
await _midiCommand
.startBluetoothCentral()
.catchError((err) {
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
content: Text(err),
));
});
if (kDebugMode) {
print("wait for init");
}
await _midiCommand
.waitUntilBluetoothIsInitialized()
.timeout(const Duration(seconds: 5), onTimeout: () {
if (kDebugMode) {
print("Failed to initialize Bluetooth");
}
});
// If bluetooth is powered on, start scanning
if (_midiCommand.bluetoothState ==
BluetoothState.poweredOn) {
_midiCommand
.startScanningForBluetoothDevices()
.catchError((err) {
if (kDebugMode) {
print("Error $err");
}
});
if (context.mounted) {
ScaffoldMessenger.of(context)
.showSnackBar(const SnackBar(
content: Text('Scanning for bluetooth devices ...'),
));
}
} else {
final messages = {
BluetoothState.unsupported:
'Bluetooth is not supported on this device.',
BluetoothState.poweredOff:
'Please switch on bluetooth and try again.',
BluetoothState.poweredOn: 'Everything is fine.',
BluetoothState.resetting:
'Currently resetting. Try again later.',
BluetoothState.unauthorized:
'This app needs bluetooth permissions. Please open settings, find your app and assign bluetooth access rights and start your app again.',
BluetoothState.unknown:
'Bluetooth is not ready yet. Try again later.',
BluetoothState.other:
'This should never happen. Please inform the developer of your app.',
};
if (context.mounted) {
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
backgroundColor: Colors.red,
content: Text(messages[_midiCommand.bluetoothState] ??
'Unknown bluetooth state: ${_midiCommand.bluetoothState}'),
));
}
}
if (kDebugMode) {
print("done");
}
// If not show a message telling users what to do
setState(() {});
},
icon: const Icon(Icons.refresh));
}),
],
),
bottomNavigationBar: Container(
padding: const EdgeInsets.all(24.0),
child: const Text(
"Tap to connnect/disconnect, long press to control.",
textAlign: TextAlign.center,
),
),
body: Center(
child: FutureBuilder(
future: _midiCommand.devices,
builder: (BuildContext context, AsyncSnapshot snapshot) {
if (snapshot.hasData && snapshot.data != null) {
var devices = snapshot.data as List<MidiDevice>;
return ListView.builder(
itemCount: devices.length,
itemBuilder: (context, index) {
MidiDevice device = devices[index];
return ListTile(
title: Text(
device.name,
style: Theme.of(context).textTheme.headlineSmall,
),
subtitle: Text(
"ins:${device.inputPorts.length} outs:${device.outputPorts.length}, ${device.id}, ${device.type}"),
leading: Icon(device.connected
? Icons.radio_button_on
: Icons.radio_button_off),
trailing: Icon(_deviceIconForType(device.type)),
onLongPress: () {
_midiCommand.stopScanningForBluetoothDevices();
Navigator.of(context)
.push(MaterialPageRoute<void>(
builder: (_) => ControllerPage(device),
))
.then((value) {
setState(() {});
});
},
onTap: () {
if (device.connected) {
if (kDebugMode) {
print("disconnect");
}
_midiCommand.disconnectDevice(device);
} else {
if (kDebugMode) {
print("connect");
}
_midiCommand.connectToDevice(device).then((_) {
if (kDebugMode) {
print("device connected async");
}
}).catchError((err) {
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
content: Text(
"Error: ${(err as PlatformException?)?.message}")));
});
}
},
);
},
);
} else {
return const CircularProgressIndicator();
}
},
),
),
),
);
}
}