STM32F103驱动oled显示屏

简介: STM32F103驱动oled显示屏


一.了解oled显示屏

oled显示屏和其他显示屏类似,不过他只有0.96英寸,屏幕较小,但是使用起来比较方便。有二种驱动方式,分别为IIC,和SPI驱动。驱动方式比较简单。IIC驱动的话只需要4根线,电源,地线,数据线,和时钟线。

二.IIC协议驱动oled显示屏

我这里使用的是IIC协议驱动oled显示屏,如果想了解IIC协议的可以看博主这篇文章:IIC协议详解

这边就直接上代码了

2.1 oled.c

#include "oled.h"
#include "stdlib.h"
#include "oledfont.h"
#include "delay.h"
u8 OLED_GRAM[144][8];
void show_led(char * msg, char * msg1,char * msg2,char * msg3)
{  
    
    
  //光照
    OLED_ShowChinese(0,0,4,16,1); //汉字隔16
  OLED_ShowChinese(16,0,5,16,1);
  
  //烟雾溶度
  OLED_ShowChinese(0,16,8,16,1); //汉字隔16
  OLED_ShowChinese(16,16,9,16,1);
    OLED_ShowChinese(32,16,10,16,1);
  OLED_ShowChinese(48,16,11,16,1);
  
  //温度
  OLED_ShowChinese(0,32,0,16,1); //汉字隔16
  OLED_ShowChinese(16,32,1,16,1);
  
  //湿度  
  OLED_ShowChinese(0,48,2,16,1); //汉字隔16
  OLED_ShowChinese(16,48,3,16,1);
    
    OLED_ShowString(32,0,(unsigned char *)msg,16,1);//16*16    
  OLED_ShowString(64,16,(unsigned char *)msg1,16,1);//16*16   
  OLED_ShowString(32,32,(unsigned char *)msg2,16,1);//16*16  
  OLED_ShowString(32,48,(unsigned char *)msg3,16,1);//16*16   
  OLED_Refresh();
    OLED_Clear();
}
//反显函数
void OLED_ColorTurn(u8 i)
{
  if(i==0)
    {
      OLED_WR_Byte(0xA6,OLED_CMD);//正常显示
    }
  if(i==1)
    {
      OLED_WR_Byte(0xA7,OLED_CMD);//反色显示
    }
}
//屏幕旋转180度
void OLED_DisplayTurn(u8 i)
{
  if(i==0)
    {
      OLED_WR_Byte(0xC8,OLED_CMD);//正常显示
      OLED_WR_Byte(0xA1,OLED_CMD);
    }
  if(i==1)
    {
      OLED_WR_Byte(0xC0,OLED_CMD);//反转显示
      OLED_WR_Byte(0xA0,OLED_CMD);
    }
}
//延时
void IIC_delay(void)
{
  u8 t=3;
  while(t--);
}
//起始信号
void I2C_Start(void)
{
  OLED_SDA_Set();
  OLED_SCL_Set();
  IIC_delay();
  OLED_SDA_Clr();
  IIC_delay();
  OLED_SCL_Clr();
  IIC_delay();
}
//结束信号
void I2C_Stop(void)
{
  OLED_SDA_Clr();
  OLED_SCL_Set();
  IIC_delay();
  OLED_SDA_Set();
}
//等待信号响应
void I2C_WaitAck(void) //测数据信号的电平
{
  OLED_SDA_Set();
  IIC_delay();
  OLED_SCL_Set();
  IIC_delay();
  OLED_SCL_Clr();
  IIC_delay();
}
//写入一个字节
void Send_Byte(u8 dat)
{
  u8 i;
  for(i=0;i<8;i++)
  {
    if(dat&0x80)//将dat的8位从最高位依次写入
    {
      OLED_SDA_Set();
    }
    else
    {
      OLED_SDA_Clr();
    }
    IIC_delay();
    OLED_SCL_Set();
    IIC_delay();
    OLED_SCL_Clr();//将时钟信号设置为低电平
    dat<<=1;
  }
}
//发送一个字节
//mode:数据/命令标志 0,表示命令;1,表示数据;
void OLED_WR_Byte(u8 dat,u8 mode)
{
  I2C_Start();
  Send_Byte(0x78);
  I2C_WaitAck();
  if(mode){Send_Byte(0x40);}
  else{Send_Byte(0x00);}
  I2C_WaitAck();
  Send_Byte(dat);
  I2C_WaitAck();
  I2C_Stop();
}
//开启OLED显示 
void OLED_DisPlay_On(void)
{
  OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  OLED_WR_Byte(0x14,OLED_CMD);//开启电荷泵
  OLED_WR_Byte(0xAF,OLED_CMD);//点亮屏幕
}
//关闭OLED显示 
void OLED_DisPlay_Off(void)
{
  OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  OLED_WR_Byte(0x10,OLED_CMD);//关闭电荷泵
  OLED_WR_Byte(0xAE,OLED_CMD);//关闭屏幕
}
//更新显存到OLED 
void OLED_Refresh(void)
{
  u8 i,n;
  for(i=0;i<8;i++)
  {
    OLED_WR_Byte(0xb0+i,OLED_CMD); //设置行起始地址
    OLED_WR_Byte(0x00,OLED_CMD);   //设置低列起始地址
    OLED_WR_Byte(0x10,OLED_CMD);   //设置高列起始地址
    I2C_Start();
    Send_Byte(0x78);
    I2C_WaitAck();
    Send_Byte(0x40);
    I2C_WaitAck();
    for(n=0;n<128;n++)
    {
      Send_Byte(OLED_GRAM[n][i]);
      I2C_WaitAck();
    }
    I2C_Stop();
  }
}
//清屏函数
void OLED_Clear(void)  //8*128
{
  u8 i,n;
  for(i=0;i<8;i++)
  {
     for(n=0;n<128;n++)
      {
       OLED_GRAM[n][i]=0;//清除所有数据
      }
  }
  //OLED_Refresh();//更新显示
}
//画点 
//x:0~127
//y:0~63
//t:1 填充 0,清空 
void OLED_DrawPoint(u8 x,u8 y,u8 t)
{
  u8 i,m,n;
  i=y/8;
  m=y%8;
  n=1<<m;
  if(t){OLED_GRAM[x][i]|=n;}
  else
  {
    OLED_GRAM[x][i]=~OLED_GRAM[x][i];
    OLED_GRAM[x][i]|=n;
    OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  }
}
//画线
//x1,y1:起点坐标
//x2,y2:结束坐标
void OLED_DrawLine(u8 x1,u8 y1,u8 x2,u8 y2,u8 mode)
{
  u16 t; 
  int xerr=0,yerr=0,delta_x,delta_y,distance;
  int incx,incy,uRow,uCol;
  delta_x=x2-x1; //计算坐标增量 
  delta_y=y2-y1;
  uRow=x1;//画线起点坐标
  uCol=y1;
  if(delta_x>0)incx=1; //设置单步方向 
  else if (delta_x==0)incx=0;//垂直线 
  else {incx=-1;delta_x=-delta_x;}
  if(delta_y>0)incy=1;
  else if (delta_y==0)incy=0;//水平线 
  else {incy=-1;delta_y=-delta_x;}
  if(delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴 
  else distance=delta_y;
  for(t=0;t<distance+1;t++)
  {
    OLED_DrawPoint(uRow,uCol,mode);//画点
    xerr+=delta_x;
    yerr+=delta_y;
    if(xerr>distance)
    {
      xerr-=distance;
      uRow+=incx;
    }
    if(yerr>distance)
    {
      yerr-=distance;
      uCol+=incy;
    }
  }
}
//x,y:圆心坐标
//r:圆的半径
void OLED_DrawCircle(u8 x,u8 y,u8 r)
{
  int a, b,num;
    a = 0;
    b = r;
    while(2 * b * b >= r * r)      
    {
        OLED_DrawPoint(x + a, y - b,1);
        OLED_DrawPoint(x - a, y - b,1);
        OLED_DrawPoint(x - a, y + b,1);
        OLED_DrawPoint(x + a, y + b,1);
 
        OLED_DrawPoint(x + b, y + a,1);
        OLED_DrawPoint(x + b, y - a,1);
        OLED_DrawPoint(x - b, y - a,1);
        OLED_DrawPoint(x - b, y + a,1);
        
        a++;
        num = (a * a + b * b) - r*r;//计算画的点离圆心的距离
        if(num > 0)
        {
            b--;
            a--;
        }
    }
}
//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//size1:选择字体 6x8/6x12/8x16/12x24
//mode:0,反色显示;1,正常显示
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size1,u8 mode)
{
  u8 i,m,temp,size2,chr1;
  u8 x0=x,y0=y;
  if(size1==8)size2=6;
  else size2=(size1/8+((size1%8)?1:0))*(size1/2);  //得到字体一个字符对应点阵集所占的字节数
  chr1=chr-' ';  //计算偏移后的值
  for(i=0;i<size2;i++)
  {
    if(size1==8)
        {temp=asc2_0806[chr1][i];} //调用0806字体
    else if(size1==12)
        {temp=asc2_1206[chr1][i];} //调用1206字体
    else if(size1==16)
        {temp=asc2_1608[chr1][i];} //调用1608字体
    else if(size1==24)
        {temp=asc2_2412[chr1][i];} //调用2412字体
    else return;
    for(m=0;m<8;m++)
    {
      if(temp&0x01)OLED_DrawPoint(x,y,mode);
      else OLED_DrawPoint(x,y,!mode);
      temp>>=1;
      y++;
    }
    x++;
    if((size1!=8)&&((x-x0)==size1/2))
    {x=x0;y0=y0+8;}
    y=y0;
  }
}
//显示字符串
//x,y:起点坐标  
//size1:字体大小 
//*chr:字符串起始地址 
//mode:0,反色显示;1,正常显示
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 size1,u8 mode)
{
  while((*chr>=' ')&&(*chr<='~'))//判断是不是非法字符!
  {
    OLED_ShowChar(x,y,*chr,size1,mode);
    if(size1==8)x+=6;
    else x+=size1/2;
    chr++;
  }
}
//m^n
u32 OLED_Pow(u8 m,u8 n)
{
  u32 result=1;
  while(n--)
  {
    result*=m;
  }
  return result;
}
//显示数字
//x,y :起点坐标
//num :要显示的数字
//len :数字的位数
//size:字体大小
//mode:0,反色显示;1,正常显示
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size1,u8 mode)
{
  u8 t,temp,m=0;
  if(size1==8)m=2;
  for(t=0;t<len;t++)
  {
    temp=(num/OLED_Pow(10,len-t-1))%10;
      if(temp==0)
      {
        OLED_ShowChar(x+(size1/2+m)*t,y,'0',size1,mode);
      }
      else 
      {
        OLED_ShowChar(x+(size1/2+m)*t,y,temp+'0',size1,mode);
      }
  }
}
//显示汉字
//x,y:起点坐标
//num:汉字对应的序号
//mode:0,反色显示;1,正常显示
void OLED_ShowChinese(u8 x,u8 y,u8 num,u8 size1,u8 mode)
{
  u8 m,temp;
  u8 x0=x,y0=y;
  u16 i,size3=(size1/8+((size1%8)?1:0))*size1;  //得到字体一个字符对应点阵集所占的字节数
  for(i=0;i<size3;i++)
  {
    if(size1==16)
        {temp= F16x16[num][i];}//调用16*16字体
    else if(size1==24)
        {temp= F16x16[num][i];}//调用24*24字体
    else if(size1==32)       
        {temp= F16x16[num][i];}//调用32*32字体
    else if(size1==64)
        {temp= F16x16[num][i];}//调用64*64字体
    else return;
    for(m=0;m<8;m++)
    {
      if(temp&0x01)OLED_DrawPoint(x,y,mode);
      else OLED_DrawPoint(x,y,!mode);
      temp>>=1;
      y++;
    }
    x++;
    if((x-x0)==size1)
    {x=x0;y0=y0+8;}
    y=y0;
  }
}
//num 显示汉字的个数
//space 每一遍显示的间隔
//mode:0,反色显示;1,正常显示
void OLED_ScrollDisplay(u8 num,u8 space,u8 mode)
{
  u8 i,n,t=0,m=0,r;
  while(1)
  {
    if(m==0)
    {
      OLED_ShowChinese(128,24,t,16,mode); //写入一个汉字保存在OLED_GRAM[][]数组中
      t++;
    }
    if(t==num)
      {
        for(r=0;r<16*space;r++)      //显示间隔
         {
          for(i=1;i<144;i++)
            {
              for(n=0;n<8;n++)
              {
                OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
              }
            }
           OLED_Refresh();
         }
        t=0;
      }
    m++;
    if(m==16){m=0;}
    for(i=1;i<144;i++)   //实现左移
    {
      for(n=0;n<8;n++)
      {
        OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
      }
    }
    OLED_Refresh();
  }
}
//x,y:起点坐标
//sizex,sizey,图片长宽
//BMP[]:要写入的图片数组
//mode:0,反色显示;1,正常显示
void OLED_ShowPicture(u8 x,u8 y,u8 sizex,u8 sizey,u8 BMP[],u8 mode)
{
  u16 j=0;
  u8 i,n,temp,m;
  u8 x0=x,y0=y;
  sizey=sizey/8+((sizey%8)?1:0);
  for(n=0;n<sizey;n++)
  {
     for(i=0;i<sizex;i++)
     {
        temp=BMP[j];
        j++;
        for(m=0;m<8;m++)
        {
          if(temp&0x01)OLED_DrawPoint(x,y,mode);
          else OLED_DrawPoint(x,y,!mode);
          temp>>=1;
          y++;
        }
        x++;
        if((x-x0)==sizex)
        {
          x=x0;
          y0=y0+8;
        }
        y=y0;
     }
   }
}
//OLED的初始化
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_OD;     //推挽输出
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
  GPIO_Init(GPIOB, &GPIO_InitStructure);    //初始化PA0,1
  GPIO_SetBits(GPIOB,GPIO_Pin_8|GPIO_Pin_9);
  
  OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
  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  Set Mapping RAM Display Start Line (0x00~0x3F)
  OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
  OLED_WR_Byte(0xCF,OLED_CMD);// Set SEG Output Current Brightness
  OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping     0xa0左右反置 0xa1正常
  OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction   0xc0上下反置 0xc8正常
  OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
  OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
  OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
  OLED_WR_Byte(0x00,OLED_CMD);//-not offset
  OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
  OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
  OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
  OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
  OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
  OLED_WR_Byte(0x12,OLED_CMD);
  OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
  OLED_WR_Byte(0x30,OLED_CMD);//Set VCOM Deselect Level
  OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
  OLED_WR_Byte(0x02,OLED_CMD);//
  OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
  OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
  OLED_Clear();
  OLED_WR_Byte(0xAF,OLED_CMD);
    
     OLED_Clear();  
   OLED_ColorTurn(0);//0正常显示,1 反色显示
   OLED_DisplayTurn(0);//0正常显示 1 屏幕翻转显示
}

