使用GUI Guider工具开发嵌入式GUI应用(5)-使用timer对象显示动画

news2025/1/9 16:37:28

使用GUI Guider工具开发嵌入式GUI应用(5)-使用timer对象显示动画

文章目录

  • 使用GUI Guider工具开发嵌入式GUI应用(5)-使用timer对象显示动画
    • 引言
    • LVGL中的timer对象
    • 基于timer对象实现仪表走针
    • 小结

引言

设计GUI的显示元素动起来,实际上是多张图按照一定的节奏和变化依次刷屏实现的。LVGL在内核设计了timer对象,用于实现周期回调的功能,开发者可以在周期调用的回调函数中,修改GUI显示内容的属性,触发LVGL显示内容的刷新,然后由LVGL的例行执行程序重新刷屏,从而实现动画效果。

LVGL中的timer对象

LVGL的后台是通过时间片轮询对LVGL的交互任务进行调度的。

在具体的MCU平台上适配LVGL时,通常会使用SysTick定时器中断服务程序调用lv_tick_inc()函数驱动LVGL内部的计数器递增,然后在主线程中调用lv_timer_handler()函数以时间片为节点执行LVGL的日常任务(输入事件、刷屏等操作)。从源码上看,整个LVGL的程序框架,就是建立在定时器刻度的轮询调度之上的,可见包含LVGL的MCU工程的顶级业务逻辑代码如下:

lv_ui guider_ui;

int main(void)
{
    BOARD_Init();
    SysTick_Config(CLOCK_SYS_FREQ / 100); /* 10ms. */

    lv_init();
    lv_port_disp_init();
    setup_ui(&guider_ui);
    events_init(&guider_ui);
    custom_init(&guider_ui);

    while (1)
    {
        lv_timer_handler(); /* 执行LVGL时间片轮询. */
    }
}

void SysTick_Handler(void)
{
    lv_tick_inc(10); /* 驱动LVGL的定时器. */
}

在LVGL中,当需要实现一些基于时间变化显示内容的效果(动画),可以借用LVGL内部的定时器对象来实现周期执行程序的效果。关于LVGL应用程序中使用定时器的API主要有两个:

  • lv_timer_create()
  • lv_timer_del()

其中,使用lv_timer_create()函数创建一个定时器实例,并会向其中传送初始化参数,指定本定时器的触发周期触发时执行的回调函数。而lv_timer_del()函数会回收这个定时器的资源(包括变量,以及挂在定时器任务列表中的指针),以减轻LVGL后台调度器的负担。

基于timer对象实现仪表走针

这里以设计一个速度表盘指针转动的样例,展示timer定时器对象的用法。实现原理,在定时器组件的回调函数中,会周期性地改变meter组件中speed_scale_1_ndline_0属性的值。当改变LVGL显示组件的任何可视化组件的属性后,LVGL会自动刷新显示内容,从而形成动画效果。

在GUI Guider的编辑页面中新建一个meter组件的对象,如图x所示。

在这里插入图片描述

图x 在GUI Guider中新建meter对象

将新建meter对象改名为meter_speed,这个名字将会在后面自定义编写代码时,用于引用这个meter对象实例。手工调整一下表盘的尺寸。

然后,选中主编辑区的页面区域(对应选中当前的screen对象),再在编辑区域的右侧,在事件(Events)页面中,单击加号,创建一个事件,编辑事件。

编辑事件的触发方式,即在什么时机产生当前编辑的事件,常见的事件触发方式有:Load Start(开始载入当前对象时)、Loaded(完成载入当前对象时)、Unload Start(开始从本对象切换到别处时)、Unloaded(完成从本对象切换到别处时)等,如图x所示。每种显示元素的对象都有各自的触发方式,甚至还有一些显示元素的对象没有触发事件可创建。

在这里插入图片描述

图x 在GUI Guider中为事件编辑触发方式

选定Trigger触发方式后,还需要选定Target目标对象。此时,目标对象的下拉列表中会自动整理出当前整个GUI Guider工程中已经创建的有所显示元素对象。如图x所示。

