0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

ここのえAdvent Calendar 2023

Day 15

MIDI Remoteを使って、自分のMIDIデバイスをCubaseに対応させる

Last updated at Posted at 2023-12-14

この記事は ここのえ Advent Calendar 2023 Day 15の記事です。

注意

この記事はDTMの音楽的な話ではなく、バリバリにコーディングする話です。
JavaScriptに対してある程度知識がある人を対象にしています。

Introduction

下手の横好きで、趣味DTMをしているここのえです。

メインDAWとしてCubaseを使っているのですが、Cubase 12から MIDI Remote という機能が追加されました。

こんな名前ですがBluetoothでリモートで動かすとかではなく、簡単に言えば「Cubaseでサードパーティ製のMIDIコントローラを使えるようにするAPI」です。

MIDI自体は共通規格ですが、ミキサー関係であれば Mackie Control Protocol などに対応しているDAWが多いものの、AKAI FIREAbleton Push のようにコントロールが特定のDAWに依存しているハードウェアも多々あります。

自分も Launchpad Pro Mk3 を使っていますが、例に漏れずコイツも LogicLive 専用のデバイスです。

再生/録音ボタンが使えなかったりするのが特に不便で今まで困っていたのですが、MIDI Remoteのおかげで対応させられそうなので、それを目標に実装してみます。

MIDI Remoteのコーディングに関する情報は本当に皆無なので、この記事がDTMer兼プログラマーの皆さんの一助になれれば幸いです。

CubaseのMIDI Remoteとは?

まず最初に、MIDI Remoteについて詳細を掘っていきます。

MIDI Remoteの機能は大きく分けて3つに分けられます。

MIDIコントロールサーフィスのエディタ

CubaseのGUIエディタ上で、直感的にデバイスの配置を定義することができます。後述の MIDI マッピングアシスタント と併用します。

image.png

MIDI Remote マッピングアシスタント

MIDIコントロールサーフィスのエディタ で設定したボタンに対して、Cubaseの特定の機能を割り当てるために使うアシスタントです。

一般的なデバイスのユースケースであれば、ノーコードでこれだけ設定すれば問題ないでしょう。

image.png

MIDI Remote API

今回の本題です。

MIDI Remote API全体像.png

MIDI Remote APIは、前述のようなCubaseの MIDI Remote の機能をスクリプトによって制御することができるAPIです。

MIDI コントロールサーフィスのエディタのデバイスのボタン配置の定義や、MIDI Remote マッピングアシスタント の各MIDI操作とCubaseの操作の対応などが、スクリプト上で可能になります。

またスクリプト上でデバイスの動作を定義するため、MIDI Remote マッピングアシスタントで出来ないような複雑な挙動を定義したり、逆にCubase側からMIDIを送り、デバイスのLEDを光らせたりすることが可能です。


MIDI Remote APIの使用言語は Javascript で、ES5 までの記法が使えます。

そうです、 ES5までしか使えませんlet, const, () => {}, class/extends, デフォルト引数, ...array, これらは 全部諦めてください

今回は本題から外れてしまうので避けますが、ES5を生で書くのは流石に面倒すぎるので、Babelとかでトランスパイルする記事をそのうち書こうと思います……

実装

実際にMIDI Remote APIを使って、LaunchpadからCubaseが操作できるようなコードを実装していきます。

今回の最終目標は、以下の機能の実装を目指します。

  • 再生ボタン
    • 再生と停止
    • 再生中は緑色に点灯させ、停止中は消灯する
  • 録音ボタン
    • 録音モードのON/OFF切り替え
    • 録音モードだったら赤色に点灯、そうでなければ消灯する

基本的にLaunchpad用のコードで解説を進めていきますが、しょせんMIDIを弄っているだけです。
CCやノート番号だけ読み替えて頂ければどんなデバイスでも対応できます。

Launchpad Pro Mk3でテストする人へ

今回はシンプルにするため、Launchpad Pro Mk3のプログラマーモード(外部から制御するモード)を対象に実装します。必要があれば、詳細はマニュアルを確認してください。

https://downloads.novationmusic.com/novation/launchpad-mk3/launchpad-pro-mk3-0

開発環境の準備

まず開発環境を構築します。
VSCodeの場合は特に気にせず、公式ドキュメント通りやれば大丈夫です。