2.2 oled.h

#ifndef __OLED_H
#define __OLED_H 
#include "sys.h"
#include "stdlib.h" 
//-----------------OLED端口定义---------------- 
#define OLED_SCL_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_8)//SCL
#define OLED_SCL_Set() GPIO_SetBits(GPIOB,GPIO_Pin_8)
#define OLED_SDA_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_9)//DIN
#define OLED_SDA_Set() GPIO_SetBits(GPIOB,GPIO_Pin_9)
#define OLED_CMD  0 //写命令
#define OLED_DATA 1 //写数据
void OLED_ClearPoint(u8 x,u8 y);
void OLED_ColorTurn(u8 i);
void OLED_DisplayTurn(u8 i);
void I2C_Start(void);
void I2C_Stop(void);
void I2C_WaitAck(void);
void Send_Byte(u8 dat);
void OLED_WR_Byte(u8 dat,u8 mode);
void OLED_DisPlay_On(void);
void OLED_DisPlay_Off(void);
void OLED_Refresh(void);
void OLED_Clear(void);
void OLED_DrawPoint(u8 x,u8 y,u8 t);
void OLED_DrawLine(u8 x1,u8 y1,u8 x2,u8 y2,u8 mode);
void OLED_DrawCircle(u8 x,u8 y,u8 r);
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size1,u8 mode);
void OLED_ShowChar6x8(u8 x,u8 y,u8 chr,u8 mode);
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 size1,u8 mode);
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size1,u8 mode);
void OLED_ShowChinese(u8 x,u8 y,u8 num,u8 size1,u8 mode);
void OLED_ScrollDisplay(u8 num,u8 space,u8 mode);
void OLED_ShowPicture(u8 x,u8 y,u8 sizex,u8 sizey,u8 BMP[],u8 mode);
void OLED_Init(void);
void show_led(char * msg, char * msg1,char * msg2,char * msg3);
#endif

