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

270 lines
6.9 KiB
C
Raw Normal View History

2017-11-23 20:42:14 +01:00
/**
2017-11-23 21:28:36 +01:00
* @file lv_font.c
2018-06-19 09:49:58 +02:00
*
2017-11-23 20:42:14 +01:00
*/
/*********************
* INCLUDES
*********************/
#include <stddef.h>
#include "lv_font.h"
2018-07-25 17:57:08 +02:00
#include "lv_log.h"
2017-11-23 20:42:14 +01:00
/*********************
* DEFINES
*********************/
/**********************
* TYPEDEFS
**********************/
/**********************
* STATIC PROTOTYPES
**********************/
/**********************
* STATIC VARIABLES
**********************/
/**********************
* GLOBAL PROTOTYPES
**********************/
/**********************
* MACROS
**********************/
/**********************
* GLOBAL FUNCTIONS
**********************/
/**
* Initialize the fonts
2017-11-23 20:42:14 +01:00
*/
2017-11-23 21:28:36 +01:00
void lv_font_init(void)
2017-11-23 20:42:14 +01:00
{
2018-10-05 17:22:49 +02:00
lv_font_builtin_init();
2017-11-23 20:42:14 +01:00
}
/**
2018-12-13 15:18:06 +01:00
* Add a font to an other to extend the character set.
* @param child the font to add
* @param parent this font will be extended. Using it later will contain the characters from `child`
2017-11-23 20:42:14 +01:00
*/
2018-06-19 09:49:58 +02:00
void lv_font_add(lv_font_t * child, lv_font_t * parent)
2017-11-23 20:42:14 +01:00
{
if(parent == NULL) return;
while(parent->next_page != NULL) {
parent = parent->next_page; /*Got to the last page and add the new font there*/
}
parent->next_page = child;
}
2018-12-13 15:18:06 +01:00
/**
* Remove a font from a character set.
* @param child the font to remove
* @param parent remove `child` from here
*/
void lv_font_remove(lv_font_t * child, lv_font_t * parent)
{
if(parent == NULL) return;
if(child == NULL) return;
while(parent->next_page != child) {
parent = parent->next_page; /*Got to the last page and add the new font there*/
}
parent->next_page = child->next_page;
}
2018-06-22 23:32:21 +02:00
/**
* Tells if font which contains `letter` is monospace or not
* @param font_p point to font
* @param letter an UNICODE character code
* @return true: the letter is monospace; false not monospace
*/
bool lv_font_is_monospace(const lv_font_t * font_p, uint32_t letter)
{
2018-10-05 17:22:49 +02:00
const lv_font_t * font_i = font_p;
int16_t w;
while(font_i != NULL) {
w = font_i->get_width(font_i, letter);
if(w >= 0) {
/*Glyph found*/
if(font_i->monospace) return true;
return false;
}
2018-06-22 23:32:21 +02:00
2018-10-05 17:22:49 +02:00
font_i = font_i->next_page;
}
2018-06-22 23:32:21 +02:00
2018-10-05 17:22:49 +02:00
return 0;
2018-06-22 23:32:21 +02:00
}
2017-11-23 20:42:14 +01:00
/**
* Return with the bitmap of a font.
* @param font_p pointer to a font
2018-06-22 23:32:21 +02:00
* @param letter an UNICODE character code
2017-11-23 20:42:14 +01:00
* @return pointer to the bitmap of the letter
*/
2017-11-23 21:28:36 +01:00
const uint8_t * lv_font_get_bitmap(const lv_font_t * font_p, uint32_t letter)
2017-11-23 20:42:14 +01:00
{
2017-11-23 21:28:36 +01:00
const lv_font_t * font_i = font_p;
2017-11-23 20:42:14 +01:00
while(font_i != NULL) {
const uint8_t * bitmap = font_i->get_bitmap(font_i, letter);
if(bitmap) return bitmap;
2017-11-23 20:42:14 +01:00
font_i = font_i->next_page;
}
return NULL;
}
/**
2018-06-22 23:32:21 +02:00
* Get the width of a letter in a font. If `monospace` is set then return with it.
2017-11-23 20:42:14 +01:00
* @param font_p pointer to a font
* @param letter an UNICODE character code
2017-11-23 20:42:14 +01:00
* @return the width of a letter
*/
2017-11-23 21:28:36 +01:00
uint8_t lv_font_get_width(const lv_font_t * font_p, uint32_t letter)
2018-06-22 23:32:21 +02:00
{
const lv_font_t * font_i = font_p;
int16_t w;
while(font_i != NULL) {
w = font_i->get_width(font_i, letter);
if(w >= 0) {
2018-10-05 17:22:49 +02:00
/*Glyph found*/
uint8_t m = font_i->monospace;
if(m) w = m;
return w;
2018-06-22 23:32:21 +02:00
}
font_i = font_i->next_page;
}
return 0;
2018-07-25 17:57:08 +02:00
2018-06-22 23:32:21 +02:00
}
/**
* Get the width of the letter without overwriting it with the `monospace` attribute
* @param font_p pointer to a font
* @param letter an UNICODE character code
* @return the width of a letter
*/
uint8_t lv_font_get_real_width(const lv_font_t * font_p, uint32_t letter)
2017-11-23 20:42:14 +01:00
{
2017-11-23 21:28:36 +01:00
const lv_font_t * font_i = font_p;
int16_t w;
2017-11-23 20:42:14 +01:00
while(font_i != NULL) {
w = font_i->get_width(font_i, letter);
if(w >= 0) return w;
2017-11-23 20:42:14 +01:00
font_i = font_i->next_page;
}
return 0;
}
2018-02-09 12:40:00 +01:00
/**
* Get the bit-per-pixel of font
* @param font pointer to font
* @param letter a letter from font (font extensions can have different bpp)
* @return bpp of the font (or font extension)
*/
uint8_t lv_font_get_bpp(const lv_font_t * font, uint32_t letter)
2018-02-09 12:40:00 +01:00
{
const lv_font_t * font_i = font;
while(font_i != NULL) {
2018-06-19 09:49:58 +02:00
if(letter >= font_i->unicode_first && letter <= font_i->unicode_last) {
return font_i->bpp;
}
font_i = font_i->next_page;
2018-02-09 12:40:00 +01:00
}
2018-06-19 09:49:58 +02:00
return 0;
2018-02-09 12:40:00 +01:00
}
/**
* Generic bitmap get function used in 'font->get_bitmap' when the font contains all characters in the range
* @param font pointer to font
* @param unicode_letter an unicode letter which bitmap should be get
* @return pointer to the bitmap or NULL if not found
*/
const uint8_t * lv_font_get_bitmap_continuous(const lv_font_t * font, uint32_t unicode_letter)
{
/*Check the range*/
if(unicode_letter < font->unicode_first || unicode_letter > font->unicode_last) return NULL;
2018-02-09 12:40:00 +01:00
uint32_t index = (unicode_letter - font->unicode_first);
return &font->glyph_bitmap[font->glyph_dsc[index].glyph_index];
}
/**
* Generic bitmap get function used in 'font->get_bitmap' when the font NOT contains all characters in the range (sparse)
* @param font pointer to font
* @param unicode_letter an unicode letter which bitmap should be get
* @return pointer to the bitmap or NULL if not found
*/
const uint8_t * lv_font_get_bitmap_sparse(const lv_font_t * font, uint32_t unicode_letter)
{
/*Check the range*/
if(unicode_letter < font->unicode_first || unicode_letter > font->unicode_last) return NULL;
2018-02-09 12:40:00 +01:00
uint32_t i;
for(i = 0; font->unicode_list[i] != 0; i++) {
if(font->unicode_list[i] == unicode_letter) {
return &font->glyph_bitmap[font->glyph_dsc[i].glyph_index];
}
}
return NULL;
}
/**
* Generic glyph width get function used in 'font->get_width' when the font contains all characters in the range
* @param font pointer to font
* @param unicode_letter an unicode letter which width should be get
* @return width of the gylph or -1 if not found
*/
int16_t lv_font_get_width_continuous(const lv_font_t * font, uint32_t unicode_letter)
{
/*Check the range*/
if(unicode_letter < font->unicode_first || unicode_letter > font->unicode_last) {
return -1;
}
uint32_t index = (unicode_letter - font->unicode_first);
return font->glyph_dsc[index].w_px;
}
/**
* Generic glyph width get function used in 'font->get_bitmap' when the font NOT contains all characters in the range (sparse)
* @param font pointer to font
* @param unicode_letter an unicode letter which width should be get
* @return width of the glyph or -1 if not found
*/
int16_t lv_font_get_width_sparse(const lv_font_t * font, uint32_t unicode_letter)
{
/*Check the range*/
if(unicode_letter < font->unicode_first || unicode_letter > font->unicode_last) return -1;
uint32_t i;
for(i = 0; font->unicode_list[i] != 0; i++) {
if(font->unicode_list[i] == unicode_letter) {
return font->glyph_dsc[i].w_px;
}
}
return -1;
}
2017-11-23 20:42:14 +01:00
/**********************
* STATIC FUNCTIONS
**********************/