1227 lines
42 KiB
C
1227 lines
42 KiB
C
/**
|
|
* @file lv_refr.c
|
|
*
|
|
*/
|
|
|
|
/*********************
|
|
* INCLUDES
|
|
*********************/
|
|
#include "../misc/lv_area_private.h"
|
|
#include "../draw/sw/lv_draw_sw_mask_private.h"
|
|
#include "../draw/lv_draw_mask_private.h"
|
|
#include "lv_obj_private.h"
|
|
#include "lv_obj_event_private.h"
|
|
#include "lv_obj_draw_private.h"
|
|
#include "lv_refr_private.h"
|
|
#include "../display/lv_display.h"
|
|
#include "../display/lv_display_private.h"
|
|
#include "../tick/lv_tick.h"
|
|
#include "../misc/lv_timer_private.h"
|
|
#include "../misc/lv_math.h"
|
|
#include "../misc/lv_profiler.h"
|
|
#include "../misc/lv_types.h"
|
|
#include "../draw/lv_draw_private.h"
|
|
#include "../font/lv_font_fmt_txt.h"
|
|
#include "../stdlib/lv_string.h"
|
|
#include "lv_global.h"
|
|
|
|
/*********************
|
|
* DEFINES
|
|
*********************/
|
|
|
|
/*Display being refreshed*/
|
|
#define disp_refr LV_GLOBAL_DEFAULT()->disp_refresh
|
|
|
|
/**********************
|
|
* TYPEDEFS
|
|
**********************/
|
|
|
|
/**********************
|
|
* STATIC PROTOTYPES
|
|
**********************/
|
|
static void lv_refr_join_area(void);
|
|
static void refr_invalid_areas(void);
|
|
static void refr_sync_areas(void);
|
|
static void refr_area(const lv_area_t * area_p);
|
|
static void refr_area_part(lv_layer_t * layer);
|
|
static lv_obj_t * lv_refr_get_top_obj(const lv_area_t * area_p, lv_obj_t * obj);
|
|
static void refr_obj_and_children(lv_layer_t * layer, lv_obj_t * top_obj);
|
|
static void refr_obj(lv_layer_t * layer, lv_obj_t * obj);
|
|
static uint32_t get_max_row(lv_display_t * disp, int32_t area_w, int32_t area_h);
|
|
static void draw_buf_flush(lv_display_t * disp);
|
|
static void call_flush_cb(lv_display_t * disp, const lv_area_t * area, uint8_t * px_map);
|
|
static void wait_for_flushing(lv_display_t * disp);
|
|
|
|
/**********************
|
|
* STATIC VARIABLES
|
|
**********************/
|
|
|
|
/**********************
|
|
* MACROS
|
|
**********************/
|
|
#if LV_USE_LOG && LV_LOG_TRACE_DISP_REFR
|
|
#define LV_TRACE_REFR(...) LV_LOG_TRACE(__VA_ARGS__)
|
|
#else
|
|
#define LV_TRACE_REFR(...)
|
|
#endif
|
|
|
|
/**********************
|
|
* GLOBAL FUNCTIONS
|
|
**********************/
|
|
|
|
/**
|
|
* Initialize the screen refresh subsystem
|
|
*/
|
|
void lv_refr_init(void)
|
|
{
|
|
}
|
|
|
|
void lv_refr_deinit(void)
|
|
{
|
|
}
|
|
|
|
void lv_refr_now(lv_display_t * disp)
|
|
{
|
|
lv_anim_refr_now();
|
|
|
|
if(disp) {
|
|
if(disp->refr_timer) lv_display_refr_timer(disp->refr_timer);
|
|
}
|
|
else {
|
|
lv_display_t * d;
|
|
d = lv_display_get_next(NULL);
|
|
while(d) {
|
|
if(d->refr_timer) lv_display_refr_timer(d->refr_timer);
|
|
d = lv_display_get_next(d);
|
|
}
|
|
}
|
|
}
|
|
|
|
void lv_obj_redraw(lv_layer_t * layer, lv_obj_t * obj)
|
|
{
|
|
lv_area_t clip_area_ori = layer->_clip_area;
|
|
lv_area_t clip_coords_for_obj;
|
|
|
|
/*Truncate the clip area to `obj size + ext size` area*/
|
|
lv_area_t obj_coords_ext;
|
|
lv_obj_get_coords(obj, &obj_coords_ext);
|
|
int32_t ext_draw_size = lv_obj_get_ext_draw_size(obj);
|
|
lv_area_increase(&obj_coords_ext, ext_draw_size, ext_draw_size);
|
|
|
|
if(!lv_area_intersect(&clip_coords_for_obj, &clip_area_ori, &obj_coords_ext)) return;
|
|
/*If the object is visible on the current clip area*/
|
|
layer->_clip_area = clip_coords_for_obj;
|
|
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_MAIN_BEGIN, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_MAIN, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_MAIN_END, layer);
|
|
#if LV_USE_REFR_DEBUG
|
|
lv_color_t debug_color = lv_color_make(lv_rand(0, 0xFF), lv_rand(0, 0xFF), lv_rand(0, 0xFF));
|
|
lv_draw_rect_dsc_t draw_dsc;
|
|
lv_draw_rect_dsc_init(&draw_dsc);
|
|
draw_dsc.bg_color = debug_color;
|
|
draw_dsc.bg_opa = LV_OPA_20;
|
|
draw_dsc.border_width = 1;
|
|
draw_dsc.border_opa = LV_OPA_30;
|
|
draw_dsc.border_color = debug_color;
|
|
lv_draw_rect(layer, &draw_dsc, &obj_coords_ext);
|
|
#endif
|
|
|
|
const lv_area_t * obj_coords;
|
|
if(lv_obj_has_flag(obj, LV_OBJ_FLAG_OVERFLOW_VISIBLE)) {
|
|
obj_coords = &obj_coords_ext;
|
|
}
|
|
else {
|
|
obj_coords = &obj->coords;
|
|
}
|
|
lv_area_t clip_coords_for_children;
|
|
bool refr_children = true;
|
|
if(!lv_area_intersect(&clip_coords_for_children, &clip_area_ori, obj_coords)) {
|
|
refr_children = false;
|
|
}
|
|
|
|
if(refr_children) {
|
|
uint32_t i;
|
|
uint32_t child_cnt = lv_obj_get_child_count(obj);
|
|
if(child_cnt == 0) {
|
|
/*If the object was visible on the clip area call the post draw events too*/
|
|
layer->_clip_area = clip_coords_for_obj;
|
|
/*If all the children are redrawn make 'post draw' draw*/
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_BEGIN, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_END, layer);
|
|
}
|
|
else {
|
|
layer->_clip_area = clip_coords_for_children;
|
|
bool clip_corner = lv_obj_get_style_clip_corner(obj, LV_PART_MAIN);
|
|
|
|
int32_t radius = 0;
|
|
if(clip_corner) {
|
|
radius = lv_obj_get_style_radius(obj, LV_PART_MAIN);
|
|
if(radius == 0) clip_corner = false;
|
|
}
|
|
|
|
if(clip_corner == false) {
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = obj->spec_attr->children[i];
|
|
refr_obj(layer, child);
|
|
}
|
|
|
|
/*If the object was visible on the clip area call the post draw events too*/
|
|
layer->_clip_area = clip_coords_for_obj;
|
|
/*If all the children are redrawn make 'post draw' draw*/
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_BEGIN, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_END, layer);
|
|
}
|
|
else {
|
|
lv_layer_t * layer_children;
|
|
lv_draw_mask_rect_dsc_t mask_draw_dsc;
|
|
lv_draw_mask_rect_dsc_init(&mask_draw_dsc);
|
|
mask_draw_dsc.radius = radius;
|
|
mask_draw_dsc.area = obj->coords;
|
|
|
|
lv_draw_image_dsc_t img_draw_dsc;
|
|
lv_draw_image_dsc_init(&img_draw_dsc);
|
|
|
|
int32_t short_side = LV_MIN(lv_area_get_width(&obj->coords), lv_area_get_height(&obj->coords));
|
|
int32_t rout = LV_MIN(radius, short_side >> 1);
|
|
|
|
lv_area_t bottom = obj->coords;
|
|
bottom.y1 = bottom.y2 - rout + 1;
|
|
if(lv_area_intersect(&bottom, &bottom, &clip_area_ori)) {
|
|
layer_children = lv_draw_layer_create(layer, LV_COLOR_FORMAT_ARGB8888, &bottom);
|
|
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = obj->spec_attr->children[i];
|
|
refr_obj(layer_children, child);
|
|
}
|
|
|
|
/*If all the children are redrawn send 'post draw' draw*/
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_BEGIN, layer_children);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST, layer_children);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_END, layer_children);
|
|
|
|
lv_draw_mask_rect(layer_children, &mask_draw_dsc);
|
|
|
|
img_draw_dsc.src = layer_children;
|
|
lv_draw_layer(layer, &img_draw_dsc, &bottom);
|
|
}
|
|
|
|
lv_area_t top = obj->coords;
|
|
top.y2 = top.y1 + rout - 1;
|
|
if(lv_area_intersect(&top, &top, &clip_area_ori)) {
|
|
layer_children = lv_draw_layer_create(layer, LV_COLOR_FORMAT_ARGB8888, &top);
|
|
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = obj->spec_attr->children[i];
|
|
refr_obj(layer_children, child);
|
|
}
|
|
|
|
/*If all the children are redrawn send 'post draw' draw*/
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_BEGIN, layer_children);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST, layer_children);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_END, layer_children);
|
|
|
|
lv_draw_mask_rect(layer_children, &mask_draw_dsc);
|
|
|
|
img_draw_dsc.src = layer_children;
|
|
lv_draw_layer(layer, &img_draw_dsc, &top);
|
|
|
|
}
|
|
|
|
lv_area_t mid = obj->coords;
|
|
mid.y1 += rout;
|
|
mid.y2 -= rout;
|
|
if(lv_area_intersect(&mid, &mid, &clip_area_ori)) {
|
|
layer->_clip_area = mid;
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = obj->spec_attr->children[i];
|
|
refr_obj(layer, child);
|
|
}
|
|
|
|
/*If all the children are redrawn make 'post draw' draw*/
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_BEGIN, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST, layer);
|
|
lv_obj_send_event(obj, LV_EVENT_DRAW_POST_END, layer);
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
layer->_clip_area = clip_area_ori;
|
|
}
|
|
|
|
void lv_inv_area(lv_display_t * disp, const lv_area_t * area_p)
|
|
{
|
|
if(!disp) disp = lv_display_get_default();
|
|
if(!disp) return;
|
|
if(!lv_display_is_invalidation_enabled(disp)) return;
|
|
|
|
/**
|
|
* There are two reasons for this issue:
|
|
* 1.LVGL API is being used across threads, such as modifying widget properties in another thread
|
|
* or within an interrupt handler during the main thread rendering process.
|
|
* 2.User-customized widget modify widget properties/styles again within the DRAW event.
|
|
*
|
|
* Therefore, ensure that LVGL is used in a single-threaded manner, or refer to
|
|
* documentation: https://docs.lvgl.io/master/porting/os.html for proper locking mechanisms.
|
|
* Additionally, ensure that only drawing-related tasks are performed within the DRAW event,
|
|
* and move widget property/style modifications to other events.
|
|
*/
|
|
LV_ASSERT_MSG(!disp->rendering_in_progress, "Invalidate area is not allowed during rendering.");
|
|
|
|
/*Clear the invalidate buffer if the parameter is NULL*/
|
|
if(area_p == NULL) {
|
|
disp->inv_p = 0;
|
|
return;
|
|
}
|
|
|
|
lv_area_t scr_area;
|
|
scr_area.x1 = 0;
|
|
scr_area.y1 = 0;
|
|
scr_area.x2 = lv_display_get_horizontal_resolution(disp) - 1;
|
|
scr_area.y2 = lv_display_get_vertical_resolution(disp) - 1;
|
|
|
|
lv_area_t com_area;
|
|
bool suc;
|
|
|
|
suc = lv_area_intersect(&com_area, area_p, &scr_area);
|
|
if(suc == false) return; /*Out of the screen*/
|
|
|
|
/*If there were at least 1 invalid area in full refresh mode, redraw the whole screen*/
|
|
if(disp->render_mode == LV_DISPLAY_RENDER_MODE_FULL) {
|
|
disp->inv_areas[0] = scr_area;
|
|
disp->inv_p = 1;
|
|
lv_display_send_event(disp, LV_EVENT_REFR_REQUEST, NULL);
|
|
return;
|
|
}
|
|
|
|
lv_result_t res = lv_display_send_event(disp, LV_EVENT_INVALIDATE_AREA, &com_area);
|
|
if(res != LV_RESULT_OK) return;
|
|
|
|
/*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], 0) != false) return;
|
|
}
|
|
|
|
/*Save the area*/
|
|
lv_area_t * tmp_area_p = &com_area;
|
|
if(disp->inv_p >= LV_INV_BUF_SIZE) { /*If no place for the area add the screen*/
|
|
disp->inv_p = 0;
|
|
tmp_area_p = &scr_area;
|
|
}
|
|
lv_area_copy(&disp->inv_areas[disp->inv_p], tmp_area_p);
|
|
disp->inv_p++;
|
|
|
|
lv_display_send_event(disp, LV_EVENT_REFR_REQUEST, NULL);
|
|
}
|
|
|
|
/**
|
|
* Get the display which is being refreshed
|
|
* @return the display being refreshed
|
|
*/
|
|
lv_display_t * lv_refr_get_disp_refreshing(void)
|
|
{
|
|
return disp_refr;
|
|
}
|
|
|
|
/**
|
|
* Get the display which is being refreshed
|
|
* @return the display being refreshed
|
|
*/
|
|
void lv_refr_set_disp_refreshing(lv_display_t * disp)
|
|
{
|
|
disp_refr = disp;
|
|
}
|
|
|
|
void lv_display_refr_timer(lv_timer_t * tmr)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
LV_TRACE_REFR("begin");
|
|
|
|
if(tmr) {
|
|
disp_refr = tmr->user_data;
|
|
/* Ensure the timer does not run again automatically.
|
|
* This is done before refreshing in case refreshing invalidates something else.
|
|
* However if the performance monitor is enabled keep the timer running to count the FPS.*/
|
|
#if LV_USE_PERF_MONITOR
|
|
lv_timer_pause(tmr);
|
|
#endif
|
|
}
|
|
else {
|
|
disp_refr = lv_display_get_default();
|
|
}
|
|
|
|
if(disp_refr == NULL) {
|
|
LV_LOG_WARN("No display registered");
|
|
return;
|
|
}
|
|
|
|
lv_draw_buf_t * buf_act = disp_refr->buf_act;
|
|
if(!(buf_act && buf_act->data && buf_act->data_size)) {
|
|
LV_LOG_WARN("No draw buffer");
|
|
return;
|
|
}
|
|
|
|
lv_display_send_event(disp_refr, LV_EVENT_REFR_START, NULL);
|
|
|
|
/*Refresh the screen's layout if required*/
|
|
LV_PROFILER_BEGIN_TAG("layout");
|
|
lv_obj_update_layout(disp_refr->act_scr);
|
|
if(disp_refr->prev_scr) lv_obj_update_layout(disp_refr->prev_scr);
|
|
|
|
lv_obj_update_layout(disp_refr->bottom_layer);
|
|
lv_obj_update_layout(disp_refr->top_layer);
|
|
lv_obj_update_layout(disp_refr->sys_layer);
|
|
LV_PROFILER_END_TAG("layout");
|
|
|
|
/*Do nothing if there is no active screen*/
|
|
if(disp_refr->act_scr == NULL) {
|
|
disp_refr->inv_p = 0;
|
|
LV_LOG_WARN("there is no active screen");
|
|
goto refr_finish;
|
|
}
|
|
|
|
lv_refr_join_area();
|
|
refr_sync_areas();
|
|
refr_invalid_areas();
|
|
|
|
if(disp_refr->inv_p == 0) goto refr_finish;
|
|
|
|
/*If refresh happened ...*/
|
|
lv_display_send_event(disp_refr, LV_EVENT_RENDER_READY, NULL);
|
|
|
|
/*In double buffered direct mode save the updated areas.
|
|
*They will be used on the next call to synchronize the buffers.*/
|
|
if(lv_display_is_double_buffered(disp_refr) && disp_refr->render_mode == LV_DISPLAY_RENDER_MODE_DIRECT) {
|
|
uint32_t i;
|
|
for(i = 0; i < disp_refr->inv_p; i++) {
|
|
if(disp_refr->inv_area_joined[i])
|
|
continue;
|
|
|
|
lv_area_t * sync_area = lv_ll_ins_tail(&disp_refr->sync_areas);
|
|
*sync_area = disp_refr->inv_areas[i];
|
|
}
|
|
}
|
|
|
|
lv_memzero(disp_refr->inv_areas, sizeof(disp_refr->inv_areas));
|
|
lv_memzero(disp_refr->inv_area_joined, sizeof(disp_refr->inv_area_joined));
|
|
disp_refr->inv_p = 0;
|
|
|
|
refr_finish:
|
|
|
|
#if LV_DRAW_SW_COMPLEX == 1
|
|
lv_draw_sw_mask_cleanup();
|
|
#endif
|
|
|
|
lv_display_send_event(disp_refr, LV_EVENT_REFR_READY, NULL);
|
|
|
|
LV_TRACE_REFR("finished");
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
/**********************
|
|
* STATIC FUNCTIONS
|
|
**********************/
|
|
|
|
/**
|
|
* Join the areas which has got common parts
|
|
*/
|
|
static void lv_refr_join_area(void)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
uint32_t join_from;
|
|
uint32_t join_in;
|
|
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;
|
|
|
|
/*Check all areas to join them in 'join_in'*/
|
|
for(join_from = 0; join_from < disp_refr->inv_p; join_from++) {
|
|
/*Handle only unjoined areas and ignore itself*/
|
|
if(disp_refr->inv_area_joined[join_from] != 0 || join_in == join_from) {
|
|
continue;
|
|
}
|
|
|
|
/*Check if the areas are on each other*/
|
|
if(lv_area_is_on(&disp_refr->inv_areas[join_in], &disp_refr->inv_areas[join_from]) == false) {
|
|
continue;
|
|
}
|
|
|
|
lv_area_join(&joined_area, &disp_refr->inv_areas[join_in], &disp_refr->inv_areas[join_from]);
|
|
|
|
/*Join two area only if the joined area size is smaller*/
|
|
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);
|
|
|
|
/*Mark 'join_form' is joined into 'join_in'*/
|
|
disp_refr->inv_area_joined[join_from] = 1;
|
|
}
|
|
}
|
|
}
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
/**
|
|
* Refresh the sync areas
|
|
*/
|
|
static void refr_sync_areas(void)
|
|
{
|
|
/*Do not sync if not direct or double buffered*/
|
|
if(disp_refr->render_mode != LV_DISPLAY_RENDER_MODE_DIRECT) return;
|
|
|
|
/*Do not sync if not double buffered*/
|
|
if(!lv_display_is_double_buffered(disp_refr)) return;
|
|
|
|
/*Do not sync if no sync areas*/
|
|
if(lv_ll_is_empty(&disp_refr->sync_areas)) return;
|
|
|
|
LV_PROFILER_BEGIN;
|
|
/*With double buffered direct mode synchronize the rendered areas to the other buffer*/
|
|
/*We need to wait for ready here to not mess up the active screen*/
|
|
wait_for_flushing(disp_refr);
|
|
|
|
/*The buffers are already swapped.
|
|
*So the active buffer is the off screen buffer where LVGL will render*/
|
|
lv_draw_buf_t * off_screen = disp_refr->buf_act;
|
|
lv_draw_buf_t * on_screen = disp_refr->buf_act == disp_refr->buf_1 ? disp_refr->buf_2 : disp_refr->buf_1;
|
|
|
|
uint32_t hor_res = lv_display_get_horizontal_resolution(disp_refr);
|
|
uint32_t ver_res = lv_display_get_vertical_resolution(disp_refr);
|
|
|
|
/*Iterate through invalidated areas to see if sync area should be copied*/
|
|
uint16_t i;
|
|
int8_t j;
|
|
lv_area_t res[4] = {0};
|
|
int8_t res_c;
|
|
lv_area_t * sync_area, * new_area, * next_area;
|
|
for(i = 0; i < disp_refr->inv_p; i++) {
|
|
/*Skip joined areas*/
|
|
if(disp_refr->inv_area_joined[i]) continue;
|
|
|
|
/*Iterate over sync areas*/
|
|
sync_area = lv_ll_get_head(&disp_refr->sync_areas);
|
|
while(sync_area != NULL) {
|
|
/*Get next sync area*/
|
|
next_area = lv_ll_get_next(&disp_refr->sync_areas, sync_area);
|
|
|
|
/*Remove intersect of redraw area from sync area and get remaining areas*/
|
|
res_c = lv_area_diff(res, sync_area, &disp_refr->inv_areas[i]);
|
|
|
|
/*New sub areas created after removing intersect*/
|
|
if(res_c != -1) {
|
|
/*Replace old sync area with new areas*/
|
|
for(j = 0; j < res_c; j++) {
|
|
new_area = lv_ll_ins_prev(&disp_refr->sync_areas, sync_area);
|
|
*new_area = res[j];
|
|
}
|
|
lv_ll_remove(&disp_refr->sync_areas, sync_area);
|
|
lv_free(sync_area);
|
|
}
|
|
|
|
/*Move on to next sync area*/
|
|
sync_area = next_area;
|
|
}
|
|
}
|
|
|
|
lv_area_t disp_area = {0, 0, (int32_t)hor_res - 1, (int32_t)ver_res - 1};
|
|
/*Copy sync areas (if any remaining)*/
|
|
for(sync_area = lv_ll_get_head(&disp_refr->sync_areas); sync_area != NULL;
|
|
sync_area = lv_ll_get_next(&disp_refr->sync_areas, sync_area)) {
|
|
/**
|
|
* @todo Resize SDL window will trigger crash because of sync_area is larger than disp_area
|
|
*/
|
|
lv_area_intersect(sync_area, sync_area, &disp_area);
|
|
lv_draw_buf_copy(off_screen, sync_area, on_screen, sync_area);
|
|
}
|
|
|
|
/*Clear sync areas*/
|
|
lv_ll_clear(&disp_refr->sync_areas);
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
/**
|
|
* Refresh the joined areas
|
|
*/
|
|
static void refr_invalid_areas(void)
|
|
{
|
|
if(disp_refr->inv_p == 0) return;
|
|
LV_PROFILER_BEGIN;
|
|
|
|
/*Find the last area which will be drawn*/
|
|
int32_t i;
|
|
int32_t last_i = 0;
|
|
for(i = disp_refr->inv_p - 1; i >= 0; i--) {
|
|
if(disp_refr->inv_area_joined[i] == 0) {
|
|
last_i = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*Notify the display driven rendering has started*/
|
|
lv_display_send_event(disp_refr, LV_EVENT_RENDER_START, NULL);
|
|
|
|
disp_refr->last_area = 0;
|
|
disp_refr->last_part = 0;
|
|
disp_refr->rendering_in_progress = true;
|
|
|
|
for(i = 0; i < (int32_t)disp_refr->inv_p; i++) {
|
|
/*Refresh the unjoined areas*/
|
|
if(disp_refr->inv_area_joined[i] == 0) {
|
|
|
|
if(i == last_i) disp_refr->last_area = 1;
|
|
disp_refr->last_part = 0;
|
|
refr_area(&disp_refr->inv_areas[i]);
|
|
}
|
|
}
|
|
|
|
disp_refr->rendering_in_progress = false;
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
/**
|
|
* Reshape the draw buffer if required
|
|
* @param layer pointer to a layer which will be drawn
|
|
*/
|
|
static void layer_reshape_draw_buf(lv_layer_t * layer)
|
|
{
|
|
lv_draw_buf_t * ret = lv_draw_buf_reshape(
|
|
layer->draw_buf,
|
|
layer->color_format,
|
|
lv_area_get_width(&layer->buf_area),
|
|
lv_area_get_height(&layer->buf_area),
|
|
0);
|
|
LV_UNUSED(ret);
|
|
LV_ASSERT_NULL(ret);
|
|
}
|
|
|
|
/**
|
|
* Refresh an area if there is Virtual Display Buffer
|
|
* @param area_p pointer to an area to refresh
|
|
*/
|
|
static void refr_area(const lv_area_t * area_p)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
lv_layer_t * layer = disp_refr->layer_head;
|
|
layer->draw_buf = disp_refr->buf_act;
|
|
|
|
#if LV_DRAW_TRANSFORM_USE_MATRIX
|
|
lv_matrix_identity(&layer->matrix);
|
|
#endif
|
|
|
|
/*With full refresh just redraw directly into the buffer*/
|
|
/*In direct mode draw directly on the absolute coordinates of the buffer*/
|
|
if(disp_refr->render_mode != LV_DISPLAY_RENDER_MODE_PARTIAL) {
|
|
layer->buf_area.x1 = 0;
|
|
layer->buf_area.y1 = 0;
|
|
layer->buf_area.x2 = lv_display_get_horizontal_resolution(disp_refr) - 1;
|
|
layer->buf_area.y2 = lv_display_get_vertical_resolution(disp_refr) - 1;
|
|
layer_reshape_draw_buf(layer);
|
|
lv_area_t disp_area;
|
|
lv_area_set(&disp_area, 0, 0, lv_display_get_horizontal_resolution(disp_refr) - 1,
|
|
lv_display_get_vertical_resolution(disp_refr) - 1);
|
|
|
|
if(disp_refr->render_mode == LV_DISPLAY_RENDER_MODE_FULL) {
|
|
disp_refr->last_part = 1;
|
|
layer->_clip_area = disp_area;
|
|
refr_area_part(layer);
|
|
}
|
|
else if(disp_refr->render_mode == LV_DISPLAY_RENDER_MODE_DIRECT) {
|
|
disp_refr->last_part = disp_refr->last_area;
|
|
layer->_clip_area = *area_p;
|
|
refr_area_part(layer);
|
|
}
|
|
LV_PROFILER_END;
|
|
return;
|
|
}
|
|
|
|
/*Normal refresh: draw the area in parts*/
|
|
/*Calculate the max row num*/
|
|
int32_t w = lv_area_get_width(area_p);
|
|
int32_t h = lv_area_get_height(area_p);
|
|
int32_t y2 = area_p->y2 >= lv_display_get_vertical_resolution(disp_refr) ?
|
|
lv_display_get_vertical_resolution(disp_refr) - 1 : area_p->y2;
|
|
|
|
int32_t max_row = get_max_row(disp_refr, w, h);
|
|
|
|
int32_t row;
|
|
int32_t row_last = 0;
|
|
lv_area_t sub_area;
|
|
for(row = area_p->y1; row + max_row - 1 <= y2; row += max_row) {
|
|
/*Calc. the next y coordinates of draw_buf*/
|
|
sub_area.x1 = area_p->x1;
|
|
sub_area.x2 = area_p->x2;
|
|
sub_area.y1 = row;
|
|
sub_area.y2 = row + max_row - 1;
|
|
layer->draw_buf = disp_refr->buf_act;
|
|
layer->buf_area = sub_area;
|
|
layer->_clip_area = sub_area;
|
|
layer_reshape_draw_buf(layer);
|
|
if(sub_area.y2 > y2) sub_area.y2 = y2;
|
|
row_last = sub_area.y2;
|
|
if(y2 == row_last) disp_refr->last_part = 1;
|
|
refr_area_part(layer);
|
|
}
|
|
|
|
/*If the last y coordinates are not handled yet ...*/
|
|
if(y2 != row_last) {
|
|
/*Calc. the next y coordinates of draw_buf*/
|
|
sub_area.x1 = area_p->x1;
|
|
sub_area.x2 = area_p->x2;
|
|
sub_area.y1 = row;
|
|
sub_area.y2 = y2;
|
|
layer->draw_buf = disp_refr->buf_act;
|
|
layer->buf_area = sub_area;
|
|
layer->_clip_area = sub_area;
|
|
layer_reshape_draw_buf(layer);
|
|
disp_refr->last_part = 1;
|
|
refr_area_part(layer);
|
|
}
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
static void refr_area_part(lv_layer_t * layer)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
disp_refr->refreshed_area = layer->_clip_area;
|
|
|
|
/* In single buffered mode wait here until the buffer is freed.
|
|
* Else we would draw into the buffer while it's still being transferred to the display*/
|
|
if(!lv_display_is_double_buffered(disp_refr)) {
|
|
wait_for_flushing(disp_refr);
|
|
}
|
|
/*If the screen is transparent initialize it when the flushing is ready*/
|
|
if(lv_color_format_has_alpha(disp_refr->color_format)) {
|
|
lv_area_t a = disp_refr->refreshed_area;
|
|
if(disp_refr->render_mode == LV_DISPLAY_RENDER_MODE_PARTIAL) {
|
|
/*The area always starts at 0;0*/
|
|
lv_area_move(&a, -disp_refr->refreshed_area.x1, -disp_refr->refreshed_area.y1);
|
|
}
|
|
|
|
lv_draw_buf_clear(layer->draw_buf, &a);
|
|
}
|
|
|
|
lv_obj_t * top_act_scr = NULL;
|
|
lv_obj_t * top_prev_scr = NULL;
|
|
|
|
/*Get the most top object which is not covered by others*/
|
|
top_act_scr = lv_refr_get_top_obj(&layer->_clip_area, lv_display_get_screen_active(disp_refr));
|
|
if(disp_refr->prev_scr) {
|
|
top_prev_scr = lv_refr_get_top_obj(&layer->_clip_area, disp_refr->prev_scr);
|
|
}
|
|
|
|
/*Draw a bottom layer background if there is no top object*/
|
|
if(top_act_scr == NULL && top_prev_scr == NULL) {
|
|
refr_obj_and_children(layer, lv_display_get_layer_bottom(disp_refr));
|
|
}
|
|
|
|
if(disp_refr->draw_prev_over_act) {
|
|
if(top_act_scr == NULL) top_act_scr = disp_refr->act_scr;
|
|
refr_obj_and_children(layer, top_act_scr);
|
|
|
|
/*Refresh the previous screen if any*/
|
|
if(disp_refr->prev_scr) {
|
|
if(top_prev_scr == NULL) top_prev_scr = disp_refr->prev_scr;
|
|
refr_obj_and_children(layer, top_prev_scr);
|
|
}
|
|
}
|
|
else {
|
|
/*Refresh the previous screen if any*/
|
|
if(disp_refr->prev_scr) {
|
|
if(top_prev_scr == NULL) top_prev_scr = disp_refr->prev_scr;
|
|
refr_obj_and_children(layer, top_prev_scr);
|
|
}
|
|
|
|
if(top_act_scr == NULL) top_act_scr = disp_refr->act_scr;
|
|
refr_obj_and_children(layer, top_act_scr);
|
|
}
|
|
|
|
/*Also refresh top and sys layer unconditionally*/
|
|
refr_obj_and_children(layer, lv_display_get_layer_top(disp_refr));
|
|
refr_obj_and_children(layer, lv_display_get_layer_sys(disp_refr));
|
|
|
|
draw_buf_flush(disp_refr);
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
/**
|
|
* Search the most top object which fully covers an area
|
|
* @param area_p pointer to an area
|
|
* @param obj the first object to start the searching (typically a screen)
|
|
* @return
|
|
*/
|
|
static lv_obj_t * lv_refr_get_top_obj(const lv_area_t * area_p, lv_obj_t * obj)
|
|
{
|
|
lv_obj_t * found_p = NULL;
|
|
|
|
if(lv_area_is_in(area_p, &obj->coords, 0) == false) return NULL;
|
|
if(lv_obj_has_flag(obj, LV_OBJ_FLAG_HIDDEN)) return NULL;
|
|
if(lv_obj_get_layer_type(obj) != LV_LAYER_TYPE_NONE) return NULL;
|
|
|
|
/*If this object is fully cover the draw area then check the children too*/
|
|
lv_cover_check_info_t info;
|
|
info.res = LV_COVER_RES_COVER;
|
|
info.area = area_p;
|
|
lv_obj_send_event(obj, LV_EVENT_COVER_CHECK, &info);
|
|
if(info.res == LV_COVER_RES_MASKED) return NULL;
|
|
|
|
int32_t i;
|
|
int32_t child_cnt = lv_obj_get_child_count(obj);
|
|
for(i = child_cnt - 1; i >= 0; i--) {
|
|
lv_obj_t * child = obj->spec_attr->children[i];
|
|
found_p = lv_refr_get_top_obj(area_p, child);
|
|
|
|
/*If a children is ok then break*/
|
|
if(found_p != NULL) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*If no better children use this object*/
|
|
if(found_p == NULL && info.res == LV_COVER_RES_COVER) {
|
|
found_p = obj;
|
|
}
|
|
|
|
return found_p;
|
|
}
|
|
|
|
/**
|
|
* Make the refreshing from an object. Draw all its children and the youngers too.
|
|
* @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 refr_obj_and_children(lv_layer_t * layer, lv_obj_t * top_obj)
|
|
{
|
|
/*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_obj == NULL) top_obj = lv_display_get_screen_active(disp_refr);
|
|
if(top_obj == NULL) return; /*Shouldn't happen*/
|
|
|
|
LV_PROFILER_BEGIN;
|
|
/*Refresh the top object and its children*/
|
|
refr_obj(layer, top_obj);
|
|
|
|
/*Draw the 'younger' sibling objects because they can be on top_obj*/
|
|
lv_obj_t * parent;
|
|
lv_obj_t * border_p = top_obj;
|
|
|
|
parent = lv_obj_get_parent(top_obj);
|
|
|
|
/*Do until not reach the screen*/
|
|
while(parent != NULL) {
|
|
bool go = false;
|
|
uint32_t i;
|
|
uint32_t child_cnt = lv_obj_get_child_count(parent);
|
|
for(i = 0; i < child_cnt; i++) {
|
|
lv_obj_t * child = parent->spec_attr->children[i];
|
|
if(!go) {
|
|
if(child == border_p) go = true;
|
|
}
|
|
else {
|
|
/*Refresh the objects*/
|
|
refr_obj(layer, child);
|
|
}
|
|
}
|
|
|
|
/*Call the post draw function of the parents of the to object*/
|
|
lv_obj_send_event(parent, LV_EVENT_DRAW_POST_BEGIN, (void *)layer);
|
|
lv_obj_send_event(parent, LV_EVENT_DRAW_POST, (void *)layer);
|
|
lv_obj_send_event(parent, LV_EVENT_DRAW_POST_END, (void *)layer);
|
|
|
|
/*The new border will be the last parents,
|
|
*so the 'younger' brothers of parent will be refreshed*/
|
|
border_p = parent;
|
|
/*Go a level deeper*/
|
|
parent = lv_obj_get_parent(parent);
|
|
}
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
static lv_result_t layer_get_area(lv_layer_t * layer, lv_obj_t * obj, lv_layer_type_t layer_type,
|
|
lv_area_t * layer_area_out, lv_area_t * obj_draw_size_out)
|
|
{
|
|
int32_t ext_draw_size = lv_obj_get_ext_draw_size(obj);
|
|
lv_obj_get_coords(obj, obj_draw_size_out);
|
|
lv_area_increase(obj_draw_size_out, ext_draw_size, ext_draw_size);
|
|
|
|
if(layer_type == LV_LAYER_TYPE_TRANSFORM) {
|
|
/*Get the transformed area and clip it to the current clip area.
|
|
*This area needs to be updated on the screen.*/
|
|
lv_area_t clip_coords_for_obj;
|
|
lv_area_t tranf_coords = *obj_draw_size_out;
|
|
lv_obj_get_transformed_area(obj, &tranf_coords, LV_OBJ_POINT_TRANSFORM_FLAG_NONE);
|
|
if(!lv_area_intersect(&clip_coords_for_obj, &layer->_clip_area, &tranf_coords)) {
|
|
return LV_RESULT_INVALID;
|
|
}
|
|
|
|
/*Transform back (inverse) the transformed area.
|
|
*It will tell which area of the non-transformed widget needs to be redrawn
|
|
*in order to cover transformed area after transformation.*/
|
|
lv_area_t inverse_clip_coords_for_obj = clip_coords_for_obj;
|
|
lv_obj_get_transformed_area(obj, &inverse_clip_coords_for_obj, LV_OBJ_POINT_TRANSFORM_FLAG_INVERSE);
|
|
if(!lv_area_intersect(&inverse_clip_coords_for_obj, &inverse_clip_coords_for_obj, obj_draw_size_out)) {
|
|
return LV_RESULT_INVALID;
|
|
}
|
|
|
|
*layer_area_out = inverse_clip_coords_for_obj;
|
|
lv_area_increase(layer_area_out, 5, 5); /*To avoid rounding error*/
|
|
}
|
|
else if(layer_type == LV_LAYER_TYPE_SIMPLE) {
|
|
lv_area_t clip_coords_for_obj;
|
|
if(!lv_area_intersect(&clip_coords_for_obj, &layer->_clip_area, obj_draw_size_out)) {
|
|
return LV_RESULT_INVALID;
|
|
}
|
|
*layer_area_out = clip_coords_for_obj;
|
|
}
|
|
else {
|
|
LV_LOG_WARN("Unhandled layer type");
|
|
return LV_RESULT_INVALID;
|
|
}
|
|
|
|
return LV_RESULT_OK;
|
|
}
|
|
|
|
static bool alpha_test_area_on_obj(lv_obj_t * obj, const lv_area_t * area)
|
|
{
|
|
/*Test for alpha by assuming there is no alpha. If it fails, fall back to rendering with alpha*/
|
|
/*If the layer area is not fully on the object, it can't fully cover it*/
|
|
if(!lv_area_is_on(area, &obj->coords)) return true;
|
|
|
|
lv_cover_check_info_t info;
|
|
info.res = LV_COVER_RES_COVER;
|
|
info.area = area;
|
|
lv_obj_send_event(obj, LV_EVENT_COVER_CHECK, &info);
|
|
if(info.res == LV_COVER_RES_COVER) return false;
|
|
else return true;
|
|
}
|
|
|
|
#if LV_DRAW_TRANSFORM_USE_MATRIX
|
|
|
|
static void refr_obj_matrix(lv_layer_t * layer, lv_obj_t * obj)
|
|
{
|
|
lv_matrix_t ori_matrix = layer->matrix;
|
|
lv_matrix_t obj_matrix;
|
|
lv_matrix_identity(&obj_matrix);
|
|
|
|
lv_point_t pivot = {
|
|
.x = lv_obj_get_style_transform_pivot_x(obj, 0),
|
|
.y = lv_obj_get_style_transform_pivot_y(obj, 0)
|
|
};
|
|
|
|
pivot.x = obj->coords.x1 + lv_pct_to_px(pivot.x, lv_area_get_width(&obj->coords));
|
|
pivot.y = obj->coords.y1 + lv_pct_to_px(pivot.y, lv_area_get_height(&obj->coords));
|
|
|
|
int32_t rotation = lv_obj_get_style_transform_rotation(obj, 0);
|
|
int32_t scale_x = lv_obj_get_style_transform_scale_x(obj, 0);
|
|
int32_t scale_y = lv_obj_get_style_transform_scale_y(obj, 0);
|
|
int32_t skew_x = lv_obj_get_style_transform_skew_x(obj, 0);
|
|
int32_t skew_y = lv_obj_get_style_transform_skew_y(obj, 0);
|
|
|
|
if(scale_x <= 0 || scale_y <= 0) {
|
|
/* NOT draw if scale is negative or zero */
|
|
return;
|
|
}
|
|
|
|
/* generate the obj matrix */
|
|
lv_matrix_translate(&obj_matrix, pivot.x, pivot.y);
|
|
if(rotation != 0) {
|
|
lv_matrix_rotate(&obj_matrix, rotation * 0.1f);
|
|
}
|
|
|
|
if(scale_x != LV_SCALE_NONE || scale_y != LV_SCALE_NONE) {
|
|
lv_matrix_scale(
|
|
&obj_matrix,
|
|
(float)scale_x / LV_SCALE_NONE,
|
|
(float)scale_y / LV_SCALE_NONE
|
|
);
|
|
}
|
|
|
|
if(skew_x != 0 || skew_y != 0) {
|
|
lv_matrix_skew(&obj_matrix, skew_x, skew_y);
|
|
}
|
|
|
|
lv_matrix_translate(&obj_matrix, -pivot.x, -pivot.y);
|
|
|
|
/* apply the obj matrix */
|
|
lv_matrix_multiply(&layer->matrix, &obj_matrix);
|
|
|
|
/* calculate clip area without transform */
|
|
lv_matrix_t matrix_reverse;
|
|
lv_matrix_inverse(&matrix_reverse, &obj_matrix);
|
|
|
|
lv_area_t clip_area = layer->_clip_area;
|
|
lv_area_t clip_area_ori = layer->_clip_area;
|
|
clip_area = lv_matrix_transform_area(&matrix_reverse, &clip_area);
|
|
|
|
/* increase the clip area by 1 pixel to avoid rounding errors */
|
|
if(!lv_matrix_is_identity_or_translation(&obj_matrix)) {
|
|
lv_area_increase(&clip_area, 1, 1);
|
|
}
|
|
|
|
layer->_clip_area = clip_area;
|
|
|
|
/* redraw obj */
|
|
lv_obj_redraw(layer, obj);
|
|
|
|
/* restore original matrix */
|
|
layer->matrix = ori_matrix;
|
|
/* restore clip area */
|
|
layer->_clip_area = clip_area_ori;
|
|
}
|
|
|
|
static bool refr_check_obj_clip_overflow(lv_layer_t * layer, lv_obj_t * obj)
|
|
{
|
|
if(lv_obj_get_style_transform_rotation(obj, 0) == 0) {
|
|
return false;
|
|
}
|
|
|
|
/*Truncate the area to the object*/
|
|
lv_area_t obj_coords;
|
|
int32_t ext_size = lv_obj_get_ext_draw_size(obj);
|
|
lv_area_copy(&obj_coords, &obj->coords);
|
|
lv_area_increase(&obj_coords, ext_size, ext_size);
|
|
|
|
lv_obj_get_transformed_area(obj, &obj_coords, LV_OBJ_POINT_TRANSFORM_FLAG_RECURSIVE);
|
|
|
|
lv_area_t clip_coords_for_obj;
|
|
if(!lv_area_intersect(&clip_coords_for_obj, &layer->_clip_area, &obj_coords)) {
|
|
return false;
|
|
}
|
|
|
|
bool has_clip = lv_memcmp(&clip_coords_for_obj, &obj_coords, sizeof(lv_area_t)) != 0;
|
|
return has_clip;
|
|
}
|
|
|
|
#endif /* LV_DRAW_TRANSFORM_USE_MATRIX */
|
|
|
|
static void refr_obj(lv_layer_t * layer, lv_obj_t * obj)
|
|
{
|
|
if(lv_obj_has_flag(obj, LV_OBJ_FLAG_HIDDEN)) return;
|
|
|
|
lv_opa_t opa = lv_obj_get_style_opa_layered(obj, 0);
|
|
if(opa < LV_OPA_MIN) return;
|
|
|
|
#if LV_DRAW_TRANSFORM_USE_MATRIX
|
|
/*If the layer opa is full then use the matrix transform*/
|
|
if(opa >= LV_OPA_MAX && !refr_check_obj_clip_overflow(layer, obj)) {
|
|
refr_obj_matrix(layer, obj);
|
|
return;
|
|
}
|
|
#endif /* LV_DRAW_TRANSFORM_USE_MATRIX */
|
|
|
|
lv_layer_type_t layer_type = lv_obj_get_layer_type(obj);
|
|
if(layer_type == LV_LAYER_TYPE_NONE) {
|
|
lv_obj_redraw(layer, obj);
|
|
}
|
|
else {
|
|
lv_area_t layer_area_full;
|
|
lv_area_t obj_draw_size;
|
|
lv_result_t res = layer_get_area(layer, obj, layer_type, &layer_area_full, &obj_draw_size);
|
|
if(res != LV_RESULT_OK) return;
|
|
|
|
/*Simple layers can be subdivided into smaller layers*/
|
|
uint32_t max_rgb_row_height = lv_area_get_height(&layer_area_full);
|
|
uint32_t max_argb_row_height = lv_area_get_height(&layer_area_full);
|
|
if(layer_type == LV_LAYER_TYPE_SIMPLE) {
|
|
int32_t w = lv_area_get_width(&layer_area_full);
|
|
uint8_t px_size = lv_color_format_get_size(disp_refr->color_format);
|
|
max_rgb_row_height = LV_DRAW_LAYER_SIMPLE_BUF_SIZE / w / px_size;
|
|
max_argb_row_height = LV_DRAW_LAYER_SIMPLE_BUF_SIZE / w / sizeof(lv_color32_t);
|
|
}
|
|
|
|
lv_area_t layer_area_act;
|
|
layer_area_act.x1 = layer_area_full.x1;
|
|
layer_area_act.x2 = layer_area_full.x2;
|
|
layer_area_act.y1 = layer_area_full.y1;
|
|
layer_area_act.y2 = layer_area_full.y1;
|
|
|
|
while(layer_area_act.y2 < layer_area_full.y2) {
|
|
/* Test with an RGB layer size (which is larger than the ARGB layer size)
|
|
* If it really doesn't need alpha use it. Else switch to the ARGB size*/
|
|
layer_area_act.y2 = layer_area_act.y1 + max_rgb_row_height - 1;
|
|
if(layer_area_act.y2 > layer_area_full.y2) layer_area_act.y2 = layer_area_full.y2;
|
|
bool area_need_alpha = alpha_test_area_on_obj(obj, &layer_area_act);
|
|
if(area_need_alpha) {
|
|
layer_area_act.y2 = layer_area_act.y1 + max_argb_row_height - 1;
|
|
if(layer_area_act.y2 > layer_area_full.y2) layer_area_act.y2 = layer_area_full.y2;
|
|
}
|
|
|
|
lv_layer_t * new_layer = lv_draw_layer_create(layer,
|
|
area_need_alpha ? LV_COLOR_FORMAT_ARGB8888 : LV_COLOR_FORMAT_NATIVE, &layer_area_act);
|
|
lv_obj_redraw(new_layer, obj);
|
|
|
|
lv_point_t pivot = {
|
|
.x = lv_obj_get_style_transform_pivot_x(obj, 0),
|
|
.y = lv_obj_get_style_transform_pivot_y(obj, 0)
|
|
};
|
|
|
|
if(LV_COORD_IS_PCT(pivot.x)) {
|
|
pivot.x = (LV_COORD_GET_PCT(pivot.x) * lv_area_get_width(&obj->coords)) / 100;
|
|
}
|
|
if(LV_COORD_IS_PCT(pivot.y)) {
|
|
pivot.y = (LV_COORD_GET_PCT(pivot.y) * lv_area_get_height(&obj->coords)) / 100;
|
|
}
|
|
|
|
lv_draw_image_dsc_t layer_draw_dsc;
|
|
lv_draw_image_dsc_init(&layer_draw_dsc);
|
|
layer_draw_dsc.pivot.x = obj->coords.x1 + pivot.x - new_layer->buf_area.x1;
|
|
layer_draw_dsc.pivot.y = obj->coords.y1 + pivot.y - new_layer->buf_area.y1;
|
|
|
|
layer_draw_dsc.opa = opa;
|
|
layer_draw_dsc.rotation = lv_obj_get_style_transform_rotation(obj, 0);
|
|
while(layer_draw_dsc.rotation > 3600) layer_draw_dsc.rotation -= 3600;
|
|
while(layer_draw_dsc.rotation < 0) layer_draw_dsc.rotation += 3600;
|
|
layer_draw_dsc.scale_x = lv_obj_get_style_transform_scale_x(obj, 0);
|
|
layer_draw_dsc.scale_y = lv_obj_get_style_transform_scale_y(obj, 0);
|
|
layer_draw_dsc.skew_x = lv_obj_get_style_transform_skew_x(obj, 0);
|
|
layer_draw_dsc.skew_y = lv_obj_get_style_transform_skew_y(obj, 0);
|
|
layer_draw_dsc.blend_mode = lv_obj_get_style_blend_mode(obj, 0);
|
|
layer_draw_dsc.antialias = disp_refr->antialiasing;
|
|
layer_draw_dsc.bitmap_mask_src = lv_obj_get_style_bitmap_mask_src(obj, 0);
|
|
layer_draw_dsc.image_area = obj_draw_size;
|
|
layer_draw_dsc.src = new_layer;
|
|
|
|
lv_draw_layer(layer, &layer_draw_dsc, &layer_area_act);
|
|
|
|
layer_area_act.y1 = layer_area_act.y2 + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
static uint32_t get_max_row(lv_display_t * disp, int32_t area_w, int32_t area_h)
|
|
{
|
|
bool has_alpha = lv_color_format_has_alpha(disp->color_format);
|
|
lv_color_format_t cf = has_alpha ? LV_COLOR_FORMAT_ARGB8888 : disp->color_format;
|
|
uint32_t stride = lv_draw_buf_width_to_stride(area_w, cf);
|
|
int32_t max_row = (uint32_t)disp->buf_act->data_size / stride;
|
|
|
|
if(max_row > area_h) max_row = area_h;
|
|
|
|
/*Round down the lines of draw_buf if rounding is added*/
|
|
lv_area_t tmp;
|
|
tmp.x1 = 0;
|
|
tmp.x2 = 0;
|
|
tmp.y1 = 0;
|
|
|
|
int32_t h_tmp = max_row;
|
|
do {
|
|
tmp.y2 = h_tmp - 1;
|
|
lv_display_send_event(disp_refr, LV_EVENT_INVALIDATE_AREA, &tmp);
|
|
|
|
/*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);
|
|
|
|
if(h_tmp <= 0) {
|
|
LV_LOG_WARN("Can't set draw_buf height using the round function. (Wrong round_cb or too "
|
|
"small draw_buf)");
|
|
return 0;
|
|
}
|
|
else {
|
|
max_row = tmp.y2 + 1;
|
|
}
|
|
|
|
return max_row;
|
|
}
|
|
|
|
/**
|
|
* Flush the content of the draw buffer
|
|
*/
|
|
static void draw_buf_flush(lv_display_t * disp)
|
|
{
|
|
/*Flush the rendered content to the display*/
|
|
lv_layer_t * layer = disp->layer_head;
|
|
|
|
while(layer->draw_task_head) {
|
|
lv_draw_dispatch_wait_for_request();
|
|
lv_draw_dispatch();
|
|
}
|
|
|
|
/* In double buffered mode wait until the other buffer is freed
|
|
* and driver is ready to receive the new buffer.
|
|
* If we need to wait here it means that the content of one buffer is being sent to display
|
|
* and other buffer already contains the new rendered image. */
|
|
if(lv_display_is_double_buffered(disp)) {
|
|
wait_for_flushing(disp_refr);
|
|
}
|
|
|
|
disp->flushing = 1;
|
|
|
|
if(disp->last_area && disp->last_part) disp->flushing_last = 1;
|
|
else disp->flushing_last = 0;
|
|
|
|
bool flushing_last = disp->flushing_last;
|
|
|
|
if(disp->flush_cb) {
|
|
call_flush_cb(disp, &disp->refreshed_area, layer->draw_buf->data);
|
|
}
|
|
/*If there are 2 buffers swap them. With direct mode swap only on the last area*/
|
|
if(lv_display_is_double_buffered(disp) && (disp->render_mode != LV_DISPLAY_RENDER_MODE_DIRECT || flushing_last)) {
|
|
if(disp->buf_act == disp->buf_1) {
|
|
disp->buf_act = disp->buf_2;
|
|
}
|
|
else {
|
|
disp->buf_act = disp->buf_1;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void call_flush_cb(lv_display_t * disp, const lv_area_t * area, uint8_t * px_map)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
LV_TRACE_REFR("Calling flush_cb on (%d;%d)(%d;%d) area with %p image pointer",
|
|
(int)area->x1, (int)area->y1, (int)area->x2, (int)area->y2, (void *)px_map);
|
|
|
|
lv_area_t offset_area = {
|
|
.x1 = area->x1 + disp->offset_x,
|
|
.y1 = area->y1 + disp->offset_y,
|
|
.x2 = area->x2 + disp->offset_x,
|
|
.y2 = area->y2 + disp->offset_y
|
|
};
|
|
|
|
lv_display_send_event(disp, LV_EVENT_FLUSH_START, &offset_area);
|
|
|
|
/*For backward compatibility support LV_COLOR_16_SWAP (from v8)*/
|
|
#if defined(LV_COLOR_16_SWAP) && LV_COLOR_16_SWAP
|
|
lv_draw_sw_rgb565_swap(px_map, lv_area_get_size(&offset_area));
|
|
#endif
|
|
|
|
disp->flush_cb(disp, &offset_area, px_map);
|
|
lv_display_send_event(disp, LV_EVENT_FLUSH_FINISH, &offset_area);
|
|
|
|
LV_PROFILER_END;
|
|
}
|
|
|
|
static void wait_for_flushing(lv_display_t * disp)
|
|
{
|
|
LV_PROFILER_BEGIN;
|
|
LV_LOG_TRACE("begin");
|
|
|
|
lv_display_send_event(disp, LV_EVENT_FLUSH_WAIT_START, NULL);
|
|
|
|
if(disp->flush_wait_cb) {
|
|
if(disp->flushing) {
|
|
disp->flush_wait_cb(disp);
|
|
}
|
|
disp->flushing = 0;
|
|
}
|
|
else {
|
|
while(disp->flushing);
|
|
}
|
|
disp->flushing_last = 0;
|
|
|
|
lv_display_send_event(disp, LV_EVENT_FLUSH_WAIT_FINISH, NULL);
|
|
|
|
LV_LOG_TRACE("end");
|
|
LV_PROFILER_END;
|
|
}
|