找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3191|回复: 1
打印 上一主题 下一主题
收起左侧

lcd12864液晶驱动源码,独创中英文混合输出 控制器7920

[复制链接]
跳转到指定楼层
楼主
ID:161768 发表于 2017-1-14 17:55 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
控制器7920
代码没啥好说,我尽量多写了注释,播放动画也很流畅的
亮点就是支持中英文混合输出,其他没啥

昨天发现座的公交车上用的也是这种12864 ,显示温度和时间...

完整源码下载:
lcd12864源码.rar (7.58 KB, 下载次数: 32)


c语言程序:
  1. #ifndef  _LCD12864_H
  2. #define         _LCD12864_H
  3. #include "my51.h"

  4. #define LCD_dataBus  P0           //总线
  5. sbit LCD_EN = P3^4;                   //使能控制
  6. sbit LCD_RS = P3^5;                   //数据_命令选择端
  7. sbit LCD_RW = P3^6;                   //读写控制
  8. sbit LCD_PSB= P3^7;                   //串并选择,H并行,L串行


  9. extern bool bShowPicFlag;  //绘图时图像显示控制(仅控制GDRAM)

  10. /*************************基本功能函数*********************************************/
  11. extern void LCD12864_init();                                        //初始化,必须置顶调用
  12. extern void LCD12864_setPos(u8 row, u8 cols);        //设置光标位置
  13. extern void LCD12864_writeByte(u8 dat);                        //写一个字节
  14. extern void LCD12864_writeCmd(u8 cmd);                        //写指令
  15. extern   u8 LCD12864_readByte();                                //读一个字节ram
  16. extern   u8 LCD12864_readIR();                                        //读暂存器
  17. extern bool LCD12864_isBusy();                                        //判忙

  18. /*************************调用基本字库显示文字**************************************/        
  19. //独创支持全角半角字符及中英混合的字符串,或字符串的子串,起始行号row(0-3)和列坐标cols(0-15),写满屏幕为止
  20. //行号4-7行是滚动区
  21. extern void LCD12864_writeData(u8 row, u8 cols,u8* pBuf,u8 dataSize);//写一堆数据
  22. extern void LCD12864_earseSomeDDRam(u8 row,u8 cols,u8 dataSize);        //擦除N字节DDRam

  23. /*************************用户自定义图标字体****************************************/
  24. extern void LCD12864_writeCGRAM(u8 userRamNum,u8* pCGRAM_userCode); //写自定义图标字体
  25. extern void LCD12864_showCGRAM(u8 row,u8 cols,u8 num) ;        //显示自定义图标字体,row(0-3),cols(0-15)
  26. extern void LCD12864_clearCGRAM(u8 CGRAM_groupNum) ;    //CGRAM清零(初始化也可清0)

  27. /*************************图像显示功能函数**********************************************/
  28. extern void LCD12864_showGDRAM(bool bShowImage);          //GDRAM绘图显示开关
  29. extern void LCD12864_clearGDRAM();                                            //液晶整个可视区的GDRAM快速清0
  30. //前4参数:起始点x(0-7)位址,y(0-63)坐标,要显示的宽度(1-128)和高度(1-64)[可显示从图像左上角开始的部分区域]
  31. //后4参数:图像代码地址,图像本身的宽度(1-128)和高度(1-64), 反白(true反白,false不反白)
  32. //画图填充GDRAM        ,注:显示的区域或者图像本身宽度必须是8的倍数
  33. extern u8 LCD12864_drawGDRAM(u8 x,u8 y,u8 width,u8 height,u8 *pImageCode,u8 imageWidth,u8 imageHight,bool bReverse);
  34. extern void LCD12864_drawDot(u8 x,u8 y,u8 flag);                   //打点,x(0-127),y(0-63),flag(0正常,1反白,2清0)
  35. extern bool LCD12864_drawXYLine(u8 x1, u8 y1,u8 x2, u8 y2,u8 flag);                  //画水平或垂直直线
  36. extern void LCD12864_drawAnyLine(u8 x1, u8 y1,u8 x2, u8 y2,u8 flag) ;          //画任意直线
  37. extern void LCD12864_drawRectangle(u8 x,u8 y ,u8 width,u8 height,u8 flag);//画矩形
  38. extern void LCD12864_drawFillRect(u8 x,u8 y, u8 width,u8 hight,u8 flag);  //填充矩形,可对矩形区反白或清0
  39. extern void LCD12864_drawVerticalSquare(u8 x,u8 y,u8 r,u8 flag)        ;                   //画站立的正方形
  40. extern void LCD12864_drawCircle(u8 x0,u8 y0,u8 r,u8 flag);                                    //画圆

  41. //打点法全屏画图有点慢了,打点法反白矩形区还可以,建议矩形范围小一些,不然比较慢
  42. //打点法效果最好,因为文字的矩形区比较小,速度很快
  43. //不过绘图法反白文字效率高,flash空间充裕的建议用绘图法

  44. /***************************全屏滚动*******************************************/
  45. //需要滚动时,用LCD12864_writeScrollData()函数,参数和LCD12864_writeData()一样
  46. extern void LCD12864_writeScrollData(u8 row,u8 cols,u8* pBuf,u8 dataSize);//写数据 (滚动模式)
  47. extern void LCD12864_setScrollPos(u8 row, u8 cols);                //设置滚动模式的坐标
  48. extern void LCD12864_showScrollCGRAM(u8 row,u8 cols,u8 CGRAM_groupNum);//用于CGRAM滚动显示前执行
  49. extern void LCD12864_startScroll(u8 scrollNum,u16 delay_ms);  //滚动开始
  50. #endif                                                                                                                                 
