Remove unnecessary differences with upstream
[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 _translate_hook(Evas_Object *obj)
946 {
947    evas_object_smart_callback_call(obj, "language,changed", NULL);
948 }
949
950 static void
951 _signal_emit_hook(Evas_Object *obj, const char *emission, const char *source)
952 {
953    Widget_Data *wd = elm_widget_data_get(obj);
954    if (!wd) return;
955    edje_object_signal_emit(wd->ent, emission, source);
956    if (wd->scroller)
957       edje_object_signal_emit(elm_smart_scroller_edje_object_get(wd->scroller),
958                               emission, source);
959 }
960
961 static void
962 _signal_callback_add_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
963 {
964    Widget_Data *wd = elm_widget_data_get(obj);
965    if (!wd) return;
966    edje_object_signal_callback_add(wd->ent, emission, source, func_cb, data);
967    if (wd->scroller)
968       edje_object_signal_callback_add(elm_smart_scroller_edje_object_get(wd->scroller),
969                                       emission, source, func_cb, data);
970 }
971
972 static void
973 _signal_callback_del_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
974 {
975    Widget_Data *wd = elm_widget_data_get(obj);
976    edje_object_signal_callback_del_full(wd->ent, emission, source, func_cb,
977                                         data);
978    if (wd->scroller)
979       edje_object_signal_callback_del_full(elm_smart_scroller_edje_object_get(wd->scroller),
980                                            emission, source, func_cb, data);
981 }
982
983 static void
984 _on_focus_region_hook(const Evas_Object *obj, Evas_Coord *x, Evas_Coord *y, Evas_Coord *w, Evas_Coord *h)
985 {
986    Widget_Data *wd = elm_widget_data_get(obj);
987    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", x, y, w, h);
988 }
989
990 static void
991 _focus_region_hook(Evas_Object *obj, Evas_Coord x, Evas_Coord y, Evas_Coord w, Evas_Coord h)
992 {
993    Widget_Data *wd = elm_widget_data_get(obj);
994    if (wd->scroll)
995       elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
996 }
997
998 static void
999 _show_region_hook(void *data, Evas_Object *obj)
1000 {
1001    Widget_Data *wd = elm_widget_data_get(data);
1002    Evas_Coord x, y, w, h;
1003    if (!wd) return;
1004    elm_widget_show_region_get(obj, &x, &y, &w, &h);
1005    if (wd->scroll)
1006      elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
1007 }
1008
1009 static void
1010 _sub_del(void *data, Evas_Object *obj, void *event_info)
1011 {
1012    Widget_Data *wd = data;
1013    Evas_Object *sub = event_info;
1014    Evas_Object *edje;
1015
1016    edje = elm_smart_scroller_edje_object_get(wd->scroller);
1017    if (sub == wd->icon)
1018      {
1019         wd->icon = NULL;
1020         if (edje)
1021           edje_object_signal_emit(edje, "elm,action,hide,icon", "elm");
1022      }
1023    else if (sub == wd->end)
1024      {
1025         wd->end = NULL;
1026         if (edje)
1027           edje_object_signal_emit(edje, "elm,action,hide,end", "elm");
1028      }
1029    _sizing_eval(obj);
1030 }
1031
1032 static void
1033 _hoversel_position(Evas_Object *obj)
1034 {
1035    Widget_Data *wd = elm_widget_data_get(obj);
1036    Evas_Coord cx, cy, cw, ch, x, y, mw, mh;
1037    if (!wd) return;
1038
1039    cx = cy = 0;
1040    cw = ch = 1;
1041    evas_object_geometry_get(wd->ent, &x, &y, NULL, NULL);
1042    if (wd->usedown)
1043      {
1044         cx = wd->downx - x;
1045         cy = wd->downy - y;
1046         cw = 1;
1047         ch = 1;
1048      }
1049    else
1050      edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text",
1051                                                &cx, &cy, &cw, &ch);
1052    evas_object_size_hint_min_get(wd->hoversel, &mw, &mh);
1053    if (cw < mw)
1054      {
1055         cx += (cw - mw) / 2;
1056         cw = mw;
1057      }
1058    if (ch < mh)
1059      {
1060         cy += (ch - mh) / 2;
1061         ch = mh;
1062      }
1063    evas_object_move(wd->hoversel, x + cx, y + cy);
1064    evas_object_resize(wd->hoversel, cw, ch);
1065 }
1066
1067 static void
1068 _move(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1069 {
1070    Widget_Data *wd = elm_widget_data_get(data);
1071
1072    if (wd->hoversel) _hoversel_position(data);
1073 }
1074
1075 static void
1076 _resize(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1077 {
1078    Widget_Data *wd = elm_widget_data_get(data);
1079    if (!wd) return;
1080
1081    if (wd->linewrap)
1082      {
1083         _sizing_eval(data);
1084      }
1085    else if (wd->scroll)
1086      {
1087         Evas_Coord vw = 0, vh = 0;
1088
1089         elm_smart_scroller_child_viewport_size_get(wd->scroller, &vw, &vh);
1090         if (vw < wd->entmw) vw = wd->entmw;
1091         if (vh < wd->entmh) vh = wd->entmh;
1092         evas_object_resize(wd->ent, vw, vh);
1093      }
1094    if (wd->hoversel) _hoversel_position(data);
1095 }
1096
1097 static void
1098 _hover_del(void *data)
1099 {
1100    Widget_Data *wd = elm_widget_data_get(data);
1101    if (!wd) return;
1102
1103    if (wd->hoversel)
1104      {
1105         evas_object_del(wd->hoversel);
1106         wd->hoversel = NULL;
1107      }
1108    wd->hovdeljob = NULL;
1109 }
1110
1111 static void
1112 _dismissed(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1113 {
1114    Widget_Data *wd = elm_widget_data_get(data);
1115    if (!wd) return;
1116    wd->usedown = 0;
1117    if (wd->hoversel) evas_object_hide(wd->hoversel);
1118    if (wd->selmode)
1119      {
1120         if (!_elm_config->desktop_entry)
1121           {
1122              if (!wd->password)
1123                edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
1124           }
1125      }
1126    elm_widget_scroll_freeze_pop(data);
1127    if (wd->hovdeljob) ecore_job_del(wd->hovdeljob);
1128    wd->hovdeljob = ecore_job_add(_hover_del, data);
1129 }
1130
1131 static void
1132 _selectall(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1133 {
1134    Widget_Data *wd = elm_widget_data_get(data);
1135    if (!wd) return;
1136    wd->selmode = EINA_TRUE;
1137    edje_object_part_text_select_none(wd->ent, "elm.text");
1138    edje_object_signal_emit(wd->ent, "elm,state,select,on", "elm");
1139    edje_object_part_text_select_all(wd->ent, "elm.text");
1140    elm_object_scroll_freeze_pop(data);
1141 }
1142
1143 static void
1144 _select(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1145 {
1146    Widget_Data *wd = elm_widget_data_get(data);
1147    if (!wd) return;
1148    wd->selmode = EINA_TRUE;
1149    edje_object_part_text_select_none(wd->ent, "elm.text");
1150    if (!_elm_config->desktop_entry)
1151      {
1152         if (!wd->password)
1153           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
1154      }
1155    edje_object_signal_emit(wd->ent, "elm,state,select,on", "elm");
1156    if (!_elm_config->desktop_entry)
1157       elm_object_scroll_freeze_pop(data);
1158       //elm_widget_scroll_hold_push(data);
1159 }
1160
1161 static void
1162 _paste(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1163 {
1164    Widget_Data *wd = elm_widget_data_get(data);
1165    if (!wd) return;
1166    evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
1167    if (wd->sel_notify_handler)
1168      {
1169 #ifdef HAVE_ELEMENTARY_X
1170         Elm_Sel_Format formats;
1171         wd->selection_asked = EINA_TRUE;
1172         formats = ELM_SEL_FORMAT_MARKUP;
1173         if (!wd->textonly)
1174           formats |= ELM_SEL_FORMAT_IMAGE;
1175         elm_selection_get(ELM_SEL_CLIPBOARD, formats, data, NULL, NULL);
1176 #endif
1177      }
1178 }
1179
1180 static void
1181 _store_selection(Elm_Sel_Type seltype, Evas_Object *obj)
1182 {
1183    Widget_Data *wd = elm_widget_data_get(obj);
1184    const char *sel;
1185    char *sel_str;
1186
1187    if (!wd) return;
1188    sel = edje_object_part_text_selection_get(wd->ent, "elm.text");
1189    sel_str = strdup(sel);
1190    if (!sel_str)
1191      return;
1192    if (wd->textonly)
1193      {
1194         while (EINA_TRUE)
1195           {
1196              char *startTag = NULL;
1197              char *endTag = NULL;
1198
1199              startTag = strstr(sel_str, "<item");
1200              if (!startTag)
1201                startTag = strstr(sel_str, "</item");
1202              if (startTag)
1203                endTag = strstr(startTag, ">");
1204              else
1205                break;
1206              if (!endTag || startTag > endTag)
1207                break;
1208
1209              size_t sindex = startTag - sel_str;
1210              size_t eindex = endTag - sel_str + 1;
1211
1212              Eina_Strbuf *buf = eina_strbuf_new();
1213              if (buf)
1214                {
1215                   eina_strbuf_append(buf, sel_str);
1216                   eina_strbuf_remove(buf, sindex, eindex);
1217                   sel_str = eina_strbuf_string_steal(buf);
1218                   eina_strbuf_free(buf);
1219                }
1220           }
1221      }
1222    elm_selection_set(seltype, obj, ELM_SEL_FORMAT_MARKUP, sel_str);
1223    if (seltype == ELM_SEL_CLIPBOARD)
1224      eina_stringshare_replace(&wd->cut_sel, sel_str);
1225    free(sel_str);
1226 }
1227
1228 static void
1229 _cut(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1230 {
1231    Widget_Data *wd = elm_widget_data_get(data);
1232
1233    /* Store it */
1234    wd->selmode = EINA_FALSE;
1235    if (!_elm_config->desktop_entry)
1236      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1237    edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1238    if (!_elm_config->desktop_entry)
1239      elm_widget_scroll_hold_pop(data);
1240    _store_selection(ELM_SEL_CLIPBOARD, data);
1241    edje_object_part_text_insert(wd->ent, "elm.text", "");
1242    edje_object_part_text_select_none(wd->ent, "elm.text");
1243 }
1244
1245 static void
1246 _copy(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1247 {
1248    Widget_Data *wd = elm_widget_data_get(data);
1249    if (!wd) return;
1250    wd->selmode = EINA_TRUE;
1251    if (!_elm_config->desktop_entry)
1252      {
1253         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1254         elm_widget_scroll_hold_pop(data);
1255      }
1256    _store_selection(ELM_SEL_CLIPBOARD, data);
1257 }
1258
1259 static void
1260 _cancel(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1261 {
1262    Widget_Data *wd = elm_widget_data_get(data);
1263    if (!wd) return;
1264    wd->selmode = EINA_FALSE;
1265    if (!_elm_config->desktop_entry)
1266      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1267    edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
1268    if (!_elm_config->desktop_entry)
1269      elm_widget_scroll_hold_pop(data);
1270    edje_object_part_text_select_none(wd->ent, "elm.text");
1271 }
1272
1273 static void
1274 _clipboard_menu(void *data, Evas_Object *obj, void *event_info __UNUSED__)
1275 {
1276    Widget_Data *wd = elm_widget_data_get(data);
1277    if (!wd) return;
1278
1279    // start for cbhm
1280 #ifdef HAVE_ELEMENTARY_X
1281    ecore_x_selection_secondary_set(elm_win_xwindow_get(obj), "",1);
1282 #endif
1283    cnpwidgetdata = data;
1284    elm_cbhm_helper_init(obj);
1285    if (elm_entry_cnp_textonly_get(obj))
1286      elm_cbhm_send_raw_data("show0");
1287    else
1288      elm_cbhm_send_raw_data("show1");
1289    // end for cbhm
1290 }
1291
1292 // start for cbhm
1293 static void
1294 _cnpinit(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1295 {
1296    Widget_Data *wd = elm_widget_data_get(data);
1297    if (!wd) return;
1298    cnpwidgetdata = data;
1299 }
1300 // end for cbhm
1301
1302
1303 static void
1304 _item_clicked(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1305 {
1306    Elm_Entry_Context_Menu_Item *it = data;
1307    Evas_Object *obj2 = it->obj;
1308    if (it->func) it->func(it->data, obj2, NULL);
1309 }
1310
1311 static void
1312 _menu_press(Evas_Object *obj)
1313 {
1314    Widget_Data *wd = elm_widget_data_get(obj);
1315    Evas_Object *top;
1316    const Eina_List *l;
1317    const Elm_Entry_Context_Menu_Item *it;
1318    if (!wd) return;
1319    if ((wd->api) && (wd->api->obj_longpress))
1320      {
1321         wd->api->obj_longpress(obj);
1322      }
1323    else if (wd->context_menu)
1324      {
1325         const char *context_menu_orientation;
1326
1327         if (wd->hoversel) evas_object_del(wd->hoversel);
1328         else elm_widget_scroll_freeze_push(obj);
1329         wd->hoversel = elm_hoversel_add(obj);
1330         context_menu_orientation = edje_object_data_get
1331            (wd->ent, "context_menu_orientation");
1332         if ((context_menu_orientation) &&
1333             (!strcmp(context_menu_orientation, "horizontal")))
1334           elm_hoversel_horizontal_set(wd->hoversel, EINA_TRUE);
1335         elm_object_style_set(wd->hoversel, "entry");
1336         elm_widget_sub_object_add(obj, wd->hoversel);
1337         elm_object_text_set(wd->hoversel, "Text");
1338         top = elm_widget_top_get(obj);
1339         if (top) elm_hoversel_hover_parent_set(wd->hoversel, top);
1340         evas_object_smart_callback_add(wd->hoversel, "dismissed", _dismissed, obj);
1341         if (!wd->selmode)
1342           {
1343              if (!wd->password)
1344                elm_hoversel_item_add(wd->hoversel, E_("Select"), NULL, ELM_ICON_NONE,
1345                                      _select, obj);
1346              if (1) // need way to detect if someone has a selection
1347                {
1348                   if (wd->editable)
1349                     elm_hoversel_item_add(wd->hoversel, E_("Paste"), NULL, ELM_ICON_NONE,
1350                                           _paste, obj);
1351                }
1352         // start for cbhm
1353              if ((!wd->password) && (wd->editable))
1354                elm_hoversel_item_add(wd->hoversel, E_("More"), NULL, ELM_ICON_NONE,
1355                                      _clipboard_menu, obj);
1356         // end for cbhm
1357           }
1358         else
1359           {
1360              if (!wd->password)
1361                {
1362                   if (wd->have_selection)
1363                     {
1364                        elm_hoversel_item_add(wd->hoversel, E_("Copy"), NULL, ELM_ICON_NONE,
1365                                              _copy, obj);
1366                        if (wd->editable)
1367                          elm_hoversel_item_add(wd->hoversel, E_("Cut"), NULL, ELM_ICON_NONE,
1368                                                _cut, obj);
1369                     }
1370                   elm_hoversel_item_add(wd->hoversel, E_("Cancel"), NULL, ELM_ICON_NONE,
1371                                         _cancel, obj);
1372         // start for cbhm
1373                   if (wd->editable)
1374                     elm_hoversel_item_add(wd->hoversel, E_("More"), NULL, ELM_ICON_NONE,
1375                                           _clipboard_menu, obj);
1376         // end for cbhm
1377                }
1378           }
1379         EINA_LIST_FOREACH(wd->items, l, it)
1380           {
1381              elm_hoversel_item_add(wd->hoversel, it->label, it->icon_file,
1382                                    it->icon_type, _item_clicked, it);
1383           }
1384         if (wd->hoversel)
1385           {
1386              _hoversel_position(obj);
1387              evas_object_show(wd->hoversel);
1388              elm_hoversel_hover_begin(wd->hoversel);
1389           }
1390         if (!_elm_config->desktop_entry)
1391           {
1392              edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
1393              edje_object_part_text_select_abort(wd->ent, "elm.text");
1394           }
1395      }
1396 }
1397
1398 static void
1399 _magnifier_hide(void *data)
1400 {
1401    Widget_Data *wd = elm_widget_data_get(data);
1402    if (!wd) return;
1403
1404    evas_object_hide(wd->mgf_bg);
1405    evas_object_hide(wd->mgf_clip);
1406
1407    if (wd->scroll)
1408      elm_smart_scroller_freeze_set(wd->scroller, EINA_FALSE);
1409 }
1410
1411 static void
1412 _magnifier_show(void *data)
1413 {
1414    Widget_Data *wd = elm_widget_data_get(data);
1415    if (!wd) return;
1416
1417    evas_object_show(wd->mgf_bg);
1418    evas_object_show(wd->mgf_clip);
1419 }
1420
1421 static void
1422 _magnifier_move(void *data)
1423 {
1424    Widget_Data *wd = elm_widget_data_get(data);
1425    if (!wd) return;
1426
1427    Evas_Coord x, y, w, h;
1428    Evas_Coord cx, cy, cw, ch, ox, oy;
1429
1430    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
1431
1432    if (wd->scroll)
1433      {
1434         evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1435         elm_smart_scroller_child_pos_get(wd->scroller, &ox, &oy);
1436         cx -= ox;
1437         cy -= oy;
1438      }
1439    else
1440      evas_object_geometry_get(data, &x, &y, &w, &h);
1441
1442    ox = oy = 0;
1443
1444    if ((cy + y) - wd->mgf_height < 0)
1445      oy = -1 * ((cy + y) - wd->mgf_height);
1446
1447    if (wd->mgf_type == _ENTRY_MAGNIFIER_FIXEDSIZE)
1448      evas_object_move(wd->mgf_bg, (cx + x + cw/2) + ox, (cy + y) - wd->mgf_height + oy);
1449    else if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1450      evas_object_move(wd->mgf_bg, x, (cy + y) - wd->mgf_height + oy);
1451    else
1452      return;
1453
1454    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);
1455 }
1456
1457 static void
1458 _magnifier_create(void *data)
1459 {
1460    Widget_Data *wd = elm_widget_data_get(data);
1461    Evas_Coord x, y, w, h, mw, mh;
1462    const char* key_data = NULL;
1463    double elm_scale;
1464
1465    if (!wd) return;
1466
1467    if (wd->mgf_proxy)
1468      {
1469         evas_object_image_source_unset(wd->mgf_proxy);
1470         evas_object_color_set(wd->mgf_proxy, 255, 255, 255, 0);
1471         evas_object_hide(wd->mgf_proxy);
1472         evas_object_clip_unset(wd->mgf_proxy);
1473         evas_object_del(wd->mgf_proxy);
1474      }
1475    if (wd->mgf_bg) evas_object_del(wd->mgf_bg);
1476    if (wd->mgf_clip) evas_object_del(wd->mgf_clip);
1477
1478    if (wd->scroll)
1479      evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1480    else
1481      evas_object_geometry_get(data, &x, &y, &w, &h);
1482
1483    if ((w <= 0) || (h <= 0))
1484      return;
1485
1486    wd->mgf_bg = edje_object_add(evas_object_evas_get(data));
1487
1488    if (wd->mgf_type == _ENTRY_MAGNIFIER_FIXEDSIZE)
1489      _elm_theme_object_set(data, wd->mgf_bg, "entry", "magnifier", "fixed-size");
1490    else if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1491      _elm_theme_object_set(data, wd->mgf_bg, "entry", "magnifier", "fill-width");
1492    else
1493      return;
1494
1495    wd->mgf_clip = evas_object_rectangle_add(evas_object_evas_get(data));
1496    evas_object_color_set(wd->mgf_clip, 255, 255, 255, 255);
1497    edje_object_part_swallow(wd->mgf_bg, "swallow", wd->mgf_clip);
1498
1499    key_data = edje_object_data_get(wd->mgf_bg, "height");
1500    if (key_data) wd->mgf_height = atoi(key_data);
1501    key_data = edje_object_data_get(wd->mgf_bg, "scale");
1502    if (key_data) wd->mgf_scale = atof(key_data);
1503
1504    elm_scale = elm_scale_get();
1505    wd->mgf_height = (int)((float)wd->mgf_height * elm_scale);
1506
1507    if (wd->mgf_type == _ENTRY_MAGNIFIER_FILLWIDTH)
1508      evas_object_resize(wd->mgf_bg, w, wd->mgf_height);
1509
1510    if (wd->scroll)
1511      {
1512         elm_smart_scroller_freeze_set(wd->scroller, EINA_TRUE);
1513         wd->mgf_proxy = evas_object_image_add(evas_object_evas_get(wd->scroller));
1514         evas_object_image_source_set(wd->mgf_proxy, wd->scroller);
1515      }
1516    else
1517      {
1518         wd->mgf_proxy = evas_object_image_add(evas_object_evas_get(data));
1519         evas_object_image_source_set(wd->mgf_proxy, data);
1520      }
1521
1522    mw = (Evas_Coord)((float)w * wd->mgf_scale);
1523    mh = (Evas_Coord)((float)h * wd->mgf_scale);
1524    if ((mw <= 0) || (mh <= 0))
1525      return;
1526
1527    evas_object_resize(wd->mgf_proxy, mw, mh);
1528    evas_object_image_fill_set(wd->mgf_proxy, 0, 0, mw, mh);
1529    evas_object_color_set(wd->mgf_proxy, 255, 255, 255, 255);
1530    evas_object_pass_events_set(wd->mgf_proxy, EINA_TRUE);
1531    evas_object_show(wd->mgf_proxy);
1532    evas_object_clip_set(wd->mgf_proxy, wd->mgf_clip);
1533
1534    evas_object_layer_set(wd->mgf_bg, EVAS_LAYER_MAX);
1535    evas_object_layer_set(wd->mgf_proxy, EVAS_LAYER_MAX);
1536 }
1537
1538 static Eina_Bool
1539 _signal_long_pressed(void *data)
1540 {
1541    Widget_Data *wd = elm_widget_data_get(data);
1542    if (!wd) return ECORE_CALLBACK_CANCEL;
1543
1544    wd->long_pressed = EINA_TRUE;
1545
1546    _cancel(data, NULL, NULL);
1547
1548    _magnifier_create(data);
1549    _magnifier_move(data);
1550    _magnifier_show(data);
1551    elm_object_scroll_freeze_push(data);
1552
1553    evas_object_smart_callback_call(data, SIG_LONGPRESSED, NULL);
1554    return ECORE_CALLBACK_CANCEL;
1555 }
1556
1557 static void
1558 _mouse_down(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1559 {
1560    Widget_Data *wd = elm_widget_data_get(data);
1561    Evas_Event_Mouse_Down *ev = event_info;
1562    if (!wd) return;
1563    if (wd->disabled) return;
1564    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return;
1565    wd->downx = ev->canvas.x;
1566    wd->downy = ev->canvas.y;
1567    wd->long_pressed = EINA_FALSE;
1568 }
1569
1570 static void
1571 _mouse_up(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1572 {
1573    Widget_Data *wd = elm_widget_data_get(data);
1574    Evas_Event_Mouse_Up *ev = event_info;
1575    if (!wd) return;
1576    if (wd->disabled) return;
1577    if (ev->button == 1)
1578      {
1579         if (!wd->double_clicked)
1580           {
1581              if ((wd->api) && (wd->api->obj_mouseup))
1582                wd->api->obj_mouseup(data);
1583           }
1584         _magnifier_hide(data);
1585         elm_object_scroll_freeze_pop(data);
1586
1587         if (wd->long_pressed)
1588           _menu_press(data);
1589      }
1590    else if (ev->button == 3)
1591      {
1592         wd->usedown = 1;
1593         _menu_press(data);
1594      }
1595 }
1596
1597 static void
1598 _mouse_move(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info)
1599 {
1600    Widget_Data *wd = elm_widget_data_get(data);
1601    Evas_Event_Mouse_Move *ev = event_info;
1602    if (!wd) return;
1603    if (wd->disabled) return;
1604
1605    if (ev->buttons == 1)
1606      {
1607         if (wd->long_pressed)
1608           {
1609              _magnifier_show(data);
1610              _magnifier_move(data);
1611           }
1612      }
1613 }
1614
1615 static const char *
1616 _getbase(Evas_Object *obj)
1617 {
1618    Widget_Data *wd = elm_widget_data_get(obj);
1619    if (!wd) return "base";
1620    if (wd->editable)
1621      {
1622         if (wd->password) return "base-password";
1623         else
1624           {
1625              if (wd->single_line) return "base-single";
1626              else
1627                {
1628                   switch (wd->linewrap)
1629                     {
1630                      case ELM_WRAP_CHAR:
1631                         return "base-charwrap";
1632                      case ELM_WRAP_WORD:
1633                         return "base";
1634                      case ELM_WRAP_MIXED:
1635                         return "base-mixedwrap";
1636                      case ELM_WRAP_NONE:
1637                      default:
1638                         return "base-nowrap";
1639                     }
1640                }
1641           }
1642      }
1643    else
1644      {
1645         if (wd->password) return "base-password";
1646         else
1647           {
1648              if (wd->single_line) return "base-single-noedit";
1649              else
1650                {
1651                   switch (wd->linewrap)
1652                     {
1653                      case ELM_WRAP_CHAR:
1654                         return "base-noedit-charwrap";
1655                      case ELM_WRAP_WORD:
1656                         return "base-noedit";
1657                      case ELM_WRAP_MIXED:
1658                         return "base-noedit-mixedwrap";
1659                      case ELM_WRAP_NONE:
1660                      default:
1661                         return "base-nowrap-noedit";
1662                     }
1663                }
1664           }
1665      }
1666    return "base";
1667 }
1668
1669
1670 static int
1671 _entry_length_get(Evas_Object *obj)
1672 {
1673    int len;
1674    const char *str = elm_entry_entry_get(obj);
1675    if (!str) return 0;
1676
1677    char *plain_str = _elm_util_mkup_to_text(str);
1678    if (!plain_str) return 0;
1679
1680    len = strlen(plain_str);
1681    free(plain_str);
1682
1683    return len;
1684 }
1685
1686 #ifndef HAVE_STRCASESTR
1687 char* _strcasestr(const char *s, const char *find)
1688 {
1689    char c, sc;
1690    size_t len;
1691
1692    if ((c = *find++) != 0) {
1693       c = tolower((unsigned char) c);
1694       len = strlen(find);
1695       do {
1696          do {
1697             if( (sc = *s++) == 0)
1698                return NULL;
1699          } while ((char)tolower((unsigned char)sc) != c);
1700       } while (strncasecmp(s, find, len) != 0);
1701       s--;
1702    }
1703    return ((char*) s);
1704 }
1705 #endif
1706
1707 static void
1708 _matchlist_show(void *data)
1709 {
1710    Widget_Data *wd = elm_widget_data_get(data);
1711    const char *text = NULL;
1712    int textlen = 0;
1713    char *str_list = NULL, *str_result = NULL;
1714    char *str_mkup = NULL, *str_front = NULL, *str_mid = NULL;
1715
1716    Eina_List *l;
1717    Eina_Bool textfound = EINA_FALSE;
1718
1719    if (!wd) return;
1720    if (elm_widget_disabled_get(data)) return;
1721
1722    wd->matchlist_job = NULL;
1723
1724    if (wd->matchlist_list_clicked)
1725      {
1726         evas_object_hide(wd->hover);
1727         wd->matchlist_list_clicked = EINA_FALSE;
1728         return;
1729      }
1730    text = elm_entry_entry_get(data);
1731    if (text == NULL)
1732      return;
1733    textlen = strlen(text);
1734
1735    if (textlen < wd->matchlist_threshold)
1736      {
1737         evas_object_hide(wd->hover);
1738         return;
1739      }
1740
1741    evas_object_hide(wd->hover);
1742
1743    if (wd->match_list)
1744      {
1745         elm_list_clear(wd->list);
1746         EINA_LIST_FOREACH(wd->match_list, l, str_list)
1747           {
1748              if (wd->matchlist_case_sensitive)
1749                str_result = strstr(str_list, text);
1750              else
1751 #ifdef HAVE_STRCASESTR
1752                 str_result = strcasestr(str_list, text);
1753 #else
1754                 str_result = _strcasestr(str_list, text);
1755 #endif
1756
1757              if (str_result)
1758                {
1759                   str_mkup = malloc(strlen(str_list) + 16);
1760                   if (str_mkup == NULL) return;
1761
1762                   textlen = strlen(str_list) - strlen(str_result);
1763                   str_front = malloc(textlen + 1);
1764                   if (str_front == NULL)
1765                     {
1766                        free(str_mkup);
1767                        return;
1768                     }
1769
1770                   memset(str_front, 0, textlen + 1);
1771                   strncpy(str_front, str_list, textlen);
1772
1773                   textlen = strlen(text);
1774                   str_mid = malloc(textlen + 1);
1775                   if (str_mid == NULL)
1776                     {
1777                        free(str_mkup);
1778                        free(str_front);
1779                        return;
1780                     }
1781
1782                   memset(str_mid, 0, textlen + 1);
1783                   strncpy(str_mid, str_list + strlen(str_front), textlen);
1784
1785                   sprintf(str_mkup, "%s<match>%s</match>%s", str_front, str_mid, str_result + strlen(text));
1786
1787                   elm_list_item_append(wd->list, str_mkup, NULL, NULL, NULL, NULL);
1788
1789                   if (str_mkup) free(str_mkup);
1790                   if (str_front) free(str_front);
1791                   if (str_mid) free(str_mid);
1792
1793                   textfound=EINA_TRUE;
1794                }
1795           }
1796      }
1797    else
1798      return;
1799
1800    if (textfound)
1801      {
1802         elm_list_go(wd->list);
1803         evas_object_show(wd->hover);
1804         evas_object_raise(wd->hover);
1805      }
1806 }
1807
1808 static void _matchlist_list_clicked( void *data, Evas_Object *obj, void *event_info )
1809 {
1810    Elm_List_Item *it = (Elm_List_Item *) elm_list_selected_item_get(obj);
1811    Widget_Data *wd = elm_widget_data_get(data);
1812    if ((it == NULL) || (wd == NULL))
1813      return;
1814
1815    const char *str = NULL;
1816    const char *text = elm_list_item_label_get(it);
1817    evas_object_smart_callback_call((Evas_Object *)data, "selected", (void *)text);
1818    if (wd->match_list)
1819      {
1820         if (text != NULL)
1821           {
1822              str = elm_entry_markup_to_utf8(text);
1823              elm_entry_entry_set(data, str);
1824              elm_entry_cursor_end_set(data);
1825              wd->matchlist_list_clicked = EINA_TRUE;
1826
1827              evas_object_smart_callback_call(data, SIG_MATCHLIST_CLICKED, elm_entry_markup_to_utf8(text));
1828              free(str);
1829           }
1830      }
1831    elm_widget_focus_set(data, EINA_TRUE);
1832 }
1833
1834 static void
1835 _entry_changed_common_handling(void *data, const char *event)
1836 {
1837    Widget_Data *wd = elm_widget_data_get(data);
1838    Evas_Coord minw, minh;
1839    if (!wd) return;
1840    wd->changed = EINA_TRUE;
1841    /* Reset the size hints which are no more relevant.
1842     * Keep the height, this is a hack, but doesn't really matter
1843     * cause we'll re-eval in a moment. */
1844    if (wd->scroll)
1845      {
1846         evas_object_size_hint_min_get(data, &minw, &minh);
1847         evas_object_size_hint_min_set(data, minw, minh);
1848      }
1849    else
1850      {
1851         evas_object_size_hint_min_get(data, NULL, &minh);
1852         evas_object_size_hint_min_set(data, -1, minh);
1853      }
1854
1855    _sizing_eval(data);
1856    if (wd->text) eina_stringshare_del(wd->text);
1857    wd->text = NULL;
1858    if (wd->password_text) eina_stringshare_del(wd->password_text);
1859    wd->password_text = NULL;
1860    _check_enable_returnkey(data);
1861    if (wd->delay_write)
1862      {
1863         ecore_timer_del(wd->delay_write);
1864         wd->delay_write = NULL;
1865      }
1866
1867    if ((wd->single_line) && (wd->match_list))
1868      {
1869         if (wd->matchlist_job) ecore_job_del(wd->matchlist_job);
1870         wd->matchlist_job = ecore_job_add(_matchlist_show, data);
1871      }
1872
1873    if ((wd->api) && (wd->api->obj_hidemenu))
1874      wd->api->obj_hidemenu(data);
1875
1876    if ((wd->autosave) && (wd->file))
1877      wd->delay_write = ecore_timer_add(2.0, _delay_write, data);
1878
1879    /* callback - this could call callbacks that delete the entry... thus...
1880     * any access to wd after this could be invalid */
1881    evas_object_smart_callback_call(data, event, NULL);
1882 }
1883
1884 static void
1885 _signal_entry_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1886 {
1887    Widget_Data *wd = elm_widget_data_get(data);
1888    if (!wd) return;
1889
1890    _entry_changed_common_handling(data, SIG_CHANGED);
1891 }
1892
1893 static void
1894 _signal_preedit_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1895 {
1896    Widget_Data *wd = elm_widget_data_get(data);
1897    if (!wd) return;
1898
1899    _entry_changed_common_handling(data, SIG_PREEDIT_CHANGED);
1900 }
1901
1902 static void
1903 _signal_handler_move_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1904 {
1905    Widget_Data *wd = elm_widget_data_get(data);
1906    if (!wd) return;
1907
1908    elm_object_scroll_freeze_push(data);
1909
1910    if ((wd->api) && (wd->api->obj_hidemenu))
1911      wd->api->obj_hidemenu(data);
1912
1913    _magnifier_create(data);
1914    _magnifier_move(data);
1915    _magnifier_show(data);
1916 }
1917
1918 static void
1919 _signal_handler_move_end(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1920 {
1921    Widget_Data *wd = elm_widget_data_get(data);
1922    if (!wd) return;
1923
1924    elm_object_scroll_freeze_pop(data);
1925
1926    if (wd->have_selection)
1927      {
1928         _magnifier_hide(data);
1929         _menu_press(data);
1930      }
1931 }
1932
1933 static void
1934 _signal_handler_moving(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1935 {
1936    _magnifier_move(data);
1937    _magnifier_show(data);
1938 }
1939
1940 static Evas_Object *
1941 _viewport_obj_get(Evas_Object *data)
1942 {
1943    Widget_Data *wd = elm_widget_data_get(data);
1944    if (!wd) return NULL;
1945
1946    if(!data || !strlen(elm_widget_type_get(data)))
1947      return NULL;
1948
1949    Evas_Coord x, y, w, h;
1950    x = y = w = h = -1;
1951
1952    if (wd->scroll)
1953      {
1954         //evas_object_geometry_get(wd->scroller, &x, &y, &w, &h);
1955         //printf(">>> wd->scroller (%d, %d, %d, %d) \n", x, y, w, h);
1956         return wd->scroller;
1957      }
1958
1959    Evas_Object *parent_obj = data;
1960
1961    while(parent_obj = elm_widget_parent_get(parent_obj))
1962      {
1963         //evas_object_geometry_get(parent_obj, &x, &y, &w, &h);
1964         //printf(">>> %s (%d, %d, %d, %d) \n", elm_widget_type_get(parent_obj), x, y, w, h);
1965         if (!strcmp(elm_widget_type_get(parent_obj), "scroller") ||
1966             !strcmp(elm_widget_type_get(parent_obj), "genlist"))
1967           return parent_obj;
1968      }
1969
1970    return NULL;
1971 }
1972
1973 static void
1974 _signal_selection_end(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1975 {
1976    _magnifier_hide(data);
1977    _menu_press(data);
1978 }
1979
1980 static void
1981 _signal_selection_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
1982 {
1983    Widget_Data *wd = elm_widget_data_get(data);
1984    const Eina_List *l;
1985    Evas_Object *entry;
1986    if (!wd) return;
1987    EINA_LIST_FOREACH(entries, l, entry)
1988      {
1989         if (entry != data) elm_entry_select_none(entry);
1990      }
1991    wd->have_selection = EINA_TRUE;
1992    wd->selmode = EINA_TRUE;
1993    evas_object_smart_callback_call(data, SIG_SELECTION_START, NULL);
1994 #ifdef HAVE_ELEMENTARY_X
1995    if (wd->sel_notify_handler)
1996      {
1997         const char *txt = elm_entry_selection_get(data);
1998         Evas_Object *top;
1999
2000         top = elm_widget_top_get(data);
2001         if ((top) && (elm_win_xwindow_get(top)))
2002           elm_selection_set(ELM_SEL_PRIMARY, data, ELM_SEL_FORMAT_MARKUP, txt);
2003      }
2004 #endif
2005
2006    if (!_elm_config->desktop_entry)
2007      edje_object_part_text_viewport_object_set(wd->ent, "elm.text", _viewport_obj_get(data));
2008 }
2009
2010 static void
2011 _signal_magnifier_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2012 {
2013    Evas_Coord cx, cy, cw, ch;
2014    Widget_Data *wd = elm_widget_data_get(data);
2015    if (!wd) return;
2016
2017    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
2018    if (!wd->deferred_recalc_job)
2019      elm_widget_show_region_set(data, cx, cy, cw, ch, EINA_FALSE);
2020    else
2021      {
2022         wd->deferred_cur = EINA_TRUE;
2023         wd->cx = cx;
2024         wd->cy = cy;
2025         wd->cw = cw;
2026         wd->ch = ch;
2027      }
2028 }
2029
2030 static void
2031 _signal_selection_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2032 {
2033    Evas_Coord cx, cy, cw, ch;
2034    Widget_Data *wd = elm_widget_data_get(data);
2035    if (!wd) return;
2036    wd->have_selection = EINA_TRUE;
2037    wd->selmode = EINA_TRUE;
2038    evas_object_smart_callback_call(data, SIG_SELECTION_CHANGED, NULL);
2039    elm_selection_set(ELM_SEL_PRIMARY, obj, ELM_SEL_FORMAT_MARKUP,
2040                      elm_entry_selection_get(data));
2041
2042    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", &cx, &cy, &cw, &ch);
2043    if (!wd->deferred_recalc_job)
2044      elm_widget_show_region_set(data, cx, cy, cw, ch, EINA_FALSE);
2045    else
2046      {
2047         wd->deferred_cur = EINA_TRUE;
2048         wd->cx = cx;
2049         wd->cy = cy;
2050         wd->cw = cw;
2051         wd->ch = ch;
2052      }
2053 }
2054
2055 static void
2056 _signal_selection_cleared(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2057 {
2058    Widget_Data *wd = elm_widget_data_get(data);
2059    if (!wd) return;
2060    if (!wd->have_selection) return;
2061    wd->have_selection = EINA_FALSE;
2062    wd->selmode = EINA_FALSE;
2063    evas_object_smart_callback_call(data, SIG_SELECTION_CLEARED, NULL);
2064    if (wd->sel_notify_handler)
2065      {
2066         if (wd->cut_sel)
2067           {
2068 #ifdef HAVE_ELEMENTARY_X
2069              Evas_Object *top;
2070
2071              top = elm_widget_top_get(data);
2072              if ((top) && (elm_win_xwindow_get(top)))
2073                elm_selection_set(ELM_SEL_PRIMARY, data, ELM_SEL_FORMAT_MARKUP,
2074                                  wd->cut_sel);
2075 #endif
2076              eina_stringshare_del(wd->cut_sel);
2077              wd->cut_sel = NULL;
2078           }
2079         else
2080           {
2081 #ifdef HAVE_ELEMENTARY_X
2082              Evas_Object *top;
2083
2084              top = elm_widget_top_get(data);
2085              if ((top) && (elm_win_xwindow_get(top)))
2086                elm_selection_clear(ELM_SEL_PRIMARY, data);
2087 #endif
2088           }
2089      }
2090
2091    if ((wd->api) && (wd->api->obj_hidemenu))
2092      {
2093         wd->api->obj_hidemenu(data);
2094      }
2095 }
2096
2097 static void
2098 _signal_entry_paste_request(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2099 {
2100    Widget_Data *wd = elm_widget_data_get(data);
2101    if (!wd) return;
2102    evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
2103    if (wd->sel_notify_handler)
2104      {
2105 #ifdef HAVE_ELEMENTARY_X
2106         Evas_Object *top;
2107
2108         top = elm_widget_top_get(data);
2109         if ((top) && (elm_win_xwindow_get(top)))
2110           {
2111              wd->selection_asked = EINA_TRUE;
2112              elm_selection_get(ELM_SEL_CLIPBOARD, ELM_SEL_FORMAT_MARKUP, data,
2113                                NULL, NULL);
2114           }
2115 #endif
2116      }
2117 }
2118
2119 static void
2120 _signal_entry_copy_notify(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2121 {
2122    Widget_Data *wd = elm_widget_data_get(data);
2123    if (!wd) return;
2124    evas_object_smart_callback_call(data, SIG_SELECTION_COPY, NULL);
2125    elm_selection_set(ELM_SEL_CLIPBOARD, obj, ELM_SEL_FORMAT_MARKUP,
2126                      elm_entry_selection_get(data));
2127 }
2128
2129 static void
2130 _signal_entry_cut_notify(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2131 {
2132    Widget_Data *wd = elm_widget_data_get(data);
2133    if (!wd) return;
2134    evas_object_smart_callback_call(data, SIG_SELECTION_CUT, NULL);
2135    elm_selection_set(ELM_SEL_CLIPBOARD, obj, ELM_SEL_FORMAT_MARKUP,
2136                      elm_entry_selection_get(data));
2137    edje_object_part_text_insert(wd->ent, "elm.text", "");
2138    wd->changed = EINA_TRUE;
2139    _sizing_eval(data);
2140 }
2141
2142 static void
2143 _signal_cursor_changed(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2144 {
2145    Widget_Data *wd = elm_widget_data_get(data);
2146    if (!wd) return;
2147    wd->cursor_pos = edje_object_part_text_cursor_pos_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
2148    wd->cur_changed = EINA_TRUE;
2149    _recalc_cursor_geometry(data);
2150 }
2151
2152 static void
2153 _signal_anchor_down(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2154 {
2155    Widget_Data *wd = elm_widget_data_get(data);
2156    if (!wd) return;
2157 }
2158
2159 static void
2160 _signal_anchor_up(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2161 {
2162    Widget_Data *wd = elm_widget_data_get(data);
2163    if (!wd) return;
2164 }
2165
2166 static void
2167 _signal_anchor_clicked(void *data, Evas_Object *obj __UNUSED__, const char *emission, const char *source __UNUSED__)
2168 {
2169    Widget_Data *wd = elm_widget_data_get(data);
2170    Elm_Entry_Anchor_Info ei;
2171    char *buf2, *p, *p2, *n;
2172    if (!wd) return;
2173    p = strrchr(emission, ',');
2174    if (p)
2175      {
2176         const Eina_List *geoms;
2177
2178         n = p + 1;
2179         p2 = p -1;
2180         while (p2 >= emission)
2181           {
2182              if (*p2 == ',') break;
2183              p2--;
2184           }
2185         p2++;
2186         buf2 = alloca(5 + p - p2);
2187         strncpy(buf2, p2, p - p2);
2188         buf2[p - p2] = 0;
2189         ei.name = n;
2190         ei.button = atoi(buf2);
2191         ei.x = ei.y = ei.w = ei.h = 0;
2192         geoms =
2193            edje_object_part_text_anchor_geometry_get(wd->ent, "elm.text", ei.name);
2194         if (geoms)
2195           {
2196              Evas_Textblock_Rectangle *r;
2197              const Eina_List *l;
2198              Evas_Coord px, py, x, y;
2199
2200              evas_object_geometry_get(wd->ent, &x, &y, NULL, NULL);
2201              evas_pointer_canvas_xy_get(evas_object_evas_get(wd->ent), &px, &py);
2202              EINA_LIST_FOREACH(geoms, l, r)
2203                {
2204                   if (((r->x + x) <= px) && ((r->y + y) <= py) &&
2205                       ((r->x + x + r->w) > px) && ((r->y + y + r->h) > py))
2206                     {
2207                        ei.x = r->x + x;
2208                        ei.y = r->y + y;
2209                        ei.w = r->w;
2210                        ei.h = r->h;
2211                        break;
2212                     }
2213                }
2214           }
2215         if (!wd->disabled)
2216           evas_object_smart_callback_call(data, SIG_ANCHOR_CLICKED, &ei);
2217      }
2218 }
2219
2220 static void
2221 _signal_anchor_move(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2222 {
2223    Widget_Data *wd = elm_widget_data_get(data);
2224    if (!wd) return;
2225 }
2226
2227 static void
2228 _signal_anchor_in(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2229 {
2230    Widget_Data *wd = elm_widget_data_get(data);
2231    if (!wd) return;
2232 }
2233
2234 static void
2235 _signal_anchor_out(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2236 {
2237    Widget_Data *wd = elm_widget_data_get(data);
2238    if (!wd) return;
2239 }
2240
2241 static void
2242 _signal_key_enter(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2243 {
2244    Widget_Data *wd = elm_widget_data_get(data);
2245    if (!wd) return;
2246    evas_object_smart_callback_call(data, SIG_ACTIVATED, NULL);
2247 }
2248
2249 static void
2250 _signal_mouse_down(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2251 {
2252    Widget_Data *wd = elm_widget_data_get(data);
2253    if (!wd) return;
2254    wd->double_clicked = EINA_FALSE;
2255    evas_object_smart_callback_call(data, SIG_PRESS, NULL);
2256
2257    if ((wd->api) && (wd->api->obj_hidemenu))
2258      wd->api->obj_hidemenu(data);
2259 }
2260
2261 static void
2262 _signal_mouse_clicked(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2263 {
2264    Widget_Data *wd = elm_widget_data_get(data);
2265    if (!wd) return;
2266    evas_object_smart_callback_call(data, SIG_CLICKED, NULL);
2267
2268    if (!_elm_config->desktop_entry && !wd->double_clicked)
2269      _cancel(data, NULL, NULL);
2270 }
2271
2272 static void
2273 _signal_mouse_double(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
2274 {
2275    Widget_Data *wd = elm_widget_data_get(data);
2276    if (!wd) return;
2277    wd->double_clicked = EINA_TRUE;
2278    evas_object_smart_callback_call(data, SIG_CLICKED_DOUBLE, NULL);
2279 }
2280
2281 #ifdef HAVE_ELEMENTARY_X
2282 static Eina_Bool
2283 _event_selection_notify(void *data, int type __UNUSED__, void *event)
2284 {
2285    Widget_Data *wd = elm_widget_data_get(data);
2286    Ecore_X_Event_Selection_Notify *ev = event;
2287    if (!wd) return ECORE_CALLBACK_PASS_ON;
2288    if ((!wd->selection_asked) && (!wd->drag_selection_asked))
2289      return ECORE_CALLBACK_PASS_ON;
2290
2291    if ((ev->selection == ECORE_X_SELECTION_CLIPBOARD) ||
2292        (ev->selection == ECORE_X_SELECTION_PRIMARY))
2293      {
2294         Ecore_X_Selection_Data_Text *text_data;
2295
2296         text_data = ev->data;
2297         if (text_data->data.content == ECORE_X_SELECTION_CONTENT_TEXT)
2298           {
2299              if (text_data->text)
2300                {
2301                   char *txt = _elm_util_text_to_mkup(text_data->text);
2302
2303                   if (txt)
2304                     {
2305                        elm_entry_entry_insert(data, txt);
2306                        free(txt);
2307                     }
2308                }
2309           }
2310         wd->selection_asked = EINA_FALSE;
2311      }
2312    else if (ev->selection == ECORE_X_SELECTION_XDND)
2313      {
2314         Ecore_X_Selection_Data_Text *text_data;
2315
2316         text_data = ev->data;
2317         if (text_data->data.content == ECORE_X_SELECTION_CONTENT_TEXT)
2318           {
2319              if (text_data->text)
2320                {
2321                   char *txt = _elm_util_text_to_mkup(text_data->text);
2322
2323                   if (txt)
2324                     {
2325                        /* Massive FIXME: this should be at the drag point */
2326                        elm_entry_entry_insert(data, txt);
2327                        free(txt);
2328                     }
2329                }
2330           }
2331         wd->drag_selection_asked = EINA_FALSE;
2332
2333         ecore_x_dnd_send_finished();
2334
2335      }
2336    return ECORE_CALLBACK_PASS_ON;
2337 }
2338
2339 static Eina_Bool
2340 _event_selection_clear(void *data __UNUSED__, int type __UNUSED__, void *event __UNUSED__)
2341 {
2342 /*
2343    Widget_Data *wd = elm_widget_data_get(data);
2344    Ecore_X_Event_Selection_Clear *ev = event;
2345    if (!wd) return ECORE_CALLBACK_PASS_ON;
2346    if (!wd->have_selection) return ECORE_CALLBACK_PASS_ON;
2347    if ((ev->selection == ECORE_X_SELECTION_CLIPBOARD) ||
2348        (ev->selection == ECORE_X_SELECTION_PRIMARY))
2349      {
2350         elm_entry_select_none(data);
2351      }
2352    return 1; */
2353
2354    // start for cbhm
2355    Evas_Object *top = elm_widget_top_get(data);
2356    Ecore_X_Event_Selection_Clear *ev = event;
2357
2358    if (!top)
2359       return ECORE_CALLBACK_PASS_ON;
2360
2361    if (ev->selection != ECORE_X_SELECTION_SECONDARY)
2362      {
2363         return ECORE_CALLBACK_PASS_ON;
2364      }
2365
2366    if (cnpwidgetdata == data)
2367      {
2368         evas_object_smart_callback_call(data, SIG_SELECTION_PASTE, NULL);
2369         elm_selection_get(ELM_SEL_SECONDARY,ELM_SEL_FORMAT_MARKUP,data,NULL,NULL);
2370      }
2371
2372    // end for cbhm
2373    return ECORE_CALLBACK_PASS_ON;
2374 }
2375
2376 static Eina_Bool
2377 _drag_drop_cb(void *data __UNUSED__, Evas_Object *obj, Elm_Selection_Data *drop)
2378 {
2379    Widget_Data *wd;
2380    Eina_Bool rv;
2381
2382    wd = elm_widget_data_get(obj);
2383    if (!wd) return EINA_FALSE;
2384    printf("Inserting at (%d,%d) %s\n",drop->x,drop->y,(char*)drop->data);
2385
2386    edje_object_part_text_cursor_copy(wd->ent, "elm.text",
2387                                      EDJE_CURSOR_MAIN,/*->*/EDJE_CURSOR_USER);
2388    rv = edje_object_part_text_cursor_coord_set(wd->ent,"elm.text",
2389                                                EDJE_CURSOR_MAIN,drop->x,drop->y);
2390    if (!rv) printf("Warning: Failed to position cursor: paste anyway\n");
2391    elm_entry_entry_insert(obj, drop->data);
2392    edje_object_part_text_cursor_copy(wd->ent, "elm.text",
2393                                      EDJE_CURSOR_USER,/*->*/EDJE_CURSOR_MAIN);
2394
2395    return EINA_TRUE;
2396 }
2397 #endif
2398
2399 static Evas_Object *
2400 _get_item(void *data, Evas_Object *edje __UNUSED__, const char *part __UNUSED__, const char *item)
2401 {
2402    Widget_Data *wd = elm_widget_data_get(data);
2403    Evas_Object *o;
2404    Eina_List *l;
2405    Elm_Entry_Item_Provider *ip;
2406
2407    EINA_LIST_FOREACH(wd->item_providers, l, ip)
2408      {
2409         o = ip->func(ip->data, data, item);
2410         if (o) return o;
2411      }
2412    if (!strncmp(item, "file://", 7))
2413      {
2414         const char *fname = item + 7;
2415
2416         o = evas_object_image_filled_add(evas_object_evas_get(data));
2417         evas_object_image_file_set(o, fname, NULL);
2418         if (evas_object_image_load_error_get(o) == EVAS_LOAD_ERROR_NONE)
2419           {
2420              evas_object_show(o);
2421           }
2422         else
2423           {
2424              evas_object_del(o);
2425              o = edje_object_add(evas_object_evas_get(data));
2426              _elm_theme_object_set(data, o, "entry/emoticon", "wtf", elm_widget_style_get(data));
2427           }
2428         return o;
2429      }
2430    o = edje_object_add(evas_object_evas_get(data));
2431    if (!_elm_theme_object_set(data, o, "entry", item, elm_widget_style_get(data)))
2432      _elm_theme_object_set(data, o, "entry/emoticon", "wtf", elm_widget_style_get(data));
2433    return o;
2434 }
2435
2436 static int
2437 _strbuf_key_value_replace(Eina_Strbuf *srcbuf, char *key, const char *value, int deleteflag)
2438 {
2439    const char *srcstring = NULL;
2440    Eina_Strbuf *repbuf = NULL, *diffbuf = NULL;
2441    char *curlocater = NULL, *replocater;
2442    char *starttag, *endtag;
2443    int tagtxtlen = 0, insertflag = 0;
2444
2445    srcstring = eina_strbuf_string_get(srcbuf);
2446    if (srcstring)
2447      curlocater = strstr(srcstring, key);
2448    if (!curlocater || !srcstring)
2449      {
2450         insertflag = 1;
2451      }
2452    else
2453      {
2454        do
2455          {
2456            starttag = strchr(srcstring, '<');
2457            endtag = strchr(srcstring, '>');
2458            tagtxtlen = endtag - starttag;
2459            if (tagtxtlen <= 0) tagtxtlen = 0;
2460            if (starttag < curlocater && curlocater < endtag) break;
2461            if (endtag != NULL && endtag+1 != NULL)
2462              srcstring = endtag+1;
2463            else
2464              break;
2465          } while (strlen(srcstring) > 1);
2466
2467        if (starttag && endtag && tagtxtlen > strlen(key))
2468          {
2469            repbuf = eina_strbuf_new();
2470            diffbuf = eina_strbuf_new();
2471            eina_strbuf_append_n(repbuf, starttag, tagtxtlen);
2472            srcstring = eina_strbuf_string_get(repbuf);
2473            curlocater = strstr(srcstring, key);
2474
2475            if (curlocater != NULL)
2476              {
2477                replocater = curlocater + strlen(key) + 1;
2478
2479                while ((*replocater) && (*replocater != ' ') && (*replocater != '>'))
2480                  replocater++;
2481
2482                if (replocater-curlocater > strlen(key)+1)
2483                  {
2484                    eina_strbuf_append_n(diffbuf, curlocater, replocater-curlocater+1);
2485                  }
2486                else
2487                  insertflag = 1;
2488              }
2489            else
2490              {
2491                insertflag = 1;
2492              }
2493            eina_strbuf_reset(repbuf);
2494          }
2495        else
2496          {
2497            insertflag = 1;
2498          }
2499      }
2500
2501    if (repbuf == NULL) repbuf = eina_strbuf_new();
2502    if (diffbuf == NULL) diffbuf = eina_strbuf_new();
2503
2504    if (insertflag)
2505      {
2506        eina_strbuf_append_printf(repbuf, "<%s=%s>", key, value);
2507        eina_strbuf_prepend(srcbuf, eina_strbuf_string_get(repbuf));
2508      }
2509    else
2510      {
2511         if (deleteflag)
2512           {
2513             eina_strbuf_prepend(diffbuf, "<");
2514             eina_strbuf_append(diffbuf, ">");
2515             eina_strbuf_replace_first(srcbuf, eina_strbuf_string_get(diffbuf), "");
2516           }
2517         else
2518           {
2519             eina_strbuf_append_printf(repbuf, "%s=%s", key, value);
2520             eina_strbuf_replace_first(srcbuf, eina_strbuf_string_get(diffbuf), eina_strbuf_string_get(repbuf));
2521           }
2522      }
2523
2524    if (repbuf) eina_strbuf_free(repbuf);
2525    if (diffbuf) eina_strbuf_free(diffbuf);
2526
2527    return 0;
2528 }
2529
2530 static int
2531 _stringshare_key_value_replace(const char **srcstring, char *key, const char *value, int deleteflag)
2532 {
2533    Eina_Strbuf *sharebuf = NULL;
2534
2535    sharebuf = eina_strbuf_new();
2536    eina_strbuf_append(sharebuf, *srcstring);
2537    _strbuf_key_value_replace(sharebuf, key, value, deleteflag);
2538    eina_stringshare_del(*srcstring);
2539    *srcstring = eina_stringshare_add(eina_strbuf_string_get(sharebuf));
2540    eina_strbuf_free(sharebuf);
2541
2542    return 0;
2543 }
2544
2545 static void
2546 _text_filter(void *data, Evas_Object *edje __UNUSED__, const char *part __UNUSED__, Edje_Text_Filter_Type type, char **text)
2547 {
2548    Widget_Data *wd = elm_widget_data_get(data);
2549    Eina_List *l;
2550    Elm_Entry_Text_Filter *tf;
2551
2552    if (type == EDJE_TEXT_FILTER_FORMAT)
2553      return;
2554
2555    EINA_LIST_FOREACH(wd->text_filters, l, tf)
2556      {
2557         tf->func(tf->data, data, text);
2558         if (!*text)
2559           break;
2560      }
2561 }
2562
2563 /* This function is used to insert text by chunks in jobs */
2564 static Eina_Bool
2565 _text_append_idler(void *data)
2566 {
2567    int start;
2568    char backup;
2569    Evas_Object *obj = (Evas_Object *) data;
2570    Widget_Data *wd = elm_widget_data_get(obj);
2571    if (wd->text) eina_stringshare_del(wd->text);
2572    wd->text = NULL;
2573    if (wd->password_text) eina_stringshare_del(wd->password_text);
2574    wd->password_text = NULL;
2575    wd->changed = EINA_TRUE;
2576
2577    start = wd->append_text_position;
2578    if (start + _CHUNK_SIZE < wd->append_text_len)
2579      {
2580         int pos = start;
2581         int tag_start, esc_start;
2582
2583         tag_start = esc_start = -1;
2584         /* Find proper markup cut place */
2585         while (pos - start < _CHUNK_SIZE)
2586           {
2587              int prev_pos = pos;
2588              Eina_Unicode tmp =
2589                 eina_unicode_utf8_get_next(wd->append_text_left, &pos);
2590              if (esc_start == -1)
2591                {
2592                   if (tmp == '<')
2593                      tag_start = prev_pos;
2594                   else if (tmp == '>')
2595                      tag_start = -1;
2596                }
2597              else if (tag_start == -1)
2598                {
2599                   if (tmp == '&')
2600                      esc_start = prev_pos;
2601                   else if (tmp == ';')
2602                      esc_start = -1;
2603                }
2604           }
2605
2606         if (tag_start >= 0)
2607           {
2608              wd->append_text_position = tag_start;
2609           }
2610         else if (esc_start >= 0)
2611           {
2612              wd->append_text_position = esc_start;
2613           }
2614         else
2615           {
2616              wd->append_text_position = pos;
2617           }
2618      }
2619    else
2620      {
2621         wd->append_text_position = wd->append_text_len;
2622      }
2623
2624    backup = wd->append_text_left[wd->append_text_position];
2625    wd->append_text_left[wd->append_text_position] = '\0';
2626
2627    edje_object_part_text_append(wd->ent, "elm.text",
2628          wd->append_text_left + start);
2629
2630    wd->append_text_left[wd->append_text_position] = backup;
2631
2632    /* If there's still more to go, renew the idler, else, cleanup */
2633    if (wd->append_text_position < wd->append_text_len)
2634      {
2635         return ECORE_CALLBACK_RENEW;
2636      }
2637    else
2638      {
2639         free(wd->append_text_left);
2640         wd->append_text_left = NULL;
2641         wd->append_text_idler = NULL;
2642         return ECORE_CALLBACK_CANCEL;
2643      }
2644 }
2645
2646 static void
2647 _add_chars_till_limit(Evas_Object *obj, char **text, int can_add, Length_Unit unit)
2648 {
2649    int i = 0, unit_size;
2650    int current_len = strlen(*text);
2651    char *new_text = *text;
2652    if (unit >= LENGTH_UNIT_LAST) return;
2653    while (*new_text)
2654      {
2655         if (*new_text == '<')
2656           {
2657              while (*new_text != '>')
2658                {
2659                   new_text++;
2660                   if (!*new_text) break;
2661                }
2662              new_text++;
2663           }
2664         else
2665           {
2666              int index = 0;
2667              if (*new_text == '&')
2668                {
2669                   while (*(new_text + index) != ';')
2670                     {
2671                        index++;
2672                        if (!*(new_text + index)) break;
2673                     }
2674                }
2675              char *markup;
2676              index = evas_string_char_next_get(new_text, index, NULL);
2677              markup = malloc(index + 1);
2678              strncpy(markup, new_text, index);
2679              markup[index] = 0;
2680              if (unit == LENGTH_UNIT_BYTE)
2681                unit_size = strlen(elm_entry_markup_to_utf8(markup));
2682              else if (unit == LENGTH_UNIT_CHAR)
2683                unit_size = evas_string_char_len_get(elm_entry_markup_to_utf8(markup));
2684              if (markup)
2685                {
2686                   free(markup);
2687                   markup = NULL;
2688                }
2689              if (can_add < unit_size)
2690                {
2691                   if (!i)
2692                     {
2693                        evas_object_smart_callback_call(obj, "maxlength,reached", NULL);
2694                        free(*text);
2695                        *text = NULL;
2696                        return;
2697                     }
2698                   can_add = 0;
2699                   strncpy(new_text, new_text + index, current_len - ((new_text + index) - *text));
2700                   current_len -= index;
2701                   (*text)[current_len] = 0;
2702                }
2703              else
2704                {
2705                   new_text += index;
2706                   can_add -= unit_size;
2707                }
2708              i++;
2709           }
2710      }
2711    evas_object_smart_callback_call(obj, "maxlength,reached", NULL);
2712 }
2713
2714 static void
2715 _elm_entry_text_set(Evas_Object *obj, const char *item, const char *entry)
2716 {
2717    int len = 0;
2718    ELM_CHECK_WIDTYPE(obj, widtype);
2719    Widget_Data *wd = elm_widget_data_get(obj);
2720    if (!wd) return;
2721    if (!entry) entry = "";
2722    if (item && strcmp(item, "default"))
2723      {
2724         edje_object_part_text_set(wd->ent, item, entry);
2725         return;
2726      }
2727    if (wd->text) eina_stringshare_del(wd->text);
2728    wd->text = NULL;
2729    if (wd->password_text) eina_stringshare_del(wd->password_text);
2730    wd->password_text = NULL;
2731    wd->changed = EINA_TRUE;
2732
2733    /* Clear currently pending job if there is one */
2734    if (wd->append_text_idler)
2735      {
2736         ecore_idler_del(wd->append_text_idler);
2737         free(wd->append_text_left);
2738         wd->append_text_left = NULL;
2739         wd->append_text_idler = NULL;
2740      }
2741
2742    len = strlen(entry);
2743    /* Split to ~_CHUNK_SIZE chunks */
2744    if (len > _CHUNK_SIZE)
2745      {
2746         wd->append_text_left = (char *) malloc(len + 1);
2747      }
2748
2749    /* If we decided to use the idler */
2750    if (wd->append_text_left)
2751      {
2752         /* Need to clear the entry first */
2753         edje_object_part_text_set(wd->ent, "elm.text", "");
2754         memcpy(wd->append_text_left, entry, len + 1);
2755         wd->append_text_position = 0;
2756         wd->append_text_len = len;
2757         wd->append_text_idler = ecore_idler_add(_text_append_idler, obj);
2758      }
2759    else
2760      {
2761         edje_object_part_text_set(wd->ent, "elm.text", entry);
2762      }
2763 }
2764
2765 static const char *
2766 _elm_entry_text_get(const Evas_Object *obj, const char *item)
2767 {
2768    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2769    Widget_Data *wd = elm_widget_data_get(obj);
2770    if (item && strcmp(item, "default")) return NULL;
2771    const char *text;
2772    if (!wd) return NULL;
2773    if (wd->password)
2774      {
2775         if(wd->password_text) return wd->password_text;
2776      }
2777    else if (wd->text) 
2778      {
2779         return wd->text;
2780      }
2781    text = edje_object_part_text_get(wd->ent, "elm.text");
2782    if (!text)
2783      {
2784         ERR("text=NULL for edje %p, part 'elm.text'", wd->ent);
2785         return NULL;
2786      }
2787    eina_stringshare_replace(&wd->text, text);
2788    if (wd->password)
2789      {
2790         const char *pw_text;
2791         pw_text = elm_entry_markup_to_utf8(wd->text);
2792         if (pw_text)
2793           {
2794              eina_stringshare_replace(&wd->password_text, pw_text);
2795              free(pw_text);
2796              return wd->password_text;
2797           }
2798      }
2799    return wd->text;
2800 }
2801
2802 EAPI Evas_Object *
2803 elm_entry_add(Evas_Object *parent)
2804 {
2805    Evas_Object *obj, *top;
2806    Evas *e;
2807    Widget_Data *wd;
2808
2809    ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
2810
2811    ELM_SET_WIDTYPE(widtype, "entry");
2812    elm_widget_type_set(obj, "entry");
2813    elm_widget_sub_object_add(parent, obj);
2814    elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
2815    elm_widget_data_set(obj, wd);
2816    elm_widget_del_hook_set(obj, _del_hook);
2817    elm_widget_del_pre_hook_set(obj, _del_pre_hook);
2818    elm_widget_theme_hook_set(obj, _theme_hook);
2819    elm_widget_disable_hook_set(obj, _disable_hook);
2820    elm_widget_signal_emit_hook_set(obj, _signal_emit_hook);
2821    elm_widget_focus_region_hook_set(obj, _focus_region_hook);
2822    elm_widget_on_focus_region_hook_set(obj, _on_focus_region_hook);
2823    elm_widget_signal_callback_add_hook_set(obj, _signal_callback_add_hook);
2824    elm_widget_signal_callback_del_hook_set(obj, _signal_callback_del_hook);
2825    elm_object_cursor_set(obj, ELM_CURSOR_XTERM);
2826    elm_widget_can_focus_set(obj, EINA_TRUE);
2827    elm_widget_highlight_ignore_set(obj, EINA_TRUE);
2828    elm_widget_text_set_hook_set(obj, _elm_entry_text_set);
2829    elm_widget_text_get_hook_set(obj, _elm_entry_text_get);
2830    elm_widget_content_set_hook_set(obj, _content_set_hook);
2831    elm_widget_content_unset_hook_set(obj, _content_unset_hook);
2832    elm_widget_content_get_hook_set(obj, _content_get_hook);
2833    elm_widget_translate_hook_set(obj, _translate_hook);
2834
2835    evas_object_smart_callback_add(obj, "sub-object-del", _sub_del, wd);
2836    wd->scroller = elm_smart_scroller_add(e);
2837    elm_widget_sub_object_add(obj, wd->scroller);
2838    elm_smart_scroller_widget_set(wd->scroller, obj);
2839    elm_smart_scroller_object_theme_set(obj, wd->scroller, "scroller", "entry",
2840                                        elm_widget_style_get(obj));
2841    evas_object_size_hint_weight_set(wd->scroller, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
2842    evas_object_size_hint_align_set(wd->scroller, EVAS_HINT_FILL, EVAS_HINT_FILL);
2843    elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
2844    elm_smart_scroller_propagate_events_set(wd->scroller, EINA_TRUE);
2845
2846    wd->linewrap     = ELM_WRAP_WORD;
2847    wd->editable     = EINA_TRUE;
2848    wd->disabled     = EINA_FALSE;
2849    wd->context_menu = EINA_TRUE;
2850    wd->autosave     = EINA_TRUE;
2851    wd->textonly     = EINA_FALSE;
2852    wd->autoperiod   = EINA_TRUE;
2853
2854    wd->ent = edje_object_add(e);
2855    elm_widget_sub_object_add(obj, wd->ent);
2856    edje_object_item_provider_set(wd->ent, _get_item, obj);
2857    edje_object_text_insert_filter_callback_add(wd->ent,"elm.text", _text_filter, obj);
2858    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOVE, _move, obj);
2859    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_DOWN,
2860                                   _mouse_down, obj);
2861    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_UP,
2862                                   _mouse_up, obj);
2863    evas_object_event_callback_add(wd->ent, EVAS_CALLBACK_MOUSE_MOVE,
2864                                   _mouse_move, obj);
2865    evas_object_event_callback_add(obj, EVAS_CALLBACK_RESIZE, _resize, obj);
2866
2867    _elm_theme_object_set(obj, wd->ent, "entry", "base", "default");
2868    edje_object_signal_callback_add(wd->ent, "entry,changed", "elm.text",
2869                                    _signal_entry_changed, obj);
2870    edje_object_signal_callback_add(wd->ent, "preedit,changed", "elm.text",
2871                                    _signal_preedit_changed, obj);
2872    edje_object_signal_callback_add(wd->ent, "handler,move,start", "elm.text",
2873                                    _signal_handler_move_start, obj);
2874    edje_object_signal_callback_add(wd->ent, "handler,move,end", "elm.text",
2875                                    _signal_handler_move_end, obj);
2876    edje_object_signal_callback_add(wd->ent, "handler,moving", "elm.text",
2877                                    _signal_handler_moving, obj);
2878    edje_object_signal_callback_add(wd->ent, "selection,start", "elm.text",
2879                                    _signal_selection_start, obj);
2880    edje_object_signal_callback_add(wd->ent, "selection,end", "elm.text",
2881                                    _signal_selection_end, obj);
2882    edje_object_signal_callback_add(wd->ent, "long,pressed", "elm.text",
2883                                    _signal_long_pressed, obj);
2884    edje_object_signal_callback_add(wd->ent, "magnifier,changed", "elm.text",
2885                                    _signal_magnifier_changed, obj);
2886    edje_object_signal_callback_add(wd->ent, "selection,changed", "elm.text",
2887                                    _signal_selection_changed, obj);
2888    edje_object_signal_callback_add(wd->ent, "selection,cleared", "elm.text",
2889                                    _signal_selection_cleared, obj);
2890    edje_object_signal_callback_add(wd->ent, "entry,paste,request", "elm.text",
2891                                    _signal_entry_paste_request, obj);
2892    edje_object_signal_callback_add(wd->ent, "entry,copy,notify", "elm.text",
2893                                    _signal_entry_copy_notify, obj);
2894    edje_object_signal_callback_add(wd->ent, "entry,cut,notify", "elm.text",
2895                                    _signal_entry_cut_notify, obj);
2896    edje_object_signal_callback_add(wd->ent, "cursor,changed", "elm.text",
2897                                    _signal_cursor_changed, obj);
2898    edje_object_signal_callback_add(wd->ent, "anchor,mouse,down,*", "elm.text",
2899                                    _signal_anchor_down, obj);
2900    edje_object_signal_callback_add(wd->ent, "anchor,mouse,up,*", "elm.text",
2901                                    _signal_anchor_up, obj);
2902    edje_object_signal_callback_add(wd->ent, "anchor,mouse,clicked,*", "elm.text",
2903                                    _signal_anchor_clicked, obj);
2904    edje_object_signal_callback_add(wd->ent, "anchor,mouse,move,*", "elm.text",
2905                                    _signal_anchor_move, obj);
2906    edje_object_signal_callback_add(wd->ent, "anchor,mouse,in,*", "elm.text",
2907                                    _signal_anchor_in, obj);
2908    edje_object_signal_callback_add(wd->ent, "anchor,mouse,out,*", "elm.text",
2909                                    _signal_anchor_out, obj);
2910    edje_object_signal_callback_add(wd->ent, "entry,key,enter", "elm.text",
2911                                    _signal_key_enter, obj);
2912    edje_object_signal_callback_add(wd->ent, "mouse,down,1", "elm.text",
2913                                    _signal_mouse_down, obj);
2914    edje_object_signal_callback_add(wd->ent, "mouse,clicked,1", "elm.text",
2915                                    _signal_mouse_clicked, obj);
2916    edje_object_signal_callback_add(wd->ent, "mouse,down,1,double", "elm.text",
2917                                    _signal_mouse_double, obj);
2918    edje_object_part_text_set(wd->ent, "elm.text", "");
2919    if (_elm_config->desktop_entry)
2920      edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_TRUE);
2921    elm_widget_resize_object_set(obj, wd->ent);
2922    _sizing_eval(obj);
2923
2924    elm_entry_input_panel_layout_set(obj, ELM_INPUT_PANEL_LAYOUT_NORMAL);
2925
2926    wd->input_panel_enable = edje_object_part_text_input_panel_enabled_get(wd->ent, "elm.text");
2927    wd->autocapital_type = edje_object_part_text_autocapital_type_get(wd->ent, "elm.text");
2928
2929 #ifdef HAVE_ELEMENTARY_X
2930    top = elm_widget_top_get(obj);
2931    if ((top) && (elm_win_xwindow_get(top)))
2932      {
2933         wd->sel_notify_handler =
2934            ecore_event_handler_add(ECORE_X_EVENT_SELECTION_NOTIFY,
2935                                    _event_selection_notify, obj);
2936         wd->sel_clear_handler =
2937            ecore_event_handler_add(ECORE_X_EVENT_SELECTION_CLEAR,
2938                                    _event_selection_clear, obj);
2939      }
2940
2941    elm_drop_target_add(obj, ELM_SEL_FORMAT_MARKUP | ELM_SEL_FORMAT_IMAGE,
2942                        _drag_drop_cb, NULL);
2943 #endif
2944
2945    entries = eina_list_prepend(entries, obj);
2946
2947    // module - find module for entry
2948    wd->api = _module(obj);
2949    // if found - hook in
2950    if ((wd->api) && (wd->api->obj_hook)) wd->api->obj_hook(obj);
2951
2952    _mirrored_set(obj, elm_widget_mirrored_get(obj));
2953    // TODO: convert Elementary to subclassing of Evas_Smart_Class
2954    // TODO: and save some bytes, making descriptions per-class and not instance!
2955    evas_object_smart_callbacks_descriptions_set(obj, _signals);
2956    return obj;
2957 }
2958
2959 EAPI void elm_entry_extension_module_data_get(Evas_Object *obj,Elm_Entry_Extension_data *ext_mod)
2960 {
2961    ELM_CHECK_WIDTYPE(obj, widtype);
2962    Widget_Data *wd = elm_widget_data_get(obj);
2963    if (!wd) return;
2964    ext_mod->cancel = _cancel;
2965    ext_mod->copy = _copy;
2966    ext_mod->cut = _cut;
2967    ext_mod->paste = _paste;
2968    ext_mod->select = _select;
2969    ext_mod->selectall = _selectall;
2970    ext_mod->ent = wd->ent;
2971    ext_mod->viewport_obj = _viewport_obj_get(obj);
2972    ext_mod->items = wd->items;
2973    ext_mod->editable = wd->editable;
2974    ext_mod->have_selection = wd->have_selection;
2975    ext_mod->password = wd->password;
2976    ext_mod->selmode = wd->selmode;
2977    ext_mod->cnpinit = _cnpinit;
2978    ext_mod->context_menu = wd->context_menu;
2979    ext_mod->textonly = wd->textonly;
2980 }
2981
2982 EAPI void
2983 elm_entry_single_line_set(Evas_Object *obj, Eina_Bool single_line)
2984 {
2985    ELM_CHECK_WIDTYPE(obj, widtype);
2986    Widget_Data *wd = elm_widget_data_get(obj);
2987    if (!wd) return;
2988    if (wd->single_line == single_line) return;
2989    wd->single_line = single_line;
2990    wd->linewrap = ELM_WRAP_NONE;
2991    elm_entry_cnp_textonly_set(obj, EINA_TRUE);
2992    _theme_hook(obj);
2993    if (wd->scroller)
2994      {
2995         if (wd->single_line)
2996           {
2997              elm_smart_scroller_policy_set(wd->scroller,
2998                                            ELM_SMART_SCROLLER_POLICY_OFF,
2999                                            ELM_SMART_SCROLLER_POLICY_OFF);
3000              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3001           }
3002         else
3003           {
3004              const Elm_Scroller_Policy map[3] =
3005                {
3006                   ELM_SMART_SCROLLER_POLICY_AUTO,
3007                   ELM_SMART_SCROLLER_POLICY_ON,
3008                   ELM_SMART_SCROLLER_POLICY_OFF
3009                };
3010              elm_smart_scroller_policy_set(wd->scroller,
3011                                            map[wd->policy_h],
3012                                            map[wd->policy_v]);
3013              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3014           }
3015         _sizing_eval(obj);
3016      }
3017 }
3018
3019 EAPI Eina_Bool
3020 elm_entry_single_line_get(const Evas_Object *obj)
3021 {
3022    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3023    Widget_Data *wd = elm_widget_data_get(obj);
3024    if (!wd) return EINA_FALSE;
3025    return wd->single_line;
3026 }
3027
3028 EAPI void
3029 elm_entry_password_set(Evas_Object *obj, Eina_Bool password)
3030 {
3031    ELM_CHECK_WIDTYPE(obj, widtype);
3032    Widget_Data *wd = elm_widget_data_get(obj);
3033    if (!wd) return;
3034    if (wd->password == password) return;
3035    wd->password = password;
3036    wd->single_line = EINA_TRUE;
3037    wd->linewrap = ELM_WRAP_NONE;
3038    _theme_hook(obj);
3039 }
3040
3041 EAPI Eina_Bool
3042 elm_entry_password_get(const Evas_Object *obj)
3043 {
3044    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3045    Widget_Data *wd = elm_widget_data_get(obj);
3046    if (!wd) return EINA_FALSE;
3047    return wd->password;
3048 }
3049
3050 EAPI void
3051 elm_entry_entry_set(Evas_Object *obj, const char *entry)
3052 {
3053    _elm_entry_text_set(obj, NULL, entry);
3054 }
3055
3056 EAPI void
3057 elm_entry_entry_append(Evas_Object *obj, const char *entry)
3058 {
3059    int len = 0;
3060    ELM_CHECK_WIDTYPE(obj, widtype);
3061    Widget_Data *wd = elm_widget_data_get(obj);
3062    if (!wd) return;
3063    if (!entry) entry = "";
3064    wd->changed = EINA_TRUE;
3065
3066    len = strlen(entry);
3067    if (wd->append_text_left)
3068      {
3069         char *tmpbuf;
3070         tmpbuf = realloc(wd->append_text_left, wd->append_text_len + len + 1);
3071         if (!tmpbuf)
3072           {
3073              /* Do something */
3074              return;
3075           }
3076         wd->append_text_left = tmpbuf;
3077         memcpy(wd->append_text_left + wd->append_text_len, entry, len + 1);
3078         wd->append_text_len += len;
3079      }
3080    else
3081      {
3082         /* FIXME: Add chunked appending here (like in entry_set) */
3083         edje_object_part_text_append(wd->ent, "elm.text", entry);
3084      }
3085 }
3086
3087 EAPI const char *
3088 elm_entry_entry_get(const Evas_Object *obj)
3089 {
3090    return _elm_entry_text_get(obj, NULL);
3091 }
3092
3093 EAPI Eina_Bool
3094 elm_entry_is_empty(const Evas_Object *obj)
3095 {
3096    /* FIXME: until there's support for that in textblock, we just check
3097     * to see if the there is text or not. */
3098    ELM_CHECK_WIDTYPE(obj, widtype) EINA_TRUE;
3099    Widget_Data *wd = elm_widget_data_get(obj);
3100    const Evas_Object *tb;
3101    //Evas_Textblock_Cursor *cur;
3102    Eina_Bool ret;
3103    if (!wd) return EINA_TRUE;
3104
3105 #if 0
3106    /* It's a hack until we get the support suggested above.
3107     * We just create a cursor, point it to the begining, and then
3108     * try to advance it, if it can advance, the tb is not empty,
3109     * otherwise it is. */
3110    tb = edje_object_part_object_get(wd->ent, "elm.text");
3111    cur = evas_object_textblock_cursor_new((Evas_Object *) tb); /* This is
3112                                                                   actually, ok for the time being, thsese hackish stuff will be removed
3113                                                                   once evas 1.0 is out*/
3114    evas_textblock_cursor_pos_set(cur, 0);
3115    ret = evas_textblock_cursor_char_next(cur);
3116    evas_textblock_cursor_free(cur);
3117
3118    return !ret;
3119 #endif
3120
3121    char *str = elm_entry_markup_to_utf8(elm_entry_entry_get(obj));
3122    if (!str) return EINA_TRUE;
3123
3124    ret = (strlen(str) == 0);
3125
3126    free(str);
3127    return ret;
3128 }
3129
3130 EAPI const char *
3131 elm_entry_selection_get(const Evas_Object *obj)
3132 {
3133    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3134    Widget_Data *wd = elm_widget_data_get(obj);
3135    if (!wd) return NULL;
3136    return edje_object_part_text_selection_get(wd->ent, "elm.text");
3137 }
3138
3139 EAPI void
3140 elm_entry_entry_insert(Evas_Object *obj, const char *entry)
3141 {
3142    ELM_CHECK_WIDTYPE(obj, widtype);
3143    Widget_Data *wd = elm_widget_data_get(obj);
3144    if (!wd) return;
3145    edje_object_part_text_insert(wd->ent, "elm.text", entry);
3146    // start for cbhm
3147 #ifdef HAVE_ELEMENTARY_X
3148    if (cnpwidgetdata == obj)
3149       ecore_x_selection_secondary_set(elm_win_xwindow_get(obj), "",1);
3150 #endif
3151    // end for cbhm
3152    wd->changed = EINA_TRUE;
3153    _sizing_eval(obj);
3154 }
3155
3156 EAPI void
3157 elm_entry_line_wrap_set(Evas_Object *obj, Elm_Wrap_Type wrap)
3158 {
3159    ELM_CHECK_WIDTYPE(obj, widtype);
3160    Widget_Data *wd = elm_widget_data_get(obj);
3161    if (!wd) return;
3162    if (wd->linewrap == wrap) return;
3163    wd->lastw = -1;
3164    wd->linewrap = wrap;
3165    _theme_hook(obj);
3166 }
3167
3168 EAPI Elm_Wrap_Type
3169 elm_entry_line_wrap_get(const Evas_Object *obj)
3170 {
3171    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3172    Widget_Data *wd = elm_widget_data_get(obj);
3173    if (!wd) return EINA_FALSE;
3174    return wd->linewrap;
3175 }
3176
3177 EAPI void
3178 elm_entry_editable_set(Evas_Object *obj, Eina_Bool editable)
3179 {
3180    ELM_CHECK_WIDTYPE(obj, widtype);
3181    Widget_Data *wd = elm_widget_data_get(obj);
3182    if (!wd) return;
3183    if (wd->editable == editable) return;
3184    wd->editable = editable;
3185    _theme_hook(obj);
3186
3187 #ifdef HAVE_ELEMENTARY_X
3188    if (editable)
3189      elm_drop_target_add(obj, ELM_SEL_FORMAT_MARKUP, _drag_drop_cb, NULL);
3190    else
3191      elm_drop_target_del(obj);
3192 #endif
3193 }
3194
3195 EAPI Eina_Bool
3196 elm_entry_editable_get(const Evas_Object *obj)
3197 {
3198    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3199    Widget_Data *wd = elm_widget_data_get(obj);
3200    if (!wd) return EINA_FALSE;
3201    return wd->editable;
3202 }
3203
3204 EAPI void
3205 elm_entry_select_none(Evas_Object *obj)
3206 {
3207    ELM_CHECK_WIDTYPE(obj, widtype);
3208    Widget_Data *wd = elm_widget_data_get(obj);
3209    if (!wd) return;
3210    if (wd->selmode)
3211      {
3212         wd->selmode = EINA_FALSE;
3213         if (!_elm_config->desktop_entry)
3214           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
3215         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
3216      }
3217    wd->have_selection = EINA_FALSE;
3218    edje_object_part_text_select_none(wd->ent, "elm.text");
3219 }
3220
3221 EAPI void
3222 elm_entry_select_all(Evas_Object *obj)
3223 {
3224    ELM_CHECK_WIDTYPE(obj, widtype);
3225    Widget_Data *wd = elm_widget_data_get(obj);
3226    if (!wd) return;
3227    if (wd->selmode)
3228      {
3229         wd->selmode = EINA_FALSE;
3230         if (!_elm_config->desktop_entry)
3231           edje_object_part_text_select_allow_set(wd->ent, "elm.text", EINA_FALSE);
3232         edje_object_signal_emit(wd->ent, "elm,state,select,off", "elm");
3233      }
3234    wd->have_selection = EINA_TRUE;
3235    edje_object_part_text_select_all(wd->ent, "elm.text");
3236 }
3237
3238 EAPI Eina_Bool
3239 elm_entry_cursor_geometry_get(const Evas_Object *obj, Evas_Coord *x, Evas_Coord *y, Evas_Coord *w, Evas_Coord *h)
3240 {
3241    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3242    Widget_Data *wd = elm_widget_data_get(obj);
3243    if (!wd) return EINA_FALSE;
3244    edje_object_part_text_cursor_geometry_get(wd->ent, "elm.text", x, y, w, h);
3245    return EINA_TRUE;
3246 }
3247
3248 EAPI Eina_Bool
3249 elm_entry_cursor_next(Evas_Object *obj)
3250 {
3251    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3252    Widget_Data *wd = elm_widget_data_get(obj);
3253    if (!wd) return EINA_FALSE;
3254    return edje_object_part_text_cursor_next(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3255 }
3256
3257 EAPI Eina_Bool
3258 elm_entry_cursor_prev(Evas_Object *obj)
3259 {
3260    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3261    Widget_Data *wd = elm_widget_data_get(obj);
3262    if (!wd) return EINA_FALSE;
3263    return edje_object_part_text_cursor_prev(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3264 }
3265
3266 EAPI Eina_Bool
3267 elm_entry_cursor_up(Evas_Object *obj)
3268 {
3269    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3270    Widget_Data *wd = elm_widget_data_get(obj);
3271    if (!wd) return EINA_FALSE;
3272    return edje_object_part_text_cursor_up(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3273 }
3274
3275 EAPI Eina_Bool
3276 elm_entry_cursor_down(Evas_Object *obj)
3277 {
3278    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3279    Widget_Data *wd = elm_widget_data_get(obj);
3280    if (!wd) return EINA_FALSE;
3281    return edje_object_part_text_cursor_down(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3282 }
3283
3284 EAPI void
3285 elm_entry_cursor_begin_set(Evas_Object *obj)
3286 {
3287    ELM_CHECK_WIDTYPE(obj, widtype);
3288    Widget_Data *wd = elm_widget_data_get(obj);
3289    if (!wd) return;
3290    edje_object_part_text_cursor_begin_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3291 }
3292
3293 EAPI void
3294 elm_entry_cursor_end_set(Evas_Object *obj)
3295 {
3296    ELM_CHECK_WIDTYPE(obj, widtype);
3297    Widget_Data *wd = elm_widget_data_get(obj);
3298    if (!wd) return;
3299    int x, y, w, h;
3300    edje_object_part_text_cursor_end_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3301    if (wd->scroll)
3302      {
3303         elm_widget_show_region_get(wd->ent, &x, &y, &w, &h);
3304         elm_smart_scroller_child_region_show(wd->scroller, x, y, w, h);
3305      }
3306 }
3307
3308 EAPI void
3309 elm_entry_cursor_line_begin_set(Evas_Object *obj)
3310 {
3311    ELM_CHECK_WIDTYPE(obj, widtype);
3312    Widget_Data *wd = elm_widget_data_get(obj);
3313    if (!wd) return;
3314    edje_object_part_text_cursor_line_begin_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3315 }
3316
3317 EAPI void
3318 elm_entry_cursor_line_end_set(Evas_Object *obj)
3319 {
3320    ELM_CHECK_WIDTYPE(obj, widtype);
3321    Widget_Data *wd = elm_widget_data_get(obj);
3322    if (!wd) return;
3323    edje_object_part_text_cursor_line_end_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3324 }
3325
3326 EAPI void
3327 elm_entry_cursor_selection_begin(Evas_Object *obj)
3328 {
3329    ELM_CHECK_WIDTYPE(obj, widtype);
3330    Widget_Data *wd = elm_widget_data_get(obj);
3331    if (!wd) return;
3332    edje_object_part_text_select_begin(wd->ent, "elm.text");
3333 }
3334
3335 EAPI void
3336 elm_entry_cursor_selection_end(Evas_Object *obj)
3337 {
3338    ELM_CHECK_WIDTYPE(obj, widtype);
3339    Widget_Data *wd = elm_widget_data_get(obj);
3340    if (!wd) return;
3341    edje_object_part_text_select_extend(wd->ent, "elm.text");
3342 }
3343
3344 EAPI Eina_Bool
3345 elm_entry_cursor_is_format_get(const Evas_Object *obj)
3346 {
3347    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3348    Widget_Data *wd = elm_widget_data_get(obj);
3349    if (!wd) return EINA_FALSE;
3350    return edje_object_part_text_cursor_is_format_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3351 }
3352
3353 EAPI Eina_Bool
3354 elm_entry_cursor_is_visible_format_get(const Evas_Object *obj)
3355 {
3356    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3357    Widget_Data *wd = elm_widget_data_get(obj);
3358    if (!wd) return EINA_FALSE;
3359    return edje_object_part_text_cursor_is_visible_format_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3360 }
3361
3362 EAPI const char *
3363 elm_entry_cursor_content_get(const Evas_Object *obj)
3364 {
3365    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3366    Widget_Data *wd = elm_widget_data_get(obj);
3367    if (!wd) return NULL;
3368    return edje_object_part_text_cursor_content_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3369 }
3370
3371 EAPI void
3372 elm_entry_cursor_pos_set(Evas_Object *obj, int pos)
3373 {
3374    ELM_CHECK_WIDTYPE(obj, widtype);
3375    Widget_Data *wd = elm_widget_data_get(obj);
3376    if (!wd) return;
3377    edje_object_part_text_cursor_pos_set(wd->ent, "elm.text", EDJE_CURSOR_MAIN, pos);
3378    edje_object_message_signal_process(wd->ent);
3379 }
3380
3381 EAPI int
3382 elm_entry_cursor_pos_get(const Evas_Object *obj)
3383 {
3384    ELM_CHECK_WIDTYPE(obj, widtype) 0;
3385    Widget_Data *wd = elm_widget_data_get(obj);
3386    if (!wd) return 0;
3387    return edje_object_part_text_cursor_pos_get(wd->ent, "elm.text", EDJE_CURSOR_MAIN);
3388 }
3389
3390 EAPI void
3391 elm_entry_selection_cut(Evas_Object *obj)
3392 {
3393    ELM_CHECK_WIDTYPE(obj, widtype);
3394    Widget_Data *wd = elm_widget_data_get(obj);
3395    if (!wd) return;
3396    _cut(obj, NULL, NULL);
3397 }
3398
3399 EAPI void
3400 elm_entry_selection_copy(Evas_Object *obj)
3401 {
3402    ELM_CHECK_WIDTYPE(obj, widtype);
3403    Widget_Data *wd = elm_widget_data_get(obj);
3404    if (!wd) return;
3405    _copy(obj, NULL, NULL);
3406 }
3407
3408 EAPI void
3409 elm_entry_selection_paste(Evas_Object *obj)
3410 {
3411    ELM_CHECK_WIDTYPE(obj, widtype);
3412    Widget_Data *wd = elm_widget_data_get(obj);
3413    if (!wd) return;
3414    _paste(obj, NULL, NULL);
3415 }
3416
3417 EAPI void
3418 elm_entry_context_menu_clear(Evas_Object *obj)
3419 {
3420    ELM_CHECK_WIDTYPE(obj, widtype);
3421    Widget_Data *wd = elm_widget_data_get(obj);
3422    Elm_Entry_Context_Menu_Item *it;
3423    if (!wd) return;
3424    EINA_LIST_FREE(wd->items, it)
3425      {
3426         eina_stringshare_del(it->label);
3427         eina_stringshare_del(it->icon_file);
3428         eina_stringshare_del(it->icon_group);
3429         free(it);
3430      }
3431 }
3432
3433 EAPI void
3434 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)
3435 {
3436    ELM_CHECK_WIDTYPE(obj, widtype);
3437    Widget_Data *wd = elm_widget_data_get(obj);
3438    Elm_Entry_Context_Menu_Item *it;
3439    if (!wd) return;
3440    it = calloc(1, sizeof(Elm_Entry_Context_Menu_Item));
3441    if (!it) return;
3442    wd->items = eina_list_append(wd->items, it);
3443    it->obj = obj;
3444    it->label = eina_stringshare_add(label);
3445    it->icon_file = eina_stringshare_add(icon_file);
3446    it->icon_type = icon_type;
3447    it->func = func;
3448    it->data = (void *)data;
3449 }
3450
3451 EAPI void
3452 elm_entry_context_menu_disabled_set(Evas_Object *obj, Eina_Bool disabled)
3453 {
3454    ELM_CHECK_WIDTYPE(obj, widtype);
3455    Widget_Data *wd = elm_widget_data_get(obj);
3456    if (!wd) return;
3457    if (wd->context_menu == !disabled) return;
3458    wd->context_menu = !disabled;
3459 }
3460
3461 EAPI Eina_Bool
3462 elm_entry_context_menu_disabled_get(const Evas_Object *obj)
3463 {
3464    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3465    Widget_Data *wd = elm_widget_data_get(obj);
3466    if (!wd) return EINA_FALSE;
3467    return !wd->context_menu;
3468 }
3469
3470 EAPI void
3471 elm_entry_item_provider_append(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3472 {
3473    ELM_CHECK_WIDTYPE(obj, widtype);
3474    Widget_Data *wd = elm_widget_data_get(obj);
3475    if (!wd) return;
3476    EINA_SAFETY_ON_NULL_RETURN(func);
3477    Elm_Entry_Item_Provider *ip = calloc(1, sizeof(Elm_Entry_Item_Provider));
3478    if (!ip) return;
3479    ip->func = func;
3480    ip->data = data;
3481    wd->item_providers = eina_list_append(wd->item_providers, ip);
3482 }
3483
3484 EAPI void
3485 elm_entry_item_provider_prepend(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3486 {
3487    ELM_CHECK_WIDTYPE(obj, widtype);
3488    Widget_Data *wd = elm_widget_data_get(obj);
3489    if (!wd) return;
3490    EINA_SAFETY_ON_NULL_RETURN(func);
3491    Elm_Entry_Item_Provider *ip = calloc(1, sizeof(Elm_Entry_Item_Provider));
3492    if (!ip) return;
3493    ip->func = func;
3494    ip->data = data;
3495    wd->item_providers = eina_list_prepend(wd->item_providers, ip);
3496 }
3497
3498 EAPI void
3499 elm_entry_item_provider_remove(Evas_Object *obj, Evas_Object *(*func) (void *data, Evas_Object *entry, const char *item), void *data)
3500 {
3501    ELM_CHECK_WIDTYPE(obj, widtype);
3502    Widget_Data *wd = elm_widget_data_get(obj);
3503    Eina_List *l;
3504    Elm_Entry_Item_Provider *ip;
3505    if (!wd) return;
3506    EINA_SAFETY_ON_NULL_RETURN(func);
3507    EINA_LIST_FOREACH(wd->item_providers, l, ip)
3508      {
3509         if ((ip->func == func) && ((!data) || (ip->data == data)))
3510           {
3511              wd->item_providers = eina_list_remove_list(wd->item_providers, l);
3512              free(ip);
3513              return;
3514           }
3515      }
3516 }
3517
3518 EAPI void
3519 elm_entry_text_filter_append(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3520 {
3521    Widget_Data *wd;
3522    Elm_Entry_Text_Filter *tf;
3523    ELM_CHECK_WIDTYPE(obj, widtype);
3524
3525    wd = elm_widget_data_get(obj);
3526
3527    EINA_SAFETY_ON_NULL_RETURN(func);
3528
3529    tf = _filter_new(func, data);
3530    if (!tf) return;
3531
3532    wd->text_filters = eina_list_append(wd->text_filters, tf);
3533 }
3534
3535 EAPI void
3536 elm_entry_text_filter_prepend(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3537 {
3538    Widget_Data *wd;
3539    Elm_Entry_Text_Filter *tf;
3540    ELM_CHECK_WIDTYPE(obj, widtype);
3541
3542    wd = elm_widget_data_get(obj);
3543
3544    EINA_SAFETY_ON_NULL_RETURN(func);
3545
3546    tf = _filter_new(func, data);
3547    if (!tf) return;
3548
3549    wd->text_filters = eina_list_prepend(wd->text_filters, tf);
3550 }
3551
3552 EAPI void
3553 elm_entry_text_filter_remove(Evas_Object *obj, void (*func) (void *data, Evas_Object *entry, char **text), void *data)
3554 {
3555    Widget_Data *wd;
3556    Eina_List *l;
3557    Elm_Entry_Text_Filter *tf;
3558    ELM_CHECK_WIDTYPE(obj, widtype);
3559
3560    wd = elm_widget_data_get(obj);
3561
3562    EINA_SAFETY_ON_NULL_RETURN(func);
3563
3564    EINA_LIST_FOREACH(wd->text_filters, l, tf)
3565      {
3566         if ((tf->func == func) && ((!data) || (tf->data == data)))
3567           {
3568              wd->text_filters = eina_list_remove_list(wd->text_filters, l);
3569              _filter_free(tf);
3570              return;
3571           }
3572      }
3573 }
3574
3575 EAPI char *
3576 elm_entry_markup_to_utf8(const char *s)
3577 {
3578    char *ss = _elm_util_mkup_to_text(s);
3579    if (!ss) ss = strdup("");
3580    return ss;
3581 }
3582
3583 EAPI char *
3584 elm_entry_utf8_to_markup(const char *s)
3585 {
3586    char *ss = _elm_util_text_to_mkup(s);
3587    if (!ss) ss = strdup("");
3588    return ss;
3589 }
3590
3591 EAPI void
3592 elm_entry_filter_limit_size(void *data, Evas_Object *entry, char **text)
3593 {
3594    Elm_Entry_Filter_Limit_Size *lim = data;
3595    char *current;
3596    int len, newlen;
3597    const char *(*text_get)(const Evas_Object *);
3598    const char *widget_type;
3599
3600    EINA_SAFETY_ON_NULL_RETURN(data);
3601    EINA_SAFETY_ON_NULL_RETURN(entry);
3602    EINA_SAFETY_ON_NULL_RETURN(text);
3603
3604    /* hack. I don't want to copy the entire function to work with
3605     * scrolled_entry */
3606    widget_type = elm_widget_type_get(entry);
3607    if (!strcmp(widget_type, "entry"))
3608      text_get = elm_entry_entry_get;
3609    else /* huh? */
3610      return;
3611
3612    current = elm_entry_markup_to_utf8(text_get(entry));
3613
3614    if (lim->max_char_count > 0)
3615      {
3616         len = evas_string_char_len_get(current);
3617         if (len >= lim->max_char_count)
3618           {
3619              evas_object_smart_callback_call(entry, "maxlength,reached", NULL);
3620              free(*text);
3621              free(current);
3622              *text = NULL;
3623              return;
3624           }
3625         newlen = evas_string_char_len_get(elm_entry_markup_to_utf8(*text));
3626         if ((len + newlen) > lim->max_char_count)
3627           _add_chars_till_limit(entry, text, (lim->max_char_count - len), LENGTH_UNIT_CHAR);
3628      }
3629    else if (lim->max_byte_count > 0)
3630      {
3631         len = strlen(current);
3632         if (len >= lim->max_byte_count)
3633           {
3634              evas_object_smart_callback_call(entry, "maxlength,reached", NULL);
3635              free(*text);
3636              free(current);
3637              *text = NULL;
3638              return;
3639           }
3640         newlen = strlen(elm_entry_markup_to_utf8(*text));
3641         if ((len + newlen) > lim->max_byte_count)
3642           _add_chars_till_limit(entry, text, (lim->max_byte_count - len), LENGTH_UNIT_BYTE);
3643      }
3644    free(current);
3645 }
3646
3647 EAPI void
3648 elm_entry_filter_accept_set(void *data, Evas_Object *entry __UNUSED__, char **text)
3649 {
3650    Elm_Entry_Filter_Accept_Set *as = data;
3651    const char *set;
3652    char *insert;
3653    Eina_Bool goes_in;
3654    int read_idx, last_read_idx = 0, read_char;
3655
3656    EINA_SAFETY_ON_NULL_RETURN(data);
3657    EINA_SAFETY_ON_NULL_RETURN(text);
3658
3659    if ((!as->accepted) && (!as->rejected))
3660      return;
3661
3662    if (as->accepted)
3663      {
3664         set = as->accepted;
3665         goes_in = EINA_TRUE;
3666      }
3667    else
3668      {
3669         set = as->rejected;
3670         goes_in = EINA_FALSE;
3671      }
3672
3673    insert = *text;
3674    read_idx = evas_string_char_next_get(*text, 0, &read_char);
3675    while (read_char)
3676      {
3677         int cmp_idx, cmp_char;
3678         Eina_Bool in_set = EINA_FALSE;
3679
3680         cmp_idx = evas_string_char_next_get(set, 0, &cmp_char);
3681         while (cmp_char)
3682           {
3683              if (read_char == cmp_char)
3684                {
3685                   in_set = EINA_TRUE;
3686                   break;
3687                }
3688              cmp_idx = evas_string_char_next_get(set, cmp_idx, &cmp_char);
3689           }
3690         if (in_set == goes_in)
3691           {
3692              int size = read_idx - last_read_idx;
3693              const char *src = (*text) + last_read_idx;
3694              if (src != insert)
3695                memcpy(insert, *text + last_read_idx, size);
3696              insert += size;
3697           }
3698         last_read_idx = read_idx;
3699         read_idx = evas_string_char_next_get(*text, read_idx, &read_char);
3700      }
3701    *insert = 0;
3702 }
3703
3704 EAPI void
3705 elm_entry_file_set(Evas_Object *obj, const char *file, Elm_Text_Format format)
3706 {
3707    ELM_CHECK_WIDTYPE(obj, widtype);
3708    Widget_Data *wd = elm_widget_data_get(obj);
3709    if (!wd) return;
3710    if (wd->delay_write)
3711      {
3712         ecore_timer_del(wd->delay_write);
3713         wd->delay_write = NULL;
3714      }
3715    if (wd->autosave) _save(obj);
3716    eina_stringshare_replace(&wd->file, file);
3717    wd->format = format;
3718    _load(obj);
3719 }
3720
3721 EAPI void
3722 elm_entry_file_get(const Evas_Object *obj, const char **file, Elm_Text_Format *format)
3723 {
3724    ELM_CHECK_WIDTYPE(obj, widtype);
3725    Widget_Data *wd = elm_widget_data_get(obj);
3726    if (!wd) return;
3727    if (file) *file = wd->file;
3728    if (format) *format = wd->format;
3729 }
3730
3731 EAPI void
3732 elm_entry_file_save(Evas_Object *obj)
3733 {
3734    ELM_CHECK_WIDTYPE(obj, widtype);
3735    Widget_Data *wd = elm_widget_data_get(obj);
3736    if (!wd) return;
3737    if (wd->delay_write)
3738      {
3739         ecore_timer_del(wd->delay_write);
3740         wd->delay_write = NULL;
3741      }
3742    _save(obj);
3743    wd->delay_write = ecore_timer_add(2.0, _delay_write, obj);
3744 }
3745
3746 EAPI void
3747 elm_entry_autosave_set(Evas_Object *obj, Eina_Bool autosave)
3748 {
3749    ELM_CHECK_WIDTYPE(obj, widtype);
3750    Widget_Data *wd = elm_widget_data_get(obj);
3751    if (!wd) return;
3752    wd->autosave = !!autosave;
3753 }
3754
3755 EAPI Eina_Bool
3756 elm_entry_autosave_get(const Evas_Object *obj)
3757 {
3758    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3759    Widget_Data *wd = elm_widget_data_get(obj);
3760    if (!wd) return EINA_FALSE;
3761    return wd->autosave;
3762 }
3763
3764 EAPI void
3765 elm_entry_cnp_textonly_set(Evas_Object *obj, Eina_Bool textonly)
3766 {
3767    Elm_Sel_Format format = ELM_SEL_FORMAT_MARKUP;
3768    ELM_CHECK_WIDTYPE(obj, widtype);
3769    Widget_Data *wd = elm_widget_data_get(obj);
3770    if (!wd) return;
3771    textonly = !!textonly;
3772    if (wd->textonly == textonly) return;
3773    wd->textonly = !!textonly;
3774    if (!textonly) format |= ELM_SEL_FORMAT_IMAGE;
3775 #ifdef HAVE_ELEMENTARY_X
3776    elm_drop_target_add(obj, format, _drag_drop_cb, NULL);
3777 #endif
3778 }
3779
3780 EAPI Eina_Bool
3781 elm_entry_cnp_textonly_get(const Evas_Object *obj)
3782 {
3783    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3784    Widget_Data *wd = elm_widget_data_get(obj);
3785    if (!wd) return EINA_FALSE;
3786    return wd->textonly;
3787 }
3788
3789 EAPI void
3790 elm_entry_scrollable_set(Evas_Object *obj, Eina_Bool scroll)
3791 {
3792    ELM_CHECK_WIDTYPE(obj, widtype);
3793    Widget_Data *wd = elm_widget_data_get(obj);
3794    if (!wd) return;
3795    scroll = !!scroll;
3796    if (wd->scroll == scroll) return;
3797    wd->scroll = scroll;
3798    if (wd->scroll)
3799      {
3800         elm_widget_sub_object_del(obj, wd->scroller);
3801         elm_widget_resize_object_set(obj, wd->scroller);
3802         elm_widget_sub_object_add(obj, wd->ent);
3803         elm_smart_scroller_child_set(wd->scroller, wd->ent);
3804         evas_object_show(wd->scroller);
3805         elm_widget_on_show_region_hook_set(obj, _show_region_hook, obj);
3806         if (wd->single_line)
3807           {
3808              elm_smart_scroller_policy_set(wd->scroller,
3809                                            ELM_SMART_SCROLLER_POLICY_OFF,
3810                                            ELM_SMART_SCROLLER_POLICY_OFF);
3811              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3812           }
3813         else
3814           {
3815              const Elm_Scroller_Policy map[3] =
3816                {
3817                   ELM_SMART_SCROLLER_POLICY_AUTO,
3818                   ELM_SMART_SCROLLER_POLICY_ON,
3819                   ELM_SMART_SCROLLER_POLICY_OFF
3820                };
3821              elm_smart_scroller_policy_set(wd->scroller,
3822                                            map[wd->policy_h],
3823                                            map[wd->policy_v]);
3824              elm_smart_scroller_bounce_allow_set(wd->scroller, EINA_FALSE, EINA_FALSE);
3825           }
3826      }
3827    else
3828      {
3829         elm_smart_scroller_child_set(wd->scroller, NULL);
3830         elm_widget_sub_object_del(obj, wd->ent);
3831         elm_widget_resize_object_set(obj, wd->ent);
3832         evas_object_smart_member_add(wd->scroller, obj);
3833         elm_widget_sub_object_add(obj, wd->scroller);
3834         evas_object_hide(wd->scroller);
3835         elm_widget_on_show_region_hook_set(obj, NULL, NULL);
3836      }
3837    wd->lastw = -1;
3838    _theme_hook(obj);
3839 }
3840
3841 EAPI Eina_Bool
3842 elm_entry_scrollable_get(const Evas_Object *obj)
3843 {
3844    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
3845    Widget_Data *wd = elm_widget_data_get(obj);
3846    if (!wd) return EINA_FALSE;
3847    return wd->scroll;
3848 }
3849
3850 EAPI void
3851 elm_entry_icon_set(Evas_Object *obj, Evas_Object *icon)
3852 {
3853    ELM_CHECK_WIDTYPE(obj, widtype);
3854    Widget_Data *wd = elm_widget_data_get(obj);
3855    Evas_Object *edje;
3856    if (!wd) return;
3857    EINA_SAFETY_ON_NULL_RETURN(icon);
3858    if (wd->icon == icon) return;
3859    if (wd->icon) evas_object_del(wd->icon);
3860    wd->icon = icon;
3861    edje = elm_smart_scroller_edje_object_get(wd->scroller);
3862    if (!edje) return;
3863    edje_object_part_swallow(edje, "elm.swallow.icon", wd->icon);
3864    edje_object_signal_emit(edje, "elm,action,show,icon", "elm");
3865    _sizing_eval(obj);
3866 }
3867
3868 EAPI Evas_Object *
3869 elm_entry_icon_get(const Evas_Object *obj)
3870 {
3871    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3872    Widget_Data *wd = elm_widget_data_get(obj);
3873    if (!wd) return NULL;
3874    return wd->icon;
3875 }
3876
3877 EAPI Evas_Object *
3878 elm_entry_icon_unset(Evas_Object *obj)
3879 {
3880    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3881    Widget_Data *wd = elm_widget_data_get(obj);
3882    Evas_Object *ret = NULL;
3883    if (!wd) return NULL;
3884    if (wd->icon)
3885      {
3886         Evas_Object *edje = elm_smart_scroller_edje_object_get(wd->scroller);
3887         if (!edje) return NULL;
3888         ret = wd->icon;
3889         edje_object_part_unswallow(edje, wd->icon);
3890         edje_object_signal_emit(edje, "elm,action,hide,icon", "elm");
3891         wd->icon = NULL;
3892         _sizing_eval(obj);
3893      }
3894    return ret;
3895 }
3896
3897 EAPI void
3898 elm_entry_icon_visible_set(Evas_Object *obj, Eina_Bool setting)
3899 {
3900    ELM_CHECK_WIDTYPE(obj, widtype);
3901    Widget_Data *wd = elm_widget_data_get(obj);
3902    if ((!wd) || (!wd->icon)) return;
3903    if (setting)
3904       evas_object_hide(wd->icon);
3905    else
3906       evas_object_show(wd->icon);
3907    _sizing_eval(obj);
3908 }
3909
3910 EAPI void
3911 elm_entry_end_set(Evas_Object *obj, Evas_Object *end)
3912 {
3913    ELM_CHECK_WIDTYPE(obj, widtype);
3914    Widget_Data *wd = elm_widget_data_get(obj);
3915    Evas_Object *edje;
3916    if (!wd) return;
3917    EINA_SAFETY_ON_NULL_RETURN(end);
3918    if (wd->end == end) return;
3919    if (wd->end) evas_object_del(wd->end);
3920    wd->end = end;
3921    edje = elm_smart_scroller_edje_object_get(wd->scroller);
3922    if (!edje) return;
3923    edje_object_part_swallow(edje, "elm.swallow.end", wd->end);
3924    edje_object_signal_emit(edje, "elm,action,show,end", "elm");
3925    _sizing_eval(obj);
3926 }
3927
3928 EAPI Evas_Object *
3929 elm_entry_end_get(const Evas_Object *obj)
3930 {
3931    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3932    Widget_Data *wd = elm_widget_data_get(obj);
3933    if (!wd) return NULL;
3934    return wd->end;
3935 }
3936
3937 EAPI Evas_Object *
3938 elm_entry_end_unset(Evas_Object *obj)
3939 {
3940    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
3941    Widget_Data *wd = elm_widget_data_get(obj);
3942    Evas_Object *ret = NULL;
3943    if (!wd) return NULL;
3944    if (wd->end)
3945      {
3946         Evas_Object *edje = elm_smart_scroller_edje_object_get(wd->scroller);
3947         if (!edje) return NULL;
3948         ret = wd->end;
3949         edje_object_part_unswallow(edje, wd->end);
3950         edje_object_signal_emit(edje, "elm,action,hide,end", "elm");
3951         wd->end = NULL;
3952         _sizing_eval(obj);
3953      }
3954    return ret;
3955 }
3956
3957 EAPI void
3958 elm_entry_end_visible_set(Evas_Object *obj, Eina_Bool setting)
3959 {
3960    ELM_CHECK_WIDTYPE(obj, widtype);
3961    Widget_Data *wd = elm_widget_data_get(obj);
3962    if ((!wd) || (!wd->end)) return;
3963    if (setting)
3964       evas_object_hide(wd->end);
3965    else
3966       evas_object_show(wd->end);
3967    _sizing_eval(obj);
3968 }
3969
3970 EAPI void
3971 elm_entry_scrollbar_policy_set(Evas_Object *obj, Elm_Scroller_Policy h, Elm_Scroller_Policy v)
3972 {
3973    ELM_CHECK_WIDTYPE(obj, widtype);
3974    Widget_Data *wd = elm_widget_data_get(obj);
3975    const Elm_Scroller_Policy map[3] =
3976      {
3977         ELM_SMART_SCROLLER_POLICY_AUTO,
3978         ELM_SMART_SCROLLER_POLICY_ON,
3979         ELM_SMART_SCROLLER_POLICY_OFF
3980      };
3981    if (!wd) return;
3982    wd->policy_h = h;
3983    wd->policy_v = v;
3984    elm_smart_scroller_policy_set(wd->scroller,
3985                                  map[wd->policy_h],
3986                                  map[wd->policy_v]);
3987 }
3988
3989 EAPI void
3990 elm_entry_bounce_set(Evas_Object *obj, Eina_Bool h_bounce, Eina_Bool v_bounce)
3991 {
3992    ELM_CHECK_WIDTYPE(obj, widtype);
3993    Widget_Data *wd = elm_widget_data_get(obj);
3994    if (!wd) return;
3995    elm_smart_scroller_bounce_allow_set(wd->scroller, h_bounce, v_bounce);
3996 }
3997
3998 EAPI void
3999 elm_entry_bounce_get(const Evas_Object *obj, Eina_Bool *h_bounce, Eina_Bool *v_bounce)
4000 {
4001    ELM_CHECK_WIDTYPE(obj, widtype);
4002    Widget_Data *wd = elm_widget_data_get(obj);
4003    if (!wd) return;
4004    elm_smart_scroller_bounce_allow_get(wd->scroller, h_bounce, v_bounce);
4005 }
4006
4007 EAPI void
4008 elm_entry_input_panel_layout_set(Evas_Object *obj, Elm_Input_Panel_Layout layout)
4009 {
4010    ELM_CHECK_WIDTYPE(obj, widtype);
4011    Widget_Data *wd = elm_widget_data_get(obj);
4012    if (!wd) return;
4013
4014    wd->input_panel_layout = layout;
4015
4016    edje_object_part_text_input_panel_layout_set(wd->ent, "elm.text", layout);
4017 }
4018
4019 EAPI Elm_Input_Panel_Layout
4020 elm_entry_input_panel_layout_get(Evas_Object *obj)
4021 {
4022    ELM_CHECK_WIDTYPE(obj, widtype) ELM_INPUT_PANEL_LAYOUT_INVALID;
4023    Widget_Data *wd = elm_widget_data_get(obj);
4024    if (!wd) return ELM_INPUT_PANEL_LAYOUT_INVALID;
4025
4026    return wd->input_panel_layout;
4027 }
4028
4029 EAPI void
4030 elm_entry_autocapital_type_set(Evas_Object *obj, Elm_Autocapital_Type autocapital_type)
4031 {
4032    ELM_CHECK_WIDTYPE(obj, widtype);
4033    Widget_Data *wd = elm_widget_data_get(obj);
4034    if (!wd) return;
4035
4036    wd->autocapital_type = autocapital_type;
4037    edje_object_part_text_autocapital_type_set(wd->ent, "elm.text", autocapital_type);
4038 }
4039
4040 EAPI Elm_Autocapital_Type
4041 elm_entry_autocapital_type_get(Evas_Object *obj)
4042 {
4043    ELM_CHECK_WIDTYPE(obj, widtype) ELM_AUTOCAPITAL_TYPE_NONE;
4044    Widget_Data *wd = elm_widget_data_get(obj);
4045    if (!wd) return ELM_AUTOCAPITAL_TYPE_NONE;
4046
4047    return wd->autocapital_type;
4048 }
4049
4050 EAPI void
4051 elm_entry_input_panel_enabled_set(Evas_Object *obj, Eina_Bool enabled)
4052 {
4053    ELM_CHECK_WIDTYPE(obj, widtype);
4054    Widget_Data *wd = elm_widget_data_get(obj);
4055    if (!wd) return;
4056
4057    wd->input_panel_enable = enabled;
4058    edje_object_part_text_input_panel_enabled_set(wd->ent, "elm.text", enabled);
4059 }
4060
4061 EINA_DEPRECATED EAPI void
4062 elm_entry_line_char_wrap_set(Evas_Object *obj, Eina_Bool wrap)
4063 {
4064    if (wrap) elm_entry_line_wrap_set(obj, ELM_WRAP_CHAR);
4065 }
4066
4067 EAPI void
4068 elm_entry_background_color_set(Evas_Object *obj, unsigned int r, unsigned int g, unsigned int b, unsigned int a)
4069 {
4070    ELM_CHECK_WIDTYPE(obj, widtype);
4071    Widget_Data *wd = elm_widget_data_get(obj);
4072    evas_object_color_set(wd->bg, r, g, b, a);
4073
4074    if (wd->bgcolor == EINA_FALSE)
4075      {
4076        wd->bgcolor = 1;
4077        edje_object_part_swallow(wd->ent, "entry.swallow.background", wd->bg);
4078      }
4079 }
4080
4081 EAPI void
4082 elm_entry_autocapitalization_set(Evas_Object *obj, Eina_Bool autocap)
4083 {
4084    ELM_CHECK_WIDTYPE(obj, widtype);
4085    Widget_Data *wd = elm_widget_data_get(obj);
4086    if (!wd) return;
4087
4088    if (autocap)
4089      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_SENTENCE;
4090    else
4091      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_NONE;
4092
4093    if (wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_URL ||
4094        wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_EMAIL)
4095      wd->autocapital_type = ELM_AUTOCAPITAL_TYPE_NONE;
4096
4097    edje_object_part_text_autocapital_type_set(wd->ent, "elm.text", wd->autocapital_type);
4098 }
4099
4100 EAPI void
4101 elm_entry_autoperiod_set(Evas_Object *obj, Eina_Bool autoperiod)
4102 {
4103    ELM_CHECK_WIDTYPE(obj, widtype);
4104    Widget_Data *wd = elm_widget_data_get(obj);
4105    if (!wd) return;
4106
4107    if (wd->password)
4108      wd->autoperiod = EINA_FALSE;
4109    else
4110      wd->autoperiod = autoperiod;
4111
4112    if (wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_URL ||
4113        wd->input_panel_layout == ELM_INPUT_PANEL_LAYOUT_EMAIL)
4114      wd->autoperiod = EINA_FALSE;
4115
4116    edje_object_part_text_autoperiod_set(wd->ent, "elm.text", wd->autoperiod);
4117 }
4118
4119 EAPI void
4120 elm_entry_autoenable_returnkey_set(Evas_Object *obj, Eina_Bool on)
4121 {
4122    ELM_CHECK_WIDTYPE(obj, widtype);
4123    Widget_Data *wd = elm_widget_data_get(obj);
4124    if (!wd) return;
4125
4126    wd->autoreturnkey = on;
4127    _check_enable_returnkey(obj);
4128 }
4129
4130 EAPI Ecore_IMF_Context *elm_entry_imf_context_get(Evas_Object *obj)
4131 {
4132    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4133    Widget_Data *wd = elm_widget_data_get(obj);
4134    if (!wd || !wd->ent) return NULL;
4135
4136    return edje_object_part_text_imf_context_get(wd->ent, "elm.text");
4137 }
4138
4139 EAPI void
4140 elm_entry_matchlist_set(Evas_Object *obj, Eina_List *match_list, Eina_Bool case_sensitive)
4141 {
4142    Widget_Data *wd = elm_widget_data_get(obj);
4143    if (!wd) return;
4144
4145    if (match_list)
4146      {
4147         Evas_Coord max_w = 9999, max_h = 9999;
4148         const char* key_data = NULL;
4149
4150         wd->matchlist_threshold = 1;
4151         wd->hover = elm_hover_add(elm_widget_parent_get(obj));
4152         elm_hover_parent_set(wd->hover, elm_widget_parent_get(obj));
4153         elm_hover_target_set(wd->hover, obj);
4154         elm_object_style_set(wd->hover, "matchlist");
4155
4156         wd->layout = elm_layout_add(wd->hover);
4157         elm_layout_theme_set(wd->layout, "entry", "matchlist", "default");
4158         wd->list = elm_list_add(wd->layout);
4159         evas_object_size_hint_weight_set(wd->list, EVAS_HINT_EXPAND, 0.0);
4160         evas_object_size_hint_align_set(wd->list, EVAS_HINT_FILL, EVAS_HINT_FILL);
4161         elm_list_mode_set(wd->list, ELM_LIST_EXPAND);
4162         elm_object_style_set(wd->list, "matchlist");
4163
4164         key_data = edje_object_data_get(elm_layout_edje_get(wd->layout), "max_width");
4165         if (key_data) max_w = atoi(key_data);
4166         key_data = edje_object_data_get(elm_layout_edje_get(wd->layout), "max_height");
4167         if (key_data) max_h = atoi(key_data);
4168
4169         elm_list_go(wd->list);
4170         evas_object_size_hint_max_set(wd->list, max_w, max_h);
4171         evas_object_smart_callback_add(wd->list, "selected", _matchlist_list_clicked, obj);
4172         elm_layout_content_set(wd->layout, "elm.swallow.content", wd->list);
4173         elm_hover_content_set(wd->hover, "bottom", wd->layout);
4174
4175         wd->match_list = match_list;
4176      }
4177    else
4178      {
4179         if (wd->hover)
4180           evas_object_del(wd->hover);
4181
4182         wd->match_list = NULL;
4183      }
4184
4185    wd->matchlist_case_sensitive = case_sensitive;
4186 }
4187
4188 EAPI void
4189 elm_entry_magnifier_type_set(Evas_Object *obj, int type)
4190 {
4191    ELM_CHECK_WIDTYPE(obj, widtype);
4192    Widget_Data *wd = elm_widget_data_get(obj);
4193    if (!wd) return;
4194
4195    wd->mgf_type = type;
4196    _magnifier_create(obj);
4197 }
4198
4199 EAPI void
4200 elm_entry_wrap_width_set(Evas_Object *obj, Evas_Coord w)
4201 {
4202    Widget_Data *wd = elm_widget_data_get(obj);
4203    if (!wd) return;
4204    if (wd->wrap_w == w) return;
4205    wd->wrap_w = w;
4206    _sizing_eval(obj);
4207 }
4208
4209 EAPI Evas_Coord
4210 elm_entry_wrap_width_get(const Evas_Object *obj)
4211 {
4212    Widget_Data *wd = elm_widget_data_get(obj);
4213    if (!wd) return 0;
4214    return wd->wrap_w;
4215 }
4216
4217 EAPI void
4218 elm_entry_fontsize_set(Evas_Object *obj, int fontsize)
4219 {
4220    ELM_CHECK_WIDTYPE(obj, widtype);
4221    Widget_Data *wd = elm_widget_data_get(obj);
4222    Eina_Strbuf *fontbuf = NULL;
4223    int removeflag = 0;
4224    const char *t;
4225
4226    if (!wd) return;
4227    t = eina_stringshare_add(elm_entry_entry_get(obj));
4228    fontbuf = eina_strbuf_new();
4229    eina_strbuf_append_printf(fontbuf, "%d", fontsize);
4230
4231    if (fontsize == 0) removeflag = 1; // remove fontsize tag
4232
4233    if (_stringshare_key_value_replace(&t, "font_size", eina_strbuf_string_get(fontbuf), removeflag) == 0)
4234      {
4235        elm_entry_entry_set(obj, t);
4236        wd->changed = 1;
4237        _sizing_eval(obj);
4238      }
4239    eina_strbuf_free(fontbuf);
4240    eina_stringshare_del(t);
4241 }
4242
4243 EAPI void
4244 elm_entry_text_color_set(Evas_Object *obj, unsigned int r, unsigned int g, unsigned int b, unsigned int a)
4245 {
4246    ELM_CHECK_WIDTYPE(obj, widtype);
4247    Widget_Data *wd = elm_widget_data_get(obj);
4248    Eina_Strbuf *colorbuf = NULL;
4249    const char *t;
4250    int len;
4251
4252    if (!wd) return;
4253    t = eina_stringshare_add(elm_entry_entry_get(obj));
4254    len = strlen(t);
4255    if (len <= 0) return;
4256    colorbuf = eina_strbuf_new();
4257    eina_strbuf_append_printf(colorbuf, "#%02X%02X%02X%02X", r, g, b, a);
4258
4259    if (_stringshare_key_value_replace(&t, "color", eina_strbuf_string_get(colorbuf), 0) == 0)
4260      {
4261        elm_entry_entry_set(obj, t);
4262        wd->changed = 1;
4263        _sizing_eval(obj);
4264      }
4265    eina_strbuf_free(colorbuf);
4266    eina_stringshare_del(t);
4267 }
4268
4269 EAPI void
4270 elm_entry_text_align_set(Evas_Object *obj, const char *alignmode)
4271 {
4272    ELM_CHECK_WIDTYPE(obj, widtype);
4273    Widget_Data *wd = elm_widget_data_get(obj);
4274    int len;
4275    const char *t;
4276
4277    if (!wd) return;
4278    t = eina_stringshare_add(elm_entry_entry_get(obj));
4279    len = strlen(t);
4280    if (len <= 0) return;
4281
4282    if (_stringshare_key_value_replace(&t, "align", alignmode, 0) == 0)
4283      elm_entry_entry_set(obj, t);
4284
4285    wd->changed = 1;
4286    _sizing_eval(obj);
4287    eina_stringshare_del(t);
4288 }