STM32 F103C8T6学习笔记16:1.3寸OLED的驱动显示日历

2024-05-02 06:28

本文主要是介绍STM32 F103C8T6学习笔记16:1.3寸OLED的驱动显示日历,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天尝试使用STM32 F103C8T6驱动显示 1.3寸的OLED,显示数字、字符串、汉字、图片等

本质与0.96寸的OLED是完全相同的原理:

而且经过我的研究发现: 1.3寸大小的OLED并未比0.96寸的有更多的显示像素点数来显示,也是128*64的像素点数显示:

也是8页(也可以称为8个水平扫描线)和128列。通过8页和128列,可以操作所有64x128个像素点。可能只是每个像素点大小变大了些吧......

//OLED的显存
//存放格式如下.
//[0]0 1 2 3 ... 127	
//[1]0 1 2 3 ... 127	
//[2]0 1 2 3 ... 127	
//[3]0 1 2 3 ... 127	
//[4]0 1 2 3 ... 127	
//[5]0 1 2 3 ... 127	
//[6]0 1 2 3 ... 127	
//[7]0 1 2 3 ... 127 			

但代码还是略有区别的,如果对驱动原理、SSD1306驱动芯片不太了解的,下面也是贴出网址链接来一起研究学习了。

本文就不将基础知识放入了,主要针对0.96寸与1.3寸的OLED驱动代码有哪些不同进行记录研究.

文章提供测试代码讲解、完整工程下载、测试效果图

目录

自己的相关文章笔记推荐:

驱动0.96寸与1.3寸OLED相同的函数:

IIC通信函数:

OLED写操作函数:

开启关闭OLED函数:

清屏函数:

 初始化SSD1306函数:

显示字符函数:

m^n函数:

 显示俩个数字函数:

 显示一个字符号串函数:

显示标准大小汉字:

显示小号汉字:

显示BMP图片:

 fill_Picture:

驱动0.96寸与1.3寸OLED不相同的函数:

坐标设置函数:

OLED打印随时会改变的变量的方法 :

(sprintf函数OLED打印方法)

测试工程下载:

测试视频:


自己的相关文章笔记推荐:

STM32 F103C8T6学习笔记8:0.96寸单色OLED显示屏显示字符_stm32f103c8t6 oled-CSDN博客

STM32 F103C8T6学习笔记9:0.96寸单色OLED显示屏—自由取模显示—显示汉字与图片_stm32f103c8t6 oled显示文字-CSDN博客

STM32 F103C8T6学习笔记11:RTC实时时钟—OLED手表日历_stm32f103c8t6显示实时时间-CSDN博客

驱动0.96寸与1.3寸OLED相同的函数:

IIC通信函数:

这部分的函数是完全一致的,毕竟都是用的IIC通信,就不会有什么不同了:

以下就简单的贴个代码,不做多余解释:

