打印
[其他ST产品]

stm32c8t6矩阵控制led亮灯 显示时间在4线oled屏幕

[复制链接]
1430|38
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主

直接上代码文章末尾讲解 注文章篇幅较长可能需要点阅读时间

首先是key.h文件

 
#define KEY_HANG1 GPIO_Pin_0
#define KEY_HANG2 GPIO_Pin_1
#define KEY_HANG3 GPIO_Pin_2
#define KEY_HANG4 GPIO_Pin_3


#define KEY1  GPIO_Pin_4
#define KEY2  GPIO_Pin_5
#define KEY3  GPIO_Pin_6
#define KEY4  GPIO_Pin_7


#define KEY1_Input  GPIO_ReadInputDataBit(KEY_PROT,KEY1)
#define KEY2_Input  GPIO_ReadInputDataBit(KEY_PROT,KEY2)
#define KEY3_Input  GPIO_ReadInputDataBit(KEY_PROT,KEY3)
#define KEY4_Input  GPIO_ReadInputDataBit(KEY_PROT,KEY4)


#define KEY_PROT GPIOA


extern  u8  flag;  
u16 *abc_read();
int sun1_read();

void KEY_Init(void);//IO初始化
void KEY_config(u8 *key);
u8 Read_KeyValue(void);
void Key_Input( u8 *key1);




使用特权

评论回复
评论
9dome猫 2022-8-22 23:20 回复TA
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 本文链接:https://blog.csdn.net/shihuntz/article/details/123933145 ———————————————— 版权声明:本文为CSDN博主「shihuntz」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.csdn.net/shihuntz/article/details/123933145 
沙发
9dome猫|  楼主 | 2022-8-22 23:23 | 只看该作者
oled.h文件
#ifndef __OLED_H
#define __OLED_H

#include "sys.h"
#include "stdlib.h"       

//-----------------OLED端口定义----------------



#define OLED_SCL_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_1)//SCL   A 0
#define OLED_SCL_Set() GPIO_SetBits(GPIOB,GPIO_Pin_1)

#define OLED_SDA_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_10)//DIN   A 1
#define OLED_SDA_Set() GPIO_SetBits(GPIOB,GPIO_Pin_10)

#define OLED_RES_Clr() GPIO_ResetBits(GPIOB,GPIO_Pin_7)//RES   A 2
#define OLED_RES_Set() GPIO_SetBits(GPIOB,GPIO_Pin_7)


#define OLED_CMD  0        //写命令
#define OLED_DATA 1        //写数据


extern  u8  flag;  

void KEY_Init(void);//IO初始化
void KEY_config(u8 *key);
u8 Read_KeyValue(void);
void Key_Input( u8 *key1);




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);

#endif


使用特权

评论回复
板凳
9dome猫|  楼主 | 2022-8-22 23:30 | 只看该作者
oled.c
#include "oled.h"
#include "stdlib.h"
#include "oledfont.h"
#include "delay.h"
#include "stm32f10x.h"
#include "usart.h"

u8 OLED_GRAM[144][8];



//反显函数
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)
{
        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=Hzk1[num][i];}//调用16*16字体
                else if(size1==24)
                                {temp=Hzk2[num][i];}//调用24*24字体
                else if(size1==32)      
                                {temp=Hzk3[num][i];}//调用32*32字体
                else if(size1==64)
                                {temp=Hzk4[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_10|GPIO_Pin_1;         
        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_10|GPIO_Pin_1);

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;         
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;                  //推挽输出              
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz                     
        GPIO_Init(GPIOB, &GPIO_InitStructure);          //初始化PA2                  
        GPIO_SetBits(GPIOB,GPIO_Pin_7);
       
        OLED_RES_Clr();
        delay_ms(200);
        OLED_RES_Set();
       
        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);
}


使用特权

评论回复
地板
9dome猫|  楼主 | 2022-8-22 23:40 | 只看该作者
key.c文件
#include "stm32f10x.h"
#include "key.h"
#include "sys.h"
#include "delay.h"
#include "usart.h"


//按键初始化函数
void KEY_Init(void) //IO初始化
{
        GPIO_InitTypeDef GPIO_InitStructure;

        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB,ENABLE);

        GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_5;  
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //推挽输出
        GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化
        GPIO_SetBits(GPIOB,GPIO_Pin_5);  
       
        GPIO_InitStructure.GPIO_Pin  = KEY1|KEY2|KEY3|KEY4;   //行  a  4 5 6 7
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //PA0设置成输入,默认上拉  
        GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.0
       
       
        GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;  //lie  
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;                                                                                                         //a 4 5 6 7
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
        GPIO_Init(GPIOA, &GPIO_InitStructure);  
        GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3);       //a 0 1 2 3
}



u8 Key = 0;
u8 flag=0;
u16 abc[9] = {0,0,0,0,0,0,0,0,0};

u8 i = 0;
u8 Flag = 0;
int sum1 = 0;

int sun1_read()
{
   return sum1;
}

u16 * abd_read();


u16 *abc_read()
{
                return &abc[0];
}