今回はWebstormで開発環境を構築してみます。
スタジオ -> MIDI Remote マネージャー を押します。
image.png

MIDI Remoteの画面が出てきたら、上部ツールバーを右クリックし、スクリプトツール にチェックを入れます。
image.png

フォルダのマークをクリックして、スクリプトの格納フォルダを開きます。このパスをWebStormでOpenします。
スクリーンショット 2023-11-16 152948.png

image.png

以下のフォルダおよび、ファイル名のルールでスクリプトを作成します。

# 規則
Local/<vendor>/<device>/<vendor>_<device>.js

# Launchpadの例
Local/Novation/Launchpad_Pro_Mk3/Novation_Launchpad_Pro_Mk3.js

WebStormはプロジェクト内のJSDocを勝手に読んでくれるので、スクリプトを開くと既に補完が動くはずです。
image.png

このままだとconstやアロー関数をうっかり使ってしまいそうなので、JavascriptのLanguage versionの指定も行っておきます。
WebStormの場合、Setting -> Languages & Frameworks -> Javascriptにあります。
image.png

デバイスのセットアップ

コーディングを始める前に、先述のスクリプトツールの左から二番目、コンソールのボタンを押して MIDI Remote スクリプトコンソール を開きます。
ここでCubase側にスクリプトをリロードさせたり、MIDIポートの識別名を確認したりできます。
image.png

image.png

Tips: デバッグ方法

普通にconsole.log()が使えます。MIDI Remote スクリプトコンソールに表示されます。

console.log(newValue.toString())

image.png

デバイスのセットアップを行うコードを書きます。

// MIDI Remote API
var api = require('midiremote_api_v1')

// デバイスのオブジェクトを作成
// makeDeviceDriver('メーカー', 'デバイス名', 'スクリプトの作成者')
var device = api.makeDeviceDriver("Novation", "Launchpad Pro Mk3", "Kokonoe")

// MIDIポートの参照を拾う
var midiIn = device.mPorts.makeMidiInput()
var midiOut = device.mPorts.makeMidiOutput()

// MIDIデバイスの識別名と対応させる
// NOTE: デバイスによってはWindowsとMacで名称が異なる場合アリ。要注意!
device.makeDetectionUnit().detectPortPair(midiIn, midiOut)
    .expectInputNameEquals('LPProMK3 MIDI')
    .expectOutputNameEquals('LPProMK3 MIDI')

レイアウトの定義

サーフィスのレイアウトを定義します。先程のセットアップのコードに引き続いて書いていきます。

// ボタンの定義
// makeButton(x座標, y座標, width, height)
var play = device.mSurface.makeButton(0, 0, 1, 1)
var rec = device.mSurface.makeButton(1, 0, 1, 1)

// サーフィスとMIDI信号のバインド
play.mSurfaceValue.mMidiBinding
    .setInputPort(midiIn).bindToControlChange(0, 20)

rec.mSurfaceValue.mMidiBinding
    .setInputPort(midiIn).bindToControlChange(0, 10)

device.mSurfaceに対して、各ボタンを作成していきます。
makeButton 以外にも makeKnob など、MIDI コントロールサーフィスのエディタ で設定できるサーフィスと同様のものがあります。

mSurfaceValue.mMidiBinding以下を使って、サーフィス要素とMIDI信号のバインドを行えます。

今回は再生ボタンが CC 20、録音ボタンが CC 10なのでそれぞれ割り当てています。

LEDを取り扱うときの注意

公式の Simple Device のサンプルコードでは、setOutputPort(midiOut) も使って、MIDI OUTもバインドしています。

play.mSurfaceValue.mMidiBinding
    .setInputPort(midiIn)
    .setOutputPort(midiOut)
    .bindToControlChange(0, 20)

MIDI OUTをバインドしてしまうと、ボタンが押されたときにCubaseがデータを勝手に送ってしまいます。

加えてsetOutputPort のMIDI信号が sendMidi() メソッドで送信するMIDIを上書きしてしまうので、自分でLEDが制御できなくなってしまいます。
LEDを使うときは、OUTはバインドしないようにしましょう

Cubaseへの操作割当、LED情報の送信

一番コアな部分の実装です。実際に操作を割り当てていきます。
加えて操作に変更があった時、MIDI信号を送ってLEDを光らせます。

// マッピングページの作成
var page = device.mMapping.makePage("Default Page")

