iOS には前から Core MIDI ってのがあって、MIDIの入出力をアプリからAPIで利用することができたんだけど、
Android は、6.0(SDK 23)から Android MIDI ってのができたらしいです。
いままで Android で MIDI を使うために Kaoru Shoji さんとか、いろいろと頑張ってるひとがいたんだけど、それがやっと OSレベルで対応されたってことです。
そして、Android MIDI では Apple の BLE MIDI 規格も使えるそうなので、それをやってみます。
Apple の BLE MIDI 規格が使えるってことは、Macとつなぐこともできるってことなんだけど、せっかくだから今回は以前作ったコレを使いました。
コレは何か簡単に説明すると、自作のMIDIコントローラーです。
8の字に空いてるところがタッチセンサーになっていて、ワニ口クリップで野菜とかバナナとかと繋いで楽器にしてしまいます。
それを BLE MIDI で iPad に送って、GarageBand で音を鳴らして遊んでたんだけど、この話はあんまり関係ないのでこのくらいにしときます。
BluetoothDevice を Scan する
まず、Android 6.0 から Ble の Scan に ACCESS_COARSE_LOCATION か ACCESS_FINE_LOCATION が必要です。
<uses-permission android:name="android.permission.BLUETOOTH"/>
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
リクエストします。
if (checkSelfPermission(Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, 0);
}
Scan して、取得した BluetoothDevice を保存します。
static final UUID MIDI_SERVICE_UUID = UUID.fromString("03B80E5A-EDE8-4B33-A751-6CE34EC4C700");
BluetoothDevice mDevice = null;
void scanStart() {
ScanSettings settings = new ScanSettings.Builder().setScanMode(ScanSettings.SCAN_MODE_LOW_LATENCY).build();
ScanFilter uuidFilter = new ScanFilter.Builder().setServiceUuid(new ParcelUuid(MIDI_SERVICE_UUID)).build();
List<ScanFilter> scanFilters = new ArrayList<>();
scanFilters.add(uuidFilter);
BluetoothAdapter.getDefaultAdapter().getBluetoothLeScanner().startScan(scanFilters, settings, mScanCallback);
}
void scanStop() {
BluetoothAdapter.getDefaultAdapter().getBluetoothLeScanner().stopScan(mScanCallback);
}
ScanCallback mScanCallback =
new ScanCallback() {
@Override
public void onScanResult(int callbackType, ScanResult result) {
mDevice = result.getDevice();
scanStop();
}
@Override
public void onBatchScanResults(List<ScanResult> results) {
if(0 < results.size()){
mDevice = results.get(0).getDevice();
}
scanStop();
}
@Override
public void onScanFailed(int errorCode) {
scanStop();
}
};
BluetoothDevice と接続する
BluetoothDevice を MIDI デバイスとして接続します。
MidiManager m = (MidiManager) getSystemService(Context.MIDI_SERVICE);
m.openBluetoothDevice(mDevice, new MidiManager.OnDeviceOpenedListener() {
@Override
public void onDeviceOpened(MidiDevice device) {
if(0 < device.getInfo().getOutputPortCount()){
MidiOutputPort outputPort = device.openOutputPort(0);
outputPort.connect(new MyReceiver());
}
}
}, new Handler(Looper.getMainLooper()));
MyReceiver で受信します。
※とりあえず ノートオン・ノートオフだけ。
class MyReceiver extends MidiReceiver {
@Override
public void onSend(byte[] data, int offset,
int count, long timestamp) throws IOException {
int i = offset;
byte message = (byte) (data[i++] & 0xF0);
byte noteNo = data[i++];
byte velocity = data[i];
if (message == (byte)0x80 || ((message == (byte)0x90 && velocity == 0))) {
Log.d(TAG, String.format("noteOff: 0x%02x", noteNo));
}else if (message == (byte)0x90) {
Log.d(TAG, String.format("noteOn: 0x%02x, velocity: 0x%02x", noteNo, velocity));
}
}
}
今回はやってませんが、送信もできるそうです。
Send a NoteOn - Android MIDI User Guide
他のアプリから使う
iOSの場合、Core MIDI に対応してるけど、BLE MIDI には対応してないアプリ(Music Studioとか)でも、GarageBand などで BLE MIDI 接続してしまえば、BLE とか関係なしに MIDI デバイスとして使えるようになります。
これが Android MIDI でもできるのか、試してみました。
BLE を使わない Android MIDI アプリをつくる
BLE を使わないので Permission は不要です。
BLE を使うと、音楽アプリなのに位置情報の許可が必要とか、ユーザーにとってわけわかんないことになってしまうので、結構重要なことです。
<!-- Permission 不要 -->
ボタンクリックしたら、接続されている MIDI デバイスの一覧を取ってきて、1個めに接続します。
@Override
public void onClick(View v) {
MidiManager m = (MidiManager) getSystemService(Context.MIDI_SERVICE);
MidiDeviceInfo[] devices = m.getDevices();
Log.d(TAG, "devices: " + Arrays.toString(devices));
if(0 < devices.length){
m.openDevice(devices[0], new MidiManager.OnDeviceOpenedListener() {
@Override
public void onDeviceOpened(MidiDevice device) {
if(0 < device.getInfo().getOutputPortCount()){
MidiOutputPort outputPort = device.openOutputPort(0);
outputPort.connect(new MyReceiver());
}
}
}, new Handler(Looper.getMainLooper()));
}
}
MyReceiver はさっきと一緒です。
BLE対応アプリで接続した後に、BLE非対応アプリで MIDI デバイス一覧を取得すると、次のように BluetoothDevice も含んでます。
※ Log 出力結果です。
devices: [MidiDeviceInfo[mType=3,mInputPortCount=1,mOutputPortCount=1,mProperties=Bundle[{name=WDMD01, bluetooth_device=C6:0E:FC:01:05:87}],mIsPrivate=false]
このあと、BLE MIDI デバイスから、ノートオン・ノートオフを送ったら、BLE非対応アプリでもちゃんと受信できました。
さらに、BLE対応アプリの方を終了しても、接続は生きてるようで、BLE非対応アプリでまだ受信できます。
これだったら GarageBand のような、標準となるアプリだけ BLE MIDI 対応してしまえば、他のアプリは対応不要になるのでいいですね。
とはいっても Android には、GarageBand がないので、Music Studio の Android版とか期待してます。
まとめ
Android MIDI は iOS の Core MIDI と同じように使える。
あ、それから
BLEには対応してませんが、Android MIDI API の公式サンプルが出たらしいです。
新しい Android Marshmallow (マシュマロ) のサンプル アプリ - Google Developer Japan Blog