基于LORA的一主多从监测系统_0.96OLED

关联:0.96OLED         hal硬件I2C         LORA

在本项目中每个节点都使用oled来显示采集到的数据以及节点状态,OLED使用I2C接口与STM32连接,这个屏幕内部驱动IC为SSD1306,SSD1306作为从机地址为0x78

发送数据:起始信号-从机地址-应答-写数据模式(0x40)-应答-数据(8bit)-结束信号

发送命令:起始信号-从机地址-应答-写命令模式(0x00)-应答-命令(8bit)-结束型号

我这里使用硬件I2C,使用HAL的I2C操作函数HAL_I2C_Mem_Write,这个函数是在阻塞模式下将大量数据写入特定的内存地址,函数原型为:HAL_StatusTypeDef HAL_I2C_Mem_Write(I2C_HandleTypeDef *hi2c, uint16_t DevAddress, uint16_t MemAddress, uint16_t MemAddSize, uint8_t *pData, uint16_t Size, uint32_t Timeout)

参数:1、I2C指针,即用I2C1 还是 I2C2… 2、器件地址uint16_t DevAddress 3、要写入的内存地址 uint16_t MemAddress 4、内存地址类型,是一个地址存8bit ,还是16bit数据 , uint16_t MemAddSize 5、要写入的数组指针uint8_t *pData 6、数据 大小 7、超时时间。下面是用这个函数封装的两个命令发送函数:

/**
* @brief  向OLED寄存器地址写一个byte的数据
* @param  addr:寄存器地址
* @param  data:要写入的数据
* @retval 无
*/
void I2C_WriteByte(uint8_t addr, uint8_t data)
{
    extern I2C_HandleTypeDef hi2c1;
    HAL_I2C_Mem_Write(&hi2c1, OLED_ADDRESS, addr, I2C_MEMADD_SIZE_8BIT, &data, 1, 10);
}

/**
 * ************************************************************************
 * @brief 写命令函数
 * @param[in] cmd  写入的命令
 * ************************************************************************
 */
void WriteCmd(unsigned char cmd)
{
    I2C_WriteByte(0x00, cmd);
}

/**
 * ************************************************************************
 * @brief 写数据函数
 * @param[in] dat  写入的数据
 * ************************************************************************
 */
void WriteDat(unsigned char dat)
{
    I2C_WriteByte(0x40, dat);
}

下面是初始化对一些参数的配置:

void OLED_Init(void)
{
    WriteCmd(0xAE); //显示关闭
    WriteCmd(0x20); //设置内存寻址模式
    WriteCmd(0x10); //00,水平寻址模式;01,垂直寻址模式;10,页寻址模式(复位);11,无效
    WriteCmd(0xb0); //设置页寻址模式的页起始地址,0-7
    WriteCmd(0xc8); //设置COM输出扫描方向
    WriteCmd(0x00); //-设置低列地址
    WriteCmd(0x10); //-设置高列地址
    WriteCmd(0x40); //-设置起始行地址
    WriteCmd(0x81); //设置对比度控制寄存器
    WriteCmd(0xff); //亮度调节 0x00~0xff
    WriteCmd(0xa1); //设置段重新映射0到127
    WriteCmd(0xa6); //设置正常显示
    WriteCmd(0xa8); //设置复用比例(1到64)
    WriteCmd(0x3F); //
    WriteCmd(0xa4); //0xa4,输出遵循RAM内容;0xa5,输出忽略RAM内容
    WriteCmd(0xd3); //设置显示偏移
    WriteCmd(0x00); //不偏移
    WriteCmd(0xd5); //--set display clock divide ratio/oscillator frequency
    WriteCmd(0xf0); //--set divide ratio
    WriteCmd(0xd9); //--set pre-charge period
    WriteCmd(0x22); //
    WriteCmd(0xda); //--set com pins hardware configuration
    WriteCmd(0x12);
    WriteCmd(0xdb); //--set vcomh
    WriteCmd(0x20); //0x20,0.77xVcc
    WriteCmd(0x8d); //设置DC-DC使能
    WriteCmd(0x14); //
    WriteCmd(0xaf); //--turn on oled panel
    OLED_CLS();
}

我们不需要去研究这个具体每项配置的作用,我们只需要关注如何显示我们所需要的,我这里提供三个接口函数,分别用来显示汉字、字符、数字,具体方法如下:

/**
 * ************************************************************************
 * @brief 中文汉字显示函数
 *
 * @param[in] x     起始点横坐标(0~127)
 * @param[in] y     起始点纵坐标(0~63)
 * @param[in] ch    汉字字模库索引
 *
 * @example OLED_ShowCN(0,0,"字");
 * ************************************************************************
 */
