forked from enlightenment/efl
2480 lines
79 KiB
C
2480 lines
79 KiB
C
#include "evas_common.h" /* Includes evas_bidi_utils stuff. */
|
|
#include "evas_private.h"
|
|
|
|
#include "Eo.h"
|
|
|
|
EAPI Eo_Op EVAS_OBJ_TEXT_BASE_ID = EO_NOOP;
|
|
|
|
#define MY_CLASS EVAS_OBJ_TEXT_CLASS
|
|
|
|
/* save typing */
|
|
#define ENFN obj->layer->evas->engine.func
|
|
#define ENDT obj->layer->evas->engine.data.output
|
|
|
|
/* private magic number for text objects */
|
|
static const char o_type[] = "text";
|
|
|
|
/* private struct for text object internal data */
|
|
typedef struct _Evas_Object_Text Evas_Object_Text;
|
|
typedef struct _Evas_Object_Text_Item Evas_Object_Text_Item;
|
|
|
|
struct _Evas_Object_Text
|
|
{
|
|
DATA32 magic;
|
|
|
|
struct {
|
|
const char *utf8_text; /* The text exposed to the API */
|
|
const char *font;
|
|
Evas_Font_Description *fdesc;
|
|
const char *source;
|
|
Evas_Font_Size size;
|
|
struct {
|
|
unsigned char r, g, b, a;
|
|
} outline, shadow, glow, glow2;
|
|
|
|
unsigned char style;
|
|
double ellipsis;
|
|
Eina_Unicode *text;
|
|
} cur, prev;
|
|
|
|
float ascent, descent;
|
|
float max_ascent, max_descent;
|
|
Evas_BiDi_Paragraph_Props *bidi_par_props;
|
|
const char *bidi_delimiters;
|
|
Evas_Object_Text_Item *items;
|
|
|
|
Evas_Font_Set *font;
|
|
|
|
struct {
|
|
Evas_Coord w, h;
|
|
Evas_Object_Text_Item *ellipsis_start;
|
|
Evas_Object_Text_Item *ellipsis_end;
|
|
} last_computed;
|
|
|
|
char changed : 1;
|
|
};
|
|
|
|
struct _Evas_Object_Text_Item
|
|
{
|
|
EINA_INLIST;
|
|
|
|
size_t text_pos;
|
|
size_t visual_pos;
|
|
Evas_Text_Props text_props;
|
|
Evas_Coord x, w, h, adv;
|
|
};
|
|
|
|
/* private methods for text objects */
|
|
static void evas_object_text_init(Evas_Object *eo_obj);
|
|
static void evas_object_text_render(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj, void *output, void *context, void *surface, int x, int y, Eina_Bool do_async);
|
|
static void evas_object_text_free(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj);
|
|
static void evas_object_text_render_pre(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj);
|
|
static void evas_object_text_render_post(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj);
|
|
|
|
static unsigned int evas_object_text_id_get(Evas_Object *eo_obj);
|
|
static unsigned int evas_object_text_visual_id_get(Evas_Object *eo_obj);
|
|
static void *evas_object_text_engine_data_get(Evas_Object *eo_obj);
|
|
|
|
static int evas_object_text_is_opaque(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj);
|
|
static int evas_object_text_was_opaque(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj);
|
|
|
|
static void evas_object_text_scale_update(Evas_Object *eo_obj);
|
|
static void _evas_object_text_recalc(Evas_Object *eo_obj, Eina_Unicode *text);
|
|
|
|
static const Evas_Object_Func object_func =
|
|
{
|
|
/* methods (compulsory) */
|
|
NULL,
|
|
evas_object_text_render,
|
|
evas_object_text_render_pre,
|
|
evas_object_text_render_post,
|
|
evas_object_text_id_get,
|
|
evas_object_text_visual_id_get,
|
|
evas_object_text_engine_data_get,
|
|
/* these are optional. NULL = nothing */
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
evas_object_text_is_opaque,
|
|
evas_object_text_was_opaque,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
evas_object_text_scale_update,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
/* the actual api call to add a rect */
|
|
/* it has no other api calls as all properties are standard */
|
|
|
|
static int
|
|
_evas_object_text_char_coords_get(const Evas_Object *eo_obj,
|
|
const Evas_Object_Text *o,
|
|
size_t pos, Evas_Coord *x, Evas_Coord *y, Evas_Coord *w, Evas_Coord *h)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text_Item *it;
|
|
|
|
EINA_INLIST_FOREACH(EINA_INLIST_GET(o->items), it)
|
|
{
|
|
if ((it->text_pos <= pos) &&
|
|
(pos < (it->text_pos + it->text_props.text_len)))
|
|
{
|
|
int ret;
|
|
ret = ENFN->font_char_coords_get(ENDT, o->font,
|
|
&it->text_props, pos - it->text_pos, x, y, w, h);
|
|
if (x) *x += it->x;
|
|
return ret;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_item_clean(Evas_Object_Text_Item *it)
|
|
{
|
|
evas_common_text_props_content_unref(&it->text_props);
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_item_del(Evas_Object_Text *o, Evas_Object_Text_Item *it)
|
|
{
|
|
if (o->last_computed.ellipsis_start == it)
|
|
o->last_computed.ellipsis_start = NULL;
|
|
else if (o->last_computed.ellipsis_end == it)
|
|
o->last_computed.ellipsis_end = NULL;
|
|
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_remove(
|
|
EINA_INLIST_GET(o->items),
|
|
EINA_INLIST_GET(it));
|
|
_evas_object_text_item_clean(it);
|
|
free(it);
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_items_clean(Evas_Object_Protected_Data *obj, Evas_Object_Text *o)
|
|
{
|
|
/* FIXME: also preserve item */
|
|
if ((o->cur.font == o->prev.font) &&
|
|
(o->cur.fdesc == o->prev.fdesc) &&
|
|
(o->cur.size == o->prev.size) &&
|
|
(!memcmp(&o->cur.outline, &o->prev.outline, sizeof (o->cur.outline))) &&
|
|
(!memcmp(&o->cur.shadow, &o->prev.shadow, sizeof (o->cur.shadow))) &&
|
|
(!memcmp(&o->cur.glow, &o->prev.glow, sizeof (o->cur.glow))) &&
|
|
(!memcmp(&o->cur.glow2, &o->prev.glow2, sizeof (o->cur.glow2))) &&
|
|
(o->cur.style == o->prev.style) &&
|
|
(obj->cur.scale == obj->prev.scale))
|
|
{
|
|
if ((o->last_computed.ellipsis_start) &&
|
|
(o->last_computed.ellipsis_start == o->items))
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_remove(EINA_INLIST_GET(o->items),
|
|
EINA_INLIST_GET(o->last_computed.ellipsis_start));
|
|
if ((o->last_computed.ellipsis_end) &&
|
|
(EINA_INLIST_GET(o->last_computed.ellipsis_end) == EINA_INLIST_GET(o->items)->last))
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_remove(EINA_INLIST_GET(o->items),
|
|
EINA_INLIST_GET(o->last_computed.ellipsis_end));
|
|
}
|
|
else
|
|
{
|
|
o->last_computed.ellipsis_start = NULL;
|
|
o->last_computed.ellipsis_end = NULL;
|
|
}
|
|
while (o->items)
|
|
{
|
|
_evas_object_text_item_del(o, o->items);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_items_clear(Evas_Object_Text *o)
|
|
{
|
|
if ((o->last_computed.ellipsis_start) &&
|
|
(o->last_computed.ellipsis_start != o->items))
|
|
{
|
|
_evas_object_text_item_clean(o->last_computed.ellipsis_start);
|
|
}
|
|
o->last_computed.ellipsis_start = NULL;
|
|
if ((o->last_computed.ellipsis_end) &&
|
|
(EINA_INLIST_GET(o->last_computed.ellipsis_end) != EINA_INLIST_GET(o->items)->last))
|
|
{
|
|
_evas_object_text_item_clean(o->last_computed.ellipsis_end);
|
|
}
|
|
o->last_computed.ellipsis_end = NULL;
|
|
while (o->items)
|
|
{
|
|
_evas_object_text_item_del(o, o->items);
|
|
}
|
|
}
|
|
|
|
#ifdef BIDI_SUPPORT
|
|
static int
|
|
_evas_object_text_it_compare_logical(const void *_it1, const void *_it2)
|
|
{
|
|
const Evas_Object_Text_Item *it1 = _it1, *it2 = _it2;
|
|
if (it1->text_pos < it2->text_pos)
|
|
return -1;
|
|
else if (it1->text_pos == it2->text_pos)
|
|
return 0;
|
|
else
|
|
return 1;
|
|
|
|
}
|
|
#endif
|
|
|
|
static int
|
|
_evas_object_text_last_up_to_pos(const Evas_Object *eo_obj,
|
|
const Evas_Object_Text *o, Evas_Coord cx, Evas_Coord cy)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text_Item *it;
|
|
|
|
#ifdef BIDI_SUPPORT
|
|
/*FIXME: not very efficient, sort the items arrays. */
|
|
/* Reorder if it's a bidi text */
|
|
if (o->bidi_par_props)
|
|
{
|
|
Eina_List *logical_it = NULL;
|
|
Evas_Object_Text_Item *i;
|
|
Eina_List *itr;
|
|
Evas_Coord x = 0;
|
|
/* Insert all to the logical list */
|
|
EINA_INLIST_FOREACH(o->items, i)
|
|
{
|
|
logical_it = eina_list_sorted_insert(logical_it,
|
|
_evas_object_text_it_compare_logical, i);
|
|
}
|
|
EINA_LIST_FOREACH(logical_it, itr, it)
|
|
{
|
|
if ((x <= cx) && (cx < x + it->adv))
|
|
{
|
|
return it->text_pos + ENFN->font_last_up_to_pos(ENDT,
|
|
o->font,
|
|
&it->text_props,
|
|
cx - x,
|
|
cy);
|
|
}
|
|
x += it->adv;
|
|
}
|
|
eina_list_free(logical_it);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
EINA_INLIST_FOREACH(EINA_INLIST_GET(o->items), it)
|
|
{
|
|
if ((it->x <= cx) && (cx < it->x + it->adv))
|
|
{
|
|
return it->text_pos + ENFN->font_last_up_to_pos(ENDT,
|
|
o->font,
|
|
&it->text_props,
|
|
cx - it->x,
|
|
cy);
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static int
|
|
_evas_object_text_char_at_coords(const Evas_Object *eo_obj,
|
|
const Evas_Object_Text *o, Evas_Coord cx, Evas_Coord cy,
|
|
Evas_Coord *rx, Evas_Coord *ry, Evas_Coord *rw, Evas_Coord *rh)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text_Item *it;
|
|
|
|
EINA_INLIST_FOREACH(EINA_INLIST_GET(o->items), it)
|
|
{
|
|
if ((it->x <= cx) && (cx < it->x + it->adv))
|
|
{
|
|
return it->text_pos + ENFN->font_char_at_coords_get(ENDT,
|
|
o->font,
|
|
&it->text_props,
|
|
cx - it->x,
|
|
cy,
|
|
rx, ry,
|
|
rw, rh);
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static Evas_Coord
|
|
_evas_object_text_horiz_advance_get(const Evas_Object *eo_obj,
|
|
const Evas_Object_Text *o)
|
|
{
|
|
Evas_Object_Text_Item *it, *last_it = NULL;
|
|
Evas_Coord adv;
|
|
(void) eo_obj;
|
|
|
|
adv = 0;
|
|
EINA_INLIST_FOREACH(EINA_INLIST_GET(o->items), it)
|
|
{
|
|
adv += it->adv;
|
|
last_it = it;
|
|
}
|
|
|
|
if (last_it && (last_it->w > last_it->adv))
|
|
adv += last_it->w - last_it->adv;
|
|
return adv;
|
|
}
|
|
|
|
static Evas_Coord
|
|
_evas_object_text_vert_advance_get(const Evas_Object *obj EINA_UNUSED,
|
|
const Evas_Object_Text *o)
|
|
{
|
|
return o->max_ascent + o->max_descent;
|
|
}
|
|
|
|
EAPI Evas_Object *
|
|
evas_object_text_add(Evas *e)
|
|
{
|
|
MAGIC_CHECK(e, Evas, MAGIC_EVAS);
|
|
return NULL;
|
|
MAGIC_CHECK_END();
|
|
Evas_Object *eo_obj = eo_add(EVAS_OBJ_TEXT_CLASS, e);
|
|
eo_unref(eo_obj);
|
|
return eo_obj;
|
|
}
|
|
|
|
static void
|
|
_constructor(Eo *eo_obj, void *class_data EINA_UNUSED, va_list *list EINA_UNUSED)
|
|
{
|
|
eo_do_super(eo_obj, eo_constructor());
|
|
evas_object_text_init(eo_obj);
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_inject(eo_obj, obj, evas_object_evas_get(eo_parent_get(eo_obj)));
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_font_source_set(Evas_Object *eo_obj, const char *font_source)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_font_source_set(font_source));
|
|
}
|
|
|
|
static void
|
|
_text_font_source_set(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Text *o = _pd;
|
|
|
|
const char *font_source = va_arg(*list, const char *);
|
|
if ((o->cur.source) && (font_source) &&
|
|
(!strcmp(o->cur.source, font_source)))
|
|
return;
|
|
/*
|
|
if (o->cur.source) eina_stringshare_del(o->cur.source);
|
|
if (font_source) o->cur.source = eina_stringshare_add(font_source);
|
|
else o->cur.source = NULL;
|
|
*/
|
|
eina_stringshare_replace(&o->cur.source, font_source);
|
|
}
|
|
|
|
EAPI const char *
|
|
evas_object_text_font_source_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return NULL;
|
|
MAGIC_CHECK_END();
|
|
const char *font_source = NULL;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_font_source_get(&font_source));
|
|
return font_source;
|
|
}
|
|
|
|
static void
|
|
_text_font_source_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
const Evas_Object_Text *o = _pd;
|
|
|
|
const char ** font_source = va_arg(*list, const char **);
|
|
*font_source = o->cur.source;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_font_set(Evas_Object *eo_obj, const char *font, Evas_Font_Size size)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_font_set(font, size));
|
|
}
|
|
|
|
static void
|
|
_text_font_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Text *o = _pd;
|
|
Eina_Bool is, was = EINA_FALSE;
|
|
Eina_Bool pass = EINA_FALSE, freeze = EINA_FALSE;
|
|
Eina_Bool source_invisible = EINA_FALSE;
|
|
Evas_Font_Description *fdesc;
|
|
|
|
const char *font = va_arg(*list, const char*);
|
|
Evas_Font_Size size = va_arg(*list, Evas_Font_Size);
|
|
|
|
if ((!font) || (size <= 0)) return;
|
|
|
|
fdesc = evas_font_desc_new();
|
|
evas_font_name_parse(fdesc, font);
|
|
if (o->cur.fdesc && !evas_font_desc_cmp(fdesc, o->cur.fdesc) &&
|
|
(size == o->cur.size))
|
|
{
|
|
evas_font_desc_unref(fdesc);
|
|
return;
|
|
}
|
|
|
|
if (o->cur.fdesc) evas_font_desc_unref(o->cur.fdesc);
|
|
o->cur.fdesc = fdesc;
|
|
|
|
o->cur.size = size;
|
|
eina_stringshare_replace(&o->cur.font, font);
|
|
o->prev.font = NULL;
|
|
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
if (!(obj->layer->evas->is_frozen))
|
|
{
|
|
pass = evas_event_passes_through(eo_obj, obj);
|
|
freeze = evas_event_freezes_through(eo_obj, obj);
|
|
source_invisible = evas_object_is_source_invisible(eo_obj, obj);
|
|
if ((!pass) && (!freeze) && (!source_invisible))
|
|
was = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y, 1, 1);
|
|
}
|
|
|
|
/* DO IT */
|
|
if (o->font)
|
|
{
|
|
evas_font_free(obj->layer->evas->evas, o->font);
|
|
o->font = NULL;
|
|
}
|
|
|
|
o->font = evas_font_load(obj->layer->evas->evas, o->cur.fdesc, o->cur.source,
|
|
(int)(((double) o->cur.size) * obj->cur.scale));
|
|
if (o->font)
|
|
{
|
|
o->ascent = ENFN->font_ascent_get(ENDT, o->font);
|
|
o->descent = ENFN->font_descent_get(ENDT, o->font);
|
|
o->max_ascent = ENFN->font_max_ascent_get(ENDT, o->font);
|
|
o->max_descent = ENFN->font_max_descent_get(ENDT, o->font);
|
|
}
|
|
else
|
|
{
|
|
o->ascent = 0;
|
|
o->descent = 0;
|
|
o->max_ascent = 0;
|
|
o->max_descent = 0;
|
|
}
|
|
_evas_object_text_recalc(eo_obj, o->cur.text);
|
|
o->changed = 1;
|
|
evas_object_change(eo_obj, obj);
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
evas_object_coords_recalc(eo_obj, obj);
|
|
if (!(obj->layer->evas->is_frozen))
|
|
{
|
|
if ((!pass) && (!freeze))
|
|
{
|
|
is = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y,
|
|
1, 1);
|
|
if ((is ^ was) && obj->cur.visible)
|
|
evas_event_feed_mouse_move(obj->layer->evas->evas,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y,
|
|
obj->layer->evas->last_timestamp,
|
|
NULL);
|
|
}
|
|
}
|
|
evas_object_inform_call_resize(eo_obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_font_get(const Evas_Object *eo_obj, const char **font, Evas_Font_Size *size)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
if (font) *font = "";
|
|
if (size) *size = 0;
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_font_get(font, size));
|
|
}
|
|
|
|
static void
|
|
_text_font_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
const Evas_Object_Text *o = _pd;
|
|
const char **font = va_arg(*list, const char **);
|
|
Evas_Font_Size *size = va_arg(*list, Evas_Font_Size *);
|
|
|
|
if (font) *font = o->cur.font;
|
|
if (size) *size = o->cur.size;
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_item_update_sizes(Evas_Object_Protected_Data *obj, Evas_Object_Text *o, Evas_Object_Text_Item *it)
|
|
{
|
|
ENFN->font_string_size_get(ENDT,
|
|
o->font,
|
|
&it->text_props,
|
|
&it->w, &it->h);
|
|
it->adv = ENFN->font_h_advance_get(ENDT, o->font,
|
|
&it->text_props);
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
* Create a new text layout item from the string and the format.
|
|
*
|
|
* @param c the context to work on - Not NULL.
|
|
* @param fmt the format to use.
|
|
* @param str the string to use.
|
|
*/
|
|
static Evas_Object_Text_Item *
|
|
_evas_object_text_item_new(Evas_Object_Protected_Data *obj,
|
|
Evas_Object_Text *o,
|
|
Evas_Font_Instance *fi, const Eina_Unicode *str,
|
|
Evas_Script_Type script,
|
|
size_t pos, size_t visual_pos, size_t len)
|
|
{
|
|
Evas_Object_Text_Item *it;
|
|
|
|
it = calloc(1, sizeof(Evas_Object_Text_Item));
|
|
it->text_pos = pos;
|
|
it->visual_pos = visual_pos;
|
|
evas_common_text_props_bidi_set(&it->text_props, o->bidi_par_props,
|
|
it->text_pos);
|
|
evas_common_text_props_script_set(&it->text_props, script);
|
|
|
|
if (fi)
|
|
{
|
|
ENFN->font_text_props_info_create(ENDT,
|
|
fi, str + pos, &it->text_props,
|
|
o->bidi_par_props, it->text_pos, len, EVAS_TEXT_PROPS_MODE_SHAPE);
|
|
_evas_object_text_item_update_sizes(obj, o, it);
|
|
}
|
|
o->items = (Evas_Object_Text_Item *)
|
|
eina_inlist_append(EINA_INLIST_GET(o->items), EINA_INLIST_GET(it));
|
|
return it;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
* Orders o->items according to the visual position.
|
|
*
|
|
* @param obj the evas object
|
|
* @param o the text object
|
|
*/
|
|
static void
|
|
_evas_object_text_item_order(Evas_Object *eo_obj, Evas_Object_Text *o)
|
|
{
|
|
(void) eo_obj;
|
|
#ifdef BIDI_SUPPORT
|
|
/*FIXME: not very efficient, sort the items arrays. */
|
|
/* Reorder if it's a bidi text */
|
|
if (o->bidi_par_props)
|
|
{
|
|
Evas_Object_Text_Item *i, *j, *min;
|
|
i = o->items;
|
|
while (i)
|
|
{
|
|
min = i;
|
|
EINA_INLIST_FOREACH(i, j)
|
|
{
|
|
if (j->visual_pos < min->visual_pos)
|
|
{
|
|
min = j;
|
|
}
|
|
}
|
|
if (min != i)
|
|
{
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_remove(EINA_INLIST_GET(o->items), EINA_INLIST_GET(min));
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_prepend_relative(EINA_INLIST_GET(o->items), EINA_INLIST_GET(min), EINA_INLIST_GET(i));
|
|
}
|
|
|
|
i = (Evas_Object_Text_Item *) EINA_INLIST_GET(min)->next;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* calculate the positions according to the order. */
|
|
{
|
|
Evas_Object_Text_Item *it = o->items;
|
|
Evas_Coord x = 0;
|
|
|
|
while (it)
|
|
{
|
|
it->x = x;
|
|
x += it->adv;
|
|
it = (Evas_Object_Text_Item *) EINA_INLIST_GET(it)->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Create ellipsis.
|
|
*/
|
|
static const Eina_Unicode _ellip_str[2] = { 0x2026, '\0' };
|
|
|
|
/* FIXME: We currently leak ellipsis items. */
|
|
static Evas_Object_Text_Item *
|
|
_layout_ellipsis_item_new(Evas_Object_Protected_Data *obj, Evas_Object_Text *o, Evas_Object_Text_Item *ti)
|
|
{
|
|
Evas_Object_Text_Item *ellip_ti;
|
|
size_t len = 1; /* The length of _ellip_str */
|
|
|
|
ellip_ti = _evas_object_text_item_new(obj, o, ti->text_props.font_instance,
|
|
_ellip_str, ti->text_props.script, 0, 0, len);
|
|
|
|
return ellip_ti;
|
|
}
|
|
|
|
/* EINA_TRUE if this item is ok and should be included, false if should be
|
|
* discarded. */
|
|
static Eina_Bool
|
|
_layout_text_item_trim(Evas_Object_Protected_Data *obj, Evas_Object_Text *o, Evas_Object_Text_Item *ti, int idx, Eina_Bool want_start)
|
|
{
|
|
Evas_Text_Props new_text_props;
|
|
if (idx >= (int) ti->text_props.len)
|
|
return EINA_FALSE;
|
|
|
|
memset(&new_text_props, 0, sizeof (new_text_props));
|
|
|
|
while (!evas_common_text_props_split(&ti->text_props, &new_text_props, idx))
|
|
idx--;
|
|
if (want_start)
|
|
{
|
|
evas_common_text_props_content_unref(&new_text_props);
|
|
}
|
|
else
|
|
{
|
|
evas_common_text_props_content_unref(&ti->text_props);
|
|
memcpy(&ti->text_props, &new_text_props, sizeof(ti->text_props));
|
|
ti->text_pos += idx;
|
|
ti->visual_pos += idx;
|
|
}
|
|
_evas_object_text_item_update_sizes(obj, o, ti);
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
* Populates o->items with the items of the text according to text
|
|
*
|
|
* @param obj the evas object
|
|
* @param o the text object
|
|
* @param text the text to layout
|
|
*/
|
|
static void
|
|
_evas_object_text_layout(Evas_Object *eo_obj, Evas_Object_Text *o, Eina_Unicode *text)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
EvasBiDiStrIndex *v_to_l = NULL;
|
|
Evas_Coord advance = 0;
|
|
size_t pos, visual_pos;
|
|
int len = eina_unicode_strlen(text);
|
|
#ifdef BIDI_SUPPORT
|
|
int par_len = len;
|
|
int *segment_idxs = NULL;
|
|
#endif
|
|
|
|
if (o->items &&
|
|
!memcmp(&o->cur, &o->prev, sizeof (o->cur)) &&
|
|
o->cur.text == text &&
|
|
obj->cur.scale == obj->prev.scale &&
|
|
o->last_computed.w == obj->cur.geometry.w &&
|
|
o->last_computed.h == obj->cur.geometry.h)
|
|
return ;
|
|
|
|
if (o->items) _evas_object_text_items_clean(obj, o);
|
|
|
|
#ifdef BIDI_SUPPORT
|
|
if (o->bidi_delimiters)
|
|
segment_idxs = evas_bidi_segment_idxs_get(text, o->bidi_delimiters);
|
|
evas_bidi_paragraph_props_unref(o->bidi_par_props);
|
|
o->bidi_par_props = evas_bidi_paragraph_props_get(text, len, segment_idxs);
|
|
evas_bidi_props_reorder_line(NULL, 0, len, o->bidi_par_props, &v_to_l);
|
|
if (segment_idxs) free(segment_idxs);
|
|
#endif
|
|
visual_pos = pos = 0;
|
|
|
|
while (len > 0)
|
|
{
|
|
Evas_Font_Instance *script_fi = NULL;
|
|
int script_len = len, tmp_cut;
|
|
Evas_Script_Type script;
|
|
tmp_cut = evas_common_language_script_end_of_run_get(
|
|
text + pos,
|
|
o->bidi_par_props,
|
|
pos, len);
|
|
if (tmp_cut > 0)
|
|
script_len = tmp_cut;
|
|
|
|
script = evas_common_language_script_type_get(text, script_len);
|
|
|
|
while (script_len > 0)
|
|
{
|
|
const Evas_Object_Text_Item *it;
|
|
Evas_Font_Instance *cur_fi = NULL;
|
|
int run_len = script_len;
|
|
if (o->font)
|
|
{
|
|
run_len = ENFN->font_run_end_get(ENDT,
|
|
o->font, &script_fi, &cur_fi,
|
|
script, text + pos, script_len);
|
|
}
|
|
#ifdef BIDI_SUPPORT
|
|
visual_pos = evas_bidi_position_logical_to_visual(
|
|
v_to_l, par_len, pos);
|
|
#else
|
|
visual_pos = pos;
|
|
#endif
|
|
it = _evas_object_text_item_new(obj, o, cur_fi, text, script,
|
|
pos, visual_pos, run_len);
|
|
|
|
advance += it->adv;
|
|
pos += run_len;
|
|
script_len -= run_len;
|
|
len -= run_len;
|
|
}
|
|
}
|
|
|
|
/* Handle ellipsis */
|
|
if ((o->cur.ellipsis >= 0.0) && (advance > obj->cur.geometry.w) && (obj->cur.geometry.w > 0))
|
|
{
|
|
Evas_Coord ellip_frame = obj->cur.geometry.w;
|
|
Evas_Object_Text_Item *start_ellip_it = NULL, *end_ellip_it = NULL;
|
|
/* Account of the ellipsis item width. As long as ellipsis != 0
|
|
* we have a left ellipsis. And the same with 1 and right. */
|
|
if (o->cur.ellipsis != 0)
|
|
{
|
|
if (o->last_computed.ellipsis_start)
|
|
{
|
|
start_ellip_it = o->last_computed.ellipsis_start;
|
|
o->items = (Evas_Object_Text_Item *)
|
|
eina_inlist_append(EINA_INLIST_GET(o->items), EINA_INLIST_GET(start_ellip_it));
|
|
}
|
|
else
|
|
{
|
|
start_ellip_it = _layout_ellipsis_item_new(obj, o, o->items);
|
|
}
|
|
o->last_computed.ellipsis_start = start_ellip_it;
|
|
ellip_frame -= start_ellip_it->adv;
|
|
}
|
|
if (o->cur.ellipsis != 1)
|
|
{
|
|
/* FIXME: Should take the last item's font and style and etc. *//* weird it's a text, should always have the same style/font */
|
|
if (o->last_computed.ellipsis_end)
|
|
{
|
|
end_ellip_it = o->last_computed.ellipsis_end;
|
|
o->items = (Evas_Object_Text_Item *)
|
|
eina_inlist_append(EINA_INLIST_GET(o->items), EINA_INLIST_GET(end_ellip_it));
|
|
}
|
|
else
|
|
{
|
|
end_ellip_it = _layout_ellipsis_item_new(obj, o, o->items);
|
|
}
|
|
o->last_computed.ellipsis_end = end_ellip_it;
|
|
ellip_frame -= end_ellip_it->adv;
|
|
}
|
|
|
|
/* The point where we should start from, going for the full
|
|
* ellip frame. */
|
|
Evas_Coord ellipsis_coord = o->cur.ellipsis * (advance - ellip_frame);
|
|
if (start_ellip_it)
|
|
{
|
|
Evas_Object_Text_Item *itr = o->items;
|
|
advance = 0;
|
|
|
|
while (itr && (advance + itr->adv < ellipsis_coord))
|
|
{
|
|
Eina_Inlist *itrn = EINA_INLIST_GET(itr)->next;
|
|
if ((itr != start_ellip_it) && (itr != end_ellip_it))
|
|
{
|
|
advance += itr->adv;
|
|
_evas_object_text_item_del(o, itr);
|
|
}
|
|
itr = (Evas_Object_Text_Item *) itrn;
|
|
}
|
|
if (itr && (itr != start_ellip_it))
|
|
{
|
|
int cut = 1 + ENFN->font_char_at_coords_get(ENDT,
|
|
o->font,
|
|
&itr->text_props,
|
|
ellipsis_coord - advance,
|
|
0,
|
|
NULL, NULL, NULL, NULL);
|
|
if (cut > 0)
|
|
{
|
|
start_ellip_it->text_pos = itr->text_pos;
|
|
start_ellip_it->visual_pos = itr->visual_pos;
|
|
if (!_layout_text_item_trim(obj, o, itr, cut, EINA_FALSE))
|
|
{
|
|
_evas_object_text_item_del(o, itr);
|
|
}
|
|
}
|
|
}
|
|
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_remove(EINA_INLIST_GET(o->items), EINA_INLIST_GET(start_ellip_it));
|
|
o->items = (Evas_Object_Text_Item *) eina_inlist_prepend(EINA_INLIST_GET(o->items), EINA_INLIST_GET(start_ellip_it));
|
|
}
|
|
|
|
if (end_ellip_it)
|
|
{
|
|
Evas_Object_Text_Item *itr = o->items;
|
|
advance = 0;
|
|
|
|
while (itr)
|
|
{
|
|
if (itr != end_ellip_it) /* was start_ellip_it */
|
|
{
|
|
if (advance + itr->adv >= ellip_frame)
|
|
{
|
|
break;
|
|
}
|
|
advance += itr->adv;
|
|
}
|
|
itr = (Evas_Object_Text_Item *) EINA_INLIST_GET(itr)->next;
|
|
}
|
|
|
|
if (itr == end_ellip_it)
|
|
{
|
|
/* FIXME: We shouldn't do anything. */
|
|
}
|
|
|
|
int cut = ENFN->font_char_at_coords_get(ENDT,
|
|
o->font,
|
|
&itr->text_props,
|
|
ellip_frame - advance,
|
|
0,
|
|
NULL, NULL, NULL, NULL);
|
|
if (cut >= 0)
|
|
{
|
|
end_ellip_it->text_pos = itr->text_pos + cut;
|
|
end_ellip_it->visual_pos = itr->visual_pos + cut;
|
|
if (_layout_text_item_trim(obj, o, itr, cut, EINA_TRUE))
|
|
{
|
|
itr = (Evas_Object_Text_Item *) EINA_INLIST_GET(itr)->next;
|
|
}
|
|
}
|
|
|
|
/* Remove the rest of the items */
|
|
while (itr)
|
|
{
|
|
Eina_Inlist *itrn = EINA_INLIST_GET(itr)->next;
|
|
if ((itr != start_ellip_it) && (itr != end_ellip_it))
|
|
_evas_object_text_item_del(o, itr);
|
|
itr = (Evas_Object_Text_Item *) itrn;
|
|
}
|
|
}
|
|
}
|
|
if (o->cur.text != text) free(o->cur.text);
|
|
o->cur.text = text;
|
|
o->prev = o->cur;
|
|
|
|
_evas_object_text_item_order(eo_obj, o);
|
|
|
|
if (v_to_l) free(v_to_l);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_ellipsis_set(Evas_Object *obj, double ellipsis)
|
|
{
|
|
MAGIC_CHECK(obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(obj, evas_obj_text_ellipsis_set(ellipsis));
|
|
}
|
|
|
|
static void
|
|
_text_resize(void *data,
|
|
Evas *e EINA_UNUSED,
|
|
Evas_Object *obj,
|
|
void *event_info EINA_UNUSED)
|
|
{
|
|
Evas_Object_Text *o = data;
|
|
|
|
_evas_object_text_recalc(obj, o->cur.text);
|
|
}
|
|
|
|
static void
|
|
_text_ellipsis_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text *o = _pd;
|
|
double ellipsis = va_arg(*list, double);
|
|
|
|
if (o->cur.ellipsis == ellipsis) return ;
|
|
|
|
o->cur.ellipsis = ellipsis;
|
|
o->changed = 1;
|
|
evas_object_change(eo_obj, obj);
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
|
|
evas_object_event_callback_del_full(eo_obj, EVAS_CALLBACK_RESIZE,
|
|
_text_resize, o);
|
|
evas_object_event_callback_add(eo_obj, EVAS_CALLBACK_RESIZE,
|
|
_text_resize, o);
|
|
}
|
|
|
|
EAPI double
|
|
evas_object_text_ellipsis_get(const Evas_Object *obj)
|
|
{
|
|
double r = 0;
|
|
|
|
MAGIC_CHECK(obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
|
|
eo_do((Eo*) obj, evas_obj_text_ellipsis_get(&r));
|
|
return r;
|
|
}
|
|
|
|
static void
|
|
_text_ellipsis_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
const Evas_Object_Text *o = _pd;
|
|
double *r = va_arg(*list, double *);
|
|
|
|
*r = o->cur.ellipsis;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_text_set(Evas_Object *eo_obj, const char *_text)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_text_set(_text));
|
|
}
|
|
|
|
static void
|
|
_text_text_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Text *o = _pd;
|
|
int is, was, len;
|
|
Eina_Unicode *text;
|
|
|
|
const char *_text = va_arg(*list, const char *);
|
|
|
|
if ((o->cur.utf8_text) && (_text) && (!strcmp(o->cur.utf8_text, _text)))
|
|
return;
|
|
text = eina_unicode_utf8_to_unicode(_text, &len);
|
|
|
|
if (!text) text = eina_unicode_strdup(EINA_UNICODE_EMPTY_STRING);
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
was = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y, 1, 1);
|
|
/* DO II */
|
|
/*Update bidi_props*/
|
|
|
|
if (o->items) _evas_object_text_items_clear(o);
|
|
|
|
_evas_object_text_recalc(eo_obj, text);
|
|
eina_stringshare_replace(&o->cur.utf8_text, _text);
|
|
o->prev.utf8_text = NULL;
|
|
|
|
if (o->cur.text != text) free(text);
|
|
|
|
o->changed = 1;
|
|
evas_object_change(eo_obj, obj);
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
evas_object_coords_recalc(eo_obj, obj);
|
|
is = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y, 1, 1);
|
|
if ((is || was) && obj->cur.visible)
|
|
evas_event_feed_mouse_move(obj->layer->evas->evas,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y,
|
|
obj->layer->evas->last_timestamp,
|
|
NULL);
|
|
evas_object_inform_call_resize(eo_obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_bidi_delimiters_set(Evas_Object *eo_obj, const char *delim)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_bidi_delimiters_set(delim));
|
|
}
|
|
|
|
static void
|
|
_text_bidi_delimiters_set(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Text *o = _pd;
|
|
|
|
const char *delim = va_arg(*list, const char *);
|
|
|
|
eina_stringshare_replace(&o->bidi_delimiters, delim);
|
|
}
|
|
|
|
EAPI const char *
|
|
evas_object_text_bidi_delimiters_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return NULL;
|
|
MAGIC_CHECK_END();
|
|
const char *delim = NULL;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_bidi_delimiters_get(&delim));
|
|
return delim;
|
|
}
|
|
|
|
static void
|
|
_text_bidi_delimiters_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
const Evas_Object_Text *o = _pd;
|
|
const char **delim = va_arg(*list, const char **);
|
|
*delim = o->bidi_delimiters;
|
|
}
|
|
|
|
EAPI const char *
|
|
evas_object_text_text_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return NULL;
|
|
MAGIC_CHECK_END();
|
|
const char *text = NULL;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_text_get(&text));
|
|
return text;
|
|
}
|
|
|
|
static void
|
|
_text_text_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
const char **text = va_arg(*list, const char **);
|
|
const Evas_Object_Text *o = _pd;
|
|
*text = o->cur.utf8_text;
|
|
}
|
|
|
|
EAPI Evas_BiDi_Direction
|
|
evas_object_text_direction_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return EVAS_BIDI_DIRECTION_NEUTRAL;
|
|
MAGIC_CHECK_END();
|
|
Evas_BiDi_Direction bidi_dir = EVAS_BIDI_DIRECTION_NEUTRAL;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_direction_get(&bidi_dir));
|
|
return bidi_dir;
|
|
}
|
|
|
|
static void
|
|
_text_direction_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_BiDi_Direction *bidi_dir = va_arg(*list, Evas_BiDi_Direction *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*bidi_dir = o->items ? o->items->text_props.bidi.dir : EVAS_BIDI_DIRECTION_NEUTRAL;
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_ascent_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord ascent = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_ascent_get(&ascent));
|
|
return ascent;
|
|
}
|
|
|
|
static void
|
|
_text_ascent_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *ascent = va_arg(*list, Evas_Coord *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*ascent = o->ascent;
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_descent_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord descent = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_descent_get(&descent));
|
|
return descent;
|
|
}
|
|
|
|
static void
|
|
_text_descent_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *descent = va_arg(*list, Evas_Coord *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*descent = o->descent;
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_max_ascent_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord max_ascent = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_max_ascent_get(&max_ascent));
|
|
return max_ascent;
|
|
}
|
|
|
|
static void
|
|
_text_max_ascent_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *max_ascent = va_arg(*list, Evas_Coord *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*max_ascent = o->max_ascent;
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_max_descent_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord max_descent = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_max_descent_get(&max_descent));
|
|
return max_descent;
|
|
}
|
|
|
|
static void
|
|
_text_max_descent_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *max_descent = va_arg(*list, Evas_Coord *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*max_descent = o->max_descent;
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_inset_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord inset = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_inset_get(&inset));
|
|
return inset;
|
|
}
|
|
|
|
static void
|
|
_text_inset_get(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Coord *inset = va_arg(*list, Evas_Coord *);
|
|
*inset = 0;
|
|
const Evas_Object_Text *o = _pd;
|
|
if (!o->font) return;
|
|
if (!o->items) return;
|
|
*inset = ENFN->font_inset_get(ENDT, o->font, &o->items->text_props);
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_horiz_advance_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord horiz = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_horiz_advance_get(&horiz));
|
|
return horiz;
|
|
}
|
|
|
|
static void
|
|
_text_horiz_advance_get(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *horiz = va_arg(*list, Evas_Coord *);
|
|
*horiz = 0;
|
|
const Evas_Object_Text *o = _pd;
|
|
if (!o->font) return;
|
|
if (!o->items) return;
|
|
*horiz = _evas_object_text_horiz_advance_get(eo_obj, o);
|
|
}
|
|
|
|
EAPI Evas_Coord
|
|
evas_object_text_vert_advance_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return 0;
|
|
MAGIC_CHECK_END();
|
|
Evas_Coord vert = 0;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_vert_advance_get(&vert));
|
|
return vert;
|
|
}
|
|
|
|
static void
|
|
_text_vert_advance_get(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord *vert = va_arg(*list, Evas_Coord *);
|
|
*vert = 0;
|
|
const Evas_Object_Text *o = _pd;
|
|
if (!o->font) return;
|
|
if (!o->items)
|
|
{
|
|
*vert = o->ascent + o->descent;
|
|
return;
|
|
}
|
|
*vert = _evas_object_text_vert_advance_get(eo_obj, o);
|
|
}
|
|
|
|
EAPI Eina_Bool
|
|
evas_object_text_char_pos_get(const Evas_Object *eo_obj, int pos, Evas_Coord *cx, Evas_Coord *cy, Evas_Coord *cw, Evas_Coord *ch)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return EINA_FALSE;
|
|
MAGIC_CHECK_END();
|
|
Eina_Bool ret = EINA_FALSE;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_char_pos_get(pos, cx, cy, cw, ch, &ret));
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
_text_char_pos_get(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
int pos = va_arg(*list, int);
|
|
Evas_Coord *cx = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *cy = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *cw = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *ch = va_arg(*list, Evas_Coord *);
|
|
Eina_Bool *ret = va_arg(*list, Eina_Bool *);
|
|
if (ret) *ret = EINA_FALSE;
|
|
|
|
const Evas_Object_Text *o = _pd;
|
|
int l = 0, r = 0, t = 0, b = 0;
|
|
int x = 0, y = 0, w = 0, h = 0;
|
|
|
|
if (!o->font) return;
|
|
if (!o->items || (pos < 0)) return;
|
|
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
|
|
Eina_Bool int_ret = _evas_object_text_char_coords_get(eo_obj, o, (size_t) pos,
|
|
&x, &y, &w, &h);
|
|
evas_text_style_pad_get(o->cur.style, &l, &r, &t, &b);
|
|
y += o->max_ascent - t;
|
|
x -= l;
|
|
if (x < 0)
|
|
{
|
|
w += x;
|
|
x = 0;
|
|
}
|
|
if ((x + w) > obj->cur.geometry.w) w = obj->cur.geometry.w - x;
|
|
if (w < 0) w = 0;
|
|
if (y < 0)
|
|
{
|
|
h += y;
|
|
y = 0;
|
|
}
|
|
if ((y + h) > obj->cur.geometry.h) h = obj->cur.geometry.h - y;
|
|
if (h < 0) h = 0;
|
|
if (cx) *cx = x;
|
|
if (cy) *cy = y;
|
|
if (cw) *cw = w + l + r;
|
|
if (ch) *ch = h + t + b;
|
|
if (ret) *ret = int_ret;
|
|
}
|
|
|
|
|
|
EAPI int
|
|
evas_object_text_last_up_to_pos(const Evas_Object *eo_obj, Evas_Coord x, Evas_Coord y)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return -1;
|
|
MAGIC_CHECK_END();
|
|
int res = -1;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_last_up_to_pos(x, y, &res));
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
_text_last_up_to_pos(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord x = va_arg(*list, Evas_Coord);
|
|
Evas_Coord y = va_arg(*list, Evas_Coord);
|
|
int *ret = va_arg(*list, int *);
|
|
*ret = -1;
|
|
|
|
const Evas_Object_Text *o = _pd;
|
|
|
|
if (!o->font) return;
|
|
if (!o->items) return;
|
|
int int_ret = _evas_object_text_last_up_to_pos(eo_obj, o, x, y - o->max_ascent);
|
|
if (ret) *ret = int_ret;
|
|
}
|
|
|
|
EAPI int
|
|
evas_object_text_char_coords_get(const Evas_Object *eo_obj, Evas_Coord x, Evas_Coord y, Evas_Coord *cx, Evas_Coord *cy, Evas_Coord *cw, Evas_Coord *ch)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return -1;
|
|
MAGIC_CHECK_END();
|
|
int res = -1;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_char_coords_get(x, y, cx, cy, cw, ch, &res));
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
_text_char_coords_get(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Coord x = va_arg(*list, Evas_Coord);
|
|
Evas_Coord y = va_arg(*list, Evas_Coord);
|
|
Evas_Coord *cx = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *cy = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *cw = va_arg(*list, Evas_Coord *);
|
|
Evas_Coord *ch = va_arg(*list, Evas_Coord *);
|
|
int *ret = va_arg(*list, int *);
|
|
if (ret) *ret = -1;
|
|
|
|
const Evas_Object_Text *o = _pd;
|
|
int l = 0, r = 0, t = 0, b = 0;
|
|
int rx = 0, ry = 0, rw = 0, rh = 0;
|
|
|
|
if (!o->font) return;
|
|
if (!o->items) return;
|
|
|
|
int int_ret = _evas_object_text_char_at_coords(eo_obj, o, x, y - o->max_ascent,
|
|
&rx, &ry, &rw, &rh);
|
|
evas_text_style_pad_get(o->cur.style, &l, &r, &t, &b);
|
|
ry += o->max_ascent - t;
|
|
rx -= l;
|
|
if (rx < 0)
|
|
{
|
|
rw += rx;
|
|
rx = 0;
|
|
}
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
if ((rx + rw) > obj->cur.geometry.w) rw = obj->cur.geometry.w - rx;
|
|
if (rw < 0) rw = 0;
|
|
if (ry < 0)
|
|
{
|
|
rh += ry;
|
|
ry = 0;
|
|
}
|
|
if ((ry + rh) > obj->cur.geometry.h) rh = obj->cur.geometry.h - ry;
|
|
if (rh < 0) rh = 0;
|
|
if (cx) *cx = rx;
|
|
if (cy) *cy = ry;
|
|
if (cw) *cw = rw + l + r;
|
|
if (ch) *ch = rh + t + b;
|
|
if (ret) *ret = int_ret;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_style_set(Evas_Object *eo_obj, Evas_Text_Style_Type style)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_style_set(style));
|
|
}
|
|
|
|
static void
|
|
_text_style_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
Evas_Text_Style_Type style = va_arg(*list, Evas_Text_Style_Type);
|
|
Evas_Object_Text *o = _pd;
|
|
int pl = 0, pr = 0, pt = 0, pb = 0, l = 0, r = 0, t = 0, b = 0;
|
|
|
|
if (o->cur.style == style) return;
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
|
|
evas_text_style_pad_get(o->cur.style, &pl, &pr, &pt, &pb);
|
|
o->cur.style = style;
|
|
evas_text_style_pad_get(o->cur.style, &l, &r, &t, &b);
|
|
if (o->items)
|
|
obj->cur.geometry.w += (l - pl) + (r - pr);
|
|
else
|
|
obj->cur.geometry.w = 0;
|
|
obj->cur.geometry.h += (t - pt) + (b - pb);
|
|
evas_object_change(eo_obj, obj);
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
}
|
|
|
|
EAPI Evas_Text_Style_Type
|
|
evas_object_text_style_get(const Evas_Object *eo_obj)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return EVAS_TEXT_STYLE_PLAIN;
|
|
MAGIC_CHECK_END();
|
|
Evas_Text_Style_Type style = EVAS_TEXT_STYLE_PLAIN;
|
|
eo_do((Eo *)eo_obj, evas_obj_text_style_get(&style));
|
|
return style;
|
|
}
|
|
|
|
static void
|
|
_text_style_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
Evas_Text_Style_Type *style = va_arg(*list, Evas_Text_Style_Type *);
|
|
const Evas_Object_Text *o = _pd;
|
|
*style = o->cur.style;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_shadow_color_set(Evas_Object *eo_obj, int r, int g, int b, int a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_shadow_color_set(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_shadow_color_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
int r = va_arg(*list, int);
|
|
int g = va_arg(*list, int);
|
|
int b = va_arg(*list, int);
|
|
int a = va_arg(*list, int);
|
|
Evas_Object_Text *o = _pd;
|
|
|
|
if ((o->cur.shadow.r == r) && (o->cur.shadow.g == g) &&
|
|
(o->cur.shadow.b == b) && (o->cur.shadow.a == a))
|
|
return;
|
|
o->cur.shadow.r = r;
|
|
o->cur.shadow.g = g;
|
|
o->cur.shadow.b = b;
|
|
o->cur.shadow.a = a;
|
|
o->changed = 1;
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_change(eo_obj, obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_shadow_color_get(const Evas_Object *eo_obj, int *r, int *g, int *b, int *a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_shadow_color_get(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_shadow_color_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
int *r = va_arg(*list, int *);
|
|
int *g = va_arg(*list, int *);
|
|
int *b = va_arg(*list, int *);
|
|
int *a = va_arg(*list, int *);
|
|
const Evas_Object_Text *o = _pd;
|
|
|
|
if (r) *r = o->cur.shadow.r;
|
|
if (g) *g = o->cur.shadow.g;
|
|
if (b) *b = o->cur.shadow.b;
|
|
if (a) *a = o->cur.shadow.a;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_glow_color_set(Evas_Object *eo_obj, int r, int g, int b, int a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_glow_color_set(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_glow_color_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
int r = va_arg(*list, int);
|
|
int g = va_arg(*list, int);
|
|
int b = va_arg(*list, int);
|
|
int a = va_arg(*list, int);
|
|
Evas_Object_Text *o = _pd;
|
|
|
|
if ((o->cur.glow.r == r) && (o->cur.glow.g == g) &&
|
|
(o->cur.glow.b == b) && (o->cur.glow.a == a))
|
|
return;
|
|
o->cur.glow.r = r;
|
|
o->cur.glow.g = g;
|
|
o->cur.glow.b = b;
|
|
o->cur.glow.a = a;
|
|
o->changed = 1;
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_change(eo_obj, obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_glow_color_get(const Evas_Object *eo_obj, int *r, int *g, int *b, int *a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
if (r) *r = 0;
|
|
if (g) *g = 0;
|
|
if (b) *b = 0;
|
|
if (a) *a = 0;
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_glow_color_get(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_glow_color_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
int *r = va_arg(*list, int *);
|
|
int *g = va_arg(*list, int *);
|
|
int *b = va_arg(*list, int *);
|
|
int *a = va_arg(*list, int *);
|
|
const Evas_Object_Text *o = _pd;
|
|
if (r) *r = o->cur.glow.r;
|
|
if (g) *g = o->cur.glow.g;
|
|
if (b) *b = o->cur.glow.b;
|
|
if (a) *a = o->cur.glow.a;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_glow2_color_set(Evas_Object *eo_obj, int r, int g, int b, int a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_glow2_color_set(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_glow2_color_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
int r = va_arg(*list, int);
|
|
int g = va_arg(*list, int);
|
|
int b = va_arg(*list, int);
|
|
int a = va_arg(*list, int);
|
|
Evas_Object_Text *o = _pd;
|
|
|
|
if ((o->cur.glow2.r == r) && (o->cur.glow2.g == g) &&
|
|
(o->cur.glow2.b == b) && (o->cur.glow2.a == a))
|
|
return;
|
|
o->cur.glow2.r = r;
|
|
o->cur.glow2.g = g;
|
|
o->cur.glow2.b = b;
|
|
o->cur.glow2.a = a;
|
|
o->changed = 1;
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_change(eo_obj, obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_glow2_color_get(const Evas_Object *eo_obj, int *r, int *g, int *b, int *a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
if (r) *r = 0;
|
|
if (g) *g = 0;
|
|
if (b) *b = 0;
|
|
if (a) *a = 0;
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_glow2_color_get(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_glow2_color_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
int *r = va_arg(*list, int *);
|
|
int *g = va_arg(*list, int *);
|
|
int *b = va_arg(*list, int *);
|
|
int *a = va_arg(*list, int *);
|
|
const Evas_Object_Text *o = _pd;
|
|
if (r) *r = o->cur.glow2.r;
|
|
if (g) *g = o->cur.glow2.g;
|
|
if (b) *b = o->cur.glow2.b;
|
|
if (a) *a = o->cur.glow2.a;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_outline_color_set(Evas_Object *eo_obj, int r, int g, int b, int a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do(eo_obj, evas_obj_text_outline_color_set(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_outline_color_set(Eo *eo_obj, void *_pd, va_list *list)
|
|
{
|
|
int r = va_arg(*list, int);
|
|
int g = va_arg(*list, int);
|
|
int b = va_arg(*list, int);
|
|
int a = va_arg(*list, int);
|
|
Evas_Object_Text *o = _pd;
|
|
if ((o->cur.outline.r == r) && (o->cur.outline.g == g) &&
|
|
(o->cur.outline.b == b) && (o->cur.outline.a == a))
|
|
return;
|
|
o->cur.outline.r = r;
|
|
o->cur.outline.g = g;
|
|
o->cur.outline.b = b;
|
|
o->cur.outline.a = a;
|
|
o->changed = 1;
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_change(eo_obj, obj);
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_outline_color_get(const Evas_Object *eo_obj, int *r, int *g, int *b, int *a)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ)
|
|
if (r) *r = 0;
|
|
if (g) *g = 0;
|
|
if (b) *b = 0;
|
|
if (a) *a = 0;
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_outline_color_get(r, g, b, a));
|
|
}
|
|
|
|
static void
|
|
_text_outline_color_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
int *r = va_arg(*list, int *);
|
|
int *g = va_arg(*list, int *);
|
|
int *b = va_arg(*list, int *);
|
|
int *a = va_arg(*list, int *);
|
|
const Evas_Object_Text *o = _pd;
|
|
if (r) *r = o->cur.outline.r;
|
|
if (g) *g = o->cur.outline.g;
|
|
if (b) *b = o->cur.outline.b;
|
|
if (a) *a = o->cur.outline.a;
|
|
}
|
|
|
|
EAPI void
|
|
evas_object_text_style_pad_get(const Evas_Object *eo_obj, int *l, int *r, int *t, int *b)
|
|
{
|
|
MAGIC_CHECK(eo_obj, Evas_Object, MAGIC_OBJ);
|
|
if (l) *l = 0;
|
|
if (r) *r = 0;
|
|
if (t) *t = 0;
|
|
if (b) *b = 0;
|
|
return;
|
|
MAGIC_CHECK_END();
|
|
eo_do((Eo *)eo_obj, evas_obj_text_style_pad_get(l, r, t, b));
|
|
}
|
|
|
|
static void
|
|
_text_style_pad_get(Eo *eo_obj EINA_UNUSED, void *_pd, va_list *list)
|
|
{
|
|
int *l = va_arg(*list, int *);
|
|
int *r = va_arg(*list, int *);
|
|
int *t = va_arg(*list, int *);
|
|
int *b = va_arg(*list, int *);
|
|
int sl = 0, sr = 0, st = 0, sb = 0;
|
|
const Evas_Object_Text *o = _pd;
|
|
/* use temps to be certain we have initialized values */
|
|
evas_text_style_pad_get(o->cur.style, &sl, &sr, &st, &sb);
|
|
if (l) *l = sl;
|
|
if (r) *r = sr;
|
|
if (t) *t = st;
|
|
if (b) *b = sb;
|
|
}
|
|
|
|
|
|
|
|
|
|
EAPI int
|
|
evas_string_char_next_get(const char *str, int pos, int *decoded)
|
|
{
|
|
int p, d;
|
|
|
|
if (decoded) *decoded = 0;
|
|
if ((!str) || (pos < 0)) return 0;
|
|
p = pos;
|
|
d = eina_unicode_utf8_get_next(str, &p);
|
|
if (decoded) *decoded = d;
|
|
return p;
|
|
}
|
|
|
|
EAPI int
|
|
evas_string_char_prev_get(const char *str, int pos, int *decoded)
|
|
{
|
|
int p, d;
|
|
|
|
if (decoded) *decoded = 0;
|
|
if ((!str) || (pos < 1)) return 0;
|
|
p = pos;
|
|
d = eina_unicode_utf8_get_prev(str, &p);
|
|
if (decoded) *decoded = d;
|
|
return p;
|
|
}
|
|
|
|
EAPI int
|
|
evas_string_char_len_get(const char *str)
|
|
{
|
|
if (!str) return 0;
|
|
return eina_unicode_utf8_get_len(str);
|
|
}
|
|
|
|
void
|
|
evas_text_style_pad_get(Evas_Text_Style_Type style, int *l, int *r, int *t, int *b)
|
|
{
|
|
int sl = 0, sr = 0, st = 0, sb = 0;
|
|
|
|
/* Don't calc anything if there's no style. */
|
|
if (style != EVAS_TEXT_STYLE_PLAIN)
|
|
{
|
|
int shad_sz = 0, shad_dst = 0, out_sz = 0;
|
|
int dx = 0, minx = 0, maxx = 0;
|
|
int dy = 0, miny = 0, maxy = 0;
|
|
Eina_Bool have_shadow = EINA_FALSE;
|
|
|
|
switch (style & EVAS_TEXT_STYLE_MASK_BASIC)
|
|
{
|
|
case EVAS_TEXT_STYLE_SHADOW:
|
|
shad_dst = 1;
|
|
have_shadow = EINA_TRUE;
|
|
break;
|
|
case EVAS_TEXT_STYLE_OUTLINE_SHADOW:
|
|
case EVAS_TEXT_STYLE_FAR_SHADOW:
|
|
shad_dst = 2;
|
|
out_sz = 1;
|
|
have_shadow = EINA_TRUE;
|
|
break;
|
|
case EVAS_TEXT_STYLE_OUTLINE_SOFT_SHADOW:
|
|
shad_dst = 1;
|
|
shad_sz = 2;
|
|
out_sz = 1;
|
|
have_shadow = EINA_TRUE;
|
|
break;
|
|
case EVAS_TEXT_STYLE_FAR_SOFT_SHADOW:
|
|
shad_dst = 2;
|
|
shad_sz = 2;
|
|
have_shadow = EINA_TRUE;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SOFT_SHADOW:
|
|
shad_dst = 1;
|
|
shad_sz = 2;
|
|
have_shadow = EINA_TRUE;
|
|
break;
|
|
case EVAS_TEXT_STYLE_GLOW:
|
|
case EVAS_TEXT_STYLE_SOFT_OUTLINE:
|
|
out_sz = 2;
|
|
break;
|
|
case EVAS_TEXT_STYLE_OUTLINE:
|
|
out_sz = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
minx = -out_sz;
|
|
maxx = out_sz;
|
|
miny = -out_sz;
|
|
maxy = out_sz;
|
|
if (have_shadow)
|
|
{
|
|
int shx1, shx2, shy1, shy2;
|
|
switch (style & EVAS_TEXT_STYLE_MASK_SHADOW_DIRECTION)
|
|
{
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM_RIGHT:
|
|
dx = 1;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM:
|
|
dx = 0;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM_LEFT:
|
|
dx = -1;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_LEFT:
|
|
dx = -1;
|
|
dy = 0;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP_LEFT:
|
|
dx = -1;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP:
|
|
dx = 0;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP_RIGHT:
|
|
dx = 1;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_RIGHT:
|
|
dx = 1;
|
|
dy = 0;
|
|
default:
|
|
break;
|
|
}
|
|
shx1 = dx * shad_dst;
|
|
shx1 -= shad_sz;
|
|
shx2 = dx * shad_dst;
|
|
shx2 += shad_sz;
|
|
if (shx1 < minx) minx = shx1;
|
|
if (shx2 > maxx) maxx = shx2;
|
|
|
|
shy1 = dy * shad_dst;
|
|
shy1 -= shad_sz;
|
|
shy2 = dy * shad_dst;
|
|
shy2 += shad_sz;
|
|
if (shy1 < miny) miny = shy1;
|
|
if (shy2 > maxy) maxy = shy2;
|
|
}
|
|
|
|
if (l) sl = *l;
|
|
if (r) sr = *r;
|
|
if (t) st = *t;
|
|
if (b) sb = *b;
|
|
|
|
if (sr < maxx) sr = maxx;
|
|
if (sl < -minx) sl = -minx;
|
|
if (sb < maxy) sb = maxy;
|
|
if (st < -miny) st = -miny;
|
|
}
|
|
|
|
if (l) *l = sl;
|
|
if (r) *r = sr;
|
|
if (t) *t = st;
|
|
if (b) *b = sb;
|
|
}
|
|
|
|
/* all nice and private */
|
|
static void
|
|
evas_object_text_init(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
/* set up default settings for this kind of object */
|
|
obj->cur.color.r = 255;
|
|
obj->cur.color.g = 255;
|
|
obj->cur.color.b = 255;
|
|
obj->cur.color.a = 255;
|
|
obj->cur.geometry.x = 0;
|
|
obj->cur.geometry.y = 0;
|
|
obj->cur.geometry.w = 0;
|
|
obj->cur.geometry.h = 0;
|
|
obj->cur.layer = 0;
|
|
/* set up object-specific settings */
|
|
obj->prev = obj->cur;
|
|
/* set up methods (compulsory) */
|
|
obj->func = &object_func;
|
|
obj->type = o_type;
|
|
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
/* alloc obj private data */
|
|
o->cur.ellipsis = -1.0;
|
|
o->prev = o->cur;
|
|
#ifdef BIDI_SUPPORT
|
|
o->bidi_par_props = evas_bidi_paragraph_props_new();
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
_destructor(Eo *eo_obj, void *_pd EINA_UNUSED, va_list *list EINA_UNUSED)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
evas_object_text_free(eo_obj, obj);
|
|
eo_do_super(eo_obj, eo_destructor());
|
|
}
|
|
|
|
static void
|
|
evas_object_text_free(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
|
|
/* free obj */
|
|
if (o->items) _evas_object_text_items_clear(o);
|
|
if (o->cur.utf8_text) eina_stringshare_del(o->cur.utf8_text);
|
|
if (o->cur.font) eina_stringshare_del(o->cur.font);
|
|
if (o->cur.fdesc) evas_font_desc_unref(o->cur.fdesc);
|
|
if (o->cur.source) eina_stringshare_del(o->cur.source);
|
|
if (o->cur.text) free(o->cur.text);
|
|
if (o->font) evas_font_free(obj->layer->evas->evas, o->font);
|
|
#ifdef BIDI_SUPPORT
|
|
evas_bidi_paragraph_props_unref(o->bidi_par_props);
|
|
#endif
|
|
}
|
|
|
|
void
|
|
evas_font_draw_async_check(Evas_Object_Protected_Data *obj,
|
|
void *data, void *context, void *surface,
|
|
Evas_Font_Set *font,
|
|
int x, int y, int w, int h, int ow, int oh,
|
|
Evas_Text_Props *intl_props, Eina_Bool do_async)
|
|
{
|
|
Eina_Bool async_unref;
|
|
|
|
async_unref = obj->layer->evas->engine.func->font_draw(data, context, surface,
|
|
font, x, y, w, h, ow, oh,
|
|
intl_props, do_async);
|
|
if (do_async && async_unref)
|
|
{
|
|
evas_common_font_glyphs_ref(intl_props->glyphs);
|
|
evas_unref_queue_glyph_put(obj->layer->evas, intl_props->glyphs);
|
|
}
|
|
}
|
|
|
|
static void
|
|
evas_object_text_render(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj, void *output, void *context, void *surface, int x, int y, Eina_Bool do_async)
|
|
{
|
|
int i, j;
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
Evas_Object_Text_Item *it;
|
|
const char vals[5][5] =
|
|
{
|
|
{0, 1, 2, 1, 0},
|
|
{1, 3, 4, 3, 1},
|
|
{2, 4, 5, 4, 2},
|
|
{1, 3, 4, 3, 1},
|
|
{0, 1, 2, 1, 0}
|
|
};
|
|
int sl = 0, st = 0;
|
|
int shad_dst, shad_sz, dx, dy, haveshad;
|
|
|
|
/* render object to surface with context, and offxet by x,y */
|
|
evas_text_style_pad_get(o->cur.style, &sl, NULL, &st, NULL);
|
|
ENFN->context_multiplier_unset(output, context);
|
|
ENFN->context_render_op_set(output, context, obj->cur.render_op);
|
|
/* FIXME: This clipping is just until we fix inset handling correctly. */
|
|
ENFN->context_clip_clip(output, context,
|
|
obj->cur.geometry.x + x,
|
|
obj->cur.geometry.y + y,
|
|
obj->cur.geometry.w,
|
|
obj->cur.geometry.h);
|
|
/*
|
|
ENFN->context_color_set(output,
|
|
context,
|
|
230, 160, 30, 100);
|
|
ENFN->rectangle_draw(output,
|
|
context,
|
|
surface,
|
|
obj->cur.geometry.x + x,
|
|
obj->cur.geometry.y + y,
|
|
obj->cur.geometry.w,
|
|
obj->cur.geometry.h);
|
|
*/
|
|
#define COLOR_ONLY_SET(object, sub, col) \
|
|
ENFN->context_color_set(output, context, \
|
|
object->sub.col.r, \
|
|
object->sub.col.g, \
|
|
object->sub.col.b, \
|
|
object->sub.col.a);
|
|
|
|
#define COLOR_SET(object, sub, col) \
|
|
if (obj->cur.clipper)\
|
|
{ \
|
|
ENFN->context_color_set(output, context, \
|
|
((int)object->sub.col.r * ((int)obj->cur.clipper->cur.cache.clip.r + 1)) >> 8, \
|
|
((int)object->sub.col.g * ((int)obj->cur.clipper->cur.cache.clip.g + 1)) >> 8, \
|
|
((int)object->sub.col.b * ((int)obj->cur.clipper->cur.cache.clip.b + 1)) >> 8, \
|
|
((int)object->sub.col.a * ((int)obj->cur.clipper->cur.cache.clip.a + 1)) >> 8); \
|
|
} \
|
|
else\
|
|
ENFN->context_color_set(output, context, \
|
|
object->sub.col.r, \
|
|
object->sub.col.g, \
|
|
object->sub.col.b, \
|
|
object->sub.col.a);
|
|
|
|
#define COLOR_SET_AMUL(object, sub, col, amul) \
|
|
if (obj->cur.clipper) \
|
|
{ \
|
|
ENFN->context_color_set(output, context, \
|
|
(((int)object->sub.col.r) * ((int)obj->cur.clipper->cur.cache.clip.r) * (amul)) / 65025, \
|
|
(((int)object->sub.col.g) * ((int)obj->cur.clipper->cur.cache.clip.g) * (amul)) / 65025, \
|
|
(((int)object->sub.col.b) * ((int)obj->cur.clipper->cur.cache.clip.b) * (amul)) / 65025, \
|
|
(((int)object->sub.col.a) * ((int)obj->cur.clipper->cur.cache.clip.a) * (amul)) / 65025); \
|
|
} \
|
|
else \
|
|
ENFN->context_color_set(output, context, \
|
|
(((int)object->sub.col.r) * (amul)) / 255, \
|
|
(((int)object->sub.col.g) * (amul)) / 255, \
|
|
(((int)object->sub.col.b) * (amul)) / 255, \
|
|
(((int)object->sub.col.a) * (amul)) / 255);
|
|
|
|
#define DRAW_TEXT(ox, oy) \
|
|
if ((o->font) && (it->text_props.len > 0)) \
|
|
evas_font_draw_async_check(obj, output, \
|
|
context, \
|
|
surface, \
|
|
o->font, \
|
|
obj->cur.geometry.x + x + sl + ox + it->x, \
|
|
obj->cur.geometry.y + y + st + oy + \
|
|
(int) \
|
|
(((o->max_ascent * obj->cur.geometry.h) / obj->cur.geometry.h)), \
|
|
obj->cur.geometry.w, \
|
|
obj->cur.geometry.h, \
|
|
obj->cur.geometry.w, \
|
|
obj->cur.geometry.h, \
|
|
&it->text_props, \
|
|
do_async);
|
|
|
|
/* shadows */
|
|
shad_dst = shad_sz = dx = dy = haveshad = 0;
|
|
switch (o->cur.style & EVAS_TEXT_STYLE_MASK_BASIC)
|
|
{
|
|
case EVAS_TEXT_STYLE_SHADOW:
|
|
case EVAS_TEXT_STYLE_OUTLINE_SOFT_SHADOW:
|
|
shad_dst = 1;
|
|
haveshad = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_OUTLINE_SHADOW:
|
|
case EVAS_TEXT_STYLE_FAR_SHADOW:
|
|
shad_dst = 2;
|
|
haveshad = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_FAR_SOFT_SHADOW:
|
|
shad_dst = 2;
|
|
shad_sz = 2;
|
|
haveshad = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SOFT_SHADOW:
|
|
shad_dst = 1;
|
|
shad_sz = 2;
|
|
haveshad = 1;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (haveshad)
|
|
{
|
|
if (shad_dst > 0)
|
|
{
|
|
switch (o->cur.style & EVAS_TEXT_STYLE_MASK_SHADOW_DIRECTION)
|
|
{
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM_RIGHT:
|
|
dx = 1;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM:
|
|
dx = 0;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_BOTTOM_LEFT:
|
|
dx = -1;
|
|
dy = 1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_LEFT:
|
|
dx = -1;
|
|
dy = 0;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP_LEFT:
|
|
dx = -1;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP:
|
|
dx = 0;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_TOP_RIGHT:
|
|
dx = 1;
|
|
dy = -1;
|
|
break;
|
|
case EVAS_TEXT_STYLE_SHADOW_DIRECTION_RIGHT:
|
|
dx = 1;
|
|
dy = 0;
|
|
default:
|
|
break;
|
|
}
|
|
dx *= shad_dst;
|
|
dy *= shad_dst;
|
|
}
|
|
}
|
|
EINA_INLIST_FOREACH(EINA_INLIST_GET(o->items), it)
|
|
{
|
|
/* Shadows */
|
|
if (haveshad)
|
|
{
|
|
switch (shad_sz)
|
|
{
|
|
case 0:
|
|
COLOR_SET(o, cur, shadow);
|
|
DRAW_TEXT(dx, dy);
|
|
break;
|
|
case 2:
|
|
for (j = 0; j < 5; j++)
|
|
{
|
|
for (i = 0; i < 5; i++)
|
|
{
|
|
if (vals[i][j] != 0)
|
|
{
|
|
COLOR_SET_AMUL(o, cur, shadow, vals[i][j] * 50);
|
|
DRAW_TEXT(i - 2 + dx, j - 2 + dy);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* glows */
|
|
if (o->cur.style == EVAS_TEXT_STYLE_GLOW)
|
|
{
|
|
for (j = 0; j < 5; j++)
|
|
{
|
|
for (i = 0; i < 5; i++)
|
|
{
|
|
if (vals[i][j] != 0)
|
|
{
|
|
COLOR_SET_AMUL(o, cur, glow, vals[i][j] * 50);
|
|
DRAW_TEXT(i - 2, j - 2);
|
|
}
|
|
}
|
|
}
|
|
COLOR_SET(o, cur, glow2);
|
|
DRAW_TEXT(-1, 0);
|
|
DRAW_TEXT(1, 0);
|
|
DRAW_TEXT(0, -1);
|
|
DRAW_TEXT(0, 1);
|
|
}
|
|
|
|
/* outlines */
|
|
if ((o->cur.style == EVAS_TEXT_STYLE_OUTLINE) ||
|
|
(o->cur.style == EVAS_TEXT_STYLE_OUTLINE_SHADOW) ||
|
|
(o->cur.style == EVAS_TEXT_STYLE_OUTLINE_SOFT_SHADOW))
|
|
{
|
|
COLOR_SET(o, cur, outline);
|
|
DRAW_TEXT(-1, 0);
|
|
DRAW_TEXT(1, 0);
|
|
DRAW_TEXT(0, -1);
|
|
DRAW_TEXT(0, 1);
|
|
}
|
|
else if (o->cur.style == EVAS_TEXT_STYLE_SOFT_OUTLINE)
|
|
{
|
|
for (j = 0; j < 5; j++)
|
|
{
|
|
for (i = 0; i < 5; i++)
|
|
{
|
|
if (((i != 2) || (j != 2)) && (vals[i][j] != 0))
|
|
{
|
|
COLOR_SET_AMUL(o, cur, outline, vals[i][j] * 50);
|
|
DRAW_TEXT(i - 2, j - 2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* normal text */
|
|
COLOR_ONLY_SET(obj, cur.cache, clip);
|
|
DRAW_TEXT(0, 0);
|
|
}
|
|
}
|
|
|
|
static void
|
|
evas_object_text_render_pre(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
int is_v = 0, was_v = 0;
|
|
/* dont pre-render the obj twice! */
|
|
if (obj->pre_render_done) return;
|
|
obj->pre_render_done = EINA_TRUE;
|
|
/* pre-render phase. this does anything an object needs to do just before
|
|
rendering. This could mean loading the image data, retrieving it from
|
|
elsewhere, decoding video etc.
|
|
Then when this is done the object needs to figure if it changed and
|
|
if so what and where and add the appropriate redraw rectangles */
|
|
/* if someone is clipping this obj - go calculate the clipper */
|
|
if (obj->cur.clipper)
|
|
{
|
|
if (obj->cur.cache.clip.dirty)
|
|
evas_object_clip_recalc(obj->cur.eo_clipper, obj->cur.clipper);
|
|
obj->cur.clipper->func->render_pre(obj->cur.eo_clipper, obj->cur.clipper);
|
|
}
|
|
/* If object size changed and ellipsis is set */
|
|
if (((o->cur.ellipsis >= 0.0 ||
|
|
o->cur.ellipsis != o->prev.ellipsis) &&
|
|
((obj->cur.geometry.w != o->last_computed.w) ||
|
|
(obj->cur.geometry.h != o->last_computed.h))) ||
|
|
(obj->cur.scale != obj->prev.scale))
|
|
{
|
|
_evas_object_text_recalc(eo_obj, o->cur.text);
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
/* now figure what changed and add draw rects
|
|
if it just became visible or invisible */
|
|
is_v = evas_object_is_visible(eo_obj, obj);
|
|
was_v = evas_object_was_visible(eo_obj, obj);
|
|
if (is_v != was_v)
|
|
{
|
|
evas_object_render_pre_visible_change(&obj->layer->evas->clip_changes,
|
|
eo_obj, is_v, was_v);
|
|
goto done;
|
|
}
|
|
if (obj->changed_map || obj->changed_src_visible)
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
/* its not visible - we accounted for it appearing or not so just abort */
|
|
if (!is_v) goto done;
|
|
/* clipper changed this is in addition to anything else for obj */
|
|
evas_object_render_pre_clipper_change(&obj->layer->evas->clip_changes, eo_obj);
|
|
/* if we restacked (layer or just within a layer) and dont clip anyone */
|
|
if (obj->restack)
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
/* if it changed color */
|
|
if ((obj->cur.color.r != obj->prev.color.r) ||
|
|
(obj->cur.color.g != obj->prev.color.g) ||
|
|
(obj->cur.color.b != obj->prev.color.b) ||
|
|
(obj->cur.color.a != obj->prev.color.a))
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
/* if it changed geometry - and obviously not visibility or color
|
|
calculate differences since we have a constant color fill
|
|
we really only need to update the differences */
|
|
if ((obj->cur.geometry.x != obj->prev.geometry.x) ||
|
|
(obj->cur.geometry.y != obj->prev.geometry.y) ||
|
|
(obj->cur.geometry.w != obj->prev.geometry.w) ||
|
|
(obj->cur.geometry.h != obj->prev.geometry.h))
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
if (obj->cur.render_op != obj->prev.render_op)
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
if (o->changed)
|
|
{
|
|
evas_object_render_pre_prev_cur_add(&obj->layer->evas->clip_changes,
|
|
eo_obj, obj);
|
|
goto done;
|
|
}
|
|
done:
|
|
evas_object_render_pre_effect_updates(&obj->layer->evas->clip_changes,
|
|
eo_obj, is_v, was_v);
|
|
}
|
|
|
|
static void
|
|
evas_object_text_render_post(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj EINA_UNUSED)
|
|
{
|
|
/* this moves the current data to the previous state parts of the object
|
|
in whatever way is safest for the object. also if we don't need object
|
|
data anymore we can free it if the object deems this is a good idea */
|
|
/* remove those pesky changes */
|
|
evas_object_clip_changes_clean(eo_obj);
|
|
/* move cur to prev safely for object data */
|
|
evas_object_cur_prev(eo_obj);
|
|
}
|
|
|
|
static unsigned int
|
|
evas_object_text_id_get(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
if (!o) return 0;
|
|
return MAGIC_OBJ_TEXT;
|
|
}
|
|
|
|
static unsigned int
|
|
evas_object_text_visual_id_get(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
if (!o) return 0;
|
|
return MAGIC_OBJ_SHAPE;
|
|
}
|
|
|
|
static void *
|
|
evas_object_text_engine_data_get(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
if (!o) return NULL;
|
|
return o->font;
|
|
}
|
|
|
|
static int
|
|
evas_object_text_is_opaque(Evas_Object *eo_obj EINA_UNUSED, Evas_Object_Protected_Data *obj EINA_UNUSED)
|
|
{
|
|
/* this returns 1 if the internal object data implies that the object is
|
|
currently fully opaque over the entire gradient it occupies */
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
evas_object_text_was_opaque(Evas_Object *eo_obj EINA_UNUSED, Evas_Object_Protected_Data *obj EINA_UNUSED)
|
|
{
|
|
/* this returns 1 if the internal object data implies that the object was
|
|
currently fully opaque over the entire gradient it occupies */
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
evas_object_text_scale_update(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
int size;
|
|
const char *font;
|
|
|
|
font = eina_stringshare_add(o->cur.font);
|
|
size = o->cur.size;
|
|
if (o->cur.font) eina_stringshare_del(o->cur.font);
|
|
o->cur.font = NULL;
|
|
o->prev.font = NULL;
|
|
o->cur.size = 0;
|
|
o->prev.size = 0;
|
|
evas_object_text_font_set(eo_obj, font, size);
|
|
}
|
|
|
|
void
|
|
_evas_object_text_rehint(Evas_Object *eo_obj)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
int is, was;
|
|
|
|
if (!o->font) return;
|
|
evas_font_load_hinting_set(obj->layer->evas->evas, o->font,
|
|
obj->layer->evas->hinting);
|
|
was = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y, 1, 1);
|
|
/* DO II */
|
|
_evas_object_text_recalc(eo_obj, o->cur.text);
|
|
o->changed = 1;
|
|
evas_object_change(eo_obj, obj);
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
evas_object_coords_recalc(eo_obj, obj);
|
|
is = evas_object_is_in_output_rect(eo_obj, obj,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y, 1, 1);
|
|
if ((is || was) && obj->cur.visible)
|
|
evas_event_feed_mouse_move(obj->layer->evas->evas,
|
|
obj->layer->evas->pointer.x,
|
|
obj->layer->evas->pointer.y,
|
|
obj->layer->evas->last_timestamp,
|
|
NULL);
|
|
evas_object_inform_call_resize(eo_obj);
|
|
}
|
|
|
|
static void
|
|
_evas_object_text_recalc(Evas_Object *eo_obj, Eina_Unicode *text)
|
|
{
|
|
Evas_Object_Protected_Data *obj = eo_data_get(eo_obj, EVAS_OBJ_CLASS);
|
|
Evas_Object_Text *o = eo_data_get(eo_obj, MY_CLASS);
|
|
|
|
if (!text) text = eina_unicode_strdup(EINA_UNICODE_EMPTY_STRING);
|
|
|
|
_evas_object_text_layout(eo_obj, o, text);
|
|
|
|
if ((o->font) && (o->items))
|
|
{
|
|
int w, h;
|
|
int l = 0, r = 0, t = 0, b = 0;
|
|
|
|
w = _evas_object_text_horiz_advance_get(eo_obj, o);
|
|
h = _evas_object_text_vert_advance_get(eo_obj, o);
|
|
evas_text_style_pad_get(o->cur.style, &l, &r, &t, &b);
|
|
obj->cur.geometry.w = w + l + r;
|
|
obj->cur.geometry.h = h + t + b;
|
|
//// obj->cur.cache.geometry.validity = 0;
|
|
}
|
|
else
|
|
{
|
|
int t = 0, b = 0;
|
|
|
|
evas_text_style_pad_get(o->cur.style, NULL, NULL, &t, &b);
|
|
obj->cur.geometry.w = 0;
|
|
obj->cur.geometry.h = o->max_ascent + o->max_descent + t + b;
|
|
//// obj->cur.cache.geometry.validity = 0;
|
|
}
|
|
o->last_computed.w = obj->cur.geometry.w;
|
|
o->last_computed.h = obj->cur.geometry.h;
|
|
evas_object_clip_dirty(eo_obj, obj);
|
|
}
|
|
|
|
static void
|
|
_class_constructor(Eo_Class *klass)
|
|
{
|
|
const Eo_Op_Func_Description func_desc[] = {
|
|
EO_OP_FUNC(EO_BASE_ID(EO_BASE_SUB_ID_CONSTRUCTOR), _constructor),
|
|
EO_OP_FUNC(EO_BASE_ID(EO_BASE_SUB_ID_DESTRUCTOR), _destructor),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_FONT_SOURCE_SET), _text_font_source_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_FONT_SOURCE_GET), _text_font_source_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_FONT_SET), _text_font_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_FONT_GET), _text_font_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_TEXT_SET), _text_text_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_BIDI_DELIMITERS_SET), _text_bidi_delimiters_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_BIDI_DELIMITERS_GET), _text_bidi_delimiters_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_TEXT_GET), _text_text_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_DIRECTION_GET), _text_direction_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_ASCENT_GET), _text_ascent_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_DESCENT_GET), _text_descent_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_MAX_ASCENT_GET), _text_max_ascent_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_MAX_DESCENT_GET), _text_max_descent_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_INSET_GET), _text_inset_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_HORIZ_ADVANCE_GET), _text_horiz_advance_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_VERT_ADVANCE_GET), _text_vert_advance_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_CHAR_POS_GET), _text_char_pos_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_LAST_UP_TO_POS), _text_last_up_to_pos),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_CHAR_COORDS_GET), _text_char_coords_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_STYLE_SET), _text_style_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_STYLE_GET), _text_style_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_SHADOW_COLOR_SET), _text_shadow_color_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_SHADOW_COLOR_GET), _text_shadow_color_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_GLOW_COLOR_SET), _text_glow_color_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_GLOW_COLOR_GET), _text_glow_color_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_GLOW2_COLOR_SET), _text_glow2_color_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_GLOW2_COLOR_GET), _text_glow2_color_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_OUTLINE_COLOR_SET), _text_outline_color_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_OUTLINE_COLOR_GET), _text_outline_color_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_STYLE_PAD_GET), _text_style_pad_get),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_ELLIPSIS_SET), _text_ellipsis_set),
|
|
EO_OP_FUNC(EVAS_OBJ_TEXT_ID(EVAS_OBJ_TEXT_SUB_ID_ELLIPSIS_GET), _text_ellipsis_get),
|
|
EO_OP_FUNC_SENTINEL
|
|
};
|
|
eo_class_funcs_set(klass, func_desc);
|
|
}
|
|
static const Eo_Op_Description op_desc[] = {
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_FONT_SOURCE_SET, "Set the font (source) file to be used on a given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_FONT_SOURCE_GET, "Get the font file's path which is being used on a given text"),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_FONT_SET, "Set the font family and size on a given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_FONT_GET, "Retrieve the font family and size in use on a given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_TEXT_SET, "Sets the text string to be displayed by the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_BIDI_DELIMITERS_SET, "Sets the BiDi delimiters used in the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_BIDI_DELIMITERS_GET, "Gets the BiDi delimiters used in the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_TEXT_GET, "Retrieves the text string currently being displayed by the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_DIRECTION_GET, "Retrieves the direction of the text currently being displayed in the text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_ASCENT_GET, "Get the ascent of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_DESCENT_GET, "Get the descent of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_MAX_ASCENT_GET, "Get the max ascent of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_MAX_DESCENT_GET, "Get the max descent of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_INSET_GET, "Get the inset of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_HORIZ_ADVANCE_GET, "Get the horizontal advance of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_VERT_ADVANCE_GET, "Get the vertical advance of the text."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_CHAR_POS_GET, "Retrieve position and dimension information of a character within a text Evas_Object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_LAST_UP_TO_POS, "Retrieves the logical position of the last char in the text up to the pos given."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_CHAR_COORDS_GET, "? evas_object_text_char_coords_get"),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_STYLE_SET, "Sets the style to apply on the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_STYLE_GET, "Retrieves the style on use on the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_SHADOW_COLOR_SET, "Sets the shadow color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_SHADOW_COLOR_GET, "Retrieves the shadow color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_GLOW_COLOR_SET, "Sets the glow color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_GLOW_COLOR_GET, "Retrieves the glow color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_GLOW2_COLOR_SET, "Sets the 'glow 2' color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_GLOW2_COLOR_GET, "Retrieves the 'glow 2' color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_OUTLINE_COLOR_SET, "Sets the outline color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_OUTLINE_COLOR_GET, "Retrieves the outline color for the given text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_STYLE_PAD_GET, "Gets the text style pad of a text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_ELLIPSIS_SET, "Gets the ellipsis of a text object."),
|
|
EO_OP_DESCRIPTION(EVAS_OBJ_TEXT_SUB_ID_ELLIPSIS_GET, "Sets the ellipsis of a text object."),
|
|
EO_OP_DESCRIPTION_SENTINEL
|
|
};
|
|
static const Eo_Class_Description class_desc = {
|
|
EO_VERSION,
|
|
"Evas_Object_Text",
|
|
EO_CLASS_TYPE_REGULAR,
|
|
EO_CLASS_DESCRIPTION_OPS(&EVAS_OBJ_TEXT_BASE_ID, op_desc, EVAS_OBJ_TEXT_SUB_ID_LAST),
|
|
NULL,
|
|
sizeof(Evas_Object_Text),
|
|
_class_constructor,
|
|
NULL
|
|
};
|
|
|
|
EO_DEFINE_CLASS(evas_object_text_class_get, &class_desc, EVAS_OBJ_CLASS, NULL);
|
|
|