mirror of
https://github.com/lvgl/lvgl.git
synced 2024-11-30 13:14:37 +08:00
194 lines
5.3 KiB
C
194 lines
5.3 KiB
C
/**
|
|
* @file font.c
|
|
*
|
|
*/
|
|
|
|
/*********************
|
|
* INCLUDES
|
|
*********************/
|
|
#include "text.h"
|
|
#include "misc/math/math_base.h"
|
|
|
|
/*********************
|
|
* DEFINES
|
|
*********************/
|
|
#define TXT_NO_BREAK_FOUND UINT16_MAX
|
|
|
|
/**********************
|
|
* TYPEDEFS
|
|
**********************/
|
|
|
|
/**********************
|
|
* STATIC PROTOTYPES
|
|
**********************/
|
|
static bool txt_is_break_char(char letter);
|
|
|
|
/**********************
|
|
* STATIC VARIABLES
|
|
**********************/
|
|
|
|
/**********************
|
|
* MACROS
|
|
**********************/
|
|
|
|
/**********************
|
|
* GLOBAL FUNCTIONS
|
|
**********************/
|
|
|
|
void txt_get_size(point_t * size_res, const char * text, const font_t * font,
|
|
uint16_t letter_space, uint16_t line_space, cord_t max_width)
|
|
{
|
|
uint32_t line_start = 0;
|
|
uint32_t new_line_start = 0;
|
|
cord_t act_line_length;
|
|
uint8_t letter_height = font_get_height(font);
|
|
size_res->x = 0;
|
|
size_res->y = 0;
|
|
|
|
/*Calc. the height and longest line*/
|
|
while (text[line_start] != '\0')
|
|
{
|
|
new_line_start += txt_get_next_line(&text[line_start], font, letter_space, max_width);
|
|
size_res->y += letter_height;
|
|
size_res->y += line_space;
|
|
|
|
/*Calculate the the longest line*/
|
|
act_line_length = txt_get_width(&text[line_start], new_line_start - line_start,
|
|
font, letter_space);
|
|
|
|
size_res->x = max(act_line_length, size_res->x);
|
|
line_start = new_line_start;
|
|
}
|
|
|
|
if(line_start != 0 && (text[line_start - 1] == '\n' || text[line_start - 1] == '\r')) {
|
|
size_res->y += letter_height + line_space;
|
|
}
|
|
|
|
/*Correction with the last line space or set the height manually if the text is empty*/
|
|
if(size_res->y == 0) size_res->y = letter_height;
|
|
else size_res->y -= line_space;
|
|
|
|
}
|
|
|
|
/**
|
|
* Get the next line of text. Check line length and break chars too.
|
|
* @param txt a '\0' terminated string
|
|
* @param font_p pointer to a font
|
|
* @param letter_space letter space
|
|
* @param max_l max line length
|
|
* @return the index of the first char of the new line
|
|
*/
|
|
uint16_t txt_get_next_line(const char * txt, const font_t * font_p,
|
|
uint16_t letter_space, cord_t max_l)
|
|
{
|
|
uint32_t i = 0;
|
|
cord_t act_l = 0;
|
|
uint16_t last_break = TXT_NO_BREAK_FOUND;
|
|
|
|
while(txt[i] != '\0') {
|
|
/*Check for new line chars*/
|
|
if(txt[i] == '\n' || txt[i] == '\r') {
|
|
/*Handle \n\r and \r\n as well*/
|
|
if(txt[i] == '\n' && txt[i + 1] == '\r') {
|
|
i++;
|
|
} else if(txt[i] == '\r' && txt[i + 1] == '\n') {
|
|
i++;
|
|
}
|
|
return i+1; /*Return with the first letter of the next line*/
|
|
|
|
} else { /*Check the actual length*/
|
|
act_l += font_get_width(font_p, txt[i]);
|
|
|
|
/*If the txt is too long then finish, this is the line end*/
|
|
if(act_l > max_l) {
|
|
/*If already a break character is found, then break there*/
|
|
if(last_break != TXT_NO_BREAK_FOUND && txt_is_break_char(txt[i]) == false) {
|
|
i = last_break;
|
|
}
|
|
|
|
while(txt[i] == ' ') i++;
|
|
|
|
/* Do not let to return without doing nothing.
|
|
* Find at least one character */
|
|
if(i == 0) i++;
|
|
|
|
return i;
|
|
}
|
|
/*If this char still can fit to this line then check if
|
|
* txt can be broken here later */
|
|
else if(txt_is_break_char(txt[i])) {
|
|
last_break = i;
|
|
last_break++;/*Go to the next char, the break char stays in this line*/
|
|
}
|
|
}
|
|
|
|
act_l += letter_space;
|
|
i++;
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
/**
|
|
* Give the length of a text with a given font
|
|
* @param txt a '\0' terminate string
|
|
* @param char_num number of characters in 'txt'
|
|
* @param font_p pointer to a font
|
|
* @param letter_space letter sapce
|
|
* @return length of a char_num long text
|
|
*/
|
|
cord_t txt_get_width(const char * txt, uint16_t char_num,
|
|
const font_t * font_p, uint16_t letter_space)
|
|
{
|
|
uint16_t i;
|
|
cord_t len = 0;
|
|
|
|
if(char_num != 0) {
|
|
for(i = 0; i < char_num; i++) {
|
|
len += font_get_width(font_p, txt[i]);
|
|
len += letter_space;
|
|
}
|
|
|
|
/*Trim closing spaces */
|
|
for(i = char_num - 1; i > 0; i--) {
|
|
if(txt[i] == ' ') {
|
|
len -= font_get_width(font_p, txt[i]);
|
|
len -= letter_space;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*Correct the last letter space,
|
|
* because thee is no letter space after the last char*/
|
|
len -= letter_space;
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
/**********************
|
|
* STATIC FUNCTIONS
|
|
**********************/
|
|
|
|
/**
|
|
* Test if char is break char or not (a text can broken here or not)
|
|
* @param letter a letter
|
|
* @return false: 'letter' is not break char
|
|
*/
|
|
static bool txt_is_break_char(char letter)
|
|
{
|
|
uint8_t i;
|
|
bool ret = false;
|
|
|
|
/*Compare the letter to TXT_BREAK_CHARS*/
|
|
for(i = 0; LV_TXT_BREAK_CHARS[i] != '\0'; i++) {
|
|
if(letter == LV_TXT_BREAK_CHARS[i]) {
|
|
ret = true; /*If match then it is break char*/
|
|
break;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|