int Wei_Shu(u8 a)
{
        int sum = 1, i = 0;
        for(;i<a;i++)
        {
        sum =        sum*10;
        }
        return sum;
}


u16 time(u16 abc[9],u8 j)
{       
        int time = 0 ,a = j ,shijian = 0;
                for(time = 0; time < j; time++ )
        {
                shijian +=  abc[time]*Wei_Shu(a--);  
        }
        return shijian;               
}

char ok = 0;
char led_liang = 0;
void Key_Input( u8 *key1)
{       
        if(ok==1){
       
       
                if(*key1 == 16 && Flag == 0)
        {
                Flag = 1;
        }
        else if(Flag == 1)
        {               
               
         if(*key1 == 16 && Flag == 1)
        {
                Flag = 0;
               
                        sum1 = (short)(time(abc,i)/10);
                        printf(" shijian =%d\r\n",sum1);
               
                         if(0<sum1)
                        {
                                sum1 = sum1*1000;
                                led_liang = 1;
                        }
                        //
                for(i=0;i<9;i++)
                        abc[i] = 0;       
                i = 0;       
         }       
                        else if(*key1<10)
                                {
                                        abc[i] = *key1;
                                        i++;
                                                if(i>9)
                                                {
                                                         i = 0;
                                          }
                }
        }
       
        else if(*key1 == 13){
                sum1 = 0;
                                        //
                for(i=0;i<9;i++)
                        abc[i] = 0;       
                i = 0;       
        }
       
       
        ok = 0;
}
       
                if(led_liang==1 && sum1>0){
                                GPIO_SetBits(GPIOB,GPIO_Pin_5);       
                                //delay_ms(1);               
                                sum1--;                       
                        }
               
                        if(sum1==0){
                                led_liang = 0;
                                GPIO_ResetBits(GPIOB,GPIO_Pin_5);       
                        }
               


}

void KEY_config(u8 *Key)
{
        char i=1,j=1;
        *Key = 0;
        for(;i<5;i++){
                         GPIO_ResetBits(GPIOA,GPIO_Pin_0<<(i-1));     //先拉低
                        if((KEY1_Input == RESET) || (KEY2_Input == RESET) || (KEY3_Input == RESET) || (KEY4_Input == RESET))
                        {
                                        delay_ms(10);
                                        for(j=1;j<5;j++){
                                                if(GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1)) == RESET){
                                                        flag = 1;
                                                        *Key = (i-1)*4+j;
                                                        while(!GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1))){};
                                                        //Key_Input(Key);
                                                                ok = 1;
                                                        break;                                               
                                                }
                                                else
                                                        flag = 0;
                                        }
                  }
                        GPIO_SetBits(GPIOA,GPIO_Pin_0<<(i-1)); //再次拉高
         }
}


使用特权

评论回复
5
9dome猫|  楼主 | 2022-8-22 23:43 | 只看该作者
main.c函数
#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
#include "exti.h"
#include "beep.h"
#include "timer.h"
#include "stm32f10x_tim.h"
#include "oled.h"


u8 Key000=0;

//定时器3中断服务程序
void TIM3_IRQHandler(void)   //TIM3中断
{
        if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)  //检查TIM3更新中断发生与否
                {
                TIM_ClearITPendingBit(TIM3, TIM_IT_Update);  //清除TIMx更新中断标志
                        KEY_config(&Key000);
                        Key_Input(&Key000);
                }
}

char a[9]={0,0,0,0,0,0,0,0,0};
char *abd_xiugai(u16 abc[9]){
        char i = 0;
  while(abc[i]!=0){
                a[i] = abc[i]  +  48;
                i++;
                if(i==7)
                        break;
                }
        return a;
}
         
char b[5];
char *sun1_read_xiu_acill(int a){
        //1234000s
       
         a = a/1000;
       
         b[3] = (char)(a%10+48);
         b[2] = (char)((a/10)%10)+48;
         b[1] = (char)((a/100)%10)+48;
         b[0] = (char)(a/1000)+48;
         b[4] = 0;
         return b;
}

int main(void)
{               
        u8 is_anxia = 0;
        delay_init();                     //延时函数初始化         
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
        uart_init(4800);         //串口初始化4800
         KEY_Init();
        //LED_Init();                                  //初始化与LED连接的硬件接口
        //LED0=0;                                        //先点亮红灯
        TIM3_Int_Init(89999,2);
        OLED_Init();
        OLED_ColorTurn(0);//0正常显示,1 反色显示
  OLED_DisplayTurn(0);//0正常显示 1 屏幕翻转显示

        while(1)
        {                        

                OLED_Clear();
               
                if(*(abc_read()+0) == 0 && sun1_read()==0)
                        OLED_ShowString(10,0,"P 16",16,1);
                else if(sun1_read()==0)
                        OLED_ShowString(10,0,abd_xiugai(abc_read()),16,1);
                else if(sun1_read()!=0)
                        OLED_ShowString(10,0,sun1_read_xiu_acill(sun1_read()),16,1);

               
                OLED_Refresh();
        }
        }

