1
0
mirror of https://github.com/lvgl/lvgl.git synced 2025-01-21 06:53:01 +08:00
lvgl/lv_obj/lv_refr.c

480 lines
13 KiB
C
Raw Normal View History

2016-06-08 07:25:08 +02:00
/**
* @file lv_refr.c
*
*/
/*********************
* INCLUDES
*********************/
2017-10-09 15:21:26 +02:00
#include <lvgl/lv_hal/lv_hal_tick.h>
2016-06-08 07:25:08 +02:00
#include "lv_conf.h"
2017-10-09 15:21:26 +02:00
#include <stddef.h>
2016-06-08 07:25:08 +02:00
#include "lv_refr.h"
#include "lv_vdb.h"
2017-10-09 15:21:26 +02:00
#include "misc/os/ptask.h"
#include "misc/mem/fifo.h"
2016-06-08 07:25:08 +02:00
/*********************
* DEFINES
*********************/
2017-07-07 18:40:35 +02:00
2016-06-08 07:25:08 +02:00
/**********************
* TYPEDEFS
**********************/
typedef struct
{
area_t area;
uint8_t joined;
}lv_join_t;
/**********************
* STATIC PROTOTYPES
**********************/
static void lv_refr_task(void * param);
static void lv_refr_join_area(void);
static void lv_refr_areas(void);
2016-06-08 07:25:08 +02:00
#if LV_VDB_SIZE == 0
static void lv_refr_area_no_vdb(const area_t * area_p);
#else
static void lv_refr_area_with_vdb(const area_t * area_p);
static void lv_refr_area_part_vdb(const area_t * area_p);
#endif
2016-10-07 11:15:46 +02:00
static lv_obj_t * lv_refr_get_top_obj(const area_t * area_p, lv_obj_t * obj);
static void lv_refr_make(lv_obj_t * top_p, const area_t * mask_p);
static void lv_refr_obj(lv_obj_t * obj, const area_t * mask_ori_p);
2016-06-08 07:25:08 +02:00
/**********************
* STATIC VARIABLES
**********************/
2017-03-20 10:23:56 +01:00
static lv_join_t inv_buf[LV_INV_FIFO_SIZE];
static uint16_t inv_buf_p;
2017-07-07 18:40:35 +02:00
static void (*monitor_cb)(uint32_t, uint32_t);
2017-07-09 15:32:49 +02:00
static uint32_t px_num;
2016-06-08 07:25:08 +02:00
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
/**
* Initialize the screen refresh subsystem
*/
void lv_refr_init(void)
{
inv_buf_p = 0;
memset(inv_buf, 0, sizeof(inv_buf));
2016-06-08 07:25:08 +02:00
ptask_t* task;
task = ptask_create(lv_refr_task, LV_REFR_PERIOD, PTASK_PRIO_MID, NULL);
2016-06-08 07:25:08 +02:00
dm_assert(task);
}
/**
* Invalidate an area
* @param area_p pointer to area which should be invalidated
*/
void lv_inv_area(const area_t * area_p)
{
2017-03-20 10:23:56 +01:00
/*Clear the invalidate buffer if the parameter is NULL*/
if(area_p == NULL) {
inv_buf_p = 0;
return;
}
2016-06-08 07:25:08 +02:00
area_t scr_area;
scr_area.x1 = 0;
scr_area.y1 = 0;
scr_area.x2 = LV_HOR_RES - 1;
scr_area.y2 = LV_VER_RES - 1;
area_t com_area;
bool suc;
suc = area_union(&com_area, area_p, &scr_area);
/*The area is truncated to the screen*/
if(suc != false)
{
#if LV_DOWNSCALE == 2
/*Rounding*/
com_area.x1 = com_area.x1 & (~0x1);
com_area.y1 = com_area.y1 & (~0x1);
com_area.x2 = com_area.x2 | 0x1;
com_area.y2 = com_area.y2 | 0x1;
#endif
/*Save only if this area is not in one of the saved areas*/
uint16_t i;
for(i = 0; i < inv_buf_p; i++) {
if(area_is_in(&com_area, &inv_buf[i].area) != false) return;
}
2016-06-08 07:25:08 +02:00
/*Save the area*/
if(inv_buf_p < LV_INV_FIFO_SIZE) {
area_cpy(&inv_buf[inv_buf_p].area,&com_area);
} else {/*If no place for the area add the screen*/
inv_buf_p = 0;
area_cpy(&inv_buf[inv_buf_p].area,&scr_area);
2016-06-08 07:25:08 +02:00
}
inv_buf_p ++;
2016-06-08 07:25:08 +02:00
}
}
2017-07-07 18:40:35 +02:00
/**
* Set a function to call after every refresh to announce the refresh time and the number of refreshed pixels
* @param cb pointer to a callback function (void my_refr_cb(uint32_t time_ms, uint32_t px_num))
2017-07-07 18:47:48 +02:00
* time_ms: refresh time in [ms]
* px_num: not the drawn pixels but the number of affected pixels of the screen
* (more pixels are drawn because of overlapping objects)
2017-07-07 18:40:35 +02:00
*/
void lv_refr_set_monitor_cb(void (*cb)(uint32_t, uint32_t))
{
monitor_cb = cb;
}
2016-06-08 07:25:08 +02:00
/**********************
* STATIC FUNCTIONS
**********************/
/**
* Called periodically to handle the refreshing
* @param param unused
2016-06-08 07:25:08 +02:00
*/
static void lv_refr_task(void * param)
2016-06-08 07:25:08 +02:00
{
2017-07-09 15:32:49 +02:00
2017-10-09 15:21:26 +02:00
uint32_t start = lv_tick_get();
2017-07-09 15:32:49 +02:00
lv_refr_join_area();
2016-06-08 07:25:08 +02:00
lv_refr_areas();
2017-07-09 15:32:49 +02:00
bool refr_done = false;
if(inv_buf_p != 0) refr_done = true;
memset(inv_buf, 0, sizeof(inv_buf));
inv_buf_p = 0;
2017-07-09 15:32:49 +02:00
/* In the callback lv_obj_inv can occur
* therefore be sure the inv_buf is cleared prior to it*/
if(refr_done != false) {
if(monitor_cb != NULL) {
2017-10-09 15:21:26 +02:00
monitor_cb(lv_tick_elaps(start), px_num);
2017-07-09 15:32:49 +02:00
}
}
2016-06-08 07:25:08 +02:00
}
/**
* Join the areas which has got common parts
*/
static void lv_refr_join_area(void)
2016-06-08 07:25:08 +02:00
{
uint32_t join_from;
uint32_t join_in;
area_t joined_area;
for(join_in = 0; join_in < inv_buf_p; join_in++) {
if(inv_buf[join_in].joined != 0) continue;
2016-06-08 07:25:08 +02:00
/*Check all areas to join them in 'join_in'*/
for(join_from = 0; join_from < inv_buf_p; join_from++) {
2016-06-08 07:25:08 +02:00
/*Handle only unjoined areas and ignore itself*/
if(inv_buf[join_from].joined != 0 || join_in == join_from) {
2016-06-08 07:25:08 +02:00
continue;
}
/*Check if the areas are on each other*/
if(area_is_on(&inv_buf[join_in].area,
&inv_buf[join_from].area) == false)
2016-06-08 07:25:08 +02:00
{
continue;
}
area_join(&joined_area, &inv_buf[join_in].area,
&inv_buf[join_from].area);
2016-06-08 07:25:08 +02:00
/*Join two area only if the joined area size is smaller*/
if(area_get_size(&joined_area) <
(area_get_size(&inv_buf[join_in].area) + area_get_size(&inv_buf[join_from].area))) {
area_cpy(&inv_buf[join_in].area, &joined_area);
2016-06-08 07:25:08 +02:00
/*Mark 'join_form' is joined into 'join_in'*/
inv_buf[join_from].joined = 1;
2016-06-08 07:25:08 +02:00
}
}
}
}
/**
* Refresh the joined areas
*/
static void lv_refr_areas(void)
2016-06-08 07:25:08 +02:00
{
2017-07-09 15:32:49 +02:00
px_num = 0;
2016-06-08 07:25:08 +02:00
uint32_t i;
for(i = 0; i < inv_buf_p; i++) {
2016-06-08 07:25:08 +02:00
/*Refresh the unjoined areas*/
if(inv_buf[i].joined == 0) {
2016-06-08 07:25:08 +02:00
/*If there is no VDB do simple drawing*/
#if LV_VDB_SIZE == 0
lv_refr_area_no_vdb(&inv_buf[i].area);
2016-06-08 07:25:08 +02:00
#else
/*If VDB is used...*/
lv_refr_area_with_vdb(&inv_buf[i].area);
2016-06-08 07:25:08 +02:00
#endif
2017-07-07 18:40:35 +02:00
if(monitor_cb != NULL) px_num += area_get_size(&inv_buf[i].area);
2016-06-08 07:25:08 +02:00
}
}
2016-06-08 07:25:08 +02:00
}
#if LV_VDB_SIZE == 0
/**
* Refresh an area if there is no Virtual Display Buffer
* @param area_p pointer to an area to refresh
*/
static void lv_refr_area_no_vdb(const area_t * area_p)
{
2016-10-07 11:15:46 +02:00
lv_obj_t * top_p;
2016-06-08 07:25:08 +02:00
/*Get top object which is not covered by others*/
top_p = lv_refr_get_top_obj(area_p, lv_scr_act());
/*Do the refreshing*/
lv_refr_make(top_p, area_p);
}
#else
/**
* Refresh an area if there is Virtual Display Buffer
* @param area_p pointer to an area to refresh
*/
static void lv_refr_area_with_vdb(const area_t * area_p)
{
/*Calculate the max row num*/
uint32_t max_row = (uint32_t) LV_VDB_SIZE / (area_get_width(area_p));
2016-06-08 07:25:08 +02:00
if(max_row > area_get_height(area_p)) max_row = area_get_height(area_p);
/*Round the row number with downscale*/
#if LV_DOWNSCALE == 2
max_row &= (~0x1);
#endif
/*Always use the full row*/
cord_t row;
cord_t row_last = 0;
2016-06-08 07:25:08 +02:00
for(row = area_p->y1; row + max_row - 1 <= area_p->y2; row += max_row) {
lv_vdb_t * vdb_p = lv_vdb_get();
2016-06-08 07:25:08 +02:00
/*Calc. the next y coordinates of VDB*/
vdb_p->area.x1 = area_p->x1;
vdb_p->area.x2 = area_p->x2;
2017-04-21 09:15:39 +02:00
vdb_p->area.y1 = row;
vdb_p->area.y2 = row + max_row - 1;
row_last = row + max_row - 1;
2016-06-08 07:25:08 +02:00
lv_refr_area_part_vdb(area_p);
}
/*If the last y coordinates are not handled yet ...*/
if(area_p->y2 != row_last) {
lv_vdb_t * vdb_p = lv_vdb_get();
2016-06-08 07:25:08 +02:00
/*Calc. the next y coordinates of VDB*/
vdb_p->area.x1 = area_p->x1;
vdb_p->area.x2 = area_p->x2;
2017-04-21 09:15:39 +02:00
vdb_p->area.y1 = row;
vdb_p->area.y2 = area_p->y2;
2016-06-08 07:25:08 +02:00
/*Refresh this part too*/
lv_refr_area_part_vdb(area_p);
}
}
/**
* Refresh a part of an area which is on the actual Virtual Display Buffer
* @param area_p pointer to an area to refresh
*/
static void lv_refr_area_part_vdb(const area_t * area_p)
{
lv_vdb_t * vdb_p = lv_vdb_get();
2016-10-07 11:15:46 +02:00
lv_obj_t * top_p;
2016-06-08 07:25:08 +02:00
/*Get the new mask from the original area and the act. VDB
It will be a part of 'area_p'*/
area_t start_mask;
2017-04-21 09:15:39 +02:00
area_union(&start_mask, area_p, &vdb_p->area);
2016-06-08 07:25:08 +02:00
/*Get the most top object which is not covered by others*/
top_p = lv_refr_get_top_obj(&start_mask, lv_scr_act());
/*Do the refreshing from the top object*/
2016-06-08 07:25:08 +02:00
lv_refr_make(top_p, &start_mask);
2017-10-10 16:17:23 +02:00
/*Also refresh top and sys layer unconditionally*/
lv_refr_make(lv_top_layer(), &start_mask);
lv_refr_make(lv_sys_layer(), &start_mask);
2016-06-08 07:25:08 +02:00
/*Flush the content of the VDB*/
lv_vdb_flush();
}
#endif /*LV_VDB_SIZE == 0*/
/**
* Search the most top object which fully covers an area
* @param area_p pointer to an area
2016-10-07 11:15:46 +02:00
* @param obj the first object to start the searching (typically a screen)
2016-06-08 07:25:08 +02:00
* @return
*/
2016-10-07 11:15:46 +02:00
static lv_obj_t * lv_refr_get_top_obj(const area_t * area_p, lv_obj_t * obj)
2016-06-08 07:25:08 +02:00
{
2016-10-07 11:15:46 +02:00
lv_obj_t * i;
lv_obj_t * found_p = NULL;
2016-06-08 07:25:08 +02:00
/*If this object is fully cover the draw area check the children too */
if(area_is_in(area_p, &obj->cords) && obj->hidden == 0)
2016-06-08 07:25:08 +02:00
{
2016-10-07 11:15:46 +02:00
LL_READ(obj->child_ll, i) {
2016-06-08 07:25:08 +02:00
found_p = lv_refr_get_top_obj(area_p, i);
/*If a children is ok then break*/
if(found_p != NULL) {
break;
}
}
/*If no better children check this object*/
2016-06-08 07:25:08 +02:00
if(found_p == NULL) {
lv_style_t * style = lv_obj_get_style(obj);
if(style->opa == OPA_COVER &&
obj->design_f(obj, area_p, LV_DESIGN_COVER_CHK) != false) {
found_p = obj;
}
2016-06-08 07:25:08 +02:00
}
}
return found_p;
}
/**
2017-07-07 18:40:35 +02:00
* Make the refreshing from an object. Draw all its children and the youngers too.
2016-06-08 07:25:08 +02:00
* @param top_p pointer to an objects. Start the drawing from it.
* @param mask_p pointer to an area, the objects will be drawn only here
*/
2016-10-07 11:15:46 +02:00
static void lv_refr_make(lv_obj_t * top_p, const area_t * mask_p)
2016-06-08 07:25:08 +02:00
{
/* Normally always will be a top_obj (at least the screen)
* but in special cases (e.g. if the screen has alpha) it won't.
* In this case use the screen directly */
if(top_p == NULL) top_p = lv_scr_act();
/*Refresh the top object and its children*/
lv_refr_obj(top_p, mask_p);
/*Draw the 'younger' sibling objects because they can be on top_obj */
2016-10-07 11:15:46 +02:00
lv_obj_t * par;
lv_obj_t * i;
lv_obj_t * border_p = top_p;
2016-06-08 07:25:08 +02:00
2016-10-07 11:15:46 +02:00
par = lv_obj_get_parent(top_p);
2016-06-08 07:25:08 +02:00
/*Do until not reach the screen*/
2016-10-07 11:15:46 +02:00
while(par != NULL) {
2016-06-08 07:25:08 +02:00
/*object before border_p has to be redrawn*/
2016-10-07 11:15:46 +02:00
i = ll_get_prev(&(par->child_ll), border_p);
2016-06-08 07:25:08 +02:00
while(i != NULL) {
/*Refresh the objects*/
lv_refr_obj(i, mask_p);
2016-10-07 11:15:46 +02:00
i = ll_get_prev(&(par->child_ll), i);
2016-06-08 07:25:08 +02:00
}
/*The new border will be there last parents,
*so the 'younger' brothers of parent will be refreshed*/
2016-10-07 11:15:46 +02:00
border_p = par;
2016-06-08 07:25:08 +02:00
/*Go a level deeper*/
2016-10-07 11:15:46 +02:00
par = lv_obj_get_parent(par);
2016-06-08 07:25:08 +02:00
}
/*Call the post draw design function of the parents of the to object*/
par = lv_obj_get_parent(top_p);
while(par != NULL) {
par->design_f(par, mask_p, LV_DESIGN_DRAW_POST);
par = lv_obj_get_parent(par);
}
2016-06-08 07:25:08 +02:00
}
/**
* Refresh an object an all of its children. (Called recursively)
2016-10-07 11:15:46 +02:00
* @param obj pointer to an object to refresh
2016-06-08 07:25:08 +02:00
* @param mask_ori_p pointer to an area, the objects will be drawn only here
*/
2016-10-07 11:15:46 +02:00
static void lv_refr_obj(lv_obj_t * obj, const area_t * mask_ori_p)
2016-06-08 07:25:08 +02:00
{
/*Do not refresh hidden objects*/
2016-10-07 11:15:46 +02:00
if(obj->hidden != 0) return;
2016-06-08 07:25:08 +02:00
bool union_ok; /* Store the return value of area_union */
/* Truncate the original mask to the coordinates of the parent
* because the parent and its children are visible only here */
2016-10-17 15:02:18 +02:00
area_t obj_mask;
2016-10-20 16:35:03 +02:00
area_t obj_ext_mask;
2016-10-17 15:02:18 +02:00
area_t obj_area;
cord_t ext_size = obj->ext_size;
lv_obj_get_cords(obj, &obj_area);
obj_area.x1 -= ext_size;
obj_area.y1 -= ext_size;
obj_area.x2 += ext_size;
obj_area.y2 += ext_size;
2016-10-20 16:35:03 +02:00
union_ok = area_union(&obj_ext_mask, mask_ori_p, &obj_area);
2016-06-08 07:25:08 +02:00
/*Draw the parent and its children only if they ore on 'mask_parent'*/
if(union_ok != false) {
/* Redraw the object */
lv_style_t * style = lv_obj_get_style(obj);
if(style->opa != OPA_TRANSP) {
2016-10-20 16:35:03 +02:00
obj->design_f(obj, &obj_ext_mask, LV_DESIGN_DRAW_MAIN);
2017-04-21 17:11:47 +02:00
//tick_wait_ms(100); /*DEBUG: Wait after every object draw to see the order of drawing*/
2016-06-08 07:25:08 +02:00
}
2016-10-20 16:35:03 +02:00
/*Create a new 'obj_mask' without 'ext_size' because the children can't be visible there*/
lv_obj_get_cords(obj, &obj_area);
union_ok = area_union(&obj_mask, mask_ori_p, &obj_area);
if(union_ok != false) {
area_t mask_child; /*Mask from obj and its child*/
lv_obj_t * child_p;
area_t child_area;
LL_READ_BACK(obj->child_ll, child_p)
{
lv_obj_get_cords(child_p, &child_area);
ext_size = child_p->ext_size;
child_area.x1 -= ext_size;
child_area.y1 -= ext_size;
child_area.x2 += ext_size;
child_area.y2 += ext_size;
/* Get the union (common parts) of original mask (from obj)
* and its child */
union_ok = area_union(&mask_child, &obj_mask, &child_area);
/*If the parent and the child has common area then refresh the child */
if(union_ok) {
/*Refresh the next children*/
lv_refr_obj(child_p, &mask_child);
}
}
2016-06-08 07:25:08 +02:00
}
2016-10-20 16:35:03 +02:00
/* If all the children are redrawn make 'post draw' design */
if(style->opa != OPA_TRANSP) {
2016-10-20 16:35:03 +02:00
obj->design_f(obj, &obj_ext_mask, LV_DESIGN_DRAW_POST);
}
2016-06-08 07:25:08 +02:00
}
}