用STM32玩OLED(显示文字、图片、动图gif等)

一位伴随我童年的演员也离开了人间,真的是感叹人生无常,希望世上病痛少一点。

开发板:STM32F407ZGT6(正点原子探索者)

OLED:技小新0.96'OLED显示器模块(4PIN)

软件:Keil MDK5.29

Gitee项目已发布,需要源码请自助下载

地址(gitee)

OLED资料网页:网页地址

例程是STM32F103和STM8的,跟我用的不一样,所以就改了例程

4pin的OLED用的是IIC通信,按着手册改了oled的scl和sda的端口(PB8和PB9)

1. 显示字符串

/*
@brief 显示字符串
@param x:起始列
y:起始页
*chr:第一个字符首地址
@retval 无
*/
void OLED_ShowString(unsigned char x,unsigned char y,unsigned char *chr)
{
unsigned char j=0; //定义变量 while (chr[j]!='\0') //如果不是最后一个字符
{
OLED_ShowChar(x,y,chr[j]); //显示字符
x+=8; //列数加8 ,一个字符的列数占8
if(x>=128){x=0;y+=2;} //如果x大于等于128,切换页,从该页的第一列显示
j++; //下一个字符
}
}

测试代码如下

#include "sys.h"
#include "delay.h"
#include "oled.h"
#include "oledfont.h"
#include "bmp.h" int main(void)
{
delay_init(168); //延时初始化
OLED_Init(); //OLED初始化
OLED_Clear(); OLED_ShowString(36,2,"Giao!!!");
while(1)
{ }
}

显示效果如下

2. 显示中文

取模软件用的是PCtoLCD2002完美版,取模方式如下

下载地址:传送门(蓝奏云)

程序方面修改了例程,让程序功能变得更直接

/*
@brief 显示中文
@param x:起始列;一个字体占16列
y:起始页;一个字体占两页
no:字体的序号
m:单个字体大小
Chn[][m]:中文数组
@retval 无
*/
void OLED_ShowCHinese(unsigned char x,unsigned char y,unsigned char no,unsigned char m,unsigned char Chn[][m])
{
unsigned char t,adder=0; //定义变量 OLED_Set_Pos(x,y); //从 x y 开始画点,先画第一页
for(t=0;t<16;t++) //循环16次,画第一页的16列
{
OLED_WR_Byte(Chn[2*no][t],OLED_DATA);//画no在数组位置的第一页16列的点
adder+=1; //数组地址加1
}
OLED_Set_Pos(x,y+1); //画第二页
for(t=0;t<16;t++)//循环16次,画第二页的16列
{
OLED_WR_Byte(Chn[2*no+1][t],OLED_DATA);//画no在数组位置的第二页16列的点
adder+=1;//数组地址加1
}
}

测试程序如下

#include "sys.h"
#include "delay.h"
#include "oled.h"
#include "oledfont.h"
#include "bmp.h" int main(void)
{
delay_init(168); //延时初始化
OLED_Init(); //OLED初始化
OLED_Clear(); OLED_ShowCHinese(16,1,0,32,Welcome);
OLED_ShowCHinese(32,1,1,32,Welcome);
OLED_ShowCHinese(48,1,2,32,Welcome);
OLED_ShowCHinese(64,1,3,32,Welcome);
OLED_ShowCHinese(80,1,4,32,Welcome);
OLED_ShowCHinese(96,1,5,32,Welcome); while(1)
{ }
}

显示效果如下

3. 显示图片

取模方式与显示中文一样,但是要转换成数组形式,就要变成单色(黑与白),而且大小应该在128*64内,我用的软件是Img2Lcd

下载地址:传送门(蓝奏云)

设置效果如下



测试图片是一本书,网上找的,大小是48*48

