Files
lvgl/src/others/xml/lv_xml_style.c
2024-12-05 10:14:39 +01:00

203 lines
6.8 KiB
C

/**
* @file lv_xml_style.c
*
*/
/*********************
* INCLUDES
*********************/
#include "../../lvgl.h"
#if LV_USE_XML
#include "lv_xml_base_types.h"
#include "lv_xml_parser.h"
#include "lv_xml_style.h"
#include "lv_xml_utils.h"
#include "lv_xml_component_private.h"
#include <string.h>
/*********************
* DEFINES
*********************/
#ifdef _MSC_VER
#define strtok_r strtok_s // Use strtok_s as an equivalent to strtok_r in Visual Studio
#endif
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
/**********************
* STATIC VARIABLES
**********************/
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
lv_state_t lv_xml_style_state_string_to_enum_value(const char * txt)
{
if(lv_streq("default", txt)) return LV_STATE_DEFAULT;
if(lv_streq("pressed", txt)) return LV_STATE_PRESSED;
if(lv_streq("checked", txt)) return LV_STATE_CHECKED;
if(lv_streq("scrolled", txt)) return LV_STATE_SCROLLED;
return 0; /*Return 0 in lack of a better option. */
}
lv_part_t lv_xml_style_part_string_to_enum_value(const char * txt)
{
if(lv_streq("main", txt)) return LV_PART_MAIN;
if(lv_streq("scrollbar", txt)) return LV_PART_SCROLLBAR;
if(lv_streq("indicator", txt)) return LV_PART_INDICATOR;
if(lv_streq("knob", txt)) return LV_PART_KNOB;
return 0; /*Return 0 in lack of a better option. */
}
void lv_xml_style_register(lv_xml_component_ctx_t * ctx, const char ** attrs)
{
const char * style_name = lv_xml_get_value_of(attrs, "name");
if(style_name == NULL) {
LV_LOG_WARN("'name' is missing from a style");
return;
}
lv_xml_style_t * xml_style = lv_ll_ins_tail(&ctx->style_ll);
lv_style_t * style = &xml_style->style;
lv_style_init(style);
xml_style->name = lv_strdup(style_name);
size_t long_name_len = lv_strlen(ctx->name) + 1 + lv_strlen(style_name) + 1;
xml_style->long_name = lv_malloc(long_name_len);
lv_snprintf((char *)xml_style->long_name, long_name_len, "%s.%s", ctx->name, style_name); /*E.g. my_button.style1*/
for(int i = 0; attrs[i]; i += 2) {
const char * name = attrs[i];
const char * value = attrs[i + 1];
if(lv_streq(name, "name")) continue;
if(lv_streq(name, "help")) continue;
if(value[0] == '#') {
const char * value_clean = &value[1];
lv_xml_const_t * c;
LV_LL_READ(&ctx->const_ll, c) {
if(lv_streq(c->name, value_clean)) {
value = c->value;
break;
}
}
}
if(lv_streq(name, "width")) lv_style_set_width(style, lv_xml_to_size(value));
else if(lv_streq(name, "height")) lv_style_set_height(style, lv_xml_to_size(value));
else if(lv_streq(name, "radius")) lv_style_set_radius(style, lv_xml_atoi(value));
else if(lv_streq(name, "bg_opa")) lv_style_set_bg_opa(style, lv_xml_atoi(value));
else if(lv_streq(name, "bg_color")) lv_style_set_bg_color(style, lv_xml_to_color(value));
else if(lv_streq(name, "border_color")) lv_style_set_border_color(style, lv_xml_to_color(value));
else if(lv_streq(name, "border_width")) lv_style_set_border_width(style, lv_xml_atoi(value));
else if(lv_streq(name, "border_opa")) lv_style_set_border_opa(style, lv_xml_atoi(value));
else if(lv_streq(name, "text_color")) lv_style_set_text_color(style, lv_xml_to_color(value));
else if(lv_streq(name, "text_font")) lv_style_set_text_font(style, lv_xml_get_font(value));
else if(lv_streq(name, "bg_image_src")) lv_style_set_bg_image_src(style, lv_xml_get_image(value));
else if(lv_streq(name, "bg_image_tiled")) lv_style_set_bg_image_tiled(style, lv_xml_to_bool(value));
else {
LV_LOG_WARN("%s style property is not supported", name);
}
}
}
const char * lv_xml_style_string_process(char * txt, lv_style_selector_t * selector)
{
*selector = 0;
char * style_name = lv_xml_split_str(&txt, ':');
char * selector_str = lv_xml_split_str(&txt, ':');
while(selector_str != NULL) {
/* Handle different states and parts */
*selector |= lv_xml_style_state_string_to_enum_value(selector_str);
*selector |= lv_xml_style_part_string_to_enum_value(selector_str);
/* Move to the next token */
selector_str = lv_xml_split_str(&txt, ':');
}
return style_name;
}
void lv_xml_style_add_to_obj(lv_xml_parser_state_t * state, lv_obj_t * obj, const char * text)
{
char * str = lv_strdup(text);
char * str_ori = str;
/* Split the string based on space and colons */
char * onestyle_str = lv_xml_split_str(&str, ' ');
while(onestyle_str != NULL) {
/* Parse the parts and states after the space */
lv_style_selector_t selector = 0;
const char * style_name = lv_xml_style_string_process(onestyle_str, &selector);
if(style_name != NULL) {
lv_xml_style_t * xml_style = NULL;
/*Resolve parameters or just find the style*/
if(style_name[0] == '$') {
/*E.g. `$pr_style` style name means use the value
*coming from the parent's `pr_style` named attribute*/
const char * name_clean = &style_name[1];
const char * parent_style_name = lv_xml_get_value_of(state->parent_attrs, name_clean);
if(parent_style_name) {
xml_style = lv_xml_get_style_by_name(state->parent_ctx, parent_style_name);
}
}
else {
xml_style = lv_xml_get_style_by_name(&state->ctx, style_name);
}
if(xml_style) {
/* Apply with the selector */
lv_obj_add_style(obj, &xml_style->style, selector);
}
}
onestyle_str = lv_xml_split_str(&str, ' ');
}
lv_free(str_ori);
}
lv_xml_style_t * lv_xml_get_style_by_name(lv_xml_component_ctx_t * ctx, const char * style_name_raw)
{
const char * style_name = strrchr(style_name_raw, '.');
if(style_name) {
char component_name[256];
size_t len = (size_t)(style_name - style_name_raw);
lv_memcpy(component_name, style_name_raw, len);
component_name[len] = '\0';
ctx = lv_xml_component_get_ctx(component_name);
style_name++; /*Skip the dot*/
}
else {
style_name = style_name_raw;
}
lv_xml_style_t * xml_style;
LV_LL_READ(&ctx->style_ll, xml_style) {
if(lv_streq(xml_style->name, style_name)) return xml_style;
}
LV_LOG_WARN("No style found with %s name", style_name_raw);
return NULL;
}
/**********************
* STATIC FUNCTIONS
**********************/
#endif /* LV_USE_XML */