实现的滚屏效果是当一屏写满时,则清空从开头接着写,不是上移的滚屏,虽然OLED有滚屏命令,但是会带水平位移效果,并且只能提前写好数据,类似于广告牌循环播放的那种.

  首先是为OLED屏划分区域. 我选择显示的字体是6*8大小的,这样可以显示多点内容,(这是最小的尺寸了,我试过4*8,太畸形了) 所以一块屏最多显示大约168个字,然后每行可显示21,共8行

用两个坐标来定位就行.显示的时候遇到回车换行也需要跳到下一行显示


#define DIS_ONE_PAGE_BUF 168 //一页的字数
#define DIS_BUFFER DIS_ONE_PAGE_BUF*8 //一页的字数 size()6*8


void OLED_Screen_Show(void){
unsigned int i,tmp,len;
len=Cal_Recv_Len(OLED_Assistant.OLED_ShowBuf_Last_DLen,OLED_Assistant.OLED_ShowBuf_DLen);
for(i=0;i<len;i++){
tmp=OLED_Screen_Buff[OLED_Assistant.OLED_ShowBuf_RPtr];
if(tmp=='\r'){//换行
OLED_Assistant.OLED_ShowBuf_RPtr+=2;
tmp=Cal_Line_Remain(OLED_Assistant.OLED_Screen_WPtr)+1;
OLED_Assistant.OLED_Screen_WPtr+=tmp;
i+=2;
continue;
}
if(OLED_Assistant.OLED_Screen_WPtr>=DIS_ONE_PAGE_BUF){
OLED_Assistant.OLED_Flush_EN=1;//屏幕写满则刷新
OLED_Assistant.OLED_Screen_WPtr=0;
} if(OLED_Assistant.OLED_Flush_EN){
OLED_Clear();
OLED_Assistant.OLED_Flush_EN=0;
}
Cal_Screen_Axis(OLED_Assistant.OLED_Screen_WPtr,&OLED_Assistant.OLED_X,&OLED_Assistant.OLED_Y);//计算坐标
OLED_ShowChar_6x8_Pos(OLED_Assistant.OLED_X,OLED_Assistant.OLED_Y,tmp);
OLED_Assistant.OLED_Screen_WPtr++;
OLED_Assistant.OLED_ShowBuf_RPtr++;
if(OLED_Assistant.OLED_ShowBuf_RPtr>=DIS_BUFFER){
OLED_Assistant.OLED_ShowBuf_RPtr=0;
}
} }

  第二个就是显存大小了,视情况而定,我是在串口空闲中断中把DMA接收的数据复制到显存中的,复制的时候通过显存已有长度来接着存储,当达到最大时,直接置0就行。因为前面的数据都显示过了,覆盖就覆盖了。

void USART2_IRQHandler()
{
if(USART_GetITStatus(USART2, USART_IT_IDLE) != RESET) //接收中断
{
//根据应用来定接收长度,每一次空闲中断,则可以通过DMA_GetCurrDataCounter拿到本次接收数据长度
USART_ReceiveData(USART2);
DMA_Cmd(DMA1_Channel6,DISABLE);/* 关闭接收DMA */
DMA_ClearFlag(DMA1_FLAG_TC6);/* 清除标志位 */
/* 重新设置传输数据长度, 要比真实数据长,防止覆盖*/
udma.Recv_Len=RECV_BUF_SIZE-DMA_GetCurrDataCounter(DMA1_Channel6);
// MemClear(udma.Send_Buffer,SEND_BUF_SIZE);
// MemCopy(udma.Recv_Buffer,udma.Send_Buffer,udma.Recv_Len);
OLED_Assistant.OLED_ShowBuf_Last_DLen=OLED_Assistant.OLED_ShowBuf_DLen;
Copy_To_ShowBuff(udma);
OLED_Assistant.OLED_Write_EN=1;
DMA_SetCurrDataCounter(DMA1_Channel6,RECV_BUF_SIZE);
/* 打开DMA */
DMA_Cmd(DMA1_Channel6,ENABLE);
// DMA_Send_EN=1;
} }

  第三个就是对于数据的处理了,关键是要能不定长的接收,并且能不定长的显示,然后什么时候刷新,什么显示接收到的内容,就需要几个控制的变量了