驱动代码就是以上这些了。

三.效果展示

目录
相关文章
|
4月前
|
传感器
手把手在STM32F103C8T6上构建可扩展可移植的DHT11驱动
【8月更文挑战第29天】本文详细介绍在STM32F103C8T6上构建可扩展且可移植的DHT11温湿度传感器驱动的步骤,包括硬件与软件准备、硬件连接、驱动代码编写及测试。通过这些步骤,可根据实际项目需求优化和扩展代码。
168 0
|
5月前
STM32Cubemx PWM驱动加湿器模拟火山喷发效果
STM32Cubemx PWM驱动加湿器模拟火山喷发效果
83 14
|
5月前
STM32Cubemx PWM驱动SG90舵机
STM32Cubemx PWM驱动SG90舵机
215 13
|
5月前
STM32CubeMX mpu6050驱动
STM32CubeMX mpu6050驱动
87 10
|
5月前
STM32CubeMX EC11旋转编码器驱动
STM32CubeMX EC11旋转编码器驱动
290 10
|
5月前
STM32CubeMX OLED驱动
STM32CubeMX OLED驱动
80 10
|
5月前
|
芯片
STM32CubeMX TM1637驱动数码管
STM32CubeMX TM1637驱动数码管
176 6
|
5月前
STM32CubeMX WS2812B灯驱动
STM32CubeMX WS2812B灯驱动
244 1
|
5月前
STM32CubeMX ULN2003步进电机驱动
STM32CubeMX ULN2003步进电机驱动
76 0
|
5月前
STM32Cubemx TB6612直流电机驱动
STM32Cubemx TB6612直流电机驱动
190 0