今天尝试使用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驱动代码有哪些不同进行记录研究.
文章提供测试代码讲解、完整工程下载、测试效果图
目录
自己的相关文章笔记推荐:
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=0
IIC_Wait_Ack();
Write_IIC_Byte(0x00); //write command
IIC_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#=0
IIC_Wait_Ack();
Write_IIC_Byte(0x40); //write data
IIC_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 ON
OLED_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 OFF
OLED_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;//速度50MHz
GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化GPIOD3,6
GPIO_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 对比度值设置为128
OLED_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/32
OLED_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);// 设置振荡器分频值为80
OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off 设置区域颜色模式关闭
OLED_WR_Byte(0x05,OLED_CMD);// 设置区域颜色模式值为05
OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period 设置预充电周期
OLED_WR_Byte(0xF1,OLED_CMD);// 设置预充电周期值为F1
OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion 设置COM引脚配置
OLED_WR_Byte(0x12,OLED_CMD);// 设置COM引脚配置值为12
OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh 设置Vcomh
OLED_WR_Byte(0x30,OLED_CMD);// 设置Vcomh值为30
OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable 设置充电泵使能
OLED_WR_Byte(0x14,OLED_CMD);// 设置充电泵使能值为14
OLED_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-page1
OLED_WR_Byte(0x02,0); //low column start address
OLED_WR_Byte(0x10,0); //high column start address
for(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的驱动显示