Support API for set/get cursor position
[platform/core/uifw/inputdelegator.git] / src / w-input-selector.cpp
index 7b08443..57320ec 100755 (executable)
@@ -30,6 +30,7 @@
 #include "w-input-keyboard.h"
 #include "w-input-stt-ise.h"
 #include "w-input-emoticon.h"
+#include "w-input-smartreply.h"
 
 #include <stt.h>
 
@@ -44,44 +45,44 @@ InputTypeData g_input_type_data;
 static Elm_Object_Item *it_empty;
 static Elm_Object_Item *it_title;
 
+static unsigned int g_smartreply_item_size = 0;                /* Current Smartreply item size */
 static unsigned int g_template_item_size = 0;          /* Current Template item size */
 
 Evas_Coord last_step; // 0 ~ 9 for gesture, 10~11 for rotary
 
 tizen_profile_t _get_tizen_profile()
 {
-   static tizen_profile_t profile = TIZEN_PROFILE_UNKNOWN;
-   if (__builtin_expect(profile != TIZEN_PROFILE_UNKNOWN, 1))
-     return profile;
-
-   char *profileName;
-   system_info_get_platform_string("http://tizen.org/feature/profile", &profileName);
-   switch (*profileName)
-     {
-      case 'm':
-      case 'M':
-         profile = TIZEN_PROFILE_MOBILE;
-         break;
-      case 'w':
-      case 'W':
-         profile = TIZEN_PROFILE_WEARABLE;
-         break;
-      case 't':
-      case 'T':
-         profile = TIZEN_PROFILE_TV;
-         break;
-      case 'i':
-      case 'I':
-         profile = TIZEN_PROFILE_IVI;
-         break;
-      default: // common or unknown ==> ALL ARE COMMON.
-         profile = TIZEN_PROFILE_COMMON;
-     }
-   free(profileName);
-
-   return profile;
-}
+       static tizen_profile_t profile = TIZEN_PROFILE_UNKNOWN;
+       if (__builtin_expect(profile != TIZEN_PROFILE_UNKNOWN, 1))
+               return profile;
+
+       char *profileName;
+       system_info_get_platform_string("http://tizen.org/feature/profile", &profileName);
+       switch (*profileName)
+       {
+               case 'm':
+               case 'M':
+                       profile = TIZEN_PROFILE_MOBILE;
+                       break;
+               case 'w':
+               case 'W':
+                       profile = TIZEN_PROFILE_WEARABLE;
+                       break;
+               case 't':
+               case 'T':
+                       profile = TIZEN_PROFILE_TV;
+                       break;
+               case 'i':
+               case 'I':
+                       profile = TIZEN_PROFILE_IVI;
+                       break;
+               default: // common or unknown ==> ALL ARE COMMON.
+                       profile = TIZEN_PROFILE_COMMON;
+       }
+       free(profileName);
 
+       return profile;
+}
 
 void _init_app_data(App_Data* app_data);
 static void _app_language_changed(app_event_info_h event_info, void *user_data);
@@ -90,11 +91,14 @@ Evas_Object* _create_genlist(Evas_Object* parent);
 void _create_genlist_items(void* user_data);
 void _create_header_items(void *user_data);
 void _update_genlist_items(void *user_data);
+unsigned int _update_smartreply_items(void *user_data);
 unsigned int _update_template_items(void *user_data);
 static void _popup_close_cb(void *data, Evas_Object *obj, void *event_info);
 static void _popup_back_cb(void *data, Evas_Object *obj, void *event_info);
 static void input_type_deinit(void);
 
+bool __compare_string(const char * src, const char * dst);
+
 void _init_app_data(App_Data* app_data)
 {
        app_data->win_main = NULL;
@@ -137,7 +141,10 @@ void init_customizing_theme(void)
        if (_WEARABLE) {
                stt_edj_path = stt_edj_path + STT_EDJ_FILE_WEARABLE;
                app_edj_path = app_edj_path + APP_EDJ_FILE_WEARABLE;
-       } else if (_MOBILE) {
+       } else if (_TV) {
+               stt_edj_path = stt_edj_path + STT_EDJ_FILE_TV;
+               app_edj_path = app_edj_path + APP_EDJ_FILE_TV;
+       } else {
                stt_edj_path = stt_edj_path + STT_EDJ_FILE_MOBILE;
                app_edj_path = app_edj_path + APP_EDJ_FILE_MOBILE;
        }
@@ -150,7 +157,7 @@ void init_customizing_theme(void)
 static Eina_Bool back_cb(void *data, Elm_Object_Item *it)
 {
        reply_to_sender_by_callback_for_back();
-       elm_exit();
+       ui_app_exit();
        return EINA_FALSE;
 }
 
@@ -166,6 +173,12 @@ static void _stt_clicked_cb(void *data, Evas_Object * obj, void *event_info)
        ise_show_stt_popup(ad);
 }
 
