2017-04-13 10:20:35 +02:00
|
|
|
/**
|
|
|
|
* @file lv_style.c
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*********************
|
|
|
|
* INCLUDES
|
|
|
|
*********************/
|
2017-07-28 13:57:56 +02:00
|
|
|
#include "lv_obj.h"
|
2019-09-24 16:30:38 +02:00
|
|
|
#include "../lv_core/lv_debug.h"
|
2017-11-23 20:42:14 +01:00
|
|
|
#include "../lv_misc/lv_mem.h"
|
2019-05-20 18:38:24 -07:00
|
|
|
#include "../lv_misc/lv_anim.h"
|
2017-04-13 10:20:35 +02:00
|
|
|
|
|
|
|
/*********************
|
|
|
|
* DEFINES
|
|
|
|
*********************/
|
2019-04-04 07:15:40 +02:00
|
|
|
#define STYLE_MIX_MAX 256
|
|
|
|
#define STYLE_MIX_SHIFT 8 /*log2(STYLE_MIX_MAX)*/
|
|
|
|
|
|
|
|
#define VAL_PROP(v1, v2, r) v1 + (((v2 - v1) * r) >> STYLE_MIX_SHIFT)
|
2019-06-06 06:05:40 +02:00
|
|
|
#define STYLE_ATTR_MIX(attr, r) \
|
|
|
|
if(start->attr != end->attr) { \
|
|
|
|
res->attr = VAL_PROP(start->attr, end->attr, r); \
|
|
|
|
} else { \
|
|
|
|
res->attr = start->attr; \
|
2019-04-04 07:15:40 +02:00
|
|
|
}
|
2017-04-13 10:20:35 +02:00
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
#define LV_STYLE_PROP_TO_ID(prop) (prop & 0xFF);
|
|
|
|
#define LV_STYLE_PROP_GET_TYPE(prop) ((prop >> 8) & 0xFF);
|
|
|
|
|
2017-04-13 10:20:35 +02:00
|
|
|
/**********************
|
|
|
|
* TYPEDEFS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* STATIC PROTOTYPES
|
|
|
|
**********************/
|
2019-12-14 23:39:26 +01:00
|
|
|
static inline int32_t get_property_index(const lv_style_t * style, lv_style_property_t prop);
|
2019-03-07 00:05:16 +01:00
|
|
|
#if LV_USE_ANIMATION
|
2019-05-20 18:38:24 -07:00
|
|
|
static void style_animator(lv_style_anim_dsc_t * dsc, lv_anim_value_t val);
|
2019-04-22 08:45:07 +02:00
|
|
|
static void style_animation_common_end_cb(lv_anim_t * a);
|
2017-11-27 17:48:54 +01:00
|
|
|
#endif
|
2017-04-13 10:20:35 +02:00
|
|
|
|
2019-12-22 22:40:02 +01:00
|
|
|
/**********************
|
|
|
|
* GLOABAL VARIABLES
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/*Basic styles*/
|
|
|
|
lv_style_t lv_style_plain;
|
|
|
|
lv_style_t lv_style_panel;
|
|
|
|
lv_style_t lv_style_panel;
|
|
|
|
lv_style_t lv_style_btn;
|
|
|
|
|
|
|
|
/*Color styles*/
|
|
|
|
lv_style_t lv_style_dark;
|
|
|
|
lv_style_t lv_style_light;
|
|
|
|
lv_style_t lv_style_red;
|
|
|
|
lv_style_t lv_style_green;
|
|
|
|
lv_style_t lv_style_blue;
|
|
|
|
|
|
|
|
/*Transparent styles*/
|
|
|
|
lv_style_t lv_style_transp;
|
|
|
|
lv_style_t lv_style_frame;
|
|
|
|
|
|
|
|
/*Padding styles*/
|
|
|
|
lv_style_t lv_style_tight;
|
|
|
|
lv_style_t lv_style_fit;
|
|
|
|
|
2017-04-13 10:20:35 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC VARIABLES
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* MACROS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* GLOBAL FUNCTIONS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**
|
2019-12-14 23:39:26 +01:00
|
|
|
* Init. the built-in styles
|
2017-04-13 10:20:35 +02:00
|
|
|
*/
|
2019-12-14 23:39:26 +01:00
|
|
|
void lv_style_built_in_init(void)
|
2017-04-13 10:20:35 +02:00
|
|
|
{
|
2019-09-27 03:28:44 +02:00
|
|
|
|
2019-12-22 22:40:02 +01:00
|
|
|
lv_mem_monitor_t mon;
|
|
|
|
lv_mem_monitor(&mon);
|
|
|
|
printf("used: %6d (%3d %%), frag: %3d %%, biggest free: %6d\n", (int)mon.total_size - mon.free_size,
|
|
|
|
mon.used_pct,
|
|
|
|
mon.frag_pct,
|
|
|
|
(int)mon.free_biggest_size);
|
|
|
|
|
|
|
|
|
|
|
|
/*Basic styles*/
|
|
|
|
lv_style_init(&lv_style_plain);
|
|
|
|
lv_style_set_value(&lv_style_plain, LV_STYLE_PAD_LEFT, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_plain, LV_STYLE_PAD_RIGHT, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_plain, LV_STYLE_PAD_TOP, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_plain, LV_STYLE_PAD_BOTTOM, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_plain, LV_STYLE_PAD_INNER, LV_DPI / 16);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_panel);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_PAD_LEFT, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_PAD_RIGHT, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_PAD_TOP, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_PAD_BOTTOM, LV_DPI / 12);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_PAD_INNER, LV_DPI / 20);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_RADIUS, LV_DPI / 16);
|
|
|
|
lv_style_set_value(&lv_style_panel, LV_STYLE_BORDER_WIDTH, LV_DPI / 50 > 0 ? LV_DPI / 50 : 1);
|
|
|
|
lv_style_set_color(&lv_style_panel, LV_STYLE_BG_COLOR, LV_COLOR_SILVER);
|
|
|
|
lv_style_set_color(&lv_style_panel, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_GRAY);
|
|
|
|
lv_style_set_color(&lv_style_panel, LV_STYLE_BORDER_COLOR, LV_COLOR_GRAY);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_btn);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_PAD_LEFT, LV_DPI / 6);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_PAD_RIGHT, LV_DPI / 6);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_PAD_TOP, LV_DPI / 10);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_PAD_BOTTOM, LV_DPI / 10);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_PAD_INNER, LV_DPI / 16);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_RADIUS, LV_DPI / 20);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_BORDER_WIDTH, LV_DPI / 50 > 0 ? LV_DPI / 50 : 1);
|
|
|
|
lv_style_set_value(&lv_style_btn, LV_STYLE_BG_GRAD_DIR, LV_GRAD_DIR_VER);
|
|
|
|
lv_style_set_color(&lv_style_btn, LV_STYLE_BG_COLOR, LV_COLOR_BLUE);
|
|
|
|
lv_style_set_color(&lv_style_btn, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_NAVY);
|
|
|
|
lv_style_set_color(&lv_style_btn, LV_STYLE_BORDER_COLOR, LV_COLOR_NAVY);
|
|
|
|
lv_style_set_color(&lv_style_btn, LV_STYLE_BG_GRAD_COLOR | LV_STYLE_STATE_PRESSED, LV_COLOR_BLACK);
|
|
|
|
lv_style_set_color(&lv_style_btn, LV_STYLE_TEXT_COLOR, LV_COLOR_WHITE);
|
|
|
|
|
|
|
|
/*Color styles*/
|
|
|
|
lv_style_init(&lv_style_light);
|
|
|
|
lv_style_set_color(&lv_style_light, LV_STYLE_BG_COLOR, LV_COLOR_SILVER);
|
|
|
|
lv_style_set_color(&lv_style_light, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_GRAY);
|
|
|
|
lv_style_set_color(&lv_style_light, LV_STYLE_BORDER_COLOR, LV_COLOR_GRAY);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_dark);
|
|
|
|
lv_style_set_color(&lv_style_dark, LV_STYLE_BG_COLOR, lv_color_hex3(0x444));
|
|
|
|
lv_style_set_color(&lv_style_dark, LV_STYLE_BG_GRAD_COLOR, lv_color_hex3(0x222));
|
|
|
|
lv_style_set_color(&lv_style_dark, LV_STYLE_BORDER_COLOR, LV_COLOR_GRAY);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_TEXT_COLOR, LV_COLOR_WHITE);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_red);
|
|
|
|
lv_style_set_color(&lv_style_red, LV_STYLE_BG_COLOR, LV_COLOR_RED);
|
|
|
|
lv_style_set_color(&lv_style_red, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_MAROON);
|
|
|
|
lv_style_set_color(&lv_style_red, LV_STYLE_BORDER_COLOR, LV_COLOR_MAROON);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_TEXT_COLOR, LV_COLOR_WHITE);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_green);
|
|
|
|
lv_style_set_color(&lv_style_green, LV_STYLE_BG_COLOR, LV_COLOR_LIME);
|
|
|
|
lv_style_set_color(&lv_style_green, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_GREEN);
|
|
|
|
lv_style_set_color(&lv_style_green, LV_STYLE_BORDER_COLOR, LV_COLOR_GREEN);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_TEXT_COLOR, LV_COLOR_WHITE);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_blue);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_BG_COLOR, LV_COLOR_BLUE);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_BG_GRAD_COLOR, LV_COLOR_NAVY);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_BORDER_COLOR, LV_COLOR_NAVY);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_BG_GRAD_COLOR | LV_STYLE_STATE_PRESSED, LV_COLOR_BLACK);
|
|
|
|
lv_style_set_color(&lv_style_blue, LV_STYLE_TEXT_COLOR, LV_COLOR_WHITE);
|
|
|
|
|
|
|
|
/*Transparent styles*/
|
|
|
|
lv_style_init(&lv_style_transp);
|
|
|
|
lv_style_set_opa(&lv_style_transp, LV_STYLE_BG_OPA, LV_OPA_TRANSP);
|
|
|
|
lv_style_set_opa(&lv_style_transp, LV_STYLE_BORDER_OPA, LV_OPA_TRANSP);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_frame);
|
|
|
|
lv_style_set_opa(&lv_style_frame, LV_STYLE_BG_OPA, LV_OPA_TRANSP);
|
|
|
|
|
|
|
|
/*Padding styles*/
|
|
|
|
lv_style_init(&lv_style_tight);
|
|
|
|
lv_style_set_value(&lv_style_tight, LV_STYLE_PAD_LEFT, 0);
|
|
|
|
lv_style_set_value(&lv_style_tight, LV_STYLE_PAD_RIGHT, 0);
|
|
|
|
lv_style_set_value(&lv_style_tight, LV_STYLE_PAD_TOP, 0);
|
|
|
|
lv_style_set_value(&lv_style_tight, LV_STYLE_PAD_BOTTOM, 0);
|
|
|
|
lv_style_set_value(&lv_style_tight, LV_STYLE_PAD_INNER, 0);
|
|
|
|
|
|
|
|
lv_style_init(&lv_style_fit);
|
|
|
|
lv_style_set_value(&lv_style_fit, LV_STYLE_PAD_LEFT, 0);
|
|
|
|
lv_style_set_value(&lv_style_fit, LV_STYLE_PAD_RIGHT, 0);
|
|
|
|
lv_style_set_value(&lv_style_fit, LV_STYLE_PAD_TOP, 0);
|
|
|
|
lv_style_set_value(&lv_style_fit, LV_STYLE_PAD_BOTTOM, 0);
|
|
|
|
|
|
|
|
lv_mem_monitor(&mon);
|
|
|
|
printf("used: %6d (%3d %%), frag: %3d %%, biggest free: %6d\n", (int)mon.total_size - mon.free_size,
|
|
|
|
mon.used_pct,
|
|
|
|
mon.frag_pct,
|
|
|
|
(int)mon.free_biggest_size);
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void lv_style_init(lv_style_t * style)
|
|
|
|
{
|
|
|
|
style->map = NULL;
|
|
|
|
style->size = 0;
|
|
|
|
style->used_groups = 0;
|
2017-04-13 10:20:35 +02:00
|
|
|
}
|
|
|
|
|
2019-12-17 09:20:40 +01:00
|
|
|
void lv_style_dsc_init(lv_style_dsc_t * style_dsc)
|
|
|
|
{
|
|
|
|
lv_style_init(&style_dsc->local);
|
|
|
|
style_dsc->classes = NULL;
|
2019-12-22 22:40:02 +01:00
|
|
|
style_dsc->class_cnt = 0;
|
2019-12-26 01:30:20 +01:00
|
|
|
memset(&style_dsc->cache, 0x00, sizeof(lv_style_cache_t));
|
|
|
|
style_dsc->cache.enabled = 1;
|
2019-12-22 22:40:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void lv_style_dsc_add_class(lv_style_dsc_t * style_dsc, lv_style_t * class)
|
|
|
|
{
|
2019-12-22 23:19:51 +01:00
|
|
|
/* Do not allocate memory for the first class.
|
|
|
|
* It can be simply stored as a pointer.*/
|
|
|
|
if(style_dsc->class_cnt == 0) {
|
|
|
|
style_dsc->classes = (lv_style_t**)class;
|
|
|
|
style_dsc->class_cnt = 1;
|
|
|
|
} else {
|
|
|
|
|
|
|
|
lv_style_t ** new_classes;
|
|
|
|
if(style_dsc->class_cnt == 1) new_classes = lv_mem_alloc(sizeof(lv_style_t *) * (style_dsc->class_cnt + 1));
|
|
|
|
else new_classes = lv_mem_realloc(style_dsc->classes, sizeof(lv_style_t *) * (style_dsc->class_cnt + 1));
|
|
|
|
LV_ASSERT_MEM(new_classes);
|
|
|
|
if(new_classes == NULL) {
|
|
|
|
LV_LOG_WARN("lv_style_dsc_add_class: couldn't add the class");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-23 01:02:19 +01:00
|
|
|
if(style_dsc->class_cnt == 1) new_classes[0] = (lv_style_t*)style_dsc->classes;
|
2019-12-22 23:19:51 +01:00
|
|
|
new_classes[style_dsc->class_cnt] = class;
|
|
|
|
|
|
|
|
style_dsc->class_cnt++;
|
|
|
|
style_dsc->classes = new_classes;
|
2019-12-22 22:40:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void lv_style_dsc_remove_class(lv_style_dsc_t * style_dsc, lv_style_t * class)
|
|
|
|
{
|
|
|
|
if(style_dsc->class_cnt == 0) return;
|
2019-12-22 23:19:51 +01:00
|
|
|
if(style_dsc->class_cnt == 1) {
|
|
|
|
if((lv_style_t*)style_dsc->classes == class) {
|
|
|
|
style_dsc->classes = NULL;
|
|
|
|
style_dsc->class_cnt = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
lv_style_t ** new_classes = lv_mem_realloc(style_dsc->classes, sizeof(lv_style_t *) * (style_dsc->class_cnt - 1));
|
|
|
|
LV_ASSERT_MEM(new_classes);
|
|
|
|
if(new_classes == NULL) {
|
|
|
|
LV_LOG_WARN("lv_style_dsc_remove_class: couldn't remove the class");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uint8_t i,j;
|
|
|
|
for(i = 0, j = 0; i < style_dsc->class_cnt; i++) {
|
|
|
|
if(style_dsc->classes[i] == class) continue;
|
|
|
|
new_classes[j] = style_dsc->classes[i];
|
|
|
|
j++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
style_dsc->class_cnt--;
|
|
|
|
style_dsc->classes = new_classes;
|
2019-12-22 22:40:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void lv_style_dsc_reset(lv_style_dsc_t * style_dsc)
|
|
|
|
{
|
2019-12-22 23:19:51 +01:00
|
|
|
if(style_dsc->class_cnt > 1) lv_mem_free(style_dsc->classes);
|
2019-12-22 22:40:02 +01:00
|
|
|
style_dsc->classes = NULL;
|
|
|
|
style_dsc->class_cnt = 0;
|
|
|
|
lv_style_reset(&style_dsc->local);
|
2019-12-22 23:19:51 +01:00
|
|
|
}
|
|
|
|
|
2019-12-26 01:30:20 +01:00
|
|
|
|
2019-12-22 22:40:02 +01:00
|
|
|
void lv_style_reset(lv_style_t * style)
|
|
|
|
{
|
|
|
|
lv_mem_free(style->map);
|
|
|
|
style->map = NULL;
|
|
|
|
style->size = 0;
|
|
|
|
style->used_groups = 0;
|
2019-12-17 09:20:40 +01:00
|
|
|
}
|
|
|
|
|
2017-05-08 10:09:41 +02:00
|
|
|
/**
|
|
|
|
* Copy a style to an other
|
|
|
|
* @param dest pointer to the destination style
|
|
|
|
* @param src pointer to the source style
|
|
|
|
*/
|
2017-10-20 10:17:02 +02:00
|
|
|
void lv_style_copy(lv_style_t * dest, const lv_style_t * src)
|
2017-05-08 10:09:41 +02:00
|
|
|
{
|
|
|
|
memcpy(dest, src, sizeof(lv_style_t));
|
|
|
|
}
|
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
void lv_style_set_value(lv_style_t * style, lv_style_property_t prop, lv_style_value_t value)
|
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
2019-12-19 11:05:04 +01:00
|
|
|
/*The property already exists but not sure it's state is the same*/
|
2019-12-14 23:39:26 +01:00
|
|
|
if(id >= 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_found;
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
|
|
|
|
attr_found.full = *(style->map + id + 1);
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFFU;
|
|
|
|
|
|
|
|
if(attr_found.bits.state == attr_goal.bits.state) {
|
|
|
|
memcpy(style->map + id + sizeof(lv_style_property_t), &value, sizeof(lv_style_value_t));
|
|
|
|
return;
|
|
|
|
}
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
2019-12-19 11:05:04 +01:00
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
/*Add new property if not exists yet*/
|
2019-12-19 11:05:04 +01:00
|
|
|
style->size += sizeof(lv_style_property_t) + sizeof(lv_style_value_t);
|
|
|
|
style->map = lv_mem_realloc(style->map, style->size);
|
|
|
|
LV_ASSERT_MEM(style->map);
|
|
|
|
if(style == NULL) return;
|
|
|
|
|
|
|
|
memcpy(style->map + style->size - (sizeof(lv_style_property_t) + sizeof(lv_style_value_t)), &prop, sizeof(lv_style_property_t));
|
|
|
|
memcpy(style->map + style->size - sizeof(lv_style_value_t), &value, sizeof(lv_style_value_t));
|
2019-12-22 22:40:02 +01:00
|
|
|
|
|
|
|
/*Set that group is used this style*/
|
|
|
|
uint16_t group = (prop >> 4) & 0xF;
|
|
|
|
style->used_groups |= 1 << group;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void lv_style_set_color(lv_style_t * style, lv_style_property_t prop, lv_color_t color)
|
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
2019-12-19 11:05:04 +01:00
|
|
|
/*The property already exists but not sure it's state is the same*/
|
2019-12-14 23:39:26 +01:00
|
|
|
if(id >= 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_found;
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
|
|
|
|
attr_found.full = *(style->map + id + 1);
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFFU;
|
|
|
|
|
|
|
|
if(attr_found.bits.state == attr_goal.bits.state) {
|
|
|
|
memcpy(style->map + id + sizeof(lv_style_property_t), &color, sizeof(lv_color_t));
|
|
|
|
return;
|
|
|
|
}
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
2019-12-19 11:05:04 +01:00
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
/*Add new property if not exists yet*/
|
2019-12-19 11:05:04 +01:00
|
|
|
style->size += sizeof(lv_style_property_t) + sizeof(lv_color_t);
|
|
|
|
style->map = lv_mem_realloc(style->map, style->size);
|
|
|
|
LV_ASSERT_MEM(style->map);
|
|
|
|
if(style == NULL) return;
|
|
|
|
|
|
|
|
memcpy(style->map + style->size - (sizeof(lv_style_property_t) + sizeof(lv_color_t)), &prop, sizeof(lv_style_property_t));
|
|
|
|
memcpy(style->map + style->size - sizeof(lv_color_t), &color, sizeof(lv_color_t));
|
2019-12-22 22:40:02 +01:00
|
|
|
|
|
|
|
/*Set that group is used this style*/
|
|
|
|
uint16_t group = (prop >> 4) & 0xF;
|
|
|
|
style->used_groups |= 1 << group;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void lv_style_set_opa(lv_style_t * style, lv_style_property_t prop, lv_opa_t opa)
|
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
2019-12-19 11:05:04 +01:00
|
|
|
/*The property already exists but not sure it's state is the same*/
|
2019-12-14 23:39:26 +01:00
|
|
|
if(id >= 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_found;
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
|
|
|
|
attr_found.full = *(style->map + id + 1);
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFFU;
|
|
|
|
|
|
|
|
if(attr_found.bits.state == attr_goal.bits.state) {
|
|
|
|
memcpy(style->map + id + sizeof(lv_style_property_t), &opa, sizeof(lv_opa_t));
|
|
|
|
return;
|
|
|
|
}
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
2019-12-19 11:05:04 +01:00
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
/*Add new property if not exists yet*/
|
2019-12-19 11:05:04 +01:00
|
|
|
style->size += sizeof(lv_style_property_t) + sizeof(lv_opa_t);
|
|
|
|
style->map = lv_mem_realloc(style->map, style->size);
|
|
|
|
LV_ASSERT_MEM(style->map);
|
|
|
|
if(style == NULL) return;
|
2019-12-14 23:39:26 +01:00
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
memcpy(style->map + style->size - (sizeof(lv_style_property_t) + sizeof(lv_opa_t)), &prop, sizeof(lv_style_property_t));
|
|
|
|
memcpy(style->map + style->size - sizeof(lv_opa_t), &opa, sizeof(lv_opa_t));
|
2019-12-22 22:40:02 +01:00
|
|
|
|
|
|
|
/*Set that group is used this style*/
|
|
|
|
uint16_t group = (prop >> 4) & 0xF;
|
|
|
|
style->used_groups |= 1 << group;
|
2019-12-19 11:05:04 +01:00
|
|
|
}
|
2019-12-14 23:39:26 +01:00
|
|
|
|
2019-12-19 22:44:18 +01:00
|
|
|
void lv_style_set_ptr(lv_style_t * style, lv_style_property_t prop, void * p)
|
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
|
|
|
/*The property already exists but not sure it's state is the same*/
|
|
|
|
if(id >= 0) {
|
|
|
|
lv_style_attr_t attr_found;
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
|
|
|
|
attr_found.full = *(style->map + id + 1);
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFFU;
|
|
|
|
|
|
|
|
if(attr_found.bits.state == attr_goal.bits.state) {
|
|
|
|
memcpy(style->map + id + sizeof(lv_style_property_t), &p, sizeof(void *));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Add new property if not exists yet*/
|
|
|
|
style->size += sizeof(lv_style_property_t) + sizeof(void *);
|
|
|
|
style->map = lv_mem_realloc(style->map, style->size);
|
|
|
|
LV_ASSERT_MEM(style->map);
|
|
|
|
if(style == NULL) return;
|
|
|
|
|
|
|
|
memcpy(style->map + style->size - (sizeof(lv_style_property_t) + sizeof(void *)), &prop, sizeof(lv_style_property_t));
|
|
|
|
memcpy(style->map + style->size - sizeof(void *), &p, sizeof(void *));
|
2019-12-22 22:40:02 +01:00
|
|
|
|
|
|
|
/*Set that group is used this style*/
|
|
|
|
uint16_t group = (prop >> 4) & 0xF;
|
|
|
|
style->used_groups |= 1 << group;
|
2019-12-19 22:44:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
/**
|
|
|
|
* Get the a property from a style.
|
|
|
|
* Take into account the style state and return the property which matches the best.
|
|
|
|
* @param style pointer to a style where to search
|
|
|
|
* @param prop the property, might contain ORed style states too
|
|
|
|
* @param res buffer to store the result
|
|
|
|
* @return the weight of the found property (how well it fits to the style state).
|
|
|
|
* Higher number is means better fit
|
|
|
|
* -1 if the not found (`res` will be undefined)
|
|
|
|
*/
|
|
|
|
int16_t lv_style_get_value(const lv_style_t * style, lv_style_property_t prop, lv_style_value_t * res)
|
2019-12-14 23:39:26 +01:00
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
|
|
|
if(id < 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
return -1;
|
2019-12-14 23:39:26 +01:00
|
|
|
} else {
|
|
|
|
memcpy(res, &style->map[id + sizeof(lv_style_property_t)], sizeof(lv_style_value_t));
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_act;
|
|
|
|
attr_act.full = style->map[id + 1];
|
|
|
|
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFF;
|
|
|
|
|
|
|
|
return attr_act.bits.state & attr_goal.bits.state;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
int16_t lv_style_get_opa(const lv_style_t * style, lv_style_property_t prop, lv_opa_t * res)
|
2019-12-14 23:39:26 +01:00
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
|
|
|
if(id < 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
return -1;
|
2019-12-14 23:39:26 +01:00
|
|
|
} else {
|
|
|
|
memcpy(res, &style->map[id + sizeof(lv_style_property_t)], sizeof(lv_opa_t));
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_act;
|
|
|
|
attr_act.full = style->map[id + 1];
|
|
|
|
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFF;
|
|
|
|
|
|
|
|
return attr_act.bits.state & attr_goal.bits.state;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
int16_t lv_style_get_color(const lv_style_t * style, lv_style_property_t prop, lv_color_t * res)
|
2019-12-14 23:39:26 +01:00
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
|
|
|
if(id < 0) {
|
2019-12-19 11:05:04 +01:00
|
|
|
return -1;
|
2019-12-14 23:39:26 +01:00
|
|
|
} else {
|
|
|
|
memcpy(res, &style->map[id + sizeof(lv_style_property_t)], sizeof(lv_color_t));
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_act;
|
|
|
|
attr_act.full = style->map[id + 1];
|
|
|
|
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFF;
|
|
|
|
|
2019-12-19 22:44:18 +01:00
|
|
|
return attr_act.bits.state & attr_goal.bits.state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int16_t lv_style_get_ptr(const lv_style_t * style, lv_style_property_t prop, void ** res)
|
|
|
|
{
|
|
|
|
int32_t id = get_property_index(style, prop);
|
|
|
|
if(id < 0) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
memcpy(res, &style->map[id + sizeof(lv_style_property_t)], sizeof(void*));
|
|
|
|
lv_style_attr_t attr_act;
|
|
|
|
attr_act.full = style->map[id + 1];
|
|
|
|
|
|
|
|
lv_style_attr_t attr_goal;
|
|
|
|
attr_goal.full = (prop >> 8) & 0xFF;
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
return attr_act.bits.state & attr_goal.bits.state;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-26 13:49:23 +02:00
|
|
|
/**
|
|
|
|
* Mix two styles according to a given ratio
|
2018-10-05 17:22:49 +02:00
|
|
|
* @param start start style
|
2018-08-26 13:49:23 +02:00
|
|
|
* @param end end style
|
|
|
|
* @param res store the result style here
|
|
|
|
* @param ratio the ratio of mix [0..256]; 0: `start` style; 256: `end` style
|
|
|
|
*/
|
2019-06-06 06:05:40 +02:00
|
|
|
void lv_style_mix(const lv_style_t * start, const lv_style_t * end, lv_style_t * res, uint16_t ratio)
|
2018-08-26 13:49:23 +02:00
|
|
|
{
|
2019-12-14 23:39:26 +01:00
|
|
|
// STYLE_ATTR_MIX(body.opa, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.radius, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.border.width, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.border.opa, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.shadow.width, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.shadow.offset.x, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.shadow.offset.y, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.shadow.spread, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.padding.left, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.padding.right, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.padding.top, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.padding.bottom, ratio);
|
|
|
|
// STYLE_ATTR_MIX(body.padding.inner, ratio);
|
|
|
|
// STYLE_ATTR_MIX(text.line_space, ratio);
|
|
|
|
// STYLE_ATTR_MIX(text.letter_space, ratio);
|
|
|
|
// STYLE_ATTR_MIX(text.opa, ratio);
|
|
|
|
// STYLE_ATTR_MIX(line.width, ratio);
|
|
|
|
// STYLE_ATTR_MIX(line.opa, ratio);
|
|
|
|
// STYLE_ATTR_MIX(image.intense, ratio);
|
|
|
|
// STYLE_ATTR_MIX(image.opa, ratio);
|
|
|
|
//
|
|
|
|
// lv_opa_t opa = ratio == STYLE_MIX_MAX ? LV_OPA_COVER : ratio;
|
|
|
|
//
|
|
|
|
// res->body.main_color = lv_color_mix(end->body.main_color, start->body.main_color, opa);
|
|
|
|
// res->body.grad_color = lv_color_mix(end->body.grad_color, start->body.grad_color, opa);
|
|
|
|
// res->body.border.color = lv_color_mix(end->body.border.color, start->body.border.color, opa);
|
|
|
|
// res->body.shadow.color = lv_color_mix(end->body.shadow.color, start->body.shadow.color, opa);
|
|
|
|
// res->text.color = lv_color_mix(end->text.color, start->text.color, opa);
|
|
|
|
// res->image.color = lv_color_mix(end->image.color, start->image.color, opa);
|
|
|
|
// res->line.color = lv_color_mix(end->line.color, start->line.color, opa);
|
|
|
|
//
|
|
|
|
// if(ratio < (STYLE_MIX_MAX >> 1)) {
|
|
|
|
// res->body.border.part = start->body.border.part;
|
|
|
|
// res->glass = start->glass;
|
|
|
|
// res->text.font = start->text.font;
|
|
|
|
// res->line.rounded = start->line.rounded;
|
|
|
|
// } else {
|
|
|
|
// res->body.border.part = end->body.border.part;
|
|
|
|
// res->glass = end->glass;
|
|
|
|
// res->text.font = end->text.font;
|
|
|
|
// res->line.rounded = end->line.rounded;
|
|
|
|
// }
|
2018-08-26 13:49:23 +02:00
|
|
|
}
|
|
|
|
|
2019-03-07 00:05:16 +01:00
|
|
|
#if LV_USE_ANIMATION
|
2018-06-15 09:29:10 +02:00
|
|
|
|
2019-05-15 18:27:26 +02:00
|
|
|
void lv_style_anim_init(lv_anim_t * a)
|
2017-07-28 13:57:56 +02:00
|
|
|
{
|
2019-05-15 18:27:26 +02:00
|
|
|
lv_anim_init(a);
|
2019-06-06 06:05:40 +02:00
|
|
|
a->start = 0;
|
|
|
|
a->end = STYLE_MIX_MAX;
|
2019-06-12 23:10:54 +02:00
|
|
|
a->exec_cb = (lv_anim_exec_xcb_t)style_animator;
|
2019-06-06 06:05:40 +02:00
|
|
|
a->path_cb = lv_anim_path_linear;
|
|
|
|
a->ready_cb = style_animation_common_end_cb;
|
|
|
|
|
|
|
|
lv_style_anim_dsc_t * dsc;
|
|
|
|
dsc = lv_mem_alloc(sizeof(lv_style_anim_dsc_t));
|
2019-09-24 23:14:17 +02:00
|
|
|
LV_ASSERT_MEM(dsc);
|
2019-06-06 06:05:40 +02:00
|
|
|
if(dsc == NULL) return;
|
|
|
|
dsc->ready_cb = NULL;
|
|
|
|
dsc->style_anim = NULL;
|
|
|
|
lv_style_copy(&dsc->style_start, &lv_style_plain);
|
|
|
|
lv_style_copy(&dsc->style_end, &lv_style_plain);
|
|
|
|
|
|
|
|
a->var = (void *)dsc;
|
2017-07-28 13:57:56 +02:00
|
|
|
}
|
2018-06-15 09:29:10 +02:00
|
|
|
|
2019-05-15 18:27:26 +02:00
|
|
|
void lv_style_anim_set_styles(lv_anim_t * a, lv_style_t * to_anim, const lv_style_t * start, const lv_style_t * end)
|
|
|
|
{
|
|
|
|
|
|
|
|
lv_style_anim_dsc_t * dsc = a->var;
|
2019-06-06 06:05:40 +02:00
|
|
|
dsc->style_anim = to_anim;
|
2019-05-15 18:27:26 +02:00
|
|
|
memcpy(&dsc->style_start, start, sizeof(lv_style_t));
|
|
|
|
memcpy(&dsc->style_end, end, sizeof(lv_style_t));
|
|
|
|
memcpy(dsc->style_anim, start, sizeof(lv_style_t));
|
|
|
|
}
|
2017-11-27 17:48:54 +01:00
|
|
|
#endif
|
2017-04-13 10:20:35 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC FUNCTIONS
|
|
|
|
**********************/
|
2019-12-14 23:39:26 +01:00
|
|
|
|
|
|
|
static inline int32_t get_property_index(const lv_style_t * style, lv_style_property_t prop)
|
|
|
|
{
|
2019-12-22 22:40:02 +01:00
|
|
|
static uint32_t stat[256];
|
|
|
|
|
|
|
|
|
|
|
|
static uint32_t s = 0;
|
|
|
|
|
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
uint8_t id_to_find = prop & 0xFF;
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr;
|
|
|
|
attr.full = (prop >> 8) & 0xFF;
|
|
|
|
|
2019-12-22 22:40:02 +01:00
|
|
|
stat[id_to_find]++;
|
|
|
|
|
|
|
|
// if(s > 1000) {
|
|
|
|
// printf("\n\n");
|
|
|
|
// s = 0;
|
|
|
|
// uint32_t i;
|
|
|
|
// for(i = 0; i < 256; i++) {
|
|
|
|
// if(stat[i] == 0) continue;
|
|
|
|
// printf("%02x;%d;\n", i, stat[i]);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
int16_t weight = -1;
|
|
|
|
int16_t id_guess = -1;
|
2019-12-14 23:39:26 +01:00
|
|
|
|
2019-12-22 22:40:02 +01:00
|
|
|
|
|
|
|
uint16_t group = (id_to_find >> 4) & 0xF;
|
|
|
|
if((style->used_groups & (1 << group)) == 0) return id_guess;
|
|
|
|
|
2019-12-14 23:39:26 +01:00
|
|
|
size_t i = 0;
|
|
|
|
while(i < style->size) {
|
2019-12-22 22:40:02 +01:00
|
|
|
// s++;
|
|
|
|
|
|
|
|
// printf("style search:%d\n", s);
|
2019-12-19 11:05:04 +01:00
|
|
|
lv_style_attr_t attr_act;
|
|
|
|
attr_act.full = style->map[i + 1];
|
2019-12-14 23:39:26 +01:00
|
|
|
if(style->map[i] == id_to_find) {
|
2019-12-19 11:05:04 +01:00
|
|
|
/*If there the state has perfectly match return this property*/
|
|
|
|
if(attr_act.bits.state == attr.bits.state) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
/* Be sure the property not specifies other state the the requested.
|
|
|
|
* E.g. For HOVER+PRESS, HOVER only is OK, but HOVER+FOCUS not*/
|
|
|
|
else if((attr_act.bits.state & (~attr.bits.state)) == 0) {
|
|
|
|
/* Use this property if it describes better the requested state than the current candidate.
|
|
|
|
* E.g. for HOVER+FOCUS+PRESS prefer HOVER+FOCUS over FOCUS*/
|
|
|
|
if(attr_act.bits.state > weight) {
|
|
|
|
weight = attr_act.bits.state;
|
|
|
|
id_guess = i;
|
|
|
|
}
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
2019-12-19 11:05:04 +01:00
|
|
|
|
|
|
|
/*Go to the next property*/
|
|
|
|
if((style->map[i] & 0xF) < LV_STYLE_ID_COLOR) i+= sizeof(lv_style_value_t);
|
|
|
|
else if((style->map[i] & 0xF) < LV_STYLE_ID_OPA) i+= sizeof(lv_color_t);
|
|
|
|
else if((style->map[i] & 0xF) < LV_STYLE_ID_PTR) i+= sizeof(lv_opa_t);
|
|
|
|
else i+= sizeof(void*);
|
|
|
|
|
|
|
|
i += sizeof(lv_style_property_t);
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
|
2019-12-19 11:05:04 +01:00
|
|
|
return id_guess;
|
2019-12-14 23:39:26 +01:00
|
|
|
}
|
|
|
|
|
2019-03-07 00:05:16 +01:00
|
|
|
#if LV_USE_ANIMATION
|
2017-07-28 13:57:56 +02:00
|
|
|
/**
|
|
|
|
* Used by the style animations to set the values of a style according to start and end style.
|
|
|
|
* @param dsc the 'animated variable' set by lv_style_anim_create()
|
2019-05-18 18:14:21 -07:00
|
|
|
* @param val the current state of the animation between 0 and LV_ANIM_RESOLUTION
|
2017-07-28 13:57:56 +02:00
|
|
|
*/
|
2019-05-20 18:38:24 -07:00
|
|
|
static void style_animator(lv_style_anim_dsc_t * dsc, lv_anim_value_t val)
|
2017-07-28 13:57:56 +02:00
|
|
|
{
|
2017-10-30 17:11:56 +01:00
|
|
|
const lv_style_t * start = &dsc->style_start;
|
2019-04-04 07:15:40 +02:00
|
|
|
const lv_style_t * end = &dsc->style_end;
|
|
|
|
lv_style_t * act = dsc->style_anim;
|
2017-07-28 13:57:56 +02:00
|
|
|
|
2018-08-26 13:49:23 +02:00
|
|
|
lv_style_mix(start, end, act, val);
|
2017-07-28 13:57:56 +02:00
|
|
|
|
2017-11-19 20:45:40 +01:00
|
|
|
lv_obj_report_style_mod(dsc->style_anim);
|
2017-12-03 00:36:31 +01:00
|
|
|
}
|
2017-10-30 17:11:56 +01:00
|
|
|
|
2017-12-03 00:36:31 +01:00
|
|
|
/**
|
|
|
|
* Called when a style animation is ready
|
|
|
|
* It called the user defined call back and free the allocated memories
|
2019-04-22 08:45:07 +02:00
|
|
|
* @param a pointer to the animation
|
2017-12-03 00:36:31 +01:00
|
|
|
*/
|
2019-04-22 08:45:07 +02:00
|
|
|
static void style_animation_common_end_cb(lv_anim_t * a)
|
2017-12-03 00:36:31 +01:00
|
|
|
{
|
|
|
|
|
2019-06-06 06:05:40 +02:00
|
|
|
(void)a; /*Unused*/
|
2019-04-22 08:45:07 +02:00
|
|
|
lv_style_anim_dsc_t * dsc = a->var; /*To avoid casting*/
|
|
|
|
|
|
|
|
if(dsc->ready_cb) dsc->ready_cb(a);
|
2017-12-03 00:36:31 +01:00
|
|
|
|
|
|
|
lv_mem_free(dsc);
|
2017-07-28 13:57:56 +02:00
|
|
|
}
|
2017-12-03 00:36:31 +01:00
|
|
|
|
2017-11-27 17:48:54 +01:00
|
|
|
#endif
|