复制代码
  1. #include "lcd12864.h"
  2. //图像与文字,图像与用户图标,像素重合时是异或关系
  3. //文字与用户图标是覆盖关系
  4. bool bShowPicFlag=false;                  //绘图时图像显示控制

  5. /*
  6. u8 LCD12864_table[]={"123456789"};
  7. */

  8. void LCD12864_drawFillRect(u8 x,u8 y, u8 width,u8 hight,u8 flag)//填充任意位置的整个矩形
  9. {//矩形起始坐标x(0-127),y(0-63),宽(1-128)高(1-64),flag有3种值,0正常填充,1反色,2清0
  10.         u8 i=0;
  11.         u8 j=0;
  12.         if(0==width||0==hight)         //矩形宽度或高度为0时返回
  13.         {
  14.                 return ;
  15.         }
  16.         if( (x+width>128 ||(y+hight)>64)  )
  17.         {
  18.                 led2=0;
  19.                 return;
  20.         }
  21.         for(j=0;j<hight;j++)
  22.         {               
  23.                 for(i=0;i<width;i++)
  24.                 {
  25.                         LCD12864_drawDot(x+i, y+j,flag);               
  26.                 }
  27.         }                        
  28. }


  29. void LCD12864_drawRectangle(u8 x,u8 y, u8 width,u8 hight,u8 flag)//画矩形
  30. {        //矩形起始坐标x(0-127),y(0-63),宽度(1-128)和高度(1-64)        flag有3种值,0正常写1,1反色,2清0
  31.         if(0==width||0==hight)         //矩形宽度或高度为0时返回
  32.         {
  33.                 return ;
  34.         }
  35.         width--;hight--;
  36.         LCD12864_drawXYLine(x, y,x+width, y, flag);
  37.         LCD12864_drawXYLine(x+width, y,x+width, y+hight, flag);
  38.         LCD12864_drawXYLine(x, y,x, y+hight, flag);
  39.         LCD12864_drawXYLine(x, y+hight,x+width, y+hight, flag);        
  40. }

  41. bool LCD12864_drawXYLine(u8 x1,u8 y1,  u8 x2, u8 y2,  u8 flag)//画水平或垂直直线
  42. {        //起始点坐标和终点坐标,x(0-127),y(0-63),  flag有3种值,0正常写1,1反色,2清0
  43.         u8 n=0;
  44.         if(flag>2|| x1>127||x2>127||y1>63||y2>63)
  45.         {
  46.                 return false;
  47.         }

  48.         if(x1==x2)
  49.         {
  50.                 for(n=0;n<abs(y2-y1)+1;n++)
  51.                 {
  52.                         LCD12864_drawDot( x1,y1+(y2>=y1?n:-n) ,flag);        
  53.                 }
  54.         }

  55.         if(y1==y2)
  56.         {
  57.                 for(n=0;n<abs(x2-x1)+1;n++)
  58.                 {
  59.                          LCD12864_drawDot(x1+(x2>=x1?n:-n),y1,flag)        ;
  60.                 }
  61.         }
  62.         return true;        
  63. }

  64. void LCD12864_drawCircle(u8 x0,u8 y0,u8 r,u8 flag)
  65. {
  66.         s8 a,b;
  67.         s8 di;
  68.         if(r>31 ||r==0) return;    //参数过滤,次液晶显示的最大圆半径为31
  69.         a=0;
  70.         b=r;
  71.         di=3-2*r;       //判断下个点位置的标志
  72.         while(a<=b)
  73.         {
  74.                 LCD12864_drawDot(x0-b,y0-a,flag);  //3           
  75.                 LCD12864_drawDot(x0+b,y0-a,flag);  //0           
  76.                 LCD12864_drawDot(x0-a,y0+b,flag);  //1      
  77.                 LCD12864_drawDot(x0-b,y0-a,flag);  //7           
  78.                 LCD12864_drawDot(x0-a,y0-b,flag);  //2            
  79.                 LCD12864_drawDot(x0+b,y0+a,flag);  //4               
  80.                 LCD12864_drawDot(x0+a,y0-b,flag);  //5
  81.                 LCD12864_drawDot(x0+a,y0+b,flag);  //6
  82.                 LCD12864_drawDot(x0-b,y0+a,flag);            
  83.                 a++;
  84.                 //使用Bresenham算法画圆     
  85.                 if(di<0)
  86.                 di +=4*a+6;
  87.                 else
  88.                 {
  89.                         di +=10+4*(a-b);   
  90.                         b--;
  91.                 }
  92.                 LCD12864_drawDot(x0+a,y0+b,flag);
  93.         }
  94. }


  95. void LCD12864_drawVerticalSquare(u8 x,u8 y,u8 r,u8 flag)        //画站立的正方形
  96. {
  97.     u8 a,b;
  98.     float c=0;
  99.     a = 0;
  100.     b = r;
  101.     c = 3 - 2*r;
  102.     while(a < b)
  103.     {
  104.         LCD12864_drawDot(x+a,y+b,flag);
  105.         LCD12864_drawDot(x-a,y+b,flag);
  106.         LCD12864_drawDot(x+a,y-b,flag);
  107.         LCD12864_drawDot(x-a,y-b,flag);
  108.         
  109.         LCD12864_drawDot(x+b,y+a,flag);
  110.         LCD12864_drawDot(x-b,y+a,flag);
  111.         LCD12864_drawDot(x+b,y-a,flag);
  112.         LCD12864_drawDot(x-b,y-a,flag);
  113.         
  114.         if(c < 0)
  115.         {
  116.             c = c+4*a + 6;
  117.         }
  118.         else
  119.         {
  120.             c= c + 4*(a - b) + 10;
  121.             b-=1;
  122.         }
  123.         a = a + 1;  //控制打点间隔
  124.         
  125.     }
  126.     if(a == b)
  127.     {
  128.         LCD12864_drawDot(x+a,y+b,flag);
  129.         LCD12864_drawDot(x-a,y+b,flag);
  130.         LCD12864_drawDot(x+a,y-b,flag);
  131.         LCD12864_drawDot(x-a,y+b,flag);
  132.         
  133.         LCD12864_drawDot(x+b,y+a,flag);
  134.         LCD12864_drawDot(x-b,y+a,flag);
  135.         LCD12864_drawDot(x+b,y-a,flag);
  136.         LCD12864_drawDot(x-b,y-a,flag);      
  137.     }
  138. }


  139. void LCD12864_drawAnyLine(u8 StartX, u8 StartY,u8 EndX, u8 EndY, u8 flag)  //画任意直线
  140. {
  141.     u8 t, distance ;      /*根据屏幕大小改变变量类型(如改为int型)*/
  142.     s16 x = 0 , y = 0 ;
  143.     s8 incx, incy, dx, dy ;
  144.         if((StartX==EndX) ||(StartY==EndY))
  145.         {
  146.                 LCD12864_drawXYLine(StartX,StartY,EndX,EndY,flag);
  147.                 return;
  148.         }

  149.     dx = EndX - StartX ;
  150.     dy = EndY - StartY ;               
  151.         incx = dx > 0 ?1:-1;
  152.         incy = dy > 0 ?1:-1;

  153.     dx = abs( dx );
  154.     dy = abs( dy );
  155.     if( dx > dy )
  156.     {
  157.         distance = dx ;
  158.     }
  159.     else
  160.     {
  161.         distance = dy ;
  162.     }
  163.     LCD12864_drawDot( StartX, StartY, flag ) ;                  //反白补点
  164.     for( t = 0 ; t <= distance+1 ; t++ )
  165.     {
  166.         LCD12864_drawDot( StartX, StartY, flag ) ;
  167.         x += dx ;
  168.         y += dy ;
  169.         if( x > distance )
  170.         {
  171.             x -= distance ;
  172.             StartX += incx ;
  173.         }
  174.         if( y > distance )
  175.         {
  176.             y -= distance ;
  177.             StartY += incy ;
  178.         }
  179.     }
  180. }

  181. void LCD12864_drawDot(u8 x, u8 y,u8 flag)        //画点,0打点,1反色,2清0
  182. {  //x(0-127),y(0-63),flag有3种值,0正常写1,1反色,2清0
  183.     u8 x_word=0;                 //水平(0-127)个像素中的哪个字,一字16位
  184.     u8 x_mode=0;                 //取余
  185.     u8 y_part=0;
  186.     u8 y_bit=0;
  187.     u8 tempH=0;
  188.     u8 tempL=0;
  189.         x_word=x>>4;            //在哪一个字(0-7)              ,x_word=x/16
  190.     x_mode=x&0x0f;      //在该字的哪一位                  ,x_mode= x%16
  191.     y_part=y>>5;        //在哪个屏0或1                          ,y_part=y/32
  192.     y_bit= y&0x1f;      //垂直方向,y_bit范围(0-31),y_bit=y%32
  193.     bShowPicFlag?LCD12864_writeCmd(0x36):LCD12864_writeCmd(0x34);
  194.     LCD12864_writeCmd(0x80+y_bit);        //垂直坐标
  195.     LCD12864_writeCmd(0x80+8*y_part+x_word);                   //水平位址
  196.   
  197.     LCD12864_readByte();
  198.     tempH=LCD12864_readByte();  //先将该字16位数据保存
  199.     tempL= LCD12864_readByte();
  200.    
  201.     LCD12864_writeCmd(0x80+y_bit);                           //重设地址,因为AC计数器变了
  202.     LCD12864_writeCmd(0x80+8*y_part+x_word);                   //水平位址
  203.         if(0==flag)                                                                                   //不反白,打1,
  204.         {
  205.             if(x_mode<8)  //如果x_mode小于8,说明点应位于该字的左边高8位中
  206.             {
  207.                 LCD12864_writeByte( tempH | bit(7- x_mode) );
  208.                 //LCD12864_writeByte(tempL);
  209.             }
  210.             else
  211.             {
  212.                 //LCD12864_writeByte(tempH);
  213.                         LCD12864_readByte();         //让AC走半步
  214.                 LCD12864_writeByte(tempL|bit(15-x_mode));
  215.             }         
  216.         }
  217.         else if(1==flag)                        //反白,该点与原来的状态相反
  218.         {
  219.                 if(x_mode<8)  //如果x_mode小于8,说明点应位于该字的左边高8位中
  220.             {
  221.                         if(tempH & bit(7- x_mode))             //原来是1
  222.                         {
  223.                                 LCD12864_writeByte( tempH&~bit(7- x_mode) );        //写0
  224.                         }
  225.                         else                                                         //原来是0
  226.                         {
  227.                                 LCD12864_writeByte( tempH | bit(7- x_mode) );        //写1
  228.                         }
  229.                
  230.             }
  231.             else
  232.             {
  233.                         LCD12864_readByte();         //让AC走半字
  234.                         if(tempL& bit(15-x_mode))          //原来是1的写0
  235.                         {
  236.                                 
  237.                                 LCD12864_writeByte(tempL&~bit(15-x_mode));         //写0
  238.                         }
  239.                         else
  240.                         {
  241.                                 LCD12864_writeByte(tempL|bit(15-x_mode));         //写0
  242.                         }
  243.                
  244.             }
  245.         }
  246.         else if(2==flag)                          //清0
  247.         {
  248.                 if(x_mode<8)  //如果x_mode小于8,说明点应位于该字的左边高8位中
  249.             {
  250.                 LCD12864_writeByte( tempH&~bit(7- x_mode) );
  251.             }
  252.             else
  253.             {
  254.                         LCD12864_readByte();         //让AC走半字
  255.                 LCD12864_writeByte(tempL&~bit(15-x_mode));
  256.             }        
  257.         }  
  258. }

  259. void LCD12864_showGDRAM(bool bShowImage)        //GDRAM图像显示开关
  260. {
  261.         if(bShowImage)                 //开启显示
  262.         {
  263.             LCD12864_writeCmd(0x36);
  264.             LCD12864_writeCmd(0x30);               
  265.         }
  266.         else                                 //关闭显示
  267.         {
  268.             LCD12864_writeCmd(0x34);
  269.             LCD12864_writeCmd(0x30);               
  270.         }
  271. }


  272. //填充GDRAM
  273. u8 LCD12864_drawGDRAM(u8 x,u8 y,u8 width,u8 height,u8* pImageCode,u8 imageWidth,u8 imageHight,bool bReverse)
  274. {//前4参数:起始点x(0-7)位址,y(0-63)坐标,要显示的宽度(1-128)和高度(1-64)[可显示从图像左上角开始的部分区域]
  275. //后4参数:图像代码地址,图像本身的宽度(1-128)和高度(1-64), 反白(true反白,false不反白)
  276.         u8 i=0;
  277.         u8 j=0;
  278.         if( height > imageHight )           //检测显示高度,宽度不检测不会乱码
  279.         {                                                           //显示的高度不能超过图片本身高度
  280.                 return 0x01;                           //也就是说可显示图像的部分区域(从图像左上角开始的部分区域)
  281.         }
  282.         width>>=3;                                           //像素宽度转化为字节个数,所以width必须是8的整数倍
  283.         imageWidth>>=3;                                   //像素宽度转化为字节个数,所以width必须是8的整数倍
  284.         if(bShowPicFlag)                                
  285.         {
  286.                 LCD12864_writeCmd(0x36);        //改写GDRAM时,开启绘图显示,可防止动画显示时闪动                 
  287.         }
  288.         else
  289.         {
  290.                 LCD12864_writeCmd(0x34);        //改写GDRAM时,关闭绘图显示
  291.         }

  292.         for(j=0;j<height;j++)                   //写GDRAM
  293.         {
  294.                 if(y+j>31)                                                        //地址变换
  295.                 {
  296.                         LCD12864_writeCmd(0x80+y+j-32);        //垂直坐标
  297.                         LCD12864_writeCmd(0x88+x);                //水平位址
  298.                 }
  299.                 else
  300.                 {
  301.                         LCD12864_writeCmd(0x80+y+j);        
  302.                         LCD12864_writeCmd(0x80+x);
  303.                 }
  304.                 for(i=0;i<width;i++)                                //水平方向写数据,带反白控制
  305.                 {
  306.                         LCD12864_writeByte(bReverse?~pImageCode[imageWidth*j+i]:pImageCode[imageWidth*j+i]);        
  307.                 }        
  308.         }
  309.         LCD12864_writeCmd(0x30);
  310.         return 0x02;
  311. }


  312. void LCD12864_clearGDRAM()              //液晶可视区的绘图GDRAM清0
  313. {   
  314.     u8 j=0;
  315.     u8 i=0;
  316.     LCD12864_writeCmd(0x34);        //扩展指令
  317.     for(j=0;j<64;j++)                       //垂直方向地址手动增加,当j=64时清整个GDram
  318.     {                                                                //我们只要清可视区的GDRAM就可以了
  319.         LCD12864_writeCmd(0x80+j);  //y轴坐标
  320.         LCD12864_writeCmd(0x80);    //x轴坐标
  321.         for(i=0;i<32;i++)           //水平方向位址自动增加
  322.         {
  323.             LCD12864_writeByte(0x00);
  324.         }
  325.     }
  326.     LCD12864_writeCmd(0x30);  //回到基本指令
  327. }         


  328. /*--------------------------------CGRAM start----------------------------------------------*/
  329. void LCD12864_clearCGRAM(u8 CGRAM_groupNum)//将用户自定义编码区CGRAM清0         
  330. {        //参数一是CGRAM的4组用户空间组号码(0~3) ,参数二是用户自定义图表或汉字的编码
  331.     u8 i,addr=0;        
  332.         bShowPicFlag?LCD12864_writeCmd(0x36):LCD12864_writeCmd(0x34);//扩展指令,绘图开关保持                                                               
  333.     LCD12864_writeCmd(0x02);            //SR等于0,允许设置卷动地址
  334.     LCD12864_writeCmd(0x30);            //恢复为8位并行,基本指令集
  335.     addr=(CGRAM_groupNum<<4)|0x40;        //将CGRAM空间号转换为相应存储地址
  336.     LCD12864_writeCmd(addr);                //定位到该位址(用户空间位址范围0x40-0x7F共128字节)
  337.     for(i=0;i<16;i++)                                //将用户自定义编码写入该16*16位元组空间
  338.     {
  339.         LCD12864_writeByte(0);                //连续写2个字节共16位
  340.         LCD12864_writeByte(0);
  341.     }   
  342. }

  343. void LCD12864_writeScrollCGRAM(u8 CGRAM_groupNum, u8* pUserCode)//将用户自定义编码写入CGRAM         
  344. {        //参数一是CGRAM的4组用户空间组号码(0~3) ,参数二是用户自定义图表或汉字的编码
  345.     u8 i,addr=0;
  346.         if(bShowPicFlag)                                
  347.         {
  348.                 LCD12864_writeCmd(0x36);        //开启绘图显示,可流畅播放动画                 
  349.         }
  350.         else
  351.         {
  352.                 LCD12864_writeCmd(0x34);        //默认关闭绘图显示
  353.         }                                
  354.     LCD12864_writeCmd(0x02);            //SR等于0,允许设置卷动地址
  355.     LCD12864_writeCmd(0x30);            //恢复为8位并行,基本指令集
  356.     addr=(CGRAM_groupNum<<4)|0x40;        //将CGRAM空间号转换为相应存储地址
  357.     LCD12864_writeCmd(addr);                //定位到该位址(用户空间位址范围0x40-0x7F共128字节)
  358.     for(i=0;i<16;i++)                                //将用户自定义编码写入该16*16位元组空间
  359.     {
  360.         LCD12864_writeByte(pUserCode[i*2]);                 //连续写2个字节共16位
  361.         LCD12864_writeByte(pUserCode[i*2+1]);
  362.     }   
  363. }

  364. void LCD12864_writeCGRAM(u8 CGRAM_groupNum, u8* pUserCode)//将用户自定义编码写入CGRAM         
  365. {        //参数一是CGRAM的4组用户空间组号码(0~3) ,参数二是用户自定义图表或汉字的编码
  366.     u8 i,addr=0;
  367.         if(bShowPicFlag)                                
  368.         {
  369.                 LCD12864_writeCmd(0x36);        //开启绘图显示,可流畅播放动画                 
  370.         }
  371.         else
  372.         {
  373.                 LCD12864_writeCmd(0x34);        //默认关闭绘图显示
  374.         }                                
  375.     LCD12864_writeCmd(0x02);            //SR等于0,允许设置卷动地址
  376.     LCD12864_writeCmd(0x30);            //恢复为8位并行,基本指令集
  377.     addr=(CGRAM_groupNum<<4)|0x40;        //将CGRAM空间号转换为相应存储地址
  378.     LCD12864_writeCmd(addr);                //定位到该位址(用户空间位址范围0x40-0x7F共128字节)
  379.     for(i=0;i<16;i++)                                //将用户自定义编码写入该16*16位元组空间
  380.     {
  381.         LCD12864_writeByte(pUserCode[i*2]);                 //连续写2个字节共16位
  382.         LCD12864_writeByte(pUserCode[i*2+1]);
  383.     }   
  384. }

  385. void LCD12864_showScrollCGRAM(u8 row,u8 cols,u8 CGRAM_groupNum)//滚动CGRAM
  386. {        //row(0-3),        cols(0-15)
  387.         //第三个参数是用户空间号码(0~3共4组空间号码),该号码乘2就是它所对应的[调用用户空间编码]
  388.     LCD12864_setScrollPos(row,cols);
  389.     LCD12864_writeByte(0x00);//4组用户空间的编码的高字节都固定为0,我猜这是为和E文ASCII码区分开
  390.     LCD12864_writeByte(CGRAM_groupNum*2);        //对应编码00h,02h,04h,06h
  391.         LCD12864_showCGRAM(row,cols,CGRAM_groupNum);
  392. }

  393. void LCD12864_showCGRAM(u8 row,u8 cols,u8 CGRAM_groupNum)//定位液晶光标,并显示自定义内容
  394. {        //row(0-3),        cols(0-15)
  395.         //第三个参数是用户空间号码(0~3共4组空间号码),该号码乘2就是它所对应的[调用用户空间编码]
  396.     LCD12864_setPos(row,cols);
  397.     LCD12864_writeByte(0x00);//4组用户空间的编码的高字节都固定为0,我猜这是为和E文ASCII码区分开
  398.     LCD12864_writeByte(CGRAM_groupNum*2);        //对应编码00h,02h,04h,06h
  399. }
  400. /*--------------------------------CGRAM end----------------------------------------------*/

  401. /*--------------------------------DDRAM start----------------------------------------------*/
  402. void LCD12864_earseSomeDDRam(u8 row,u8 cols,u8 dataSize) //擦除N个字节DDRam
  403. {        //row(0-3),cols(0-15),如果起始地址是汉字的低字节,则会一同擦除汉字的高字节
  404.         LCD12864_setPos(row, cols);                                   //定位                                                           
  405.         if(cols%2!=0)                                                           //如果从奇数列开始
  406.         {
  407.                 LCD12864_readByte();                                   //空读一次,让位址指针移动半字
  408.                 if(LCD12864_readByte()>127)                           //检测高位是否是汉字码
  409.                 {
  410.                         LCD12864_setPos(row, cols);                   //是汉字码的话要擦除,不然要乱码
  411.                         LCD12864_writeByte(' ');                   //其实是写空格,看不到了就等于擦除了
  412.                 }                                                                           //连续写2个0的话是乱码
  413.         }        
  414.         while(dataSize--)                                                   //擦除
  415.         {
  416.                 if(cols)                                                           //cols>0
  417.                 {
  418.                         if(0==cols%16)                                           //如果一行满了
  419.                         {
  420.                                 row++;                                                   //准备将光标移到下一行
  421.                                 cols=0;                                                   //列坐标置于行首
  422.                                 LCD12864_setPos(row, cols);           //设置新光标
  423.                         }                                
  424.                 }
  425.                 LCD12864_writeByte(' ');                           //其实是写空格,但为啥不写0呢
  426.                 cols++;                                                                   //因为0与是CGRAM重码了,写2个0会乱码的
  427.         }
  428.         
  429. }


  430. /*****************************************************************************************
  431. pBuf如果用来修饰字符串,dataSize=strlen(pBuf);
  432. pBuf如果是一个字符数组,dataSize=sizeof(pBuf);
  433. strlen()虽然也可以用来计算字符数组长度,但遇到'\0'时就会返回,不会再计算后面的其他字符
  434. 在VC中strlen()只能计算字符串长度,不能计算字符数组,否则编译出错
  435. sizeof("你好5"),sizeof("你好56"),最终液晶光标位置是一样的,故不要用sizeof计算字符串
  436. *****************************************************************************************/
  437. void LCD12864_writeData(u8 row,u8 cols,u8* pBuf,u8 dataSize)//写数据
  438. {        //支持全角半角字符及中英混合的字符串,也可写入字符串的子串,(行坐标0~3,列坐标0~15)
  439.         u8 flag=0;                                                                        //液晶低字节ram数据检测标志,0不检测
  440.     LCD12864_setPos(row, cols);                                        //设置光标   
  441.         if(cols%2!=0)                                                                //列号不为偶数(汉字在液晶上要偶对齐)
  442.         {        //要让位址空移一个字节,执行下面这句读操作后,效果是达到了
  443.                 LCD12864_readByte();//但AC值没变,我怀疑除了这个AC字型指针,另有一个标志位没公开)
  444.                 flag=1;                                                                        //此时需要检测液晶低位字节ram        
  445.         }                                                                                        //因为高位字节现在可能是汉字码

  446.     while(dataSize--)                                                        //循环处理目标字节流
  447.     {                                                                                       
  448.                 if(0==cols%2)                                                        //偶对齐时,对于ram高字节
  449.                 {
  450.                         if(*pBuf>127)                                                //如果写入ram的高字节是汉字码
  451.                         {
  452.                                 flag=0;                                                        //下一次不要检测低字节,因为肯定是汉字了
  453.                         }
  454.                         else                                                                //如果高字节是半角字符
  455.                         {
  456.                                 flag=1;                                                        //若在低字节ram写入汉字就乱码了,故检测
  457.                         }                                
  458.                 }

  459.                 if(cols%2!=0)                                                    //对于液晶低字节
  460.                 {
  461.                         if(flag)                                                        //如果要检测低字节
  462.                         {
  463.                                 if(*pBuf>127)                                        //如果低字节是汉字码
  464.                                 {
  465.                                         LCD12864_writeByte(0x20);        //插入一个空格
  466.                                         cols++;                                                //字节计数器++
  467.                                         flag=0;                                                //清检测标志
  468.                                 }                                       
  469.                         }        
  470.                 }      
  471.                
  472.                 if(cols)                                                                //行尾检测
  473.                 {
  474.                         if(0==cols%16)                                                //如果一行满了
  475.                         {
  476.                                 row++;                                                        //准备将光标移到下一行
  477.                                 cols=0;                                                        //列坐标置于行首
  478.                                 LCD12864_setPos(row, cols);                //设置新位址
  479.                         }                                
  480.                 }

  481.                 LCD12864_writeByte(*pBuf++);                        //终于可以写数据了
  482.                 cols++;                                                                    //列号累加
  483.     }
  484. }


  485. //滚屏模式的写数据函数
  486. void LCD12864_writeScrollData(u8 row,u8 cols,u8* pBuf,u8 dataSize)//写数据,卷动模式
  487. {        //支持全角半角字符及中英混合的字符串,也可写入字符串的子串,(行坐标0~3,列坐标0~15)
  488.         u8 flag=0;                                                                        //液晶低字节ram数据检测标志,0不检测
  489.         LCD12864_writeData(row,cols,pBuf,dataSize);
  490.     LCD12864_setScrollPos(row, cols);                                        //设置光标   
  491.         if(cols%2!=0)                                                                //列号不为偶数(汉字在液晶上要偶对齐)
  492.         {        //要让位址空移一个字节,执行下面这句读操作后,效果是达到了
  493.                 LCD12864_readByte();//但AC值没变,我怀疑除了这个AC字型指针,另有一个标志位没公开)
  494.                 flag=1;                                                                        //此时需要检测液晶低位字节ram        
  495.         }                                                                                        //因为高位字节现在可能是汉字码

  496.     while(dataSize--)                                                        //循环处理目标字节流
  497.     {                                                                                       
  498.                 if(0==cols%2)                                                        //偶对齐时,对于ram高字节
  499.                 {
  500.                         if(*pBuf>127)                                                //如果写入ram的高字节是汉字码
  501.                         {
  502.                                 flag=0;                                                        //下一次不要检测低字节,因为肯定是汉字了
  503.                         }
  504.                         else                                                                //如果高字节是半角字符
  505.                         {
  506.                                 flag=1;                                                        //若在低字节ram写入汉字就乱码了,故检测
  507.                         }                                
  508.                 }

  509.                 if(cols%2!=0)                                                    //对于液晶低字节
  510.                 {
  511.                         if(flag)                                                        //如果要检测低字节
  512.                         {
  513.                                 if(*pBuf>127)                                        //如果低字节是汉字码
  514.                                 {
  515.                                         LCD12864_writeByte(0x20);        //插入一个空格
  516.                                         cols++;                                                //字节计数器++
  517.                                         flag=0;                                                //清检测标志
  518.                                 }                                       
  519.                         }        
  520.                 }      
  521.                
  522.                 if(cols)                                                                //行尾检测
  523.                 {
  524.                         if(0==cols%16)                                                //如果一行满了
  525.                         {
  526.                                 row++;                                                        //准备将光标移到下一行
  527.                                 cols=0;                                                        //列坐标置于行首
  528.                                 LCD12864_setScrollPos(row, cols);                //设置新位址
  529.                         }                                
  530.                 }

  531.                 LCD12864_writeByte(*pBuf++);                        //终于可以写数据了
  532.                 cols++;                                                                    //列号累加
  533.     }
  534. }

  535. void  LCD12864_startScroll(u8 scrollNum,u16 delay_ms)  //滚动
  536. {        //scrollNum建议值为64
  537.         u8 i;
  538.         LCD12864_writeCmd(0x34);         
  539.         LCD12864_writeCmd(0x03);
  540.         for(i=0x40;i<0x40+scrollNum;i++)
  541.         {  
  542.                 LCD12864_writeCmd(i); //设置卷动地址
  543.                 delayms(delay_ms);                    //实际使用时建议用定时器处理
  544.         }
  545.         LCD12864_writeCmd(0x40);  //补滚一行
  546.         LCD12864_writeCmd(0x30);
  547. }

  548. void LCD12864_setScrollPos(u8 row, u8 cols)                //设置光标位置 ,卷动模式
  549. {                                                                                                //row行坐标0~3
  550.         u8 newPos=0;                                                                //cols列坐标0~15
  551.         switch(row)
  552.         {
  553.                 case 0:
  554.                 {
  555.                         row=0xa8;        
  556.                 }
  557.                 break;
  558.                 case 1:
  559.                 {
  560.                         row=0xb8;
  561.                 }
  562.                 break;
  563.                 case 2:
  564.                 {
  565.                         row=0xa0;
  566.                 }
  567.                 break;
  568.                 case 3:
  569.                 {
  570.                         row=0xb0;
  571.                 }
  572.                 break;
  573.                 default:                //如果需要检测行坐标范围,可在这里加代码
  574.                 break;                        
  575.         }
  576.         newPos=row+cols/2;        //液晶写指令坐标只能8级,我的函数中列坐标是16级的,支持半角全角混合
  577.         LCD12864_writeCmd(newPos);
  578. }


  579. void LCD12864_setPos(u8 row, u8 cols)                        //设置光标位置
  580. {                                                                                                //row行坐标0~3
  581.         u8 newPos=0;                                                                //cols列坐标0~15
  582.         switch(row)
  583.         {
  584.                 case 0:
  585.                 {
  586.                         row=0x80;        
  587.                 }
  588.                 break;
  589.                 case 1:
  590.                 {
  591.                         row=0x90;
  592.                 }
  593.                 break;
  594.                 case 2:
  595.                 {
  596.                         row=0x88;
  597.                 }
  598.                 break;
  599.                 case 3:
  600.                 {
  601.                         row=0x98;
  602.                 }
  603.                 break;
  604.                 case 4:
  605.                 {
  606.                         row=0xa0;
  607.                 }
  608.                 break;
  609.                 case 5:
  610.                 {
  611.                         row=0xb0;
  612.                 }
  613.                 break;
  614.                 case 6:
  615.                 {
  616.                         row=0xa8;
  617.                 }
  618.                 break;
  619.                 case 7:
  620.                 {
  621.                         row=0xb8;
  622.                 }
  623.                 break;

  624.                 default:                //如果需要检测行坐标范围,可在这里加代码
  625.                 break;                        
  626.         }
  627.         newPos=row+cols/2;        //液晶写指令坐标只能8级,我的函数中列坐标是16级的,支持半角全角混合
  628.         LCD12864_writeCmd(newPos);
  629. }
  630. /*
  631. void LCD12864_setPos(u8 row, u8 cols)                        //设置光标位置
  632. {                                                                                                //row行坐标0~3
  633.         u8 newPos=0;                                                                //cols列坐标0~15
  634.         switch(row)
  635.         {
  636.                 case 0:
  637.                 {
  638.                         row=0x80;        
  639.                 }
  640.                 break;
  641.                 case 1:
  642.                 {
  643.                         row=0x90;
  644.                 }
  645.                 break;
  646.                 case 2:
  647.                 {
  648.                         row=0x88;
  649.                 }
  650.                 break;
  651.                 case 3:
  652.                 {
  653.                         row=0x98;
  654.                 }
  655.                 break;
  656.                 default:                //如果需要检测行坐标范围,可在这里加代码
  657.                 break;                        
  658.         }
  659.         newPos=row+cols/2;        //液晶写指令坐标只能8级,我的函数中列坐标是16级的,支持半角全角混合
  660.         LCD12864_writeCmd(newPos);
  661. }
  662.          */



  663. void LCD12864_init()                                  //初始化
  664. {
  665.         delayms(40);                                          //rst由低到高后保持40ms以上,我们的rst接VCC
  666.         LCD_PSB= 1;                                                    //选择并口方式

  667.         LCD12864_writeCmd( B(110000) );          //0x30,启用基本指令集
  668.         delayXus(15);                                          //要求延时100us以上,(8+6x)*1.085=106us

  669.         LCD12864_writeCmd( B(110000) );   //0x30,要求写2次该指令
  670.         delayXus(5);                                           //要求延时37us以上,(8+6x)*1.085=41us

  671.     LCD12864_writeCmd( B(1100) );          //0x0f,整体显示,游标,游标反白
  672.         delayXus(15);                                          //要求延时100us以上

  673.         LCD12864_writeCmd( B(0001) );          //0x01,清屏指令,整屏幕写满空格
  674.         delayms(10);                                           //要求延时10ms以上

  675.         LCD12864_writeCmd( B(110) );          //0x06,进入模式设置,游标自动指向下一位置,
  676.                                                 
  677.         delayms(5);                                                    //手册上没说这里要延时,额,还是加上吧
  678. }


  679. void LCD12864_writeCmd(u8 cmd)                    //写指令
  680. {
  681.         while(LCD12864_isBusy());
  682.         LCD_EN=0;                                                   //使能 拉低
  683.         LCD_RW=0;                                                   //写
  684.         LCD_RS=0;_nop_();                                  //命令
  685.                
  686.         LCD_EN=1;                                                  //使能
  687.         LCD_dataBus=cmd;                                  //送指令
  688.         _nop_();_nop_();                              //稳定

  689.         LCD_EN=0;_nop_();                           //取走
  690. }

  691. void LCD12864_writeByte(u8 dat)                     //写一个字节
  692. {
  693.         while(LCD12864_isBusy());
  694.         LCD_EN=0;                                 //使能先拉低
  695.         LCD_RW=0;                                 //写
  696.         LCD_RS=1;_nop_();                 //数据
  697.         
  698.         LCD_EN=1;
  699.         LCD_dataBus=dat;
  700.         _nop_();_nop_();                 //延时大于1.5us

  701.         LCD_EN=0;_nop_();                 //下降沿取走数据
  702. }

  703. u8 LCD12864_readByte()                  //读数据暂存器Data Register
  704. {                                                         //用的时候要空操作一次
  705.         u8 temp=0;
  706.         while(LCD12864_isBusy());//忙检测
  707.         LCD_dataBus=0xff;                 //用总线读数据时必须先置为输入模式
  708.         LCD_EN=0;                                 //使能线拉低
  709.         LCD_RW=1;                                 //读
  710.         LCD_RS=1;_nop_();                 //数据                                         
  711.         LCD_EN=1;_nop_();                 //使能
  712.         temp=LCD_dataBus;                 //取走数据
  713.         
  714.         _nop_();
  715.         LCD_EN=0;                                 //使能恢复
  716.         return temp;
  717. }

  718. bool LCD12864_isBusy()                   //检测液晶是否忙
  719. {
  720.         if(LCD12864_readIR() & 0x80) //检测BF位
  721.         {
  722.                 return TRUE;                  //忙
  723.         }
  724.         return FALSE;                          //不忙
  725. }

  726. u8 LCD12864_readIR()                 //读指令暂存器Instruction Register
  727. {
  728.         u8 temp=0;
  729.         LCD_EN=0;                                 //使能准备
  730.         LCD_RW=1;                                 //读
  731.         LCD_RS=0;_nop_();                 //命令字
  732.         LCD_dataBus=0xff;               //准备输入
  733.         LCD_EN=1;_nop_();             //使能
  734.         temp=LCD_dataBus;                 //提取数据

  735.         _nop_();
  736.         LCD_EN=0;                                 //使能拉低
  737.         return temp;                  
  738. }

复制代码




分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:137736 发表于 2022-8-30 12:49 | 只看该作者
太好了,生成了头文件,使用很方便,感谢
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表