unsigned char BMP1[] =
{
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x80,0xC0,0xE0,0x60,0x60,0x60,0x60,0x60,0x60,0xC0,0xC0,0xC0,0xC0,
0x80,0x80,0x80,0x80,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0xC0,0xE0,0x70,0x38,
0x1C,0x0E,0x07,0x73,0x79,0x58,0xC8,0xD8,0xD8,0xD8,0x98,0x90,0xB0,0xB0,0x30,0x30,
0x60,0xE1,0xE1,0xE1,0x01,0x03,0x03,0x03,0x83,0xC6,0xE6,0x7E,0x1C,0x00,0x00,0x00,
0x00,0x00,0x00,0x00,0x80,0xC0,0xE0,0x70,0x38,0x0C,0x07,0x03,0x01,0x00,0x00,0x00,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x01,0x01,0x01,0x01,0x03,
0x03,0x83,0xC1,0xE0,0x70,0x38,0x9E,0xE7,0x7F,0x0F,0x03,0xFF,0xFF,0x00,0x00,0x00,
0x00,0x00,0x00,0xFE,0xFF,0x03,0xF8,0xF8,0x0C,0x0C,0x0C,0x18,0x18,0x18,0x18,0x30,
0x30,0x30,0x60,0x60,0x60,0xC0,0xC0,0xC0,0xC0,0x80,0x80,0xC0,0xE0,0x70,0x1C,0x8E,
0xC7,0xE3,0x71,0x3C,0x8E,0xC7,0xE3,0x79,0x1C,0x0E,0x07,0x03,0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x01,0x07,0x0E,0x1C,0x3B,0x33,0x36,0x66,0x66,0x6C,0xCC,0xCC,0xD8,
0xD8,0x98,0xB0,0xB0,0x30,0x30,0x60,0x60,0x60,0xC0,0xE1,0x70,0x38,0x9C,0xCE,0xE7,
0x71,0x38,0x1E,0x07,0x03,0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
0x00,0x01,0x01,0x01,0x03,0x03,0x03,0x03,0x06,0x06,0x06,0x06,0x07,0x03,0x01,0x00,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,/*"C:\Users\long\Desktop\#′y′|àí\Pic\book1.bmp",0*/
};

这个就是例程里的,没什么要改的

程序如下:

/*
@brief 显示图片
@param x0:起始列地址
y0:起始页地址
x1:终止列地址
y1:终止页地址
BMP[]:存放图片代码的数组
@retval 无
*/
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; //判断终止页是否为8的整数倍
else y=y1/8+1; for(y=y0;y<y1;y++) //从起始页开始,画到终止页
{
OLED_Set_Pos(x0,y); //在页的起始列开始画
for(x=x0;x<x1;x++) //画x1 - x0 列
{
OLED_WR_Byte(BMP[j++],OLED_DATA); //画图片的点
}
}
}

值得注意的是函数的第一个和第三个参数,起始列地址和终止列地址,如果和图片的分辨率对不上,那就会显示出来一坨散沙,看不出样子

测试程序如下:

#include "sys.h"
#include "delay.h"
#include "oled.h"
#include "oledfont.h"
#include "bmp.h" int main(void)
{
delay_init(168); //延时初始化
OLED_Init(); //OLED初始化
OLED_Clear(); OLED_DrawBMP(40,2,88,8,BMP1); while(1)
{ }
}

显示效果如下

4. 显示动图

在网上找了一些512*512的GIF图,这样是没办法用的,需要降低分辨率和调成单色,而且还要提取帧

512*512动图下载地址:传送门(蓝奏云)

在这里用到的软件有很多,我用的是一个叫IrfanView的软件,这个软件可以快速的查看图片,也可以提取帧,批量操作图片,真滴好好用。

IrfanView下载地址:官网

传送门(蓝奏云)

使用教程:教程(bilibili)

取模之后的数组比较大,就不发上来了

显示动图程序如下

