[C&P popup] add search button & change clipboard icon to text
[framework/uifw/elementary.git] / src / modules / ctxpopup_copypasteUI / copypaste.c
index 0ad6794..d35e4ca 100644 (file)
@@ -1,6 +1,7 @@
 #include <Elementary.h>
 #include "elm_module_priv.h"
 #include "elm_priv.h"
+#include <appsvc/appsvc.h>
 
 Elm_Entry_Extension_data *ext_mod;
 static int _mod_hook_count = 0;
@@ -17,28 +18,84 @@ struct _Elm_Entry_Context_Menu_Item
    void *data;
 };
 
+static void _ctxpopup_hide(Evas_Object *popup);
+static void
+_entry_move(void *data, Evas *e, Evas_Object *obj, void *event_info)
+{
+   _ctxpopup_hide(ext_mod->popup);
+}
+
+static void
+_ctxpopup_hide(Evas_Object *popup)
+{
+   evas_object_hide(popup);
+   evas_object_event_callback_del(ext_mod->caller, EVAS_CALLBACK_MOVE, _entry_move);
+   ext_mod->caller = NULL;
+}
+
 static void
 _ctxpopup_position(Evas_Object *obj)
 {
    if(!ext_mod) return;
 
-   Evas_Coord cx, cy, cw, ch, x, y, mw, mh;
-   evas_object_geometry_get(ext_mod->ent, &x, &y, NULL, NULL);
-   edje_object_part_text_cursor_geometry_get(ext_mod->ent, "elm.text",
-                                             &cx, &cy, &cw, &ch);
-   evas_object_size_hint_min_get(ext_mod->popup, &mw, &mh);
-   if (cw < mw)
+   Evas_Coord cx, cy, cw, ch, x, y, w, h;
+   if (!edje_object_part_text_selection_geometry_get(ext_mod->ent, "elm.text", &x, &y, &w, &h))
      {
-        cx += (cw - mw) / 2;
-        cw = mw;
+        evas_object_geometry_get(ext_mod->ent, &x, &y, NULL, NULL);
+        edje_object_part_text_cursor_geometry_get(ext_mod->ent, "elm.text",
+                                                  &cx, &cy, &cw, &ch);
+        evas_object_size_hint_min_get(ext_mod->popup, &w, &h);
+        if (cw < w)
+          {
+             cx += (cw - w) / 2;
+             cw = w;
+          }
+        if (ch < h)
+          {
+             cy += (ch - h) / 2;
+             ch = h;
+          }
+        evas_object_move(ext_mod->popup, x + cx, y + cy);
+        evas_object_resize(ext_mod->popup, cw, ch);
      }
-   if (ch < mh)
+   else
      {
-        cy += (ch - mh) / 2;
-        ch = mh;
+        if (ext_mod->viewport_rect.x != -1 || ext_mod->viewport_rect.y != -1
+            || ext_mod->viewport_rect.w != -1 || ext_mod->viewport_rect.h != -1)
+          {
+             Evas_Coord vx, vy, vw, vh, x2, y2;
+             x2 = x + w;
+             y2 = y + h;
+             vx = ext_mod->viewport_rect.x;
+             vy = ext_mod->viewport_rect.y;
+             vw = ext_mod->viewport_rect.w;
+             vh = ext_mod->viewport_rect.h;
+
+             if (x < vx) x = vx;
+             if (y < vy) y = vy;
+             if (x2 > vx + vw) x2 = vx + vw;
+             if (y2 > vy + vh) y2 = vy + vh;
+             w = x2 - x;
+             h = y2 - y;
+          }
+        else
+          {
+             Evas_Coord sw, sh, x2, y2;
+             x2 = x + w;
+             y2 = y + h;
+             ecore_x_window_size_get(ecore_x_window_root_first_get(), &sw, &sh);
+
+             if (x < 0) x = 0;
+             if (y < 0) y = 0;
+             if (x2 > sw) x2 = sw;
+             if (y2 > sh) y2 = sh;
+             w = x2 - x;
+             h = y2 - y;
+          }
+        cx = x + (w / 2);
+        cy = y + (h / 2);
+        evas_object_move(ext_mod->popup, cx, cy);
      }
-   evas_object_move(ext_mod->popup, x + cx, y + cy);
-   evas_object_resize(ext_mod->popup, cw, ch);
 }
 
 static void