在这里插入图片描述

图x 在GUI Guider中为事件编辑目标对象

这里需要重点说明的是,触发方式描述的当前在编辑区域中选中的对象的动作,当选定触发方式后,本事件的标签名称也就自动改成了<当前对象名>_<触发方式>。目标对象反映的是即将改变属性的对象,对应部分可选的属性也在事件页面下方展示出来,可在图形页面中配置。当然,并不是所有的可编辑属性都被做成了在对话框中可配置的,本例就在最后一行,选择了执行自定义程序,在弹出的代码编辑对话框中添加了创建定时器对象timer_meter_speed的语句。如图x所示。

在这里插入图片描述

图x 在GUI Guider中为事件编辑自定义执行的程序代码

图x中所示,自定义编写C源码,当触发Screen的Loaded事件时,将执行图中代码,创建timer_meter_speed定时器对象实例。保存工程后生成代码,对应地,可在GUI Guider工程中的./generated/event_init.c文件中看到对应生成的代码。

#include "events_init.h"
#include <stdio.h>
#include "lvgl.h"
#include "custom.h"

static lv_timer_t * timer_meter_speed;

void events_init(lv_ui *ui)
{
}

static void screen_event_handler(lv_event_t *e)
{
	lv_event_code_t code = lv_event_get_code(e);
	switch (code)
	{
	case LV_EVENT_SCREEN_LOADED:
	{
		timer_meter_speed = lv_timer_create(timer_meter_speed_cb, 100, &guider_ui);
	}
		break;
	case LV_EVENT_SCREEN_UNLOADED:
	{
		lv_timer_del(timer_meter_speed);
	}
		break;
	default:
		break;
	}
}

void events_init_screen(lv_ui *ui)
{
	lv_obj_add_event_cb(ui->screen, screen_event_handler, LV_EVENT_ALL, ui);
}

但是回调函数timer_meter_speed_cb()仍需要开发者自行在custom.c文件中自行创建,GUI Guider不会自动更新。代码如下:

static int32_t speed = 50;
static bool is_increase = true;


/**
 * Create a demo application
 */

void custom_init(lv_ui *ui)
{
    /* Add your codes here */
}

void timer_meter_speed_cb(lv_timer_t *t)
{
    lv_ui * gui = t->user_data;
    lv_meter_set_indicator_value(gui->screen_meter_speed, gui->screen_meter_speed_scale_1_ndline_0, speed);

    if (speed >= 90)
    {
        is_increase = false;
    }
    if (speed <= 20)
    {
        is_increase = true;
    }
    if (is_increase)
    {
        speed++;
    }
    else
    {
        speed--;
    }
}

这里实现的内容是,在MCU上运行LVGL时,一旦显示完成当前的屏幕页面后,立即创建定时器对象timer_meter_speed,这个定时器将会每隔100ms调用一次回调函数timer_meter_speed_cb(),在timer_meter_speed_cb()函数中,更新变量speed的值,然后通过lv_meter_set_indicator_value()函数,改变screen_meter_speed对象中screen_meter_speed_scale_1_ndline_0属性的值。每隔100ms改变一次screen_meter_speed对象的属性值,将会触发显示内容的刷新,从而在显示屏上显现出动画效果。

将GUI Guider生成的程序下载到MCU之前,可以先在GUI Guider的工程中模拟运行一次,一方面可以帮忙排除编译错误(GUI Guider使用armgcc将LVGL和生成的源码一起编译),另一方面也方便预览运行效果。使用LVGL组件的过程不涉及到对具体MCU硬件的依赖,因此完全可以在模拟环境中预览实际执行的效果。如图x所示。

在这里插入图片描述

图x 预览在GUI Guider中创建的表盘对象

最后,编译Keil工程,下载到MCU开发板。可以看到在MCU开发板上运行动图的效果,如图x所示。

在这里插入图片描述

图x 在MCU开发板上运行转动指针的表盘

小结

