2009-10-16 08:03:50 +08:00
|
|
|
/*
|
|
|
|
* File : font.c
|
|
|
|
* This file is part of RT-Thread RTOS
|
|
|
|
* COPYRIGHT (C) 2006 - 2009, RT-Thread Development Team
|
|
|
|
*
|
|
|
|
* The license and distribution terms for this file may be
|
|
|
|
* found in the file LICENSE in this distribution or at
|
|
|
|
* http://www.rt-thread.org/license/LICENSE
|
|
|
|
*
|
|
|
|
* Change Logs:
|
|
|
|
* Date Author Notes
|
|
|
|
* 2009-10-16 Bernard first version
|
|
|
|
*/
|
|
|
|
#include <rtgui/font.h>
|
|
|
|
#include <rtgui/dc.h>
|
|
|
|
|
|
|
|
static rtgui_list_t _rtgui_font_list;
|
|
|
|
static struct rtgui_font* rtgui_default_font;
|
|
|
|
|
|
|
|
extern struct rtgui_font rtgui_font_asc16;
|
2010-01-21 18:05:06 +08:00
|
|
|
extern struct rtgui_font rtgui_font_arial16;
|
2009-10-16 08:03:50 +08:00
|
|
|
extern struct rtgui_font rtgui_font_asc12;
|
2010-01-21 18:05:06 +08:00
|
|
|
extern struct rtgui_font rtgui_font_arial12;
|
2009-10-16 08:03:50 +08:00
|
|
|
#ifdef RTGUI_USING_FONTHZ
|
|
|
|
extern struct rtgui_font rtgui_font_hz16;
|
|
|
|
extern struct rtgui_font rtgui_font_hz12;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void rtgui_font_system_init()
|
|
|
|
{
|
|
|
|
rtgui_list_init(&(_rtgui_font_list));
|
|
|
|
|
|
|
|
/* set default font to NULL */
|
2009-11-15 22:47:26 +08:00
|
|
|
rtgui_default_font = RT_NULL;
|
2009-10-16 08:03:50 +08:00
|
|
|
|
|
|
|
#ifdef RTGUI_USING_FONT16
|
|
|
|
rtgui_font_system_add_font(&rtgui_font_asc16);
|
|
|
|
#ifdef RTGUI_USING_FONTHZ
|
|
|
|
rtgui_font_system_add_font(&rtgui_font_hz16);
|
|
|
|
#endif
|
2009-11-15 22:47:26 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef RTGUI_USING_FONT12
|
2009-10-16 08:03:50 +08:00
|
|
|
rtgui_font_system_add_font(&rtgui_font_asc12);
|
|
|
|
#ifdef RTGUI_USING_FONTHZ
|
|
|
|
rtgui_font_system_add_font(&rtgui_font_hz12);
|
|
|
|
#endif
|
2009-11-15 22:47:26 +08:00
|
|
|
#endif
|
2009-10-16 08:03:50 +08:00
|
|
|
|
2009-11-15 22:47:26 +08:00
|
|
|
#ifdef RTGUI_USING_FONT12
|
2010-01-21 18:05:06 +08:00
|
|
|
if (rtgui_default_font == RT_NULL)
|
|
|
|
rtgui_font_set_defaut(&rtgui_font_asc12);
|
2010-01-18 07:43:20 +08:00
|
|
|
#endif
|
|
|
|
#ifdef RTGUI_USING_FONT16
|
2010-01-21 18:05:06 +08:00
|
|
|
if (rtgui_default_font == RT_NULL)
|
|
|
|
rtgui_font_set_defaut(&rtgui_font_asc16);
|
2009-11-15 22:47:26 +08:00
|
|
|
#endif
|
2009-10-16 08:03:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void rtgui_font_system_add_font(struct rtgui_font* font)
|
|
|
|
{
|
|
|
|
rtgui_list_init(&(font->list));
|
|
|
|
rtgui_list_append(&_rtgui_font_list, &(font->list));
|
2009-11-15 22:47:26 +08:00
|
|
|
|
|
|
|
/* init font */
|
|
|
|
if (font->engine->font_init != RT_NULL)
|
|
|
|
font->engine->font_init(font);
|
|
|
|
|
|
|
|
/* first refer, load it */
|
|
|
|
if (font->engine->font_load != RT_NULL)
|
|
|
|
font->engine->font_load(font);
|
2009-10-16 08:03:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void rtgui_font_system_remove_font(struct rtgui_font* font)
|
|
|
|
{
|
|
|
|
rtgui_list_remove(&_rtgui_font_list, &(font->list));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct rtgui_font* rtgui_font_default()
|
|
|
|
{
|
|
|
|
return rtgui_default_font;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtgui_font_set_defaut(struct rtgui_font* font)
|
|
|
|
{
|
|
|
|
rtgui_default_font = font;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct rtgui_font* rtgui_font_refer(const rt_uint8_t* family, rt_uint16_t height)
|
|
|
|
{
|
|
|
|
/* search font */
|
|
|
|
struct rtgui_list_node* node;
|
|
|
|
struct rtgui_font* font;
|
|
|
|
|
|
|
|
rtgui_list_foreach(node, &_rtgui_font_list)
|
|
|
|
{
|
|
|
|
font = rtgui_list_entry(node, struct rtgui_font, list);
|
|
|
|
if ((rt_strncmp((const char*)font->family, (const char*)family, RTGUI_NAME_MAX) == 0) &&
|
|
|
|
font->height == height)
|
|
|
|
{
|
|
|
|
font->refer_count ++;
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return RT_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtgui_font_derefer(struct rtgui_font* font)
|
|
|
|
{
|
|
|
|
RT_ASSERT(font != RT_NULL);
|
|
|
|
|
|
|
|
font->refer_count --;
|
|
|
|
|
|
|
|
/* no refer, remove font */
|
|
|
|
if (font->refer_count == 0)
|
|
|
|
{
|
|
|
|
rtgui_font_system_remove_font(font);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* draw a text */
|
2010-02-18 07:56:49 +08:00
|
|
|
void rtgui_font_draw(struct rtgui_font* font, struct rtgui_dc* dc, const char* text, rt_ubase_t len, struct rtgui_rect* rect)
|
2009-10-16 08:03:50 +08:00
|
|
|
{
|
|
|
|
RT_ASSERT(font != RT_NULL);
|
|
|
|
|
|
|
|
if (font->engine != RT_NULL &&
|
|
|
|
font->engine->font_draw_text != RT_NULL)
|
|
|
|
{
|
|
|
|
font->engine->font_draw_text(font, dc, text, len, rect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-18 07:56:49 +08:00
|
|
|
int rtgui_font_get_string_width(struct rtgui_font* font, const char* text)
|
2009-10-16 08:03:50 +08:00
|
|
|
{
|
|
|
|
rtgui_rect_t rect;
|
|
|
|
|
|
|
|
/* get metrics */
|
|
|
|
rtgui_font_get_metrics(font, text, &rect);
|
|
|
|
|
|
|
|
return rect.x2 - rect.x1;
|
|
|
|
}
|
|
|
|
|
2010-02-18 07:56:49 +08:00
|
|
|
void rtgui_font_get_metrics(struct rtgui_font* font, const char* text, rtgui_rect_t* rect)
|
2009-10-16 08:03:50 +08:00
|
|
|
{
|
|
|
|
RT_ASSERT(font != RT_NULL);
|
|
|
|
|
|
|
|
if (font->engine != RT_NULL &&
|
|
|
|
font->engine->font_get_metrics != RT_NULL)
|
|
|
|
{
|
|
|
|
font->engine->font_get_metrics(font, text, rect);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* no font engine found, set rect to zero */
|
|
|
|
rt_memset(rect, 0, sizeof(rtgui_rect_t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-18 07:56:49 +08:00
|
|
|
static void rtgui_bitmap_font_draw_text(struct rtgui_font* font, struct rtgui_dc* dc, const char* text, rt_ubase_t len, struct rtgui_rect* rect);
|
|
|
|
static void rtgui_bitmap_font_get_metrics(struct rtgui_font* font, const char* text, rtgui_rect_t* rect);
|
2009-10-16 08:03:50 +08:00
|
|
|
struct rtgui_font_engine bmp_font_engine =
|
|
|
|
{
|
|
|
|
RT_NULL,
|
|
|
|
RT_NULL,
|
|
|
|
rtgui_bitmap_font_draw_text,
|
|
|
|
rtgui_bitmap_font_get_metrics
|
|
|
|
};
|
|
|
|
|
|
|
|
void rtgui_bitmap_font_draw_char(struct rtgui_font_bitmap* font, struct rtgui_dc* dc, const char ch,
|
|
|
|
rtgui_rect_t* rect)
|
|
|
|
{
|
2010-01-21 18:05:06 +08:00
|
|
|
const rt_uint8_t* font_ptr;
|
2009-10-16 08:03:50 +08:00
|
|
|
rt_uint16_t x, y, w, h;
|
|
|
|
register rt_base_t i, j;
|
|
|
|
|
|
|
|
x = rect->x1;
|
|
|
|
y = rect->y1;
|
|
|
|
|
|
|
|
/* check first and last char */
|
|
|
|
if (ch < font->first_char || ch > font->last_char) return;
|
2010-01-21 18:05:06 +08:00
|
|
|
font_ptr = font->bmp + (ch - font->first_char) * font->height;
|
2009-10-16 08:03:50 +08:00
|
|
|
|
|
|
|
w = (font->width + x > rect->x2)? rect->x2 - rect->x1 : font->width;
|
|
|
|
h = (font->height + y > rect->y2)? rect->y2 - rect->y1 : font->height;
|
|
|
|
|
|
|
|
for (i = 0; i < h; i ++ )
|
|
|
|
{
|
|
|
|
for (j = 0; j < w; j ++)
|
|
|
|
{
|
|
|
|
if ( ((font_ptr[i] >> (7-j)) & 0x01) != 0)
|
|
|
|
{
|
|
|
|
/* draw a pixel */
|
|
|
|
rtgui_dc_draw_point(dc, j + x, i + y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-18 07:56:49 +08:00
|
|
|
static void rtgui_bitmap_font_draw_text(struct rtgui_font* font, struct rtgui_dc* dc, const char* text, rt_ubase_t len, struct rtgui_rect* rect)
|
2009-10-16 08:03:50 +08:00
|
|
|
{
|
|
|
|
struct rtgui_font_bitmap* bmp_font = (struct rtgui_font_bitmap*)(font->data);
|
|
|
|
|
|
|
|
RT_ASSERT(bmp_font != RT_NULL);
|
|
|
|
|
|
|
|
while (len-- && rect->x1 < rect->x2)
|
|
|
|
{
|
|
|
|
rtgui_bitmap_font_draw_char(bmp_font, dc, *text, rect);
|
|
|
|
|
|
|
|
/* move x to next character */
|
|
|
|
rect->x1 += bmp_font->width;
|
|
|
|
text ++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-18 07:56:49 +08:00
|
|
|
static void rtgui_bitmap_font_get_metrics(struct rtgui_font* font, const char* text, rtgui_rect_t* rect)
|
2009-10-16 08:03:50 +08:00
|
|
|
{
|
|
|
|
struct rtgui_font_bitmap* bmp_font = (struct rtgui_font_bitmap*)(font->data);
|
|
|
|
|
|
|
|
RT_ASSERT(bmp_font != RT_NULL);
|
|
|
|
|
|
|
|
/* set metrics rect */
|
|
|
|
rect->x1 = rect->y1 = 0;
|
|
|
|
rect->x2 = bmp_font->width * (rt_int16_t)rt_strlen((const char*)text);
|
|
|
|
rect->y2 = bmp_font->height;
|
|
|
|
}
|