@@ -47,7 +104,7 @@ _select_all(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->selectall(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
 }
 
 static void
@@ -56,7 +113,7 @@ _select(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->select(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
 }
 
 static void
@@ -65,7 +122,7 @@ _paste(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->paste(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
 }
 
 static void
@@ -74,7 +131,7 @@ _cut(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->cut(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
    elm_object_scroll_freeze_pop(ext_mod->popup);
 }
 
@@ -84,7 +141,7 @@ _copy(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->copy(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
    elm_object_scroll_freeze_pop(ext_mod->popup);
 }
 
@@ -94,11 +151,36 @@ _cancel(void *data, Evas_Object *obj, void *event_info)
    if(!ext_mod) return;
 
    ext_mod->cancel(data,obj,event_info);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
    elm_object_scroll_freeze_pop(ext_mod->popup);
 }
 
 static void
+_search_menu(void *data, Evas_Object *obj, void *event_info)
+{
+   if(!ext_mod) return;
+
+   int ret;
+   bundle *b = bundle_create();
+   if (!b)
+     {
+        //printf("bundle_create() failed\n");
+        return;
+     }
+
+   appsvc_set_operation(b, APPSVC_OPERATION_SEARCH);
+   if (ext_mod->selmode)
+     {
+        char *selection = elm_entry_selection_get(ext_mod->caller);
+        if (selection)
+          appsvc_add_data(b, APPSVC_DATA_KEYWORD, selection);
+     }
+   appsvc_run_service(b, 0, NULL, NULL);
+   bundle_free(b);
+   _ctxpopup_hide(obj);
+}
+
+static void
 _clipboard_menu(void *data, Evas_Object *obj, void *event_info)
 {
    if(!ext_mod) return;
@@ -109,11 +191,11 @@ _clipboard_menu(void *data, Evas_Object *obj, void *event_info)
 #endif
    ext_mod->cnpinit(data,obj,event_info);
    elm_cbhm_helper_init(obj);
-   if (ext_mod->textonly)
+   if (ext_mod->cnp_mode != ELM_CNP_MODE_MARKUP)
      elm_cbhm_send_raw_data("show0");
    else
      elm_cbhm_send_raw_data("show1");
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
    // end for cbhm
 }
 
@@ -124,7 +206,7 @@ _item_clicked(void *data, Evas_Object *obj, void *event_info)
    Evas_Object *obj2 = it->obj;
 
    if (it->func) it->func(it->data, obj2, NULL);
-   evas_object_hide(obj);
+   _ctxpopup_hide(obj);
 }
 
 static void
@@ -186,11 +268,17 @@ obj_longpress(Evas_Object *obj)
    const char *context_menu_orientation;
    char buf[255];
    Evas_Object* icon;
+   Elm_Object_Item *added_item = NULL;
 
    /*update*/
    elm_entry_extension_module_data_get(obj,ext_mod);
    if (ext_mod->context_menu)
      {
+#ifdef HAVE_ELEMENTARY_X
+        int cbhm_count = -1;
+        if (elm_cbhm_helper_init(obj))
+          cbhm_count = elm_cbhm_get_count();
+#endif
         if (ext_mod->popup) evas_object_del(ext_mod->popup);
         //else elm_widget_scroll_freeze_push(obj);
         top = elm_widget_top_get(obj);
@@ -215,26 +303,36 @@ obj_longpress(Evas_Object *obj)
                {
                   if (!elm_entry_is_empty(obj))
                     {
-                       elm_ctxpopup_item_append(ext_mod->popup, "Select", NULL, _select, obj );
-                       elm_ctxpopup_item_append(ext_mod->popup, "Select All", NULL, _select_all, obj );
+                       added_item = elm_ctxpopup_item_append(ext_mod->popup, "Select", NULL, _select, obj );
+                       added_item = elm_ctxpopup_item_append(ext_mod->popup, "Select All", NULL, _select_all, obj );
                     }
                }
+
+#ifdef HAVE_ELEMENTARY_X
+             if (cbhm_count)
+#else
              if (1) // need way to detect if someone has a selection
+#endif
                {
                   if (ext_mod->editable)
-                    elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
+                    added_item = elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
                }
              //elm_ctxpopup_item_append(wd->ctxpopup, NULL, "Selectall",_select_all, obj );
              // start for cbhm
+#ifdef HAVE_ELEMENTARY_X
+             if ((!ext_mod->password) && (ext_mod->editable) && (cbhm_count))
+#else
              if ((!ext_mod->password) && (ext_mod->editable))
+#endif
                {
-                  icon = elm_icon_add(ext_mod->popup);
-                  snprintf(buf, sizeof(buf), "%s/images/copypaste_icon_clipboard.png", PACKAGE_DATA_DIR);
-                  elm_icon_file_set(icon, buf, NULL);
-                  elm_ctxpopup_item_append(ext_mod->popup, NULL, icon, _clipboard_menu, obj);
+                  added_item = elm_ctxpopup_item_append(ext_mod->popup, "Clipboard", NULL, _clipboard_menu, obj);  // Clipboard
                   //elm_ctxpopup_item_append(ext_mod->popup, "More", NULL, _clipboard_menu, obj );
                }
              // end for cbhm
+             icon = elm_icon_add(ext_mod->popup);
+             snprintf(buf, sizeof(buf), "%s/images/copy&paste_icon_search.png", PACKAGE_DATA_DIR);
+             elm_icon_file_set(icon, buf, NULL);
+             added_item = elm_ctxpopup_item_append(ext_mod->popup, NULL, icon, _search_menu, obj);  // Search
           }
         else
           {
@@ -242,50 +340,66 @@ obj_longpress(Evas_Object *obj)
                {
                   if (ext_mod->have_selection)
                     {
-                       elm_ctxpopup_item_append(ext_mod->popup, "Copy", NULL, _copy, obj );
+                       added_item = elm_ctxpopup_item_append(ext_mod->popup, "Copy", NULL, _copy, obj );
                        if (ext_mod->editable)
-                         elm_ctxpopup_item_append(ext_mod->popup, "Cut", NULL, _cut, obj );
+                         added_item = elm_ctxpopup_item_append(ext_mod->popup, "Cut", NULL, _cut, obj );
+#ifdef HAVE_ELEMENTARY_X
+                       if (ext_mod->editable && cbhm_count)
+#else
                        if (ext_mod->editable)
-                         elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
+#endif
+                         added_item = elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
                     }
                   else
                     {
                        _cancel(obj,ext_mod->popup,NULL);
                        if (!elm_entry_is_empty(obj))
                          {
-                            elm_ctxpopup_item_append(ext_mod->popup, "Select", NULL, _select, obj );
-                            elm_ctxpopup_item_append(ext_mod->popup, "Select All", NULL, _select_all, obj );
+                            added_item = elm_ctxpopup_item_append(ext_mod->popup, "Select", NULL, _select, obj );
+                            added_item = elm_ctxpopup_item_append(ext_mod->popup, "Select All", NULL, _select_all, obj );
                          }
+#ifdef HAVE_ELEMENTARY_X
+                       if (cbhm_count)
+#else
                        if (1) // need way to detect if someone has a selection
+#endif
                          {
                             if (ext_mod->editable)
-                              elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
+                              added_item = elm_ctxpopup_item_append(ext_mod->popup, "Paste", NULL, _paste, obj );
                          }
                     }
                   // start for cbhm
+#ifdef HAVE_ELEMENTARY_X
+                  if (ext_mod->editable && cbhm_count)
+#else
                   if (ext_mod->editable)
+#endif
                     {
-                       icon = elm_icon_add(ext_mod->popup);
-                       snprintf(buf, sizeof(buf), "%s/images/copypaste_icon_clipboard.png", PACKAGE_DATA_DIR);
-                       elm_icon_file_set(icon, buf, NULL);
-                       elm_ctxpopup_item_append(ext_mod->popup, NULL, icon, _clipboard_menu, obj);
+                       added_item = elm_ctxpopup_item_append(ext_mod->popup, "Clipboard", NULL, _clipboard_menu, obj);  // Clipboard
                        //elm_ctxpopup_item_append(ext_mod->popup, "More", NULL, _clipboard_menu, obj );
                     }
                   // end for cbhm
+                  icon = elm_icon_add(ext_mod->popup);
+                  snprintf(buf, sizeof(buf), "%s/images/copy&paste_icon_search.png", PACKAGE_DATA_DIR);
+                  elm_icon_file_set(icon, buf, NULL);
+                  added_item = elm_ctxpopup_item_append(ext_mod->popup, NULL, icon, _search_menu, obj);  // Search
                }
           }
         EINA_LIST_FOREACH(ext_mod->items, l, it)
           {
-             elm_ctxpopup_item_append(ext_mod->popup, it->label, NULL, _item_clicked, it );
+             added_item = elm_ctxpopup_item_append(ext_mod->popup, it->label, NULL, _item_clicked, it );
           }
-        if (ext_mod->popup)
+        if (ext_mod->popup && added_item)
           {
              elm_object_scroll_freeze_push(ext_mod->popup);
              _ctxpopup_position(obj);
              evas_object_show(ext_mod->popup);
+             ext_mod->caller = obj;
+             evas_object_event_callback_add(obj, EVAS_CALLBACK_MOVE, _entry_move, ext_mod->popup);
           }
+        else
+          ext_mod->caller = NULL;
      }
-   ext_mod->longpress_timer = NULL;
 }
 
 EAPI void
@@ -299,11 +413,9 @@ obj_mouseup(Evas_Object *obj)
 EAPI void
 obj_hidemenu(Evas_Object *obj)
 {
-   if (!obj || !ext_mod)
+   if (!obj || !ext_mod || obj != ext_mod->caller)
      return;
 
-   evas_object_hide(ext_mod->popup);
+   _ctxpopup_hide(ext_mod->popup);
    // if (ext_mod->popup) evas_object_del(ext_mod->popup);
 }
-
-