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

591 lines
19 KiB
C
Raw Normal View History

2016-06-08 07:25:08 +02:00
/**
* @file lv_refr.c
2018-06-19 09:49:58 +02:00
*
2016-06-08 07:25:08 +02:00
*/
/*********************
* INCLUDES
*********************/
2017-10-09 15:21:26 +02:00
#include <stddef.h>
2016-06-08 07:25:08 +02:00
#include "lv_refr.h"
2019-02-12 12:21:34 +01:00
#include "lv_disp.h"
2017-11-26 23:57:39 +01:00
#include "../lv_hal/lv_hal_tick.h"
2019-01-29 04:58:18 +01:00
#include "../lv_hal/lv_hal_disp.h"
2017-11-23 20:42:14 +01:00
#include "../lv_misc/lv_task.h"
#include "../lv_misc/lv_mem.h"
2019-02-12 12:21:34 +01:00
#include "../lv_misc/lv_gc.h"
#include "../lv_draw/lv_draw.h"
2016-06-08 07:25:08 +02:00
#if defined(LV_GC_INCLUDE)
2019-04-04 07:15:40 +02:00
#include LV_GC_INCLUDE
#endif /* LV_ENABLE_GC */
2016-06-08 07:25:08 +02:00
/*********************
* DEFINES
*********************/
2019-05-09 06:04:03 +02:00
/* Draw translucent random colored areas on the invalidated (redrawn) areas*/
2019-06-06 06:05:40 +02:00
#define MASK_AREA_DEBUG 0
2017-07-07 18:40:35 +02:00
2016-06-08 07:25:08 +02:00
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
static void lv_refr_join_area(void);
static void lv_refr_areas(void);
2019-02-13 01:40:22 +01:00
static void lv_refr_area(const lv_area_t * area_p);
static void lv_refr_area_part(const lv_area_t * area_p);
2017-11-23 21:28:36 +01:00
static lv_obj_t * lv_refr_get_top_obj(const lv_area_t * area_p, lv_obj_t * obj);
static void lv_refr_obj_and_children(lv_obj_t * top_p, const lv_area_t * mask_p);
2017-11-23 21:28:36 +01:00
static void lv_refr_obj(lv_obj_t * obj, const lv_area_t * mask_ori_p);
2019-02-13 01:40:22 +01:00
static void lv_refr_vdb_flush(void);
2016-06-08 07:25:08 +02:00
/**********************
* STATIC VARIABLES
**********************/
2017-07-09 15:32:49 +02:00
static uint32_t px_num;
2019-04-04 07:15:40 +02:00
static lv_disp_t * disp_refr; /*Display being refreshed*/
2016-06-08 07:25:08 +02:00
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
/**
* Initialize the screen refresh subsystem
*/
void lv_refr_init(void)
2018-06-19 09:49:58 +02:00
{
/*Nothing to do*/
2016-06-08 07:25:08 +02:00
}
2018-09-20 21:59:44 +02:00
/**
* Redraw the invalidated areas now.
2019-04-04 07:15:40 +02:00
* Normally the redrawing is periodically executed in `lv_task_handler` but a long blocking process
* can prevent the call of `lv_task_handler`. In this case if the the GUI is updated in the process
* (e.g. progress bar) this function can be called when the screen should be updated.
* @param disp pointer to display to refresh. NULL to refresh all displays.
2018-09-20 21:59:44 +02:00
*/
void lv_refr_now(lv_disp_t * disp)
2018-09-20 21:59:44 +02:00
{
if(disp) {
lv_disp_refr_task(disp->refr_task);
} else {
lv_disp_t * d;
d = lv_disp_get_next(NULL);
while(d) {
lv_disp_refr_task(d->refr_task);
d = lv_disp_get_next(d);
}
2019-05-07 12:15:02 +02:00
}
2018-09-20 21:59:44 +02:00
}
2016-06-08 07:25:08 +02:00
/**
* Invalidate an area on display to redraw it
* @param area_p pointer to area which should be invalidated (NULL: delete the invalidated areas)
2019-04-04 07:15:40 +02:00
* @param disp pointer to display where the area should be invalidated (NULL can be used if there is
* only one display)
2016-06-08 07:25:08 +02:00
*/
void lv_inv_area(lv_disp_t * disp, const lv_area_t * area_p)
2016-06-08 07:25:08 +02:00
{
2019-02-20 10:16:33 +01:00
if(!disp) disp = lv_disp_get_default();
if(!disp) return;
2017-03-20 10:23:56 +01:00
/*Clear the invalidate buffer if the parameter is NULL*/
if(area_p == NULL) {
disp->inv_p = 0;
2017-03-20 10:23:56 +01:00
return;
}
2018-06-19 09:49:58 +02:00
2017-11-23 21:28:36 +01:00
lv_area_t scr_area;
2016-06-08 07:25:08 +02:00
scr_area.x1 = 0;
scr_area.y1 = 0;
2019-03-30 06:23:41 +01:00
scr_area.x2 = lv_disp_get_hor_res(disp) - 1;
scr_area.y2 = lv_disp_get_ver_res(disp) - 1;
2018-06-19 09:49:58 +02:00
lv_area_t com_area;
2016-06-08 07:25:08 +02:00
bool suc;
suc = lv_area_intersect(&com_area, area_p, &scr_area);
2016-06-08 07:25:08 +02:00
/*The area is truncated to the screen*/
2018-02-16 13:16:40 +01:00
if(suc != false) {
if(disp->driver.rounder_cb) disp->driver.rounder_cb(&disp->driver, &com_area);
2018-06-19 09:49:58 +02:00
/*Save only if this area is not in one of the saved areas*/
uint16_t i;
for(i = 0; i < disp->inv_p; i++) {
if(lv_area_is_in(&com_area, &disp->inv_areas[i]) != false) return;
2018-06-19 09:49:58 +02:00
}
2016-06-08 07:25:08 +02:00
/*Save the area*/
if(disp->inv_p < LV_INV_BUF_SIZE) {
lv_area_copy(&disp->inv_areas[disp->inv_p], &com_area);
2019-04-04 07:15:40 +02:00
} else { /*If no place for the area add the screen*/
disp->inv_p = 0;
lv_area_copy(&disp->inv_areas[disp->inv_p], &scr_area);
2016-06-08 07:25:08 +02:00
}
2019-04-04 07:15:40 +02:00
disp->inv_p++;
2016-06-08 07:25:08 +02:00
}
}
2019-02-12 12:21:34 +01:00
/**
* Get the display which is being refreshed
* @return the display being refreshed
*/
lv_disp_t * lv_refr_get_disp_refreshing(void)
{
return disp_refr;
}
/**
* Set the display which is being refreshed.
* It shouldn1t be used directly by the user.
* It can be used to trick the drawing functions about there is an active display.
* @param the display being refreshed
*/
void lv_refr_set_disp_refreshing(lv_disp_t * disp)
{
disp_refr = disp;
}
2016-06-08 07:25:08 +02:00
/**
* Called periodically to handle the refreshing
* @param task pointer to the task itself
2016-06-08 07:25:08 +02:00
*/
void lv_disp_refr_task(lv_task_t * task)
2016-06-08 07:25:08 +02:00
{
2019-02-12 12:21:34 +01:00
LV_LOG_TRACE("lv_refr_task: started");
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
disp_refr = task->user_data;
2019-01-29 04:58:18 +01:00
lv_refr_join_area();
2018-06-19 09:49:58 +02:00
lv_refr_areas();
/*If refresh happened ...*/
if(disp_refr->inv_p != 0) {
2019-04-04 07:15:40 +02:00
/*In true double buffered mode copy the refreshed areas to the new VDB to keep it up to
* date*/
if(lv_disp_is_true_double_buf(disp_refr)) {
lv_disp_buf_t * vdb = lv_disp_get_buf(disp_refr);
2019-02-13 01:40:22 +01:00
/*Flush the content of the VDB*/
lv_refr_vdb_flush();
2019-02-13 01:40:22 +01:00
2019-04-04 07:15:40 +02:00
/* With true double buffering the flushing should be only the address change of the
* current frame buffer. Wait until the address change is ready and copy the changed
* content to the other frame buffer (new active VDB) to keep the buffers synchronized*/
2019-06-06 06:05:40 +02:00
while(vdb->flushing)
;
2019-02-13 01:40:22 +01:00
2019-04-04 07:15:40 +02:00
uint8_t * buf_act = (uint8_t *)vdb->buf_act;
uint8_t * buf_ina = (uint8_t *)vdb->buf_act == vdb->buf1 ? vdb->buf2 : vdb->buf1;
2019-02-13 01:40:22 +01:00
lv_coord_t hres = lv_disp_get_hor_res(disp_refr);
uint16_t a;
for(a = 0; a < disp_refr->inv_p; a++) {
if(disp_refr->inv_area_joined[a] == 0) {
lv_coord_t y;
2019-04-04 07:15:40 +02:00
uint32_t start_offs =
2019-06-06 06:05:40 +02:00
(hres * disp_refr->inv_areas[a].y1 + disp_refr->inv_areas[a].x1) * sizeof(lv_color_t);
uint32_t line_length = lv_area_get_width(&disp_refr->inv_areas[a]) * sizeof(lv_color_t);
2019-02-13 01:40:22 +01:00
for(y = disp_refr->inv_areas[a].y1; y <= disp_refr->inv_areas[a].y2; y++) {
memcpy(buf_act + start_offs, buf_ina + start_offs, line_length);
start_offs += hres * sizeof(lv_color_t);
2019-02-12 12:21:34 +01:00
}
}
}
2019-04-04 07:15:40 +02:00
} /*End of true double buffer handling*/
2018-12-29 09:12:32 +01:00
/*Clean up*/
memset(disp_refr->inv_areas, 0, sizeof(disp_refr->inv_areas));
memset(disp_refr->inv_area_joined, 0, sizeof(disp_refr->inv_area_joined));
disp_refr->inv_p = 0;
2018-12-29 09:12:32 +01:00
/*Call monitor cb if present*/
if(disp_refr->driver.monitor_cb) {
disp_refr->driver.monitor_cb(&disp_refr->driver, lv_tick_elaps(start), px_num);
2017-07-09 15:32:49 +02:00
}
}
2018-07-25 20:39:24 +02:00
lv_mem_buf_free_all();
2019-06-26 15:56:41 +02:00
2019-02-12 12:21:34 +01:00
LV_LOG_TRACE("lv_refr_task: ready");
2016-06-08 07:25:08 +02:00
}
/**********************
* STATIC FUNCTIONS
**********************/
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;
2017-11-23 21:28:36 +01:00
lv_area_t joined_area;
for(join_in = 0; join_in < disp_refr->inv_p; join_in++) {
if(disp_refr->inv_area_joined[join_in] != 0) continue;
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*Check all areas to join them in 'join_in'*/
for(join_from = 0; join_from < disp_refr->inv_p; join_from++) {
2016-06-08 07:25:08 +02:00
/*Handle only unjoined areas and ignore itself*/
if(disp_refr->inv_area_joined[join_from] != 0 || join_in == join_from) {
2016-06-08 07:25:08 +02:00
continue;
}
/*Check if the areas are on each other*/
2019-06-06 06:05:40 +02:00
if(lv_area_is_on(&disp_refr->inv_areas[join_in], &disp_refr->inv_areas[join_from]) == false) {
2016-06-08 07:25:08 +02:00
continue;
}
2018-06-19 09:49:58 +02:00
2019-06-06 06:05:40 +02:00
lv_area_join(&joined_area, &disp_refr->inv_areas[join_in], &disp_refr->inv_areas[join_from]);
2016-06-08 07:25:08 +02:00
/*Join two area only if the joined area size is smaller*/
2019-06-06 06:05:40 +02:00
if(lv_area_get_size(&joined_area) < (lv_area_get_size(&disp_refr->inv_areas[join_in]) +
lv_area_get_size(&disp_refr->inv_areas[join_from]))) {
lv_area_copy(&disp_refr->inv_areas[join_in], &joined_area);
2016-06-08 07:25:08 +02:00
/*Mark 'join_form' is joined into 'join_in'*/
disp_refr->inv_area_joined[join_from] = 1;
2016-06-08 07:25:08 +02:00
}
2018-06-19 09:49:58 +02:00
}
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 < disp_refr->inv_p; i++) {
2016-06-08 07:25:08 +02:00
/*Refresh the unjoined areas*/
if(disp_refr->inv_area_joined[i] == 0) {
2019-02-13 01:40:22 +01:00
lv_refr_area(&disp_refr->inv_areas[i]);
2019-02-20 23:58:13 +01:00
if(disp_refr->driver.monitor_cb) px_num += lv_area_get_size(&disp_refr->inv_areas[i]);
2016-06-08 07:25:08 +02:00
}
}
}
/**
* Refresh an area if there is Virtual Display Buffer
* @param area_p pointer to an area to refresh
*/
2019-02-13 01:40:22 +01:00
static void lv_refr_area(const lv_area_t * area_p)
2016-06-08 07:25:08 +02:00
{
2019-04-04 07:15:40 +02:00
/*True double buffering: there are two screen sized buffers. Just redraw directly into a
* buffer*/
2019-02-20 23:58:13 +01:00
if(lv_disp_is_true_double_buf(disp_refr)) {
lv_disp_buf_t * vdb = lv_disp_get_buf(disp_refr);
2019-04-04 07:15:40 +02:00
vdb->area.x1 = 0;
vdb->area.x2 = lv_disp_get_hor_res(disp_refr) - 1;
vdb->area.y1 = 0;
vdb->area.y2 = lv_disp_get_ver_res(disp_refr) - 1;
2019-02-13 01:40:22 +01:00
lv_refr_area_part(area_p);
}
2019-02-13 01:40:22 +01:00
/*The buffer is smaller: refresh the area in parts*/
else {
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp_refr);
2019-02-13 01:40:22 +01:00
/*Calculate the max row num*/
2019-06-06 06:05:40 +02:00
lv_coord_t w = lv_area_get_width(area_p);
lv_coord_t h = lv_area_get_height(area_p);
lv_coord_t y2 =
area_p->y2 >= lv_disp_get_ver_res(disp_refr) ? y2 = lv_disp_get_ver_res(disp_refr) - 1 : area_p->y2;
2019-02-13 01:40:22 +01:00
2019-04-04 07:15:40 +02:00
int32_t max_row = (uint32_t)vdb->size / w;
2019-02-13 01:40:22 +01:00
if(max_row > h) max_row = h;
/*Round down the lines of VDB if rounding is added*/
2019-02-20 23:58:13 +01:00
if(disp_refr->driver.rounder_cb) {
2019-02-13 01:40:22 +01:00
lv_area_t tmp;
tmp.x1 = 0;
tmp.x2 = 0;
tmp.y1 = 0;
lv_coord_t h_tmp = max_row;
2019-02-13 01:40:22 +01:00
do {
tmp.y2 = h_tmp - 1;
2019-04-02 12:16:15 +02:00
disp_refr->driver.rounder_cb(&disp_refr->driver, &tmp);
2019-04-02 06:49:05 +02:00
/*If this height fits into `max_row` then fine*/
if(lv_area_get_height(&tmp) <= max_row) break;
/*Decrement the height of the area until it fits into `max_row` after rounding*/
h_tmp--;
} while(h_tmp > 0);
2019-02-13 01:40:22 +01:00
if(h_tmp <= 0) {
2019-04-04 07:15:40 +02:00
LV_LOG_WARN("Can't set VDB height using the round function. (Wrong round_cb or to "
"small VDB)");
2019-02-13 01:40:22 +01:00
return;
} else {
max_row = tmp.y2 + 1;
}
2018-09-12 09:03:48 +02:00
}
2019-02-13 01:40:22 +01:00
/*Always use the full row*/
lv_coord_t row;
lv_coord_t row_last = 0;
2019-04-04 07:15:40 +02:00
for(row = area_p->y1; row + max_row - 1 <= y2; row += max_row) {
2019-02-13 01:40:22 +01:00
/*Calc. the next y coordinates of VDB*/
vdb->area.x1 = area_p->x1;
vdb->area.x2 = area_p->x2;
vdb->area.y1 = row;
vdb->area.y2 = row + max_row - 1;
if(vdb->area.y2 > y2) vdb->area.y2 = y2;
row_last = vdb->area.y2;
lv_refr_area_part(area_p);
2018-10-05 17:22:49 +02:00
}
2019-02-13 01:40:22 +01:00
/*If the last y coordinates are not handled yet ...*/
if(y2 != row_last) {
/*Calc. the next y coordinates of VDB*/
vdb->area.x1 = area_p->x1;
vdb->area.x2 = area_p->x2;
vdb->area.y1 = row;
vdb->area.y2 = y2;
2016-06-08 07:25:08 +02:00
2019-02-13 01:40:22 +01:00
/*Refresh this part too*/
lv_refr_area_part(area_p);
}
2016-06-08 07:25:08 +02:00
}
}
/**
* Refresh a part of an area which is on the actual Virtual Display Buffer
* @param area_p pointer to an area to refresh
*/
2019-02-13 01:40:22 +01:00
static void lv_refr_area_part(const lv_area_t * area_p)
2016-06-08 07:25:08 +02:00
{
2019-02-13 01:40:22 +01:00
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp_refr);
2019-02-13 01:40:22 +01:00
2019-04-04 07:15:40 +02:00
/*In non double buffered mode, before rendering the next part wait until the previous image is
* flushed*/
2019-02-20 23:58:13 +01:00
if(lv_disp_is_double_buf(disp_refr) == false) {
2019-06-06 06:05:40 +02:00
while(vdb->flushing)
;
2018-10-05 17:22:49 +02:00
}
2019-02-13 01:40:22 +01:00
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'*/
2017-11-23 21:28:36 +01:00
lv_area_t start_mask;
2019-02-13 01:40:22 +01:00
lv_area_intersect(&start_mask, area_p, &vdb->area);
2016-06-08 07:25:08 +02:00
/*Get the most top object which is not covered by others*/
2019-02-20 23:58:13 +01:00
top_p = lv_refr_get_top_obj(&start_mask, lv_disp_get_scr_act(disp_refr));
2016-06-08 07:25:08 +02:00
/*Do the refreshing from the top object*/
lv_refr_obj_and_children(top_p, &start_mask);
2017-10-10 16:17:23 +02:00
/*Also refresh top and sys layer unconditionally*/
2019-02-20 23:58:13 +01:00
lv_refr_obj_and_children(lv_disp_get_layer_top(disp_refr), &start_mask);
lv_refr_obj_and_children(lv_disp_get_layer_sys(disp_refr), &start_mask);
2017-10-10 16:17:23 +02:00
2018-12-29 09:12:32 +01:00
/* In true double buffered mode flush only once when all areas were rendered.
* In normal mode flush after every area */
2019-02-20 23:58:13 +01:00
if(lv_disp_is_true_double_buf(disp_refr) == false) {
2019-02-13 01:40:22 +01:00
lv_refr_vdb_flush();
}
2016-06-08 07:25:08 +02:00
}
/**
* 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)
2018-06-19 09:49:58 +02:00
* @return
2016-06-08 07:25:08 +02:00
*/
2017-11-23 21:28:36 +01:00
static lv_obj_t * lv_refr_get_top_obj(const lv_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 * found_p = NULL;
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*If this object is fully cover the draw area check the children too */
2018-06-19 09:49:58 +02:00
if(lv_area_is_in(area_p, &obj->coords) && obj->hidden == 0) {
2019-09-06 19:53:39 +02:00
lv_design_res_t design_res = obj->design_cb(obj, area_p, LV_DESIGN_COVER_CHK);
if(design_res == LV_DESIGN_RES_MASKED) return NULL;
lv_obj_t * i;
2019-04-04 07:15:40 +02:00
LV_LL_READ(obj->child_ll, i)
{
2016-06-08 07:25:08 +02:00
found_p = lv_refr_get_top_obj(area_p, i);
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*If a children is ok then break*/
if(found_p != NULL) {
break;
}
}
2018-06-19 09:49:58 +02:00
/*If no better children check this object*/
2016-06-08 07:25:08 +02:00
if(found_p == NULL) {
2019-12-14 23:39:26 +01:00
if(lv_obj_get_style_value(obj, LV_STYLE_BG_OPA) == LV_OPA_COVER && design_res == LV_DESIGN_RES_COVER &&
lv_obj_get_opa_scale(obj) == LV_OPA_COVER) {
found_p = obj;
}
2016-06-08 07:25:08 +02:00
}
}
2018-06-19 09:49:58 +02:00
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
*/
static void lv_refr_obj_and_children(lv_obj_t * top_p, const lv_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 */
2019-02-20 23:58:13 +01:00
if(top_p == NULL) top_p = lv_disp_get_scr_act(disp_refr);
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*Refresh the top object and its children*/
lv_refr_obj(top_p, mask_p);
2018-06-19 09:49:58 +02:00
/*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 * 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*/
lv_obj_t * i = lv_ll_get_prev(&(par->child_ll), border_p);
2016-06-08 07:25:08 +02:00
2018-06-19 09:49:58 +02:00
while(i != NULL) {
2016-06-08 07:25:08 +02:00
/*Refresh the objects*/
lv_refr_obj(i, mask_p);
2017-11-24 17:48:47 +01:00
i = lv_ll_get_prev(&(par->child_ll), i);
2018-06-19 09:49:58 +02:00
}
/*Call the post draw design function of the parents of the to object*/
par->design_cb(par, mask_p, LV_DESIGN_DRAW_POST);
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
}
}
/**
* 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
*/
2017-11-23 21:28:36 +01:00
static void lv_refr_obj(lv_obj_t * obj, const lv_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;
2018-06-19 09:49:58 +02:00
2019-04-04 07:15:40 +02:00
bool union_ok; /* Store the return value of area_union */
2016-06-08 07:25:08 +02:00
/* Truncate the original mask to the coordinates of the parent
* because the parent and its children are visible only here */
2017-11-23 21:28:36 +01:00
lv_area_t obj_mask;
lv_area_t obj_ext_mask;
lv_area_t obj_area;
lv_coord_t ext_size = obj->ext_draw_pad;
lv_obj_get_coords(obj, &obj_area);
2016-10-17 15:02:18 +02:00
obj_area.x1 -= ext_size;
obj_area.y1 -= ext_size;
obj_area.x2 += ext_size;
obj_area.y2 += ext_size;
union_ok = lv_area_intersect(&obj_ext_mask, mask_ori_p, &obj_area);
2018-06-19 09:49:58 +02:00
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 */
obj->design_cb(obj, &obj_ext_mask, LV_DESIGN_DRAW_MAIN);
2016-06-08 07:25:08 +02:00
2019-05-09 06:04:03 +02:00
#if MASK_AREA_DEBUG
static lv_color_t debug_color = LV_COLOR_RED;
2019-11-02 21:30:51 +01:00
LV_STYLE_CREATE(style_debug, &lv_style_plain);
style_debug.body.main_color = debug_color;
style_debug.body.grad_color = debug_color;
style_debug.body.border.width = 2;
style_debug.body.border.color.full = (debug_color.full + 0x13) * 9;
lv_draw_rect(&obj_ext_mask, &obj_ext_mask, &style_debug, LV_OPA_50);
2019-05-09 06:04:03 +02:00
debug_color.full *= 17;
debug_color.full += 0xA1;
#endif
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_coords(obj, &obj_area);
union_ok = lv_area_intersect(&obj_mask, mask_ori_p, &obj_area);
2016-10-20 16:35:03 +02:00
if(union_ok != false) {
2018-06-19 09:49:58 +02:00
lv_area_t mask_child; /*Mask from obj and its child*/
lv_obj_t * child_p;
lv_area_t child_area;
2019-04-04 07:15:40 +02:00
LV_LL_READ_BACK(obj->child_ll, child_p)
{
2018-06-19 09:49:58 +02:00
lv_obj_get_coords(child_p, &child_area);
ext_size = child_p->ext_draw_pad;
2018-06-19 09:49:58 +02:00
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 = lv_area_intersect(&mask_child, &obj_mask, &child_area);
2018-06-19 09:49:58 +02:00
/*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 */
obj->design_cb(obj, &obj_ext_mask, LV_DESIGN_DRAW_POST);
2016-06-08 07:25:08 +02:00
}
}
2019-02-13 01:40:22 +01:00
/**
* Flush the content of the VDB
*/
static void lv_refr_vdb_flush(void)
{
2019-03-08 07:54:36 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp_refr);
2019-02-13 01:40:22 +01:00
2019-04-04 07:15:40 +02:00
/*In double buffered mode wait until the other buffer is flushed before flushing the current
* one*/
2019-03-08 07:54:36 +01:00
if(lv_disp_is_double_buf(disp_refr)) {
2019-06-27 07:16:15 +02:00
while(vdb->flushing)
;
2019-02-13 01:40:22 +01:00
}
2019-02-20 10:16:33 +01:00
vdb->flushing = 1;
2019-02-13 01:40:22 +01:00
/*Flush the rendered content to the display*/
lv_disp_t * disp = lv_refr_get_disp_refreshing();
if(disp->driver.flush_cb) disp->driver.flush_cb(&disp->driver, &vdb->area, vdb->buf_act);
2019-02-13 01:40:22 +01:00
if(vdb->buf1 && vdb->buf2) {
2019-06-27 07:16:15 +02:00
if(vdb->buf_act == vdb->buf1)
vdb->buf_act = vdb->buf2;
else
vdb->buf_act = vdb->buf1;
2019-02-13 01:40:22 +01:00
}
}