之前用STM32做過按鍵矩陣,沒加任何上拉下拉的5個OUT_PIN和5個IN_PIN的那種,最近使用51單片機STC11F04E再次制作5*6的按鍵矩陣,因為IO口不夠所以特地使用了IO拓展IC-----CH423S。因為官方沒有提供按鍵矩陣的例程,所以我走的點彎路,希望這篇帖子能幫助諸位。
CH423除去電源和地以及2個i2c控制腳外共有16個通用輸出腳可以推挽輸出和開漏輸出,8個雙向輸出IO口可以輸入也可以輸出。
在使用STM32做按鍵掃描時我一般是讓5個OUT_PIN輪流輸出高電平,然后讀取5個IN_PIN腳的電平。如下:
for(i = 0;i<5;i++)
{
out_pin_1(i);//第i號引腳輸出高電平,其他引腳輸出低電平
for(j = 0;j<5;j++)
{
if(in_pin(j))//判斷第J號引腳是否檢測到高電平
{
delay_ms(10);//延時消抖
if(in_pin(j))//再次判定
{
key_num = i*6+j;//按鍵賦值
while(in_pin(j)){;}//等待按鍵彈起
break;//見此到按鍵,按鍵又彈起來了,當然退出了,我又不做組合按鍵
}else{;}//第一次見此到按鍵值10ms后檢測不到,判定為誤觸,無效
}
}
}
以上是STM32的按鍵掃描,這里如果in_pin引腳懸空讀出來的值是0,如果被按下的話值就是!0(非零)。
我用相同的思路在CH423上發現錯的離譜,在CH423里面懸空腳讀取時他的電平是1,也就是說CH423里面除非IO口和GND像連接,否則他讀取出來的值就是1,因此想要用CH423S做按鍵矩陣,在按鍵掃描時只能檢測低電平。
unsigned int CHECK_KEY(void)
{
unsigned char key_num1,key_num2,i;
for(i = 0;i < 5;i++)
{
CH423_WriteByte(0x4400|((0Xfe << i)+1));//設置低8位開漏輸出命令
delay_ms(5);
key_num1 = CH423_ReadByte(); // 讀取數據
if(key_num1 != 0xff)
{
_nop_();_nop_();_nop_();_nop_();_nop_();
key_num2 = CH423_ReadByte();
if(key_num1 == key_num2){
while(CH423_ReadByte() != 0xff){_nop_();_nop_();_nop_();_nop_();_nop_();}
return (key_num1 | (0x01 << (i+8)));
}
else key_num1 = 0;
}
}
return 0;
}
|