LVGL庫入門教程01-移植到STM32(觸摸屏)

来源:https://www.cnblogs.com/frozencandles/archive/2022/06/13/16370160.html
-Advertisement-
Play Games

鏡像下載、功能變數名稱解析、時間同步請點擊 阿裡雲開源鏡像站 ifconfig 命令可以用於查看、配置、啟用或禁用指定的網路介面,還可以用來配置網卡的IP地址、掩碼、廣播地址、網關等,功能很豐富 功能雖然豐富,但是如果你沒有安裝呢? 嘿嘿嘿 如果沒有安裝ifconfig管理命令的話,直接運行ifconfig ...


LVGL庫移植STM32

LVGL庫簡介

LVGL(Light and Versatile Graphics Library)是一個免費、開源的嵌入式圖形庫,可以創建豐富、美觀的界面,具有許多可以自定義樣式的控制項,支持按鍵或觸摸響應,支持中文字元,並且記憶體占用較低。可以在 https://lvgl.io/demos 使用網頁端體驗 LVGL 的動態效果,再決定是否需要使用 LVGL 。

LVGL 使用 C 語言編寫,可以用在樹莓派、ESP32 、STM32 等單片機上,並支持各種中大型屏幕(只需要提供屏幕的繪圖 API 即可)。LVGL 的官網地址為:https://lvgl.io/ ,GitHub 地址為:https://github.com/lvgl/lvgl

LVGL 提供了許多示常式序,還提供了 PC 端的模擬器,這都加快了 LVGL 的開發效率。

移植LVGL

LVGL 並沒有隻針對哪一個單片機和哪一個屏幕,事實上它甚至在 PC 機也能運行的起來。完整的 LVGL 的移植可以參考官方文檔的介紹 https://docs.lvgl.io/master/porting/index.html 。在移植前,請自行瞭解單片機及屏幕的使用方法並提供介面程式。

建立工程

接下來以 STM32 系列單片機為例介紹 LVGL 的移植,不同單片機的移植過程也可以參考以下步驟。下表給出了 LVGL 所需的配置,在使用 LVGL 前請確保單片機性能滿足要求:

Name Minimal Recommended
Architecture 16, 32 or 64 bit microcontroller or processor
Clock > 16 MHz > 48 MHz
Flash/ROM > 64 kB > 180 kB
Static RAM > 16 kB > 48 kB
Draw buffer > 1 ×hor. res. pixels > 1/10 screen size
Compiler C99 or newer

註意:使用 Keil5 請開啟“C99 Mode”,否則會編譯不通過。還在使用 Keil4 的請升級或更換編譯器。

首先,在 https://github.com/lvgl/lvgl 下載或克隆整個工程。LVGL 的最新版本是 LVGL 8.2 ,註意 LVGL 7 已經不再更新,LVGL 7 和 8 之間庫結構發生較大改變,編寫出的代碼並不很相容,並且 LVGL 7 的示例代碼和模擬器似乎已經在 GitHub 上下架了。本教程以 LVGL 8 為例,移植 LVGL 7 的話可以參考,但一些細節需要註意調整。建議使用最新版本,否則無法得到完整的工具鏈支持。

使用 Keil 的開發者請註意,LVGL 8 似乎不能在 ARM CC v5 下編譯成功,請更新編譯器版本為 ARM CC v6 。

使用 STM32 的開發者還需要註意,STM32 標準庫無法使用 ARM CC v6 編譯,請使用 HAL 庫或更換編譯工具鏈(如 LLVM-clang 或 GCC-none-eabi )

接下來自行準備一個單片機工程,在 User 或其它等效的目錄中,然後新建目錄 lvgl 併進入,從克隆得到的 LVGL 工程中複製以下文件或目錄到其中:

demos
examples
src
lvgl.h
lv_conf_template.h

如果不需要使用官方提供的示例代碼,可以不複製 demos 目錄。

接下來,將 lv_conf_template.h 重命名為 lv_conf.h ,並移動到上一級目錄中。

註意:LVGL 庫的目錄比較複雜,頭文件引用相對混亂,在沒有充分明白正在做什麼之前,請不要隨意修改文件夾名或變更文件位置。

回到上一級目錄,打開 lv_conf.h ,將開頭的 #if 0 條件編譯取消,啟用文件包含的配置:

/* clang-format off */
#if 1 /*Set it to "1" to enable content*/

該配置文件還有幾處需要調整的地方,首先最前面(第 27 行)的一個巨集定義表示顯示屏的顏色深度,需要根據不同的顯示屏做調整:

/*Color depth: 1 (1 byte per pixel), 8 (RGB332), 16 (RGB565), 32 (ARGB8888)*/
#define LV_COLOR_DEPTH 16