void OLED_ShowChinese(signed short int x, signed short int y, unsigned char* ch)
{
    if (x >= 0 && x < SCREEN_COLUMN && y >= 0 && y < SCREEN_ROW) {
        int32_t  len = 0,offset = sizeof(F16x16_CN[0].index);

        while(ch[len] != '\0')
        {
            if(x >= 127 || (127-x < 16))//8个汉字显示||剩余列小于16不能显示完整字符,换行显示
            {
                x = 0;
                y += 16;
                if(63 - y < 16) // 不足以显示一行时不显示
                    break;
            }

            //需要处理输入数据大于显示数据的问题
            for(unsigned char i = 0; i < sizeof(F16x16_CN)/sizeof(GB2312_CN); i++)
            {
                if(((F16x16_CN[i].index[0] == ch[len]) && (F16x16_CN[i].index[1] == ch[len+1]))){
                        for(unsigned char m = 0; m < 2; m++)    //页
                        {
                            for(unsigned char n = 0; n < 16; n++) // 列
                            {
                                for(unsigned char j = 0; j < 8; j++)    // 行
                                {
                                    OLED_SetPixel(x+n, y+j+m*8, (F16x16_CN[i].encoder[n+m*16] >> j) & 0x01);
                                }
                            }
                        }
                        x += 16;
                        len += offset;
                        break;
                }
                else if(F16x16_CN[i].index[0] == ch[len] && ch[len] == 0x20){
                    for(unsigned char m = 0; m < 2; m++)
                    {
                        for(unsigned char n = 0; n < 16; n++)
                        {
                            for(unsigned char j = 0; j < 8; j++)
                            {
                                OLED_SetPixel(x+n, y+j+m*8, (F16x16_CN[i].encoder[n+m*16] >> j) & 0x01);
                            }
                        }
                    }
                    x += 16;
                    len++;
                    break;
                }
            }
        }
    }
    OLED_RefreshRAM();
}


/**
 * ************************************************************************
 * @brief BMP图片显示函数
 *
 * @param[in] x0    起始点横坐标(0~127)
 * @param[in] y0    起始点纵坐标(0~63)
 * @param[in] L     BMP图片宽度
 * @param[in] H     BMP图片高度
 * @param[in] BMP   图片取模地址
 *
 * @example OLED_ShowBMP(0,0,52,48,(unsigned char *)astronaut_0);
 * ************************************************************************
 */
void OLED_ShowBMP(signed short int x0,signed short int y0,signed short int L,signed short int H,const unsigned char BMP[])
{
    if (x0 >= 0 && x0 < SCREEN_COLUMN && x0+L <= SCREEN_ROW &&\
        y0 >= 0 && y0 < SCREEN_COLUMN && y0+H <= SCREEN_ROW) {

        unsigned char *p = (unsigned char *)BMP;
        for(signed short int y = y0; y < y0+H; y+=8)
        {
            for(signed short int x = x0; x < x0+L; x++)
            {
                for(signed short int i = 0; i < 8; i++)
                {
                    OLED_SetPixel(x, y+i, ((*p) >> i) & 0x01);
                }
                p++;
            }
        }
    }

    OLED_RefreshRAM();
}



/**
 * ************************************************************************
 * @brief 数字显示函数
 *
 * @param[in] x          起始点横坐标(0~127)
 * @param[in] y          起始点纵坐标(0~63)
 * @param[in] number     要显示的数字(可以是整数或浮点数)
 * @param[in] TextSize   字符大小(1:6*8;2:8*16)
 * @param[in] decimalPlaces   小数位数(例如:2 表示显示两位小数)
 *
 * ************************************************************************
 */
void OLED_ShowNumber(signed short int x, signed short int y, float number, unsigned char TextSize, unsigned char decimalPlaces)
{
    char buffer[20]; // 预留空间以存放数字转换为字符串后的结果,包括符号和终止符
    // 构造格式字符串,%.*f 表示动态设置小数位数
    sprintf(buffer, "%.*f", decimalPlaces, number); // 将浮点数转换为字符串
    OLED_ShowStr(x, y, (unsigned char *)buffer, TextSize); // 调用显示字符串的函数
}

通过以上接口函数,我们可以控制在屏幕上显示我们想要显示的东西

void oledUIShow(void){
    OLED_ShowStr(0, 0, "DEVICE.1", 1);
    OLED_ShowStr(62, 0, "ADDR:0x0A", 1);
    OLED_ShowChinese(0, 16, "温度");
    OLED_ShowChinese(68, 16, "湿度");
    OLED_ShowStr(118, 16, "%", 2);
    OLED_ShowChinese(0, 40, "光照");
    OLED_ShowChinese(54, 40, "气压");
}

效果如下图所示:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值