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"
|
|
|
|
#include "lv_vdb.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"
|
2016-06-08 07:25:08 +02:00
|
|
|
|
|
|
|
/*********************
|
|
|
|
* DEFINES
|
|
|
|
*********************/
|
2018-12-29 09:12:32 +01:00
|
|
|
#ifndef LV_INV_FIFO_SIZE
|
|
|
|
#define LV_INV_FIFO_SIZE 32 /*The average count of objects on a screen */
|
|
|
|
#endif
|
2017-07-07 18:40:35 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* TYPEDEFS
|
|
|
|
**********************/
|
2018-06-19 09:49:58 +02:00
|
|
|
typedef struct {
|
2017-11-23 21:28:36 +01:00
|
|
|
lv_area_t area;
|
2016-06-08 07:25:08 +02:00
|
|
|
uint8_t joined;
|
2018-06-19 09:49:58 +02:00
|
|
|
} lv_join_t;
|
2016-06-08 07:25:08 +02:00
|
|
|
|
|
|
|
/**********************
|
|
|
|
* STATIC PROTOTYPES
|
|
|
|
**********************/
|
2017-01-06 13:30:57 +01:00
|
|
|
static void lv_refr_task(void * param);
|
2016-12-21 14:49:23 +01:00
|
|
|
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
|
2017-11-23 21:28:36 +01:00
|
|
|
static void lv_refr_area_no_vdb(const lv_area_t * area_p);
|
2016-06-08 07:25:08 +02:00
|
|
|
#else
|
2017-11-23 21:28:36 +01:00
|
|
|
static void lv_refr_area_with_vdb(const lv_area_t * area_p);
|
|
|
|
static void lv_refr_area_part_vdb(const lv_area_t * area_p);
|
2016-06-08 07:25:08 +02:00
|
|
|
#endif
|
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);
|
2017-12-03 00:36:31 +01:00
|
|
|
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);
|
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;
|
2018-02-24 11:55:39 +01:00
|
|
|
static void (*monitor_cb)(uint32_t, uint32_t); /*Monitor the rendering time*/
|
2018-06-19 09:49:58 +02:00
|
|
|
static void (*round_cb)(lv_area_t *); /*If set then called to modify invalidated areas for special display controllers*/
|
2017-07-09 15:32:49 +02:00
|
|
|
static uint32_t px_num;
|
2019-02-12 12:21:34 +01: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
|
|
|
{
|
2016-12-21 14:49:23 +01:00
|
|
|
inv_buf_p = 0;
|
|
|
|
memset(inv_buf, 0, sizeof(inv_buf));
|
|
|
|
|
2018-06-19 09:49:58 +02:00
|
|
|
lv_task_t * task;
|
2017-11-24 17:48:47 +01:00
|
|
|
task = lv_task_create(lv_refr_task, LV_REFR_PERIOD, LV_TASK_PRIO_MID, NULL);
|
2018-05-17 17:07:21 +02:00
|
|
|
lv_task_ready(task); /*Be sure the screen will be refreshed immediately on start up*/
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
|
2018-09-20 21:59:44 +02:00
|
|
|
/**
|
|
|
|
* Redraw the invalidated areas now.
|
2018-12-26 07:58:06 +01:00
|
|
|
* Normally the redrawing is periodically executed in `lv_task_handler` but a long blocking process can
|
2018-09-20 21:59:44 +02:00
|
|
|
* prevent the call of `lv_task_handler`. In this case if the the GUI is updated in the process (e.g. progress bar)
|
2018-12-26 07:58:06 +01:00
|
|
|
* this function can be called when the screen should be updated.
|
2018-09-20 21:59:44 +02:00
|
|
|
*/
|
|
|
|
void lv_refr_now(void)
|
|
|
|
{
|
|
|
|
lv_refr_task(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**
|
|
|
|
* Invalidate an area
|
|
|
|
* @param area_p pointer to area which should be invalidated
|
|
|
|
*/
|
2017-11-23 21:28:36 +01:00
|
|
|
void lv_inv_area(const lv_area_t * area_p)
|
2016-06-08 07:25:08 +02:00
|
|
|
{
|
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;
|
|
|
|
}
|
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-02-07 19:17:10 +01:00
|
|
|
scr_area.x2 = LV_HOR_RES_MAX - 1;
|
|
|
|
scr_area.y2 = LV_VER_RES_MAX - 1;
|
2018-06-19 09:49:58 +02:00
|
|
|
|
|
|
|
lv_area_t com_area;
|
2016-06-08 07:25:08 +02:00
|
|
|
bool suc;
|
|
|
|
|
2018-07-17 16:40:52 +02:00
|
|
|
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) {
|
2018-02-24 11:55:39 +01:00
|
|
|
if(round_cb) round_cb(&com_area);
|
2018-01-09 12:38:52 -06:00
|
|
|
|
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 < inv_buf_p; i++) {
|
|
|
|
if(lv_area_is_in(&com_area, &inv_buf[i].area) != false) return;
|
|
|
|
}
|
2016-12-21 14:49:23 +01:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Save the area*/
|
2018-06-19 09:49:58 +02:00
|
|
|
if(inv_buf_p < LV_INV_FIFO_SIZE) {
|
|
|
|
lv_area_copy(&inv_buf[inv_buf_p].area, &com_area);
|
|
|
|
} else {/*If no place for the area add the screen*/
|
|
|
|
inv_buf_p = 0;
|
|
|
|
lv_area_copy(&inv_buf[inv_buf_p].area, &scr_area);
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
2018-06-19 09:49:58 +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
|
2017-08-04 19:09:29 +02:00
|
|
|
* (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;
|
|
|
|
}
|
|
|
|
|
2018-02-24 11:55:39 +01:00
|
|
|
/**
|
|
|
|
* Called when an area is invalidated to modify the coordinates of the area.
|
|
|
|
* Special display controllers may require special coordinate rounding
|
|
|
|
* @param cb pointer to the a function which will modify the area
|
|
|
|
*/
|
2018-06-19 09:49:58 +02:00
|
|
|
void lv_refr_set_round_cb(void(*cb)(lv_area_t *))
|
2018-02-24 11:55:39 +01:00
|
|
|
{
|
|
|
|
round_cb = cb;
|
|
|
|
}
|
|
|
|
|
2018-02-24 15:36:06 +01:00
|
|
|
/**
|
|
|
|
* Get the number of areas in the buffer
|
|
|
|
* @return number of invalid areas
|
|
|
|
*/
|
|
|
|
uint16_t lv_refr_get_buf_size(void)
|
|
|
|
{
|
|
|
|
return inv_buf_p;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pop (delete) the last 'num' invalidated areas from the buffer
|
|
|
|
* @param num number of areas to delete
|
|
|
|
*/
|
|
|
|
void lv_refr_pop_from_buf(uint16_t num)
|
|
|
|
{
|
|
|
|
if(inv_buf_p < num) inv_buf_p = 0;
|
|
|
|
else inv_buf_p -= num;
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC FUNCTIONS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called periodically to handle the refreshing
|
2017-01-06 13:30:57 +01:00
|
|
|
* @param param unused
|
2016-06-08 07:25:08 +02:00
|
|
|
*/
|
2017-01-06 13:30:57 +01:00
|
|
|
static void lv_refr_task(void * param)
|
2016-06-08 07:25:08 +02:00
|
|
|
{
|
2017-12-02 20:43:50 +01:00
|
|
|
(void)param;
|
2018-07-25 20:39:24 +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
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
LL_READ(LV_GC_ROOT(_lv_disp_ll), disp_refr) {
|
|
|
|
LV_LOG_TRACE("lv_refr_task: refreshing a display");
|
2019-01-29 04:58:18 +01:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
lv_refr_join_area();
|
2018-06-19 09:49:58 +02:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
lv_refr_areas();
|
2016-12-21 14:49:23 +01:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
/*If refresh happened ...*/
|
|
|
|
if(inv_buf_p != 0) {
|
2017-07-09 15:32:49 +02:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
/*In true double buffered mode copy the refreshed areas to the new VDB to keep it up to date*/
|
|
|
|
#if LV_VDB_TRUE_DOUBLE_BUFFERED
|
|
|
|
lv_vdb_t * vdb_p = lv_vdb_get();
|
|
|
|
vdb_p->area.x1 = 0;
|
|
|
|
vdb_p->area.x2 = LV_HOR_RES-1;
|
|
|
|
vdb_p->area.y1 = 0;
|
|
|
|
vdb_p->area.y2 = LV_VER_RES - 1;
|
|
|
|
|
|
|
|
/*Flush the content of the VDB*/
|
|
|
|
lv_vdb_flush();
|
|
|
|
|
|
|
|
/* 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 active content to the other frame buffer (new active VDB)
|
|
|
|
* The changes will be written to the new VDB.*/
|
|
|
|
lv_vdb_t * vdb_act = lv_vdb_get_active();
|
|
|
|
lv_vdb_t * vdb_ina = lv_vdb_get_inactive();
|
|
|
|
|
|
|
|
uint8_t * buf_act = (uint8_t *) vdb_act->buf;
|
|
|
|
uint8_t * buf_ina = (uint8_t *) vdb_ina->buf;
|
|
|
|
|
|
|
|
uint16_t a;
|
|
|
|
for(a = 0; a < inv_buf_p; a++) {
|
|
|
|
if(inv_buf[a].joined == 0) {
|
|
|
|
lv_coord_t y;
|
|
|
|
uint32_t start_offs = ((LV_HOR_RES * inv_buf[a].area.y1 + inv_buf[a].area.x1) * LV_VDB_PX_BPP) >> 3;
|
|
|
|
uint32_t line_length = (lv_area_get_width(&inv_buf[a].area) * LV_VDB_PX_BPP) >> 3;
|
|
|
|
|
|
|
|
for(y = inv_buf[a].area.y1; y <= inv_buf[a].area.y2; y++) {
|
|
|
|
memcpy(buf_act + start_offs, buf_ina + start_offs, line_length);
|
|
|
|
start_offs += (LV_HOR_RES * LV_VDB_PX_BPP) >> 3;
|
|
|
|
}
|
2019-01-04 08:33:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
#endif
|
2018-12-29 09:12:32 +01:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
/*Clean up*/
|
|
|
|
memset(inv_buf, 0, sizeof(inv_buf));
|
|
|
|
inv_buf_p = 0;
|
2018-12-29 09:12:32 +01:00
|
|
|
|
2019-02-12 12:21:34 +01:00
|
|
|
/*Call monitor cb if present*/
|
|
|
|
if(monitor_cb != NULL) {
|
|
|
|
monitor_cb(lv_tick_elaps(start), px_num);
|
|
|
|
}
|
2017-07-09 15:32:49 +02:00
|
|
|
}
|
|
|
|
}
|
2018-07-25 20:39:24 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Join the areas which has got common parts
|
|
|
|
*/
|
2016-12-21 14:49:23 +01:00
|
|
|
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;
|
2016-12-21 14:49:23 +01:00
|
|
|
for(join_in = 0; join_in < inv_buf_p; join_in++) {
|
|
|
|
if(inv_buf[join_in].joined != 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'*/
|
2016-12-21 14:49:23 +01:00
|
|
|
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*/
|
2016-12-21 14:49:23 +01:00
|
|
|
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*/
|
2017-11-23 21:28:36 +01:00
|
|
|
if(lv_area_is_on(&inv_buf[join_in].area,
|
2018-06-19 09:49:58 +02:00
|
|
|
&inv_buf[join_from].area) == false) {
|
2016-06-08 07:25:08 +02:00
|
|
|
continue;
|
|
|
|
}
|
2018-06-19 09:49:58 +02:00
|
|
|
|
2017-11-23 21:28:36 +01:00
|
|
|
lv_area_join(&joined_area, &inv_buf[join_in].area,
|
2018-06-19 09:49:58 +02:00
|
|
|
&inv_buf[join_from].area);
|
2016-06-08 07:25:08 +02:00
|
|
|
|
|
|
|
/*Join two area only if the joined area size is smaller*/
|
2018-06-19 09:49:58 +02:00
|
|
|
if(lv_area_get_size(&joined_area) <
|
|
|
|
(lv_area_get_size(&inv_buf[join_in].area) + lv_area_get_size(&inv_buf[join_from].area))) {
|
2017-11-28 16:15:13 +01:00
|
|
|
lv_area_copy(&inv_buf[join_in].area, &joined_area);
|
2016-06-08 07:25:08 +02:00
|
|
|
|
|
|
|
/*Mark 'join_form' is joined into 'join_in'*/
|
2016-12-21 14:49:23 +01:00
|
|
|
inv_buf[join_from].joined = 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
|
|
|
|
*/
|
2016-12-21 14:49:23 +01:00
|
|
|
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;
|
2017-06-17 09:08:54 +02:00
|
|
|
|
2016-12-21 14:49:23 +01:00
|
|
|
for(i = 0; i < inv_buf_p; i++) {
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Refresh the unjoined areas*/
|
2016-12-21 14:49:23 +01:00
|
|
|
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
|
2016-12-21 14:49:23 +01:00
|
|
|
lv_refr_area_no_vdb(&inv_buf[i].area);
|
2016-06-08 07:25:08 +02:00
|
|
|
#else
|
|
|
|
/*If VDB is used...*/
|
2016-12-21 14:49:23 +01:00
|
|
|
lv_refr_area_with_vdb(&inv_buf[i].area);
|
2016-06-08 07:25:08 +02:00
|
|
|
#endif
|
2017-11-23 21:28:36 +01:00
|
|
|
if(monitor_cb != NULL) px_num += lv_area_get_size(&inv_buf[i].area);
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-17 09:08:54 +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
|
|
|
|
*/
|
2017-11-23 21:28:36 +01:00
|
|
|
static void lv_refr_area_no_vdb(const lv_area_t * area_p)
|
2016-06-08 07:25:08 +02:00
|
|
|
{
|
2016-10-07 11:15:46 +02:00
|
|
|
lv_obj_t * top_p;
|
2018-06-19 09:49:58 +02:00
|
|
|
|
|
|
|
/*Get top object which is not covered by others*/
|
2017-10-20 15:37:50 +02:00
|
|
|
top_p = lv_refr_get_top_obj(area_p, lv_scr_act());
|
2018-06-19 09:49:58 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Do the refreshing*/
|
2017-12-03 00:36:31 +01:00
|
|
|
lv_refr_obj_and_children(top_p, area_p);
|
2018-09-04 07:25:07 +02:00
|
|
|
|
|
|
|
/*Also refresh top and sys layer unconditionally*/
|
2018-09-23 21:54:55 +02:00
|
|
|
lv_refr_obj_and_children(lv_layer_top(), area_p);
|
|
|
|
lv_refr_obj_and_children(lv_layer_sys(), area_p);
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Refresh an area if there is Virtual Display Buffer
|
|
|
|
* @param area_p pointer to an area to refresh
|
|
|
|
*/
|
2017-11-23 21:28:36 +01:00
|
|
|
static void lv_refr_area_with_vdb(const lv_area_t * area_p)
|
2016-06-08 07:25:08 +02:00
|
|
|
{
|
2018-12-29 09:12:32 +01:00
|
|
|
|
|
|
|
#if LV_VDB_TRUE_DOUBLE_BUFFERED == 0
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Calculate the max row num*/
|
2017-12-07 19:22:23 +01:00
|
|
|
lv_coord_t w = lv_area_get_width(area_p);
|
|
|
|
lv_coord_t h = lv_area_get_height(area_p);
|
2019-02-07 19:17:10 +01:00
|
|
|
lv_coord_t y2 = area_p->y2 >= lv_disp_get_ver_res(NULL) ? y2 = lv_disp_get_ver_res(NULL) - 1 : area_p->y2;
|
2017-12-07 19:22:23 +01:00
|
|
|
|
2018-10-05 14:44:17 +02:00
|
|
|
int32_t max_row = (uint32_t) LV_VDB_SIZE / w;
|
2018-08-04 01:46:00 +02:00
|
|
|
|
2018-02-15 10:12:28 +01:00
|
|
|
if(max_row > h) max_row = h;
|
2016-06-08 07:25:08 +02:00
|
|
|
|
2018-08-04 01:46:00 +02:00
|
|
|
|
|
|
|
/*Round down the lines of VDB if rounding is added*/
|
|
|
|
if(round_cb) {
|
2018-10-05 17:22:49 +02:00
|
|
|
lv_area_t tmp;
|
|
|
|
tmp.x1 = 0;
|
|
|
|
tmp.x2 = 0;
|
|
|
|
tmp.y1 = 0;
|
|
|
|
tmp.y2 = max_row;
|
|
|
|
|
|
|
|
lv_coord_t y_tmp = max_row;
|
|
|
|
do {
|
|
|
|
tmp.y2 = y_tmp;
|
|
|
|
round_cb(&tmp);
|
|
|
|
y_tmp --; /*Decrement the number of line until it is rounded to a smaller (or equal) value then the original. */
|
|
|
|
} while(lv_area_get_height(&tmp) > max_row && y_tmp != 0);
|
|
|
|
|
|
|
|
if(y_tmp == 0) {
|
|
|
|
LV_LOG_WARN("Can't set VDB height using the round function. (Wrong round_cb or to small VDB)");
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
max_row = tmp.y2 + 1;
|
|
|
|
}
|
2018-08-04 01:46:00 +02:00
|
|
|
}
|
|
|
|
|
2017-08-04 19:09:29 +02:00
|
|
|
/*Always use the full row*/
|
2018-09-25 11:16:37 +02:00
|
|
|
lv_coord_t row;
|
2017-11-23 21:28:36 +01:00
|
|
|
lv_coord_t row_last = 0;
|
2017-12-20 16:51:34 +01:00
|
|
|
for(row = area_p->y1; row + max_row - 1 <= y2; row += max_row) {
|
2017-08-04 19:09:29 +02:00
|
|
|
lv_vdb_t * vdb_p = lv_vdb_get();
|
2018-09-12 09:03:48 +02:00
|
|
|
if(!vdb_p) {
|
|
|
|
LV_LOG_WARN("Invalid VDB pointer");
|
|
|
|
return;
|
|
|
|
}
|
2017-08-04 19:09:29 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Calc. the next y coordinates of VDB*/
|
2017-08-04 19:09:29 +02:00
|
|
|
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;
|
2017-12-20 16:51:34 +01:00
|
|
|
if(vdb_p->area.y2 > y2) vdb_p->area.y2 = y2;
|
|
|
|
row_last = vdb_p->area.y2;
|
2016-06-08 07:25:08 +02:00
|
|
|
lv_refr_area_part_vdb(area_p);
|
|
|
|
}
|
2018-06-19 09:49:58 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/*If the last y coordinates are not handled yet ...*/
|
2017-12-20 16:51:34 +01:00
|
|
|
if(y2 != row_last) {
|
2017-08-04 19:09:29 +02:00
|
|
|
lv_vdb_t * vdb_p = lv_vdb_get();
|
2018-09-12 09:03:48 +02:00
|
|
|
if(!vdb_p) {
|
2018-10-05 17:22:49 +02:00
|
|
|
LV_LOG_WARN("Invalid VDB pointer");
|
|
|
|
return;
|
|
|
|
}
|
2017-08-04 19:09:29 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/*Calc. the next y coordinates of VDB*/
|
2017-08-04 19:09:29 +02:00
|
|
|
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;
|
2017-12-20 16:51:34 +01:00
|
|
|
vdb_p->area.y2 = y2;
|
2016-06-08 07:25:08 +02:00
|
|
|
|
|
|
|
/*Refresh this part too*/
|
|
|
|
lv_refr_area_part_vdb(area_p);
|
|
|
|
}
|
2018-12-29 09:12:32 +01:00
|
|
|
#else
|
|
|
|
lv_vdb_t * vdb_p = lv_vdb_get();
|
|
|
|
vdb_p->area.x1 = 0;
|
|
|
|
vdb_p->area.x2 = LV_HOR_RES-1;
|
|
|
|
vdb_p->area.y1 = 0;
|
|
|
|
vdb_p->area.y2 = LV_VER_RES - 1;
|
|
|
|
lv_refr_area_part_vdb(area_p);
|
|
|
|
#endif
|
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
|
|
|
|
*/
|
2017-11-23 21:28:36 +01:00
|
|
|
static void lv_refr_area_part_vdb(const lv_area_t * area_p)
|
2016-06-08 07:25:08 +02:00
|
|
|
{
|
|
|
|
lv_vdb_t * vdb_p = lv_vdb_get();
|
2018-09-12 09:03:48 +02:00
|
|
|
if(!vdb_p) {
|
2018-10-05 17:22:49 +02:00
|
|
|
LV_LOG_WARN("Invalid VDB pointer");
|
|
|
|
return;
|
|
|
|
}
|
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;
|
2018-07-17 16:40:52 +02:00
|
|
|
lv_area_intersect(&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*/
|
2019-02-12 12:21:34 +01:00
|
|
|
top_p = lv_refr_get_top_obj(&start_mask, lv_scr_act(disp_refr));
|
2016-06-08 07:25:08 +02:00
|
|
|
|
2017-01-01 22:54:59 +01:00
|
|
|
/*Do the refreshing from the top object*/
|
2017-12-03 00:36:31 +01:00
|
|
|
lv_refr_obj_and_children(top_p, &start_mask);
|
2017-06-19 12:55:02 +02:00
|
|
|
|
2017-10-10 16:17:23 +02:00
|
|
|
/*Also refresh top and sys layer unconditionally*/
|
2019-02-12 12:21:34 +01:00
|
|
|
lv_refr_obj_and_children(lv_layer_top(disp_refr), &start_mask);
|
|
|
|
lv_refr_obj_and_children(lv_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 */
|
|
|
|
#if LV_VDB_TRUE_DOUBLE_BUFFERED == 0
|
2018-06-19 09:49:58 +02:00
|
|
|
/*Flush the content of the VDB*/
|
2016-06-08 07:25:08 +02:00
|
|
|
lv_vdb_flush();
|
2018-12-29 09:12:32 +01:00
|
|
|
#endif
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#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)
|
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 * i;
|
|
|
|
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) {
|
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);
|
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
|
|
|
|
2016-12-21 14:49:23 +01:00
|
|
|
/*If no better children check this object*/
|
2016-06-08 07:25:08 +02:00
|
|
|
if(found_p == NULL) {
|
2017-04-13 10:20:35 +02:00
|
|
|
lv_style_t * style = lv_obj_get_style(obj);
|
2017-11-23 21:28:36 +01:00
|
|
|
if(style->body.opa == LV_OPA_COVER &&
|
2018-06-19 09:49:58 +02:00
|
|
|
obj->design_func(obj, area_p, LV_DESIGN_COVER_CHK) != false &&
|
|
|
|
lv_obj_get_opa_scale(obj) == LV_OPA_COVER) {
|
2016-12-21 14:49:23 +01:00
|
|
|
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
|
|
|
|
*/
|
2017-12-03 00:36:31 +01:00
|
|
|
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-12 12:21:34 +01:00
|
|
|
if(top_p == NULL) top_p = lv_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
|
|
|
|
2017-01-01 22:54:59 +01: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 * 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*/
|
2017-11-24 17:48:47 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
2017-01-01 22:54:59 +01:00
|
|
|
|
|
|
|
/*Call the post draw design function of the parents of the to object*/
|
|
|
|
par = lv_obj_get_parent(top_p);
|
|
|
|
while(par != NULL) {
|
2017-10-20 10:17:02 +02:00
|
|
|
par->design_func(par, mask_p, LV_DESIGN_DRAW_POST);
|
2017-01-01 22:54:59 +01: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
|
|
|
|
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 */
|
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_size;
|
2017-10-20 10:17:02 +02:00
|
|
|
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;
|
2018-07-17 16:40:52 +02:00
|
|
|
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) {
|
|
|
|
|
2017-04-13 10:20:35 +02:00
|
|
|
/* Redraw the object */
|
2018-11-21 14:00:19 +01:00
|
|
|
obj->design_func(obj, &obj_ext_mask, LV_DESIGN_DRAW_MAIN);
|
2018-12-22 19:08:03 +01:00
|
|
|
//usleep(5 * 1000); /*DEBUG: Wait after every object draw to see the order of drawing*/
|
2018-11-21 14:00:19 +01:00
|
|
|
|
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*/
|
2017-10-20 10:17:02 +02:00
|
|
|
lv_obj_get_coords(obj, &obj_area);
|
2018-07-17 16:40:52 +02:00
|
|
|
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;
|
|
|
|
LL_READ_BACK(obj->child_ll, child_p) {
|
|
|
|
lv_obj_get_coords(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 */
|
2018-07-17 16:40:52 +02:00
|
|
|
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-04 09:45:39 +02:00
|
|
|
|
2016-10-20 16:35:03 +02:00
|
|
|
/* If all the children are redrawn make 'post draw' design */
|
2018-11-21 14:00:19 +01:00
|
|
|
obj->design_func(obj, &obj_ext_mask, LV_DESIGN_DRAW_POST);
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
}
|