timer对象本身只是一个实现周期调度的机制,开发者可以在周期回调的函数内部,修改和重置显示各种对象的属性,触发LVGL刷屏,从而实现动画的效果。

(未完待续。。。)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/890249.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Websocket原理和实践

一、概述 1.websocket是什么&#xff1f; WebSocket是一种在单个TCP连接上进行全双工通信的协议。WebSocket使得客户端和服务器之间的数据交换变得更加简单&#xff0c;允许服务端主动向客户端推送数据。在WebSocket API中&#xff0c;浏览器和服务器只需要完成一次握手&…

如何快速优化 CnosDB 数据库性能与延迟:使用 Jaeger 分布式追踪系统

在正式的生产环境中&#xff0c;数据库的性能和延迟对于确保系统的稳定和高效运行至关重要。特别是在与 CnosDB 数据库进行交互时&#xff0c;更深入地了解其表现变得尤为重要。这时Jaeger 分布式追踪系统发挥了巨大的作用。在本篇博客中&#xff0c;我们将深入探讨如何通过使用…

ATA-4000系列高压功率放大器——应用场景介绍

ATA-4000系列是一款理想的可放大交、直流信号的高压功率放大器。最大输出310Vp-p(155Vp)电压&#xff0c;452Wp功率&#xff0c;可以驱动高压功率型负载。电压增益&#xff0c;直流偏置数控精细可调&#xff0c;为客户提供了丰富的测试选择。 图&#xff1a;ATA-4000系列高压功…

ndk开发-交叉编译

为什么要使用交叉编译&#xff1a; 在linux系统一般使用c c编译可执行程序或者so库文件。该程序只能在当前linux系统执行&#xff0c;为了将生成文件可以再android平台运行&#xff0c;必须使用交叉编译。ndk中提供了跟多android平台交叉编译链&#xff0c;所以首先下载ndk工具…

FPGA应用学习笔记-----布图布线

分割可以将运行时间惊人地减少到三个小时更小的布局布线操作&#xff0c;主要的结构不影响另一个&#xff01;和增量设计流程一样 关键路径布图&#xff1a; 对于不同的模块有不同的电路和不同的关键路径&#xff0c; 布图没有主要的分割&#xff0c;布图由两个小的区域组成&a…

KDD 2023 获奖论文公布,港中文、港科大等获最佳论文奖

ACM SIGKDD&#xff08;国际数据挖掘与知识发现大会&#xff0c;KDD&#xff09;是数据挖掘领域历史最悠久、规模最大的国际顶级学术会议&#xff0c;也是首个引入大数据、数据科学、预测分析、众包等概念的会议。 今年&#xff0c;第29届 KDD 大会于上周在美国加州长滩圆满结…

C语言入门教程,C语言学习教程(非常详细)第五章 循环结构与选择结构

C语言if else语句详解 前面我们看到的代码都是顺序执行的&#xff0c;也就是先执行第一条语句&#xff0c;然后是第二条、第三条……一直到最后一条语句&#xff0c;这称为顺序结构。 但是对于很多情况&#xff0c;顺序结构的代码是远远不够的&#xff0c;比如一个程序限制了只…

【Javaswing课设源码】学生信息管理 Mysql课程设计 管理员 教师 学生

文章目录 系统介绍 系统介绍 大学时代弄的一个课设&#xff0c;当时百度[学长敲代码]找的代做&#xff0c;代码思路很清晰&#xff0c;完全按照我的功能需求去做的&#xff0c;主要是价格便宜&#xff0c;真的爱了&#xff0c;现在回头学习也是不错的一个项目。大概内容如下 本…

springboot里 用zxing 生成二维码

引入pom <!--二维码依赖--><dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.3.3</version></dependency><dependency><groupId>com.google.zxing</groupId>…

【分类讨论】CF1674 E

Problem - E - Codeforces 题意&#xff1a; 思路&#xff1a; 样例&#xff1a; 这种分类讨论的题&#xff0c;主要是去看答案的最终来源是哪几种情况&#xff0c;这几种情况得不重不漏 Code&#xff1a; #include <bits/stdc.h>#define int long longusing i64 lon…