/*
void TIM3_IRQHandler(void)   //TIM3中断
{
        if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)  //检查TIM3更新中断发生与否
                {
                TIM_ClearITPendingBit(TIM3, TIM_IT_Update  );  //清除TIMx更新中断标志
       
                                Key_Input(&key);
                }
}*/

使用特权

评论回复
6
9dome猫|  楼主 | 2022-8-22 23:46 | 只看该作者
先来个总述

接线方式  矩阵键盘的8个引脚对应GPIOA 0---7

                led灯对应 GPIOB 5     可以替换为一个LS298N模块控制电机 或者其他的很多东西

                 oled屏幕对应接线 scl - GPIO B1

                                               sda-GPIOB7

                                                VCC和GND随意了

使用特权

评论回复
7
9dome猫|  楼主 | 2022-8-22 23:47 | 只看该作者
接下来是对此代码的详解

        首先是对于key.h 文件的解释我们对于矩阵键盘对应IO口的一个声明,之后我们可以将对应的IO口进行简化方便使用,也使我们的代码更有可读性。

对于oled.h也是同理,这就不在啰嗦了,应该都懂吧

使用特权

评论回复
8
9dome猫|  楼主 | 2022-8-22 23:48 | 只看该作者
对于key.c 文件是本文章的核心部分

        首先是对于矩阵键盘八个对应引脚的初始化这里我们选择扫描的方式来进行按键检测,但是之后我遇到了一个大问题 那就是延时的问题 会发现在我们使用oled后,时间会变得不可控。

使用特权

评论回复
9
9dome猫|  楼主 | 2022-8-22 23:50 | 只看该作者
在这里我们对于八个IO口选择 4个推挽输出 4个上拉输出的模式  在这里我将led灯对应的IO口初始化也放在这里了。对于io口的初始化就不在详细的说明 ,没什么好说的 只需要注意和.h文件的对应关系

使用特权

评论回复
10
9dome猫|  楼主 | 2022-8-22 23:51 | 只看该作者
之后我们来说说

void KEY_config(u8 *Key)

使用特权

评论回复
11
9dome猫|  楼主 | 2022-8-22 23:52 | 只看该作者
对于此函数需要一丁点的位操作知识 用到了一点<<    >>的操作
 
void KEY_config(u8 *Key)
{
        char i=1,j=1;
        *Key = 0;
        for(;i<5;i++){
                         GPIO_ResetBits(GPIOA,GPIO_Pin_0<<(i-1));     //先拉低
                        if((KEY1_Input == RESET) || (KEY2_Input == RESET) || (KEY3_Input == RESET) || (KEY4_Input == RESET))
                        {
                                        delay_ms(10);
                                        for(j=1;j<5;j++){
                                                if(GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1)) == RESET){
                                                        flag = 1;
                                                        *Key = (i-1)*4+j;
                                                        while(!GPIO_ReadInputDataBit(KEY_PROT,GPIO_Pin_4<<(j-1))){};
                                                        //Key_Input(Key);
                                                                ok = 1;
                                                        break;                                               
                                                }
                                                else
                                                        flag = 0;
                                        }
                  }
                        GPIO_SetBits(GPIOA,GPIO_Pin_0<<(i-1)); //再次拉高
         }
}

使用特权

评论回复
12
9dome猫|  楼主 | 2022-8-22 23:53 | 只看该作者

使用特权

评论回复
13
9dome猫|  楼主 | 2022-8-22 23:54 | 只看该作者
我们在初始化时将4路IO口设置为默认上拉  也就是一直输出高电平

所以我们只需要循环检测 电平的高低 如果有按键按下 那么图上边的一排1 必然有一路变为低电平

我们只需要在循环扫描一列4路哪一路为低电平  如果2路都为低电平则可以确认按键的位置 相当于在按键位置花了一个十字 确认位置。

使用特权

评论回复
14
9dome猫|  楼主 | 2022-8-22 23:55 | 只看该作者
好了原理已经说清了,借来下进入我们的代码部分

我们循环拉低四行的电平 那么比如我们再按下第一个按键时可以知道第一列的电平也会被拉低

这就对应了我们的前五行代码

之间插入一个按键消抖操作

使用特权

评论回复
15
Uriah| | 2022-9-30 10:30 | 只看该作者

待向GPIO(通用I/O端口)的输入从0变为1时,程序可以一定的间隔来检查GPIO的状态

使用特权

评论回复
16
Bblythe| | 2022-9-30 13:29 | 只看该作者

CPLD解密,DSP解密都习惯称为单片机解密

使用特权

评论回复
17
saservice| | 2022-10-1 12:56 | 只看该作者
stm32驱动oled如何实现反色显示表示选中功能

使用特权

评论回复
18
kkzz| | 2022-10-1 13:26 | 只看该作者
stm32怎样才能同时驱动OLED和LCD两个屏

使用特权

评论回复
19
qiufengsd| | 2022-10-1 13:52 | 只看该作者
stm32硬件IIC不好用               

使用特权

评论回复
20
jkl21| | 2022-10-1 14:25 | 只看该作者
stm32 怎么向oled输出数据

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

133

主题

1406

帖子

2

粉丝