不想看《跟着Hugh学开发--51单片机篇》pdf版本?没关系,网页版 markdown 格式为你准好了!
OLED 应用场合非常的多。我们这里选用中景园的0.96 OLED 作为实例,讲解 OLED 的驱动与使用。
图 20-1 0.96 OLED 外形图
0.96 OLED 外形,正反面如上图所示,除了屏幕用于显示之外,反面有4个插针引脚。引脚定义如下表所示:
表 20-1 接口定义
0.96 OLED 接口定义
序号 | 符号 | 说明 |
---|---|---|
1 | GND | 电源地 |
2 | VCC | 电源正3.3~5V |
3 | SCL | I^2^C 时钟线 |
4 | SDA | I^2^C 数据线 |
由上表可知,我们这里使用的 OLED ,采用 I^2^C 通信协议。因此,只需要研究 OLED 的驱动芯片的 I^2^C 协议,再结合前面课程学习过的 I^2^C 通信,即可实现对显示器的控制了。
图 20-2 0.96 OLED 驱动
该款 OLED 使用了 SSD1306 作为液晶的驱动芯片,如上图。
图 20-3 0.96 OLED 像素布局
液晶显示器的布局如上图所示,每一个方块就是一个像素:
总共有 64 行, 128 列,共计 128 * 64 = 8192 像素,每个像素可以有两种状态,亮或灭。
每8行组成一页,总共 64/8=8页,页也称为 Page。
左上角的第一个像素,为第0行,第0列。右下角的最后一个像素,为第 63 行,第 127 列。
前8行称为 Page0,最后8称为 Page7。
驱动芯片 SSD1306 规定:
每次写入数据的最小单位为1个字节,即8位;
而且每次写入必须是在同一页内,不能跨页;
写入的数据按列写入,下面举例说明。
例子:
往 page0 的第0列写入一个字节的数据 0x01= 0b0000,0001,显示结果如下图所示:
图 20-4 0.96 写入一个字节的数据
如上图所示, Page0 对应屏幕的第 0-7 行。第0行、第0列将写入数据的最低位 D0,即0。第7行、第0列将写入数据的最高位 D7,即1。因为写入的数据为 0x01,显示屏的第7行,第0列的单个像素将点亮,整个屏幕效果为,只有一个像素点亮。
上面是0.96 OLED 的最小操作单位,我们可以选择 0-127 列中的任意列, 0-7 页的任意页写入一个字节的数据。
这里值得注意的是, OLED 的最小操作单位,并不是单个像素,而是一次最小操作某一列的8个像素。我们可以组合最小操作单位,实现字符、数字、中文、字符串以及图像的显示。
图 20-5 OLED 驱动 SSD1306 写数据时序
如上图所示,驱动步骤入下:
启动 I^2^C 通信;
写入从设备地址;
写入控制字节,可选;
写入一个字节数据,可选;
写入控制字节, 1Byte;
写入数据,≥0Byte;
结束 I^2^C 通信。
如上述步骤所示,其中步骤3,4可选,我们使用最精简的模式跳过步骤3,4。根据步骤5中控制字的不同,步骤6的内容将表示数据或者命令。
当步骤5写入 0x00,表示6为命令;
当步骤5写入 0x40,表示6为数据。
根据上述内容,我们编写 OLED 的驱动函数如下, I^2^C 的底层驱动在前面章节学习过,我们这里直接调用:
/**********************************************
// IIC Write Command
**********************************************/
void Write_IIC_Command(unsigned char IIC_Command)
{
Start_I2C();
Wr_I2C(0x78); //Slave address,SA0=0
Wr_I2C(0x00); //write command
Wr_I2C(IIC_Command);
Stop_I2C();
}
/**********************************************
// IIC Write Data
**********************************************/
void Write_IIC_Data(unsigned char IIC_Data)
{
Start_I2C();
Wr_I2C(0x78); //D/C#=0; R/W#=0
Wr_I2C(0x40); //write data
Wr_I2C(IIC_Data);
Stop_I2C();
}
/**********************************************
// Write OLED
**********************************************/
void OLED_WR_Byte(unsigned dat,unsigned cmd)
{
if(cmd) Write_IIC_Data(dat);
else Write_IIC_Command(dat);
}
图 20-6 OLED 驱动代码编写
上图中 Write_IIC_Command() , Write_IIC_Data()分别写命令和数据函数,并将其组合成了 OLED 的最小操作函数 OLED _WR_Byte(),后续所用的操作均以这个函数为基础。
通过写入相应的值来初始化 OLED 以及清除屏幕。相应的指令从驱动 SSD1306 的技术文档可以查阅,我们这里直接上代码 OLED _Init()、 OLED _Clear() :
/*****************************************************************
* OLED 初始化
* 为 OLED 正常工作做准备
***************************************************************/
void OLED_Init(void)
{
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
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)
OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty
OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction
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);//
OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off
OLED_WR_Byte(0x05,OLED_CMD);//
OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period
OLED_WR_Byte(0xF1,OLED_CMD);//
OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion
OLED_WR_Byte(0x12,OLED_CMD);//
OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh
OLED_WR_Byte(0x30,OLED_CMD);//
OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable
OLED_WR_Byte(0x14,OLED_CMD);//
OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel
}
图 20-7 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 (0x00,OLED_CMD); // 设置显示位置 --- 列低地址
OLED_WR_Byte (0x10,OLED_CMD); // 设置显示位置 --- 列高地址
for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA);
} // 更新显示
}
图 20-8 OLED 清屏
如何在显示屏上显示一个字符?首先,我们需要制作一个字符的点阵。以字符 “1” 为例子讲解,字符点阵的制作。
图 20-9 字符 “0” 的显示效果
如上图所示,字符 “1” 大小为 8x 16 像素,即占用空间为8列, 16 行。结合前面讲解的内容,我们可以按列整理成数据,第一列可以整合成两个字节的数据,依次类推按列整合成数据如下:
0x00,0x10,0x10,0xF8,0x00,0x00,0x00,0x00,0x00,0x20,0x20,0x3F,0x20,0x20,0x00,0x00
调用 OLED 的最小操作函数把上述字节写入 OLED 即可。为了方便,我们把所有可打印字符的数据整理成一个数组,在数组中查表即可,如下图所示。
图 20-10 字符点阵数组
/*****************************************************************
* 0.96 OLED 128 *64 像素,即:128 列 x 64 行
* 64 行被均分成了8页,最开始8行为第0页,最后8行为第7页。
*
* 函数功能为:从x列,y页开始显示一个字符,Char_Size 为字体大小。
*
* x:0-127(列)
* y:0-7 (页)
* Char_Size: 16 (字体大小为:8列 x 16 行),其他(6列 x 8行)
* flag,反白显示,默认为0,正常显示。1:反白显示。
*****************************************************************/
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 Char_Size,bit flag)
{
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++)
{
if(flag == 0) OLED_WR_Byte( F8X16[c*16+i],OLED_DATA);
else OLED_WR_Byte(~F8X16[c*16+i],OLED_DATA);
}
OLED_Set_Pos(x,y+1);
for(i=0;i<8;i++)
{
if(flag == 0) OLED_WR_Byte( F8X16[c*16+i+8],OLED_DATA);
else OLED_WR_Byte(~F8X16[c*16+i+8],OLED_DATA);
}
}
else
{
OLED_Set_Pos(x,y);
for(i=0;i<6;i++)
{
if(flag == 0) OLED_WR_Byte( F6x8[c][i],OLED_DATA);
else OLED_WR_Byte(~F6x8[c][i],OLED_DATA);
}
}
}
图 20-11 写字符函数
上述函数中,可以选择两种字符,大小分别为 8x16 像素和 6x8 像素,分别放在数组符F 8x16 []和F 6x8 []中。
同样的道理,可以制作不同的字体放入数组中。
使用工具软件 PCtoLCD2002 生成字符点阵,使用方法如下:
图 20-12 字模软件生成字符点阵
有了前面的基础,写入数字就比较简单了,即把数字各个位转换成字符,然后写入即可, OLED_ShowNum()代码如下:
/*****************************************************************
*
* 函数功能为:m的n次方
*
*****************************************************************/
u32 oled_pow(u8 m,u8 n)
{
u32 result=1;
while(n--)result*=m;
return result;
}
/*****************************************************************
* 0.96 OLED 128 *64 像素,即:128 列 x 64 行
* 64 行被均分成了8页,最开始8行为第0页,最后8行为第7页。
*
* 函数功能为:从x列,y页开始显示数字。
*
* x:0-127(列)
* y:0-7 (页)
* len:数字长度
* Char_Size: 16 (字体大小为:8列 x 16 行),其他(6列 x 8行)
*
***************************************************************/
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 Char_Size,bit flag)
{
u8 t,temp,RowChar,CharSize;
u8 enshow=0;
if(Char_Size==16)// 判断字体大小
{
RowChar = 8;
CharSize= 16;
}
else
{
RowChar = 6;
CharSize= 8;
}
for(t=0;t<len;t++)
{
temp=(num/oled_pow(10,len-t-1))%10;
if(enshow==0&&t<(len-1))
{
if(temp==0)
{ // 第一个非0数字之前用空格替代
OLED_ShowChar(x+RowChar*t,y,' ',CharSize,flag);
** continue**;
}
else enshow=1;
}
OLED_ShowChar(x+RowChar*t,y,temp+'0',CharSize,flag);
}
}
图 20-13 显示数字
同样利用字模软件生成中文字模,并放入数字 Hzk[][]中,显示中文函数如下所示:
/********************************************************************
* 0.96 OLED 128 *64 像素,即:128 列 x 64 行
* 64 行被均分成了8页,最开始8行为第0页,最后8行为第7页。
*
* 函数功能为:从x列,y页开始显示汉字,汉字大小 16x16
*
* x:0-127(列)
* y:0-7 (页)
* *chr:字符串首地址
* Num :第 Num 个汉字,汉字定义在 oledfont.h的数组 HzK[]中。
*
****************************************************************/
void OLED_ShowCHinese(u8 x,u8 y,u8 Num,bit flag)
{
u8 t,adder=0;
OLED_Set_Pos(x,y);
for(t=0;t<16;t++)
{
if(flag == 0) OLED_WR_Byte( Hzk[2*Num][t],OLED_DATA);
else OLED_WR_Byte(~Hzk[2*Num][t],OLED_DATA);
adder+=1;
}
OLED_Set_Pos(x,y+1);
for(t=0;t<16;t++)
{
if(flag == 0) OLED_WR_Byte( Hzk[2*Num+1][t],OLED_DATA);
else OLED_WR_Byte(~Hzk[2*Num+1][t],OLED_DATA);
adder+=1;
}
}
图 20-14 显示中文
将字符组合,实现字符串显示,函数如下:
/*************************************************************
* 0.96 OLED 128 *64 像素,即:128 列 x 64 行
* 64 行被均分成了8页,最开始8行为第0页,最后8行为第7页。
*
* 函数功能为:从x列,y页开始显示字符串。
*
* x:0-127(列)
* y:0-7 (页)
* *chr:字符串首地址
* Char_Size: 16 (字体大小为:8列 x 16 行),其他(6列 x 8行)
*
****************************************************************/
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 Char_Size,bit flag)
{
u8 CharSize,RowChar,j=0;
if(Char_Size==16)
{
RowChar = 8;
CharSize= 16;
}
else
{
RowChar = 6;
CharSize= 8;
}
while (chr[j]!='\0')
{
OLED_ShowChar(x,y,chr[j],CharSize,flag);
x+=RowChar;
if(x>(120-RowChar))// 行尾不足一个字,换行
{
x=0;
y+=2;
}
j++;
}
}
图 20-15 显示字符串
利用字模软件将 128x64 像素的图片生成字模,并存储于 BMP[]数组中,图片显示函数如下:
/****************************************************************
* 0.96 OLED 128 *64 像素,即:128 列 x 64 行
* 64 行被均分成了8页,最开始8行为第0页,最后8行为第7页。
*
* 函数功能为:从 x0 列, y0 页开始显示图片,坐标
*
* x:0-127(列)
* y:0-7 (页)
* *chr:字符串首地址
* BMP[]:图片字模数组
*
***************************************************************/
void OLED_DrawBMP(u8 x0,u8 y0,u8 BMP[],bit flag)
{
unsigned int j=0;
unsigned char x,y;
for(y=y0;y<8;y++)
{
OLED_Set_Pos(x0,y);
for(x=x0;x<128;x++)
{
if(flag == 0) OLED_WR_Byte( BMP[j++],OLED_DATA);
else OLED_WR_Byte(~BMP[j++],OLED_DATA);
}
}
}
图 20-16 显示图片
将上述代码放入 oled.c, oled.h,字符字模和图片字模数组分别放入 oledfont.h和 bmp.h文件中。建立工程,在 main()函数中进行功能测试,部分代码如下:
/*****************************************
*
*0.96 OLED 字符显示测试
*
*******************************************/
OLED_ShowChar( 0,0,'A',16,0);
OLED_ShowChar( 8,0,'B',16,0);
OLED_ShowChar(16,0,'C',16,0);
OLED_ShowChar(24,0,'D',16,0);
OLED_ShowChar( 0,2,'A',8,0);
OLED_ShowChar( 8,2,'B',8,0);
OLED_ShowChar(16,2,'C',8,0);
OLED_ShowChar(24,2,'D',8,0);
OLED_ShowString(25,6,"Char Test!",16,1);
delayms(5000);
OLED_Clear();// 清除屏幕
/*****************************************
*
*0.96 OLED 数字显示测试
*
*******************************************/
OLED_ShowNum( 0,1,12,2,16,0);
OLED_ShowNum( 48,1,34,2,16,0);
OLED_ShowNum( 96,1,56,2,16,0);
OLED_ShowString(25,6,"Num Test!",16,1);
delayms(5000);
OLED_Clear();// 清除屏幕
/*****************************************
*
*0.96 OLED 中文显示测试
*
*******************************************/
OLED_ShowCHinese(22 ,3,1,0);// 不
OLED_ShowCHinese(22+16,3,2,0);// 见
OLED_ShowCHinese(22+32,3,3,0);// 不
OLED_ShowCHinese(22+48,3,4,0);// 散
OLED_ShowCHinese(22+64,3,5,0);//!
OLED_ShowString(25,6,"CHN Test!",16,1);
delayms(5000);
OLED_Clear();// 清除屏幕
/*****************************************
*
*0.96 OLED 字符串显示测试
*
*******************************************/
OLED_ShowString(0,2,"Nebula-Pi,RYMCU!",16,0);
OLED_ShowString(25,6,"Str Test!",16,1);
delayms(5000);
OLED_Clear();// 清除屏幕
/*****************************************
*
*0.96 OLED 图片显示测试
*
*******************************************/
OLED_DrawBMP(0,0,Logo,0);// 显示图片
OLED_ShowString(25,6,"PIC Test!",16,1);
delayms(5000);
图 20-17 综合显示部分代码
OLED 与开发板 Nebula Pi 连接关系如下图所示,将 OLED 靠左插入 LCD1602 的前4个排孔中。另外,左上角的跳线帽选择 OLED 。
OLED 的 I^2^C 总线与单片机 I/O 口连接情况,如图 20-19 所示。需要将 I^2^C 的 SCL, SDA 定义更新,如图 20-20 所示。
图 20-18 OLED 与开发板连接
图 20-19 OLED 与开发板硬件连接关系
图 20-20 OLED 的 II2C 引脚定义
字符显示与数字显示如下:
图 20-21 OLED 字符及数字显示结果
中文显示与字符串显示如下:
图 20-22 OLED 中文及字符串显示结果
图片显示如下:
图 20-23 OLED 图片显示
将 DS18B20 采集的温度, DS1302 的时间分别显示在 OLED 上,并实现三个按键可设置时间,蜂鸣器整点报时功能,编写相应代码,代码详细见本章程序工程。编译后,自行下载测试功能,部分综合显示结果如下图所示:
图 20-24 综合显示结果
本章讲解了0.96 OLED 显示原理,驱动编写。