如果屏幕的顏色深度不一致,一定要修改該巨集。LVGL 會根據該巨集創建合適的顏色定義,如果與實際不一致會造成顯示時顏色錯亂。

如果設置為 8 ,代表使用 8 位的顏色,其中 RBG 色值各占 3 、3 、2 位;如果設置為 16 ,則 RBG 色值各占 5 、6 、5 位,這是許多 TFT 屏採用的顏色格式;32 則是 PC 機和移動設備都使用的帶透明度的 32bit 點陣圖,RGB 色值和透明度各占一個位元組。

第 52 行中還有一處表示最大占用記憶體量的巨集,可以根據實際單片機的情況自行修改,只要保證大於註釋中寫的 2kB 就行。

    /*Size of the memory available for `lv_mem_alloc()` in bytes (>= 2kB)*/
    #define LV_MEM_SIZE (32U * 1024U)          /*[bytes]*/

除此之外,在第 273 和 280 行還有這麼兩個巨集定義,如果將它們設置為 1 ,那麼可以在屏幕的左下角和右下角顯示當前記憶體占用和幀率,非常適合性能分析:

#define LV_USE_PERF_MONITOR 0
#define LV_USE_MEM_MONITOR 0

其它設置可以對照註釋和文檔修改。

接下來開始導入工程文件,這一步需要將 lvgl/src 中除了 draw 目錄中的所有文件全部導入,而 draw 目錄中除了根目錄的 .c 文件外,只導入 sw 目錄中的源文件。LVGL 8 的目錄深度較大,請耐心添加,細心檢查,不要遺漏文件。

使用 STM32 單片機的話還需要註意在啟動文件中修改堆、棧大小,至少各設置 8kB 空間:

Stack_Size      EQU     0x00002000
Heap_Size       EQU     0x00002000

全部添加完成之後,嘗試編譯整個工程,應該是可以零 error 通過了。

使用 ARM CC v6 可能會發生 __aeabi_assert 符號未定義的問題,可以在整個項目管理中提前定義巨集 NDEBUG 禁用該符號。

顯示設備的API對接

LVGL 只提供了繪圖的演算法,其它內容需要自行編寫。LVGL 提供的介面在 lvgl/examples/porting 目錄中,該目錄有如下文件:

  • lv_port_disp :顯示設備介面
  • lv_port_indev :輸入設備介面
  • lv_port_fs :文件系統介面

將各個文件名結尾的 template 去除。接下來先編寫顯示設備的介面,至少確保能顯示一些東西來。


lv_port_disp.c 及其頭文件中,首先需要去除條件編譯,啟用這部分內容:

/*Copy this file as "lv_port_disp.h" and set this value to "1" to enable content*/
#if 1

由於之前重命名過頭文件,因此在源文件中也需要修改對應的名稱:

#include "lv_port_disp.h"

源文件在巨集定義區域中有兩個巨集定義,需要修改為實際的顯示屏尺寸。改過了之後記得把 #warning 預處理語句去除了:

#ifndef MY_DISP_HOR_RES
    //#warning Please define or replace the macro MY_DISP_HOR_RES with the actual screen width, default value 320 is used for now.
    #define MY_DISP_HOR_RES    320
#endif
/* ... same as above ... */

lv_port_disp_init() 是一個最頂層的初始化顯示設備的函數,在主函數中需要調用它一次性初始化顯示設備的功能。該函數的修改方式註釋里已經寫的較為清楚了,接下來提供一個修改示例。

首先將 91~102 行的兩個提供顯示緩存的語句全部註釋或刪除,只保留 /* Example for 1) */ 。然後修改 114~115 行的兩個數值為實際的屏幕清晰度。

    /* Example for 1) */
    static lv_disp_draw_buf_t draw_buf_dsc_1;
    static lv_color_t buf_1[MY_DISP_HOR_RES * 10];                          /*A buffer for 10 rows*/
    lv_disp_draw_buf_init(&draw_buf_dsc_1, buf_1, NULL, MY_DISP_HOR_RES * 10);   /*Initialize the display buffer*/

    /*Set the resolution of the display*/
    disp_drv.hor_res = 320;
    disp_drv.ver_res = 240;

該文件內還有兩個函數 disp_init()disp_flush() ,需要提供實際顯示設備的介面。

disp_init() 中,需要提供屏幕的初始化代碼,如果已經在外部初始化過可以忽略。

disp_flush() 中,需要在註釋的位置根據提供的參數繪製一個像素點,。這一過程也可以使用填充函數獲得更快的速度,甚至可以使用 GPU 等加速等方式完成,具體如何編寫代碼可以參考註釋。例如,測試用的屏幕是這樣逐個繪製像素點,從而填充一塊區域的:

/* ... */
    for(y = area->y1; y <= area->y2; y++) {
        for(x = area->x1; x <= area->x2; x++) {
            ILI9341_SetFrontColor(&ili9341, color_p->full);
            ILI9341_DrawPixel(&ili9341, x, y);
            color_p++;
        }
    }
/* ... */

至此,API 移植便結束了。接下來可以編寫程式測試 LVGL 的效果了。

LVGL的初始化

在使用 LVGL 前,需要調用以下兩個函數完成 LVGL 庫的初始化以及 LVGL 顯示設備介面的初始化:

lv_init();
lv_port_disp_init();

然後就可以繪製圖形了。這裡提供了一段簡單的代碼,可以繪製一個按鈕:

lv_obj_t* btn = lv_btn_create(lv_scr_act()); 
lv_obj_set_pos(btn, 10, 10);
lv_obj_set_size(btn, 120, 50);
lv_obj_t* label = lv_label_create(btn);
lv_label_set_text(label, "Button");
lv_obj_center(label);

繪製完之後,還需要在主迴圈中調用 lv_task_handler() 函數,這樣繪製的內容才能實時更新到屏幕上:

while (1) {
    /* ... */
    lv_task_handler();
}

然後將編譯得到的結果下載到單片機內,就可以在屏幕上看到一個按鈕了:

image

LVGL輸入設備移植

上文介紹瞭如何移植顯示設備。但是 LVGL 是一個用戶界面庫,光有顯示設備,不能做一些用戶交互的功能還是不太夠,因此就需要使用輸入設備。

LVGL 支持 5 種類型的輸入設備:

  • Touchpad :觸摸屏
  • Mouse :滑鼠
  • Keypad :鍵盤
  • Encoder :編碼器
  • Button :按鍵

在移植時,不要搞錯了輸入設備的類型,否則 LVGL 無法對輸入作出響應。

LVGL 對輸入設備的介面全部存放在 lv_port_indev.c 及其頭文件中。接下來以觸摸屏為例介紹輸入設備的移植,不同設備的 API 有一定區別,在移植時請以官方文檔為主。

首先,需要去掉兩個文件中的 #if 0 條件編譯,啟用兩個文件。

lv_port_indev.c 中,包含了 5 種設備的 API ,但它們不可能都用到,因此需要裁剪無用的函數和定義。尤其是在初始化函數 lv_port_indev_init() 中,如果不去除無用設備的初始化語句,那麼在調用時可能會出現問題。

源碼在註釋中已經著重強調了不同 API 的分區,只需要根據分區保留需要的代碼即可。

根據代碼的思路(精簡後的源碼不長,而且抽象程度較高,完全可以讀懂),接下來實現三個函數的功能。

首先是 touchpad_init() ,在這裡需要對輸入設備做初始化,就像上文對觸摸屏做初始化一樣。

touchpad_is_pressed() 中,需要提供一個顯示屏觸摸函數,判斷是否發生了觸摸事件:

static bool touchpad_is_pressed(void) {
    if (XPT2046_TouchDetect() == TOUCH_PRESSED)
        return true;
    return false;
}

如果發生了觸摸事件,那麼會進入 touchpad_get_xy() 函數中,獲取觸摸點坐標:

static void touchpad_get_xy(lv_coord_t * x, lv_coord_t * y) {
    static XPT2046_Coordinate coord = { -1, -1, -1, -1 };
    XPT2046_Get_TouchedPoint(&xpt2046, &coord);
    (*x) = coord.x;
    (*y) = coord.y;
}

如果這幾個函數都編寫正確,那麼理論上已經可以實現輸入功能了。不過在此之前,還有一個關鍵的步驟:LVGL 使用一個 tick 系統管理全局事件,它就像 LVGL 的心跳一樣,如果沒有這個心跳就無法檢測事件。

為了給 LVGL 提供心跳,需要不斷調用 lv_tick_inc() 函數,該函數的參數為每次心跳的毫秒間隔:

while (1)
{
    lv_tick_inc(1);
    lv_task_handler();
    delay_ms(1);
}

使用單片機時更推薦使用定時器完成該函數的調用,設置定時器溢出時間為 1 毫秒後在定時器中斷函數內調用它。


接下來提供一個示例,可以檢測輸入設備是否能正常使用。首先在 main 函數的開頭執行輸入設備的初始化:

lv_port_indev_init();

然後編寫如下函數:

static void btn_event_cb(lv_event_t* e) {
    lv_event_code_t code = lv_event_get_code(e);
    lv_obj_t* btn = lv_event_get_target(e);
    if (code == LV_EVENT_CLICKED) {
        static uint8_t cnt = 0;
        cnt++;
        lv_obj_t* label = lv_obj_get_child(btn, 0);
        lv_label_set_text_fmt(label, "Button: %d", cnt);
    }
}