// ボタンとCubase操作のマッピング
page.makeValueBinding(play.mSurfaceValue, page.mHostAccess.mTransport.mValue.mStart).setTypeToggle()
page.makeValueBinding(rec.mSurfaceValue, page.mHostAccess.mTransport.mValue.mRecord).setTypeToggle()

// LEDの処理
play.mSurfaceValue.mOnProcessValueChange = function (context, newValue) {
    if(newValue === 1) {  // 再生中の時
        midiOut.sendMidi(context, [0xb0, 20, 25])
    } else {  // 停止中の時
        midiOut.sendMidi(context, [0xb0, 20, 0])
    }
}

rec.mSurfaceValue.mOnProcessValueChange = function (context, newValue) {
    if(newValue === 1) {  // 再生中の時
        midiOut.sendMidi(context, [0xb0, 10, 5])
    } else {  // 停止中の時
        midiOut.sendMidi(context, [0xb0, 10, 0])
    }
}

一つずつ見ていきます。

マッピングページ

// マッピングページの作成
var page = device.mMapping.makePage("Default Page")

MIDI Remoteにはマッピングページという概念があり、これを切り替えることによって各ボタンの挙動を変更することができます。

image.png

裏を返せば、各ボタンの挙動はマッピングページをベースにしているため、マッピングページを作成しないとボタンに動作を設定できません。

このコードでは MR_DeviceDriver.mMapping.makePage("ページ名") でページを作成しています。

ボタンとCubase操作のマッピング

// makeValueBinding(サーフィスのvalue, Cubase操作のvalue)
page.makeValueBinding(play.mSurfaceValue, page.mHostAccess.mTransport.mValue.mStart).setTypeToggle()

makeValueBindingでボタンとCubaseの操作をバインドすることができます。
バインドされるとMIDI Remoteのサーフィス表示も自動的に変更されます。今回は再生・録音の2つのボタンを用意しているので、それぞれに対象のアイコンが表示されています。

image.png

また再生・録音についてはON/OFFのトグル式なので、.setTypeToggle() を使って押すたびに切り替わるように設定します。

LEDの処理 (MIDI信号の送信)

// LEDの処理
// context: device
play.mSurfaceValue.mOnProcessValueChange = function (context, newValue) {
    if(newValue === 1) {  // 再生中の時
        midiOut.sendMidi(context, [0xb0, 20, 25])
    } else {  // 停止中の時
        midiOut.sendMidi(context, [0xb0, 20, 0])
    }
}

mSurfaceValue.mOnProcessValueChangefunction を当てることで、ボタンにバインドされた値が変更されたときにcallbackが走るようになります。
第一引数には対象のデバイスの参照、第二引数に変更された後の値が返ってくるので、これを使って処理を行います。

上記の例では再生を割り当てていますが、再生時は1、停止時は0 が返ってきます。フェーダーやパンの値をバインドしていれば、それに対応したnumber型の値が返ってきます。

今回はLEDを光らせるために、Launchpadの仕様に基づいてデータを送信しています。

sendMidi()の送信内容について

sendMidi() の第二引数ですが、そのまま生のMIDIメッセージを書いて送る必要があります。
第二引数のarray(3)がMIDIメッセージの第1, 第2, 第3バイトにあたります。

今回の例では、channel 0CC 2025(緑色)を送信しています。

  • 第1バイト 0xb0 => b = CC, 0 MIDI Channel
  • 第2バイト 20 => CC番号
  • 第3バイト 25 => 対象のCCに送るValue

詳細はこちらのページが分かりやすいです。
https://www.g200kg.com/jp/docs/dic/controlchange.html



まとめ

MIDI Remote APIの情報についてはほとんどネット上に落ちておらず、基本的には公式のドキュメント頼りだったので、折角なので書いてみました。

Cubase 12以降しか使えないこともあり、対応したCubaseを持っている方が少ないという問題はありますが、コーディングしてCubaseを自由自在に操れるのはとても魅力的です。感覚として近いところだと、自作キーボードの実装やってる時と同じようなワクワク感があります。

最近発売された Cubase 13 では大幅にUI・操作性の改善も入りました。プログラマーかつDTMerかつ"Cubase": "^12.0"なユーザに限られてしまいますが、ぜひ一度触ってみてください!

おまけ

今回使ったコードのGistです。

参考

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?