1602屏配合紅外遙控(實現光標移動)與對1602新的認識
作者:胡永明 來源:本站原創 點擊數:
… 更新時間:2013年12月10日 【字體:
大 中 小】
1、效果圖
2、代碼
3、對1602新的認識(小結)
效果圖
.jpeg)
代碼
工程文件下載:http://www.zg4o1577.cn/f/gxde_c.rar
#include <reg51.h>
#include<intrins.h>
sbit RS = P2^4; //定義端口
sbit RW = P2^5;
sbit EN = P2^6;
unsigned char sj[33]; //接收脈沖時間數組
unsigned char ac; //l602AC值,(顯示緩存地址)
unsigned char i; //脈沖個數記錄
unsigned char mcsj; //脈沖時間(大于0.56ms小于1.125ms為0,大于1.125ms小于2.25ms)
bit MC=0; //接收紅外脈沖開始標志(0:脈沖已經結束,1:脈沖剛開始)
bit JS=0; //脈沖接收結束標志位(1標志接收結束)
bit JM=0; //解碼完成標志位(1:解碼完成)
void Delay(unsigned char f);
void dsq_0() interrupt 1 using 1 //定時器T0中斷服務函數
{
mcsj++; //256
}
void wbzd_0() interrupt 0 //外部中斷服務函數
{
if(MC)
{
if(mcsj>32) //判斷是不是引導碼。(如果是i=0)
i=0;
sj[i]=mcsj; //把脈沖時間存入sj這個數組里
mcsj=0; //清空脈沖時間準備接收下一個脈沖時間
i++;
if(i==33) //判斷是否接收完脈沖時間
{
i=0;
JS = 1; //接收完成標志位置1
MC=0; //紅外脈沖結束
}
}
else
{
MC=1; //紅外脈沖開始
mcsj=0; //清空脈沖時間
}
}
void csh_dsq_0() //初始化定時器0
{
TMOD = 0x02;
TH0=0x00; //定時器0的重裝數據
TL0=0x00; //初始化
ET0=1; //打開定時器0中斷
TR0=1; //啟用定時器0
}
void csh_wbzd_0() //初始化外部中斷0
{
IT0=1; //外部中斷0下降沿觸發
EX0=1; //啟用外部中斷0
EA=1; //打開總中斷
}
void hwjm(unsigned char *p) //紅外解碼函數
{
unsigned char i,j,k=1;
for(i=0;i<4;i++) //4組數據的計數
{
for(j=0;j<8;j++) //每組數據中的8位數據計算
{
p[i] >>= 1; //數據右移一位
if(sj[k]>7) //脈沖時間大于7的就是1
p[i] |= 0x80;
k++;
}
}
JS = 0; //分析完成清零JS
JM = 1; //解碼完成JM置1
}
void Delayus(unsigned char t) // us級別延時
{
while(--t);
}
void Delayms(unsigned char t)// ms級別延時
{
while(t--)
{
//大致延時1mS
Delayus(245);
Delayus(245);
}
}
bit m_1602(bit i) //判斷1602是否忙(參數i=1:讀取AC值。i=0:不讀AC值)
{
P0 = 0xFF; //準備讀取
RS = 0;
RW = 1;
EN = 0;
_nop_();
EN = 1; //產生高電平
if(i)
ac = P0 & 0x7f; //讀取AC值
return (bit)(P0 & 0x80);
}
void x_1602(bit i,unsigned char j) //參數一是寫(0、寫指令 1、寫數據),參數二是寫入的8位數據
{
while(m_1602(0))
{
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
}
RS = i;
RW = 0;
EN = 1;
P0 = j;
_nop_();
EN = 0; //產生下降沿
}
void qp_1602() //清屏函數
{
x_1602(0,0x01); //第一個參數是:寫入的類型(0、寫指令 1、寫數據),第一個參數是:寫入的數據
Delayms(5);
}
//顯示字符
void zf_1602(unsigned char x,unsigned char y,unsigned dat) //參數一是顯示的列,參數二是顯示的行,參數三是顯示的數據
{
if(y==0)
{
x_1602(0,(0x80+x)); //第一行
}
else
{
x_1602(0,(0xc0+x)); //第二行
}
x_1602(1,dat); //寫入數據
}
void zfc_1602(unsigned char x,unsigned char y,unsigned char *dat)//參數一是顯示的列,參數二是顯示的行,參數三是顯示的數據(注:數據用\0結尾)
{
if(y==0)
{
x_1602(0,(0x80+x)); //第一行
}
else
{
x_1602(0,(0xc0+x)); //第二行
}
while(*dat) //&:取地址 *:取值
{
x_1602(1,*dat);
dat ++;
}
}
void xhc(unsigned char *p) //紅外按鍵匹配函數
{
switch(p[2]) //匹配按鍵
{
case 0x16:x_1602(1,'0');break; //按鍵0
case 0x0c:x_1602(1,'1');break; //按鍵1
case 0x18:x_1602(1,'2');break; //按鍵2
case 0x5e:x_1602(1,'3');break; //按鍵3
case 0x08:x_1602(1,'4');break; //按鍵4
case 0x1c:x_1602(1,'5');break; //按鍵5
case 0x5a:x_1602(1,'6');break; //按鍵6
case 0x42:x_1602(1,'7');break; //按鍵7
case 0x52:x_1602(1,'8');break; //按鍵8
case 0x4a:x_1602(1,'9');break; //按鍵9*/
case 0x44: //光標左移
{
m_1602(1); //讀取AC值(此函數也是讀忙值函數)
if(ac != 0x00 && ac != 0x40) //限制光標左移的邊界
x_1602(0,0x10); //光標左移
break;
}
case 0x40: //光標右移
{
m_1602(1); //讀取AC值(此函數也是讀忙值函數)
if(ac != 0x0f && ac != 0x4f)//限制光標右移的邊界
x_1602(0,0x16); //光標右移
break;
}
case 0x43: //換行
{
unsigned char io1; //記錄光標所在的列
bit io2; //記錄光標所在的行
m_1602(1); //讀取ac值
io2 = ac & 0x40; //取出行
io1 = ac & 0x0f; //取出列
if(io2) //io2=1是第一行,io2=0是第二行
{
x_1602(0,(0x80+io1));
}
else
x_1602(0,(0xC0+io1));
break;
}
case 0x09:x_1602(0,0x1);break; //清屏
}
JM=0;
}
void csh_1602() //初始化1602
{
x_1602(0,0x38); //顯示模式設置
Delayms(5);
x_1602(0,0x38);
Delayms(5);
x_1602(0,0x38);
Delayms(5);
x_1602(0,0x38);
x_1602(0,0x0f); //顯示光標
x_1602(0,0x01); //顯示清屏
x_1602(0,0x06); //顯示光標移動設置
}
void main()
{
unsigned char jmsj[4];
csh_1602();
qp_1602();
csh_wbzd_0();
csh_dsq_0();
while(1)
{
zfc_1602(1,0,la); //寫入字符串
while(1)
{
if(JS) //脈沖接收結束后調用解碼函數解碼
{hwjm(jmsj);}
if(JM) //解碼完成后調用按鍵匹配函數
{xhc(jmsj);}
}
}
}
小結
1、1602的AC值其實就是前一次的地址設置
2、讀出AC地址與寫入AC有本質的區別
以上的意思是 如下:
讀出的AC地址是:4BH(75)
但是如果我寫入的地址直接寫入 : 4BH(75)這個那就是錯的。
這里就涉及到一個寫入命令了 。。如下圖:
.jpeg)
所以要設置4B這個AC地址就要+上80H這個值。。
3、光標所在的位置就是AC地址的位置
4、這個程序有個小的BUG。
BUG就是:在第15個字符位置輸入一個字符后光標會消失
造成的 原因是AC值大于了 2FH 與 67H 這個兩個 地址。