typedef struct{
/*
OLED显存指针:用于定位显存读取位置//此指针非彼指针,只是比喻,下同
OLED显存数据长度:用于数据存储(到达最大值,变为0,直接覆盖)
OLED显存上一次数据长度:用于计算接收的数据长度
OLED屏幕位置指针:用于定位当前显示位置
OLED屏幕位置X坐标 COLUMN
OLED屏幕位置Y坐标 ROW
OLED刷新使能标识
OLED显示内容使能标识
*/
unsigned int OLED_ShowBuf_RPtr;
unsigned int OLED_ShowBuf_DLen;
unsigned int OLED_ShowBuf_Last_DLen;
unsigned int OLED_Screen_WPtr;
unsigned char OLED_X;//column (0-20)
unsigned char OLED_Y;//row (0-7)
unsigned char OLED_Flush_EN;
unsigned char OLED_Write_EN;
}OLED_Show_Handler; extern OLED_Show_Handler OLED_Assistant;//初始化全为0就可
void Copy_To_ShowBuff(U_DMA udma){//复制DMA接收的串口数据,在空闲中断中调用
unsigned int i;
for(i=0;i<udma.Recv_Len;i++){
OLED_Screen_Buff[OLED_Assistant.OLED_ShowBuf_DLen]=udma.Recv_Buffer[i];//每次接收多少数据就写入多少数据
OLED_Assistant.OLED_ShowBuf_DLen++;
if(OLED_Assistant.OLED_ShowBuf_DLen==DIS_BUFFER){
OLED_Assistant.OLED_ShowBuf_DLen=0; //重新从头开始覆盖写入,因为最前面的内容显示过了,可以覆盖
}
} } unsigned char Cal_Line_Remain(unsigned int arg){//计算不足一行的数据的长度,用于换行时增加屏幕指针偏移的
unsigned char remain;
remain=arg-(unsigned char)(arg/20)*20;
remain=20-remain;
return remain;
}
void Cal_Screen_Axis(unsigned int wptr,unsigned char *x,unsigned char *y){//计算需要显示的坐标
*y=(unsigned char)(wptr/20);
*x=wptr-(*y)*20;
} unsigned int Cal_Recv_Len(unsigned int last_len,unsigned int new_len){//计算接收到的数据长度,用于不定长显示
unsigned int recv_len;
if(new_len>=last_len){ //防止循环接收时的接收指针跑到上一次接收指针前面
recv_len=new_len-last_len;
}
else{
recv_len=(DIS_BUFFER-last_len)+recv_len;
}
return recv_len;
}

最新文章

  1. 调戏OpenShift:一个免费能干的云平台
  2. 交换机的交换原理、mac学习机制和老化机制
  3. JDK源码解读之toUnsignedString
  4. Linux网络编程-SIGPIPE信号导致的程序退出问题
  5. [mysql] timestamp自动更新和初始化
  6. CentOS 6.5上MySQL安装部署与入门。
  7. Intrusion Analysis Learning
  8. iOS 设置button中图文位置
  9. thinking in java之Collections工具类的使用
  10. API网关
  11. 转-[Python 学习]2.5版yield之学习心得
  12. Poj 3667
  13. 关于FileSystemXmlApplicationContext和ClassPathXmlApplicationContext路径问题
  14. Python3——MP3播放器
  15. 解决flask的502错误:upstream prematurely closed connection while reading response header from upstream
  16. RequestHolder工具类
  17. 使用python语言计算n的阶乘
  18. angular2 bootstrap modal
  19. HDU 4763 Theme Section (2013长春网络赛1005,KMP)
  20. 使用Windows防火墙禁止Ping入

热门文章

  1. C#代码整洁之道读后总结与感想
  2. django框架之drf(部分讲解)
  3. 9月28日——while循环知识补充、for循环、break及continue生效范围和range方法的介绍
  4. 洛谷 P2629 好消息,坏消息(单调队列)
  5. Django框架之drf:9、接口文档,coreapi的使用,JWT原理、介绍、快速使用、定制、认证
  6. Vue29 $nextTick
  7. 亲测有效! Wondershare UniConverterV14.1.7 Wondershare PDFelement Professional V9.3.3 含(win/mac版)
  8. fast planner总结
  9. 冰河指南AI技术社区基于ChatGPT正式启动运营
  10. 加密,各种加密,耙梳加密算法(Encryption)种类以及开发场景中的运用(Python3.10)