2

我正在尝试使用 RtMidi 编写一个 C++ 应用程序,以通过我的 Tascam FireOne MIDI 控制器向 Ableton Live 发送控制信号。到目前为止,我已经成功地通过我的 MIDI 控制器使用“a”和“s”按键向我的数码钢琴发送了 Note On + Off 信号、Volume Up + Down 信号等。

// midiout.cpp

#include <iostream>
using namespace std;

#include <signal.h>
#include <windows.h>
#include <conio.h>
#include "RtMidi.h"

int main()
{
    std::vector<unsigned char> message;

    int i, keyPress;
    int nPorts;
    char input;

    RtMidiOut *midiout = 0;

    // midiOUT
    try {
        midiout = new RtMidiOut();

        // Check available ports.
        nPorts = midiout->getPortCount();
        if ( nPorts == 0 ) {
            cout << "No ports available!" << endl;
            goto cleanup;
        }

        // List Available Ports
        cout << "\nPort Count = " << nPorts << endl;
        cout << "Available Output Ports\n-----------------------------\n";
        for( i=0; i<nPorts; i++ )
        {
            try {
                cout << "  Output Port Number " << i << " : " << midiout->getPortName(i) << endl;
            }
            catch(RtError &error) {
                error.printMessage();
                goto cleanup;
            }
        }

        cout << "\nSelect an output port number :" << endl;
        cin >> keyPress;

        while( keyPress < 0 || keyPress >= midiout->getPortCount() )
        {
            cout << "\nIncorrect selection. Please try again :" << endl;
            cin >> keyPress;
        }

        // Open Selected Port
        midiout->openPort( keyPress );

        keyPress = NULL;

        bool done = false;

        cout << "Press a key to generate a message, press 'Esc' to exit" << endl;

        while(!done)
        {
            keyPress = _getch();
            input = keyPress;
            cout << input << " is: " << keyPress << endl;

            switch ( keyPress )
            {
                case 97 : 
                // Process for keypress = a
                    // Note On: 144, 60, 90
                    message.push_back( 144 );
                    message.push_back( 60 );
                    message.push_back( 90 );
                    midiout->sendMessage( &message );
                    break;
                case 115 : 
                    // Process for keypress = s
                    // Note Off: 128, 60, 90
                    message.push_back( 128 );
                    message.push_back( 60 );
                    message.push_back( 90 );
                    midiout->sendMessage( &message );
                    break;
                case 27 : 
                    // Process for keypress = esc
                    done = true;
                    break;
            }
            message.clear();
            keyPress = NULL;
        }
    }
    catch(RtError &error) {
        error.printMessage();
        exit( EXIT_FAILURE );
    }

    cleanup:
        delete midiout;

  return 0;
}

我尝试以与上述相同的方式发送控制信号,但这次使用消息字节中的控制值代替音符开或音符关值。

当ableton live运行时,我按下一个键来发送信号,但应用程序锁定并且不会返回到while循环的开始以接收来自下一个按键的输入。

编辑:我刚刚注意到,即使上面的代码(通常运行良好)在运行ableton live并且我按下一个键时也会冻结。

进一步编辑: 我下载了一个名为 MIDI Monitor 的非常简洁的应用程序,它可以监控正在传输的 MIDI 数据:http: //obds.free.fr/midimon - 我的 MIDI 控制器设备有两个端口 -> 一个用于 MIDI,一个用于控制. 当我监控控制时,我可以发送 MIDI 信号,反之亦然。但是,例如,如果我正在监视控制并尝试发送一些 CC 类型的数据,程序会锁定。这可能是设备驱动程序问题吗?——</p>

有谁知道这里出了什么问题?

4

2 回答 2

1

只有一条评论 - 您的异常处理有点奇怪。

我会将整个代码(初始化和所有)包装在一个try/catch(RtError &err)块中,并丢失大多数其他 try/catch 块。

特别是,我不知道你的东西会达到什么效果,如果抛出catch(char * str)你根本就没有捕捉到。openPort()

于 2011-05-24T15:04:18.193 回答
0

首先,尝试发送不同的 CC 并将其映射到 Ableton 中的任意控件,以查看它是否正常工作。您尝试更改的音量控制与常规 CC 的行为略有不同。具体来说,您应该查看MIDI 组织推荐的递增/递减控制器实践(注意 0x60 == 96),即当它们编写时:

此参数分别使用 MSB 和 LSB,MSB 以半音为单位调节灵敏度,LSB 以音分为单位调节灵敏度。一些制造商甚至可能忽略对 LSB 的调整。

于 2011-05-24T14:33:02.630 回答