Fixed prevent issues:56236,38820,30368,24310,28344,56426,25511,22275,24788,45558...
[framework/uifw/elementary.git] / src / lib / elm_entry.c
1 #include <Elementary.h>
2 #include <Elementary_Cursor.h>
3 #include "elm_priv.h"
4 #include "elm_module_priv.h"
5 #include "els_scroller.h"
6
7 /* Maximum chunk size to be inserted to the entry at once
8  * FIXME: This size is arbitrary, should probably choose a better size.
9  * Possibly also find a way to set it to a low value for weak computers,
10  * and to a big value for better computers. */
11 #define _CHUNK_SIZE 10000
12
13 typedef struct _Mod_Api Mod_Api;
14
15 typedef struct _Widget_Data Widget_Data;
16 typedef struct _Elm_Entry_Context_Menu_Item Elm_Entry_Context_Menu_Item;
17 typedef struct _Elm_Entry_Item_Provider Elm_Entry_Item_Provider;
18 typedef struct _Elm_Entry_Text_Filter Elm_Entry_Text_Filter;
19
20 struct _Widget_Data
21 {
22    Evas_Object *ent, *scroller, *end, *icon;
23    Evas_Object *bg;
24    Evas_Object *hoversel;
25    Evas_Object *hover;
26    Evas_Object *layout;
27    Evas_Object *list;
28    Evas_Object *mgf_proxy;
29    Evas_Object *mgf_clip;
30    Evas_Object *mgf_bg;
31    Evas_Coord mgf_height;
32    float mgf_scale;
33    int mgf_type;
34    Ecore_Job *deferred_recalc_job;
35    Ecore_Event_Handler *sel_notify_handler;
36    Ecore_Event_Handler *sel_clear_handler;
37    Ecore_Timer *delay_write;
38    /* for deferred appending */
39    Ecore_Idler *append_text_idler;
40    char *append_text_left;
41    int append_text_position;
42    int append_text_len;
43    /* Only for clipboard */
44    const char *cut_sel;
45    const char *text;
46    const char *password_text;
47    Evas_Coord wrap_w;
48    const char *file;
49    Elm_Text_Format format;
50    Evas_Coord lastw, entmw, entmh;
51    Evas_Coord downx, downy;
52    Evas_Coord cx, cy, cw, ch;
53    Eina_List *items;
54    Eina_List *item_providers;
55    Eina_List *text_filters;
56    Eina_List *match_list;
57    Ecore_Job *matchlist_job;
58    int matchlist_threshold;
59    Ecore_Job *hovdeljob;
60    Mod_Api *api; // module api if supplied
61    int cursor_pos;
62    Elm_Scroller_Policy policy_h, policy_v;
63    Elm_Wrap_Type linewrap;
64    Elm_Input_Panel_Layout input_panel_layout;
65    Elm_Autocapital_Type autocapital_type;
66    Eina_Bool changed : 1;
67    Eina_Bool single_line : 1;
68    Eina_Bool password : 1;
69    Eina_Bool editable : 1;
70    Eina_Bool selection_asked : 1;
71    Eina_Bool have_selection : 1;
72    Eina_Bool selmode : 1;
73    Eina_Bool deferred_cur : 1;
74    Eina_Bool cur_changed : 1;
75    Eina_Bool disabled : 1;
76    Eina_Bool double_clicked : 1;
77    Eina_Bool long_pressed : 1;
78    Eina_Bool context_menu : 1;
79    Eina_Bool drag_selection_asked : 1;
80    Eina_Bool bgcolor : 1;
81    Eina_Bool can_write : 1;
82    Eina_Bool autosave : 1;
83    Eina_Bool textonly : 1;
84    Eina_Bool usedown : 1;
85    Eina_Bool scroll : 1;
86    Eina_Bool input_panel_enable : 1;
87    Eina_Bool autoreturnkey : 1;
88    Eina_Bool autoperiod : 1;
89    Eina_Bool matchlist_list_clicked : 1;
90    Eina_Bool matchlist_case_sensitive : 1;
91 };
92
93 struct _Elm_Entry_Context_Menu_Item
94 {
95    Evas_Object *obj;
96    const char *label;
97    const char *icon_file;
98    const char *icon_group;
99    Elm_Icon_Type icon_type;
100    Evas_Smart_Cb func;
101    void *data;
102 };
103
104 struct _Elm_Entry_Item_Provider
105 {
106    Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item);
107    void *data;
108 };
109
110 struct _Elm_Entry_Text_Filter
111 {
112    void (*func) (void *data, Evas_Object *entry, char **text);
113    void *data;
114 };
115
116 typedef enum _Length_Unit
117 {
118    LENGTH_UNIT_CHAR,
119    LENGTH_UNIT_BYTE,
120    LENGTH_UNIT_LAST
121 } Length_Unit;
122
123 static const char *widtype = NULL;
124 // start for cbhm
125 static Evas_Object *cnpwidgetdata = NULL;
126 // end for cbhm
127
128 #ifdef HAVE_ELEMENTARY_X
129 static Eina_Bool _drag_drop_cb(void *data, Evas_Object *obj, Elm_Selection_Data *);
130 #endif
131 static void _del_hook(Evas_Object *obj);
132 static void _mirrored_set(Evas_Object *obj, Eina_Bool rtl);
133 static void _theme_hook(Evas_Object *obj);
134 static void _disable_hook(Evas_Object *obj);
135 static void _sizing_eval(Evas_Object *obj);
136 static void _on_focus_hook(void *data, Evas_Object *obj);
137 static void _content_set_hook(Evas_Object *obj, const char *part, Evas_Object *content);
138 static Evas_Object *_content_unset_hook(Evas_Object *obj, const char *part);
139 static Evas_Object *_content_get_hook(const Evas_Object *obj, const char *part);
140 static void _resize(void *data, Evas *e, Evas_Object *obj, void *event_info);
141 static const char *_getbase(Evas_Object *obj);
142 static void _signal_entry_changed(void *data, Evas_Object *obj, const char *emission, const char *source);
143 static void _signal_selection_start(void *data, Evas_Object *obj, const char *emission, const char *source);
144 static void _signal_selection_end(void *data, Evas_Object *obj, const char *emission, const char *source);
145 static void _signal_selection_changed(void *data, Evas_Object *obj, const char *emission, const char *source);
146 static void _signal_selection_cleared(void *data, Evas_Object *obj, const char *emission, const char *source);
147 static void _signal_handler_move_start(void *data, Evas_Object *obj, const char *emission, const char *source);
148 static void _signal_handler_move_end(void *data, Evas_Object *obj, const char *emission, const char *source);
149 static void _signal_handler_moving(void *data, Evas_Object *obj, const char *emission, const char *source);
150 static void _signal_entry_paste_request(void *data, Evas_Object *obj, const char *emission, const char *source);
151 static void _signal_entry_copy_notify(void *data, Evas_Object *obj, const char *emission, const char *source);
152 static void _signal_entry_cut_notify(void *data, Evas_Object *obj, const char *emission, const char *source);
153 static void _signal_cursor_changed(void *data, Evas_Object *obj, const char *emission, const char *source);
154 static void _add_chars_till_limit(Evas_Object *obj, char **text, int can_add, Length_Unit unit);
155 static int _strbuf_key_value_replace(Eina_Strbuf *srcbuf, char *key, const char *value, int deleteflag);
156 static int _stringshare_key_value_replace(const char **srcstring, char *key, const char *value, int deleteflag);
157 static int _entry_length_get(Evas_Object *obj);
158 static void _magnifier_create(void *data);
159 static void _magnifier_show(void *data);
160 static void _magnifier_hide(void *data);
161 static void _magnifier_move(void *data);
162
163 static const char SIG_CHANGED[] = "changed";
164 static const char SIG_ACTIVATED[] = "activated";
165 static const char SIG_PRESS[] = "press";
166 static const char SIG_LONGPRESSED[] = "longpressed";
167 static const char SIG_CLICKED[] = "clicked";
168 static const char SIG_CLICKED_DOUBLE[] = "clicked,double";
169 static const char SIG_FOCUSED[] = "focused";
170 static const char SIG_UNFOCUSED[] = "unfocused";
171 static const char SIG_SELECTION_PASTE[] = "selection,paste";
172 static const char SIG_SELECTION_COPY[] = "selection,copy";
173 static const char SIG_SELECTION_CUT[] = "selection,cut";
174 static const char SIG_SELECTION_START[] = "selection,start";
175 static const char SIG_SELECTION_CHANGED[] = "selection,changed";
176 static const char SIG_SELECTION_CLEARED[] = "selection,cleared";
177 static const char SIG_CURSOR_CHANGED[] = "cursor,changed";
178 static const char SIG_ANCHOR_CLICKED[] = "anchor,clicked";
179 static const char SIG_MATCHLIST_CLICKED[] = "matchlist,clicked";
180 static const char SIG_PREEDIT_CHANGED[] = "preedit,changed";
181 static const Evas_Smart_Cb_Description _signals[] = {
182   {SIG_CHANGED, ""},
183   {SIG_ACTIVATED, ""},
184   {SIG_PRESS, ""},
185   {SIG_LONGPRESSED, ""},
186   {SIG_CLICKED, ""},
187   {SIG_CLICKED_DOUBLE, ""},
188   {SIG_FOCUSED, ""},
189   {SIG_UNFOCUSED, ""},
190   {SIG_SELECTION_PASTE, ""},
191   {SIG_SELECTION_COPY, ""},
192   {SIG_SELECTION_CUT, ""},
193   {SIG_SELECTION_START, ""},
194   {SIG_SELECTION_CHANGED, ""},
195   {SIG_SELECTION_CLEARED, ""},
196   {SIG_CURSOR_CHANGED, ""},
197   {SIG_ANCHOR_CLICKED, ""},
198   {SIG_PREEDIT_CHANGED, ""},
199   {SIG_MATCHLIST_CLICKED, ""},
200   {NULL, NULL}
201 };
202
203 typedef enum _Elm_Entry_Magnifier_Type
204 {
205    _ENTRY_MAGNIFIER_FIXEDSIZE = 0,
206    _ENTRY_MAGNIFIER_FILLWIDTH,
207    _ENTRY_MAGNIFIER_CIRCULAR,
208 } Elm_Entry_Magnifier_Type;
209
210
211 static Eina_List *entries = NULL;
212
213 struct _Mod_Api
214 {
215    void (*obj_hook) (Evas_Object *obj);
216    void (*obj_unhook) (Evas_Object *obj);
217    void (*obj_longpress) (Evas_Object *obj);
218    void (*obj_hidemenu) (Evas_Object *obj);
219    void (*obj_mouseup) (Evas_Object *obj);
220 };
221
222 static Mod_Api *
223 _module(Evas_Object *obj __UNUSED__)
224 {
225    static Elm_Module *m = NULL;
226    if (m) goto ok; // already found - just use
227    if (!(m = _elm_module_find_as("entry/api"))) return NULL;
228    // get module api
229    m->api = malloc(sizeof(Mod_Api));
230    if (!m->api) return NULL;
231    ((Mod_Api *)(m->api)      )->obj_hook = // called on creation
232       _elm_module_symbol_get(m, "obj_hook");
233    ((Mod_Api *)(m->api)      )->obj_unhook = // called on deletion
234       _elm_module_symbol_get(m, "obj_unhook");
235    ((Mod_Api *)(m->api)      )->obj_longpress = // called on long press menu
236       _elm_module_symbol_get(m, "obj_longpress");
237    ((Mod_Api *)(m->api)      )->obj_hidemenu = // called on hide menu
238       _elm_module_symbol_get(m, "obj_hidemenu");
239    ((Mod_Api *)(m->api)      )->obj_mouseup = // called on mouseup
240       _elm_module_symbol_get(m, "obj_mouseup");
241 ok: // ok - return api
242    return m->api;
243 }
244
245 static char *
246 _buf_append(char *buf, const char *str, int *len, int *alloc)
247 {
248    int len2 = strlen(str);
249    if ((*len + len2) >= *alloc)
250      {
251         char *buf2 = realloc(buf, *alloc + len2 + 512);
252         if (!buf2) return NULL;
253         buf = buf2;
254         *alloc += (512 + len2);
255      }
256    strcpy(buf + *len, str);
257    *len += len2;
258    return buf;
259 }
260
261 static char *
262 _load_file(const char *file)
263 {
264    FILE *f;
265    size_t size;
266    int alloc = 0, len = 0;
267    char *text = NULL, buf[16384 + 1];
268
269    f = fopen(file, "rb");
270    if (!f) return NULL;
271    while ((size = fread(buf, 1, sizeof(buf) - 1, f)))
272      {
273         char *tmp_text;
274         buf[size] = 0;
275         tmp_text = _buf_append(text, buf, &len, &alloc);
276         if (!tmp_text) break;
277         text = tmp_text;
278      }
279    fclose(f);
280    return text;
281 }
282
283 static char *
284 _load_plain(const char *file)
285 {
286    char *text;
287
288    text = _load_file(file);
289    if (text)
290      {
291         char *text2;
292
293         text2 = elm_entry_utf8_to_markup(text);
294         free(text);
295         return text2;
296      }
297    return NULL;
298 }
299
300 static void
301 _load(Evas_Object *obj)
302 {
303    Widget_Data *wd = elm_widget_data_get(obj);
304    char *text;
305    if (!wd) return;
306    if (!wd->file)
307      {
308         elm_entry_entry_set(obj, "");
309         return;
310      }
311    switch (wd->format)
312      {
313       case ELM_TEXT_FORMAT_PLAIN_UTF8:
314          text = _load_plain(wd->file);
315          break;
316       case ELM_TEXT_FORMAT_MARKUP_UTF8:
317          text = _load_file(wd->file);
318          break;
319       default:
320          text = NULL;
321          break;
322      }
323    if (text)
324      {
325         elm_entry_entry_set(obj, text);
326         free(text);
327      }
328    else
329      elm_entry_entry_set(obj, "");
330 }
331
332 static void
333 _save_markup_utf8(const char *file, const char *text)
334 {
335    FILE *f;
336
337    if ((!text) || (!text[0]))
338      {
339         ecore_file_unlink(file);
340         return;
341      }
342    f = fopen(file, "wb");
343    if (!f)
344      {
345         // FIXME: report a write error
346         return;
347      }
348    fputs(text, f); // FIXME: catch error
349    fclose(f);
350 }
351
352 static void
353 _save_plain_utf8(const char *file, const char *text)
354 {
355    char *text2;
356
357    text2 = elm_entry_markup_to_utf8(text);
358    if (!text2)
359      return;
360    _save_markup_utf8(file, text2);
361    free(text2);
362 }
363
364 static void
365 _save(Evas_Object *obj)
366 {
367    Widget_Data *wd = elm_widget_data_get(obj);
368    if (!wd) return;
369    if (!wd->file) return;
370    switch (wd->format)
371      {
372       case ELM_TEXT_FORMAT_PLAIN_UTF8:
373          _save_plain_utf8(wd->file, elm_entry_entry_get(obj));
374          break;
375       case ELM_TEXT_FORMAT_MARKUP_UTF8:
376          _save_markup_utf8(wd->file, elm_entry_entry_get(obj));
377          break;
378       default:
379          break;
380      }
381 }
382
383 static Eina_Bool
384 _delay_write(void *data)
385 {
386    Widget_Data *wd = elm_widget_data_get(data);
387    if (!wd) return ECORE_CALLBACK_CANCEL;
388    _save(data);
389    wd->delay_write = NULL;
390    return ECORE_CALLBACK_CANCEL;
391 }
392
393 static Elm_Entry_Text_Filter *
394 _filter_new(void (*func) (void *data, Evas_Object *entry, char **text), void *data)
395 {
396    Elm_Entry_Text_Filter *tf = ELM_NEW(Elm_Entry_Text_Filter);
397    if (!tf) return NULL;
398
399    tf->func = func;
400    if (func == elm_entry_filter_limit_size)
401      {
402         Elm_Entry_Filter_Limit_Size *lim = data, *lim2;
403
404         if (!data)
405           {
406              free(tf);
407              return NULL;
408           }
409         lim2 = malloc(sizeof(Elm_Entry_Filter_Limit_Size));
410         if (!lim2)
411           {
412              free(tf);
413              return NULL;
414           }
415         memcpy(lim2, lim, sizeof(Elm_Entry_Filter_Limit_Size));
416         tf->data = lim2;
417      }
418    else if (func == elm_entry_filter_accept_set)
419      {
420         Elm_Entry_Filter_Accept_Set *as = data, *as2;
421
422         if (!data)
423           {
424              free(tf);
425              return NULL;
426           }
427         as2 = malloc(sizeof(Elm_Entry_Filter_Accept_Set));
428         if (!as2)
429           {
430              free(tf);
431              return NULL;
432           }
433         if (as->accepted)
434           as2->accepted = eina_stringshare_add(as->accepted);
435         else
436           as2->accepted = NULL;
437         if (as->rejected)
438           as2->rejected = eina_stringshare_add(as->rejected);
439         else
440           as2->rejected = NULL;
441         tf->data = as2;
442      }
443    else
444      tf->data = data;
445    return tf;
446 }
447
448 static void
449 _filter_free(Elm_Entry_Text_Filter *tf)
450 {
451    if (tf->func == elm_entry_filter_limit_size)
452      {
453         Elm_Entry_Filter_Limit_Size *lim = tf->data;
454         if (lim) free(lim);
455      }
456    else if (tf->func == elm_entry_filter_accept_set)
457      {
458         Elm_Entry_Filter_Accept_Set *as = tf->data;
459         if (as)
460           {
461              if (as->accepted) eina_stringshare_del(as->accepted);
462              if (as->rejected) eina_stringshare_del(as->rejected);
463              free(as);
464           }
465      }
466    free(tf);
467 }
468
469 static void
470 _del_pre_hook(Evas_Object *obj)
471 {
472    Widget_Data *wd = elm_widget_data_get(obj);
473    if (!wd) return;
474    if (wd->delay_write)
475      {
476         ecore_timer_del(wd->delay_write);
477         wd->delay_write = NULL;
478         if (wd->autosave) _save(obj);
479      }
480 }
481
482 static void
483 _del_hook(Evas_Object *obj)
484 {
485    Widget_Data *wd = elm_widget_data_get(obj);
486    Elm_Entry_Context_Menu_Item *it;
487    Elm_Entry_Item_Provider *ip;
488    Elm_Entry_Text_Filter *tf;
489
490    if (wd->file) eina_stringshare_del(wd->file);
491
492    if (wd->hovdeljob) ecore_job_del(wd->hovdeljob);
493    if ((wd->api) && (wd->api->obj_unhook)) wd->api->obj_unhook(obj); // module - unhook
494
495    entries = eina_list_remove(entries, obj);
496 #ifdef HAVE_ELEMENTARY_X
497    if (wd->sel_notify_handler)
498      ecore_event_handler_del(wd->sel_notify_handler);
499    if (wd->sel_clear_handler)
500      ecore_event_handler_del(wd->sel_clear_handler);
501 #endif
502    if (wd->cut_sel) eina_stringshare_del(wd->cut_sel);
503    if (wd->text) eina_stringshare_del(wd->text);
504    if (wd->password_text) eina_stringshare_del(wd->password_text);
505    if (wd->bg) evas_object_del(wd->bg);
506    if (wd->deferred_recalc_job) ecore_job_del(wd->deferred_recalc_job);
507    if (wd->append_text_idler)
508      {
509         ecore_idler_del(wd->append_text_idler);
510         free(wd->append_text_left);
511         wd->append_text_left = NULL;
512         wd->append_text_idler = NULL;
513      }
514    if (wd->matchlist_job) ecore_job_del(wd->matchlist_job);
515    if (wd->mgf_proxy) evas_object_del(wd->mgf_proxy);
516    if (wd->mgf_bg) evas_object_del(wd->mgf_bg);
517    if (wd->mgf_clip) evas_object_del(wd->mgf_clip);
518
519    EINA_LIST_FREE(wd->items, it)
520      {
521         eina_stringshare_del(it->label);
522         eina_stringshare_del(it->icon_file);
523         eina_stringshare_del(it->icon_group);
524         free(it);
525      }
526    EINA_LIST_FREE(wd->item_providers, ip)
527      {
528         free(ip);
529      }
530    EINA_LIST_FREE(wd->text_filters, tf)
531      {
532         _filter_free(tf);
533      }
534    if (wd->delay_write) ecore_timer_del(wd->delay_write);
535    free(wd);
536 }
537
538 static void
539 _mirrored_set(Evas_Object *obj, Eina_Bool rtl)
540 {
541    Widget_Data *wd = elm_widget_data_get(obj);
542    edje_object_mirrored_set(wd->ent, rtl);
543 }
544
545 static void
546 _theme_hook(Evas_Object *obj)
547 {
548    Widget_Data *wd = elm_widget_data_get(obj);
549    const char *t;
550    _elm_widget_mirrored_reload(obj);
551    _mirrored_set(obj, elm_widget_mirrored_get(obj));
552
553    t = eina_stringshare_add(elm_entry_entry_get(obj));
554    _elm_theme_object_set(obj, wd->ent, "entry", _getbase(obj), elm_widget_style_get(obj));
555    if (_elm_config->desktop_entry)
556      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
557    elm_entry_entry_set(obj, t);
558    eina_stringshare_del(t);
559    if (elm_widget_disabled_get(obj))
560      edje_object_signal_emit(wd->ent, "elm,state,disabled", "elm");
561    edje_object_part_text_input_panel_layout_set(wd->ent, "elm.text", wd->input_panel_layout);
562    edje_object_part_text_autocapital_type_set(wd->ent, "elm.text", wd->autocapital_type);
563    edje_object_part_text_input_panel_enabled_set(wd->ent, "elm.text", wd->input_panel_enable);
564    elm_entry_cursor_pos_set(obj, wd->cursor_pos);
565    if (elm_widget_focus_get(obj))
566      edje_object_signal_emit(wd->ent, "elm,action,focus", "elm");
567    edje_object_message_signal_process(wd->ent);
568    edje_object_scale_set(wd->ent, elm_widget_scale_get(obj) * _elm_config->scale);
569    elm_smart_scroller_mirrored_set(wd->scroller, elm_widget_mirrored_get(obj));
570    elm_smart_scroller_object_theme_set(obj, wd->scroller, "scroller", "entry",
571                                        elm_widget_style_get(obj));
572    if (wd->scroll)
573      {
574         const char *str;
575         Evas_Object *edj;
576
577         edj = elm_smart_scroller_edje_object_get(wd->scroller);
578         str = edje_object_data_get(edj, "focus_highlight");
579         if ((str) && (!strcmp(str, "on")))
580            elm_widget_highlight_in_theme_set(obj, EINA_TRUE);
581         else
582            elm_widget_highlight_in_theme_set(obj, EINA_FALSE);
583      }
584    _sizing_eval(obj);
585 }
586
587 static void
588 _disable_hook(Evas_Object *obj)
589 {
590    Widget_Data *wd = elm_widget_data_get(obj);
591
592    if (elm_widget_disabled_get(obj))
593      {
594         edje_object_signal_emit(wd->ent, "elm,state,disabled", "elm");
595         wd->disabled = EINA_TRUE;
596      }
597    else
598      {
599         edje_object_signal_emit(wd->ent, "elm,state,enabled", "elm");
600         wd->disabled = EINA_FALSE;
601      }
602 }
603
604 static void
605 _recalc_cursor_geometry(Evas_Object *obj)
606 {
607    Widget_Data *wd = elm_widget_data_get(obj);
608    if (!wd) return;
609    evas_object_smart_callback_call(obj, SIG_CURSOR_CHANGED, NULL);
610    if (!elm_object_focus_get(obj)) return;
611    if (!wd->deferred_recalc_job)
612      {
613         Evas_Coord cx, cy, cw, ch;
614         edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text",
615               &cx, &cy, &cw, &ch);
616         if (wd->cur_changed)
617           {
618              elm_widget_show_region_set(obj, cx, cy, cw, ch, EINA_FALSE);
619              wd->cur_changed = EINA_FALSE;
620           }
621      }
622    else
623       wd->deferred_cur = EINA_TRUE;
624 }
625
626 static void
627 _elm_win_recalc_job(void *data)
628 {
629    Widget_Data *wd = elm_widget_data_get(data);
630    Evas_Coord minh = -1, resw = -1, minw = -1, fw = 0, fh = 0;
631    if (!wd) return;
632    wd->deferred_recalc_job = NULL;
633
634    evas_object_geometry_get(wd->ent, NULL, NULL, &resw, NULL);
635    edje_object_size_min_restricted_calc(wd->ent, &minw, &minh, resw, 0);
636    edje_object_size_min_restricted_calc(wd->ent, &fw, &fh, 0, 0);
637    elm_coords_finger_size_adjust(1, &fw, 1, &minh);
638    wd->entmw = fw;
639    wd->entmh = minh;
640    /* This is a hack to workaround the way min size hints are treated.
641     * If the minimum width is smaller than the restricted width, it means
642     * the mininmum doesn't matter. */
643    if (minw <= resw)
644      {
645         Evas_Coord ominw = -1;
646         evas_object_size_hint_min_get(data, &ominw, NULL);
647         minw = ominw;
648      }
649
650    elm_coords_finger_size_adjust(1, &fw, 1, &fh);
651    if (wd->scroll)
652      {
653         Evas_Coord vmw = 0, vmh = 0;
654
655         edje_object_size_min_calc
656            (elm_smart_scroller_edje_object_get(wd->scroller),
657                &vmw, &vmh);
658         if (wd->single_line)
659           {
660              evas_object_size_hint_min_set(data, vmw, minh + vmh);
661              evas_object_size_hint_max_set(data, -1, minh + vmh);
662           }
663         else
664           {
665              evas_object_size_hint_min_set(data, vmw, vmh);
666              evas_object_size_hint_max_set(data, -1, -1);
667           }
668      }
669    else
670      {
671         if (wd->single_line)
672           {
673              evas_object_size_hint_min_set(data, minw, minh);
674              evas_object_size_hint_max_set(data, -1, minh);
675           }
676         else
677           {
678              evas_object_size_hint_min_set(data, fw, minh);
679              evas_object_size_hint_max_set(data, -1, -1);
680           }
681      }
682
683    if ((wd->deferred_cur) && (elm_object_focus_get(data)))
684      {
685         Evas_Coord cx, cy, cw, ch;
686         edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text",
687                                                   &cx, &cy, &cw, &ch);
688         if (wd->cur_changed)
689           {
690              elm_widget_show_region_set(data, cx, cy, cw, ch, EINA_FALSE);
691              wd->cur_changed = EINA_FALSE;
692           }
693      }
694 }
695
696 static void
697 _sizing_eval(Evas_Object *obj)
698 {
699    Widget_Data *wd = elm_widget_data_get(obj);
700    Evas_Coord minw = -1, minh = -1;
701    Evas_Coord resw, resh;
702    if (!wd) return;
703
704    evas_object_geometry_get(obj, NULL, NULL, &resw, &resh);
705    if (wd->linewrap)
706      {
707         if ((resw == wd->lastw) && (!wd->changed)) return;
708         wd->changed = EINA_FALSE;
709         wd->lastw = resw;
710         if (wd->scroll)
711           {
712              Evas_Coord vw = 0, vh = 0, vmw = 0, vmh = 0, w = -1, h = -1;
713
714              evas_object_resize(wd->scroller, resw, resh);
715              edje_object_size_min_calc
716                 (elm_smart_scroller_edje_object_get(wd->scroller),
717                     &vmw, &vmh);
718              elm_smart_scroller_child_viewport_size_get(wd->scroller, &vw, &vh);
719              edje_object_size_min_restricted_calc(wd->ent, &minw, &minh, vw, 0);
720              wd->entmw = minw;
721              wd->entmh = minh;
722              elm_coords_finger_size_adjust(1, &minw, 1, &minh);
723
724              if ((minw > 0) && (vw < minw)) vw = minw;
725              if (minh > vh) vh = minh;
726
727              if (wd->single_line) h = vmh + minh;
728              else h = vmh;
729              evas_object_resize(wd->ent, vw, vh);
730              evas_object_size_hint_min_get(obj, &w, NULL);
731              evas_object_size_hint_min_set(obj, w, h);
732              if (wd->single_line)
733                 evas_object_size_hint_max_set(obj, -1, h);
734              else
735                 evas_object_size_hint_max_set(obj, -1, -1);
736           }
737         else
738           {
739              if (wd->deferred_recalc_job) ecore_job_del(wd->deferred_recalc_job);
740              wd->deferred_recalc_job = ecore_job_add(_elm_win_recalc_job, obj);
741           }
742      }
743    else
744      {
745         if (!wd->changed) return;
746         wd->changed = EINA_FALSE;
747         wd->lastw = resw;
748         if (wd->scroll)
749           {
750              Evas_Coord vw = 0, vh = 0, vmw = 0, vmh = 0, w = -1, h = -1;
751
752              edje_object_size_min_calc(wd->ent, &minw, &minh);
753              wd->entmw = minw;
754              wd->entmh = minh;
755              elm_coords_finger_size_adjust(1, &minw, 1, &minh);
756
757              elm_smart_scroller_child_viewport_size_get(wd->scroller, &vw, &vh);
758
759              if ((minw > 0) && (vw < minw)) vw = minw;
760              if (minh > 0) vh = minh;
761
762              evas_object_resize(wd->ent, vw, vh);
763              edje_object_size_min_calc
764                 (elm_smart_scroller_edje_object_get(wd->scroller),
765                     &vmw, &vmh);
766              if (wd->single_line) h = vmh + minh;
767              else h = vmh;
768              evas_object_size_hint_min_get(obj, &w, NULL);
769              evas_object_size_hint_min_set(obj, w, h);
770              if (wd->single_line)
771                 evas_object_size_hint_max_set(obj, -1, h);
772              else
773                 evas_object_size_hint_max_set(obj, -1, -1);
774           }
775         else
776           {
777              edje_object_size_min_calc(wd->ent, &minw, &minh);
778              wd->entmw = minw;
779              wd->entmh = minh;
780              elm_coords_finger_size_adjust(1, &minw, 1, &minh);
781              evas_object_size_hint_min_set(obj, minw, minh);
782              if (wd->single_line)
783                 evas_object_size_hint_max_set(obj, -1, minh);
784              else
785                 evas_object_size_hint_max_set(obj, -1, -1);
786           }
787      }
788
789    _recalc_cursor_geometry(obj);
790 }
791
792 static void
793 _check_enable_returnkey(Evas_Object *obj)
794 {
795    Widget_Data *wd = elm_widget_data_get(obj);
796    if (!wd) return;
797
798    Ecore_IMF_Context *ic = elm_entry_imf_context_get(obj);
799    if (!ic) return;
800
801    if (!wd->autoreturnkey) return;
802
803    if (_entry_length_get(obj) == 0)
804      {
805         ecore_imf_context_input_panel_key_disabled_set(ic, ECORE_IMF_INPUT_PANEL_LAYOUT_NORMAL, ECORE_IMF_INPUT_PANEL_KEY_ENTER, EINA_TRUE);
806      }
807    else
808      {
809         ecore_imf_context_input_panel_key_disabled_set(ic, ECORE_IMF_INPUT_PANEL_LAYOUT_NORMAL, ECORE_IMF_INPUT_PANEL_KEY_ENTER, EINA_FALSE);
810      }
811 }
812
813 static void
814 _on_focus_hook(void *data __UNUSED__, Evas_Object *obj)
815 {
816    Widget_Data *wd = elm_widget_data_get(obj);
817    Evas_Object *top = elm_widget_top_get(obj);
818    if (!wd) return;
819    if (elm_widget_focus_get(obj))
820      {
821         printf("[Elm_entry::Focused] obj : %p\n", obj);
822         evas_object_focus_set(wd->ent, EINA_TRUE);
823         edje_object_signal_emit(wd->ent, "elm,action,focus", "elm");
824         if (top && wd->input_panel_enable)
825           elm_win_keyboard_mode_set(top, ELM_WIN_KEYBOARD_ON);
826         evas_object_smart_callback_call(obj, SIG_FOCUSED, NULL);
827         _check_enable_returnkey(obj);
828         wd->mgf_type = _ENTRY_MAGNIFIER_FIXEDSIZE;
829      }
830    else
831      {
832         printf("[Elm_entry::Unfocused] obj : %p\n", obj);
833         edje_object_signal_emit(wd->ent, "elm,action,unfocus", "elm");
834         evas_object_focus_set(wd->ent, EINA_FALSE);
835         if (top && wd->input_panel_enable)
836           elm_win_keyboard_mode_set(top, ELM_WIN_KEYBOARD_OFF);
837         evas_object_smart_callback_call(obj, SIG_UNFOCUSED, NULL);
838
839         if ((wd->api) && (wd->api->obj_hidemenu))
840           {
841              wd->api->obj_hidemenu(obj);
842           }
843      }
844 }
845
846
847 static void
848 _content_set_hook(Evas_Object *obj, const char *part, Evas_Object *content)
849 {
850    Widget_Data *wd = elm_widget_data_get(obj);
851    Evas_Object *edje;
852    if ((!wd) || (!content)) return;
853
854    if (wd->scroll)
855      {
856         edje = elm_smart_scroller_edje_object_get(wd->scroller);
857         if (!strcmp(part, "elm.swallow.icon"))
858           {
859              if (wd->icon)
860                evas_object_del(wd->icon);
861              wd->icon = content;
862              edje_object_signal_emit(edje, "elm,action,show,icon", "elm");
863           }
864         else if (!strcmp(part, "elm.swallow.end"))
865           {
866              if (wd->end)
867                evas_object_del(wd->end);
868              wd->end = content;
869              edje_object_signal_emit(edje, "elm,action,show,end", "elm");
870           }
871      }
872    else
873      edje = wd->ent;
874    evas_event_freeze(evas_object_evas_get(obj));
875    elm_widget_sub_object_add(obj, content);
876    edje_object_part_swallow(edje, part, content);
877    _sizing_eval(obj);
878    evas_event_thaw(evas_object_evas_get(obj));
879    evas_event_thaw_eval(evas_object_evas_get(obj));
880 }
881
882 static Evas_Object *
883 _content_unset_hook(Evas_Object *obj, const char *part)
884 {
885    Widget_Data *wd = elm_widget_data_get(obj);
886    Evas_Object *content, *edje;
887    if (!wd) return NULL;
888
889    if (wd->scroll)
890      {
891         edje = elm_smart_scroller_edje_object_get(wd->scroller);
892         if (!strcmp(part, "elm.swallow.icon"))
893           {
894              wd->icon = NULL;
895              edje_object_signal_emit(edje, "elm,action,hide,icon", "elm");
896           }
897         else if (!strcmp(part, "elm.swallow.end"))
898           {
899              wd->end = NULL;
900              edje_object_signal_emit(edje, "elm,action,hide,end", "elm");
901           }
902      }
903    else
904      edje = wd->ent;
905
906    content = edje_object_part_swallow_get(edje, part);
907    edje_object_part_swallow(edje, part, NULL);
908    if (!content) return NULL;
909    evas_event_freeze(evas_object_evas_get(obj));
910    elm_widget_sub_object_del(obj, content);
911    edje_object_part_unswallow(wd->ent, content);
912    _sizing_eval(obj);
913    evas_event_thaw(evas_object_evas_get(obj));
914    evas_event_thaw_eval(evas_object_evas_get(obj));
915
916    return content;
917 }
918
919 static Evas_Object *
920 _content_get_hook(const Evas_Object *obj, const char *part)
921 {
922    Widget_Data *wd = elm_widget_data_get(obj);
923    Evas_Object *content = NULL, *edje;
924    if (!wd) return NULL;
925
926    if (wd->scroll)
927      {
928         if (!strcmp(part, "elm.swallow.icon"))
929           return wd->icon;
930         if (!strcmp(part, "elm.swallow.end"))
931           return wd->end;
932
933         edje = elm_smart_scroller_edje_object_get(wd->scroller);
934      }
935    else
936      edje = wd->ent;
937
938    if (edje)
939      content = edje_object_part_swallow_get(edje, part);
940    return content;
941 }
942
943
944 static void
945 _signal_emit_hook(Evas_Object *obj, const char *emission, const char *source)
946 {
947    Widget_Data *wd = elm_widget_data_get(obj);
948    if (!wd) return;
949    edje_object_signal_emit(wd->ent, emission, source);
950    if (wd->scroller)
951       edje_object_signal_emit(elm_smart_scroller_edje_object_get(wd->scroller),
952                               emission, source);
953 }
954
955 static void
956 _signal_callback_add_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
957 {
958    Widget_Data *wd = elm_widget_data_get(obj);
959    if (!wd) return;
960    edje_object_signal_callback_add(wd->ent, emission, source, func_cb, data);
961    if (wd->scroller)
962       edje_object_signal_callback_add(elm_smart_scroller_edje_object_get(wd->scroller),
963                                       emission, source, func_cb, data);
964 }
965
966 static void
967 _signal_callback_del_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
968 {
969    Widget_Data *wd = elm_widget_data_get(obj);
970    edje_object_signal_callback_del_full(wd->ent, emission, source, func_cb,
971                                         data);
972    if (wd->scroller)
973       edje_object_signal_callback_del_full(elm_smart_scroller_edje_object_get(wd->scroller),
974                                            emission, source, func_cb, data);
975 }
976
977 static void
978 _on_focus_region_hook(const Evas_Object *obj, Evas_Coord *x, Evas_Coord *y, Evas_Coord *w, Evas_Coord *h)
979 {
980    Widget_Data *wd = elm_widget_data_get(obj);
981    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", x, y, w, h);
982 }
983
984 static void
985 _focus_region_hook(Evas_Object *obj, Evas_Coord x, Evas_Coord y, Evas_Coord w, Evas_Coord h)
986 {
987    Widget_Data *wd = elm_widget_data_get(obj);
988    if (wd->scroll)
989       elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
990 }
991
992 static void
993 _show_region_hook(void *data, Evas_Object *obj)
994 {
995    Widget_Data *wd = elm_widget_data_get(data);
996    Evas_Coord x, y, w, h;
997    if (!wd) return;
998    elm_widget_show_region_get(obj, &x, &y, &w, &h);
999    if (wd->scroll)
1000      elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
1001 }
1002
1003 static void
1004 _sub_del(void *data, Evas_Object *obj, void *event_info)
1005 {
1006    Widget_Data *wd = data;
1007    Evas_Object *sub = event_info;
1008    Evas_Object *edje;
1009
1010    edje = elm_smart_scroller_edje_object_get(wd->scroller);
1011    if (sub == wd->icon)
1012      {
1013         wd->icon = NULL;
1014         if (edje)
1015           edje_object_signal_emit(edje, "elm,action,hide,icon", "elm");
1016      }
1017    else if (sub == wd->end)
1018      {
1019         wd->end = NULL;
1020         if (edje)
1021           edje_object_signal_emit(edje, "elm,action,hide,end", "elm");
1022      }
1023    _sizing_eval(obj);
1024 }
1025
1026 static void
1027 _hoversel_position(Evas_Object *obj)
1028 {
1029    Widget_Data *wd = elm_widget_data_get(obj);
1030    Evas_Coord cx, cy, cw, ch, x, y, mw, mh;
1031    if (!wd) return;
1032
1033    cx = cy = 0;
1034    cw = ch = 1;
1035    evas_object_geometry_get(wd->ent, &x, &y, NULL, NULL);
1036    if (wd->usedown)
1037      {
1038         cx = wd->downx - x;
1039         cy = wd->downy - y;
1040         cw = 1;
1041         ch = 1;
1042      }
1043    else
1044      edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text",
1045                                                &cx, &cy, &cw, &ch);
1046    evas_object_size_hint_min_get(wd->hoversel, &mw, &mh);
1047    if (cw < mw)
1048      {
1049         cx += (cw - mw) / 2;
1050         cw = mw;
1051      }
1052    if (ch < mh)
1053      {
1054         cy += (ch - mh) / 2;
1055         ch = mh;
1056      }
1057    evas_object_move(wd->hoversel, x + cx, y + cy);
1058    evas_object_resize(wd->hoversel, cw, ch);
1059 }
1060
1061 static void
1062 _move(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1063 {
1064    Widget_Data *wd = elm_widget_data_get(data);
1065
1066    if (wd->hoversel) _hoversel_position(data);
1067 }
1068
1069 static void
1070 _resize(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1071 {
1072    Widget_Data *wd = elm_widget_data_get(data);
1073    if (!wd) return;
1074
1075    if (wd->linewrap)
1076      {
1077         _sizing_eval(data);
1078      }
1079    else if (wd->scroll)
1080      {
1081         Evas_Coord vw = 0, vh = 0;
1082
1083         elm_smart_scroller_child_viewport_size_get(wd->scroller, &vw, &vh);
1084         if (vw < wd->entmw) vw = wd->entmw;
1085         if (vh < wd->entmh) vh = wd->entmh;
1086         evas_object_resize(wd->ent, vw, vh);
1087      }
1088    if (wd->hoversel) _hoversel_position(data);
1089 }
1090
1091 static void
1092 _hover_del(void *data)
1093 {
1094    Widget_Data *wd = elm_widget_data_get(data);
1095    if (!wd) return;
1096
1097    if (wd->hoversel)
1098      {
1099         evas_object_del(wd->hoversel);
1100         wd->hoversel = NULL;
1101      }
1102    wd->hovdeljob = NULL;
1103 }
1104
1105 static void
1106 _dismissed(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1107 {
1108    Widget_Data *wd = elm_widget_data_get(data);
1109    if (!wd) return;
1110    wd->usedown = 0;
1111    if (wd->hoversel) evas_object_hide(wd->hoversel);
1112    if (wd->selmode)
1113      {
1114         if (!_elm_config->desktop_entry)
1115           {
1116              if (!wd->password)
1117                edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
1118           }
1119      }
1120    elm_widget_scroll_freeze_pop(data);
1121    if (wd->hovdeljob) ecore_job_del(wd->hovdeljob);
1122    wd->hovdeljob = ecore_job_add(_hover_del, data);
1123 }
1124
1125 static void
1126 _selectall(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1127 {
1128    Widget_Data *wd = elm_widget_data_get(data);
1129    if (!wd) return;
1130    wd->selmode = EINA_TRUE;
1131    edje_object_part_text_select_none(wd->ent, "elm.text");
1132    edje_object_signal_emit(wd->ent, "elm,state,select,on", "elm");
1133    edje_object_part_text_select_all(wd->ent, "elm.text");
1134    elm_object_scroll_freeze_pop(data);
1135 }
1136
1137 static void
1138 _select(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1139 {
1140    Widget_Data *wd = elm_widget_data_get(data);
1141    if (!wd) return;
1142    wd->selmode = EINA_TRUE;
1143    edje_object_part_text_select_none(wd->ent, "elm.text");
1144    if (!_elm_config->desktop_entry)
1145      {
1146         if (!wd->password)
1147           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
1148      }
1149    edje_object_signal_emit(wd->ent, "elm,state,select,on", "elm");
1150    if (!_elm_config->desktop_entry)
1151       elm_object_scroll_freeze_pop(data);
1152       //elm_widget_scroll_hold_push(data);
1153 }
1154
1155 static void
1156 _paste(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1157 {
1158    Widget_Data *wd = elm_widget_data_get(data);
1159    if (!wd) return;
1160    evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
1161    if (wd->sel_notify_handler)
1162      {
1163 #ifdef HAVE_ELEMENTARY_X
1164         Elm_Sel_Format formats;
1165         wd->selection_asked = EINA_TRUE;
1166         formats = ELM_SEL_FORMAT_MARKUP;
1167         if (!wd->textonly)
1168           formats |= ELM_SEL_FORMAT_IMAGE;
1169         elm_selection_get(ELM_SEL_CLIPBOARD, formats, data, NULL, NULL);
1170 #endif
1171      }
1172 }
1173
1174 static void
1175 _store_selection(Elm_Sel_Type seltype, Evas_Object *obj)
1176 {
1177    Widget_Data *wd = elm_widget_data_get(obj);
1178    const char *sel;
1179    char *sel_str;
1180
1181    if (!wd) return;
1182    sel = edje_object_part_text_selection_get(wd->ent, "elm.text");
1183    sel_str = strdup(sel);
1184    if (!sel_str)
1185      return;
1186    if (wd->textonly)
1187      {
1188         while (EINA_TRUE)
1189           {
1190              char *startTag = NULL;
1191              char *endTag = NULL;
1192
1193              startTag = strstr(sel_str, "<item");
1194              if (!startTag)
1195                startTag = strstr(sel_str, "</item");
1196              if (startTag)
1197                endTag = strstr(startTag, ">");
1198              else
1199                break;
1200              if (!endTag || startTag > endTag)
1201                break;
1202
1203              size_t sindex = startTag - sel_str;
1204              size_t eindex = endTag - sel_str + 1;
1205
1206              Eina_Strbuf *buf = eina_strbuf_new();
1207              if (buf)
1208                {
1209                   eina_strbuf_append(buf, sel_str);
1210                   eina_strbuf_remove(buf, sindex, eindex);
1211                   sel_str = eina_strbuf_string_steal(buf);
1212                   eina_strbuf_free(buf);
1213                }
1214           }
1215      }
1216    elm_selection_set(seltype, obj, ELM_SEL_FORMAT_MARKUP, sel_str);
1217    if (seltype == ELM_SEL_CLIPBOARD)
1218      eina_stringshare_replace(&wd->cut_sel, sel_str);
1219    free(sel_str);
1220 }
1221
1222 static void
1223 _cut(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1224 {
1225    Widget_Data *wd = elm_widget_data_get(data);
1226
1227    /* Store it */
1228    wd->selmode = EINA_FALSE;
1229    if (!_elm_config->desktop_entry)
1230      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1231    edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1232    if (!_elm_config->desktop_entry)
1233      elm_widget_scroll_hold_pop(data);
1234    _store_selection(ELM_SEL_CLIPBOARD, data);
1235    edje_object_part_text_insert(wd->ent, "elm.text", "");
1236    edje_object_part_text_select_none(wd->ent, "elm.text");
1237 }
1238
1239 static void
1240 _copy(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1241 {
1242    Widget_Data *wd = elm_widget_data_get(data);
1243    if (!wd) return;
1244    wd->selmode = EINA_TRUE;
1245    if (!_elm_config->desktop_entry)
1246      {
1247         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1248         elm_widget_scroll_hold_pop(data);
1249      }
1250    _store_selection(ELM_SEL_CLIPBOARD, data);
1251 }
1252
1253 static void
1254 _cancel(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1255 {
1256    Widget_Data *wd = elm_widget_data_get(data);
1257    if (!wd) return;
1258    wd->selmode = EINA_FALSE;
1259    if (!_elm_config->desktop_entry)
1260      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1261    edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1262    if (!_elm_config->desktop_entry)
1263      elm_widget_scroll_hold_pop(data);
1264    edje_object_part_text_select_none(wd->ent, "elm.text");
1265 }
1266
1267 static void
1268 _clipboard_menu(void *data, Evas_Object *obj, void *event_info __UNUSED__)
1269 {
1270    Widget_Data *wd = elm_widget_data_get(data);
1271    if (!wd) return;
1272
1273    // start for cbhm
1274 #ifdef HAVE_ELEMENTARY_X
1275    ecore_x_selection_secondary_set(elm_win_xwindow_get(obj), "",1);
1276 #endif
1277    cnpwidgetdata = data;
1278    elm_cbhm_helper_init(obj);
1279    if (elm_entry_cnp_textonly_get(obj))
1280      elm_cbhm_send_raw_data("show0");
1281    else
1282      elm_cbhm_send_raw_data("show1");
1283    // end for cbhm
1284 }
1285
1286 // start for cbhm
1287 static void
1288 _cnpinit(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1289 {
1290    Widget_Data *wd = elm_widget_data_get(data);
1291    if (!wd) return;
1292    cnpwidgetdata = data;
1293 }
1294 // end for cbhm
1295
1296
1297 static void
1298 _item_clicked(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1299 {
1300    Elm_Entry_Context_Menu_Item *it = data;
1301    Evas_Object *obj2 = it->obj;
1302    if (it->func) it->func(it->data, obj2, NULL);
1303 }
1304
1305 static void
1306 _menu_press(Evas_Object *obj)
1307 {
1308    Widget_Data *wd = elm_widget_data_get(obj);
1309    Evas_Object *top;
1310    const Eina_List *l;
1311    const Elm_Entry_Context_Menu_Item *it;
1312    if (!wd) return;
1313    if ((wd->api) && (wd->api->obj_longpress))
1314      {
1315         wd->api->obj_longpress(obj);
1316      }
1317    else if (wd->context_menu)
1318      {
1319         const char *context_menu_orientation;
1320
1321         if (wd->hoversel) evas_object_del(wd->hoversel);
1322         else elm_widget_scroll_freeze_push(obj);
1323         wd->hoversel = elm_hoversel_add(obj);
1324         context_menu_orientation = edje_object_data_get
1325            (wd->ent, "context_menu_orientation");
1326         if ((context_menu_orientation) &&
1327             (!strcmp(context_menu_orientation, "horizontal")))
1328           elm_hoversel_horizontal_set(wd->hoversel, EINA_TRUE);
1329         elm_object_style_set(wd->hoversel, "entry");
1330         elm_widget_sub_object_add(obj, wd->hoversel);
1331         elm_object_text_set(wd->hoversel, "Text");
1332         top = elm_widget_top_get(obj);
1333         if (top) elm_hoversel_hover_parent_set(wd->hoversel, top);
1334         evas_object_smart_callback_add(wd->hoversel, "dismissed", _dismissed, obj);
1335         if (!wd->selmode)
1336           {
1337              if (!wd->password)
1338                elm_hoversel_item_add(wd->hoversel, E_("Select"), NULL, ELM_ICON_NONE,
1339                                      _select, obj);
1340              if (1) // need way to detect if someone has a selection
1341                {
1342                   if (wd->editable)
1343                     elm_hoversel_item_add(wd->hoversel, E_("Paste"), NULL, ELM_ICON_NONE,
1344                                           _paste, obj);
1345                }
1346         // start for cbhm
1347              if ((!wd->password) && (wd->editable))
1348                elm_hoversel_item_add(wd->hoversel, E_("More"), NULL, ELM_ICON_NONE,
1349                                      _clipboard_menu, obj);
1350         // end for cbhm
1351           }
1352         else
1353           {
1354              if (!wd->password)
1355                {
1356                   if (wd->have_selection)
1357                     {
1358                        elm_hoversel_item_add(wd->hoversel, E_("Copy"), NULL, ELM_ICON_NONE,
1359                                              _copy, obj);
1360                        if (wd->editable)
1361                          elm_hoversel_item_add(wd->hoversel, E_("Cut"), NULL, ELM_ICON_NONE,
1362                                                _cut, obj);
1363                     }
1364                   elm_hoversel_item_add(wd->hoversel, E_("Cancel"), NULL, ELM_ICON_NONE,
1365                                         _cancel, obj);
1366         // start for cbhm
1367                   if (wd->editable)
1368                     elm_hoversel_item_add(wd->hoversel, E_("More"), NULL, ELM_ICON_NONE,
1369                                           _clipboard_menu, obj);
1370         // end for cbhm
1371                }
1372           }
1373         EINA_LIST_FOREACH(wd->items, l, it)
1374           {
1375              elm_hoversel_item_add(wd->hoversel, it->label, it->icon_file,
1376                                    it->icon_type, _item_clicked, it);
1377           }
1378         if (wd->hoversel)
1379           {
1380              _hoversel_position(obj);
1381              evas_object_show(wd->hoversel);
1382              elm_hoversel_hover_begin(wd->hoversel);
1383           }
1384         if (!_elm_config->desktop_entry)
1385           {
1386              edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1387              edje_object_part_text_select_abort(wd->ent, "elm.text");
1388           }
1389      }
1390 }
1391
1392 static void
1393 _magnifier_hide(void *data)
1394 {
1395    Widget_Data *wd = elm_widget_data_get(data);
1396    if (!wd) return;
1397
1398    evas_object_hide(wd->mgf_bg);
1399    evas_object_hide(wd->mgf_clip);
1400
1401    if (wd->scroll)
1402      elm_smart_scroller_freeze_set(wd->scroller, EINA_FALSE);
1403 }
1404
1405 static void
1406 _magnifier_show(void *data)
1407 {
1408    Widget_Data *wd = elm_widget_data_get(data);
1409    if (!wd) return;
1410
1411    evas_object_show(wd->mgf_bg);
1412    evas_object_show(wd->mgf_clip);
1413 }
1414
1415 static void
1416 _magnifier_move(void *data)
1417 {
1418    Widget_Data *wd = elm_widget_data_get(data);
1419    if (!wd) return;
1420
1421    Evas_Coord x, y, w, h;
1422    Evas_Coord cx, cy, cw, ch, ox, oy;
1423
1424    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
1425
1426    if (wd->scroll)
1427      {
1428         evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1429         elm_smart_scroller_child_pos_get(wd->scroller, &ox, &oy);
1430         cx -= ox;
1431         cy -= oy;
1432      }
1433    else
1434      evas_object_geometry_get(data, &x, &y, &w, &h);
1435
1436    ox = oy = 0;
1437
1438    if ((cy + y) - wd->mgf_height < 0)
1439      oy = -1 * ((cy + y) - wd->mgf_height);
1440
1441    if (wd->mgf_type == _ENTRY_MAGNIFIER_FIXEDSIZE)
1442      evas_object_move(wd->mgf_bg, (cx + x + cw/2) + ox, (cy + y) - wd->mgf_height + oy);
1443    else if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1444      evas_object_move(wd->mgf_bg, x, (cy + y) - wd->mgf_height + oy);
1445    else
1446      return;
1447
1448    evas_object_move(wd->mgf_proxy, (1 - wd->mgf_scale) * cx + x + ox, (1 - wd->mgf_scale) * cy + y - wd->mgf_height/2 - ch/2 + oy);
1449 }
1450
1451 static void
1452 _magnifier_create(void *data)
1453 {
1454    Widget_Data *wd = elm_widget_data_get(data);
1455    Evas_Coord x, y, w, h, mw, mh;
1456    const char* key_data = NULL;
1457    double elm_scale;
1458
1459    if (!wd) return;
1460
1461    if (wd->mgf_proxy)
1462      {
1463         evas_object_image_source_unset(wd->mgf_proxy);
1464         evas_object_color_set(wd->mgf_proxy, 255, 255, 255, 0);
1465         evas_object_hide(wd->mgf_proxy);
1466         evas_object_clip_unset(wd->mgf_proxy);
1467         evas_object_del(wd->mgf_proxy);
1468      }
1469    if (wd->mgf_bg) evas_object_del(wd->mgf_bg);
1470    if (wd->mgf_clip) evas_object_del(wd->mgf_clip);
1471
1472    if (wd->scroll)
1473      evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1474    else
1475      evas_object_geometry_get(data, &x, &y, &w, &h);
1476
1477    if ((w <= 0) || (h <= 0))
1478      return;
1479
1480    wd->mgf_bg = edje_object_add(evas_object_evas_get(data));
1481
1482    if (wd->mgf_type == _ENTRY_MAGNIFIER_FIXEDSIZE)
1483      _elm_theme_object_set(data, wd->mgf_bg, "entry", "magnifier", "fixed-size");
1484    else if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1485      _elm_theme_object_set(data, wd->mgf_bg, "entry", "magnifier", "fill-width");
1486    else
1487      return;
1488
1489    wd->mgf_clip = evas_object_rectangle_add(evas_object_evas_get(data));
1490    evas_object_color_set(wd->mgf_clip, 255, 255, 255, 255);
1491    edje_object_part_swallow(wd->mgf_bg, "swallow", wd->mgf_clip);
1492
1493    key_data = edje_object_data_get(wd->mgf_bg, "height");
1494    if (key_data) wd->mgf_height = atoi(key_data);
1495    key_data = edje_object_data_get(wd->mgf_bg, "scale");
1496    if (key_data) wd->mgf_scale = atof(key_data);
1497
1498    elm_scale = elm_scale_get();
1499    wd->mgf_height = (int)((float)wd->mgf_height * elm_scale);
1500
1501    if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1502      evas_object_resize(wd->mgf_bg, w, wd->mgf_height);
1503
1504    if (wd->scroll)
1505      {
1506         elm_smart_scroller_freeze_set(wd->scroller, EINA_TRUE);
1507         wd->mgf_proxy = evas_object_image_add(evas_object_evas_get(wd->scroller));
1508         evas_object_image_source_set(wd->mgf_proxy, wd->scroller);
1509      }
1510    else
1511      {
1512         wd->mgf_proxy = evas_object_image_add(evas_object_evas_get(data));
1513         evas_object_image_source_set(wd->mgf_proxy, data);
1514      }
1515
1516    mw = (Evas_Coord)((float)w * wd->mgf_scale);
1517    mh = (Evas_Coord)((float)h * wd->mgf_scale);
1518    if ((mw <= 0) || (mh <= 0))
1519      return;
1520
1521    evas_object_resize(wd->mgf_proxy, mw, mh);
1522    evas_object_image_fill_set(wd->mgf_proxy, 0, 0, mw, mh);
1523    evas_object_color_set(wd->mgf_proxy, 255, 255, 255, 255);
1524    evas_object_pass_events_set(wd->mgf_proxy, EINA_TRUE);
1525    evas_object_show(wd->mgf_proxy);
1526    evas_object_clip_set(wd->mgf_proxy, wd->mgf_clip);
1527
1528    evas_object_layer_set(wd->mgf_bg, EVAS_LAYER_MAX);
1529    evas_object_layer_set(wd->mgf_proxy, EVAS_LAYER_MAX);
1530 }
1531
1532 static Eina_Bool
1533 _signal_long_pressed(void *data)
1534 {
1535    Widget_Data *wd = elm_widget_data_get(data);
1536    if (!wd) return ECORE_CALLBACK_CANCEL;
1537
1538    wd->long_pressed = EINA_TRUE;
1539
1540    _cancel(data, NULL, NULL);
1541
1542    _magnifier_create(data);
1543    _magnifier_move(data);
1544    _magnifier_show(data);
1545    elm_object_scroll_freeze_push(data);
1546
1547    evas_object_smart_callback_call(data, SIG_LONGPRESSED, NULL);
1548    return ECORE_CALLBACK_CANCEL;
1549 }
1550
1551 static void
1552 _mouse_down(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1553 {
1554    Widget_Data *wd = elm_widget_data_get(data);
1555    Evas_Event_Mouse_Down *ev = event_info;
1556    if (!wd) return;
1557    if (wd->disabled) return;
1558    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return;
1559    wd->downx = ev->canvas.x;
1560    wd->downy = ev->canvas.y;
1561    wd->long_pressed = EINA_FALSE;
1562 }
1563
1564 static void
1565 _mouse_up(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1566 {
1567    Widget_Data *wd = elm_widget_data_get(data);
1568    Evas_Event_Mouse_Up *ev = event_info;
1569    if (!wd) return;
1570    if (wd->disabled) return;
1571    if (ev->button == 1)
1572      {
1573         if (!wd->double_clicked)
1574           {
1575              if ((wd->api) && (wd->api->obj_mouseup))
1576                wd->api->obj_mouseup(data);
1577           }
1578         _magnifier_hide(data);
1579         elm_object_scroll_freeze_pop(data);
1580
1581         if (wd->long_pressed)
1582           _menu_press(data);
1583      }
1584    else if (ev->button == 3)
1585      {
1586         wd->usedown = 1;
1587         _menu_press(data);
1588      }
1589 }
1590
1591 static void
1592 _mouse_move(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1593 {
1594    Widget_Data *wd = elm_widget_data_get(data);
1595    Evas_Event_Mouse_Move *ev = event_info;
1596    if (!wd) return;
1597    if (wd->disabled) return;
1598
1599    if (ev->buttons == 1)
1600      {
1601         if (wd->long_pressed)
1602           {
1603              _magnifier_show(data);
1604              _magnifier_move(data);
1605           }
1606      }
1607 }
1608
1609 static const char *
1610 _getbase(Evas_Object *obj)
1611 {
1612    Widget_Data *wd = elm_widget_data_get(obj);
1613    if (!wd) return "base";
1614    if (wd->editable)
1615      {
1616         if (wd->password) return "base-password";
1617         else
1618           {
1619              if (wd->single_line) return "base-single";
1620              else
1621                {
1622                   switch (wd->linewrap)
1623                     {
1624                      case ELM_WRAP_CHAR:
1625                         return "base-charwrap";
1626                      case ELM_WRAP_WORD:
1627                         return "base";
1628                      case ELM_WRAP_MIXED:
1629                         return "base-mixedwrap";
1630                      case ELM_WRAP_NONE:
1631                      default:
1632                         return "base-nowrap";
1633                     }
1634                }
1635           }
1636      }
1637    else
1638      {
1639         if (wd->password) return "base-password";
1640         else
1641           {
1642              if (wd->single_line) return "base-single-noedit";
1643              else
1644                {
1645                   switch (wd->linewrap)
1646                     {
1647                      case ELM_WRAP_CHAR:
1648                         return "base-noedit-charwrap";
1649                      case ELM_WRAP_WORD:
1650                         return "base-noedit";
1651                      case ELM_WRAP_MIXED:
1652                         return "base-noedit-mixedwrap";
1653                      case ELM_WRAP_NONE:
1654                      default:
1655                         return "base-nowrap-noedit";
1656                     }
1657                }
1658           }
1659      }
1660    return "base";
1661 }
1662
1663
1664 static int
1665 _entry_length_get(Evas_Object *obj)
1666 {
1667    int len;
1668    const char *str = elm_entry_entry_get(obj);
1669    if (!str) return 0;
1670
1671    char *plain_str = _elm_util_mkup_to_text(str);
1672    if (!plain_str) return 0;
1673
1674    len = strlen(plain_str);
1675    free(plain_str);
1676
1677    return len;
1678 }
1679
1680 #ifndef HAVE_STRCASESTR
1681 char* _strcasestr(const char *s, const char *find)
1682 {
1683    char c, sc;
1684    size_t len;
1685
1686    if ((c = *find++) != 0) {
1687       c = tolower((unsigned char) c);
1688       len = strlen(find);
1689       do {
1690          do {
1691             if( (sc = *s++) == 0)
1692                return NULL;
1693          } while ((char)tolower((unsigned char)sc) != c);
1694       } while (strncasecmp(s, find, len) != 0);
1695       s--;
1696    }
1697    return ((char*) s);
1698 }
1699 #endif
1700
1701 static void
1702 _matchlist_show(void *data)
1703 {
1704    Widget_Data *wd = elm_widget_data_get(data);
1705    const char *text = NULL;
1706    int textlen = 0;
1707    char *str_list = NULL, *str_result = NULL;
1708    char *str_mkup = NULL, *str_front = NULL, *str_mid = NULL;
1709
1710    Eina_List *l;
1711    Eina_Bool textfound = EINA_FALSE;
1712
1713    if (!wd) return;
1714    if (elm_widget_disabled_get(data)) return;
1715
1716    wd->matchlist_job = NULL;
1717
1718    if (wd->matchlist_list_clicked)
1719      {
1720         evas_object_hide(wd->hover);
1721         wd->matchlist_list_clicked = EINA_FALSE;
1722         return;
1723      }
1724    text = elm_entry_entry_get(data);
1725    if (text == NULL)
1726      return;
1727    textlen = strlen(text);
1728
1729    if (textlen < wd->matchlist_threshold)
1730      {
1731         evas_object_hide(wd->hover);
1732         return;
1733      }
1734
1735    evas_object_hide(wd->hover);
1736
1737    if (wd->match_list)
1738      {
1739         elm_list_clear(wd->list);
1740         EINA_LIST_FOREACH(wd->match_list, l, str_list)
1741           {
1742              if (wd->matchlist_case_sensitive)
1743                str_result = strstr(str_list, text);
1744              else
1745 #ifdef HAVE_STRCASESTR
1746                 str_result = strcasestr(str_list, text);
1747 #else
1748                 str_result = _strcasestr(str_list, text);
1749 #endif
1750
1751              if (str_result)
1752                {
1753                   str_mkup = malloc(strlen(str_list) + 16);
1754                   if (str_mkup == NULL) return;
1755
1756                   textlen = strlen(str_list) - strlen(str_result);
1757                   str_front = malloc(textlen + 1);
1758                   if (str_front == NULL)
1759                     {
1760                        free(str_mkup);
1761                        return;
1762                     }
1763
1764                   memset(str_front, 0, textlen + 1);
1765                   strncpy(str_front, str_list, textlen);
1766
1767                   textlen = strlen(text);
1768                   str_mid = malloc(textlen + 1);
1769                   if (str_mid == NULL)
1770                     {
1771                        free(str_mkup);
1772                        free(str_front);
1773                        return;
1774                     }
1775
1776                   memset(str_mid, 0, textlen + 1);
1777                   strncpy(str_mid, str_list + strlen(str_front), textlen);
1778
1779                   sprintf(str_mkup, "%s<match>%s</match>%s", str_front, str_mid, str_result + strlen(text));
1780
1781                   elm_list_item_append(wd->list, str_mkup, NULL, NULL, NULL, NULL);
1782
1783                   if (str_mkup) free(str_mkup);
1784                   if (str_front) free(str_front);
1785                   if (str_mid) free(str_mid);
1786
1787                   textfound=EINA_TRUE;
1788                }
1789           }
1790      }
1791    else
1792      return;
1793
1794    if (textfound)
1795      {
1796         elm_list_go(wd->list);
1797         evas_object_show(wd->hover);
1798         evas_object_raise(wd->hover);
1799      }
1800 }
1801
1802 static void _matchlist_list_clicked( void *data, Evas_Object *obj, void *event_info )
1803 {
1804    Elm_List_Item *it = (Elm_List_Item *) elm_list_selected_item_get(obj);
1805    Widget_Data *wd = elm_widget_data_get(data);
1806    if ((it == NULL) || (wd == NULL))
1807      return;
1808
1809    const char *str = NULL;
1810    const char *text = elm_list_item_label_get(it);
1811    evas_object_smart_callback_call((Evas_Object *)data, "selected", (void *)text);
1812    if (wd->match_list)
1813      {
1814         if (text != NULL)
1815           {
1816              str = elm_entry_markup_to_utf8(text);
1817              elm_entry_entry_set(data, str);
1818              elm_entry_cursor_end_set(data);
1819              wd->matchlist_list_clicked = EINA_TRUE;
1820
1821              evas_object_smart_callback_call(data, SIG_MATCHLIST_CLICKED, elm_entry_markup_to_utf8(text));
1822              free(str);
1823           }
1824      }
1825    elm_widget_focus_set(data, EINA_TRUE);
1826 }
1827
1828 static void
1829 _entry_changed_common_handling(void *data, const char *event)
1830 {
1831    Widget_Data *wd = elm_widget_data_get(data);
1832    Evas_Coord minw, minh;
1833    if (!wd) return;
1834    wd->changed = EINA_TRUE;
1835    /* Reset the size hints which are no more relevant.
1836     * Keep the height, this is a hack, but doesn't really matter
1837     * cause we'll re-eval in a moment. */
1838    if (wd->scroll)
1839      {
1840         evas_object_size_hint_min_get(data, &minw, &minh);
1841         evas_object_size_hint_min_set(data, minw, minh);
1842      }
1843    else
1844      {
1845         evas_object_size_hint_min_get(data, NULL, &minh);
1846         evas_object_size_hint_min_set(data, -1, minh);
1847      }
1848
1849    _sizing_eval(data);
1850    if (wd->text) eina_stringshare_del(wd->text);
1851    wd->text = NULL;
1852    if (wd->password_text) eina_stringshare_del(wd->password_text);
1853    wd->password_text = NULL;
1854    _check_enable_returnkey(data);
1855    if (wd->delay_write)
1856      {
1857         ecore_timer_del(wd->delay_write);
1858         wd->delay_write = NULL;
1859      }
1860
1861    if ((wd->single_line) && (wd->match_list))
1862      {
1863         if (wd->matchlist_job) ecore_job_del(wd->matchlist_job);
1864         wd->matchlist_job = ecore_job_add(_matchlist_show, data);
1865      }
1866
1867    if ((wd->api) && (wd->api->obj_hidemenu))
1868      wd->api->obj_hidemenu(data);
1869
1870    if ((wd->autosave) && (wd->file))
1871      wd->delay_write = ecore_timer_add(2.0, _delay_write, data);
1872
1873    /* callback - this could call callbacks that delete the entry... thus...
1874     * any access to wd after this could be invalid */
1875    evas_object_smart_callback_call(data, event, NULL);
1876 }
1877
1878 static void
1879 _signal_entry_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1880 {
1881    Widget_Data *wd = elm_widget_data_get(data);
1882    if (!wd) return;
1883
1884    _entry_changed_common_handling(data, SIG_CHANGED);
1885 }
1886
1887 static void
1888 _signal_preedit_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1889 {
1890    Widget_Data *wd = elm_widget_data_get(data);
1891    if (!wd) return;
1892
1893    _entry_changed_common_handling(data, SIG_PREEDIT_CHANGED);
1894 }
1895
1896 static void
1897 _signal_handler_move_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1898 {
1899    Widget_Data *wd = elm_widget_data_get(data);
1900    if (!wd) return;
1901
1902    elm_object_scroll_freeze_push(data);
1903
1904    if ((wd->api) && (wd->api->obj_hidemenu))
1905      wd->api->obj_hidemenu(data);
1906
1907    _magnifier_create(data);
1908    _magnifier_move(data);
1909    _magnifier_show(data);
1910 }
1911
1912 static void
1913 _signal_handler_move_end(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1914 {
1915    Widget_Data *wd = elm_widget_data_get(data);
1916    if (!wd) return;
1917
1918    elm_object_scroll_freeze_pop(data);
1919
1920    if (wd->have_selection)
1921      {
1922         _magnifier_hide(data);
1923         _menu_press(data);
1924      }
1925 }
1926
1927 static void
1928 _signal_handler_moving(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1929 {
1930    _magnifier_move(data);
1931    _magnifier_show(data);
1932 }
1933
1934 static Evas_Object *
1935 _viewport_obj_get(Evas_Object *data)
1936 {
1937    Widget_Data *wd = elm_widget_data_get(data);
1938    if (!wd) return NULL;
1939
1940    if(!data || !strlen(elm_widget_type_get(data)))
1941      return NULL;
1942
1943    Evas_Coord x, y, w, h;
1944    x = y = w = h = -1;
1945
1946    if (wd->scroll)
1947      {
1948         //evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1949         //printf(">>> wd->scroller (%d, %d, %d, %d) \n", x, y, w, h);
1950         return wd->scroller;
1951      }
1952
1953    Evas_Object *parent_obj = data;
1954
1955    while(parent_obj = elm_widget_parent_get(parent_obj))
1956      {
1957         //evas_object_geometry_get(parent_obj, &x, &y, &w, &h);
1958         //printf(">>> %s (%d, %d, %d, %d) \n", elm_widget_type_get(parent_obj), x, y, w, h);
1959         if (!strcmp(elm_widget_type_get(parent_obj), "scroller") ||
1960             !strcmp(elm_widget_type_get(parent_obj), "genlist"))
1961           return parent_obj;
1962      }
1963
1964    return NULL;
1965 }
1966
1967 static void
1968 _signal_selection_end(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1969 {
1970    _magnifier_hide(data);
1971    _menu_press(data);
1972 }
1973
1974 static void
1975 _signal_selection_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1976 {
1977    Widget_Data *wd = elm_widget_data_get(data);
1978    const Eina_List *l;
1979    Evas_Object *entry;
1980    if (!wd) return;
1981    EINA_LIST_FOREACH(entries, l, entry)
1982      {
1983         if (entry != data) elm_entry_select_none(entry);
1984      }
1985    wd->have_selection = EINA_TRUE;
1986    wd->selmode = EINA_TRUE;
1987    evas_object_smart_callback_call(data, SIG_SELECTION_START, NULL);
1988 #ifdef HAVE_ELEMENTARY_X
1989    if (wd->sel_notify_handler)
1990      {
1991         const char *txt = elm_entry_selection_get(data);
1992         Evas_Object *top;
1993
1994         top = elm_widget_top_get(data);
1995         if ((top) && (elm_win_xwindow_get(top)))
1996           elm_selection_set(ELM_SEL_PRIMARY, data, ELM_SEL_FORMAT_MARKUP, txt);
1997      }
1998 #endif
1999
2000    if (!_elm_config->desktop_entry)
2001      edje_object_part_text_viewport_object_set(wd->ent, "elm.text", _viewport_obj_get(data));
2002 }
2003
2004 static void
2005 _signal_magnifier_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2006 {
2007    Evas_Coord cx, cy, cw, ch;
2008    Widget_Data *wd = elm_widget_data_get(data);
2009    if (!wd) return;
2010
2011    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
2012    if (!wd->deferred_recalc_job)
2013      elm_widget_show_region_set(data, cx, cy, cw, ch, EINA_FALSE);
2014    else
2015      {
2016         wd->deferred_cur = EINA_TRUE;
2017         wd->cx = cx;
2018         wd->cy = cy;
2019         wd->cw = cw;
2020         wd->ch = ch;
2021      }
2022 }
2023
2024 static void
2025 _signal_selection_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2026 {
2027    Evas_Coord cx, cy, cw, ch;
2028    Widget_Data *wd = elm_widget_data_get(data);
2029    if (!wd) return;
2030    wd->have_selection = EINA_TRUE;
2031    wd->selmode = EINA_TRUE;
2032    evas_object_smart_callback_call(data, SIG_SELECTION_CHANGED, NULL);
2033    elm_selection_set(ELM_SEL_PRIMARY, obj, ELM_SEL_FORMAT_MARKUP,
2034                      elm_entry_selection_get(data));
2035
2036    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
2037    if (!wd->deferred_recalc_job)
2038      elm_widget_show_region_set(data, cx, cy, cw, ch, EINA_FALSE);
2039    else
2040      {
2041         wd->deferred_cur = EINA_TRUE;
2042         wd->cx = cx;
2043         wd->cy = cy;
2044         wd->cw = cw;
2045         wd->ch = ch;
2046      }
2047 }
2048
2049 static void
2050 _signal_selection_cleared(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2051 {
2052    Widget_Data *wd = elm_widget_data_get(data);
2053    if (!wd) return;
2054    if (!wd->have_selection) return;
2055    wd->have_selection = EINA_FALSE;
2056    wd->selmode = EINA_FALSE;
2057    evas_object_smart_callback_call(data, SIG_SELECTION_CLEARED, NULL);
2058    if (wd->sel_notify_handler)
2059      {
2060         if (wd->cut_sel)
2061           {
2062 #ifdef HAVE_ELEMENTARY_X
2063              Evas_Object *top;
2064
2065              top = elm_widget_top_get(data);
2066              if ((top) && (elm_win_xwindow_get(top)))
2067                elm_selection_set(ELM_SEL_PRIMARY, data, ELM_SEL_FORMAT_MARKUP,
2068                                  wd->cut_sel);
2069 #endif
2070              eina_stringshare_del(wd->cut_sel);
2071              wd->cut_sel = NULL;
2072           }
2073         else
2074           {
2075 #ifdef HAVE_ELEMENTARY_X
2076              Evas_Object *top;
2077
2078              top = elm_widget_top_get(data);
2079              if ((top) && (elm_win_xwindow_get(top)))
2080                elm_selection_clear(ELM_SEL_PRIMARY, data);
2081 #endif
2082           }
2083      }
2084
2085    if ((wd->api) && (wd->api->obj_hidemenu))
2086      {
2087         wd->api->obj_hidemenu(data);
2088      }
2089 }
2090
2091 static void
2092 _signal_entry_paste_request(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2093 {
2094    Widget_Data *wd = elm_widget_data_get(data);
2095    if (!wd) return;
2096    evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
2097    if (wd->sel_notify_handler)
2098      {
2099 #ifdef HAVE_ELEMENTARY_X
2100         Evas_Object *top;
2101
2102         top = elm_widget_top_get(data);
2103         if ((top) && (elm_win_xwindow_get(top)))
2104           {
2105              wd->selection_asked = EINA_TRUE;
2106              elm_selection_get(ELM_SEL_CLIPBOARD, ELM_SEL_FORMAT_MARKUP, data,
2107                                NULL, NULL);
2108           }
2109 #endif
2110      }
2111 }
2112
2113 static void
2114 _signal_entry_copy_notify(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2115 {
2116    Widget_Data *wd = elm_widget_data_get(data);
2117    if (!wd) return;
2118    evas_object_smart_callback_call(data, SIG_SELECTION_COPY, NULL);
2119    elm_selection_set(ELM_SEL_CLIPBOARD, obj, ELM_SEL_FORMAT_MARKUP,
2120                      elm_entry_selection_get(data));
2121 }
2122
2123 static void
2124 _signal_entry_cut_notify(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2125 {
2126    Widget_Data *wd = elm_widget_data_get(data);
2127    if (!wd) return;
2128    evas_object_smart_callback_call(data, SIG_SELECTION_CUT, NULL);
2129    elm_selection_set(ELM_SEL_CLIPBOARD, obj, ELM_SEL_FORMAT_MARKUP,
2130                      elm_entry_selection_get(data));
2131    edje_object_part_text_insert(wd->ent, "elm.text", "");
2132    wd->changed = EINA_TRUE;
2133    _sizing_eval(data);
2134 }
2135
2136 static void
2137 _signal_cursor_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2138 {
2139    Widget_Data *wd = elm_widget_data_get(data);
2140    if (!wd) return;
2141    wd->cursor_pos = edje_object_part_text_cursor_pos_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
2142    wd->cur_changed = EINA_TRUE;
2143    _recalc_cursor_geometry(data);
2144 }
2145
2146 static void
2147 _signal_anchor_down(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2148 {
2149    Widget_Data *wd = elm_widget_data_get(data);
2150    if (!wd) return;
2151 }
2152
2153 static void
2154 _signal_anchor_up(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2155 {
2156    Widget_Data *wd = elm_widget_data_get(data);
2157    if (!wd) return;
2158 }
2159
2160 static void
2161 _signal_anchor_clicked(void *data, Evas_Object *obj __UNUSED__, const char *emission, const char *source __UNUSED__)
2162 {
2163    Widget_Data *wd = elm_widget_data_get(data);
2164    Elm_Entry_Anchor_Info ei;
2165    char *buf2, *p, *p2, *n;
2166    if (!wd) return;
2167    p = strrchr(emission, ',');
2168    if (p)
2169      {
2170         const Eina_List *geoms;
2171
2172         n = p + 1;
2173         p2 = p -1;
2174         while (p2 >= emission)
2175           {
2176              if (*p2 == ',') break;
2177              p2--;
2178           }
2179         p2++;
2180         buf2 = alloca(5 + p - p2);
2181         strncpy(buf2, p2, p - p2);
2182         buf2[p - p2] = 0;
2183         ei.name = n;
2184         ei.button = atoi(buf2);
2185         ei.x = ei.y = ei.w = ei.h = 0;
2186         geoms =
2187            edje_object_part_text_anchor_geometry_get(wd->ent, "elm.text", ei.name);
2188         if (geoms)
2189           {
2190              Evas_Textblock_Rectangle *r;
2191              const Eina_List *l;
2192              Evas_Coord px, py, x, y;
2193
2194              evas_object_geometry_get(wd->ent, &x, &y, NULL, NULL);
2195              evas_pointer_canvas_xy_get(evas_object_evas_get(wd->ent), &px, &py);
2196              EINA_LIST_FOREACH(geoms, l, r)
2197                {
2198                   if (((r->x + x) <= px) && ((r->y + y) <= py) &&
2199                       ((r->x + x + r->w) > px) && ((r->y + y + r->h) > py))
2200                     {
2201                        ei.x = r->x + x;
2202                        ei.y = r->y + y;
2203                        ei.w = r->w;
2204                        ei.h = r->h;
2205                        break;
2206                     }
2207                }
2208           }
2209         if (!wd->disabled)
2210           evas_object_smart_callback_call(data, SIG_ANCHOR_CLICKED, &ei);
2211      }
2212 }
2213
2214 static void
2215 _signal_anchor_move(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2216 {
2217    Widget_Data *wd = elm_widget_data_get(data);
2218    if (!wd) return;
2219 }
2220
2221 static void
2222 _signal_anchor_in(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2223 {
2224    Widget_Data *wd = elm_widget_data_get(data);
2225    if (!wd) return;
2226 }
2227
2228 static void
2229 _signal_anchor_out(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2230 {
2231    Widget_Data *wd = elm_widget_data_get(data);
2232    if (!wd) return;
2233 }
2234
2235 static void
2236 _signal_key_enter(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2237 {
2238    Widget_Data *wd = elm_widget_data_get(data);
2239    if (!wd) return;
2240    evas_object_smart_callback_call(data, SIG_ACTIVATED, NULL);
2241 }
2242
2243 static void
2244 _signal_mouse_down(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2245 {
2246    Widget_Data *wd = elm_widget_data_get(data);
2247    if (!wd) return;
2248    wd->double_clicked = EINA_FALSE;
2249    evas_object_smart_callback_call(data, SIG_PRESS, NULL);
2250
2251    if ((wd->api) && (wd->api->obj_hidemenu))
2252      wd->api->obj_hidemenu(data);
2253 }
2254
2255 static void
2256 _signal_mouse_clicked(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2257 {
2258    Widget_Data *wd = elm_widget_data_get(data);
2259    if (!wd) return;
2260    evas_object_smart_callback_call(data, SIG_CLICKED, NULL);
2261
2262    if (!_elm_config->desktop_entry && !wd->double_clicked)
2263      _cancel(data, NULL, NULL);
2264 }
2265
2266 static void
2267 _signal_mouse_double(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2268 {
2269    Widget_Data *wd = elm_widget_data_get(data);
2270    if (!wd) return;
2271    wd->double_clicked = EINA_TRUE;
2272    evas_object_smart_callback_call(data, SIG_CLICKED_DOUBLE, NULL);
2273 }
2274
2275 #ifdef HAVE_ELEMENTARY_X
2276 static Eina_Bool
2277 _event_selection_notify(void *data, int type __UNUSED__, void *event)
2278 {
2279    Widget_Data *wd = elm_widget_data_get(data);
2280    Ecore_X_Event_Selection_Notify *ev = event;
2281    if (!wd) return ECORE_CALLBACK_PASS_ON;
2282    if ((!wd->selection_asked) && (!wd->drag_selection_asked))
2283      return ECORE_CALLBACK_PASS_ON;
2284
2285    if ((ev->selection == ECORE_X_SELECTION_CLIPBOARD) ||
2286        (ev->selection == ECORE_X_SELECTION_PRIMARY))
2287      {
2288         Ecore_X_Selection_Data_Text *text_data;
2289
2290         text_data = ev->data;
2291         if (text_data->data.content == ECORE_X_SELECTION_CONTENT_TEXT)
2292           {
2293              if (text_data->text)
2294                {
2295                   char *txt = _elm_util_text_to_mkup(text_data->text);
2296
2297                   if (txt)
2298                     {
2299                        elm_entry_entry_insert(data, txt);
2300                        free(txt);
2301                     }
2302                }
2303           }
2304         wd->selection_asked = EINA_FALSE;
2305      }
2306    else if (ev->selection == ECORE_X_SELECTION_XDND)
2307      {
2308         Ecore_X_Selection_Data_Text *text_data;
2309
2310         text_data = ev->data;
2311         if (text_data->data.content == ECORE_X_SELECTION_CONTENT_TEXT)
2312           {
2313              if (text_data->text)
2314                {
2315                   char *txt = _elm_util_text_to_mkup(text_data->text);
2316
2317                   if (txt)
2318                     {
2319                        /* Massive FIXME: this should be at the drag point */
2320                        elm_entry_entry_insert(data, txt);
2321                        free(txt);
2322                     }
2323                }
2324           }
2325         wd->drag_selection_asked = EINA_FALSE;
2326
2327         ecore_x_dnd_send_finished();
2328
2329      }
2330    return ECORE_CALLBACK_PASS_ON;
2331 }
2332
2333 static Eina_Bool
2334 _event_selection_clear(void *data __UNUSED__, int type __UNUSED__, void *event __UNUSED__)
2335 {
2336 /*
2337    Widget_Data *wd = elm_widget_data_get(data);
2338    Ecore_X_Event_Selection_Clear *ev = event;
2339    if (!wd) return ECORE_CALLBACK_PASS_ON;
2340    if (!wd->have_selection) return ECORE_CALLBACK_PASS_ON;
2341    if ((ev->selection == ECORE_X_SELECTION_CLIPBOARD) ||
2342        (ev->selection == ECORE_X_SELECTION_PRIMARY))
2343      {
2344         elm_entry_select_none(data);
2345      }
2346    return 1; */
2347
2348    // start for cbhm
2349    Evas_Object *top = elm_widget_top_get(data);
2350    Ecore_X_Event_Selection_Clear *ev = event;
2351
2352    if (!top)
2353       return ECORE_CALLBACK_PASS_ON;
2354
2355    if (ev->selection != ECORE_X_SELECTION_SECONDARY)
2356      {
2357         return ECORE_CALLBACK_PASS_ON;
2358      }
2359
2360    if (cnpwidgetdata == data)
2361      {
2362         evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
2363         elm_selection_get(ELM_SEL_SECONDARY,ELM_SEL_FORMAT_MARKUP,data,NULL,NULL);
2364      }
2365
2366    // end for cbhm
2367    return ECORE_CALLBACK_PASS_ON;
2368 }
2369
2370 static Eina_Bool
2371 _drag_drop_cb(void *data __UNUSED__, Evas_Object *obj, Elm_Selection_Data *drop)
2372 {
2373    Widget_Data *wd;
2374    Eina_Bool rv;
2375
2376    wd = elm_widget_data_get(obj);
2377    if (!wd) return EINA_FALSE;
2378    printf("Inserting at (%d,%d) %s\n",drop->x,drop->y,(char*)drop->data);
2379
2380    edje_object_part_text_cursor_copy(wd->ent, "elm.text",
2381                                      EDJE_CURSOR_MAIN,/*->*/EDJE_CURSOR_USER);
2382    rv = edje_object_part_text_cursor_coord_set(wd->ent,"elm.text",
2383                                                EDJE_CURSOR_MAIN,drop->x,drop->y);
2384    if (!rv) printf("Warning: Failed to position cursor: paste anyway\n");
2385    elm_entry_entry_insert(obj, drop->data);
2386    edje_object_part_text_cursor_copy(wd->ent, "elm.text",
2387                                      EDJE_CURSOR_USER,/*->*/EDJE_CURSOR_MAIN);
2388
2389    return EINA_TRUE;
2390 }
2391 #endif
2392
2393 static Evas_Object *
2394 _get_item(void *data, Evas_Object *edje __UNUSED__, const char *part __UNUSED__, const char *item)
2395 {
2396    Widget_Data *wd = elm_widget_data_get(data);
2397    Evas_Object *o;
2398    Eina_List *l;
2399    Elm_Entry_Item_Provider *ip;
2400
2401    EINA_LIST_FOREACH(wd->item_providers, l, ip)
2402      {
2403         o = ip->func(ip->data, data, item);
2404         if (o) return o;
2405      }
2406    if (!strncmp(item, "file://", 7))
2407      {
2408         const char *fname = item + 7;
2409
2410         o = evas_object_image_filled_add(evas_object_evas_get(data));
2411         evas_object_image_file_set(o, fname, NULL);
2412         if (evas_object_image_load_error_get(o) == EVAS_LOAD_ERROR_NONE)
2413           {
2414              evas_object_show(o);
2415           }
2416         else
2417           {
2418              evas_object_del(o);
2419              o = edje_object_add(evas_object_evas_get(data));
2420              _elm_theme_object_set(data, o, "entry/emoticon", "wtf", elm_widget_style_get(data));
2421           }
2422         return o;
2423      }
2424    o = edje_object_add(evas_object_evas_get(data));
2425    if (!_elm_theme_object_set(data, o, "entry", item, elm_widget_style_get(data)))
2426      _elm_theme_object_set(data, o, "entry/emoticon", "wtf", elm_widget_style_get(data));
2427    return o;
2428 }
2429
2430 static int
2431 _strbuf_key_value_replace(Eina_Strbuf *srcbuf, char *key, const char *value, int deleteflag)
2432 {
2433    const char *srcstring = NULL;
2434    Eina_Strbuf *repbuf = NULL, *diffbuf = NULL;
2435    char *curlocater = NULL, *replocater;
2436    char *starttag, *endtag;
2437    int tagtxtlen = 0, insertflag = 0;
2438
2439    srcstring = eina_strbuf_string_get(srcbuf);
2440    if (srcstring)
2441      curlocater = strstr(srcstring, key);
2442    if (!curlocater || !srcstring)
2443      {
2444         insertflag = 1;
2445      }
2446    else
2447      {
2448        do
2449          {
2450            starttag = strchr(srcstring, '<');
2451            endtag = strchr(srcstring, '>');
2452            tagtxtlen = endtag - starttag;
2453            if (tagtxtlen <= 0) tagtxtlen = 0;
2454            if (starttag < curlocater && curlocater < endtag) break;
2455            if (endtag != NULL && endtag+1 != NULL)
2456              srcstring = endtag+1;
2457            else
2458              break;
2459          } while (strlen(srcstring) > 1);
2460
2461        if (starttag && endtag && tagtxtlen > strlen(key))
2462          {
2463            repbuf = eina_strbuf_new();
2464            diffbuf = eina_strbuf_new();
2465            eina_strbuf_append_n(repbuf, starttag, tagtxtlen);
2466            srcstring = eina_strbuf_string_get(repbuf);
2467            curlocater = strstr(srcstring, key);
2468
2469            if (curlocater != NULL)
2470              {
2471                replocater = curlocater + strlen(key) + 1;
2472
2473                while ((*replocater) && (*replocater != ' ') && (*replocater != '>'))
2474                  replocater++;
2475
2476                if (replocater-curlocater > strlen(key)+1)
2477                  {
2478                    eina_strbuf_append_n(diffbuf, curlocater, replocater-curlocater+1);
2479                  }
2480                else
2481                  insertflag = 1;
2482              }
2483            else
2484              {
2485                insertflag = 1;
2486              }
2487            eina_strbuf_reset(repbuf);
2488          }
2489        else
2490          {
2491            insertflag = 1;
2492          }
2493      }
2494
2495    if (repbuf == NULL) repbuf = eina_strbuf_new();
2496    if (diffbuf == NULL) diffbuf = eina_strbuf_new();
2497
2498    if (insertflag)
2499      {
2500        eina_strbuf_append_printf(repbuf, "<%s=%s>", key, value);
2501        eina_strbuf_prepend(srcbuf, eina_strbuf_string_get(repbuf));
2502      }
2503    else
2504      {
2505         if (deleteflag)
2506           {
2507             eina_strbuf_prepend(diffbuf, "<");
2508             eina_strbuf_append(diffbuf, ">");
2509             eina_strbuf_replace_first(srcbuf, eina_strbuf_string_get(diffbuf), "");
2510           }
2511         else
2512           {
2513             eina_strbuf_append_printf(repbuf, "%s=%s", key, value);
2514             eina_strbuf_replace_first(srcbuf, eina_strbuf_string_get(diffbuf), eina_strbuf_string_get(repbuf));
2515           }
2516      }
2517
2518    if (repbuf) eina_strbuf_free(repbuf);
2519    if (diffbuf) eina_strbuf_free(diffbuf);
2520
2521    return 0;
2522 }
2523
2524 static int
2525 _stringshare_key_value_replace(const char **srcstring, char *key, const char *value, int deleteflag)
2526 {
2527    Eina_Strbuf *sharebuf = NULL;
2528
2529    sharebuf = eina_strbuf_new();
2530    eina_strbuf_append(sharebuf, *srcstring);
2531    _strbuf_key_value_replace(sharebuf, key, value, deleteflag);
2532    eina_stringshare_del(*srcstring);
2533    *srcstring = eina_stringshare_add(eina_strbuf_string_get(sharebuf));
2534    eina_strbuf_free(sharebuf);
2535
2536    return 0;
2537 }
2538
2539 static void
2540 _text_filter(void *data, Evas_Object *edje __UNUSED__, const char *part __UNUSED__, Edje_Text_Filter_Type type, char **text)
2541 {
2542    Widget_Data *wd = elm_widget_data_get(data);
2543    Eina_List *l;
2544    Elm_Entry_Text_Filter *tf;
2545
2546    if (type == EDJE_TEXT_FILTER_FORMAT)
2547      return;
2548
2549    EINA_LIST_FOREACH(wd->text_filters, l, tf)
2550      {
2551         tf->func(tf->data, data, text);
2552         if (!*text)
2553           break;
2554      }
2555 }
2556
2557 /* This function is used to insert text by chunks in jobs */
2558 static Eina_Bool
2559 _text_append_idler(void *data)
2560 {
2561    int start;
2562    char backup;
2563    Evas_Object *obj = (Evas_Object *) data;
2564    Widget_Data *wd = elm_widget_data_get(obj);
2565    if (wd->text) eina_stringshare_del(wd->text);
2566    wd->text = NULL;
2567    if (wd->password_text) eina_stringshare_del(wd->password_text);
2568    wd->password_text = NULL;
2569    wd->changed = EINA_TRUE;
2570
2571    start = wd->append_text_position;
2572    if (start + _CHUNK_SIZE < wd->append_text_len)
2573      {
2574         int pos = start;
2575         int tag_start, esc_start;
2576
2577         tag_start = esc_start = -1;
2578         /* Find proper markup cut place */
2579         while (pos - start < _CHUNK_SIZE)
2580           {
2581              int prev_pos = pos;
2582              Eina_Unicode tmp =
2583                 eina_unicode_utf8_get_next(wd->append_text_left, &pos);
2584              if (esc_start == -1)
2585                {
2586                   if (tmp == '<')
2587                      tag_start = prev_pos;
2588                   else if (tmp == '>')
2589                      tag_start = -1;
2590                }
2591              else if (tag_start == -1)
2592                {
2593                   if (tmp == '&')
2594                      esc_start = prev_pos;
2595                   else if (tmp == ';')
2596                      esc_start = -1;
2597                }
2598           }
2599
2600         if (tag_start >= 0)
2601           {
2602              wd->append_text_position = tag_start;
2603           }
2604         else if (esc_start >= 0)
2605           {
2606              wd->append_text_position = esc_start;
2607           }
2608         else
2609           {
2610              wd->append_text_position = pos;
2611           }
2612      }
2613    else
2614      {
2615         wd->append_text_position = wd->append_text_len;
2616      }
2617
2618    backup = wd->append_text_left[wd->append_text_position];
2619    wd->append_text_left[wd->append_text_position] = '\0';
2620
2621    edje_object_part_text_append(wd->ent, "elm.text",
2622          wd->append_text_left + start);
2623
2624    wd->append_text_left[wd->append_text_position] = backup;
2625
2626    /* If there's still more to go, renew the idler, else, cleanup */
2627    if (wd->append_text_position < wd->append_text_len)
2628      {
2629         return ECORE_CALLBACK_RENEW;
2630      }
2631    else
2632      {
2633         free(wd->append_text_left);
2634         wd->append_text_left = NULL;
2635         wd->append_text_idler = NULL;
2636         return ECORE_CALLBACK_CANCEL;
2637      }
2638 }
2639
2640 static void
2641 _add_chars_till_limit(Evas_Object *obj, char **text, int can_add, Length_Unit unit)
2642 {
2643    int i = 0, unit_size;
2644    int current_len = strlen(*text);
2645    char *new_text = *text;
2646    if (unit >= LENGTH_UNIT_LAST) return;
2647    while (*new_text)
2648      {
2649         if (*new_text == '<')
2650           {
2651              while (*new_text != '>')
2652                {
2653                   new_text++;
2654                   if (!*new_text) break;
2655                }
2656              new_text++;
2657           }
2658         else
2659           {
2660              int index = 0;
2661              if (*new_text == '&')
2662                {
2663                   while (*(new_text + index) != ';')
2664                     {
2665                        index++;
2666                        if (!*(new_text + index)) break;
2667                     }
2668                }
2669              char *markup;
2670              index = evas_string_char_next_get(new_text, index, NULL);
2671              markup = malloc(index + 1);
2672              strncpy(markup, new_text, index);
2673              markup[index] = 0;
2674              if (unit == LENGTH_UNIT_BYTE)
2675                unit_size = strlen(elm_entry_markup_to_utf8(markup));
2676              else if (unit == LENGTH_UNIT_CHAR)
2677                unit_size = evas_string_char_len_get(elm_entry_markup_to_utf8(markup));
2678              if (markup)
2679                {
2680                   free(markup);
2681                   markup = NULL;
2682                }
2683              if (can_add < unit_size)
2684                {
2685                   if (!i)
2686                     {
2687                        evas_object_smart_callback_call(obj, "maxlength,reached", NULL);
2688                        free(*text);
2689                        *text = NULL;
2690                        return;
2691                     }
2692                   can_add = 0;
2693                   strncpy(new_text, new_text + index, current_len - ((new_text + index) - *text));
2694                   current_len -= index;
2695                   (*text)[current_len] = 0;
2696                }
2697              else
2698                {
2699                   new_text += index;
2700                   can_add -= unit_size;
2701                }
2702              i++;
2703           }
2704      }
2705    evas_object_smart_callback_call(obj, "maxlength,reached", NULL);
2706 }
2707
2708 static void
2709 _elm_entry_text_set(Evas_Object *obj, const char *item, const char *entry)
2710 {
2711    int len = 0;
2712    ELM_CHECK_WIDTYPE(obj, widtype);
2713    Widget_Data *wd = elm_widget_data_get(obj);
2714    if (!wd) return;
2715    if (!entry) entry = "";
2716    if (item && strcmp(item, "default"))
2717      {
2718         edje_object_part_text_set(wd->ent, item, entry);
2719         return;
2720      }
2721    if (wd->text) eina_stringshare_del(wd->text);
2722    wd->text = NULL;
2723    if (wd->password_text) eina_stringshare_del(wd->password_text);
2724    wd->password_text = NULL;
2725    wd->changed = EINA_TRUE;
2726
2727    /* Clear currently pending job if there is one */
2728    if (wd->append_text_idler)
2729      {
2730         ecore_idler_del(wd->append_text_idler);
2731         free(wd->append_text_left);
2732         wd->append_text_left = NULL;
2733         wd->append_text_idler = NULL;
2734      }
2735
2736    len = strlen(entry);
2737    /* Split to ~_CHUNK_SIZE chunks */
2738    if (len > _CHUNK_SIZE)
2739      {
2740         wd->append_text_left = (char *) malloc(len + 1);
2741      }
2742
2743    /* If we decided to use the idler */
2744    if (wd->append_text_left)
2745      {
2746         /* Need to clear the entry first */
2747         edje_object_part_text_set(wd->ent, "elm.text", "");
2748         memcpy(wd->append_text_left, entry, len + 1);
2749         wd->append_text_position = 0;
2750         wd->append_text_len = len;
2751         wd->append_text_idler = ecore_idler_add(_text_append_idler, obj);
2752      }
2753    else
2754      {
2755         edje_object_part_text_set(wd->ent, "elm.text", entry);
2756      }
2757 }
2758
2759 static const char *
2760 _elm_entry_text_get(const Evas_Object *obj, const char *item)
2761 {
2762    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2763    Widget_Data *wd = elm_widget_data_get(obj);
2764    if (item && strcmp(item, "default")) return NULL;
2765    const char *text;
2766    if (!wd) return NULL;
2767    if (wd->password)
2768      {
2769         if(wd->password_text) return wd->password_text;
2770      }
2771    else if (wd->text) 
2772      {
2773         return wd->text;
2774      }
2775    text = edje_object_part_text_get(wd->ent, "elm.text");
2776    if (!text)
2777      {
2778         ERR("text=NULL for edje %p, part 'elm.text'", wd->ent);
2779         return NULL;
2780      }
2781    eina_stringshare_replace(&wd->text, text);
2782    if (wd->password)
2783      {
2784         const char *pw_text;
2785         pw_text = elm_entry_markup_to_utf8(wd->text);
2786         if (pw_text)
2787           {
2788              eina_stringshare_replace(&wd->password_text, pw_text);
2789              free(pw_text);
2790              return wd->password_text;
2791           }
2792      }
2793    return wd->text;
2794 }
2795
2796 EAPI Evas_Object *
2797 elm_entry_add(Evas_Object *parent)
2798 {
2799    Evas_Object *obj, *top;
2800    Evas *e;
2801    Widget_Data *wd;
2802
2803    ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
2804
2805    ELM_SET_WIDTYPE(widtype, "entry");
2806    elm_widget_type_set(obj, "entry");
2807    elm_widget_sub_object_add(parent, obj);
2808    elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
2809    elm_widget_data_set(obj, wd);
2810    elm_widget_del_hook_set(obj, _del_hook);
2811    elm_widget_del_pre_hook_set(obj, _del_pre_hook);
2812    elm_widget_theme_hook_set(obj, _theme_hook);
2813    elm_widget_disable_hook_set(obj, _disable_hook);
2814    elm_widget_signal_emit_hook_set(obj, _signal_emit_hook);
2815    elm_widget_focus_region_hook_set(obj, _focus_region_hook);
2816    elm_widget_on_focus_region_hook_set(obj, _on_focus_region_hook);
2817    elm_widget_signal_callback_add_hook_set(obj, _signal_callback_add_hook);
2818    elm_widget_signal_callback_del_hook_set(obj, _signal_callback_del_hook);
2819    elm_object_cursor_set(obj, ELM_CURSOR_XTERM);
2820    elm_widget_can_focus_set(obj, EINA_TRUE);
2821    elm_widget_highlight_ignore_set(obj, EINA_TRUE);
2822    elm_widget_text_set_hook_set(obj, _elm_entry_text_set);
2823    elm_widget_text_get_hook_set(obj, _elm_entry_text_get);
2824    elm_widget_content_set_hook_set(obj, _content_set_hook);
2825    elm_widget_content_unset_hook_set(obj, _content_unset_hook);
2826    elm_widget_content_get_hook_set(obj, _content_get_hook);
2827
2828    evas_object_smart_callback_add(obj, "sub-object-del", _sub_del, wd);
2829    wd->scroller = elm_smart_scroller_add(e);
2830    elm_widget_sub_object_add(obj, wd->scroller);
2831    elm_smart_scroller_widget_set(wd->scroller, obj);
2832    elm_smart_scroller_object_theme_set(obj, wd->scroller, "scroller", "entry",
2833                                        elm_widget_style_get(obj));
2834    evas_object_size_hint_weight_set(wd->scroller, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
2835    evas_object_size_hint_align_set(wd->scroller, EVAS_HINT_FILL, EVAS_HINT_FILL);
2836    elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
2837    elm_smart_scroller_propagate_events_set(wd->scroller, EINA_TRUE);
2838
2839    wd->linewrap     = ELM_WRAP_WORD;
2840    wd->editable     = EINA_TRUE;
2841    wd->disabled     = EINA_FALSE;
2842    wd->context_menu = EINA_TRUE;
2843    wd->autosave     = EINA_TRUE;
2844    wd->textonly     = EINA_FALSE;
2845    wd->autoperiod   = EINA_TRUE;
2846
2847    wd->ent = edje_object_add(e);
2848    elm_widget_sub_object_add(obj, wd->ent);
2849    edje_object_item_provider_set(wd->ent, _get_item, obj);
2850    edje_object_text_insert_filter_callback_add(wd->ent,"elm.text", _text_filter, obj);
2851    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOVE, _move, obj);
2852    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_DOWN,
2853                                   _mouse_down, obj);
2854    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_UP,
2855                                   _mouse_up, obj);
2856    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_MOVE,
2857                                   _mouse_move, obj);
2858    evas_object_event_callback_add(obj, EVAS_CALLBACK_RESIZE, _resize, obj);
2859
2860    _elm_theme_object_set(obj, wd->ent, "entry", "base", "default");
2861    edje_object_signal_callback_add(wd->ent, "entry,changed", "elm.text",
2862                                    _signal_entry_changed, obj);
2863    edje_object_signal_callback_add(wd->ent, "preedit,changed", "elm.text",
2864                                    _signal_preedit_changed, obj);
2865    edje_object_signal_callback_add(wd->ent, "handler,move,start", "elm.text",
2866                                    _signal_handler_move_start, obj);
2867    edje_object_signal_callback_add(wd->ent, "handler,move,end", "elm.text",
2868                                    _signal_handler_move_end, obj);
2869    edje_object_signal_callback_add(wd->ent, "handler,moving", "elm.text",
2870                                    _signal_handler_moving, obj);
2871    edje_object_signal_callback_add(wd->ent, "selection,start", "elm.text",
2872                                    _signal_selection_start, obj);
2873    edje_object_signal_callback_add(wd->ent, "selection,end", "elm.text",
2874                                    _signal_selection_end, obj);
2875    edje_object_signal_callback_add(wd->ent, "long,pressed", "elm.text",
2876                                    _signal_long_pressed, obj);
2877    edje_object_signal_callback_add(wd->ent, "magnifier,changed", "elm.text",
2878                                    _signal_magnifier_changed, obj);
2879    edje_object_signal_callback_add(wd->ent, "selection,changed", "elm.text",
2880                                    _signal_selection_changed, obj);
2881    edje_object_signal_callback_add(wd->ent, "selection,cleared", "elm.text",
2882                                    _signal_selection_cleared, obj);
2883    edje_object_signal_callback_add(wd->ent, "entry,paste,request", "elm.text",
2884                                    _signal_entry_paste_request, obj);
2885    edje_object_signal_callback_add(wd->ent, "entry,copy,notify", "elm.text",
2886                                    _signal_entry_copy_notify, obj);
2887    edje_object_signal_callback_add(wd->ent, "entry,cut,notify", "elm.text",
2888                                    _signal_entry_cut_notify, obj);
2889    edje_object_signal_callback_add(wd->ent, "cursor,changed", "elm.text",
2890                                    _signal_cursor_changed, obj);
2891    edje_object_signal_callback_add(wd->ent, "anchor,mouse,down,*", "elm.text",
2892                                    _signal_anchor_down, obj);
2893    edje_object_signal_callback_add(wd->ent, "anchor,mouse,up,*", "elm.text",
2894                                    _signal_anchor_up, obj);
2895    edje_object_signal_callback_add(wd->ent, "anchor,mouse,clicked,*", "elm.text",
2896                                    _signal_anchor_clicked, obj);
2897    edje_object_signal_callback_add(wd->ent, "anchor,mouse,move,*", "elm.text",
2898                                    _signal_anchor_move, obj);
2899    edje_object_signal_callback_add(wd->ent, "anchor,mouse,in,*", "elm.text",
2900                                    _signal_anchor_in, obj);
2901    edje_object_signal_callback_add(wd->ent, "anchor,mouse,out,*", "elm.text",
2902                                    _signal_anchor_out, obj);
2903    edje_object_signal_callback_add(wd->ent, "entry,key,enter", "elm.text",
2904                                    _signal_key_enter, obj);
2905    edje_object_signal_callback_add(wd->ent, "mouse,down,1", "elm.text",
2906                                    _signal_mouse_down, obj);
2907    edje_object_signal_callback_add(wd->ent, "mouse,clicked,1", "elm.text",
2908                                    _signal_mouse_clicked, obj);
2909    edje_object_signal_callback_add(wd->ent, "mouse,down,1,double", "elm.text",
2910                                    _signal_mouse_double, obj);
2911    edje_object_part_text_set(wd->ent, "elm.text", "");
2912    if (_elm_config->desktop_entry)
2913      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
2914    elm_widget_resize_object_set(obj, wd->ent);
2915    _sizing_eval(obj);
2916
2917    elm_entry_input_panel_layout_set(obj, ELM_INPUT_PANEL_LAYOUT_NORMAL);
2918
2919    wd->input_panel_enable = edje_object_part_text_input_panel_enabled_get(wd->ent, "elm.text");
2920    wd->autocapital_type = edje_object_part_text_autocapital_type_get(wd->ent, "elm.text");
2921
2922 #ifdef HAVE_ELEMENTARY_X
2923    top = elm_widget_top_get(obj);
2924    if ((top) && (elm_win_xwindow_get(top)))
2925      {
2926         wd->sel_notify_handler =
2927            ecore_event_handler_add(ECORE_X_EVENT_SELECTION_NOTIFY,
2928                                    _event_selection_notify, obj);
2929         wd->sel_clear_handler =
2930            ecore_event_handler_add(ECORE_X_EVENT_SELECTION_CLEAR,
2931                                    _event_selection_clear, obj);
2932      }
2933
2934    elm_drop_target_add(obj, ELM_SEL_FORMAT_MARKUP | ELM_SEL_FORMAT_IMAGE,
2935                        _drag_drop_cb, NULL);
2936 #endif
2937
2938    entries = eina_list_prepend(entries, obj);
2939
2940    // module - find module for entry
2941    wd->api = _module(obj);
2942    // if found - hook in
2943    if ((wd->api) && (wd->api->obj_hook)) wd->api->obj_hook(obj);
2944
2945    _mirrored_set(obj, elm_widget_mirrored_get(obj));
2946    // TODO: convert Elementary to subclassing of Evas_Smart_Class
2947    // TODO: and save some bytes, making descriptions per-class and not instance!
2948    evas_object_smart_callbacks_descriptions_set(obj, _signals);
2949    return obj;
2950 }
2951
2952 EAPI void elm_entry_extension_module_data_get(Evas_Object *obj,Elm_Entry_Extension_data *ext_mod)
2953 {
2954    ELM_CHECK_WIDTYPE(obj, widtype);
2955    Widget_Data *wd = elm_widget_data_get(obj);
2956    if (!wd) return;
2957    ext_mod->cancel = _cancel;
2958    ext_mod->copy = _copy;
2959    ext_mod->cut = _cut;
2960    ext_mod->paste = _paste;
2961    ext_mod->select = _select;
2962    ext_mod->selectall = _selectall;
2963    ext_mod->ent = wd->ent;
2964    ext_mod->viewport_obj = _viewport_obj_get(obj);
2965    ext_mod->items = wd->items;
2966    ext_mod->editable = wd->editable;
2967    ext_mod->have_selection = wd->have_selection;
2968    ext_mod->password = wd->password;
2969    ext_mod->selmode = wd->selmode;
2970    ext_mod->cnpinit = _cnpinit;
2971    ext_mod->context_menu = wd->context_menu;
2972    ext_mod->textonly = wd->textonly;
2973 }
2974
2975 EAPI void
2976 elm_entry_single_line_set(Evas_Object *obj, Eina_Bool single_line)
2977 {
2978    ELM_CHECK_WIDTYPE(obj, widtype);
2979    Widget_Data *wd = elm_widget_data_get(obj);
2980    if (!wd) return;
2981    if (wd->single_line == single_line) return;
2982    wd->single_line = single_line;
2983    wd->linewrap = ELM_WRAP_NONE;
2984    elm_entry_cnp_textonly_set(obj, EINA_TRUE);
2985    _theme_hook(obj);
2986    if (wd->scroller)
2987      {
2988         if (wd->single_line)
2989           {
2990              elm_smart_scroller_policy_set(wd->scroller,
2991                                            ELM_SMART_SCROLLER_POLICY_OFF,
2992                                            ELM_SMART_SCROLLER_POLICY_OFF);
2993              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
2994           }
2995         else
2996           {
2997              const Elm_Scroller_Policy map[3] =
2998                {
2999                   ELM_SMART_SCROLLER_POLICY_AUTO,
3000                   ELM_SMART_SCROLLER_POLICY_ON,
3001                   ELM_SMART_SCROLLER_POLICY_OFF
3002                };
3003              elm_smart_scroller_policy_set(wd->scroller,
3004                                            map[wd->policy_h],
3005                                            map[wd->policy_v]);
3006              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3007           }
3008         _sizing_eval(obj);
3009      }
3010 }
3011
3012 EAPI Eina_Bool
3013 elm_entry_single_line_get(const Evas_Object *obj)
3014 {
3015    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3016    Widget_Data *wd = elm_widget_data_get(obj);
3017    if (!wd) return EINA_FALSE;
3018    return wd->single_line;
3019 }
3020
3021 EAPI void
3022 elm_entry_password_set(Evas_Object *obj, Eina_Bool password)
3023 {
3024    ELM_CHECK_WIDTYPE(obj, widtype);
3025    Widget_Data *wd = elm_widget_data_get(obj);
3026    if (!wd) return;
3027    if (wd->password == password) return;
3028    wd->password = password;
3029    wd->single_line = EINA_TRUE;
3030    wd->linewrap = ELM_WRAP_NONE;
3031    _theme_hook(obj);
3032 }
3033
3034 EAPI Eina_Bool
3035 elm_entry_password_get(const Evas_Object *obj)
3036 {
3037    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3038    Widget_Data *wd = elm_widget_data_get(obj);
3039    if (!wd) return EINA_FALSE;
3040    return wd->password;
3041 }
3042
3043 EAPI void
3044 elm_entry_entry_set(Evas_Object *obj, const char *entry)
3045 {
3046    _elm_entry_text_set(obj, NULL, entry);
3047 }
3048
3049 EAPI void
3050 elm_entry_entry_append(Evas_Object *obj, const char *entry)
3051 {
3052    int len = 0;
3053    ELM_CHECK_WIDTYPE(obj, widtype);
3054    Widget_Data *wd = elm_widget_data_get(obj);
3055    if (!wd) return;
3056    if (!entry) entry = "";
3057    wd->changed = EINA_TRUE;
3058
3059    len = strlen(entry);
3060    if (wd->append_text_left)
3061      {
3062         char *tmpbuf;
3063         tmpbuf = realloc(wd->append_text_left, wd->append_text_len + len + 1);
3064         if (!tmpbuf)
3065           {
3066              /* Do something */
3067              return;
3068           }
3069         wd->append_text_left = tmpbuf;
3070         memcpy(wd->append_text_left + wd->append_text_len, entry, len + 1);
3071         wd->append_text_len += len;
3072      }
3073    else
3074      {
3075         /* FIXME: Add chunked appending here (like in entry_set) */
3076         edje_object_part_text_append(wd->ent, "elm.text", entry);
3077      }
3078 }
3079
3080 EAPI const char *
3081 elm_entry_entry_get(const Evas_Object *obj)
3082 {
3083    return _elm_entry_text_get(obj, NULL);
3084 }
3085
3086 EAPI Eina_Bool
3087 elm_entry_is_empty(const Evas_Object *obj)
3088 {
3089    /* FIXME: until there's support for that in textblock, we just check
3090     * to see if the there is text or not. */
3091    ELM_CHECK_WIDTYPE(obj, widtype) EINA_TRUE;
3092    Widget_Data *wd = elm_widget_data_get(obj);
3093    const Evas_Object *tb;
3094    //Evas_Textblock_Cursor *cur;
3095    Eina_Bool ret;
3096    if (!wd) return EINA_TRUE;
3097
3098 #if 0
3099    /* It's a hack until we get the support suggested above.
3100     * We just create a cursor, point it to the begining, and then
3101     * try to advance it, if it can advance, the tb is not empty,
3102     * otherwise it is. */
3103    tb = edje_object_part_object_get(wd->ent, "elm.text");
3104    cur = evas_object_textblock_cursor_new((Evas_Object *) tb); /* This is
3105                                                                   actually, ok for the time being, thsese hackish stuff will be removed
3106                                                                   once evas 1.0 is out*/
3107    evas_textblock_cursor_pos_set(cur, 0);
3108    ret = evas_textblock_cursor_char_next(cur);
3109    evas_textblock_cursor_free(cur);
3110
3111    return !ret;
3112 #endif
3113
3114    char *str = elm_entry_markup_to_utf8(elm_entry_entry_get(obj));
3115    if (!str) return EINA_TRUE;
3116
3117    ret = (strlen(str) == 0);
3118
3119    free(str);
3120    return ret;
3121 }
3122
3123 EAPI const char *
3124 elm_entry_selection_get(const Evas_Object *obj)
3125 {
3126    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3127    Widget_Data *wd = elm_widget_data_get(obj);
3128    if (!wd) return NULL;
3129    return edje_object_part_text_selection_get(wd->ent, "elm.text");
3130 }
3131
3132 EAPI void
3133 elm_entry_entry_insert(Evas_Object *obj, const char *entry)
3134 {
3135    ELM_CHECK_WIDTYPE(obj, widtype);
3136    Widget_Data *wd = elm_widget_data_get(obj);
3137    if (!wd) return;
3138    edje_object_part_text_insert(wd->ent, "elm.text", entry);
3139    // start for cbhm
3140 #ifdef HAVE_ELEMENTARY_X
3141    if (cnpwidgetdata == obj)
3142       ecore_x_selection_secondary_set(elm_win_xwindow_get(obj), "",1);
3143 #endif
3144    // end for cbhm
3145    wd->changed = EINA_TRUE;
3146    _sizing_eval(obj);
3147 }
3148
3149 EAPI void
3150 elm_entry_line_wrap_set(Evas_Object *obj, Elm_Wrap_Type wrap)
3151 {
3152    ELM_CHECK_WIDTYPE(obj, widtype);
3153    Widget_Data *wd = elm_widget_data_get(obj);
3154    if (!wd) return;
3155    if (wd->linewrap == wrap) return;
3156    wd->lastw = -1;
3157    wd->linewrap = wrap;
3158    _theme_hook(obj);
3159 }
3160
3161 EAPI Elm_Wrap_Type
3162 elm_entry_line_wrap_get(const Evas_Object *obj)
3163 {
3164    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3165    Widget_Data *wd = elm_widget_data_get(obj);
3166    if (!wd) return EINA_FALSE;
3167    return wd->linewrap;
3168 }
3169
3170 EAPI void
3171 elm_entry_editable_set(Evas_Object *obj, Eina_Bool editable)
3172 {
3173    ELM_CHECK_WIDTYPE(obj, widtype);
3174    Widget_Data *wd = elm_widget_data_get(obj);
3175    if (!wd) return;
3176    if (wd->editable == editable) return;
3177    wd->editable = editable;
3178    _theme_hook(obj);
3179
3180 #ifdef HAVE_ELEMENTARY_X
3181    if (editable)
3182      elm_drop_target_add(obj, ELM_SEL_FORMAT_MARKUP, _drag_drop_cb, NULL);
3183    else
3184      elm_drop_target_del(obj);
3185 #endif
3186 }
3187
3188 EAPI Eina_Bool
3189 elm_entry_editable_get(const Evas_Object *obj)
3190 {
3191    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3192    Widget_Data *wd = elm_widget_data_get(obj);
3193    if (!wd) return EINA_FALSE;
3194    return wd->editable;
3195 }
3196
3197 EAPI void
3198 elm_entry_select_none(Evas_Object *obj)
3199 {
3200    ELM_CHECK_WIDTYPE(obj, widtype);
3201    Widget_Data *wd = elm_widget_data_get(obj);
3202    if (!wd) return;
3203    if (wd->selmode)
3204      {
3205         wd->selmode = EINA_FALSE;
3206         if (!_elm_config->desktop_entry)
3207           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
3208         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
3209      }
3210    wd->have_selection = EINA_FALSE;
3211    edje_object_part_text_select_none(wd->ent, "elm.text");
3212 }
3213
3214 EAPI void
3215 elm_entry_select_all(Evas_Object *obj)
3216 {
3217    ELM_CHECK_WIDTYPE(obj, widtype);
3218    Widget_Data *wd = elm_widget_data_get(obj);
3219    if (!wd) return;
3220    if (wd->selmode)
3221      {
3222         wd->selmode = EINA_FALSE;
3223         if (!_elm_config->desktop_entry)
3224           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
3225         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
3226      }
3227    wd->have_selection = EINA_TRUE;
3228    edje_object_part_text_select_all(wd->ent, "elm.text");
3229 }
3230
3231 EAPI Eina_Bool
3232 elm_entry_cursor_geometry_get(const Evas_Object *obj, Evas_Coord *x, Evas_Coord *y, Evas_Coord *w, Evas_Coord *h)
3233 {
3234    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3235    Widget_Data *wd = elm_widget_data_get(obj);
3236    if (!wd) return EINA_FALSE;
3237    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", x, y, w, h);
3238    return EINA_TRUE;
3239 }
3240
3241 EAPI Eina_Bool
3242 elm_entry_cursor_next(Evas_Object *obj)
3243 {
3244    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3245    Widget_Data *wd = elm_widget_data_get(obj);
3246    if (!wd) return EINA_FALSE;
3247    return edje_object_part_text_cursor_next(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3248 }
3249
3250 EAPI Eina_Bool
3251 elm_entry_cursor_prev(Evas_Object *obj)
3252 {
3253    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3254    Widget_Data *wd = elm_widget_data_get(obj);
3255    if (!wd) return EINA_FALSE;
3256    return edje_object_part_text_cursor_prev(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3257 }
3258
3259 EAPI Eina_Bool
3260 elm_entry_cursor_up(Evas_Object *obj)
3261 {
3262    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3263    Widget_Data *wd = elm_widget_data_get(obj);
3264    if (!wd) return EINA_FALSE;
3265    return edje_object_part_text_cursor_up(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3266 }
3267
3268 EAPI Eina_Bool
3269 elm_entry_cursor_down(Evas_Object *obj)
3270 {
3271    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3272    Widget_Data *wd = elm_widget_data_get(obj);
3273    if (!wd) return EINA_FALSE;
3274    return edje_object_part_text_cursor_down(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3275 }
3276
3277 EAPI void
3278 elm_entry_cursor_begin_set(Evas_Object *obj)
3279 {
3280    ELM_CHECK_WIDTYPE(obj, widtype);
3281    Widget_Data *wd = elm_widget_data_get(obj);
3282    if (!wd) return;
3283    edje_object_part_text_cursor_begin_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3284 }
3285
3286 EAPI void
3287 elm_entry_cursor_end_set(Evas_Object *obj)
3288 {
3289    ELM_CHECK_WIDTYPE(obj, widtype);
3290    Widget_Data *wd = elm_widget_data_get(obj);
3291    if (!wd) return;
3292    int x, y, w, h;
3293    edje_object_part_text_cursor_end_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3294    if (wd->scroll)
3295      {
3296         elm_widget_show_region_get(wd->ent, &x, &y, &w, &h);
3297         elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
3298      }
3299 }
3300
3301 EAPI void
3302 elm_entry_cursor_line_begin_set(Evas_Object *obj)
3303 {
3304    ELM_CHECK_WIDTYPE(obj, widtype);
3305    Widget_Data *wd = elm_widget_data_get(obj);
3306    if (!wd) return;
3307    edje_object_part_text_cursor_line_begin_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3308 }
3309
3310 EAPI void
3311 elm_entry_cursor_line_end_set(Evas_Object *obj)
3312 {
3313    ELM_CHECK_WIDTYPE(obj, widtype);
3314    Widget_Data *wd = elm_widget_data_get(obj);
3315    if (!wd) return;
3316    edje_object_part_text_cursor_line_end_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3317 }
3318
3319 EAPI void
3320 elm_entry_cursor_selection_begin(Evas_Object *obj)
3321 {
3322    ELM_CHECK_WIDTYPE(obj, widtype);
3323    Widget_Data *wd = elm_widget_data_get(obj);
3324    if (!wd) return;
3325    edje_object_part_text_select_begin(wd->ent, "elm.text");
3326 }
3327
3328 EAPI void
3329 elm_entry_cursor_selection_end(Evas_Object *obj)
3330 {
3331    ELM_CHECK_WIDTYPE(obj, widtype);
3332    Widget_Data *wd = elm_widget_data_get(obj);
3333    if (!wd) return;
3334    edje_object_part_text_select_extend(wd->ent, "elm.text");
3335 }
3336
3337 EAPI Eina_Bool
3338 elm_entry_cursor_is_format_get(const Evas_Object *obj)
3339 {
3340    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3341    Widget_Data *wd = elm_widget_data_get(obj);
3342    if (!wd) return EINA_FALSE;
3343    return edje_object_part_text_cursor_is_format_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3344 }
3345
3346 EAPI Eina_Bool
3347 elm_entry_cursor_is_visible_format_get(const Evas_Object *obj)
3348 {
3349    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3350    Widget_Data *wd = elm_widget_data_get(obj);
3351    if (!wd) return EINA_FALSE;
3352    return edje_object_part_text_cursor_is_visible_format_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3353 }
3354
3355 EAPI const char *
3356 elm_entry_cursor_content_get(const Evas_Object *obj)
3357 {
3358    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3359    Widget_Data *wd = elm_widget_data_get(obj);
3360    if (!wd) return NULL;
3361    return edje_object_part_text_cursor_content_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3362 }
3363
3364 EAPI void
3365 elm_entry_cursor_pos_set(Evas_Object *obj, int pos)
3366 {
3367    ELM_CHECK_WIDTYPE(obj, widtype);
3368    Widget_Data *wd = elm_widget_data_get(obj);
3369    if (!wd) return;
3370    edje_object_part_text_cursor_pos_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN, pos);
3371    edje_object_message_signal_process(wd->ent);
3372 }
3373
3374 EAPI int
3375 elm_entry_cursor_pos_get(const Evas_Object *obj)
3376 {
3377    ELM_CHECK_WIDTYPE(obj, widtype) 0;
3378    Widget_Data *wd = elm_widget_data_get(obj);
3379    if (!wd) return 0;
3380    return edje_object_part_text_cursor_pos_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3381 }
3382
3383 EAPI void
3384 elm_entry_selection_cut(Evas_Object *obj)
3385 {
3386    ELM_CHECK_WIDTYPE(obj, widtype);
3387    Widget_Data *wd = elm_widget_data_get(obj);
3388    if (!wd) return;
3389    _cut(obj, NULL, NULL);
3390 }
3391
3392 EAPI void
3393 elm_entry_selection_copy(Evas_Object *obj)
3394 {
3395    ELM_CHECK_WIDTYPE(obj, widtype);
3396    Widget_Data *wd = elm_widget_data_get(obj);
3397    if (!wd) return;
3398    _copy(obj, NULL, NULL);
3399 }
3400
3401 EAPI void
3402 elm_entry_selection_paste(Evas_Object *obj)
3403 {
3404    ELM_CHECK_WIDTYPE(obj, widtype);
3405    Widget_Data *wd = elm_widget_data_get(obj);
3406    if (!wd) return;
3407    _paste(obj, NULL, NULL);
3408 }
3409
3410 EAPI void
3411 elm_entry_context_menu_clear(Evas_Object *obj)
3412 {
3413    ELM_CHECK_WIDTYPE(obj, widtype);
3414    Widget_Data *wd = elm_widget_data_get(obj);
3415    Elm_Entry_Context_Menu_Item *it;
3416    if (!wd) return;
3417    EINA_LIST_FREE(wd->items, it)
3418      {
3419         eina_stringshare_del(it->label);
3420         eina_stringshare_del(it->icon_file);
3421         eina_stringshare_del(it->icon_group);
3422         free(it);
3423      }
3424 }
3425
3426 EAPI void
3427 elm_entry_context_menu_item_add(Evas_Object *obj, const char *label, const char *icon_file, Elm_Icon_Type icon_type, Evas_Smart_Cb func, const void *data)
3428 {
3429    ELM_CHECK_WIDTYPE(obj, widtype);
3430    Widget_Data *wd = elm_widget_data_get(obj);
3431    Elm_Entry_Context_Menu_Item *it;
3432    if (!wd) return;
3433    it = calloc(1, sizeof(Elm_Entry_Context_Menu_Item));
3434    if (!it) return;
3435    wd->items = eina_list_append(wd->items, it);
3436    it->obj = obj;
3437    it->label = eina_stringshare_add(label);
3438    it->icon_file = eina_stringshare_add(icon_file);
3439    it->icon_type = icon_type;
3440    it->func = func;
3441    it->data = (void *)data;
3442 }
3443
3444 EAPI void
3445 elm_entry_context_menu_disabled_set(Evas_Object *obj, Eina_Bool disabled)
3446 {
3447    ELM_CHECK_WIDTYPE(obj, widtype);
3448    Widget_Data *wd = elm_widget_data_get(obj);
3449    if (!wd) return;
3450    if (wd->context_menu == !disabled) return;
3451    wd->context_menu = !disabled;
3452 }
3453
3454 EAPI Eina_Bool
3455 elm_entry_context_menu_disabled_get(const Evas_Object *obj)
3456 {
3457    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3458    Widget_Data *wd = elm_widget_data_get(obj);
3459    if (!wd) return EINA_FALSE;
3460    return !wd->context_menu;
3461 }
3462
3463 EAPI void
3464 elm_entry_item_provider_append(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3465 {
3466    ELM_CHECK_WIDTYPE(obj, widtype);
3467    Widget_Data *wd = elm_widget_data_get(obj);
3468    if (!wd) return;
3469    EINA_SAFETY_ON_NULL_RETURN(func);
3470    Elm_Entry_Item_Provider *ip = calloc(1, sizeof(Elm_Entry_Item_Provider));
3471    if (!ip) return;
3472    ip->func = func;
3473    ip->data = data;
3474    wd->item_providers = eina_list_append(wd->item_providers, ip);
3475 }
3476
3477 EAPI void
3478 elm_entry_item_provider_prepend(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3479 {
3480    ELM_CHECK_WIDTYPE(obj, widtype);
3481    Widget_Data *wd = elm_widget_data_get(obj);
3482    if (!wd) return;
3483    EINA_SAFETY_ON_NULL_RETURN(func);
3484    Elm_Entry_Item_Provider *ip = calloc(1, sizeof(Elm_Entry_Item_Provider));
3485    if (!ip) return;
3486    ip->func = func;
3487    ip->data = data;
3488    wd->item_providers = eina_list_prepend(wd->item_providers, ip);
3489 }
3490
3491 EAPI void
3492 elm_entry_item_provider_remove(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3493 {
3494    ELM_CHECK_WIDTYPE(obj, widtype);
3495    Widget_Data *wd = elm_widget_data_get(obj);
3496    Eina_List *l;
3497    Elm_Entry_Item_Provider *ip;
3498    if (!wd) return;
3499    EINA_SAFETY_ON_NULL_RETURN(func);
3500    EINA_LIST_FOREACH(wd->item_providers, l, ip)
3501      {
3502         if ((ip->func == func) && ((!data) || (ip->data == data)))
3503           {
3504              wd->item_providers = eina_list_remove_list(wd->item_providers, l);
3505              free(ip);
3506              return;
3507           }
3508      }
3509 }
3510
3511 EAPI void
3512 elm_entry_text_filter_append(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3513 {
3514    Widget_Data *wd;
3515    Elm_Entry_Text_Filter *tf;
3516    ELM_CHECK_WIDTYPE(obj, widtype);
3517
3518    wd = elm_widget_data_get(obj);
3519
3520    EINA_SAFETY_ON_NULL_RETURN(func);
3521
3522    tf = _filter_new(func, data);
3523    if (!tf) return;
3524
3525    wd->text_filters = eina_list_append(wd->text_filters, tf);
3526 }
3527
3528 EAPI void
3529 elm_entry_text_filter_prepend(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3530 {
3531    Widget_Data *wd;
3532    Elm_Entry_Text_Filter *tf;
3533    ELM_CHECK_WIDTYPE(obj, widtype);
3534
3535    wd = elm_widget_data_get(obj);
3536
3537    EINA_SAFETY_ON_NULL_RETURN(func);
3538
3539    tf = _filter_new(func, data);
3540    if (!tf) return;
3541
3542    wd->text_filters = eina_list_prepend(wd->text_filters, tf);
3543 }
3544
3545 EAPI void
3546 elm_entry_text_filter_remove(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3547 {
3548    Widget_Data *wd;
3549    Eina_List *l;
3550    Elm_Entry_Text_Filter *tf;
3551    ELM_CHECK_WIDTYPE(obj, widtype);
3552
3553    wd = elm_widget_data_get(obj);
3554
3555    EINA_SAFETY_ON_NULL_RETURN(func);
3556
3557    EINA_LIST_FOREACH(wd->text_filters, l, tf)
3558      {
3559         if ((tf->func == func) && ((!data) || (tf->data == data)))
3560           {
3561              wd->text_filters = eina_list_remove_list(wd->text_filters, l);
3562              _filter_free(tf);
3563              return;
3564           }
3565      }
3566 }
3567
3568 EAPI char *
3569 elm_entry_markup_to_utf8(const char *s)
3570 {
3571    char *ss = _elm_util_mkup_to_text(s);
3572    if (!ss) ss = strdup("");
3573    return ss;
3574 }
3575
3576 EAPI char *
3577 elm_entry_utf8_to_markup(const char *s)
3578 {
3579    char *ss = _elm_util_text_to_mkup(s);
3580    if (!ss) ss = strdup("");
3581    return ss;
3582 }
3583
3584 EAPI void
3585 elm_entry_filter_limit_size(void *data, Evas_Object *entry, char **text)
3586 {
3587    Elm_Entry_Filter_Limit_Size *lim = data;
3588    char *current;
3589    int len, newlen;
3590    const char *(*text_get)(const Evas_Object *);
3591    const char *widget_type;
3592
3593    EINA_SAFETY_ON_NULL_RETURN(data);
3594    EINA_SAFETY_ON_NULL_RETURN(entry);
3595    EINA_SAFETY_ON_NULL_RETURN(text);
3596
3597    /* hack. I don't want to copy the entire function to work with
3598     * scrolled_entry */
3599    widget_type = elm_widget_type_get(entry);
3600    if (!strcmp(widget_type, "entry"))
3601      text_get = elm_entry_entry_get;
3602    else /* huh? */
3603      return;
3604
3605    current = elm_entry_markup_to_utf8(text_get(entry));
3606
3607    if (lim->max_char_count > 0)
3608      {
3609         len = evas_string_char_len_get(current);
3610         if (len >= lim->max_char_count)
3611           {
3612              evas_object_smart_callback_call(entry, "maxlength,reached", NULL);
3613              free(*text);
3614              free(current);
3615              *text = NULL;
3616              return;
3617           }
3618         newlen = evas_string_char_len_get(elm_entry_markup_to_utf8(*text));
3619         if ((len + newlen) > lim->max_char_count)
3620           _add_chars_till_limit(entry, text, (lim->max_char_count - len), LENGTH_UNIT_CHAR);
3621      }
3622    else if (lim->max_byte_count > 0)
3623      {
3624         len = strlen(current);
3625         if (len >= lim->max_byte_count)
3626           {
3627              evas_object_smart_callback_call(entry, "maxlength,reached", NULL);
3628              free(*text);
3629              free(current);
3630              *text = NULL;
3631              return;
3632           }
3633         newlen = strlen(elm_entry_markup_to_utf8(*text));
3634         if ((len + newlen) > lim->max_byte_count)
3635           _add_chars_till_limit(entry, text, (lim->max_byte_count - len), LENGTH_UNIT_BYTE);
3636      }
3637    free(current);
3638 }
3639
3640 EAPI void
3641 elm_entry_filter_accept_set(void *data, Evas_Object *entry __UNUSED__, char **text)
3642 {
3643    Elm_Entry_Filter_Accept_Set *as = data;
3644    const char *set;
3645    char *insert;
3646    Eina_Bool goes_in;
3647    int read_idx, last_read_idx = 0, read_char;
3648
3649    EINA_SAFETY_ON_NULL_RETURN(data);
3650    EINA_SAFETY_ON_NULL_RETURN(text);
3651
3652    if ((!as->accepted) && (!as->rejected))
3653      return;
3654
3655    if (as->accepted)
3656      {
3657         set = as->accepted;
3658         goes_in = EINA_TRUE;
3659      }
3660    else
3661      {
3662         set = as->rejected;
3663         goes_in = EINA_FALSE;
3664      }
3665
3666    insert = *text;
3667    read_idx = evas_string_char_next_get(*text, 0, &read_char);
3668    while (read_char)
3669      {
3670         int cmp_idx, cmp_char;
3671         Eina_Bool in_set = EINA_FALSE;
3672
3673         cmp_idx = evas_string_char_next_get(set, 0, &cmp_char);
3674         while (cmp_char)
3675           {
3676              if (read_char == cmp_char)
3677                {
3678                   in_set = EINA_TRUE;
3679                   break;
3680                }
3681              cmp_idx = evas_string_char_next_get(set, cmp_idx, &cmp_char);
3682           }
3683         if (in_set == goes_in)
3684           {
3685              int size = read_idx - last_read_idx;
3686              const char *src = (*text) + last_read_idx;
3687              if (src != insert)
3688                memcpy(insert, *text + last_read_idx, size);
3689              insert += size;
3690           }
3691         last_read_idx = read_idx;
3692         read_idx = evas_string_char_next_get(*text, read_idx, &read_char);
3693      }
3694    *insert = 0;
3695 }
3696
3697 EAPI void
3698 elm_entry_file_set(Evas_Object *obj, const char *file, Elm_Text_Format format)
3699 {
3700    ELM_CHECK_WIDTYPE(obj, widtype);
3701    Widget_Data *wd = elm_widget_data_get(obj);
3702    if (!wd) return;
3703    if (wd->delay_write)
3704      {
3705         ecore_timer_del(wd->delay_write);
3706         wd->delay_write = NULL;
3707      }
3708    if (wd->autosave) _save(obj);
3709    eina_stringshare_replace(&wd->file, file);
3710    wd->format = format;
3711    _load(obj);
3712 }
3713
3714 EAPI void
3715 elm_entry_file_get(const Evas_Object *obj, const char **file, Elm_Text_Format *format)
3716 {
3717    ELM_CHECK_WIDTYPE(obj, widtype);
3718    Widget_Data *wd = elm_widget_data_get(obj);
3719    if (!wd) return;
3720    if (file) *file = wd->file;
3721    if (format) *format = wd->format;
3722 }
3723
3724 EAPI void
3725 elm_entry_file_save(Evas_Object *obj)
3726 {
3727    ELM_CHECK_WIDTYPE(obj, widtype);
3728    Widget_Data *wd = elm_widget_data_get(obj);
3729    if (!wd) return;
3730    if (wd->delay_write)
3731      {
3732         ecore_timer_del(wd->delay_write);
3733         wd->delay_write = NULL;
3734      }
3735    _save(obj);
3736    wd->delay_write = ecore_timer_add(2.0, _delay_write, obj);
3737 }
3738
3739 EAPI void
3740 elm_entry_autosave_set(Evas_Object *obj, Eina_Bool autosave)
3741 {
3742    ELM_CHECK_WIDTYPE(obj, widtype);
3743    Widget_Data *wd = elm_widget_data_get(obj);
3744    if (!wd) return;
3745    wd->autosave = !!autosave;
3746 }
3747
3748 EAPI Eina_Bool
3749 elm_entry_autosave_get(const Evas_Object *obj)
3750 {
3751    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3752    Widget_Data *wd = elm_widget_data_get(obj);
3753    if (!wd) return EINA_FALSE;
3754    return wd->autosave;
3755 }
3756
3757 EAPI void
3758 elm_entry_cnp_textonly_set(Evas_Object *obj, Eina_Bool textonly)
3759 {
3760    Elm_Sel_Format format = ELM_SEL_FORMAT_MARKUP;
3761    ELM_CHECK_WIDTYPE(obj, widtype);
3762    Widget_Data *wd = elm_widget_data_get(obj);
3763    if (!wd) return;
3764    textonly = !!textonly;
3765    if (wd->textonly == textonly) return;
3766    wd->textonly = !!textonly;
3767    if (!textonly) format |= ELM_SEL_FORMAT_IMAGE;
3768 #ifdef HAVE_ELEMENTARY_X
3769    elm_drop_target_add(obj, format, _drag_drop_cb, NULL);
3770 #endif
3771 }
3772
3773 EAPI Eina_Bool
3774 elm_entry_cnp_textonly_get(const Evas_Object *obj)
3775 {
3776    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3777    Widget_Data *wd = elm_widget_data_get(obj);
3778    if (!wd) return EINA_FALSE;
3779    return wd->textonly;
3780 }
3781
3782 EAPI void
3783 elm_entry_scrollable_set(Evas_Object *obj, Eina_Bool scroll)
3784 {
3785    ELM_CHECK_WIDTYPE(obj, widtype);
3786    Widget_Data *wd = elm_widget_data_get(obj);
3787    if (!wd) return;
3788    scroll = !!scroll;
3789    if (wd->scroll == scroll) return;
3790    wd->scroll = scroll;
3791    if (wd->scroll)
3792      {
3793         elm_widget_sub_object_del(obj, wd->scroller);
3794         elm_widget_resize_object_set(obj, wd->scroller);
3795         elm_widget_sub_object_add(obj, wd->ent);
3796         elm_smart_scroller_child_set(wd->scroller, wd->ent);
3797         evas_object_show(wd->scroller);
3798         elm_widget_on_show_region_hook_set(obj, _show_region_hook, obj);
3799         if (wd->single_line)
3800           {
3801              elm_smart_scroller_policy_set(wd->scroller,
3802                                            ELM_SMART_SCROLLER_POLICY_OFF,
3803                                            ELM_SMART_SCROLLER_POLICY_OFF);
3804              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3805           }
3806         else
3807           {
3808              const Elm_Scroller_Policy map[3] =
3809                {
3810                   ELM_SMART_SCROLLER_POLICY_AUTO,
3811                   ELM_SMART_SCROLLER_POLICY_ON,
3812                   ELM_SMART_SCROLLER_POLICY_OFF
3813                };
3814              elm_smart_scroller_policy_set(wd->scroller,
3815                                            map[wd->policy_h],
3816                                            map[wd->policy_v]);
3817              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3818           }
3819      }
3820    else
3821      {
3822         elm_smart_scroller_child_set(wd->scroller, NULL);
3823         elm_widget_sub_object_del(obj, wd->ent);
3824         elm_widget_resize_object_set(obj, wd->ent);
3825         evas_object_smart_member_add(wd->scroller, obj);
3826         elm_widget_sub_object_add(obj, wd->scroller);
3827         evas_object_hide(wd->scroller);
3828         elm_widget_on_show_region_hook_set(obj, NULL, NULL);
3829      }
3830    wd->lastw = -1;
3831    _theme_hook(obj);
3832 }
3833
3834 EAPI Eina_Bool
3835 elm_entry_scrollable_get(const Evas_Object *obj)
3836 {
3837    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3838    Widget_Data *wd = elm_widget_data_get(obj);
3839    if (!wd) return EINA_FALSE;
3840    return wd->scroll;
3841 }
3842
3843 EAPI void
3844 elm_entry_icon_set(Evas_Object *obj, Evas_Object *icon)
3845 {
3846    ELM_CHECK_WIDTYPE(obj, widtype);
3847    Widget_Data *wd = elm_widget_data_get(obj);
3848    Evas_Object *edje;
3849    if (!wd) return;
3850    EINA_SAFETY_ON_NULL_RETURN(icon);
3851    if (wd->icon == icon) return;
3852    if (wd->icon) evas_object_del(wd->icon);
3853    wd->icon = icon;
3854    edje = elm_smart_scroller_edje_object_get(wd->scroller);
3855    if (!edje) return;
3856    edje_object_part_swallow(edje, "elm.swallow.icon", wd->icon);
3857    edje_object_signal_emit(edje, "elm,action,show,icon", "elm");
3858    _sizing_eval(obj);
3859 }
3860
3861 EAPI Evas_Object *
3862 elm_entry_icon_get(const Evas_Object *obj)
3863 {
3864    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3865    Widget_Data *wd = elm_widget_data_get(obj);
3866    if (!wd) return NULL;
3867    return wd->icon;
3868 }
3869
3870 EAPI Evas_Object *
3871 elm_entry_icon_unset(Evas_Object *obj)
3872 {
3873    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3874    Widget_Data *wd = elm_widget_data_get(obj);
3875    Evas_Object *ret = NULL;
3876    if (!wd) return NULL;
3877    if (wd->icon)
3878      {
3879         Evas_Object *edje = elm_smart_scroller_edje_object_get(wd->scroller);
3880         if (!edje) return NULL;
3881         ret = wd->icon;
3882         edje_object_part_unswallow(edje, wd->icon);
3883         edje_object_signal_emit(edje, "elm,action,hide,icon", "elm");
3884         wd->icon = NULL;
3885         _sizing_eval(obj);
3886      }
3887    return ret;
3888 }
3889
3890 EAPI void
3891 elm_entry_icon_visible_set(Evas_Object *obj, Eina_Bool setting)
3892 {
3893    ELM_CHECK_WIDTYPE(obj, widtype);
3894    Widget_Data *wd = elm_widget_data_get(obj);
3895    if ((!wd) || (!wd->icon)) return;
3896    if (setting)
3897       evas_object_hide(wd->icon);
3898    else
3899       evas_object_show(wd->icon);
3900    _sizing_eval(obj);
3901 }
3902
3903 EAPI void
3904 elm_entry_end_set(Evas_Object *obj, Evas_Object *end)
3905 {
3906    ELM_CHECK_WIDTYPE(obj, widtype);
3907    Widget_Data *wd = elm_widget_data_get(obj);
3908    Evas_Object *edje;
3909    if (!wd) return;
3910    EINA_SAFETY_ON_NULL_RETURN(end);
3911    if (wd->end == end) return;
3912    if (wd->end) evas_object_del(wd->end);
3913    wd->end = end;
3914    edje = elm_smart_scroller_edje_object_get(wd->scroller);
3915    if (!edje) return;
3916    edje_object_part_swallow(edje, "elm.swallow.end", wd->end);
3917    edje_object_signal_emit(edje, "elm,action,show,end", "elm");
3918    _sizing_eval(obj);
3919 }
3920
3921 EAPI Evas_Object *
3922 elm_entry_end_get(const Evas_Object *obj)
3923 {
3924    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3925    Widget_Data *wd = elm_widget_data_get(obj);
3926    if (!wd) return NULL;
3927    return wd->end;
3928 }
3929
3930 EAPI Evas_Object *
3931 elm_entry_end_unset(Evas_Object *obj)
3932 {
3933    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3934    Widget_Data *wd = elm_widget_data_get(obj);
3935    Evas_Object *ret = NULL;
3936    if (!wd) return NULL;
3937    if (wd->end)
3938      {
3939         Evas_Object *edje = elm_smart_scroller_edje_object_get(wd->scroller);
3940         if (!edje) return NULL;
3941         ret = wd->end;
3942         edje_object_part_unswallow(edje, wd->end);
3943         edje_object_signal_emit(edje, "elm,action,hide,end", "elm");
3944         wd->end = NULL;
3945         _sizing_eval(obj);
3946      }
3947    return ret;
3948 }
3949
3950 EAPI void
3951 elm_entry_end_visible_set(Evas_Object *obj, Eina_Bool setting)
3952 {
3953    ELM_CHECK_WIDTYPE(obj, widtype);
3954    Widget_Data *wd = elm_widget_data_get(obj);
3955    if ((!wd) || (!wd->end)) return;
3956    if (setting)
3957       evas_object_hide(wd->end);
3958    else
3959       evas_object_show(wd->end);
3960    _sizing_eval(obj);
3961 }
3962
3963 EAPI void
3964 elm_entry_scrollbar_policy_set(Evas_Object *obj, Elm_Scroller_Policy h, Elm_Scroller_Policy v)
3965 {
3966    ELM_CHECK_WIDTYPE(obj, widtype);
3967    Widget_Data *wd = elm_widget_data_get(obj);
3968    const Elm_Scroller_Policy map[3] =
3969      {
3970         ELM_SMART_SCROLLER_POLICY_AUTO,
3971         ELM_SMART_SCROLLER_POLICY_ON,
3972         ELM_SMART_SCROLLER_POLICY_OFF
3973      };
3974    if (!wd) return;
3975    wd->policy_h = h;
3976    wd->policy_v = v;
3977    elm_smart_scroller_policy_set(wd->scroller,
3978                                  map[wd->policy_h],
3979                                  map[wd->policy_v]);
3980 }
3981
3982 EAPI void
3983 elm_entry_bounce_set(Evas_Object *obj, Eina_Bool h_bounce, Eina_Bool v_bounce)
3984 {
3985    ELM_CHECK_WIDTYPE(obj, widtype);
3986    Widget_Data *wd = elm_widget_data_get(obj);
3987    if (!wd) return;
3988    elm_smart_scroller_bounce_allow_set(wd->scroller, h_bounce, v_bounce);
3989 }
3990
3991 EAPI void
3992 elm_entry_bounce_get(const Evas_Object *obj, Eina_Bool *h_bounce, Eina_Bool *v_bounce)
3993 {
3994    ELM_CHECK_WIDTYPE(obj, widtype);
3995    Widget_Data *wd = elm_widget_data_get(obj);
3996    if (!wd) return;
3997    elm_smart_scroller_bounce_allow_get(wd->scroller, h_bounce, v_bounce);
3998 }
3999
4000 EAPI void
4001 elm_entry_input_panel_layout_set(Evas_Object *obj, Elm_Input_Panel_Layout layout)
4002 {
4003    ELM_CHECK_WIDTYPE(obj, widtype);
4004    Widget_Data *wd = elm_widget_data_get(obj);
4005    if (!wd) return;
4006
4007    wd->input_panel_layout = layout;
4008
4009    edje_object_part_text_input_panel_layout_set(wd->ent, "elm.text", layout);
4010 }
4011
4012 EAPI Elm_Input_Panel_Layout
4013 elm_entry_input_panel_layout_get(Evas_Object *obj)
4014 {
4015    ELM_CHECK_WIDTYPE(obj, widtype) ELM_INPUT_PANEL_LAYOUT_INVALID;
4016    Widget_Data *wd = elm_widget_data_get(obj);
4017    if (!wd) return ELM_INPUT_PANEL_LAYOUT_INVALID;
4018
4019    return wd->input_panel_layout;
4020 }
4021
4022 EAPI void
4023 elm_entry_autocapital_type_set(Evas_Object *obj, Elm_Autocapital_Type autocapital_type)
4024 {
4025    ELM_CHECK_WIDTYPE(obj, widtype);
4026    Widget_Data *wd = elm_widget_data_get(obj);
4027    if (!wd) return;
4028
4029    wd->autocapital_type = autocapital_type;
4030    edje_object_part_text_autocapital_type_set(wd->ent, "elm.text", autocapital_type);
4031 }
4032
4033 EAPI Elm_Autocapital_Type
4034 elm_entry_autocapital_type_get(Evas_Object *obj)
4035 {
4036    ELM_CHECK_WIDTYPE(obj, widtype) ELM_AUTOCAPITAL_TYPE_NONE;
4037    Widget_Data *wd = elm_widget_data_get(obj);
4038    if (!wd) return ELM_AUTOCAPITAL_TYPE_NONE;
4039
4040    return wd->autocapital_type;
4041 }
4042
4043 EAPI void
4044 elm_entry_input_panel_enabled_set(Evas_Object *obj, Eina_Bool enabled)
4045 {
4046    ELM_CHECK_WIDTYPE(obj, widtype);
4047    Widget_Data *wd = elm_widget_data_get(obj);
4048    if (!wd) return;
4049
4050    wd->input_panel_enable = enabled;
4051    edje_object_part_text_input_panel_enabled_set(wd->ent, "elm.text", enabled);
4052 }
4053
4054 EINA_DEPRECATED EAPI void
4055 elm_entry_line_char_wrap_set(Evas_Object *obj, Eina_Bool wrap)
4056 {
4057    if (wrap) elm_entry_line_wrap_set(obj, ELM_WRAP_CHAR);
4058 }
4059
4060 EAPI void
4061 elm_entry_background_color_set(Evas_Object *obj, unsigned int r, unsigned int g, unsigned int b, unsigned int a)
4062 {
4063    ELM_CHECK_WIDTYPE(obj, widtype);
4064    Widget_Data *wd = elm_widget_data_get(obj);
4065    evas_object_color_set(wd->bg, r, g, b, a);
4066
4067    if (wd->bgcolor == EINA_FALSE)
4068      {
4069        wd->bgcolor = 1;
4070        edje_object_part_swallow(wd->ent, "entry.swallow.background", wd->bg);
4071      }
4072 }
4073
4074 EAPI void
4075 elm_entry_autocapitalization_set(Evas_Object *obj, Eina_Bool autocap)
4076 {
4077    ELM_CHECK_WIDTYPE(obj, widtype);
4078    Widget_Data *wd = elm_widget_data_get(obj);
4079    if (!wd) return;
4080
4081    if (autocap)
4082      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_SENTENCE;
4083    else
4084      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_NONE;
4085
4086    if (wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_URL ||
4087        wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_EMAIL)
4088      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_NONE;
4089
4090    edje_object_part_text_autocapital_type_set(wd->ent, "elm.text", wd->autocapital_type);
4091 }
4092
4093 EAPI void
4094 elm_entry_autoperiod_set(Evas_Object *obj, Eina_Bool autoperiod)
4095 {
4096    ELM_CHECK_WIDTYPE(obj, widtype);
4097    Widget_Data *wd = elm_widget_data_get(obj);
4098    if (!wd) return;
4099
4100    if (wd->password)
4101      wd->autoperiod = EINA_FALSE;
4102    else
4103      wd->autoperiod = autoperiod;
4104
4105    if (wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_URL ||
4106        wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_EMAIL)
4107      wd->autoperiod = EINA_FALSE;
4108
4109    edje_object_part_text_autoperiod_set(wd->ent, "elm.text", wd->autoperiod);
4110 }
4111
4112 EAPI void
4113 elm_entry_autoenable_returnkey_set(Evas_Object *obj, Eina_Bool on)
4114 {
4115    ELM_CHECK_WIDTYPE(obj, widtype);
4116    Widget_Data *wd = elm_widget_data_get(obj);
4117    if (!wd) return;
4118
4119    wd->autoreturnkey = on;
4120    _check_enable_returnkey(obj);
4121 }
4122
4123 EAPI Ecore_IMF_Context *elm_entry_imf_context_get(Evas_Object *obj)
4124 {
4125    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4126    Widget_Data *wd = elm_widget_data_get(obj);
4127    if (!wd || !wd->ent) return NULL;
4128
4129    return edje_object_part_text_imf_context_get(wd->ent, "elm.text");
4130 }
4131
4132 EAPI void
4133 elm_entry_matchlist_set(Evas_Object *obj, Eina_List *match_list, Eina_Bool case_sensitive)
4134 {
4135    Widget_Data *wd = elm_widget_data_get(obj);
4136    if (!wd) return;
4137
4138    if (match_list)
4139      {
4140         Evas_Coord max_w = 9999, max_h = 9999;
4141         const char* key_data = NULL;
4142
4143         wd->matchlist_threshold = 1;
4144         wd->hover = elm_hover_add(elm_widget_parent_get(obj));
4145         elm_hover_parent_set(wd->hover, elm_widget_parent_get(obj));
4146         elm_hover_target_set(wd->hover, obj);
4147         elm_object_style_set(wd->hover, "matchlist");
4148
4149         wd->layout = elm_layout_add(wd->hover);
4150         elm_layout_theme_set(wd->layout, "entry", "matchlist", "default");
4151         wd->list = elm_list_add(wd->layout);
4152         evas_object_size_hint_weight_set(wd->list, EVAS_HINT_EXPAND, 0.0);
4153         evas_object_size_hint_align_set(wd->list, EVAS_HINT_FILL, EVAS_HINT_FILL);
4154         elm_list_mode_set(wd->list, ELM_LIST_EXPAND);
4155         elm_object_style_set(wd->list, "matchlist");
4156
4157         key_data = edje_object_data_get(elm_layout_edje_get(wd->layout), "max_width");
4158         if (key_data) max_w = atoi(key_data);
4159         key_data = edje_object_data_get(elm_layout_edje_get(wd->layout), "max_height");
4160         if (key_data) max_h = atoi(key_data);
4161
4162         elm_list_go(wd->list);
4163         evas_object_size_hint_max_set(wd->list, max_w, max_h);
4164         evas_object_smart_callback_add(wd->list, "selected", _matchlist_list_clicked, obj);
4165         elm_layout_content_set(wd->layout, "elm.swallow.content", wd->list);
4166         elm_hover_content_set(wd->hover, "bottom", wd->layout);
4167
4168         wd->match_list = match_list;
4169      }
4170    else
4171      {
4172         if (wd->hover)
4173           evas_object_del(wd->hover);
4174
4175         wd->match_list = NULL;
4176      }
4177
4178    wd->matchlist_case_sensitive = case_sensitive;
4179 }
4180
4181 EAPI void
4182 elm_entry_magnifier_type_set(Evas_Object *obj, int type)
4183 {
4184    ELM_CHECK_WIDTYPE(obj, widtype);
4185    Widget_Data *wd = elm_widget_data_get(obj);
4186    if (!wd) return;
4187
4188    wd->mgf_type = type;
4189    _magnifier_create(obj);
4190 }
4191
4192 EAPI void
4193 elm_entry_wrap_width_set(Evas_Object *obj, Evas_Coord w)
4194 {
4195    Widget_Data *wd = elm_widget_data_get(obj);
4196    if (!wd) return;
4197    if (wd->wrap_w == w) return;
4198    wd->wrap_w = w;
4199    _sizing_eval(obj);
4200 }
4201
4202 EAPI Evas_Coord
4203 elm_entry_wrap_width_get(const Evas_Object *obj)
4204 {
4205    Widget_Data *wd = elm_widget_data_get(obj);
4206    if (!wd) return 0;
4207    return wd->wrap_w;
4208 }
4209
4210 EAPI void
4211 elm_entry_fontsize_set(Evas_Object *obj, int fontsize)
4212 {
4213    ELM_CHECK_WIDTYPE(obj, widtype);
4214    Widget_Data *wd = elm_widget_data_get(obj);
4215    Eina_Strbuf *fontbuf = NULL;
4216    int removeflag = 0;
4217    const char *t;
4218
4219    if (!wd) return;
4220    t = eina_stringshare_add(elm_entry_entry_get(obj));
4221    fontbuf = eina_strbuf_new();
4222    eina_strbuf_append_printf(fontbuf, "%d", fontsize);
4223
4224    if (fontsize == 0) removeflag = 1; // remove fontsize tag
4225
4226    if (_stringshare_key_value_replace(&t, "font_size", eina_strbuf_string_get(fontbuf), removeflag) == 0)
4227      {
4228        elm_entry_entry_set(obj, t);
4229        wd->changed = 1;
4230        _sizing_eval(obj);
4231      }
4232    eina_strbuf_free(fontbuf);
4233    eina_stringshare_del(t);
4234 }
4235
4236 EAPI void
4237 elm_entry_text_color_set(Evas_Object *obj, unsigned int r, unsigned int g, unsigned int b, unsigned int a)
4238 {
4239    ELM_CHECK_WIDTYPE(obj, widtype);
4240    Widget_Data *wd = elm_widget_data_get(obj);
4241    Eina_Strbuf *colorbuf = NULL;
4242    const char *t;
4243    int len;
4244
4245    if (!wd) return;
4246    t = eina_stringshare_add(elm_entry_entry_get(obj));
4247    len = strlen(t);
4248    if (len <= 0) return;
4249    colorbuf = eina_strbuf_new();
4250    eina_strbuf_append_printf(colorbuf, "#%02X%02X%02X%02X", r, g, b, a);
4251
4252    if (_stringshare_key_value_replace(&t, "color", eina_strbuf_string_get(colorbuf), 0) == 0)
4253      {
4254        elm_entry_entry_set(obj, t);
4255        wd->changed = 1;
4256        _sizing_eval(obj);
4257      }
4258    eina_strbuf_free(colorbuf);
4259    eina_stringshare_del(t);
4260 }
4261
4262 EAPI void
4263 elm_entry_text_align_set(Evas_Object *obj, const char *alignmode)
4264 {
4265    ELM_CHECK_WIDTYPE(obj, widtype);
4266    Widget_Data *wd = elm_widget_data_get(obj);
4267    int len;
4268    const char *t;
4269
4270    if (!wd) return;
4271    t = eina_stringshare_add(elm_entry_entry_get(obj));
4272    len = strlen(t);
4273    if (len <= 0) return;
4274
4275    if (_stringshare_key_value_replace(&t, "align", alignmode, 0) == 0)
4276      elm_entry_entry_set(obj, t);
4277
4278    wd->changed = 1;
4279    _sizing_eval(obj);
4280    eina_stringshare_del(t);
4281 }