+static void _input_smartreply_notify_cb(void *user_data)
+{
+       ecore_main_loop_iterate();
+       _update_genlist_items((void *)app_data);
+}
+
 static void _input_template_notify_cb(void *user_data)
 {
        _update_genlist_items((void *)app_data);
@@ -195,6 +208,26 @@ static void _keyboard_clicked_cb(void *data, Evas_Object * obj, void *event_info
        input_keyboard_launch(ad->win_main, data);
 }
 
+static void __ise_smartreply_gl_sel(void *data, Evas_Object *obj, void *event_info)
+{
+       Elm_Object_Item *item = (Elm_Object_Item *) event_info;
+
+       if (item) {
+               elm_genlist_item_selected_set(item, EINA_FALSE);
+
+               int index = (intptr_t) elm_object_item_data_get(item);
+
+               int type;
+               char *reply = input_smartreply_get_nth_item(index, &type);
+               if (reply) {
+                       input_smartreply_send_feedback(reply);
+                       reply_to_sender_by_callback(reply, "smartreply", NULL, NULL);
+                       free(reply);
+                       elm_exit();
+               }
+       }
+}
+
 static void __ise_template_gl_sel(void *data, Evas_Object *obj, void *event_info)
 {
        Elm_Object_Item *item = (Elm_Object_Item *) event_info;
@@ -207,8 +240,8 @@ static void __ise_template_gl_sel(void *data, Evas_Object *obj, void *event_info
                const std::vector<TemplateData>  template_list = input_template_get_list();
 
                if (index < (int)template_list.size()) {
-                       reply_to_sender_by_callback(gettext(template_list[index].text.c_str()), "template");
-                       elm_exit();
+                       reply_to_sender_by_callback(gettext(template_list[index].text.c_str()), "template", NULL, NULL);
+                       ui_app_exit();
                }
        }
 }
@@ -262,6 +295,8 @@ static Evas_Object * __ise_gl_2button_content_get(void *data, Evas_Object *obj,
                string path = get_resource_path();
                if (_WEARABLE)
                        path = path + "wearable/";
+               else if (_TV)
+                       path = path + "tv/";
                else
                        path = path + "mobile/";
 
@@ -346,6 +381,8 @@ static Evas_Object * __ise_gl_3button_content_get(void *data, Evas_Object *obj,
                string path = get_resource_path();
                if (_WEARABLE)
                        path = path + "wearable/";
+               else if (_TV)
+                       path = path + "tv/";
                else
                        path = path + "mobile/";
 
@@ -401,6 +438,23 @@ static Evas_Object * __ise_gl_3button_content_get(void *data, Evas_Object *obj,
        return NULL;
 }
 
+static char * __ise_smartreply_gl_text_get(void *data, Evas_Object *obj, const char *part)
+{
+       if(!strcmp(part, "elm.text")) {
+               int index;
+               char *reply = NULL;
+               index = (intptr_t)data;
+               if (index < 0)
+                       return NULL;
+
+               reply = input_smartreply_get_nth_item(index, NULL);
+               if (reply == NULL)
+                       return NULL;
+
+               return reply;
+       }
+       return NULL;
+}
 
 static void __ise_gl_lang_changed(void *data, Evas_Object *obj, void *event_info)
 {
@@ -425,7 +479,7 @@ void set_source_caller_app_id(app_control_h app_control)
        }
 }
 
-void reply_to_sender_by_callback(const char *value, const char *type)
+void reply_to_sender_by_callback(const char *value, const char *type, const char *path[], const char *cursor_position)
 {
        PRINTFUNC(DLOG_DEBUG, "");
 
@@ -440,6 +494,13 @@ void reply_to_sender_by_callback(const char *value, const char *type)
                if (type)
                        app_control_add_extra_data(app_control, "reply_type", type);
 
+               if (path != NULL) {
+                       app_control_add_extra_data_array(app_control, APP_CONTROL_DATA_PATH, path, 1);
+               }
+
+               if (cursor_position != NULL)
+                       app_control_add_extra_data(app_control, "cursor_position_get", cursor_position);
+
                set_source_caller_app_id(app_control);
 
                ret = app_control_reply_to_launch_request(app_control, app_data->source_app_control, APP_CONTROL_RESULT_SUCCEEDED);
@@ -521,6 +582,8 @@ void show_popup_toast(const char *text, bool check_img)
                string path = get_resource_path();
                if (_WEARABLE)
                        path = path + "wearable/";
+               else if (_TV)
+                       path = path + "tv/";
                else
                        path = path + "mobile/";
                string path_ic = path + "/images/toast_check_icon.png";
@@ -561,15 +624,15 @@ void _back_to_genlist_for_selector()
 
        if (!app_data) return;
 
-       if (app_data->app_type == APP_TYPE_STT || app_data->app_type == APP_TYPE_EMOTICON){
-               PRINTFUNC(DLOG_DEBUG, "launched as STT/EMOTICON mode, So exit here.");
-               reply_to_sender_by_callback(NULL, NULL);
-               elm_exit();
+       if (_WEARABLE) {
+               Evas_Object *circle_genlist = (Evas_Object *) evas_object_data_get(app_data->genlist, "circle");
+               eext_rotary_object_event_activated_set(circle_genlist, EINA_TRUE);
+       }
+       if (app_data->app_type == APP_TYPE_STT || app_data->app_type == APP_TYPE_EMOTICON || app_data->app_type == APP_TYPE_KEYBOARD){
+               PRINTFUNC(DLOG_DEBUG, "launched as STT/EMOTICON/KEYBOARD mode, So exit here.");
+               reply_to_sender_by_callback(NULL, NULL, NULL, NULL);
+               ui_app_exit();
        }
-#ifdef _WEARABLE
-       Evas_Object *circle_genlist = (Evas_Object *) evas_object_data_get(app_data->genlist, "circle");
-       eext_rotary_object_event_activated_set(circle_genlist, EINA_TRUE);
-#endif
 }
 
 static void _item_realized(void *data, Evas_Object *obj, void *event_info) //called when list scrolled
@@ -584,12 +647,12 @@ Evas_Object* _create_genlist(Evas_Object* navi)
                return NULL;
 
        elm_genlist_mode_set(genlist, ELM_LIST_COMPRESS);
-#ifdef _WEARABLE
-       Evas_Object* circle_object_genlist = eext_circle_object_genlist_add(genlist, app_data->circle_surface);
-       eext_circle_object_genlist_scroller_policy_set(circle_object_genlist, ELM_SCROLLER_POLICY_OFF, ELM_SCROLLER_POLICY_OFF);
-       evas_object_data_set(genlist, "circle", (void *) circle_object_genlist);
-       eext_rotary_object_event_activated_set(circle_object_genlist, EINA_TRUE);
-#endif
+       if (_WEARABLE) {
+               Evas_Object* circle_object_genlist = eext_circle_object_genlist_add(genlist, app_data->circle_surface);
+               eext_circle_object_genlist_scroller_policy_set(circle_object_genlist, ELM_SCROLLER_POLICY_OFF, ELM_SCROLLER_POLICY_OFF);
+               evas_object_data_set(genlist, "circle", (void *) circle_object_genlist);
+               eext_rotary_object_event_activated_set(circle_object_genlist, EINA_TRUE);
+       }
        evas_object_size_hint_weight_set(genlist, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
        evas_object_size_hint_align_set(genlist, EVAS_HINT_FILL, EVAS_HINT_FILL);
 
@@ -600,12 +663,15 @@ Evas_Object* _create_genlist(Evas_Object* navi)
 
        show_gl_focus(EINA_FALSE);
 
+       const char *item_style = NULL;
+       if (_WEARABLE)
+               item_style = "empty";
        Elm_Object_Item *nf_main_item = elm_naviframe_item_push(navi,
             NULL,
             NULL,
             NULL,
             genlist,
-            "empty");
+            item_style);
 
        elm_naviframe_item_pop_cb_set(nf_main_item, back_cb, app_data);
        evas_object_smart_callback_add(genlist, "realized", _item_realized, NULL);
@@ -760,6 +826,11 @@ void _update_genlist_items(void *user_data)
 
        _create_header_items(user_data);
 
+       if (input_smartreply_is_enabled())
+               g_smartreply_item_size = _update_smartreply_items(user_data);
+       else
+               g_smartreply_item_size = 0;
+
        g_template_item_size = _update_template_items(user_data);
 
        /* Update genlist item position */
@@ -767,11 +838,107 @@ void _update_genlist_items(void *user_data)
        elm_genlist_item_show(item, ELM_GENLIST_ITEM_SCROLLTO_MIDDLE);
 }
 
+unsigned int _update_smartreply_items(void *user_data)
+{
+       App_Data* app_data;
+
+       Elm_Object_Item *first;
+       Elm_Object_Item *menu;
+
+       unsigned int i = 0;
+       unsigned int len = 0;
+       unsigned int item_size = 0;
+
+       app_data = (App_Data *)user_data;
+
+       if (app_data == NULL) {
+               PRINTFUNC(DLOG_ERROR, "Can not get app_data");
+               return item_size;
+       }
+
+       if (app_data->genlist == NULL) {
+               /* smartreply will update when genlist is exist only */
+               PRINTFUNC(DLOG_ERROR, "Can not get getlist");
+               return item_size;
+       }
+
+       /* Move to smartreply */
+       first = elm_genlist_first_item_get(app_data->genlist);
+       menu = elm_genlist_item_next_get(first);
+
+       if (input_smartreply_is_enabled() == false)
+               return item_size;
+
+       /* Append newly added smartreply list */
+       len = input_smartreply_get_reply_num();
+
+       if (len > 0) {
+               const std::vector<TemplateData>  template_list = input_template_get_list();
+
+               Elm_Genlist_Item_Class *itc;
+
+               itc = elm_genlist_item_class_new();
+               itc->item_style = "1text";
+               itc->func.text_get = __ise_smartreply_gl_text_get;
+               itc->func.content_get = NULL;
+               itc->func.state_get = NULL;
+               itc->func.del = NULL;
+
+               for (i = 0; i < len; i++) {
+                       char *reply = "hello";
+                       unsigned int j;
+                       bool matched;
+
+                       matched = false;
+
+                       int type;
+                       reply = input_smartreply_get_nth_item(i, &type);
+                       if (reply == NULL)
+                               continue;
+                       PRINTFUNC(DLOG_DEBUG, "SmartReply = [%d]%s", i, reply);
+
+                       for (j = 0; j < template_list.size(); j++) {
+                               const char *template_str;
+
+                               if (template_list[j].use_gettext)
+                                       template_str = gettext(template_list[j].text.c_str());
+                               else
+                                       template_str = template_list[j].text.c_str();
+
+                               if (!__compare_string(reply, template_str)) {
+                                       matched = true;
+                                       break;
+                               }
+                       }
+
+                       if (matched == true)
+                               continue;
+
+                       elm_genlist_item_append(app_data->genlist,
+                                               itc,
+                                               (void *)(uintptr_t)i,
+                                               NULL,
+                                               ELM_GENLIST_ITEM_NONE,
+                                               __ise_smartreply_gl_sel,
+                                               app_data);
+
+                       item_size++;
+                       if (item_size >= 3)
+                               break;
+               }
+               elm_genlist_item_class_free(itc);
+       }
+       return item_size;
+}
+
+
 unsigned int _update_template_items(void *user_data)
 {
        App_Data* app_data;
 
        Elm_Object_Item *first;
+       Elm_Object_Item *menu;
+       Elm_Object_Item *pos;
 
        unsigned int i = 0;
        unsigned int item_size = 0;
@@ -792,7 +959,20 @@ unsigned int _update_template_items(void *user_data)
        first = elm_genlist_first_item_get(app_data->genlist);
        if (first == NULL)
                return 0;
-       elm_genlist_item_next_get(first);
+       menu = elm_genlist_item_next_get(first);
+
+       pos = menu;
+       /* move to smartreply next if it need */
+       if (input_smartreply_is_enabled()) {
+               unsigned int j = 0;
+
+               for (j = 0; j < g_smartreply_item_size; j++) {
+                       if (pos == NULL)
+                               break;
+
+                       pos = elm_genlist_item_next_get(pos);
+               }
+       }
 
        /* Append New Template list */
        const std::vector<TemplateData> template_list = input_template_get_list();
@@ -809,7 +989,7 @@ unsigned int _update_template_items(void *user_data)
                itc->func.del = NULL;
 
                for (i = 0; i < template_list.size(); i++) {
-                       elm_genlist_item_append(app_data->genlist,
+                       pos = elm_genlist_item_append(app_data->genlist,
                                        itc,
                                        (void *)(uintptr_t)i,
                                        NULL,
@@ -824,6 +1004,34 @@ unsigned int _update_template_items(void *user_data)
        return item_size;
 }
 
+bool __compare_string(const char * smart_str, const char * template_str)
+{
+       int src_len = 0;
+       int dst_len = 0;
+       bool ret = true;
+
+       if(!smart_str||!template_str)
+               return ret;
+
+       src_len = strlen(smart_str);
+       dst_len = strlen(template_str);
+
+       if(smart_str[src_len-1] == '.')
+               src_len--;
+
+       while(template_str[dst_len-1] == ' ')
+               dst_len--;
+       if(template_str[dst_len-1] == '.')
+               dst_len--;
+
+       dst_len = (dst_len > src_len)? dst_len : src_len;
+
+       if(!strncmp(smart_str, template_str, (size_t)dst_len))
+               ret = false;
+
+       return (ret);
+}
+
 bool _app_create(void* user_data)
 {
        int width = 1000, height = 1000;
@@ -832,9 +1040,8 @@ bool _app_create(void* user_data)
        Evas_Object* conform = NULL;
        Evas_Object* bg = NULL;
        Evas_Object* window = NULL;
-#ifdef _WEARABLE
-       Eext_Circle_Surface *surface;
-#endif
+       Eext_Circle_Surface *surface = NULL;
+
        if (!user_data) {
                return false;
        }
@@ -869,11 +1076,10 @@ bool _app_create(void* user_data)
        evas_object_size_hint_weight_set(layout, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
 
        conform = elm_conformant_add(window);
-#ifdef _WEARABLE
-       surface = eext_circle_surface_conformant_add(conform);
-#endif
-       evas_object_size_hint_weight_set(conform, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
+       if (_WEARABLE)
+               surface = eext_circle_surface_conformant_add(conform);
 
+       evas_object_size_hint_weight_set(conform, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
 
        elm_win_resize_object_add(window, conform);
        elm_object_content_set(conform, layout);
@@ -884,9 +1090,7 @@ bool _app_create(void* user_data)
        app_data->win_main = window;
        app_data->conform = conform;
        app_data->layout_main = layout;
-#ifdef _WEARABLE
        app_data->circle_surface = surface;
-#endif
        app_data->app_type = APP_TYPE_SELECT_MODE;
 
        Evas_Object *naviframe = elm_naviframe_add(layout);
@@ -907,9 +1111,46 @@ void _app_service(app_control_h service, void* user_data)
        char **input_type_array = NULL;
        int input_type_array_len = -1;
        bool is_extra_data_array = false;
+       char *mime_type = NULL;
 
        app_control_clone(&(app_data->source_app_control), service);
        app_data->reply_type = REPLY_APP_NORMAL;
+       app_data->mime_type = MIME_TYPE_ALL;
+
+       ret = app_control_get_mime(service, &mime_type);
+       if (ret != APP_CONTROL_ERROR_NONE) {
+               LOGD("Fail to get mime type : %d", ret);
+       } else {
+               if (mime_type) {
+                       LOGD("mime type = %s", mime_type);
+                       if (!strncmp(mime_type, "image/", strlen("image/"))) {
+                               app_data->mime_type = MIME_TYPE_IMAGE;
+                               app_data->app_type = APP_TYPE_DRAWING;
+                               launch_drawing_app((void *)app_data);
+                               if (mime_type)
+                                       free(mime_type);
+                               goto ACTIVATE;
+                       } else if(!strncmp(mime_type, "audio/", strlen("audio/"))) {
+                               app_data->mime_type = MIME_TYPE_AUDIO;
+                               app_data->app_type = APP_TYPE_STT;
+                               _stt_clicked_cb((void *)app_data, NULL, NULL);
+                               if (mime_type)
+                                       free(mime_type);
+                               goto ACTIVATE;
+                       } else if(!strncmp(mime_type, "text/", strlen("text/"))) {
+                               app_data->mime_type = MIME_TYPE_TEXT;
+                       }
+               }
+       }
+       if (mime_type)
+               free(mime_type);
+
+       if (_TV || _MOBILE) {
+               app_data->app_type = APP_TYPE_KEYBOARD;
+               input_keyboard_init(service);
+               _keyboard_clicked_cb((void *)app_data, NULL, NULL);
+               goto ACTIVATE;
+       }
 
        ret = app_control_is_extra_data_array(service, APP_CONTROL_DATA_INPUT_TYPE, &is_extra_data_array);
        if ( is_extra_data_array == true) {
@@ -943,6 +1184,12 @@ void _app_service(app_control_h service, void* user_data)
 
        input_keyboard_init(service);
 
+       input_smartreply_init(service);
+       input_smartreply_set_notify(_input_smartreply_notify_cb, NULL);
+
+       if (input_smartreply_is_enabled())
+               input_smartreply_get_reply_async();
+
        input_template_init(service);
        input_template_set_notify(_input_template_notify_cb,  NULL);
 
@@ -983,6 +1230,7 @@ void _app_terminate(void* user_data)
                free(app_data->shared_res_path);
 
        input_keyboard_deinit();
+       input_smartreply_deinit();
 
        input_template_unset_notify();
        input_template_deinit();