DIY制作MIDI腳踏控制器
MIDI控制器,可以用來控制支持MIDI協議的軟件效果器,如guitar rig。MIDI控制器和鍵盤一樣可以控制guitar rig,但是鍵盤無法在后臺對guitar rig 進行控制。而midi協議的控制器不受影響 ,不管軟件是在前臺運行還是后臺運行都可以接受到MIDI信號的控制。
先說調試過程中遇到的問題:
1.guitar rig 軟件必須在正確連接asio聲卡的情況下才會接受MIDI信號,所以調試過程中必須保證你的聲卡正常連接。
2.硬件電路中的兩個反向器可以不接但是接了也沒問題而且大多數的MIDI鍵盤電路都接了這個反相器。
3.MIDI信號的發送,剛開始我只發送了ON信號結果發現 guitar rig軟件只能識別一次這個信號,第二次就控制不了了,所以MIDI協議的信號必須發送一個ON信號,再發送一個OFF 信號就正常了。
4.由于USB端口的電流比較低,所以不要接太多的無所謂的電路上去。會供電不足。
以下是一個簡單的測試電路和測試程序以供大家學習和參考。
硬件:AT89S52單片機,usb轉MIDI線,單片機學習開發板,74HC04,300歐電阻2個。
單片機軟件程序:軟件里有做了4個按鍵的代碼 如果需要8個按鍵可以把scankey函數寫兩遍并修改對應的發送代碼即可獲得更多的按鍵。
制作出來的實物圖如下:
152507njojwwutjv9ju1t1.jpg (46.16 KB, 下載次數: 85)
下載附件
2019-4-2 17:19 上傳
電路原理圖如下:
152438qwxxy8ttc8dz0dcr.jpg (121.2 KB, 下載次數: 72)
下載附件
2019-4-2 17:19 上傳
152503ub1e191b4asyh8fa.jpg (135.79 KB, 下載次數: 84)
下載附件
2019-4-2 17:19 上傳
#include<reg52.h>
#define uchar unsigned char
#define uint unsigned int
/*****************************************************************
延遲1ms子函數
******************************************************************/
void delay1ms(uint i)
{
uchar j;
while(i--)
{
for(j=0;j<115;j++) //1ms基準延時程序
{
;
}
}
}
/*************************************************************
初始化子函數
***************************************************************/
void init()
{
TMOD=0x21; //T/C1工作于8位自動裝入狀態//
TL1=0xff; // T/C1常數,確定波特率//
TH1=0xff; //波特率為 31.25k
SCON=0x50; //設串口工作于方式1,//
PCON=0x80; //相當于SMOD=1;//
TR1=1; //T/C1開始計數//
}
/*********************************************************
串口發送子函數
*********************************************************/
void send(uchar *p)
{
uchar i;
TR1 =1;
for(i=0;i<3;i++) //當i=1 發送CC,當i=2 發送kk,當i=3發送vv
{
SBUF=*p;
while(TI==0);
TI = 0;
p++;
}
TR1 =0;
}
/**********************************************************
鍵盤掃描子函數
**********************************************************/
void scankey()
{
uchar p[] = {0x90,0x30,0x64, //按鍵1的命令,在guitar rig 里顯示的是48
0x91,0x32,0x64,//按鍵2的命令,在guitar rig 里顯示的是52
0x92,0x34,0x64,//按鍵3的命令,在guitar rig 里顯示的是53
0x93,0x35,0x64,//按鍵4的命令,在guitar rig 里顯示的是50
0x80,0x30,0x40,
0x81,0x32,0x40,
0x82,0x34,0x40,
0x83,0x35,0x40,
};
uchar temp;
P1=0xfe;//給P1口11111110
temp=P1;
temp=temp&0xf0;//取出高四位值
while(temp!=0xf0)
{
delay1ms(5);
temp=P1;//此時 p1口值不等于0xf0,等于按鍵按下后的值
temp=temp&0xf0;
while(temp!=0xf0)
{
temp=P1; //此時 p1口值不等于0xf0,等于按鍵按下后的值
while(temp==0xee)
{
send(p);
while(temp!=0xf0)//等待按鍵釋放
{
temp=P1;
temp=temp&0xf0;
}
send(p+12);
}
while(temp==0xde)
{
send(p+3);
while(temp!=0xf0)//等待按鍵釋放
{
temp=P1;
temp=temp&0xf0;
}
send(p+15);
}
while(temp==0xbe)
{
send(p+6);
while(temp!=0xf0)//等待按鍵釋放
{
temp=P1;
temp=temp&0xf0;
}
send(p+18);
}
while(temp==0x7e)
{
send(p+9);
while(temp!=0xf0)//等待按鍵釋放
{
temp=P1;
temp=temp&0xf0;
}
send(p+21);
}
}
}
}
void main()
{
init();//初始化
while(1)
{
scankey();
}
}
|