尚硅谷css3笔记

目录 一、新增长度单位 二、新增盒子属性 1.border-box 怪异盒模型 2.resize 调整盒子大小 3.box-shadow 盒子阴影 案例&#xff1a;鼠标悬浮盒子上时&#xff0c;盒子有一个过度的阴影效果 三、新增背景属性 1.background-origin 设置背景图的原点 2.background-clip 设置背…

基于IMX6ULLmini的linux裸机开发系列一:汇编点亮LED

思来想去还是决定记录一下点灯&#xff0c;毕竟万物皆点灯嘛 编程步骤 使能GPIO时钟 设置引脚复用为GPIO 设置引脚属性(上下拉、速率、驱动能力) 控制GPIO引脚输出高低电平 使能GPIO时钟 其实和32差不多 先找到控制LED灯的引脚&#xff0c;也就是原理图 文件名 C:/Us…

自动提示功能消失解决方案

如果绿叶子是不可点击状态&#xff0c;可以点一下列表中的配置文件

43、TCP报文(一)

本节内容开始&#xff0c;我们正式学习TCP协议中具体的一些原理。首先&#xff0c;最重要的内容仍然是这个协议的封装结构和首部格式&#xff0c;因为这里面牵扯到一些环环相扣的知识点&#xff0c;例如ACK、SYN等等&#xff0c;如果这些内容不能很好的理解&#xff0c;那么后续…

A. Copil Copac Draws Trees(Codeforces Round 875 (Div. 1))

Copil Copac is given a list of n − 1 n-1 n−1 edges describing a tree of n n n vertices. He decides to draw it using the following algorithm: Step 0 0 0: Draws the first vertex (vertex 1 1 1). Go to step 1 1 1.Step 1 1 1: For every edge in the inpu…

号外号外,最经典的16S数据库Greengenes2更新啦!!!

没错&#xff0c;这是真的&#xff0c;沉积十年之后&#xff0c;多样性研究中最经典的16S数据库——Greengenes数据库&#xff0c;竟&#xff01;然&#xff01;更&#xff01;新&#xff01;了&#xff01;惊不惊喜&#xff01;意不意外&#xff01; 遥想当年小编还是一个小白…

vue 数字递增(滚动从0到)

使用 html <Incremental :startVal"0" :endVal"1000" :duration"500" />js&#xff1a; import Incremental from /utils/num/numViewjs let lastTime 0 const prefixes webkit moz ms o.split( ) // 各浏览器前缀let requestAnimatio…

基于YOLOv5n/s/m不同参数量级模型开发构建茶叶嫩芽检测识别模型,使用pruning剪枝技术来对模型进行轻量化处理,探索不同剪枝水平下模型性能影响

今天有点时间就想着之前遗留的一个问题正好拿过来做一下看看&#xff0c;主要的目的就是想要对训练好的目标检测模型进行剪枝处理&#xff0c;这里就以茶叶嫩芽检测数据场景为例了&#xff0c;在我前面的博文中已经有过相关的实践介绍了&#xff0c;感兴趣的话可以自行移步阅读…

QT的设计器介绍

设计器介绍 Qt制作 UI 界面&#xff0c;一般可以通过UI制作工具QtDesigner和纯代码编写两种方式来实现。纯代码实现暂时在这里不阐述了在后续布局章节详细说明&#xff0c;QtDesigner已经继承到开发环境中&#xff0c;在工程中直接双击ui文件就可以直接在QtDesigner设计器中打…

AtCoder Beginner Contest 314 E题题解

文章目录 Roulettes问题建模问题分析1.分析每个转盘对所求的作用2.从集合的角度思考每个积分的贡献代码 Roulettes 问题建模 给定n个轮盘&#xff0c;每个轮盘上有p个积分&#xff0c;每次转动轮盘需要一定的代价&#xff0c;在转动轮盘后可以等概率获得p个积分中的一个&#…