1.epaper-drive.h
#ifndef EPD_DRIVER_H
#define EPD_DRIVER_H
#include <stdint.h>
/**
* @brief 初始化墨水屏
*/
void epd_init(void);
/**
* @brief 设置局部刷新窗口
* @param x 起始X坐标
* @param y 起始Y坐标
* @param w 宽度
* @param h 高度
*/
void epd_set_partial_window(uint16_t x, uint16_t y, uint16_t w, uint16_t h);
/**
* @brief 向局部区域写入数据
* @param data 数据缓冲区指针
* @param size 数据大小
*/
void epd_write_partial_data(const uint8_t *data, uint32_t size);
/**
* @brief 执行局部刷新显示
*/
void epd_partial_display_update(void);
#endif // EPD_DRIVER_H
2.epaper_drive.c代码
#include "nrf_drv_spi.h"
#include "nrf_gpio.h"
#include "nrf_delay.h"
#include "app_error.h"
// 定义引脚连接
#define EPD_SPI_INSTANCE 0
#define EPD_CS_PIN 2
#define EPD_DC_PIN 3
#define EPD_RESET_PIN 4
#define EPD_BUSY_PIN 5
// 屏幕尺寸定义
#define EPD_WIDTH 296
#define EPD_HEIGHT 128
static const nrf_drv_spi_t spi = NRF_DRV_SPI_INSTANCE(EPD_SPI_INSTANCE);
// 初始化SPI通信
static void spi_init(void)
{
nrf_drv_spi_config_t spi_config = NRF_DRV_SPI_DEFAULT_CONFIG;
spi_config.ss_pin = EPD_CS_PIN;
spi_config.miso_pin = NRF_DRV_SPI_PIN_NOT_USED;
spi_config.frequency = NRF_SPI_FREQ_4M;
APP_ERROR_CHECK(nrf_drv_spi_init(&spi, &spi_config, NULL, NULL));
}
// 发送命令
static void epd_send_command(uint8_t command)
{
nrf_gpio_pin_clear(EPD_DC_PIN);
nrf_drv_spi_transfer(&spi, &command, 1, NULL, 0);
}
// 发送数据
static void epd_send_data(uint8_t data)
{
nrf_gpio_pin_set(EPD_DC_PIN);
nrf_drv_spi_transfer(&spi, &data, 1, NULL, 0);
}
// 等待墨水屏就绪
static void epd_wait_until_idle(void)
{
while(nrf_gpio_pin_read(EPD_BUSY_PIN) == 1) {
nrf_delay_ms(10);
}
nrf_delay_ms(200);
}
// 墨水屏硬件复位
static void epd_reset(void)
{
nrf_gpio_pin_set(EPD_RESET_PIN);
nrf_delay_ms(200);
nrf_gpio_pin_clear(EPD_RESET_PIN);
nrf_delay_ms(10);
nrf_gpio_pin_set(EPD_RESET_PIN);
nrf_delay_ms(200);
}
// 初始化墨水屏
void epd_init(void)
{
// 配置控制引脚
nrf_gpio_pin_dir_set(EPD_CS_PIN, NRF_GPIO_PIN_DIR_OUTPUT);
nrf_gpio_pin_dir_set(EPD_DC_PIN, NRF_GPIO_PIN_DIR_OUTPUT);
nrf_gpio_pin_dir_set(EPD_RESET_PIN, NRF_GPIO_PIN_DIR_OUTPUT);
nrf_gpio_pin_dir_set(EPD_BUSY_PIN, NRF_GPIO_PIN_DIR_INPUT);
// 初始化SPI
spi_init();
// 复位并初始化屏幕
epd_reset();
epd_send_command(0x12); // 软复位
epd_wait_until_idle();
epd_send_command(0x01); // 设置驱动电压
epd_send_data(0x03);
epd_send_data(0x00);
epd_send_data(0x2b);
epd_send_data(0x2b);
epd_send_command(0x3C); // 边框波形
epd_send_data(0x01);
// 其他初始化命令...
}
// 设置局部刷新区域
void epd_set_partial_window(uint16_t x, uint16_t y, uint16_t w, uint16_t h)
{
epd_send_command(0x44); // 设置RAM X地址开始/结束位置
epd_send_data((x >> 3) & 0xFF);
epd_send_data(((x + w - 1) >> 3) & 0xFF);
epd_send_command(0x45); // 设置RAM Y地址开始/结束位置
epd_send_data(y & 0xFF);
epd_send_data((y >> 8) & 0xFF);
epd_send_data((y + h - 1) & 0xFF);
epd_send_data(((y + h - 1) >> 8) & 0xFF);
epd_send_command(0x4E); // 设置RAM X地址计数器
epd_send_data((x >> 3) & 0xFF);
epd_send_command(0x4F); // 设置RAM Y地址计数器
epd_send_data(y & 0xFF);
epd_send_data((y >> 8) & 0xFF);
}
// 局部刷新显示
void epd_partial_display_update(void)
{
epd_send_command(0x22); // 显示更新控制2
epd_send_data(0x0F); // 启用局部刷新
epd_send_command(0x20); // 触发显示更新
epd_wait_until_idle();
}
// 向局部区域写入数据
void epd_write_partial_data(const uint8_t *data, uint32_t size)
{
epd_send_command(0x24); // 向RAM写入黑色/白色数据
for (uint32_t i = 0; i < size; i++) {
epd_send_data(data[i]);
}
}
3.main.c代码
#include "nrf_drv_rtc.h"
#include "nrf_drv_clock.h"
#include "nrf_gpio.h"
#include "nrf_delay.h"
#include "epaper_driver.h"
#include <stdio.h>
static const nrf_drv_rtc_t rtc = NRF_DRV_RTC_INSTANCE(0);
static char time_str[16];
static uint8_t current_hour = 255;
static uint8_t current_minute = 255;
static uint8_t current_second = 255;
// RTC中断处理函数
static void rtc_handler(nrf_drv_rtc_int_type_t int_type)
{
if (int_type == NRF_DRV_RTC_INT_TICK)
{
uint32_t seconds = nrf_drv_rtc_counter_get(&rtc);
uint8_t hour = (seconds / 3600) % 24;
uint8_t minute = (seconds / 60) % 60;
uint8_t second = seconds % 60;
// 仅在分钟变化时更新显示,减少刷新次数
if (minute != current_minute)
{
current_hour = hour;
current_minute = minute;
current_second = second;
// 更新时钟显示
snprintf(time_str, sizeof(time_str), "%02d:%02d:%02d", hour, minute, second);
update_clock_display(time_str);
}
}
}
// 初始化RTC实时时钟
static void rtc_init(void)
{
nrf_drv_rtc_config_t config = NRF_DRV_RTC_DEFAULT_CONFIG;
config.prescaler = 32768/1 - 1; // 1Hz时钟
APP_ERROR_CHECK(nrf_drv_rtc_init(&rtc, &config, rtc_handler));
nrf_drv_rtc_tick_enable(&rtc, true);
nrf_drv_rtc_enable(&rtc);
}
// 更新时钟显示
static void update_clock_display(const char *time_str)
{
// 清屏
epd_set_partial_window(0, 0, EPD_WIDTH, EPD_HEIGHT);
// 准备时钟显示数据
uint8_t clock_buffer[EPD_WIDTH * EPD_HEIGHT / 8] = {0};
// 在缓冲区上绘制时钟(这里简化为文字显示,实际应用需要字模库)
draw_text(clock_buffer, 50, 80, time_str, FONT_SIZE_24);
// 更新显示
epd_write_partial_data(clock_buffer, sizeof(clock_buffer));
epd_partial_display_update();
}
// 绘制文字(简化版,实际应用需要字模库支持)
static void draw_text(uint8_t *buffer, uint16_t x, uint16_t y, const char *text, uint8_t font_size)
{
// 这里需要根据字模库实现文字绘制
// 简化示例,实际应用需要替换为真实的字模绘制代码
// ...
}
int main(void)
{
// 初始化外设
nrf_drv_clock_init();
nrf_drv_clock_lfclk_request(NULL);
while(!nrf_drv_clock_lfclk_is_running()) {}
// 初始化墨水屏
epd_init();
// 初始化RTC
rtc_init();
// 显示初始时间
update_clock_display("00:00:00");
// 进入低功耗模式
while(1)
{
__WFI();
}
}