/*
@brief 显示动图
@param x0:起始列地址
y0:起始页地址
x1:终止列地址
y1:终止页地址
k: 帧个数
m: 单帧数组大小
BMP[][m]:存放动图代码的数组
@retval 无
*/
void OLED_DrawGIF(unsigned char x0, unsigned char y0,unsigned char x1, unsigned char y1, unsigned char k, int m, unsigned char GIF[][m])
{
unsigned int j=0; //定义变量
unsigned char x,y,i; //定义变量 if(y1%8==0) y=y1/8; //判断终止页是否为8的整数倍
else y=y1/8+1;
for (i=0;i<k;i++) //从第一帧开始画
{
j = 0;
for(y=y0;y<y1;y++) //从起始页开始,画到终止页
{
OLED_Set_Pos(x0,y); //在页的起始列开始画 for(x=x0;x<x1;x++) //画x1 - x0 列
{ OLED_WR_Byte(GIF[i][j++],OLED_DATA); //画图片的点
}
}
//delay_ms(80);//人为制造卡顿??? }
}

测试程序如下

#include "sys.h"
#include "delay.h"
#include "oled.h"
#include "oledfont.h"
#include "bmp.h" int main(void)
{
delay_init(168); //延时初始化
OLED_Init(); //OLED初始化
OLED_Clear(); while(1)
{
OLED_DrawGIF(32,0,96,8,31,512,Pencil);
}
}

显示效果如下:

5. 总结测试

测试代码:

#include "sys.h"
#include "delay.h"
#include "oled.h"
#include "oledfont.h"
#include "bmp.h" int main(void)
{
delay_init(168); //延时初始化
OLED_Init(); //OLED初始化
OLED_Clear(); OLED_ShowCHinese(16,1,0,32,Welcome);
OLED_ShowCHinese(32,1,1,32,Welcome);
OLED_ShowCHinese(48,1,2,32,Welcome);
OLED_ShowCHinese(64,1,3,32,Welcome);
OLED_ShowCHinese(80,1,4,32,Welcome);
OLED_ShowCHinese(96,1,5,32,Welcome);
delay_ms(500);
OLED_Clear(); OLED_ShowCHinese(16,1,0,32,Introduced);
OLED_ShowCHinese(32,1,1,32,Introduced);
OLED_ShowCHinese(48,1,2,32,Introduced);
OLED_ShowCHinese(64,1,3,32,Introduced); OLED_ShowCHinese(16,3,4,32,Introduced);
OLED_ShowCHinese(32,3,5,32,Introduced);
OLED_ShowCHinese(48,3,6,32,Introduced);
OLED_ShowCHinese(64,3,7,32,Introduced);
OLED_ShowCHinese(80,3,8,32,Introduced);
OLED_ShowCHinese(96,3,9,32,Introduced);
delay_ms(1000);
OLED_Clear(); while(1)
{
OLED_DrawGIF(32,0,96,8,31,512,Pencil);
OLED_DrawGIF(32,0,96,8,31,512,Pencil);
delay_ms(200);
OLED_DrawGIF(32,0,96,8,18,512,Rocket);
OLED_DrawGIF(32,0,96,8,18,512,Rocket);
delay_ms(200);
OLED_DrawGIF(32,0,96,8,28,512,Bell);
OLED_DrawGIF(32,0,96,8,28,512,Bell);
delay_ms(200); }
}

效果如下

玩耍OLED(动图)

挺好玩的对不对?来制作你的动图吧!


祝大家新的一年里健健康康!

用STM32玩OLED(显示文字、图片、动图gif等)的更多相关文章

  1. php生成文字图片效果

    php生成文字图片效果最近看到php的GD功能,试着做了一个基本生成文字图片效果的代码: 显示文字图片页面:demo.php<?php$str = $_REQUEST['str'] ? $_RE ...

  2. 用STM32玩SR04(测距、串口显示、OLED显示)

    目录 用STM32玩SR04(测距.串口显示.OLED显示) 超声波模块使用 SR04初始化 SR04使用串口打印数据 SR04使用OLED来传输数据,并显示在OLED上 用STM32玩SR04(测距 ...

  3. 致敬学长!J20航模遥控器开源项目计划【开局篇】 | 先做一个开机界面 | MATLAB图像二值化 | Img2Lcd图片取模 | OLED显示图片

    我们的开源宗旨:自由 协调 开放 合作 共享 拥抱开源,丰富国内开源生态,开展多人运动,欢迎加入我们哈~ 和一群志同道合的人,做自己所热爱的事! 项目开源地址:https://github.com/C ...

  4. Qt中在图片上叠加显示文字

    Qt中在图片上叠加显示文字   QCustLabel::QCustLabel(QWidget *parent):QLabel(parent){ setPixmap(QPixmap(QString::f ...

  5. VS2003 下GridControl的列显示成图片+文字的形式实现

    public RC_CustomerSolicitListUC() { // 该调用是 Windows.Forms 窗体设计器所必需的. InitializeComponent(); // TODO: ...

  6. C#跑马灯,图片滚动,后台获取图片地址。动态绑定图片,imag显示文字

    下面附下载地址. http://download.csdn.net/download/njxiaogui/10002058 1.跑马灯效果,图片连续循环滚动,图片下面并可附文字描述,图片是从数据库中获 ...

  7. 玩转X-CTR100 l STM32F4 l OLED显示-SSD1306无字库

    我造轮子,你造车,创客一起造起来!塔克创新资讯[塔克社区 www.xtark.cn ][塔克博客 www.cnblogs.com/xtark/ ]      OLED显示屏具有自发光特性,不需要背光, ...

  8. CSS实现文字半透明显示在图片上方法

    CSS实现文字半透明显示在图片上方法 在css中文字半透明我们会需要使用滤镜效果也就是css中的filter:alpha来实现了,下面来看两个文字显示在图片上并且半透明的例子. CSS让一行文字显示在 ...

  9. iOS View自定义窍门——UIButton实现上显示图片,下显示文字

    “UIButton实现上显示图片,下显示文字”这个需求相信大家在开发中都或多或少会遇见.比如自定义分享View的时候.当然,也可以封装一个item,上边imageView,下边一个label.但是既然 ...

随机推荐

  1. java中接口interface可以持有多个类的共享常量

    3.接口持有多个类的共享常量  接口另一主要功能,马克-to-win: 可以使用接口来引入多个类的共享常量.所有的这些变量名都将作为常量看待.所有定义在接口中的常量都默认为public.static和 ...

  2. PAT B1031查验身份证

    一个合法的身份证号码由17位地区.日期编号和顺序编号加1位校验码组成.校验码的计算规则如下: 首先对前17位数字加权求和,权重分配为:{7,9,10,5,8,4,2,1,6,3,7,9,10,5,8, ...

  3. java中如何能把一个字符串转成日期对象

    题目3.1: 把一个字符串转成日期对象 当我们想根据输入字符串得到一个日期对象时我们不知道,应该以什么格式写这个字符串,才能被系统正确解析,一种聪明的做法是,马克-to-win,我们先 把日期对象根据 ...

  4. java重载时自动转换咋回事?举例说明

    当一个重载的方法被调用时,Java在调用方法的参数和方法的自变量之间寻找匹配.    但是,这种匹配并不总是精确的.只有在找不到精确匹配时,Java的自动转换才会起作用. (如果定义了test(int ...

  5. Python pip下载慢的解决方法

    国外的源下载速度实在是太慢了 可以使用国内的一些镜像网站安装 使用cmd命令 格式:pip install -i 网站 库 例如: 国内的一些镜像网站 清华大学:https://pypi.tuna.t ...

  6. Go Slice Tricks Cheat Sheet、Go 切片使用小妙招

    AppendVector. Copy. Cut. Delete. Delete without preserving order. Cut (GC). Delete (GC). Delete with ...

  7. ArrayList扩容问题

    今天上午上课在看JavaSE的面经,其中有问关于ArrayList和LinkedList的区别,就突然思考到,既然ArrayList是采用数组形式存储数据,对比我们自己使用到的数组,为什么ArrayL ...

  8. Java学习day35(《大道至简》读后感)

    对于大一之前的我来说,并不明白计算机这个专业要做的是什么,在我的眼中敲敲键盘打打字就是这个专业的全部:对于现在的我而言,这样的想法显然是十分幼稚的. 当初高考完填报志愿时选择了这门专业,也并不是因为自 ...

  9. 『现学现忘』Git基础 — 4、Git下载与安装

    目录 1.Git下载 2.Git在Windows下的详细安装 3.验证Git是否安装成功 1.Git下载 进入官方地址下载Git客户端:https://git-scm.com/download/win ...

  10. Codeforces Round #754 (Div. 2), problem: (A) A.M. Deviation泪目 万万没想到狂wa是因为这

    Problem - A - Codeforces 题目 题意很简单每次操作可以使得a1 a2  a3任意两个数分别+1  -1 求最后使得a+c-2b绝对值的最小值 BUG就是最后忽略了-2和2这一点 ...