//-----------------OLED IIC端口定义----------------  					   #define OLED_SCLK_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_8)//SCL
#define OLED_SCLK_Set() GPIO_SetBits(GPIOB,GPIO_Pin_8)#define OLED_SDIN_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_9)//SDA
#define OLED_SDIN_Set() GPIO_SetBits(GPIOB,GPIO_Pin_9)#define OLED_CMD  0	//写命令
#define OLED_DATA 1	//写数据
/**********************************************
//IIC Start
**********************************************/
void IIC_Start(void)
{OLED_SCLK_Set() ;OLED_SDIN_Set();OLED_SDIN_Clr();OLED_SCLK_Clr();
}/**********************************************
//IIC Stop
**********************************************/
void IIC_Stop(void)
{
OLED_SCLK_Set() ;
//	OLED_SCLK_Clr();OLED_SDIN_Clr();OLED_SDIN_Set();}void IIC_Wait_Ack(void)
{//GPIOB->CRH &= 0XFFF0FFFF;	//设置PB12为上拉输入模式//GPIOB->CRH |= 0x00080000;
//	OLED_SDA = 1;
//	delay_us(1);//OLED_SCL = 1;//delay_us(50000);
/*	while(1){if(!OLED_SDA)				//判断是否接收到OLED 应答信号{//GPIOB->CRH &= 0XFFF0FFFF;	//设置PB12为通用推免输出模式//GPIOB->CRH |= 0x00030000;return;}}
*/OLED_SCLK_Set() ;OLED_SCLK_Clr();
}
/**********************************************
// IIC Write byte
**********************************************/void Write_IIC_Byte(unsigned char IIC_Byte)
{unsigned char i;unsigned char m,da;da=IIC_Byte;OLED_SCLK_Clr();for(i=0;i<8;i++)		{m=da;//	OLED_SCLK_Clr();m=m&0x80;if(m==0x80){OLED_SDIN_Set();}else OLED_SDIN_Clr();da=da<<1;OLED_SCLK_Set();OLED_SCLK_Clr();}}
/**********************************************
// IIC Write Command
**********************************************/
void Write_IIC_Command(unsigned char IIC_Command)
{IIC_Start();Write_IIC_Byte(0x78);            //Slave address,SA0=0IIC_Wait_Ack();	Write_IIC_Byte(0x00);			//write commandIIC_Wait_Ack();	Write_IIC_Byte(IIC_Command); IIC_Wait_Ack();	IIC_Stop();
}
/**********************************************
// IIC Write Data
**********************************************/
void Write_IIC_Data(unsigned char IIC_Data)
{IIC_Start();Write_IIC_Byte(0x78);			//D/C#=0; R/W#=0IIC_Wait_Ack();	Write_IIC_Byte(0x40);			//write dataIIC_Wait_Ack();	Write_IIC_Byte(IIC_Data);IIC_Wait_Ack();	IIC_Stop();
}

OLED写操作函数:

 这部分也是完全一致,这是对OLED进行命令操作以及写入控制屏幕最基础的函数了:

//起到了有命令写命令,有数据写数据的效果:
void OLED_WR_Byte(unsigned dat,unsigned cmd)
{if(cmd){Write_IIC_Data(dat);}else {Write_IIC_Command(dat);}
}

开启关闭OLED函数:

 通过写命令开启关闭OLED显示,这与驱动芯片SSD1306相关,因此命令相同,无需改动:

//开启OLED显示    
void OLED_Display_On(void)
{OLED_WR_Byte(0X8D,OLED_CMD);  //SET DCDC命令OLED_WR_Byte(0X14,OLED_CMD);  //DCDC ONOLED_WR_Byte(0XAF,OLED_CMD);  //DISPLAY ON
}
//关闭OLED显示     
void OLED_Display_Off(void)
{OLED_WR_Byte(0X8D,OLED_CMD);  //SET DCDC命令OLED_WR_Byte(0X10,OLED_CMD);  //DCDC OFFOLED_WR_Byte(0XAE,OLED_CMD);  //DISPLAY OFF
}

清屏函数:

意味着它们的硬件特性和像素组织方式在清屏操作上是相似的 :命令相同,无需改动

页面和列结构:这两个尺寸的OLED屏幕可能都采用了类似的页面和列结构来组织像素命令和数据格式:    两个OLED屏幕可能使用相同的命令和数据格式来设置像素点的状态。

但清屏函数相同并不意味着两个OLED屏幕在所有方面都是相同

//清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!	  
void OLED_Clear(void)  
{  u8 i,n;		    for(i=0;i<8;i++)  {  OLED_WR_Byte (0xb0+i,OLED_CMD);    //设置页地址(0~7)OLED_WR_Byte (0x02,OLED_CMD);      //设置显示位置—列低地址OLED_WR_Byte (0x10,OLED_CMD);      //设置显示位置—列高地址   for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA); } //更新显示
}

 初始化SSD1306函数:

//初始化SSD1306					    
void OLED_Init(void)
{ 	GPIO_InitTypeDef  GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);	 //使能A端口时钟GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8|GPIO_Pin_9;	 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 //推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHzGPIO_Init(GPIOB, &GPIO_InitStructure);	  //初始化GPIOD3,6GPIO_SetBits(GPIOB,GPIO_Pin_8|GPIO_Pin_9);	delay_ms(800);OLED_WR_Byte(0xAE,OLED_CMD);//--display off                    关闭显示屏OLED_WR_Byte(0x00,OLED_CMD);//---set low column address        设置低列地址 OLED_WR_Byte(0x10,OLED_CMD);//---set high column address       设置高列地址OLED_WR_Byte(0x40,OLED_CMD);//--set start line address         设置起始行地址OLED_WR_Byte(0xB0,OLED_CMD);//--set page address               设置页地址OLED_WR_Byte(0x81,OLED_CMD); // contract control               设置对比度控制OLED_WR_Byte(0xFF,OLED_CMD);//--128                            对比度值设置为128OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap                设置段重映射OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse               设置正常显示/反向显示OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)   设置多路复用比率(1到64)OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty                      设置Duty值为1/32OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction               设置COM扫描方向OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset              设置显示偏移OLED_WR_Byte(0x00,OLED_CMD);//                                 设置显示起始行OLED_WR_Byte(0xD5,OLED_CMD);//set osc division                 设置振荡器分频OLED_WR_Byte(0x80,OLED_CMD);//                                 设置振荡器分频值为80OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off          设置区域颜色模式关闭OLED_WR_Byte(0x05,OLED_CMD);//                                 设置区域颜色模式值为05OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period            设置预充电周期OLED_WR_Byte(0xF1,OLED_CMD);//                                 设置预充电周期值为F1OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion        设置COM引脚配置OLED_WR_Byte(0x12,OLED_CMD);//                                 设置COM引脚配置值为12OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh                        设置VcomhOLED_WR_Byte(0x30,OLED_CMD);//                                 设置Vcomh值为30OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable           设置充电泵使能OLED_WR_Byte(0x14,OLED_CMD);//                                 设置充电泵使能值为14OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel             打开显示屏l
}  

显示字符函数:

//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//mode:0,反白显示;1,正常显示				 
//size:选择字体 16/12 
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 Char_Size)
{      	unsigned char c=0,i=0;	c=chr-' ';//得到偏移后的值			if(x>Max_Column-1){x=0;y=y+2;}if(Char_Size ==16){OLED_Set_Pos(x,y);	for(i=0;i<8;i++)OLED_WR_Byte(F8X16[c*16+i],OLED_DATA);OLED_Set_Pos(x,y+1);for(i=0;i<8;i++)OLED_WR_Byte(F8X16[c*16+i+8],OLED_DATA);}else {	OLED_Set_Pos(x,y);for(i=0;i<6;i++)OLED_WR_Byte(F6x8[c][i],OLED_DATA);}
}

m^n函数:

//m^n函数
u32 oled_pow(u8 m,u8 n)
{u32 result=1;	 while(n--)result*=m;    return result;
}				

 显示俩个数字函数:

//显示2个数字
//x,y :起点坐标	 
//len :数字的位数
//size:字体大小
//mode:模式	0,填充模式;1,叠加模式
//num:数值(0~4294967295);	 		  
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size2)
{         	u8 t,temp;u8 enshow=0;						   for(t=0;t<len;t++){temp=(num/oled_pow(10,len-t-1))%10;if(enshow==0&&t<(len-1)){if(temp==0){OLED_ShowChar(x+(size2/2)*t,y,' ',size2);continue;}else enshow=1; }OLED_ShowChar(x+(size2/2)*t,y,temp+'0',size2); }
} 

 显示一个字符号串函数:

//显示一个字符号串
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 Char_Size)
{unsigned char j=0;while (chr[j]!='\0'){		OLED_ShowChar(x,y,chr[j],Char_Size);x+=8;if(x>120){x=0;y+=2;}j++;}
}