void lv_example(void) {
    lv_obj_t* btn = lv_btn_create(lv_scr_act());
    lv_obj_set_pos(btn, 10, 10);
    lv_obj_set_size(btn, 120, 50);
    lv_obj_add_event_cb(btn, btn_event_cb, LV_EVENT_ALL, NULL);
    lv_obj_t* label = lv_label_create(btn);
    lv_label_set_text(label, "Button");
    lv_obj_center(label);
}

在主函數中調用 lv_example() ,編譯後下載到單片機內,可以得到一個和上一個示例相同的按鈕,但是每次點擊之後,按鈕的文本都會發生變化:

image

其餘內容將第一時間更新於:http://frozencandles.fun/archives/307

使用LVGL模擬器

LVGL 是一個圖形庫,那麼在繪製圖形時就免不了需要對繪製結果做一些微調。那麼每次微調都需要將程式下載到單片機去顯然是麻煩的選擇,不過幸好 LVGL 提供了模擬器,可以在 PC 端上直接生成可交互的界面,無需下載即可查看繪製效果。

LVGL 可以在各個平臺上模擬,完整的模擬器使用指南可以參照 https://docs.lvgl.io/master/get-started/platforms/pc-simulator.html 。接下來以 Windows 平臺基於 Visual Studio 的模擬為例介紹通用的使用方法。

首先,在 https://github.com/lvgl/lv_port_win_visual_studio 中下載 Visual Studio 工程源碼。註意,在 LVGL.Simulator 目錄中包含 3 個外部的倉庫,需要將它們一併下載並放在正確的位置。

然後,使用 Visual Studio 打開 LVGL.Simulator.sln 工程,點擊編譯即可得到 GUI 可執行文件。

image

需要註意的是,Visual Studio 提供的模擬器是使用 C++ 編寫的,如果需要自定義函數,需要在頭文件中使用

#ifdef __cplusplus
extern "C" {
#endif
/* ... function prototypes ... */
#ifdef __cplusplus
}
#endif

將函數原型包圍起來,否則在使用 C 語言符號時會出錯。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • scrapy爬蟲框架 簡介 通過實戰快速入門scrapy爬蟲框架 scrapy爬蟲框架入門簡介 下載scrapy pip install scrapy 創建項目 scrapy startproject spiderTest1 創建爬蟲 cd .\spiderTest1 scrapy genspide ...
  • 線上問題年年有,今年特別多。記幾次線上慘痛的踩坑記錄,希望大家以史為鑒。 1. 包裝類型自動解箱導致空指針異常 public int getId() { Integer id = null; return id; } 如果調用上面的方法會發生什麼?id是Integer類型,而方法的返回值int類型, ...
  • 排版雖然只是繪圖中的輔助功能,但是好的排版能夠提高圖形的表現力。 讓人一眼看出圖形要表達的內容和數據,而不用去費力的睜大眼睛去圖中查找。 如果說 matplotlib的坐標系統,各種基礎元素,包括柱狀圖,折線圖,散點圖等它的硬實力, 那麼,排版就是 matplotlib 的軟實力,結合排版,能讓分析 ...
  • 本文是clickhouse專欄第五篇,更多內容請關註本號歷史文章! 一、數據類型表 clickhouse內置了很多的column數據類型,可以通過查詢system.data_type_families這張表獲取版本的所有支持的數據類型。下文中第一列是欄位類型,第二列表示該類型的欄位類型是否區分大小寫 ...
  • #region 輸入數字判斷是否是數字 #region 方法實現 /*不管是實參或形參,都在類型中開闢了空間的; 方法的功能一定要單一; 如GetMax(int n1,int n2); 方法中最忌諱的就是提示用戶輸入的字眼。 ///PS:最新版的沒有namespace這些,目前我想到的調用方法和類就 ...
  • #region 調用 /* 我們在main函數中調用Test()函數,我們管main函數稱為調用者, Test函數稱為被調用者. 如果被調用者想要得到調用者的值: 1) 傳遞參數; 2) 使用靜態欄位來模擬全局變數; 如果調用者想要得到被調用者的值: 1) 返回值; */ #endregion na ...
  • #region 複習 /* 常量:一旦賦值,不能被重新賦值; 枚舉:規範開發; 結構:為了一次性聲明多個不同類型的變數(實際為欄位); 數組:為了一次性聲明多個相同類型的變數 通過下標或索引訪問數組中元素 數組的取值和賦值 冒泡排序:兩個for迴圈 Array.Sort(nums);Array.Re ...
  • 鏡像下載、功能變數名稱解析、時間同步請點擊 阿裡雲開源鏡像站 1.先查看本機的系統信息 [root@h0436 h0436 zlong]# cat /etc/redhat-release 2.進入yum.repos.d [root@h0436 zlong]# cd /etc/yum.repos.d 3.查看 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...