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

697 lines
25 KiB
C
Raw Normal View History

2016-06-08 07:25:08 +02:00
/**
2019-02-13 01:40:22 +01:00
* @file lv_draw_basic.c
2018-06-19 09:49:58 +02:00
*
2016-06-08 07:25:08 +02:00
*/
2017-10-09 15:21:26 +02:00
2019-02-13 01:40:22 +01:00
#include "lv_draw_basic.h"
2017-10-09 15:21:26 +02:00
2017-03-20 07:07:05 +01:00
#include <stdbool.h>
#include <stdint.h>
#include <string.h>
2019-02-12 12:21:34 +01:00
#include "../lv_core/lv_refr.h"
2019-02-10 11:06:47 +01:00
#include "../lv_hal/lv_hal.h"
2017-11-23 20:42:14 +01:00
#include "../lv_misc/lv_area.h"
#include "../lv_misc/lv_font.h"
#include "../lv_misc/lv_color.h"
2018-07-25 17:57:08 +02:00
#include "../lv_misc/lv_log.h"
2017-06-26 16:20:05 +02:00
2016-06-08 07:25:08 +02:00
#include <stddef.h>
#include "lv_draw.h"
2016-06-08 07:25:08 +02:00
/*********************
* INCLUDES
*********************/
/*********************
* DEFINES
*********************/
2019-04-04 16:44:16 +02:00
/*Always fill < 50 px with 'sw_color_fill' because of the hw. init overhead*/
#define VFILL_HW_ACC_SIZE_LIMIT 50
2016-06-08 07:25:08 +02:00
#ifndef LV_ATTRIBUTE_MEM_ALIGN
#define LV_ATTRIBUTE_MEM_ALIGN
#endif
2016-06-08 07:25:08 +02:00
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
2017-11-28 16:15:13 +01:00
static void sw_mem_blend(lv_color_t * dest, const lv_color_t * src, uint32_t length, lv_opa_t opa);
2019-04-04 07:15:40 +02:00
static void sw_color_fill(lv_area_t * mem_area, lv_color_t * mem, const lv_area_t * fill_area,
lv_color_t color, lv_opa_t opa);
2016-06-08 07:25:08 +02:00
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP
2019-04-04 07:15:40 +02:00
static inline lv_color_t color_mix_2_alpha(lv_color_t bg_color, lv_opa_t bg_opa,
lv_color_t fg_color, lv_opa_t fg_opa);
2018-09-20 22:16:03 +02:00
#endif
2016-06-08 07:25:08 +02:00
/**********************
* STATIC VARIABLES
**********************/
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
2017-04-24 12:08:24 +02:00
/**
* Put a pixel in the Virtual Display Buffer
* @param x pixel x coordinate
* @param y pixel y coordinate
* @param mask_p fill only on this mask (truncated to VDB area)
* @param color pixel color
* @param opa opacity of the area (0..255)
*/
2019-04-04 07:15:40 +02:00
void lv_draw_px(lv_coord_t x, lv_coord_t y, const lv_area_t * mask_p, lv_color_t color,
lv_opa_t opa)
2017-04-24 12:08:24 +02:00
{
if(opa < LV_OPA_MIN) return;
if(opa > LV_OPA_MAX) opa = LV_OPA_COVER;
2017-04-24 12:08:24 +02:00
/*Pixel out of the mask*/
2019-04-04 07:15:40 +02:00
if(x < mask_p->x1 || x > mask_p->x2 || y < mask_p->y1 || y > mask_p->y2) {
2017-04-24 12:08:24 +02:00
return;
}
2019-04-04 07:15:40 +02:00
lv_disp_t * disp = lv_refr_get_disp_refreshing();
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp);
2019-04-04 07:15:40 +02:00
uint32_t vdb_width = lv_area_get_width(&vdb->area);
2017-04-24 12:08:24 +02:00
/*Make the coordinates relative to VDB*/
2019-02-13 01:40:22 +01:00
x -= vdb->area.x1;
y -= vdb->area.y1;
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
disp->driver.set_px_cb(&disp->driver, (uint8_t *)vdb->buf_act, vdb_width, x, y, color, opa);
2018-06-19 09:49:58 +02:00
} else {
2019-02-13 01:40:22 +01:00
lv_color_t * vdb_px_p = vdb->buf_act;
vdb_px_p += y * vdb_width + x;
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP == 0
if(opa == LV_OPA_COVER) {
*vdb_px_p = color;
} else {
*vdb_px_p = lv_color_mix(color, *vdb_px_p, opa);
}
2018-09-20 22:16:03 +02:00
#else
*vdb_px_p = color_mix_2_alpha(*vdb_px_p, (*vdb_px_p).alpha, color, opa);
#endif
2017-04-24 12:08:24 +02:00
}
}
2016-06-08 07:25:08 +02:00
/**
* Fill an area in the Virtual Display Buffer
* @param cords_p coordinates of the area to fill
2017-04-24 12:08:24 +02:00
* @param mask_p fill only o this mask (truncated to VDB area)
2016-06-08 07:25:08 +02:00
* @param color fill color
* @param opa opacity of the area (0..255)
*/
2019-04-04 07:15:40 +02:00
void lv_draw_fill(const lv_area_t * cords_p, const lv_area_t * mask_p, lv_color_t color,
lv_opa_t opa)
2016-06-08 07:25:08 +02:00
{
if(opa < LV_OPA_MIN) return;
if(opa > LV_OPA_MAX) opa = LV_OPA_COVER;
2017-11-23 21:28:36 +01:00
lv_area_t res_a;
2016-06-08 07:25:08 +02:00
bool union_ok;
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*Get the union of cord and mask*/
/* The mask is already truncated to the vdb size
* in 'lv_refr_area_with_vdb' function */
union_ok = lv_area_intersect(&res_a, cords_p, mask_p);
2018-06-19 09:49:58 +02:00
2016-06-08 07:25:08 +02:00
/*If there are common part of the three area then draw to the vdb*/
if(union_ok == false) {
return;
}
2019-04-04 07:15:40 +02:00
lv_disp_t * disp = lv_refr_get_disp_refreshing();
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp);
2019-02-12 12:21:34 +01:00
2019-04-04 07:15:40 +02:00
lv_area_t vdb_rel_a; /*Stores relative coordinates on vdb*/
2019-02-13 01:40:22 +01:00
vdb_rel_a.x1 = res_a.x1 - vdb->area.x1;
vdb_rel_a.y1 = res_a.y1 - vdb->area.y1;
vdb_rel_a.x2 = res_a.x2 - vdb->area.x1;
vdb_rel_a.y2 = res_a.y2 - vdb->area.y1;
2019-02-13 01:40:22 +01:00
lv_color_t * vdb_buf_tmp = vdb->buf_act;
2019-04-04 07:15:40 +02:00
uint32_t vdb_width = lv_area_get_width(&vdb->area);
/*Move the vdb_tmp to the first row*/
vdb_buf_tmp += vdb_width * vdb_rel_a.y1;
#if LV_USE_GPU
2019-04-04 07:15:40 +02:00
static LV_ATTRIBUTE_MEM_ALIGN lv_color_t
color_array_tmp[LV_HOR_RES_MAX]; /*Used by 'lv_disp_mem_blend'*/
static lv_coord_t last_width = -1;
lv_coord_t w = lv_area_get_width(&vdb_rel_a);
2017-11-29 15:21:47 +01:00
/*Don't use hw. acc. for every small fill (because of the init overhead)*/
if(w < VFILL_HW_ACC_SIZE_LIMIT) {
2019-02-13 01:40:22 +01:00
sw_color_fill(&vdb->area, vdb->buf_act, &vdb_rel_a, color, opa);
2017-11-29 15:21:47 +01:00
}
/*Not opaque fill*/
else if(opa == LV_OPA_COVER) {
2017-11-28 16:15:13 +01:00
/*Use hw fill if present*/
2019-04-05 06:55:35 +02:00
if(disp->driver.mem_fill_cb) {
2019-04-22 08:45:07 +02:00
disp->driver.mem_fill_cb(&disp->driver, vdb->buf_act, &vdb->area, &vdb_rel_a, color);
2017-11-28 16:15:13 +01:00
}
/*Use hw blend if present and the area is not too small*/
2017-11-29 15:21:47 +01:00
else if(lv_area_get_height(&vdb_rel_a) > VFILL_HW_ACC_SIZE_LIMIT &&
2019-04-05 06:55:35 +02:00
disp->driver.mem_blend_cb) {
2018-03-07 10:49:25 +01:00
/*Fill a one line sized buffer with a color and blend this later*/
2017-12-06 09:48:52 +01:00
if(color_array_tmp[0].full != color.full || last_width != w) {
2017-11-28 16:15:13 +01:00
uint16_t i;
for(i = 0; i < w; i++) {
2017-12-06 09:48:52 +01:00
color_array_tmp[i].full = color.full;
2017-11-28 16:15:13 +01:00
}
last_width = w;
}
2018-03-07 10:49:25 +01:00
/*Blend the filled line to every line VDB line-by-line*/
2017-11-28 16:15:13 +01:00
lv_coord_t row;
2018-06-19 09:49:58 +02:00
for(row = vdb_rel_a.y1; row <= vdb_rel_a.y2; row++) {
2019-04-22 08:45:07 +02:00
disp->driver.mem_blend_cb(&disp->driver, &vdb_buf_tmp[vdb_rel_a.x1], color_array_tmp, w, opa);
2017-11-28 16:15:13 +01:00
vdb_buf_tmp += vdb_width;
2017-10-18 13:47:35 +02:00
}
2017-06-26 16:20:05 +02:00
2017-10-18 13:47:35 +02:00
}
2017-11-28 16:15:13 +01:00
/*Else use sw fill if no better option*/
else {
2019-02-13 01:40:22 +01:00
sw_color_fill(&vdb->area, vdb->buf_act, &vdb_rel_a, color, opa);
2017-11-28 16:15:13 +01:00
}
2017-11-28 16:15:13 +01:00
}
2017-11-29 15:21:47 +01:00
/*Fill with opacity*/
else {
/*Use hw blend if present*/
2019-04-05 06:55:35 +02:00
if(disp->driver.mem_blend_cb) {
2017-12-06 09:48:52 +01:00
if(color_array_tmp[0].full != color.full || last_width != w) {
2017-11-28 16:15:13 +01:00
uint16_t i;
for(i = 0; i < w; i++) {
2017-12-06 09:48:52 +01:00
color_array_tmp[i].full = color.full;
2017-11-28 16:15:13 +01:00
}
last_width = w;
}
lv_coord_t row;
2018-06-19 09:49:58 +02:00
for(row = vdb_rel_a.y1; row <= vdb_rel_a.y2; row++) {
2019-04-22 08:45:07 +02:00
disp->driver.mem_blend_cb(&disp->driver, &vdb_buf_tmp[vdb_rel_a.x1], color_array_tmp, w, opa);
2017-11-28 16:15:13 +01:00
vdb_buf_tmp += vdb_width;
}
2017-11-29 15:21:47 +01:00
}
/*Use sw fill with opa if no better option*/
else {
2019-02-13 01:40:22 +01:00
sw_color_fill(&vdb->area, vdb->buf_act, &vdb_rel_a, color, opa);
2017-10-18 13:47:35 +02:00
}
2017-06-26 16:20:05 +02:00
}
#else
2019-02-13 01:40:22 +01:00
sw_color_fill(&vdb->area, vdb->buf_act, &vdb_rel_a, color, opa);
#endif
2016-06-08 07:25:08 +02:00
}
/**
* Draw a letter in the Virtual Display Buffer
* @param pos_p left-top coordinate of the latter
2017-04-24 12:08:24 +02:00
* @param mask_p the letter will be drawn only on this area (truncated to VDB area)
2018-06-19 09:49:58 +02:00
* @param font_p pointer to font
2016-06-08 07:25:08 +02:00
* @param letter a letter to draw
* @param color color of letter
* @param opa opacity of letter (0..255)
*/
2019-04-04 07:15:40 +02:00
void lv_draw_letter(const lv_point_t * pos_p, const lv_area_t * mask_p, const lv_font_t * font_p,
uint32_t letter, lv_color_t color, lv_opa_t opa)
2018-02-05 11:27:08 +01:00
{
/*clang-format off*/
const uint8_t bpp1_opa_table[2] = { 0, 255}; /*Opacity mapping with bpp = 1 (Just for compatibility)*/
2019-04-04 07:15:40 +02:00
const uint8_t bpp2_opa_table[4] = {0, 85, 170, 255}; /*Opacity mapping with bpp = 2*/
const uint8_t bpp4_opa_table[16] = {0, 17, 34, 51, /*Opacity mapping with bpp = 4*/
68, 85, 102, 119,
136, 153, 170, 187,
204, 221, 238, 255};
/*clang-format on*/
if(opa < LV_OPA_MIN) return;
if(opa > LV_OPA_MAX) opa = LV_OPA_COVER;
2018-07-25 17:57:08 +02:00
if(font_p == NULL) {
LV_LOG_WARN("Font: character's bitmap not found");
return;
2018-07-25 17:57:08 +02:00
}
2019-05-01 16:43:32 +02:00
lv_font_glyph_dsc_t g;
2019-05-29 06:40:19 +02:00
bool g_ret = lv_font_get_glyph_dsc(font_p, &g, letter, '\0');
2019-05-01 16:43:32 +02:00
if(g_ret == false) return;
2018-02-05 11:27:08 +01:00
2019-06-05 12:25:49 +02:00
lv_coord_t pos_x = pos_p->x + g.ofs_x;
2019-06-02 11:56:57 +02:00
lv_coord_t pos_y = pos_p->y + (font_p->line_height - font_p->base_line) - g.box_h - g.ofs_y;
2018-06-22 23:32:21 +02:00
const uint8_t * bpp_opa_table;
uint8_t bitmask_init;
uint8_t bitmask;
2018-02-05 11:27:08 +01:00
2019-05-01 16:43:32 +02:00
switch(g.bpp) {
2018-06-19 09:49:58 +02:00
case 1:
bpp_opa_table = bpp1_opa_table;
bitmask_init = 0x80;
2018-06-19 09:49:58 +02:00
break;
case 2:
bpp_opa_table = bpp2_opa_table;
bitmask_init = 0xC0;
2018-06-19 09:49:58 +02:00
break;
case 4:
bpp_opa_table = bpp4_opa_table;
bitmask_init = 0xF0;
2018-06-19 09:49:58 +02:00
break;
case 8:
bpp_opa_table = NULL;
bitmask_init = 0xFF;
2019-04-04 07:15:40 +02:00
break; /*No opa table, pixel value will be used directly*/
default: return; /*Invalid bpp. Can't render the letter*/
2018-02-05 11:27:08 +01:00
}
2019-04-22 05:21:35 +02:00
const uint8_t * map_p = lv_font_get_glyph_bitmap(font_p, letter);
if(map_p == NULL) return;
2016-06-08 07:25:08 +02:00
/*If the letter is completely out of mask don't draw it */
2019-05-01 16:43:32 +02:00
if(pos_x + g.box_w < mask_p->x1 || pos_x > mask_p->x2 || pos_y + g.box_h < mask_p->y1 ||
2019-04-04 07:15:40 +02:00
pos_y > mask_p->y2)
return;
2016-06-08 07:25:08 +02:00
2019-04-04 07:15:40 +02:00
lv_disp_t * disp = lv_refr_get_disp_refreshing();
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp);
2018-09-12 09:03:48 +02:00
2019-04-04 07:15:40 +02:00
lv_coord_t vdb_width = lv_area_get_width(&vdb->area);
2019-02-13 01:40:22 +01:00
lv_color_t * vdb_buf_tmp = vdb->buf_act;
2017-11-23 21:28:36 +01:00
lv_coord_t col, row;
2019-05-01 16:43:32 +02:00
uint8_t width_byte_scr = g.box_w >> 3; /*Width in bytes (on the screen finally) (e.g. w = 11 -> 2 bytes wide)*/
if(g.box_w & 0x7) width_byte_scr++;
2019-06-01 21:46:05 +02:00
uint8_t width_bit = g.box_w * g.bpp; /*Letter width in bits*/
/* Calculate the col/row start/end on the map*/
2018-06-22 23:32:21 +02:00
lv_coord_t col_start = pos_x >= mask_p->x1 ? 0 : mask_p->x1 - pos_x;
2019-05-01 16:43:32 +02:00
lv_coord_t col_end = pos_x + g.box_w <= mask_p->x2 ? g.box_w : mask_p->x2 - pos_x + 1;
2018-06-22 23:32:21 +02:00
lv_coord_t row_start = pos_y >= mask_p->y1 ? 0 : mask_p->y1 - pos_y;
2019-05-01 16:43:32 +02:00
lv_coord_t row_end = pos_y + g.box_h <= mask_p->y2 ? g.box_h : mask_p->y2 - pos_y + 1;
/*Set a pointer on VDB to the first pixel of the letter*/
2019-04-04 07:15:40 +02:00
vdb_buf_tmp += ((pos_y - vdb->area.y1) * vdb_width) + pos_x - vdb->area.x1;
/*If the letter is partially out of mask the move there on VDB*/
vdb_buf_tmp += (row_start * vdb_width) + col_start;
/*Move on the map too*/
2019-06-01 21:46:05 +02:00
uint32_t bit_ofs = (row_start * width_bit) + (col_start * g.bpp);
map_p += bit_ofs>> 3;
2017-03-20 07:07:05 +01:00
2018-02-05 11:27:08 +01:00
uint8_t letter_px;
lv_opa_t px_opa;
2019-06-01 21:46:05 +02:00
uint16_t col_bit;
col_bit = bit_ofs & 0x7; /* "& 0x7" equals to "% 8" just faster */
2019-04-04 07:15:40 +02:00
for(row = row_start; row < row_end; row++) {
bitmask = bitmask_init >> col_bit;
2019-04-04 07:15:40 +02:00
for(col = col_start; col < col_end; col++) {
2019-05-01 16:43:32 +02:00
letter_px = (*map_p & bitmask) >> (8 - col_bit - g.bpp);
2018-02-05 11:27:08 +01:00
if(letter_px != 0) {
if(opa == LV_OPA_COVER) {
2019-05-01 16:43:32 +02:00
px_opa = g.bpp == 8 ? letter_px : bpp_opa_table[letter_px];
} else {
2019-05-01 16:43:32 +02:00
px_opa = g.bpp == 8 ? (uint16_t)((uint16_t)letter_px * opa) >> 8
2019-04-04 07:15:40 +02:00
: (uint16_t)((uint16_t)bpp_opa_table[letter_px] * opa) >> 8;
}
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
disp->driver.set_px_cb(&disp->driver, (uint8_t *)vdb->buf_act, vdb_width,
2019-04-04 07:15:40 +02:00
(col + pos_x) - vdb->area.x1,
(row + pos_y) - vdb->area.y1, color, px_opa);
2019-05-07 05:40:01 +02:00
} else if (vdb_buf_tmp->full != color.full) {
if(px_opa > LV_OPA_MAX) *vdb_buf_tmp = color;
2019-05-03 19:25:58 +02:00
else if(px_opa > LV_OPA_MIN) {
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP == 0
*vdb_buf_tmp = lv_color_mix(color, *vdb_buf_tmp, px_opa);
2018-09-20 22:16:03 +02:00
#else
*vdb_buf_tmp =
color_mix_2_alpha(*vdb_buf_tmp, (*vdb_buf_tmp).alpha, color, px_opa);
2018-09-20 22:16:03 +02:00
#endif
}
}
2016-06-08 07:25:08 +02:00
}
2018-02-05 11:27:08 +01:00
vdb_buf_tmp++;
2017-03-20 07:07:05 +01:00
2019-05-01 16:43:32 +02:00
if(col_bit < 8 - g.bpp) {
col_bit += g.bpp;
bitmask = bitmask >> g.bpp;
2018-06-19 09:49:58 +02:00
} else {
2018-02-05 11:27:08 +01:00
col_bit = 0;
bitmask = bitmask_init;
2019-04-04 07:15:40 +02:00
map_p++;
}
2016-06-08 07:25:08 +02:00
}
2019-06-01 21:46:05 +02:00
col_bit += ((g.box_w - col_end) + col_start) * g.bpp;
2019-06-01 21:46:05 +02:00
map_p += (col_bit >> 3);
col_bit = col_bit & 0x7;
2019-04-04 07:15:40 +02:00
vdb_buf_tmp += vdb_width - (col_end - col_start); /*Next row in VDB*/
2016-06-08 07:25:08 +02:00
}
}
/**
* Draw a color map to the display (image)
2016-06-08 07:25:08 +02:00
* @param cords_p coordinates the color map
2017-04-24 12:08:24 +02:00
* @param mask_p the map will drawn only on this area (truncated to VDB area)
2017-11-23 21:28:36 +01:00
* @param map_p pointer to a lv_color_t array
* @param opa opacity of the map
* @param chroma_keyed true: enable transparency of LV_IMG_LV_COLOR_TRANSP color pixels
* @param alpha_byte true: extra alpha byte is inserted for every pixel
* @param recolor mix the pixels with this color
* @param recolor_opa the intense of recoloring
2016-06-08 07:25:08 +02:00
*/
2019-04-04 07:15:40 +02:00
void lv_draw_map(const lv_area_t * cords_p, const lv_area_t * mask_p, const uint8_t * map_p,
lv_opa_t opa, bool chroma_key, bool alpha_byte, lv_color_t recolor,
lv_opa_t recolor_opa)
2016-06-08 07:25:08 +02:00
{
if(opa < LV_OPA_MIN) return;
if(opa > LV_OPA_MAX) opa = LV_OPA_COVER;
2017-11-23 21:28:36 +01:00
lv_area_t masked_a;
2016-06-08 07:25:08 +02:00
bool union_ok;
2016-06-08 07:25:08 +02:00
/*Get the union of map size and mask*/
/* The mask is already truncated to the vdb size
2019-04-04 07:15:40 +02:00
* in 'lv_refr_area_with_vdb' function */
union_ok = lv_area_intersect(&masked_a, cords_p, mask_p);
2016-06-08 07:25:08 +02:00
/*If there are common part of the three area then draw to the vdb*/
2019-04-04 07:15:40 +02:00
if(union_ok == false) return;
2018-02-09 12:40:00 +01:00
/*The pixel size in byte is different if an alpha byte is added too*/
uint8_t px_size_byte = alpha_byte ? LV_IMG_PX_SIZE_ALPHA_BYTE : sizeof(lv_color_t);
2016-06-08 07:25:08 +02:00
/*If the map starts OUT of the masked area then calc. the first pixel*/
2018-02-09 12:40:00 +01:00
lv_coord_t map_width = lv_area_get_width(cords_p);
2016-06-08 07:25:08 +02:00
if(cords_p->y1 < masked_a.y1) {
2019-04-04 07:15:40 +02:00
map_p += (uint32_t)map_width * ((masked_a.y1 - cords_p->y1)) * px_size_byte;
2016-06-08 07:25:08 +02:00
}
if(cords_p->x1 < masked_a.x1) {
2018-02-09 12:40:00 +01:00
map_p += (masked_a.x1 - cords_p->x1) * px_size_byte;
2016-06-08 07:25:08 +02:00
}
2019-04-04 07:15:40 +02:00
lv_disp_t * disp = lv_refr_get_disp_refreshing();
2019-02-20 23:58:13 +01:00
lv_disp_buf_t * vdb = lv_disp_get_buf(disp);
2019-02-13 01:40:22 +01:00
2018-02-09 12:40:00 +01:00
/*Stores coordinates relative to the current VDB*/
2019-02-13 01:40:22 +01:00
masked_a.x1 = masked_a.x1 - vdb->area.x1;
masked_a.y1 = masked_a.y1 - vdb->area.y1;
masked_a.x2 = masked_a.x2 - vdb->area.x1;
masked_a.y2 = masked_a.y2 - vdb->area.y1;
2016-06-08 07:25:08 +02:00
2019-04-04 07:15:40 +02:00
lv_coord_t vdb_width = lv_area_get_width(&vdb->area);
2019-02-13 01:40:22 +01:00
lv_color_t * vdb_buf_tmp = vdb->buf_act;
2019-04-04 07:15:40 +02:00
vdb_buf_tmp += (uint32_t)vdb_width * masked_a.y1; /*Move to the first row*/
vdb_buf_tmp += (uint32_t)masked_a.x1; /*Move to the first col*/
2016-06-08 07:25:08 +02:00
2018-02-09 12:40:00 +01:00
lv_coord_t row;
lv_coord_t map_useful_w = lv_area_get_width(&masked_a);
2018-02-09 12:40:00 +01:00
/*The simplest case just copy the pixels into the VDB*/
2019-04-04 07:15:40 +02:00
if(chroma_key == false && alpha_byte == false && opa == LV_OPA_COVER &&
recolor_opa == LV_OPA_TRANSP) {
/*Use the custom VDB write function is exists*/
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
lv_coord_t col;
for(row = masked_a.y1; row <= masked_a.y2; row++) {
for(col = 0; col < map_useful_w; col++) {
lv_color_t px_color = *((lv_color_t *)&map_p[(uint32_t)col * px_size_byte]);
2019-04-04 07:15:40 +02:00
disp->driver.set_px_cb(&disp->driver, (uint8_t *)vdb->buf_act, vdb_width,
col + masked_a.x1, row, px_color, opa);
}
2019-04-04 07:15:40 +02:00
map_p += map_width * px_size_byte; /*Next row on the map*/
}
}
/*Normal native VDB*/
else {
for(row = masked_a.y1; row <= masked_a.y2; row++) {
#if LV_USE_GPU
2019-04-05 06:55:35 +02:00
if(disp->driver.mem_blend_cb == false) {
sw_mem_blend(vdb_buf_tmp, (lv_color_t *)map_p, map_useful_w, opa);
} else {
2019-04-22 08:45:07 +02:00
disp->driver.mem_blend_cb(&disp->driver, vdb_buf_tmp, (lv_color_t *)map_p, map_useful_w, opa);
}
#else
sw_mem_blend(vdb_buf_tmp, (lv_color_t *)map_p, map_useful_w, opa);
#endif
2019-04-04 07:15:40 +02:00
map_p += map_width * px_size_byte; /*Next row on the map*/
vdb_buf_tmp += vdb_width; /*Next row on the VDB*/
}
}
2016-06-08 07:25:08 +02:00
}
2018-02-09 12:40:00 +01:00
/*In the other cases every pixel need to be checked one-by-one*/
2017-03-09 11:23:28 +01:00
else {
2017-11-23 21:28:36 +01:00
lv_coord_t col;
2019-04-04 07:15:40 +02:00
lv_color_t last_img_px = LV_COLOR_BLACK;
2018-02-27 11:43:14 +01:00
lv_color_t recolored_px = lv_color_mix(recolor, last_img_px, recolor_opa);
2018-02-09 12:40:00 +01:00
for(row = masked_a.y1; row <= masked_a.y2; row++) {
for(col = 0; col < map_useful_w; col++) {
2019-04-04 07:15:40 +02:00
lv_opa_t opa_result = opa;
uint8_t * px_color_p = (uint8_t *)&map_p[(uint32_t)col * px_size_byte];
lv_color_t px_color;
2018-02-09 12:40:00 +01:00
/*Calculate with the pixel level alpha*/
if(alpha_byte) {
#if LV_COLOR_DEPTH == 8 || LV_COLOR_DEPTH == 1
2018-06-19 09:49:58 +02:00
px_color.full = px_color_p[0];
#elif LV_COLOR_DEPTH == 16
2019-04-04 07:15:40 +02:00
/*Because of Alpha byte 16 bit color can start on odd address which can cause
* crash*/
2018-06-19 09:49:58 +02:00
px_color.full = px_color_p[0] + (px_color_p[1] << 8);
#elif LV_COLOR_DEPTH == 32
2018-06-19 09:49:58 +02:00
px_color = *((lv_color_t *)px_color_p);
#endif
lv_opa_t px_opa = *(px_color_p + LV_IMG_PX_SIZE_ALPHA_BYTE - 1);
2019-04-04 07:15:40 +02:00
if(px_opa == LV_OPA_TRANSP)
continue;
else if(px_opa != LV_OPA_COVER)
opa_result = (uint32_t)((uint32_t)px_opa * opa_result) >> 8;
2018-06-19 09:49:58 +02:00
} else {
px_color = *((lv_color_t *)px_color_p);
}
/*Handle chroma key*/
if(chroma_key && px_color.full == disp->driver.color_chroma_key.full) continue;
2018-02-09 12:40:00 +01:00
2018-02-16 12:55:05 +01:00
/*Re-color the pixel if required*/
if(recolor_opa != LV_OPA_TRANSP) {
2019-04-04 07:15:40 +02:00
if(last_img_px.full != px_color.full) { /*Minor acceleration: calculate only for
new colors (save the last)*/
last_img_px = px_color;
recolored_px = lv_color_mix(recolor, last_img_px, recolor_opa);
}
/*Handle custom VDB write is present*/
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
2019-04-04 07:15:40 +02:00
disp->driver.set_px_cb(&disp->driver, (uint8_t *)vdb->buf_act, vdb_width,
col + masked_a.x1, row, recolored_px, opa_result);
}
/*Normal native VDB write*/
else {
2019-04-04 07:15:40 +02:00
if(opa_result == LV_OPA_COVER)
vdb_buf_tmp[col].full = recolored_px.full;
else
vdb_buf_tmp[col] =
lv_color_mix(recolored_px, vdb_buf_tmp[col], opa_result);
}
2018-02-16 12:55:05 +01:00
} else {
/*Handle custom VDB write is present*/
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
2019-04-04 07:15:40 +02:00
disp->driver.set_px_cb(&disp->driver, (uint8_t *)vdb->buf_act, vdb_width,
col + masked_a.x1, row, px_color, opa_result);
}
/*Normal native VDB write*/
else {
2019-04-04 07:15:40 +02:00
if(opa_result == LV_OPA_COVER)
vdb_buf_tmp[col] = px_color;
2018-09-20 22:16:03 +02:00
else {
#if LV_COLOR_SCREEN_TRANSP == 0
vdb_buf_tmp[col] = lv_color_mix(px_color, vdb_buf_tmp[col], opa_result);
#else
2019-04-04 07:15:40 +02:00
vdb_buf_tmp[col] = color_mix_2_alpha(
vdb_buf_tmp[col], vdb_buf_tmp[col].alpha, px_color, opa_result);
2018-09-20 22:16:03 +02:00
#endif
}
}
2018-02-16 12:55:05 +01:00
}
2017-03-09 11:23:28 +01:00
}
2018-02-09 12:40:00 +01:00
2019-04-04 07:15:40 +02:00
map_p += map_width * px_size_byte; /*Next row on the map*/
vdb_buf_tmp += vdb_width; /*Next row on the VDB*/
2017-03-09 11:23:28 +01:00
}
}
2016-06-08 07:25:08 +02:00
}
/**********************
* STATIC FUNCTIONS
**********************/
2017-06-26 16:20:05 +02:00
/**
2017-11-28 16:15:13 +01:00
* Blend pixels to destination memory using opacity
2017-06-26 16:20:05 +02:00
* @param dest a memory address. Copy 'src' here.
* @param src pointer to pixel map. Copy it to 'dest'.
* @param length number of pixels in 'src'
2017-11-23 21:28:36 +01:00
* @param opa opacity (0, LV_OPA_TRANSP: transparent ... 255, LV_OPA_COVER, fully cover)
2017-06-26 16:20:05 +02:00
*/
2017-11-28 16:15:13 +01:00
static void sw_mem_blend(lv_color_t * dest, const lv_color_t * src, uint32_t length, lv_opa_t opa)
2017-06-26 16:20:05 +02:00
{
2017-11-23 21:28:36 +01:00
if(opa == LV_OPA_COVER) {
memcpy(dest, src, length * sizeof(lv_color_t));
2017-06-26 16:20:05 +02:00
} else {
2017-12-02 20:43:50 +01:00
uint32_t col;
2017-06-26 16:20:05 +02:00
for(col = 0; col < length; col++) {
dest[col] = lv_color_mix(src[col], dest[col], opa);
}
2017-06-26 16:20:05 +02:00
}
}
/**
*
* @param mem_area coordinates of 'mem' memory area
2017-11-22 13:26:20 +01:00
* @param mem a memory address. Considered to a rectangular window according to 'mem_area'
* @param fill_area coordinates of an area to fill. Relative to 'mem_area'.
* @param color fill color
2017-11-23 21:28:36 +01:00
* @param opa opacity (0, LV_OPA_TRANSP: transparent ... 255, LV_OPA_COVER, fully cover)
*/
2019-04-04 07:15:40 +02:00
static void sw_color_fill(lv_area_t * mem_area, lv_color_t * mem, const lv_area_t * fill_area,
lv_color_t color, lv_opa_t opa)
{
/*Set all row in vdb to the given color*/
2017-11-23 21:28:36 +01:00
lv_coord_t row;
2017-12-02 20:43:50 +01:00
lv_coord_t col;
2017-11-28 16:15:13 +01:00
lv_coord_t mem_width = lv_area_get_width(mem_area);
2019-02-12 12:21:34 +01:00
lv_disp_t * disp = lv_refr_get_disp_refreshing();
2019-02-20 23:58:13 +01:00
if(disp->driver.set_px_cb) {
for(col = fill_area->x1; col <= fill_area->x2; col++) {
for(row = fill_area->y1; row <= fill_area->y2; row++) {
2019-04-04 07:15:40 +02:00
disp->driver.set_px_cb(&disp->driver, (uint8_t *)mem, mem_width, col, row, color,
opa);
}
}
} else {
2019-04-04 07:15:40 +02:00
mem += fill_area->y1 * mem_width; /*Go to the first row*/
/*Run simpler function without opacity*/
if(opa == LV_OPA_COVER) {
/*Fill the first row with 'color'*/
for(col = fill_area->x1; col <= fill_area->x2; col++) {
mem[col] = color;
}
/*Copy the first row to all other rows*/
lv_color_t * mem_first = &mem[fill_area->x1];
2019-04-04 07:15:40 +02:00
lv_coord_t copy_size = (fill_area->x2 - fill_area->x1 + 1) * sizeof(lv_color_t);
mem += mem_width;
for(row = fill_area->y1 + 1; row <= fill_area->y2; row++) {
memcpy(&mem[fill_area->x1], mem_first, copy_size);
mem += mem_width;
}
}
/*Calculate with alpha too*/
else {
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP == 0
2019-04-04 07:15:40 +02:00
lv_color_t bg_tmp = LV_COLOR_BLACK;
lv_color_t opa_tmp = lv_color_mix(color, bg_tmp, opa);
2018-09-20 22:16:03 +02:00
#endif
for(row = fill_area->y1; row <= fill_area->y2; row++) {
for(col = fill_area->x1; col <= fill_area->x2; col++) {
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP == 0
/*If the bg color changed recalculate the result color*/
if(mem[col].full != bg_tmp.full) {
2019-04-04 07:15:40 +02:00
bg_tmp = mem[col];
opa_tmp = lv_color_mix(color, bg_tmp, opa);
}
2018-09-20 22:16:03 +02:00
mem[col] = opa_tmp;
2018-09-20 22:16:03 +02:00
#else
mem[col] = color_mix_2_alpha(mem[col], mem[col].alpha, color, opa);
#endif
}
mem += mem_width;
}
}
}
}
2018-09-20 22:16:03 +02:00
#if LV_COLOR_SCREEN_TRANSP
/**
* Mix two colors. Both color can have alpha value. It requires ARGB888 colors.
* @param bg_color background color
* @param bg_opa alpha of the background color
* @param fg_color foreground color
* @param fg_opa alpha of the foreground color
* @return the mixed color. the alpha channel (color.alpha) contains the result alpha
*/
2019-04-04 07:15:40 +02:00
static inline lv_color_t color_mix_2_alpha(lv_color_t bg_color, lv_opa_t bg_opa,
lv_color_t fg_color, lv_opa_t fg_opa)
2018-09-20 22:16:03 +02:00
{
/* Pick the foreground if it's fully opaque or the Background is fully transparent*/
if(fg_opa == LV_OPA_COVER || bg_opa <= LV_OPA_MIN) {
fg_color.ch.alpha = fg_opa;
2018-09-20 22:16:03 +02:00
return fg_color;
}
/*Transparent foreground: use the Background*/
else if(fg_opa <= LV_OPA_MIN) {
return bg_color;
}
/*Opaque background: use simple mix*/
2018-10-05 17:22:49 +02:00
else if(bg_opa >= LV_OPA_MAX) {
2018-09-20 22:16:03 +02:00
return lv_color_mix(fg_color, bg_color, fg_opa);
}
/*Both colors have alpha. Expensive calculation need to be applied*/
else {
/*Save the parameters and the result. If they will be asked again don't compute again*/
2019-04-04 07:15:40 +02:00
static lv_opa_t fg_opa_save = 0;
static lv_opa_t bg_opa_save = 0;
static lv_color_t fg_color_save = {{0}};
static lv_color_t bg_color_save = {{0}};
2019-04-04 07:15:40 +02:00
static lv_color_t c = {{0}};
2018-09-20 22:16:03 +02:00
2019-04-04 07:15:40 +02:00
if(fg_opa != fg_opa_save || bg_opa != bg_opa_save || fg_color.full != fg_color_save.full ||
bg_color.full != bg_color_save.full) {
fg_opa_save = fg_opa;
bg_opa_save = bg_opa;
fg_color.full = fg_color_save.full;
bg_color.full = bg_color_save.full;
2019-04-04 07:15:40 +02:00
/*Info:
* https://en.wikipedia.org/wiki/Alpha_compositing#Analytical_derivation_of_the_over_operator*/
2018-09-20 22:16:03 +02:00
lv_opa_t alpha_res = 255 - ((uint16_t)((uint16_t)(255 - fg_opa) * (255 - bg_opa)) >> 8);
if(alpha_res == 0) {
2019-04-04 07:15:40 +02:00
while(1)
;
2018-09-20 22:16:03 +02:00
}
2019-04-04 07:15:40 +02:00
lv_opa_t ratio = (uint16_t)((uint16_t)fg_opa * 255) / alpha_res;
c = lv_color_mix(fg_color, bg_color, ratio);
c.ch.alpha = alpha_res;
2018-09-20 22:16:03 +02:00
}
return c;
}
}
2018-09-20 22:16:03 +02:00
#endif /*LV_COLOR_SCREEN_TRANSP*/