显示标准大小汉字:

这个函数需要自己在取模软件取出16*16像素点大小的汉字模(通过函数中循环次数可知)

每个汉字高度会占用二列(一共八列)


//显示汉字
void OLED_ShowCHinese(u8 x,u8 y,u8 no)
{      			    u8 t,adder=0;OLED_Set_Pos(x,y);	for(t=0;t<16;t++){OLED_WR_Byte(Hzk[2*no][t],OLED_DATA);adder+=1;}	OLED_Set_Pos(x,y+1);	for(t=0;t<16;t++){	OLED_WR_Byte(Hzk[2*no+1][t],OLED_DATA);adder+=1;}					
}

效果如图中的 年、星期:

显示小号汉字:

这个是我自己杜撰的一个可以显示小一号汉字的函数:

这个函数需要自己在取模软件取出12*8像素点大小的汉字模(通过函数中循环次数可知)

每个汉字高度会占用一列(一共八列)


void OLED_ShowCHinese_small(u8 x,u8 y,u8 no)
{u8 t,adder=0;OLED_Set_Pos(x,y);	for(t=0;t<12;t++){OLED_WR_Byte(Hzk_m[no][t],OLED_DATA);adder+=1;}	
}

 效果如图中的 月、日:

显示BMP图片:

/***********功能描述:显示显示BMP图片128×64起始点坐标(x,y),x的范围0~127,y为页的范围0~7*****************/
//x0 传入图像显示起始点x坐标
//y0 传入图像显示起始点y坐标
//x1 传入图像x横向像素个数 + x0
//y1 传入图像y纵向像素个数 占用的页数 
//BMP[] 传入图像数组        
void OLED_DrawBMP(unsigned char x0, unsigned char y0,unsigned char x1, unsigned char y1,unsigned char BMP[])
{ 	unsigned int j=0;unsigned char x,y;if(y1%8==0) y=y1/8;else y=y1/8+1;for(y=y0;y<y1;y++){OLED_Set_Pos(x0,y);for(x=x0;x<x1;x++){      OLED_WR_Byte(BMP[j++],OLED_DATA);	}}
}

 fill_Picture:

/********************************************
// fill_Picture
********************************************/
void fill_picture(unsigned char fill_Data)
{unsigned char m,n;for(m=0;m<8;m++){OLED_WR_Byte(0xb0+m,0);		//page0-page1OLED_WR_Byte(0x02,0);		//low column start addressOLED_WR_Byte(0x10,0);		//high column start addressfor(n=0;n<128;n++){OLED_WR_Byte(fill_Data,1);}}
}

驱动0.96寸与1.3寸OLED不相同的函数:

坐标设置函数:

两个OLED屏幕的坐标设置函数之所以不同,主要是因为它们的硬件特性和寻址模式不同。在编写驱动程序时,需要根据OLED屏幕的数据手册或者技术规格来确定正确的寻址方式和起始偏移。对于不同的屏幕,即使它们的尺寸不同,也可能需要使用类似的函数结构,但内部的计算或偏移值可能不同。 

 在这个函数中,x坐标的值在发送列地址之前被增加了2。这可能是因为1.3寸OLED屏幕有某种特定的寻址方式或者起始偏移,所以需要调整x坐标的起始值。

//1.3寸OLED 坐标设置void OLED_Set_Pos(unsigned char x, unsigned char y) 
{ OLED_WR_Byte(0xb0+y,OLED_CMD);OLED_WR_Byte((((x+2)&0xf0)>>4)|0x10,OLED_CMD);OLED_WR_Byte(((x+2)&0x0f),OLED_CMD); 
}     	

 这个函数在发送列地址之前没有调整x坐标的值。这可能是因为0.96寸OLED屏幕没有那样的起始偏移或者它的寻址模式与1.3寸的不同。

//0.96寸OLED 坐标设置
void OLED_Set_Pos(unsigned char x, unsigned char y) 
{ OLED_WR_Byte(0xb0+y,OLED_CMD);OLED_WR_Byte(((x&0xf0)>>4)|0x10,OLED_CMD);OLED_WR_Byte((x&0x0f),OLED_CMD); 
}   

OLED打印随时会改变的变量的方法 :

(sprintf函数OLED打印方法)

首先需要一个字符数组承接拼接处理的一串字符:

	   char  buf[20];   //用于暂存oled数据

使用sprintf函数将tm结构体中的年份(tm_year)转换为字符串,并保存到buf数组中。tm是一个指向struct tm的指针,这个结构体用于表示日期和时间

		//打印年sprintf(buf,"%d",tm->tm_year);OLED_ShowString(75,0,(u8 *)buf,16);OLED_ShowCHinese(75+16*2,0,0);  //打印中文“年”//打印时间:sprintf(buf,"%02d:%02d:%02d",tm->tm_hour,tm->tm_min,tm->tm_sec);		OLED_ShowString(64,2,(u8 *)buf,16);		          //打印日期:sprintf(buf,"%02d",tm->tm_mon);	   OLED_ShowString(75,4,(u8 *)buf,12);	//打印月	OLED_ShowCHinese_small(75+14,4,0);  //打印中文月	    sprintf(buf,"%02d",tm->tm_mday);		 OLED_ShowString(75+14+12,4,(u8 *)buf,12);	//打印日OLED_ShowCHinese_small(75+14+12+14,4,1);  //打印中文日

测试工程下载:

https://download.csdn.net/download/qq_64257614/89213223?spm=1001.2014.3001.5503

测试视频:

1.3寸OLED的驱动显示

这篇关于STM32 F103C8T6学习笔记16:1.3寸OLED的驱动显示日历的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/953579

相关文章

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

如何设置vim永久显示行号

《如何设置vim永久显示行号》在Linux环境下,vim默认不显示行号,这在程序编译出错时定位错误语句非常不便,通过修改vim配置文件vimrc,可以在每次打开vim时永久显示行号... 目录设置vim永久显示行号1.临时显示行号2.永www.chinasem.cn久显示行号总结设置vim永久显示行号在li

电脑显示hdmi无信号怎么办? 电脑显示器无信号的终极解决指南

《电脑显示hdmi无信号怎么办?电脑显示器无信号的终极解决指南》HDMI无信号的问题却让人头疼不已,遇到这种情况该怎么办?针对这种情况,我们可以采取一系列步骤来逐一排查并解决问题,以下是详细的方法... 无论你是试图为笔记本电脑设置多个显示器还是使用外部显示器,都可能会弹出“无HDMI信号”错误。此消息可能

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h