[els_webview]send mouse_down and mouse_up using smart callback's event value
[framework/uifw/elementary.git] / src / lib / els_webview.c
1 /*
2  *
3  * vim:ts=8:sw=3:sts=8:noexpandtab:cino=>5n-3f0^-2{2
4  */
5 #include <Elementary.h>
6 #include "elm_priv.h"
7
8 #ifdef ELM_EWEBKIT
9 #include <EWebKit.h>
10 #include <cairo.h>
11
12 #define SMART_NAME "els_webview"
13 #define API_ENTRY Smart_Data *sd; sd = evas_object_smart_data_get(obj); if ((!obj) || (!sd) || (evas_object_type_get(obj) && strcmp(evas_object_type_get(obj), SMART_NAME)))
14 #define INTERNAL_ENTRY Smart_Data *sd; sd = evas_object_smart_data_get(obj); if (!sd) return;
15 #define EWK_VIEW_PRIV_GET_OR_RETURN(sd, ptr, ...)                   \
16    Ewk_View_Private_Data* ptr = ((Ewk_View_Smart_Data*)sd)->_priv; \
17    if (!ptr) \
18    {                                                     \
19       ERR("no private data for object %p (%s)",               \
20          ((Ewk_View_Smart_Data*)sd)->self,                           \
21          evas_object_type_get(((Ewk_View_Smart_Data*)sd)->self));    \
22       return __VA_ARGS__;                                         \
23    }
24
25 #define EWEBKIT_PATH "/usr/lib/libewebkit.so"
26 #define CAIRO_PATH "/usr/lib/libcairo.so.2"
27
28 #define MINIMAP_WIDTH 120
29 #define MINIMAP_HEIGHT 200
30
31 #define USE_MAX_TUC_20MB
32
33 #ifdef USE_MAX_TUC_20MB
34 #define MAX_TUC 1024*1024*20
35 #else
36 #define MAX_TUC 1024*1024*10
37 #endif
38 #define MAX_URI 512
39 #define MOBILE_DEFAULT_ZOOM_RATIO 1.5f
40
41 #define WEBVIEW_EDJ "/usr/share/edje/ewebview.edj"
42 #define WEBKIT_EDJ "/usr/share/edje/webkit.edj"
43 #define WEBVIEW_THEME_EDJ "/usr/share/edje/ewebview-theme.edj"
44
45 #define DEFAULT_LAYOUT_WIDTH 1024
46 #define MIN_ZOOM_RATIO 0.09f
47 #define MAX_ZOOM_RATIO 4.0f
48 #define ZOOM_OUT_BOUNCING 0.85f
49 #define ZOOM_IN_BOUNCING 1.25f
50 #define BOUNCING_DISTANCE 400
51
52 //                      "<!--<body bgcolor=#4c4c4c text=white text-align=left>-->"
53 #define NOT_FOUND_PAGE_HEADER "<html>" \
54                         "<head><title>Page Not Found</title></head>" \
55                         "<body bgcolor=white text=black text-align=left>" \
56                         "<center>" \
57                         "<table>" \
58                         "<tr><td><h1>Page Not Found<br/></td></tr>" \
59                         "<meta name='viewport' content='width=device-width, initial-scale=1.0, user-scalable=no'>" \
60                         "<tr><td>" \
61                         "<script type='text/javascript'>"\
62                         "var s = "
63
64 #define NOT_FOUND_PAGE_FOOTER ";" \
65                         "var failingUrl = s.substring(s.indexOf(\"?\"\)+1, s.lastIndexOf(\"?\"\));" \
66                         "document.write(\"<p><tr><td><h2>URL: \" + unescape(failingUrl) + \"</h2></td></tr>\");" \
67                         "var errorDesc = s.substring(s.lastIndexOf(\"?\")+1, s.length);" \
68                         "document.write(\"<tr><td><h2>Error: \" + unescape(errorDesc) + \"</h2></td></tr>\");" \
69                         "document.write(\"<tr><td><h3>Google: <form method=\'get\' action=\'http://www.google.com/custom\'><input type=text name=\'q\' size=15 maxlength=100 value=\'\"+ unescape(failingUrl)+\"\'> <input type=submit name=\'sa\' value=Search></form></h3></td></tr>\");" \
70                         "</script>" \
71                         "</td></tr>" \
72                         "</table>" \
73                         "</h1>" \
74                         "</body>" \
75                         "</html>"
76
77 #define NEED_TO_REMOVE
78 //#define DBG printf
79 //#define ERR printf
80
81 typedef struct _Smart_Data Smart_Data;
82
83 struct _Smart_Data {
84      Ewk_View_Smart_Data base; //default data
85
86      Evas_Object* widget;
87      Ecore_Job *move_calc_job;
88      Ecore_Job *resize_calc_job;
89      Eina_Hash* mime_func_hash;
90      int locked_dx;
91      int locked_dy;
92      unsigned char bounce_horiz : 1;
93      unsigned char bounce_vert : 1;
94      unsigned char events_feed : 1;
95      unsigned char auto_fitting : 1;
96      unsigned char mouse_clicked : 1;
97
98      /* ewk functions */
99      void (*ewk_view_theme_set)(Evas_Object *, const char *);
100      Evas_Object *(*ewk_view_frame_main_get)(const Evas_Object *);
101      Eina_Bool (*ewk_view_uri_set)(Evas_Object *, const char *);
102      float (*ewk_view_zoom_get)(const Evas_Object *);
103      const char * (*ewk_view_uri_get)(const Evas_Object *o);
104      Eina_Bool (*ewk_view_zoom_set)(Evas_Object *, float, Evas_Coord, Evas_Coord);
105      Eina_Bool (*ewk_view_zoom_weak_set)(Evas_Object *, float, Evas_Coord, Evas_Coord);
106      Eina_Bool (*ewk_view_zoom_text_only_set)(Evas_Object *, Eina_Bool);
107      Eina_Bool (*ewk_view_zoom_cairo_scaling_get)(const Evas_Object *);
108      Eina_Bool (*ewk_view_zoom_cairo_scaling_set)(Evas_Object *, Eina_Bool);
109      void (*ewk_view_viewport_get)(Evas_Object *, int *, int *, float *, float *, float *, Eina_Bool *);
110      Eina_Bool (*ewk_view_zoom_range_set)(Evas_Object *, float, float);
111      void (*ewk_view_user_scalable_set)(Evas_Object *, Eina_Bool);
112      Eina_Bool (*ewk_view_pre_render_region)(Evas_Object *, Evas_Coord, Evas_Coord, Evas_Coord, Evas_Coord, float);
113      void (*ewk_view_pre_render_cancel)(Evas_Object *);
114      Eina_Bool (*ewk_view_enable_render)(const Evas_Object *);
115      Eina_Bool (*ewk_view_disable_render)(const Evas_Object *);
116      void (*ewk_view_javascript_suspend)(Evas_Object *);
117      void (*ewk_view_javascript_resume)(Evas_Object *);
118      void (*ewk_view_fixed_layout_size_set)(Evas_Object *, Evas_Coord, Evas_Coord);
119      Eina_Bool (*ewk_view_setting_enable_plugins_get)(const Evas_Object *);
120      void (*ewk_view_pause_and_or_hide_plugins)(Evas_Object *, Eina_Bool, Eina_Bool);
121      Eina_Bool (*ewk_view_suspend_request)(Evas_Object *);
122      Eina_Bool (*ewk_view_resume_request)(Evas_Object *);
123      Eina_Bool (*ewk_view_select_none)(Evas_Object *);
124      Eina_Bool (*ewk_view_get_smart_zoom_rect)(Evas_Object *, int, int, const Evas_Event_Mouse_Up *, Eina_Rectangle *);
125      Eina_Bool (*ewk_view_paint_contents)(Ewk_View_Private_Data *, cairo_t *, const Eina_Rectangle *);
126      Eina_Bool (*ewk_view_stop)(Evas_Object *);
127      Ewk_Tile_Unused_Cache *(*ewk_view_tiled_unused_cache_get)(const Evas_Object *);
128      void (*ewk_view_tiled_unused_cache_set)(Evas_Object *, Ewk_Tile_Unused_Cache *);
129      void (*ewk_tile_unused_cache_max_set)(Ewk_Tile_Unused_Cache *, size_t);
130      size_t (*ewk_tile_unused_cache_max_get)(const Ewk_Tile_Unused_Cache *);
131      size_t (*ewk_tile_unused_cache_used_get)(const Ewk_Tile_Unused_Cache *);
132      size_t (*ewk_tile_unused_cache_flush)(Ewk_Tile_Unused_Cache *, size_t);
133      void (*ewk_tile_unused_cache_auto_flush)(Ewk_Tile_Unused_Cache *);
134      char * (*ewk_page_check_point_for_keyboard)(Evas_Object *, int, int, Eina_Bool *);
135      Eina_Bool (*ewk_page_check_point)(Evas_Object *, int, int, Evas_Event_Mouse_Down *, Eina_Bool *, Eina_Bool *, char **, char **, char **);
136      char ** (*ewk_page_dropdown_get_options)(Evas_Object *, int, int, int *, int *);
137      Eina_Bool (*ewk_page_dropdown_set_current_index)(Evas_Object *, int, int, int);
138      Eina_Bool (*ewk_frame_contents_size_get)(const Evas_Object *, Evas_Coord *, Evas_Coord *);
139      Ewk_Hit_Test * (*ewk_frame_hit_test_new)(const Evas_Object *, int, int);
140      Eina_Bool (*ewk_frame_feed_mouse_down)(Evas_Object *, const Evas_Event_Mouse_Down *);
141      Eina_Bool (*ewk_frame_feed_mouse_up)(Evas_Object *, const Evas_Event_Mouse_Up *);
142      Eina_Bool (*ewk_frame_visible_content_geometry_get)(const Evas_Object *, Eina_Bool, int *, int *, int *, int *);
143      Eina_Bool (*ewk_frame_scroll_pos_get)(const Evas_Object *, int *, int *);
144      void (*ewk_frame_hit_test_free)(Ewk_Hit_Test *);
145      Eina_Bool (*ewk_frame_contents_set)(Evas_Object *, const char *, size_t, const char *, const char *, const char *);
146      Eina_Bool (*ewk_frame_select_closest_word)(Evas_Object *, int, int, int *, int *, int *, int *, int *, int *);
147      Eina_Bool (*ewk_frame_selection_handlers_get)(Evas_Object *, int *, int *, int *, int *, int *, int *);
148      Eina_Bool (*ewk_frame_selection_left_set)(Evas_Object *, int, int, int *, int *, int *);
149      Eina_Bool (*ewk_frame_selection_right_set)(Evas_Object *, int, int, int *, int *, int *);
150      Eina_Bool (*ewk_frame_feed_focus_in)(Evas_Object *);
151      Eina_Bool (*ewk_frame_scroll_add)(Evas_Object *, int, int);
152      unsigned int (*ewk_view_imh_get)(Evas_Object *);
153      Ecore_IMF_Context* (*ewk_view_core_imContext_get)(Evas_Object *);
154      void (*ewk_set_show_geolocation_permission_dialog_callback)(ewk_show_geolocation_permission_dialog_callback);
155      void (*ewk_set_geolocation_sharing_allowed)(void *, Eina_Bool);
156
157      /* cairo functions */
158      cairo_t * (*cairo_create)(cairo_surface_t *);
159      void (*cairo_destroy)(cairo_t *);
160      void (*cairo_paint)(cairo_t *);
161      void (*cairo_stroke)(cairo_t *cr); 
162      void (*cairo_scale)(cairo_t *, double, double);
163      void (*cairo_rectangle)(cairo_t *, double, double, double, double);
164      void (*cairo_set_source_rgb)(cairo_t *, double, double, double);
165      cairo_status_t (*cairo_surface_status)(cairo_surface_t *);
166      void (*cairo_surface_destroy)(cairo_surface_t *);
167      void (*cairo_set_line_width)(cairo_t *, double);
168      void (*cairo_set_source_surface)(cairo_t *, cairo_surface_t *, double, double);
169      cairo_status_t (*cairo_surface_write_to_png)(cairo_surface_t *, const char *);
170      cairo_surface_t * (*cairo_image_surface_create)(cairo_format_t, int, int);
171      void (*cairo_set_antialias)(cairo_t *, cairo_antialias_t);
172      cairo_surface_t * (*cairo_image_surface_create_for_data)(unsigned char *, cairo_format_t, int, int, int);
173
174      /* add user data */
175      struct {
176           Evas_Object* eo;
177           Evas_Object* content;
178           int cw, ch;
179      } minimap;
180
181      struct {
182           char** options;
183           int option_cnt;
184           int option_idx;
185      } dropdown;
186
187      struct {
188           Evas_Point basis;    // basis point of zoom
189           int finger_distance; // distance between two finger
190           int zooming_level;
191           float zooming_rate;
192           float zoom_rate_at_start;
193           float zoom_rate_to_set;
194           Evas_Point scroll_at_start;
195           Evas_Point scroll_to_set;
196           float init_zoom_rate;
197           float min_zoom_rate; //content based minimum
198           float max_zoom_rate;
199           Eina_Bool scalable;
200      } zoom;
201
202      struct {
203           int w, h;
204      } content;
205
206      struct {
207           int default_w;
208           int w, h;
209      } layout;
210
211      Ecore_Animator* smart_zoom_animator;
212
213      Evas_Point pan_s;
214      Evas_Event_Mouse_Down mouse_down_copy;
215      Evas_Event_Mouse_Up mouse_up_copy;
216
217      cairo_surface_t* thumbnail;
218      float current_zoom_level;
219
220      Eina_Bool tiled;
221      Eina_Bool on_panning;
222      Eina_Bool on_zooming;
223      Eina_Bool is_mobile_page;
224
225      Eina_Bool use_text_selection;
226      Eina_Bool text_selection_on;
227      struct {
228           Evas_Coord_Rectangle front;
229           Evas_Coord_Rectangle back;
230           Evas_Point front_handle;
231           Evas_Point back_handle;
232           Eina_Bool front_handle_moving;
233           Eina_Bool back_handle_moving;
234      } text_selection;
235      void* touch_obj;
236
237      Ecore_Idler *flush_and_pre_render_idler;
238      Eina_Bool use_zoom_bouncing;
239 };
240
241 /* local subsystem functions */
242 static void      _resize_calc_job(void *data);
243 static void      _move_calc_job(void *data);
244 static void      _smart_show(Evas_Object* obj);
245 static void      _smart_hide(Evas_Object* obj);
246 static void      _smart_resize(Evas_Object* obj, Evas_Coord w, Evas_Coord h);
247 static void      _smart_move(Evas_Object* obj, Evas_Coord x, Evas_Coord y);
248 #ifdef DEBUG
249 static void      _smart_calculate(Evas_Object* obj);
250 #endif
251 static Eina_Bool _smart_mouse_down(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Down* ev);
252 static Eina_Bool _smart_mouse_up(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Up* ev);
253 static Eina_Bool _smart_mouse_move(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Move* ev);
254 static void      _smart_add_console_message(Ewk_View_Smart_Data *esd, const char *message, unsigned int lineNumber, const char *sourceID);
255 static void      _smart_run_javascript_alert(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message);
256 static Eina_Bool _smart_run_javascript_confirm(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message);
257 static Eina_Bool _smart_run_javascript_prompt(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message, const char *defaultValue, char **value);
258 static Eina_Bool _smart_should_interrupt_javascript(Ewk_View_Smart_Data *esd);
259 static Eina_Bool _smart_run_open_panel(Ewk_View_Smart_Data *esd, Evas_Object *frame, Eina_Bool allows_multiple_files, const Eina_List *suggested_filenames, Eina_List **selected_filenames);
260 static Eina_Bool _smart_navigation_policy_decision(Ewk_View_Smart_Data *esd, Ewk_Frame_Resource_Request *request);
261 static void      _view_on_mouse_down(void* data, Evas* e, Evas_Object* o, void* event_info);
262 static void      _view_on_mouse_up(void* data, Evas* e, Evas_Object* o, void* event_info);
263 static void      _smart_load_started(void* data, Evas_Object* webview, void* error);
264 static void      _smart_load_finished(void* data, Evas_Object* webview, void* arg);
265 static void      _smart_load_error(void* data, Evas_Object* webview, void* arg);
266 static void      _smart_viewport_changed(void* data, Evas_Object* webview, void* arg);
267 static void      _smart_input_method_changed(void* data, Evas_Object* webview, void* arg);
268 static void      _smart_page_layout_info_set(Smart_Data *sd, float init_zoom_rate, float min_zoom_rate, float max_zoom_rate, Eina_Bool scalable);
269 static void      _smart_contents_size_changed(void* data, Evas_Object* frame, void* arg);
270 static void      _smart_load_nonemptylayout_finished(void* data, Evas_Object* frame, void* arg);
271 static void      _smart_cb_view_created(void* data, Evas_Object* webview, void* arg);
272 static void      _smart_add(Evas_Object* obj);
273 static void      _smart_del(Evas_Object* o);
274 static void      _directional_pre_render(Evas_Object* webview, int dx, int dy);
275 static void      _smart_cb_mouse_down(void* data, Evas_Object* webview, void* ev);
276 static void      _smart_cb_mouse_up(void* data, Evas_Object* webview, void* ev);
277 static void      _smart_cb_mouse_tap(void* data, Evas_Object* webview, void* ev);
278 static void      _smart_cb_pan_start(void* data, Evas_Object* webview, void* ev);
279 static void      _smart_cb_pan_by(void* data, Evas_Object* webview, void* ev);
280 static void      _smart_cb_pan_stop(void* data, Evas_Object* webview, void* ev);
281 static void      _smart_cb_select_closest_word(void* data, Evas_Object* webview, void* ev);
282 static void      _smart_cb_unselect_closest_word(void* data, Evas_Object* webview, void* ev);
283 static void      _suspend_all(Smart_Data *sd, Eina_Bool hidePlugin);
284 static void      _resume_all(Smart_Data *sd, Eina_Bool hidePlugin);
285 static void      _zoom_start(Smart_Data* sd, int centerX, int centerY, int distance);
286 static void      _zoom_move(Smart_Data* sd, int centerX, int centerY, int distance);
287 static void      _zoom_stop(Smart_Data* sd);
288 static void      _adjust_to_contents_boundary(Evas_Object* webview, int* to_x, int* to_y, int from_x, int from_y, float new_zoom_rate);
289 static int       _smart_zoom_animator(void* data);
290 static void      _smart_cb_pinch_zoom_start(void* data, Evas_Object* webview, void* event_info);
291 static void      _smart_cb_pinch_zoom_move(void* data, Evas_Object* webview, void* event_info);
292 static void      _smart_cb_pinch_zoom_stop(void* data, Evas_Object* webview, void* event_info);
293 static void      _smart_cb_vertical_zoom_start(void* data, Evas_Object* webview, void* event_info);
294 static void      _smart_cb_vertical_zoom_move(void* data, Evas_Object* webview, void* event_info);
295 static void      _smart_cb_vertical_zoom_stop(void* data, Evas_Object* webview, void* event_info);
296 static void      _smart_cb_smart_zoom(void* data, Evas_Object* webview, void* event_info);
297 static void      _zoom_to_rect(Smart_Data *sd, int x, int y);
298 static void      _text_selection_init(Evas_Object* parent);
299 static void      _text_selection_show(void);
300 static void      _text_selection_hide(Smart_Data *sd);
301 static void      _text_selection_set_front_info(Smart_Data *sd, int x, int y, int height);
302 static void      _text_selection_set_back_info(Smart_Data *sd, int x, int y, int height);
303 static Eina_Bool _text_selection_handle_pressed(Smart_Data *sd, int x, int y);
304 static void      _text_selection_update_position(Smart_Data *sd, int x, int y);
305 static void      _text_selection_move_by(Smart_Data *sd, int dx, int dy);
306 static void      _minimap_update_detail(Evas_Object* minimap, Smart_Data *sd, cairo_surface_t* src, int srcW, int srcH, Eina_Rectangle* visibleRect);
307 static void      _minimap_update(Evas_Object* minimap, Smart_Data *sd, cairo_surface_t* src, int minimapW, int minimapH);
308 static cairo_surface_t* _image_clone_get(Smart_Data *sd, int* minimap_w, int* minimap_h);
309 static void      _unzoom_position(Evas_Object* webview, int x, int y, int* ux, int* uy);
310 static void      _coords_evas_to_ewk(Evas_Object* webview, int x, int y, int* ux, int* uy);
311 static void      _coords_ewk_to_evas(Evas_Object* webview, int x, int y, int* ux, int* uy);
312 static void      _update_min_zoom_rate(Evas_Object *obj);
313 static void      _geolocation_permission_callback(void *geolocation_obj, const char* url);
314
315 /* local subsystem globals */
316 static Evas_Smart *_smart = NULL;
317 static Ewk_View_Smart_Class _parent_sc = EWK_VIEW_SMART_CLASS_INIT_NULL;
318
319 /* ewk functions */
320 static void *ewk_handle;
321 static void *cairo_handle;
322
323 static Ewk_Tile_Unused_Cache *ewk_tile_cache = NULL;
324 static ewk_tile_cache_ref_count = 0;
325
326 static Evas_Object *obj = NULL;
327
328 /* externally accessible functions */
329 Evas_Object*
330 _elm_smart_webview_add(Evas *evas, Eina_Bool tiled)
331 {
332    Evas_Object* webview;
333    int (*ewk_init)(void) = NULL;
334    void (*ewk_dnet_open)(void) = NULL;
335    Eina_Bool (*ewk_view_single_smart_set)(Ewk_View_Smart_Class *) = NULL;
336    Eina_Bool (*ewk_view_tiled_smart_set)(Ewk_View_Smart_Class *) = NULL;
337
338    if (!_smart)
339      {
340         ewk_handle = dlopen(EWEBKIT_PATH, RTLD_LAZY);
341         if (ewk_handle == NULL)
342           {
343              ERR("could not initialize ewk \n");
344              return NULL;
345           }
346         cairo_handle = dlopen(CAIRO_PATH, RTLD_LAZY);
347         if (cairo_handle == NULL)
348           {
349              ERR("could not initialize cairo \n");
350              return NULL;
351           }
352
353         // init ewk
354         if (!ewk_init)
355           ewk_init = (int (*)())dlsym(ewk_handle, "ewk_init");
356         ewk_init();
357
358         if (!ewk_dnet_open)
359           ewk_dnet_open = (void (*)())dlsym(ewk_handle, "ewk_dnet_open");
360         ewk_dnet_open();
361
362         /* create subclass */
363         static Ewk_View_Smart_Class _api = EWK_VIEW_SMART_CLASS_INIT_NAME_VERSION(SMART_NAME);
364
365         if (tiled)
366           {
367              if (!ewk_view_tiled_smart_set)
368                ewk_view_tiled_smart_set = (Eina_Bool (*)(Ewk_View_Smart_Class *))dlsym(ewk_handle, "ewk_view_tiled_smart_set");
369              ewk_view_tiled_smart_set(&_api);
370              if (EINA_UNLIKELY(!_parent_sc.sc.add))
371                ewk_view_tiled_smart_set(&_parent_sc);
372
373           } else {
374                if (!ewk_view_single_smart_set)
375                  ewk_view_single_smart_set = (Eina_Bool (*)(Ewk_View_Smart_Class *))dlsym(ewk_handle, "ewk_view_single_smart_set");
376                ewk_view_single_smart_set(&_api);
377                if (EINA_UNLIKELY(!_parent_sc.sc.add))
378                  ewk_view_single_smart_set(&_parent_sc);
379           }
380
381         _api.sc.add     = _smart_add;
382         _api.sc.del     = _smart_del;
383         _api.sc.show    = _smart_show;
384         _api.sc.hide    = _smart_hide;
385         _api.sc.resize  = _smart_resize;
386         _api.sc.move  = _smart_move;
387 #ifdef DEBUG
388         _api.sc.calculate = _smart_calculate;
389 #endif
390         _api.mouse_down = _smart_mouse_down;
391         _api.mouse_up   = _smart_mouse_up  ;
392         _api.mouse_move = _smart_mouse_move;
393
394         _api.add_console_message = _smart_add_console_message;
395         _api.run_javascript_alert = _smart_run_javascript_alert;
396         _api.run_javascript_confirm = _smart_run_javascript_confirm;
397         _api.run_javascript_prompt = _smart_run_javascript_prompt;
398         _api.should_interrupt_javascript = _smart_should_interrupt_javascript;
399         _api.run_open_panel = _smart_run_open_panel;
400         //_api.navigation_policy_decision = _smart_navigation_policy_decision;
401
402         _smart = evas_smart_class_new(&_api.sc);
403         elm_theme_overlay_add(NULL, WEBVIEW_THEME_EDJ);
404
405      }
406
407    if (!_smart)
408      {
409         ERR("could not create smart class\n");
410         return NULL;
411      }
412
413    webview = evas_object_smart_add(evas, _smart);
414    if (!webview)
415      {
416         ERR("could not create smart object for webview");
417         return NULL;
418      }
419    obj = webview;
420
421    // set tiled and unused cache 
422    Smart_Data* sd = evas_object_smart_data_get(webview);
423    if (sd)
424      {
425         sd->tiled = tiled;
426         if (sd->tiled)
427           {
428              if (ewk_tile_cache_ref_count == 0)
429                {
430                   if (!sd->ewk_view_tiled_unused_cache_get)
431                     sd->ewk_view_tiled_unused_cache_get = (Ewk_Tile_Unused_Cache *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_get");
432                   ewk_tile_cache = sd->ewk_view_tiled_unused_cache_get(webview);
433
434                   if (!sd->ewk_tile_unused_cache_max_set)
435                     sd->ewk_tile_unused_cache_max_set = (void (*)(Ewk_Tile_Unused_Cache *, size_t))dlsym(ewk_handle, "ewk_tile_unused_cache_max_set");
436                   sd->ewk_tile_unused_cache_max_set(ewk_tile_cache, MAX_TUC);
437                } else {
438                     if (!sd->ewk_view_tiled_unused_cache_set)
439                       sd->ewk_view_tiled_unused_cache_set = (void (*)(Evas_Object *, Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_set");
440                     sd->ewk_view_tiled_unused_cache_set(webview, ewk_tile_cache);
441                }
442              ++ewk_tile_cache_ref_count;
443              //size_t mem = ewk_tile_unused_cache_used_get(ewk_tile_cache);
444              //DBG("%s: Used cache: %d (%dkB)", __func__, mem, (mem/1024));
445           }
446      }
447
448    return webview;
449 }
450
451 void
452 _elm_smart_webview_events_feed_set(Evas_Object* obj, Eina_Bool feed)
453 {
454    API_ENTRY return;
455    sd->events_feed = feed;
456 }
457
458 Eina_Bool
459 _elm_smart_webview_events_feed_get(Evas_Object* obj)
460 {
461    API_ENTRY return EINA_FALSE;
462    return sd->events_feed;
463 }
464
465 void
466 _elm_smart_webview_auto_fitting_set(Evas_Object* obj, Eina_Bool enable)
467 {
468    API_ENTRY return;
469    sd->auto_fitting = enable;
470 }
471
472 Eina_Bool
473 _elm_smart_webview_auto_fitting_get(Evas_Object *obj)
474 {
475    API_ENTRY return EINA_FALSE;
476    return sd->auto_fitting;
477 }
478
479 Evas_Object *
480 _elm_smart_webview_minimap_get(Evas_Object* obj)
481 {
482    DBG("%s\n", __func__);
483    API_ENTRY return NULL;
484
485    if (sd->minimap.eo != NULL) return sd->minimap.eo;
486
487    sd->minimap.eo = edje_object_add(evas_object_evas_get(obj));
488    edje_object_file_set(sd->minimap.eo, WEBVIEW_EDJ, "minimap");
489
490    sd->minimap.content = evas_object_image_add(evas_object_evas_get(sd->minimap.eo));
491    evas_object_size_hint_align_set(sd->minimap.content, 0.5, 0.5);
492    evas_object_image_colorspace_set(sd->minimap.content, EVAS_COLORSPACE_ARGB8888);
493    evas_object_image_alpha_set(sd->minimap.content, EINA_FALSE);
494
495    Evas_Object* box = evas_object_box_add(evas_object_evas_get(sd->minimap.eo));
496    evas_object_box_append(box, sd->minimap.content);
497    evas_object_show(sd->minimap.content);
498    edje_object_part_swallow(sd->minimap.eo, "swallow.content", box);
499
500    return sd->minimap.eo;
501 }
502
503 void
504 _elm_smart_webview_uri_set(Evas_Object* obj, const char* uri)
505 {
506    API_ENTRY return;
507
508    char full_uri[MAX_URI] = "";
509    printf("<< uri [%s] >>\n", uri);
510
511    if (uri == NULL)
512      return;
513
514    // check uri string
515    int len = strlen(uri);
516    if (len)
517      {
518         if (strstr(uri, "://") == NULL) {
519              strncpy(full_uri, "http://", 7);
520              full_uri[7] = '\0';
521              len = (len >= (MAX_URI - 7)) ? (MAX_URI - 8) : len;
522              strncat(full_uri, uri, len);
523         } else {
524              len = (len >= MAX_URI) ? (MAX_URI - 1) : len;
525              strncpy(full_uri, uri, len);
526              full_uri[len] = '\0';
527         }
528
529         printf("<< full uri [%s] >>\n", full_uri);
530         if (!sd->ewk_view_uri_set)
531           sd->ewk_view_uri_set = (Eina_Bool (*)(Evas_Object *, const char *))dlsym(ewk_handle, "ewk_view_uri_set");
532         sd->ewk_view_uri_set(obj, full_uri);
533      }
534 }
535
536 void
537 _elm_smart_webview_widget_set(Evas_Object *obj, Evas_Object *wid)
538 {
539    API_ENTRY return;
540    sd->widget = wid;
541 }
542
543 void
544 _elm_smart_webview_bounce_allow_set(Evas_Object* obj, Eina_Bool horiz, Eina_Bool vert)
545 {
546    API_ENTRY return;
547    sd->bounce_horiz = horiz;
548    sd->bounce_vert = vert;
549 }
550
551 void
552 _elm_smart_webview_mime_callback_set(Evas_Object* obj, const char *mime, Elm_WebView_Mime_Cb func)
553 {
554    API_ENTRY return;
555    if (!sd->mime_func_hash)
556      sd->mime_func_hash = eina_hash_pointer_new(NULL);
557
558    if (!func)
559      eina_hash_del(sd->mime_func_hash, mime, func);
560    else
561      eina_hash_add(sd->mime_func_hash, mime, func);
562 }
563
564 void
565 _elm_smart_webview_default_layout_width_set(Evas_Object *obj, int width)
566 {
567    API_ENTRY return;
568    sd->layout.default_w = width;
569 }
570
571 int
572 _flush_and_pre_render(void *data)
573 {
574    Evas_Object *obj = (Evas_Object *)data;
575    API_ENTRY return ECORE_CALLBACK_CANCEL;
576
577    if (!sd->ewk_view_tiled_unused_cache_get)
578      sd->ewk_view_tiled_unused_cache_get = (Ewk_Tile_Unused_Cache *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_get");
579    if (!sd->ewk_tile_unused_cache_used_get)
580      sd->ewk_tile_unused_cache_used_get = (size_t (*)(const Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_used_get");
581    if (!sd->ewk_tile_unused_cache_flush)
582      sd->ewk_tile_unused_cache_flush = (size_t (*)(Ewk_Tile_Unused_Cache *, size_t))dlsym(ewk_handle, "ewk_tile_unused_cache_flush");
583
584    Ewk_Tile_Unused_Cache *tuc = sd->ewk_view_tiled_unused_cache_get(obj);
585    sd->ewk_tile_unused_cache_flush(tuc, sd->ewk_tile_unused_cache_used_get(tuc));
586    _directional_pre_render(obj, 0, 0);
587
588    sd->flush_and_pre_render_idler = NULL;
589
590    return ECORE_CALLBACK_CANCEL;
591 }
592
593 /* local subsystem functions */
594 static void
595 _smart_show(Evas_Object* obj)
596 {
597    DBG("%s\n", __func__);
598    INTERNAL_ENTRY;
599
600    _elm_smart_touch_start(sd->touch_obj);
601    _parent_sc.sc.show(obj);
602 }
603
604 static void
605 _smart_hide(Evas_Object* obj)
606 {
607    DBG("%s\n", __func__);
608    INTERNAL_ENTRY;
609
610    _elm_smart_touch_stop(sd->touch_obj);
611    _parent_sc.sc.hide(obj);
612 }
613
614 static void
615 _smart_resize(Evas_Object* obj, Evas_Coord w, Evas_Coord h)
616 {
617    DBG("%s\n", __func__);
618    INTERNAL_ENTRY;
619
620    Evas_Coord ow, oh;
621    evas_object_geometry_get(obj, NULL, NULL, &ow, &oh);
622    if ((ow == w) && (oh == h)) return;
623    if (sd->resize_calc_job) ecore_job_del(sd->resize_calc_job);
624    sd->resize_calc_job = ecore_job_add(_resize_calc_job, obj);
625 }
626
627 static void
628 _resize_calc_job(void *data)
629 {
630    Evas_Object *obj = data;
631    INTERNAL_ENTRY;
632
633    int object_w, object_h;
634    evas_object_geometry_get(obj, NULL, NULL, &object_w, &object_h);
635    object_w = (object_w % 10) ? (object_w / 10 * 10 + 10) : object_w;
636
637    if (sd->is_mobile_page)
638      {
639         int old_layout_w = sd->layout.w;
640         sd->layout.w = object_w / sd->zoom.init_zoom_rate;
641         sd->layout.h = object_h / sd->zoom.init_zoom_rate;
642         if (old_layout_w != sd->layout.w)
643           {
644              if (!sd->ewk_view_fixed_layout_size_set)
645                sd->ewk_view_fixed_layout_size_set = (void (*)(Evas_Object *, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_fixed_layout_size_set");
646              sd->ewk_view_fixed_layout_size_set(obj, sd->layout.w, sd->layout.h);
647           }
648      }
649    else
650      {
651         if (!sd->ewk_view_zoom_get)
652           sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
653         if (!sd->ewk_view_zoom_set)
654           sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
655
656         _update_min_zoom_rate(obj);
657
658         // set zoom
659         if (sd->ewk_view_zoom_get(obj) < sd->zoom.min_zoom_rate)
660           sd->ewk_view_zoom_set(obj, sd->zoom.min_zoom_rate, 0, 0);
661      }
662
663    // call preRender by timer, because we can not get the correct visible_content of frame
664    // when call it directly.
665    if (!sd->ewk_view_uri_get)
666      sd->ewk_view_uri_get = (const char * (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_uri_get");
667    const char *url = sd->ewk_view_uri_get(obj);
668    if (url && strcmp(url, "") != 0 && sd->flush_and_pre_render_idler == NULL)
669      {
670         sd->flush_and_pre_render_idler = ecore_idler_add(_flush_and_pre_render, obj);
671      }
672
673    sd->resize_calc_job = NULL;
674    _parent_sc.sc.resize(obj, object_w, object_h);
675 }
676
677 static void
678 _move_calc_job(void *data)
679 {
680    Evas_Object *obj = data;
681    INTERNAL_ENTRY;
682    int x, y;
683    evas_object_geometry_get(obj, &x, &y, NULL, NULL);
684    sd->move_calc_job = NULL;
685    _parent_sc.sc.move(obj, x, y);
686 }
687
688 static void
689 _smart_move(Evas_Object* obj, Evas_Coord x, Evas_Coord y)
690 {
691    DBG("%s\n", __func__);
692    INTERNAL_ENTRY;
693
694    if (sd->move_calc_job) ecore_job_del(sd->move_calc_job);
695    sd->move_calc_job = ecore_job_add(_move_calc_job, obj);
696 }
697
698 #ifdef DEBUG
699 static void
700 _smart_calculate(Evas_Object* obj)
701 {
702    _parent_sc.sc.calculate(obj);
703 }
704 #endif
705
706 static Eina_Bool
707 _smart_mouse_down(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Down* ev)
708 {
709    DBG("[NATIVE]%s is called\n", __func__);
710    Smart_Data *sd = (Smart_Data *)esd;
711    sd->mouse_down_copy = *ev;
712
713    if (sd->events_feed)
714      {
715         _suspend_all(sd, EINA_FALSE);
716         sd->mouse_clicked = EINA_TRUE;
717         return _parent_sc.mouse_down(esd, ev);
718      }
719    else return EINA_TRUE;
720 }
721
722 static Eina_Bool
723 _smart_mouse_up(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Up* ev)
724 {
725    DBG("[NATIVE]%s is called\n", __func__);
726    Smart_Data *sd = (Smart_Data *)esd;
727    sd->mouse_up_copy = *ev;
728
729    if (sd->events_feed)
730      {
731         _resume_all(sd, EINA_FALSE);
732         //check if user hold touch
733         if (ev && (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD))
734           {
735              return EINA_TRUE;
736           }
737
738         Eina_Bool ret = _parent_sc.mouse_up(esd, ev);
739         sd->mouse_clicked = EINA_FALSE;
740         return ret;
741      }
742    else
743      return EINA_TRUE;
744 }
745
746 static Eina_Bool
747 _smart_mouse_move(Ewk_View_Smart_Data *esd, const Evas_Event_Mouse_Move* ev)
748 {
749    Smart_Data *sd = (Smart_Data *)esd;
750    if (sd->events_feed) _parent_sc.mouse_move(esd, ev);
751    else return EINA_TRUE;
752 }
753
754 static void
755 _smart_add_console_message(Ewk_View_Smart_Data *esd, const char *message, unsigned int lineNumber, const char *sourceID)
756 {
757    //TODO
758 }
759
760 static void
761 _smart_run_javascript_alert(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message)
762 {
763    Evas_Object *popup;
764    popup = elm_popup_add(esd->self);
765    evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
766    elm_popup_desc_set(popup, message);
767    elm_popup_buttons_add(popup, 1, "Ok", ELM_POPUP_RESPONSE_OK, NULL);
768    evas_object_show(popup);
769 }
770
771 static Eina_Bool
772 _smart_run_javascript_confirm(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message)
773 {
774    Evas_Object *popup;
775    popup = elm_popup_add(esd->self);
776    evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
777    elm_popup_desc_set(popup, message);
778    elm_popup_buttons_add(popup, 2, "Ok", ELM_POPUP_RESPONSE_OK, "Cancel", ELM_POPUP_RESPONSE_CANCEL, NULL);
779
780    int ret = elm_popup_run(popup);
781    evas_object_del(popup);
782    switch (ret)
783      {
784       case ELM_POPUP_RESPONSE_OK:
785          return EINA_TRUE;
786       case ELM_POPUP_RESPONSE_CANCEL:
787          return EINA_FALSE;
788       default:
789          elm_exit();
790      }
791    return EINA_FALSE;
792 }
793
794 static Eina_Bool
795 _smart_run_javascript_prompt(Ewk_View_Smart_Data *esd, Evas_Object *frame, const char *message, const char *defaultValue, char **value)
796 {
797    //FIXME: it's not work
798    Evas_Object *popup;
799    Evas_Object *box, *entry, *label;
800
801    popup = elm_popup_add(esd->self);
802    elm_object_style_set(popup, "customstyle");
803    evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
804    elm_popup_buttons_add(popup, 2, "Ok", ELM_POPUP_RESPONSE_OK, "Cancel", ELM_POPUP_RESPONSE_CANCEL, NULL);
805
806    box = elm_box_add(popup);
807    evas_object_size_hint_weight_set(box, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
808    evas_object_size_hint_fill_set(box, EVAS_HINT_FILL, EVAS_HINT_FILL);
809    evas_object_show(box);
810
811    label = elm_label_add(box);
812    evas_object_size_hint_weight_set(label, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
813    elm_label_label_set(label, message);
814    elm_box_pack_start(box, label);
815    evas_object_show(label);
816
817    entry = elm_entry_add(box);
818    evas_object_size_hint_weight_set(entry, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
819    elm_entry_single_line_set(entry, EINA_TRUE);
820    elm_entry_entry_set(entry, defaultValue);
821    elm_box_pack_end(box, entry);
822    evas_object_show(entry);
823
824    int ret = elm_popup_run(popup);
825    *value = strdup("temp");
826    evas_object_del(popup);
827
828    return EINA_FALSE;
829 }
830
831 static Eina_Bool
832 _smart_should_interrupt_javascript(Ewk_View_Smart_Data *esd)
833 {
834    //TODO
835    return EINA_FALSE;
836 }
837
838 static Eina_Bool 
839 _smart_run_open_panel(Ewk_View_Smart_Data *esd, Evas_Object *frame, Eina_Bool allows_multiple_files, const Eina_List *suggested_filenames, Eina_List **selected_filenames)
840 {
841    //TODO
842    return EINA_FALSE;
843 }
844
845 static Eina_Bool
846 _smart_navigation_policy_decision(Ewk_View_Smart_Data *esd, Ewk_Frame_Resource_Request *request)
847 {
848    char *protocol_hack;
849    Smart_Data *sd = (Smart_Data*)esd;
850    if (!sd->mime_func_hash)
851      return EINA_FALSE;
852
853    protocol_hack = strstr(request->url, ":");
854    *protocol_hack = '\0';
855    Elm_WebView_Mime_Cb func = (Elm_WebView_Mime_Cb) eina_hash_find(sd->mime_func_hash, request->url);
856    *protocol_hack = ':';
857
858    if (!func)
859      {
860         if (strncmp(request->url, "http", 4) == 0
861               || strncmp(request->url, "https", 5) == 0
862               || strncmp(request->url, "file", 4) == 0)
863           return EINA_TRUE;
864         return EINA_FALSE;
865      }
866    else
867      return func(esd->self);
868 }
869
870 #ifdef NEED_TO_REMOVE
871 // TODO: temporary mouse callback until the webkit engine can receive mouse events
872 static void
873 _view_on_mouse_down(void* data, Evas* e, Evas_Object* o, void* event_info)
874 {
875    Evas_Event_Mouse_Down* ev = (Evas_Event_Mouse_Down*)event_info;
876    Ewk_View_Smart_Data* sd = (Ewk_View_Smart_Data*)data;
877    EINA_SAFETY_ON_NULL_RETURN(sd->api);
878    EINA_SAFETY_ON_NULL_RETURN(sd->api->mouse_down);
879    sd->api->mouse_down(sd, ev);
880 }
881
882 static void
883 _view_on_mouse_up(void* data, Evas* e, Evas_Object* o, void* event_info)
884 {
885    Evas_Event_Mouse_Up* ev = (Evas_Event_Mouse_Up*)event_info;
886    Ewk_View_Smart_Data* sd = (Ewk_View_Smart_Data*)data;
887    EINA_SAFETY_ON_NULL_RETURN(sd->api);
888    EINA_SAFETY_ON_NULL_RETURN(sd->api->mouse_up);
889    sd->api->mouse_up(sd, ev);
890 }
891 #endif
892
893 static void
894 _smart_load_started(void* data, Evas_Object* webview, void* error)
895 {
896    DBG("%s is called\n", __func__);
897    Smart_Data *sd = (Smart_Data *)data;
898    if (!sd) return;
899
900    if (!sd->ewk_view_user_scalable_set)
901      sd->ewk_view_user_scalable_set = (void (*)(Evas_Object *, Eina_Bool))dlsym(ewk_handle, "ewk_view_user_scalable_set");
902
903    // set default layout and zoom level
904    sd->is_mobile_page = EINA_FALSE;
905    sd->layout.w = -1;
906    sd->layout.h = -1;
907    sd->zoom.init_zoom_rate = 1.0f;
908    sd->zoom.scalable = EINA_TRUE;
909    sd->ewk_view_user_scalable_set(webview, EINA_TRUE);
910 }
911
912 static void
913 _smart_load_finished(void* data, Evas_Object* webview, void* arg)
914 {
915    DBG("%s is called\n", __func__);
916    Smart_Data* sd = (Smart_Data *)data;
917    if (!sd) return;
918
919    // if error, call loadNotFoundPage
920    Ewk_Frame_Load_Error *error = (Ewk_Frame_Load_Error *) arg;
921    int errorCode = (error)? error->code: 0;
922    if ( errorCode != 0 && errorCode != -999 )
923      { // 0 ok, -999 request cancelled
924         DBG( "page not found:, [code: %d] [domain: %s] [description: %s] [failing_url: %s] \n",
925               error->code, error->domain, error->description, error->failing_url);
926         //ecore_job_add(loadNotFoundPage, (void *)this);
927         return;
928      }
929
930    if (sd->auto_fitting == EINA_TRUE)
931      {
932         if (!sd->ewk_view_zoom_set)
933           sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
934         sd->ewk_view_zoom_set(webview, sd->zoom.min_zoom_rate, 0, 0);
935      }
936
937    // update thumbnail and minimap
938    if (sd->thumbnail != NULL)
939      {
940         if (!sd->cairo_surface_destroy)
941           sd->cairo_surface_destroy = (void (*)(cairo_surface_t *))dlsym(cairo_handle, "cairo_surface_destroy");
942         sd->cairo_surface_destroy(sd->thumbnail);
943      }
944    sd->thumbnail = _image_clone_get(sd, &(sd->minimap.cw), &(sd->minimap.ch));
945
946    if (sd->tiled)
947      _directional_pre_render(sd->base.self, 0, 0);
948
949    if (sd->minimap.eo != NULL)
950      {
951    _minimap_update(sd->minimap.content, sd, sd->thumbnail,
952          sd->minimap.cw, sd->minimap.ch);
953      }
954 }
955
956 static void
957 _smart_load_error(void* data, Evas_Object* webview, void* arg)
958 {
959    DBG("%s is called\n", __func__);
960    Smart_Data* sd = (Smart_Data *)data;
961    char szBuffer[2048];
962    if (!sd) return;
963
964    // if error, call loadNotFoundPage
965    Ewk_Frame_Load_Error *error = (Ewk_Frame_Load_Error *) arg;
966    int errorCode = (error)? error->code: 0;
967    if ( errorCode != 0 && errorCode != -999 )
968      { // 0 ok, -999 request cancelled
969         //char szStrBuffer[1024];
970         //snprintf(szStrBuffer, 1024, "page not found:, [code: %d] [domain: %s] [description: %s] [failing_url: %s] \n",
971         //      error->code, error->domain, error->description, error->failing_url);
972         //DBG(szStrBuffer);
973
974         //ecore_job_add(loadNotFoundPage, (void *)this);
975         if (!sd->ewk_view_stop)
976           sd->ewk_view_stop = (Eina_Bool (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_stop");
977         sd->ewk_view_stop(webview);
978
979         if (!sd->ewk_view_frame_main_get)
980           sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
981
982         if (!sd->ewk_frame_contents_set)
983           sd->ewk_frame_contents_set = (Eina_Bool (*)(Evas_Object *, const char *, size_t, const char *, const char *, const char *))dlsym(ewk_handle, "ewk_frame_contents_set");
984
985         snprintf(szBuffer, 2048, NOT_FOUND_PAGE_HEADER "\"?%s?%s\"" NOT_FOUND_PAGE_FOOTER, error->failing_url, error->description);
986         //sd->ewk_frame_contents_set(sd->ewk_view_frame_main_get(webview), szStrBuffer, 0, NULL, NULL, NULL);
987         sd->ewk_frame_contents_set(error->frame, szBuffer, 0, NULL, NULL, NULL);
988         return;
989      }
990 }
991
992 static void
993 _smart_viewport_changed(void* data, Evas_Object* webview, void* arg)
994 {
995    DBG("%s is called\n", __func__);
996    Smart_Data* sd = (Smart_Data *)data;
997    if (!sd) return;
998
999    // check for mobile page
1000    int layout_w, layout_h;
1001    float init_zoom_rate, max_zoom_rate, min_zoom_rate;
1002    Eina_Bool scalable;
1003
1004    if (!sd->ewk_view_viewport_get)
1005      sd->ewk_view_viewport_get = (void (*)(Evas_Object *, int *, int *, float *, float *, float *, Eina_Bool *))dlsym(ewk_handle, "ewk_view_viewport_get");
1006    sd->ewk_view_viewport_get(webview, &layout_w, &layout_h,
1007          &init_zoom_rate, &max_zoom_rate, &min_zoom_rate, &scalable);
1008
1009    int object_w, object_h;
1010    evas_object_geometry_get(webview, NULL, NULL, &object_w, &object_h);
1011    object_w = (object_w % 10) ? (object_w / 10 * 10 + 10) : object_w;
1012
1013    // if layout width is bigger than object width, we regard current page to not the mobile page
1014    // for bbc.co.uk
1015    if (layout_w > object_w)
1016      {
1017         sd->layout.w = layout_w;
1018         return;
1019      }
1020
1021    // if there is no layout_w and url does not have mobile keyword, it is the desktop site.
1022    if (layout_w <= 0)
1023      {
1024         if (!sd->ewk_view_uri_get)
1025           sd->ewk_view_uri_get = (const char * (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_uri_get");
1026         const char *url = sd->ewk_view_uri_get(webview);
1027         if ((url && (strstr(url, "://m.") != NULL
1028                     || strstr(url, "://wap.") != NULL
1029                     || strstr(url, ".m.") != NULL
1030                     || strstr(url, "/mobile/i") != NULL))) // For www.bbc.co.uk/mobile/i site
1031           {
1032              min_zoom_rate = MIN_ZOOM_RATIO;
1033              max_zoom_rate = MAX_ZOOM_RATIO;
1034              scalable = 1;
1035           }
1036         else
1037           {
1038              return;
1039           }
1040      }
1041
1042    // set data for mobile page
1043    sd->is_mobile_page = EINA_TRUE;
1044    _smart_page_layout_info_set(sd, MOBILE_DEFAULT_ZOOM_RATIO, min_zoom_rate, max_zoom_rate, scalable);
1045 }
1046
1047 //#ifdef PROFUSION_INPUT_PATCH
1048 /**
1049  * Rotaion modes
1050  * @see appcore_set_rotation_cb(), appcore_get_rotation_state()
1051  */
1052 enum appcore_rm {
1053      APPCORE_RM_UNKNOWN, /**< Unknown mode */
1054      APPCORE_RM_PORTRAIT_NORMAL , /**< Portrait mode */
1055      APPCORE_RM_PORTRAIT_REVERSE , /**< Portrait upside down mode */
1056      APPCORE_RM_LANDSCAPE_NORMAL , /**< Left handed landscape mode */
1057      APPCORE_RM_LANDSCAPE_REVERSE ,  /**< Right handed landscape mode */
1058 };
1059 /*
1060 static void
1061 updateIMFOrientation( Ecore_IMF_Context *ctx )
1062 {
1063    if ( !ctx )
1064      return;
1065
1066    enum appcore_rm current_state = APPCORE_RM_UNKNOWN;
1067    int ret = appcore_get_rotation_state(&current_state);
1068
1069    switch (current_state)
1070      {
1071       case APPCORE_RM_PORTRAIT_NORMAL:
1072         ecore_imf_context_input_panel_orient_set(ctx, ECORE_IMF_INPUT_PANEL_ORIENT_NONE);
1073         break;
1074       case APPCORE_RM_PORTRAIT_REVERSE:
1075         ecore_imf_context_input_panel_orient_set(ctx, ECORE_IMF_INPUT_PANEL_ORIENT_180);
1076         break;
1077       case APPCORE_RM_LANDSCAPE_NORMAL:
1078         ecore_imf_context_input_panel_orient_set(ctx, ECORE_IMF_INPUT_PANEL_ORIENT_90_CW);
1079         break;
1080       case APPCORE_RM_LANDSCAPE_REVERSE:
1081         ecore_imf_context_input_panel_orient_set(ctx, ECORE_IMF_INPUT_PANEL_ORIENT_90_CCW);
1082         break;
1083      }
1084
1085    // call to show needed
1086    if ( ecore_imf_context_input_panel_state_get(ctx) == ECORE_IMF_INPUT_PANEL_STATE_SHOW )
1087      ecore_imf_context_input_panel_show(ctx);
1088 }
1089 */
1090
1091 static void
1092 _smart_input_method_changed(void* data, Evas_Object* webview, void* arg)
1093 {
1094    DBG("%s is called\n", __func__);
1095    Smart_Data* sd = (Smart_Data *)data;
1096    if (!sd) return;
1097
1098    if (sd->ewk_view_core_imContext_get == NULL)
1099      sd->ewk_view_core_imContext_get = (Ecore_IMF_Context* (*)(Evas_Object *)) dlsym(ewk_handle, "ewk_view_core_imContext_get");
1100
1101    Ecore_IMF_Context* imContext = sd->ewk_view_core_imContext_get(webview);
1102    Eina_Bool active = (Eina_Bool)arg;
1103    if (active && sd->mouse_clicked)
1104      {
1105         static unsigned int lastImh = 0;//FIXME
1106         if (sd->ewk_view_imh_get == NULL)
1107           sd->ewk_view_imh_get = (unsigned int (*)(Evas_Object *)) dlsym(ewk_handle, "ewk_view_imh_get");
1108         unsigned int imh = sd->ewk_view_imh_get(webview);
1109         if (ecore_imf_context_input_panel_state_get(imContext) != ECORE_IMF_INPUT_PANEL_STATE_SHOW || lastImh != imh)
1110           {
1111              lastImh = imh;
1112              //currentPage->reactToInputFieldTap(view, currentPage->getLastClickInfo().x, currentPage->getLastClickInfo().y);
1113              //updateIMFOrientation( imContext );
1114              ecore_imf_context_input_panel_reset (imContext);
1115              switch (imh)
1116                {
1117                 case EWK_IMH_TELEPHONE: ecore_imf_context_input_panel_layout_set(imContext, ECORE_IMF_INPUT_PANEL_LAYOUT_PHONENUMBER); break;
1118                 case EWK_IMH_NUMBER: ecore_imf_context_input_panel_layout_set(imContext, ECORE_IMF_INPUT_PANEL_LAYOUT_NUMBER); break;
1119                 case EWK_IMH_EMAIL: ecore_imf_context_input_panel_layout_set(imContext, ECORE_IMF_INPUT_PANEL_LAYOUT_EMAIL); break;
1120                 case EWK_IMH_URL: ecore_imf_context_input_panel_layout_set(imContext, ECORE_IMF_INPUT_PANEL_LAYOUT_URL); break;
1121                 default: ecore_imf_context_input_panel_layout_set(imContext, ECORE_IMF_INPUT_PANEL_LAYOUT_NORMAL);
1122                }
1123              DBG("ecore_imf_context_input_panel_show");
1124              ecore_imf_context_focus_in(imContext);
1125              ecore_imf_context_client_canvas_set(imContext, evas_object_evas_get(sd->base.self));
1126              ecore_imf_context_input_panel_show (imContext);
1127           }
1128      }
1129    else
1130      {
1131         DBG("ecore_imf_context_input_panel_hide");
1132         ecore_imf_context_input_panel_hide (imContext);
1133      }
1134 }
1135 //#endif
1136
1137 static void _smart_page_layout_info_set(Smart_Data *sd, float init_zoom_rate, float min_zoom_rate, float max_zoom_rate, Eina_Bool scalable)
1138 {
1139    Evas_Object* webview = sd->base.self;
1140
1141    int object_w, object_h;
1142    evas_object_geometry_get(webview, NULL, NULL, &object_w, &object_h);
1143    object_w = (object_w % 10) ? (object_w / 10 * 10 + 10) : object_w;
1144
1145    sd->zoom.init_zoom_rate = init_zoom_rate;
1146    sd->layout.w = object_w / sd->zoom.init_zoom_rate;
1147    sd->layout.h = object_h / sd->zoom.init_zoom_rate;
1148    sd->zoom.scalable = scalable;
1149    if (scalable)
1150      {
1151         sd->zoom.min_zoom_rate = (min_zoom_rate <= sd->zoom.init_zoom_rate) ? sd->zoom.init_zoom_rate : min_zoom_rate;
1152         sd->zoom.max_zoom_rate = (max_zoom_rate <= sd->zoom.init_zoom_rate) ? sd->zoom.init_zoom_rate : max_zoom_rate;
1153         if (sd->zoom.max_zoom_rate < sd->zoom.min_zoom_rate)
1154           sd->zoom.max_zoom_rate = sd->zoom.min_zoom_rate;
1155      }
1156    else
1157      {
1158         sd->zoom.min_zoom_rate = init_zoom_rate;
1159         sd->zoom.max_zoom_rate = init_zoom_rate;
1160      }
1161 }
1162
1163 static void
1164 _smart_contents_size_changed(void* data, Evas_Object* frame, void* arg)
1165 {
1166    Smart_Data* sd = (Smart_Data *)data;
1167    if (!sd) return;
1168
1169    Evas_Object* webview = sd->base.self;
1170
1171    Evas_Coord* size = (Evas_Coord*)arg;
1172    if (!size || size[0] == 0)
1173      return;
1174
1175    _update_min_zoom_rate(sd->base.self);
1176 }
1177
1178 static void
1179 _smart_load_nonemptylayout_finished(void* data, Evas_Object* frame, void* arg)
1180 {
1181    DBG("%s is called\n", __func__);
1182    Smart_Data* sd = (Smart_Data *)data;
1183    if (!sd) return;
1184
1185    Evas_Object* webview = sd->base.self;
1186
1187    if (!sd->ewk_view_user_scalable_set)
1188      sd->ewk_view_user_scalable_set = (void (*)(Evas_Object *, Eina_Bool))dlsym(ewk_handle, "ewk_view_user_scalable_set");
1189    if (!sd->ewk_view_zoom_range_set)
1190      sd->ewk_view_zoom_range_set = (Eina_Bool (*)(Evas_Object *, float, float))dlsym(ewk_handle, "ewk_view_zoom_range_set");
1191    if (!sd->ewk_view_fixed_layout_size_set)
1192      sd->ewk_view_fixed_layout_size_set = (void (*)(Evas_Object *, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_fixed_layout_size_set");
1193
1194    sd->ewk_view_zoom_range_set(webview, MIN_ZOOM_RATIO, MAX_ZOOM_RATIO);
1195
1196    // set default layout size
1197    int object_w, object_h;
1198    evas_object_geometry_get(webview, NULL, NULL, &object_w, &object_h);
1199    object_w = (object_w % 10) ? (object_w / 10 * 10 + 10) : object_w;
1200    sd->ewk_view_fixed_layout_size_set(webview, object_w, object_h);
1201
1202    sd->ewk_view_user_scalable_set(webview, EINA_TRUE);
1203
1204    // set zoom and layout
1205    if (sd->is_mobile_page)
1206      {
1207         if (!sd->ewk_view_zoom_set)
1208           sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
1209         if (!sd->ewk_frame_contents_size_get)
1210           sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
1211         if (!sd->ewk_view_uri_get)
1212           sd->ewk_view_uri_get = (const char * (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_uri_get");
1213
1214         sd->ewk_view_zoom_set(webview, sd->zoom.init_zoom_rate, 0, 0);
1215         sd->ewk_view_fixed_layout_size_set(webview, sd->layout.w, sd->layout.h);
1216
1217         int content_w;
1218         sd->ewk_frame_contents_size_get(frame, &content_w, NULL);
1219
1220         const char *url = sd->ewk_view_uri_get(webview);
1221         if ((content_w > sd->layout.w && !strstr(url, "docs.google.com"))
1222               || strstr(url, "maps.google.com/maps/m"))
1223           {
1224              // set page layout info, zoom and layout again
1225              _smart_page_layout_info_set(sd, 1.0f, sd->zoom.min_zoom_rate, sd->zoom.max_zoom_rate, sd->zoom.scalable);
1226              sd->ewk_view_zoom_range_set(webview, MIN_ZOOM_RATIO, MAX_ZOOM_RATIO);
1227              sd->ewk_view_zoom_set(webview, sd->zoom.init_zoom_rate, 0, 0);
1228              sd->ewk_view_fixed_layout_size_set(webview, sd->layout.w, sd->layout.h);
1229           }
1230         if (sd->use_zoom_bouncing)
1231           {
1232              float min_zoom_rate = sd->zoom.min_zoom_rate * ZOOM_OUT_BOUNCING;
1233              if (min_zoom_rate <= 0) min_zoom_rate = MIN_ZOOM_RATIO;
1234              float max_zoom_rate = sd->zoom.max_zoom_rate * ZOOM_IN_BOUNCING;
1235              sd->ewk_view_zoom_range_set(webview, min_zoom_rate, max_zoom_rate);
1236           }
1237         else
1238           {
1239              sd->ewk_view_zoom_range_set(webview, sd->zoom.min_zoom_rate, sd->zoom.max_zoom_rate);
1240           }
1241
1242      } else {
1243           sd->zoom.min_zoom_rate = MIN_ZOOM_RATIO;
1244           sd->zoom.max_zoom_rate = MAX_ZOOM_RATIO;
1245           if (sd->layout.w <= 0) sd->layout.w = sd->layout.default_w;
1246           sd->layout.h = object_h;
1247
1248           if (!sd->ewk_view_zoom_set)
1249             sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
1250           sd->ewk_view_zoom_set(webview, sd->zoom.init_zoom_rate, 0, 0);
1251           sd->ewk_view_fixed_layout_size_set(webview, sd->layout.w, sd->layout.h);
1252
1253           _update_min_zoom_rate(webview);
1254      }
1255
1256    sd->ewk_view_user_scalable_set(webview, sd->zoom.scalable);
1257 }
1258
1259 static void
1260 _smart_cb_view_created(void* data, Evas_Object* webview, void* arg)
1261 {
1262    printf("%s is called\n", __func__);
1263    Smart_Data* sd = (Smart_Data *)data;
1264    if (!sd) return;
1265    *((Evas_Object**)arg) = webview;
1266 }
1267
1268 static void
1269 _smart_add(Evas_Object* obj)
1270 {
1271    DBG("%s\n", __func__);
1272    Smart_Data* sd;
1273
1274    sd = calloc(1, sizeof(Smart_Data));
1275    if (!sd) return;
1276    evas_object_smart_data_set(obj, sd);
1277    _parent_sc.sc.add(obj);
1278
1279    sd->resize_calc_job = NULL;
1280    sd->move_calc_job = NULL;
1281    sd->thumbnail = NULL;
1282    sd->minimap.eo = NULL;
1283    sd->dropdown.options = NULL;
1284    sd->dropdown.option_cnt = 0;
1285    sd->use_text_selection = EINA_FALSE;
1286    sd->text_selection_on = EINA_FALSE;
1287    sd->events_feed = EINA_FALSE;
1288    sd->touch_obj = _elm_smart_touch_add(evas_object_evas_get(obj));
1289    sd->layout.default_w = DEFAULT_LAYOUT_WIDTH;
1290
1291    sd->ewk_view_theme_set = (void (*)(Evas_Object *, const char *))dlsym(ewk_handle, "ewk_view_theme_set");
1292    sd->ewk_view_theme_set(obj, WEBKIT_EDJ);
1293
1294    // set geolocation callback
1295    sd->ewk_set_show_geolocation_permission_dialog_callback = (void (*)(ewk_show_geolocation_permission_dialog_callback))dlsym(ewk_handle, "ewk_set_show_geolocation_permission_dialog_callback");
1296    sd->ewk_set_show_geolocation_permission_dialog_callback(_geolocation_permission_callback);
1297
1298    sd->ewk_view_zoom_text_only_set = (Eina_Bool (*)(Evas_Object *, Eina_Bool))dlsym(ewk_handle, "ewk_view_zoom_text_only_set");
1299    sd->ewk_view_zoom_text_only_set(obj, EINA_FALSE);
1300    sd->ewk_view_zoom_cairo_scaling_set = (Eina_Bool (*)(Evas_Object *, Eina_Bool))dlsym(ewk_handle, "ewk_view_zoom_cairo_scaling_set");
1301    sd->ewk_view_zoom_cairo_scaling_set(obj, EINA_TRUE);
1302    sd->flush_and_pre_render_idler = NULL;
1303    sd->use_zoom_bouncing = EINA_TRUE;
1304
1305 #ifdef NEED_TO_REMOVE
1306    // TODO: temporary add the mouse callbacks until the webkit engine can receive mouse events
1307    evas_object_event_callback_add(obj, EVAS_CALLBACK_MOUSE_DOWN, _view_on_mouse_down, sd);
1308    evas_object_event_callback_add(obj, EVAS_CALLBACK_MOUSE_UP, _view_on_mouse_up, sd);
1309 #endif
1310
1311    evas_object_smart_callback_add(obj, "load,started", _smart_load_started, sd);
1312    evas_object_smart_callback_add(obj, "load,finished", _smart_load_finished, sd);
1313    evas_object_smart_callback_add(obj, "load,error", _smart_load_error, sd);
1314    evas_object_smart_callback_add(obj, "viewport,changed", _smart_viewport_changed, sd);
1315    evas_object_smart_callback_add(obj, "inputmethod,changed", _smart_input_method_changed, sd);
1316
1317    evas_object_smart_callback_add(obj, "webview,created", _smart_cb_view_created, sd); // I need to consider more
1318
1319    if (!(sd->ewk_view_frame_main_get))
1320      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
1321    evas_object_smart_callback_add(sd->ewk_view_frame_main_get(obj), "contents,size,changed",
1322          _smart_contents_size_changed, sd);
1323    evas_object_smart_callback_add(sd->ewk_view_frame_main_get(obj), "load,nonemptylayout,finished",
1324          _smart_load_nonemptylayout_finished, sd);
1325
1326    evas_object_smart_callback_add(obj, "one,press", _smart_cb_mouse_down, sd);
1327    evas_object_smart_callback_add(obj, "one,release", _smart_cb_mouse_up, sd);
1328    evas_object_smart_callback_add(obj, "one,single,tap", _smart_cb_mouse_tap, sd);
1329    evas_object_smart_callback_add(obj, "one,long,press", _smart_cb_select_closest_word, sd);
1330    evas_object_smart_callback_add(obj, "one,double,tap", _smart_cb_smart_zoom, sd);
1331    evas_object_smart_callback_add(obj, "one,move,start", _smart_cb_pan_start, sd);
1332    evas_object_smart_callback_add(obj, "one,move", _smart_cb_pan_by, sd);
1333    evas_object_smart_callback_add(obj, "one,move,end", _smart_cb_pan_stop, sd);
1334    evas_object_smart_callback_add(obj, "two,move,start", _smart_cb_pinch_zoom_start, sd);
1335    evas_object_smart_callback_add(obj, "two,move", _smart_cb_pinch_zoom_move, sd);
1336    evas_object_smart_callback_add(obj, "two,move,end", _smart_cb_pinch_zoom_stop, sd);
1337
1338    evas_object_size_hint_weight_set(obj, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
1339    evas_object_size_hint_align_set(obj, EVAS_HINT_FILL, EVAS_HINT_FILL);
1340
1341    _elm_smart_touch_child_set(sd->touch_obj, obj);
1342    _text_selection_init(obj);
1343 }
1344
1345 static void
1346 _smart_del(Evas_Object* obj)
1347 {
1348    DBG("%s\n", __func__);
1349    INTERNAL_ENTRY;
1350
1351    if (sd->minimap.eo != NULL)
1352      {
1353         evas_object_del(sd->minimap.eo);
1354         sd->minimap.eo = NULL;
1355      }
1356
1357    if (sd->minimap.content != NULL)
1358      {
1359         evas_object_del(sd->minimap.content);
1360         sd->minimap.content = NULL;
1361      }
1362
1363    _parent_sc.sc.del(obj);
1364
1365    if (--ewk_tile_cache_ref_count == 0)
1366      ewk_tile_cache = NULL;
1367 }
1368
1369 static void
1370 _directional_pre_render(Evas_Object* obj, int dx, int dy)
1371 {
1372    INTERNAL_ENTRY;
1373
1374    if (!sd->ewk_view_frame_main_get)
1375      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
1376    if (!sd->ewk_frame_visible_content_geometry_get)
1377      sd->ewk_frame_visible_content_geometry_get = (Eina_Bool (*)(const Evas_Object *, Eina_Bool, int *, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_visible_content_geometry_get");
1378    int x, y, w, h;
1379    sd->ewk_frame_visible_content_geometry_get(sd->ewk_view_frame_main_get(obj), false, &x, &y, &w, &h);
1380    DBG("visible content: (%d, %d, %d, %d)", x, y, w, h);
1381
1382    typedef enum { up, down, left, right, up_left, up_right, down_left, down_right, undefined } Directions;
1383    Directions direction = undefined;
1384
1385 #ifdef USE_MAX_TUC_20MB
1386    if (!sd->ewk_view_zoom_get)
1387      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
1388    float zoom = sd->ewk_view_zoom_get(obj);
1389 #endif
1390
1391    if (dx == 0 && dy <  0) direction = down;
1392    if (dx >  0 && dy <  0) direction = down_left;
1393    if (dx >  0 && dy == 0) direction = left;
1394    if (dx >  0 && dy >  0) direction = up_left;
1395    if (dx == 0 && dy >  0) direction = up;
1396    if (dx <  0 && dy >  0) direction = up_right;
1397    if (dx <  0 && dy == 0) direction = right;
1398    if (dx <  0 && dy <  0) direction = down_right;
1399
1400 #ifdef USE_MAX_TUC_20MB
1401    const float DIRECTION_PLAIN_CX = 2.0/zoom;
1402    const float DIRECTION_CROSS_CX = 1.0/zoom;
1403    const float DIRECTION_UNDEFINED_CX_LEVEL_1 = 0.5/zoom;
1404    const float DIRECTION_UNDEFINED_CX_LEVEL_2 = 0.8/zoom;
1405 #else
1406    const float DIRECTION_PLAIN_CX = 1.5;
1407    const float DIRECTION_CROSS_CX = 0.7;
1408    const float DIRECTION_UNDEFINED_CX_LEVEL_1 = 0.3;
1409    const float DIRECTION_UNDEFINED_CX_LEVEL_2 = 0.6;
1410    const float DIRECTION_UNDEFINED_CX_LEVEL_3 = 0.8;
1411 #endif
1412
1413    int p_x = x, p_y = y, p_w = w, p_h = h;
1414
1415    switch (direction) {
1416       case up:
1417          DBG("Direction: up");
1418          p_y = y - h * DIRECTION_PLAIN_CX;
1419          p_h = h * DIRECTION_PLAIN_CX;
1420          break;
1421       case up_right:
1422          DBG("Direction: up_right");
1423          p_w = w + w * DIRECTION_CROSS_CX;
1424          p_y = y - h * DIRECTION_CROSS_CX;
1425          p_h = h + h * DIRECTION_CROSS_CX;
1426          break;
1427       case right:
1428          DBG("Direction: right");
1429          p_x = x + w;
1430          p_w = w * DIRECTION_PLAIN_CX;
1431          break;
1432       case down_right:
1433          DBG("Direction: down_right");
1434          p_w = w + w * DIRECTION_CROSS_CX;
1435          p_h = h + h * DIRECTION_CROSS_CX;
1436          break;
1437       case down:
1438          DBG("Direction: down");
1439          p_y = y + h;
1440          p_h = h * DIRECTION_PLAIN_CX;
1441          break;
1442       case down_left:
1443          DBG("Direction: down_left");
1444          p_x = x - w * DIRECTION_CROSS_CX;
1445          p_w = w + w * DIRECTION_CROSS_CX;
1446          p_h = h + h * DIRECTION_CROSS_CX;
1447          break;
1448       case left:
1449          DBG("Direction: left");
1450          p_x = x - w * DIRECTION_PLAIN_CX;
1451          p_w = w * DIRECTION_PLAIN_CX;
1452          break;
1453       case up_left:
1454          DBG("Direction: left_up");
1455          p_x = x - w * DIRECTION_CROSS_CX;
1456          p_w = w + w * DIRECTION_CROSS_CX;
1457          p_y = y - h * DIRECTION_CROSS_CX;
1458          p_h = h + h * DIRECTION_CROSS_CX;
1459          break;
1460       case undefined:
1461          DBG("Direction: undefined");
1462          break;
1463       default:
1464          DBG("Shouldn't happen!!");
1465    }
1466
1467 #ifndef USE_MAX_TUC_20MB
1468    if (!sd->ewk_view_zoom_get)
1469      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
1470    float zoom = sd->ewk_view_zoom_get(obj);
1471 #endif
1472
1473    // cancel the previously scheduled pre-rendering
1474    // This makes sense especilaly for zooming operation - when user
1475    // finishes zooming, and pre-render for the previous zoom was
1476    // not finished, it doesn't make sense to continue pre-rendering for the previous zoom
1477    if (!sd->ewk_view_pre_render_cancel)
1478      sd->ewk_view_pre_render_cancel = (void (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_pre_render_cancel");
1479    sd->ewk_view_pre_render_cancel(obj);
1480
1481    if (!sd->ewk_view_pre_render_region)
1482      sd->ewk_view_pre_render_region = (Eina_Bool (*)(Evas_Object *, Evas_Coord, Evas_Coord, Evas_Coord, Evas_Coord, float))dlsym(ewk_handle, "ewk_view_pre_render_region");
1483
1484    if (direction != undefined)
1485      {
1486         /* Queue tiles in the direction of the last panning */
1487         DBG("pre rendering - directional - content: (%d, %d, %d, %d), zoom %.3f",p_x, p_y, p_w, p_h, zoom);
1488
1489         sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, p_h, zoom);
1490         //dbg_draw_scaled_area(obj, 0, p_x, p_y, p_w, p_h);
1491      }
1492    else
1493      {
1494         DBG("pre rendering - directional - skipped");
1495         //dbg_draw_scaled_area(obj, 0, 0, 0, 0, 0);
1496      }
1497
1498 #ifdef USE_MAX_TUC_20MB
1499    int content_w=0, content_h=0;
1500    int center_x=0,center_y=0;
1501    int tmp_h=0;
1502
1503    if (!sd->ewk_frame_contents_size_get)
1504      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
1505    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(obj), &content_w, &content_h);
1506
1507    p_w = content_w;
1508    p_h = content_h;
1509
1510    size_t size = (size_t)roundf(p_w * zoom * p_h * zoom * 4);
1511    Eina_Bool  toggle = EINA_FALSE;
1512
1513    while(size > (MAX_TUC*0.8))
1514      {
1515         if(toggle)
1516           {
1517              p_h = p_h -32;
1518           }
1519         else
1520           {
1521              p_w = p_w - 32;
1522              if(p_w < w)
1523                {
1524                   p_w = w;
1525                   toggle = EINA_TRUE;
1526                }
1527           }
1528         size = (size_t)roundf(p_w * zoom * p_h * zoom * 4);
1529      }
1530
1531    center_x = (int)roundf(x + w/2);
1532    center_y = (int)roundf(y + h/2);
1533
1534    tmp_h = p_h* DIRECTION_UNDEFINED_CX_LEVEL_1;
1535    p_x = center_x - (int)roundf(p_w/2);
1536    p_y = center_y - (int)roundf(tmp_h/2);
1537    if(p_x < 0) p_x = 0;
1538    if(p_y < 0) p_y = 0;
1539    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, tmp_h, zoom);
1540
1541    tmp_h = p_h* DIRECTION_UNDEFINED_CX_LEVEL_2;
1542    p_x = center_x - (int)roundf(p_w/2);
1543    p_y = center_y - (int)roundf(tmp_h/2);
1544    if(p_x < 0) p_x = 0;
1545    if(p_y < 0) p_y = 0;
1546    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, tmp_h, zoom);
1547
1548    p_x = center_x - (int)roundf(p_w/2);
1549    p_y = center_y - (int)roundf(p_h/2);
1550    if(p_x < 0) p_x = 0;
1551    if(p_y < 0) p_y = 0;
1552    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, p_h, zoom);
1553 #else
1554    /* Queue tiles in a small rectangle around the viewport */
1555    p_x = x - w * DIRECTION_UNDEFINED_CX_LEVEL_1;
1556    p_y = y - h * DIRECTION_UNDEFINED_CX_LEVEL_1;
1557    p_w = w + 2.0 * w * DIRECTION_UNDEFINED_CX_LEVEL_1;
1558    p_h = h + 2.0 * h * DIRECTION_UNDEFINED_CX_LEVEL_1;
1559    DBG("pre rendering - small - content: (%d, %d, %d, %d), zoom %.3f", p_x, p_y, p_w, p_h, zoom);
1560    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, p_h, zoom);
1561    //dbg_draw_scaled_area(obj, 1, p_x, p_y, p_w, p_h);
1562
1563    /* Queue tiles in a medium rectangle around the viewport */
1564    p_x = x - w * DIRECTION_UNDEFINED_CX_LEVEL_2;
1565    p_y = y - h * DIRECTION_UNDEFINED_CX_LEVEL_2;
1566    p_w = w + 2.0 * w * DIRECTION_UNDEFINED_CX_LEVEL_2;
1567    p_h = h + 2.0 * h * DIRECTION_UNDEFINED_CX_LEVEL_2;
1568    DBG("pre rendering - medium - content: (%d, %d, %d, %d), zoom %.3f", p_x, p_y, p_w, p_h, zoom);
1569    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, p_h, zoom);
1570    //dbg_draw_scaled_area(obj, 2, p_x, p_y, p_w, p_h);
1571
1572    /* Queue tiles in a large rectangle around the viewport */
1573    p_x = x - w * DIRECTION_UNDEFINED_CX_LEVEL_3;
1574    p_y = y - h * DIRECTION_UNDEFINED_CX_LEVEL_3;
1575    p_w = w + 2.0 * w * DIRECTION_UNDEFINED_CX_LEVEL_3;
1576    p_h = h + 2.0 * h * DIRECTION_UNDEFINED_CX_LEVEL_3;
1577    DBG("pre rendering - large - content: (%d, %d, %d, %d), zoom %.3f", p_x, p_y, p_w, p_h, zoom);
1578    sd->ewk_view_pre_render_region(obj, p_x, p_y, p_w, p_h, zoom);
1579    //dbg_draw_scaled_area(obj, 3, p_x, p_y, p_w, p_h);
1580 #endif
1581
1582    /* Log some statistics */
1583    /*
1584       int v_w, v_h;
1585       evas_object_geometry_get(obj, NULL, NULL, &v_w, &v_h);
1586       Ewk_Tile_Unused_Cache *tuc = ewk_view_tiled_unused_cache_get(obj);
1587       size_t used = ewk_tile_unused_cache_used_get(tuc);
1588       size_t max = ewk_tile_unused_cache_max_get(tuc);
1589    // Will this work for non cairo scaling?
1590    int est = (zoomRatio*p_w * zoomRatio*p_h - v_w * v_h) * 4; // 4 bytes per pixel
1591    DBG("pre rendering - Cache max = %.1fMB  Cache used = %.1fMB  Estimated size of pre-render area: %.1fMB\n", 
1592    max/1024.0/1024.0, used/1024.0/1024.0, est/1024.0/1024.0);
1593    if (est > max)
1594    DBG("WARNING!! estimated size of pre-render are is larger than the cache size. This will result in inefficient use of cache!");
1595    */
1596 }
1597
1598 static void
1599 _smart_cb_mouse_down(void* data, Evas_Object* webview, void* ev)
1600 {
1601    DBG("%s\n", __func__);
1602    Smart_Data* sd = (Smart_Data *)data;
1603    if (!sd) return;
1604    if (sd->events_feed == EINA_TRUE) return;
1605    //Evas_Point* point = (Evas_Point*)ev;
1606
1607    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE) return;
1608
1609 #ifdef NEED_TO_REMOVE
1610    evas_object_focus_set(webview, EINA_TRUE);
1611    if (!sd->ewk_view_frame_main_get)
1612      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
1613    if (!sd->ewk_frame_feed_focus_in)
1614      sd->ewk_frame_feed_focus_in = (Eina_Bool (*)(Evas_Object *))dlsym(ewk_handle, "ewk_frame_feed_focus_in");
1615    sd->ewk_frame_feed_focus_in(sd->ewk_view_frame_main_get(webview));
1616 #endif
1617
1618    sd->mouse_clicked = EINA_TRUE;
1619    Evas_Event_Mouse_Down mouse_down = sd->mouse_down_copy;
1620    Evas_Point* point = (Evas_Point*)ev;
1621    mouse_down.canvas.x = point->x;
1622    mouse_down.canvas.y = point->y;
1623    _parent_sc.mouse_down((Ewk_View_Smart_Data*)sd, &mouse_down);
1624
1625 #if 0 // comment out below code until it is completed
1626    if (sd->bounce_horiz)
1627      elm_widget_drag_lock_x_set(sd->widget, EINA_TRUE);
1628    if (sd->bounce_vert)
1629      elm_widget_drag_lock_y_set(sd->widget, EINA_TRUE);
1630 #endif
1631 }
1632
1633 static void
1634 _smart_cb_mouse_up(void* data, Evas_Object* webview, void* ev)
1635 {
1636    DBG("%s\n", __func__);
1637    Smart_Data* sd = (Smart_Data *)data;
1638    if (!sd) return;
1639    if (sd->events_feed == EINA_TRUE) return;
1640
1641    Evas_Point* point = (Evas_Point*)ev;
1642    DBG(" argument : (%d, %d)\n", point->x, point->y);
1643 }
1644
1645 static void
1646 _smart_cb_mouse_tap(void* data, Evas_Object* webview, void* ev)
1647 {
1648    DBG("%s\n", __func__);
1649    Smart_Data* sd = (Smart_Data *)data;
1650    if (!sd) return;
1651    if (sd->events_feed == EINA_TRUE) return;
1652
1653    Evas_Point* point = (Evas_Point*)ev;
1654    DBG(" argument : (%d, %d)\n", point->x, point->y);
1655
1656    // check for video link
1657    int ewk_x, ewk_y;
1658    _coords_evas_to_ewk(webview, point->x, point->y, &ewk_x, &ewk_y);
1659    Eina_Bool have_link = EINA_FALSE;
1660    Eina_Bool have_image = EINA_FALSE;
1661    char *link_url = NULL, *link_text = NULL, *image_url = NULL;
1662    if (!sd->ewk_page_check_point)
1663      sd->ewk_page_check_point = (Eina_Bool (*)(Evas_Object *, int, int, Evas_Event_Mouse_Down *, Eina_Bool *, Eina_Bool *, char **, char **, char **))dlsym(ewk_handle, "ewk_page_check_point");
1664    sd->ewk_page_check_point(webview, ewk_x, ewk_y, &sd->mouse_down_copy,
1665          &have_link, &have_image, &link_url, &link_text, &image_url);
1666    if (link_url) free(link_url);
1667    if (link_text) free(link_text);
1668    if (image_url) free(image_url);
1669
1670    //TODO: below code is not based on open source (need to check and refactor)
1671    int x = 0, y = 0;
1672    _unzoom_position(webview, point->x, point->y, &x, &y);
1673
1674    // check for input field
1675    if (!sd->ewk_page_check_point_for_keyboard)
1676      sd->ewk_page_check_point_for_keyboard = (char * (*)(Evas_Object *, int, int, Eina_Bool *))dlsym(ewk_handle, "ewk_page_check_point_for_keyboard");
1677    if (!sd->ewk_page_dropdown_get_options)
1678      sd->ewk_page_dropdown_get_options = (char ** (*)(Evas_Object *, int, int, int *, int *))dlsym(ewk_handle, "ewk_page_dropdown_get_options");
1679
1680    Eina_Bool have_input_field;
1681    sd->ewk_page_check_point_for_keyboard(webview, x, y, &have_input_field);
1682    if (have_input_field == EINA_TRUE)
1683      {
1684         _zoom_to_rect(sd, point->x, point->y);
1685
1686         // check whether it is radio
1687      }
1688    else if (NULL != (sd->dropdown.options = sd->ewk_page_dropdown_get_options(webview, x, y,
1689                &sd->dropdown.option_cnt, &sd->dropdown.option_idx)))
1690      {
1691         Evas* evas;
1692         evas = evas_object_evas_get(webview);
1693
1694         // TODO: we have to show list instead of discpicker
1695         /* below code is deprecated
1696         Evas_Object* discpicker = elm_discpicker_add(webview);
1697         if (discpicker)
1698           {
1699              // set items
1700              int i;
1701              Elm_Discpicker_Item* item;
1702              for (i = 0; i < sd->dropdown.option_cnt; i++)
1703                {
1704                   item = elm_discpicker_item_append(discpicker, sd->dropdown.options[i], NULL, NULL);
1705                   if (i == sd->dropdown.option_idx)
1706                     {
1707                        elm_discpicker_item_selected_set(item);
1708                     }
1709                }
1710
1711              // selected callback
1712              void discpicker_selected_cb(void* data, Evas_Object* obj, void* event_info)
1713                {
1714                   Smart_Data* sd = (Smart_Data *)data;
1715                   if (!sd) return;
1716                   Evas_Object* webview = sd->base.self;
1717
1718                   int x = 0, y = 0;
1719                   Evas_Point* point = &sd->mouse_up_copy.output;
1720                   _unzoom_position(webview, point->x, point->y, &x, &y);
1721
1722                   Elm_Discpicker_Item* item = event_info;
1723                   const char *selected_label = elm_discpicker_item_label_get(item);
1724                   int selected_index;
1725                   for (selected_index = 0; selected_index < sd->dropdown.option_cnt; selected_index++)
1726                     {
1727                        if (!strcmp(selected_label, sd->dropdown.options[selected_index]))
1728                          {
1729                             break;
1730                          }
1731                     }
1732                   printf("<< selected [%d | %s] >>\n", selected_index, selected_label);
1733                   if (!sd->ewk_page_dropdown_set_current_index)
1734                     sd->ewk_page_dropdown_set_current_index = (Eina_Bool (*)(Evas_Object *, int, int, int))dlsym(ewk_handle, "ewk_page_dropdown_set_current_index");
1735                   sd->ewk_page_dropdown_set_current_index(webview, x, y, selected_index);
1736                   //evas_object_del(obj);
1737                }
1738
1739              // show discpicker
1740              evas_object_smart_callback_add(discpicker, "selected", discpicker_selected_cb, sd);
1741              elm_discpicker_row_height_set(discpicker, 80);
1742              evas_object_resize(discpicker, 480, 400);
1743              evas_object_move(discpicker, 0, 400);
1744              evas_object_show(discpicker);
1745           }
1746           */
1747      }
1748
1749    if (sd->use_text_selection  == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
1750      {
1751         _smart_cb_unselect_closest_word(sd, webview, NULL);
1752         return;
1753      }
1754
1755    Evas_Event_Mouse_Up mouse_up = sd->mouse_up_copy;
1756    mouse_up.canvas.x = point->x;
1757    mouse_up.canvas.y = point->y;
1758    _parent_sc.mouse_up((Ewk_View_Smart_Data*)sd, &mouse_up);
1759    sd->mouse_clicked = EINA_FALSE;
1760 }
1761
1762 static void
1763 _smart_cb_pan_start(void* data, Evas_Object* webview, void* ev)
1764 {
1765    DBG("%s\n", __func__);
1766    Smart_Data* sd = (Smart_Data *)data;
1767    if (!sd) return;
1768    Evas_Point* point = (Evas_Point*)ev;
1769
1770    if (sd->events_feed == EINA_TRUE) return;
1771
1772    sd->pan_s = *point;
1773    sd->on_panning = EINA_TRUE;
1774
1775    if (sd->use_text_selection  == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
1776      {
1777         if (_text_selection_handle_pressed(sd, point->x, point->y))
1778           _elm_smart_touch_is_one_drag_mode_enable(sd->touch_obj, EINA_FALSE);
1779      }
1780
1781    _suspend_all(sd, EINA_FALSE);
1782
1783    sd->locked_dx = 0;
1784    sd->locked_dy = 0;
1785 }
1786
1787 static void
1788 _smart_cb_pan_by(void* data, Evas_Object* webview, void* ev)
1789 {
1790    //DBG("%s\n", __func__);
1791    Smart_Data* sd = (Smart_Data *)data;
1792    if (!sd) return;
1793    Evas_Point* point = (Evas_Point*)ev;
1794
1795    if (sd->events_feed == EINA_TRUE)
1796      {
1797         Evas* evas = evas_object_evas_get(webview);
1798         Evas_Modifier *modifiers = (Evas_Modifier *)evas_key_modifier_get(evas);
1799         Evas_Lock *locks = (Evas_Lock *)evas_key_lock_get(evas);
1800
1801         Evas_Event_Mouse_Move event_move;
1802         event_move.buttons = 1;
1803         event_move.cur.output.x = point->x;
1804         event_move.cur.output.y = point->y;
1805         event_move.cur.canvas.x = point->x;
1806         event_move.cur.canvas.y = point->y;
1807         event_move.data = NULL;
1808         event_move.modifiers = modifiers;
1809         event_move.locks = locks;
1810         event_move.timestamp = ecore_loop_time_get();
1811         event_move.event_flags = EVAS_EVENT_FLAG_NONE;
1812         event_move.dev = NULL;
1813
1814         _parent_sc.mouse_move((Ewk_View_Smart_Data*)sd, &event_move);
1815         return;
1816      }
1817    if (sd->on_panning == EINA_FALSE) return;
1818
1819    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
1820      {
1821         if (sd->text_selection.front_handle_moving == EINA_TRUE
1822               || sd->text_selection.back_handle_moving == EINA_TRUE)
1823           {
1824              _text_selection_update_position(sd, point->x, point->y);
1825              return;
1826           }
1827      }
1828
1829    if (!sd->ewk_frame_scroll_pos_get)
1830      sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
1831
1832    int dx = sd->pan_s.x - point->x;
1833    int dy = sd->pan_s.y - point->y;
1834
1835    if (!sd->ewk_view_frame_main_get)
1836      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
1837
1838    int old_x, old_y;
1839    sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(webview), &old_x, &old_y);
1840
1841    if (!sd->ewk_frame_contents_size_get)
1842      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
1843
1844    int content_w, content_h;
1845    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(webview), &content_w, &content_h);
1846    if (!sd->ewk_view_zoom_get)
1847      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
1848    float zoom = sd->ewk_view_zoom_get(webview);
1849    content_w *= zoom;
1850    content_h *= zoom;
1851    DBG("<< ========content [%d, %d] new pos [%d, %d] >>\n", content_w, content_h, old_x + dx, old_y + dy);
1852
1853 #if 0 // comment out below code until it is completed
1854    Eina_Bool locked = EINA_FALSE;
1855    if (!elm_widget_drag_lock_x_get(sd->widget))
1856      {
1857         if ((old_x + dx) >= 0 && (old_x + dx) <=content_w)
1858           elm_widget_drag_lock_x_set(sd->widget, EINA_TRUE);
1859         else if ((sd->locked_dx > 0 && (sd->locked_dx + dx) <= 0)
1860               || (sd->locked_dx < 0 && (sd->locked_dx + dx) >= 0))
1861           {
1862              elm_widget_drag_lock_x_set(sd->widget, EINA_TRUE);
1863              DBG("===============<< widget x lock >>\n");
1864              dx += sd->locked_dx;
1865           }
1866         else
1867           {
1868              sd->locked_dx += dx;
1869              locked = EINA_TRUE;
1870           }
1871      }
1872    if (!elm_widget_drag_lock_y_get(sd->widget))
1873      {
1874         if ((old_y + dy) >= 0 && (old_y + dy) <= content_h)
1875           elm_widget_drag_lock_y_set(sd->widget, EINA_TRUE);
1876         else if ((sd->locked_dy > 0 && (sd->locked_dy + dy) <= 0)
1877               || (sd->locked_dy < 0 && (sd->locked_dy + dy) >= 0))
1878           {
1879              elm_widget_drag_lock_y_set(sd->widget, EINA_TRUE);
1880              DBG("===============<< widget y lock >>\n");
1881              dy += sd->locked_dy;
1882           }
1883         else
1884           {
1885              sd->locked_dy += dy;
1886              locked = EINA_TRUE;
1887           }
1888      }
1889
1890    if (locked) return;
1891 #endif
1892
1893    if (!sd->ewk_view_frame_main_get)
1894      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
1895    if (!sd->ewk_frame_scroll_add)
1896      sd->ewk_frame_scroll_add = (Eina_Bool (*)(Evas_Object *, int, int))dlsym(ewk_handle, "ewk_frame_scroll_add");
1897    sd->ewk_frame_scroll_add(sd->ewk_view_frame_main_get(webview), dx, dy);
1898
1899    _minimap_update(sd->minimap.content, sd, sd->thumbnail,
1900          sd->minimap.cw, sd->minimap.ch);
1901    sd->pan_s = *point;
1902
1903    int new_x, new_y;
1904    sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(webview), &new_x, &new_y);
1905
1906    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
1907      _text_selection_move_by(sd, old_x - new_x, old_y - new_y);
1908
1909 #if 0 // comment out below code until it is completed
1910    if (!sd->bounce_horiz &&
1911          (dx && elm_widget_drag_lock_x_get(sd->widget) && (old_x == new_x)))
1912      {
1913         sd->locked_dx = dx - (old_x - new_x);
1914         elm_widget_drag_lock_x_set(sd->widget, EINA_FALSE);
1915         DBG("===============<< widget x unlock >>\n");
1916      }
1917
1918    if (!sd->bounce_vert &&
1919          (dy && elm_widget_drag_lock_y_get(sd->widget) && (old_y == new_y)))
1920      {
1921         sd->locked_dy = dy - (old_y - new_y);
1922         elm_widget_drag_lock_y_set(sd->widget, EINA_FALSE);
1923         DBG("===============<< widget y unlock >>\n");
1924      }
1925 #endif
1926 }
1927
1928 static void
1929 _smart_cb_pan_stop(void* data, Evas_Object* webview, void* ev)
1930 {
1931    DBG("%s\n", __func__);
1932    Smart_Data* sd = (Smart_Data *)data;
1933    if (!sd) return;
1934    if (sd->events_feed == EINA_TRUE) return;
1935
1936    Evas_Point* point = (Evas_Point*)ev;
1937    sd->on_panning = EINA_FALSE;
1938
1939    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
1940      {
1941         if (sd->text_selection.front_handle_moving == EINA_TRUE
1942               || sd->text_selection.back_handle_moving == EINA_TRUE)
1943           _elm_smart_touch_is_one_drag_mode_enable(sd->touch_obj, EINA_TRUE);
1944         sd->text_selection.front_handle_moving = EINA_FALSE;
1945         sd->text_selection.back_handle_moving = EINA_FALSE;
1946      }
1947
1948    _resume_all(sd, EINA_FALSE);
1949
1950    if (sd->tiled)
1951      {
1952         if (!sd->ewk_view_tiled_unused_cache_get)
1953           sd->ewk_view_tiled_unused_cache_get = (Ewk_Tile_Unused_Cache *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_get");
1954         if (!sd->ewk_tile_unused_cache_used_get)
1955           sd->ewk_tile_unused_cache_used_get = (size_t (*)(const Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_used_get");
1956         if (!sd->ewk_tile_unused_cache_max_get)
1957           sd->ewk_tile_unused_cache_max_get = (size_t (*)(const Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_max_get");
1958         Ewk_Tile_Unused_Cache *tuc = sd->ewk_view_tiled_unused_cache_get(webview);
1959         size_t used = sd->ewk_tile_unused_cache_used_get(tuc);
1960         size_t max = sd->ewk_tile_unused_cache_max_get(tuc);
1961         DBG("[%s] max = %d  used = %d \n", __func__, max, used);
1962         if (used > max)
1963           {
1964              if (!sd->ewk_tile_unused_cache_auto_flush)
1965                sd->ewk_tile_unused_cache_auto_flush = (void (*)(Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_auto_flush");
1966              sd->ewk_tile_unused_cache_auto_flush(tuc);
1967           }
1968         _directional_pre_render(webview,
1969               (sd->mouse_down_copy.canvas.x - point->x), (sd->mouse_down_copy.canvas.y - point->y));
1970      }
1971
1972 #if 0 // comment out below code until it is completed
1973    if (!sd->bounce_horiz && elm_widget_drag_lock_x_get(sd->widget))
1974      {
1975         DBG("==============<< widget x unlock >>\n");
1976         elm_widget_drag_lock_x_set(sd->widget, EINA_FALSE);
1977      }
1978
1979    if (!sd->bounce_vert && elm_widget_drag_lock_y_get(sd->widget))
1980      {
1981         DBG("==============<< widget y unlock >>\n");
1982         elm_widget_drag_lock_y_set(sd->widget, EINA_FALSE);
1983      }
1984 #endif
1985 }
1986
1987 static void
1988 _smart_cb_select_closest_word(void* data, Evas_Object* webview, void* ev)
1989 {
1990    DBG("%s\n", __func__);
1991    Smart_Data* sd = (Smart_Data *)data;
1992    if (!sd) return;
1993    if (sd->events_feed == EINA_TRUE) return;
1994
1995    Evas_Point* point = (Evas_Point*)ev;
1996
1997    if (sd->use_text_selection == EINA_FALSE) return;
1998
1999    int x, y;
2000    _coords_evas_to_ewk(webview, point->x, point->y, &x, &y);
2001
2002    if (!sd->ewk_view_frame_main_get)
2003      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2004    if (!sd->ewk_frame_select_closest_word)
2005      sd->ewk_frame_select_closest_word = (Eina_Bool (*)(Evas_Object *, int, int, int *, int *, int *, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_select_closest_word");
2006    int tx, ty, th, bx, by, bh;
2007    Eina_Bool ret = sd->ewk_frame_select_closest_word(sd->ewk_view_frame_main_get(webview), x, y,
2008          &tx, &ty, &th, &bx, &by, &bh);
2009    if (ret)
2010      {
2011         _coords_ewk_to_evas(webview, tx, ty, &tx, &ty);
2012         _coords_ewk_to_evas(webview, bx, by, &bx, &by);
2013         _text_selection_show();
2014         _text_selection_set_front_info(sd, tx, ty, th);
2015         _text_selection_set_back_info(sd, bx, by, bh);
2016         sd->text_selection_on = EINA_TRUE;
2017      }
2018 }
2019
2020 static void
2021 _smart_cb_unselect_closest_word(void* data, Evas_Object* webview, void* ev)
2022 {
2023    DBG("%s\n", __func__);
2024    Smart_Data* sd = (Smart_Data *)data;
2025    if (!sd) return;
2026
2027    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
2028      {
2029         _text_selection_hide(sd);
2030         if (!sd->ewk_view_select_none)
2031           sd->ewk_view_select_none = (Eina_Bool (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_select_none");
2032         sd->ewk_view_select_none(webview);
2033         sd->text_selection_on = EINA_FALSE;
2034      }
2035 }
2036
2037 // zoom
2038 static const int ZOOM_STEP_TRESHOLD = 20;
2039 static const float ZOOM_STEP_PER_PIXEL = 0.005f;
2040
2041 #define ZOOM_FRAMERATE 60
2042 #define N_COSINE 18
2043 static const float cosine[N_COSINE] =
2044 { 1.0f, 0.99f, 0.96f, 0.93f, 0.88f, 0.82f, 0.75f, 0.67f, 0.59f, 0.5f,
2045    0.41f, 0.33f, 0.25f, 0.18f, 0.12f, 0.07f, 0.01f, 0.0f };
2046 static int smart_zoom_index = N_COSINE - 1;
2047
2048 #define INPUT_LOCATION_X 20
2049 #define INPUT_LOCATION_Y 50
2050 #define INPUT_ZOOM_RATIO 2.5
2051
2052 static void
2053 _suspend_all(Smart_Data *sd, Eina_Bool hidePlugin)
2054 {
2055    Evas_Object *webview = sd->base.self;
2056
2057    // javascript suspend
2058    if (!sd->ewk_view_javascript_suspend)
2059      sd->ewk_view_javascript_suspend = (void (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_javascript_suspend");
2060    sd->ewk_view_javascript_suspend(webview);
2061
2062    // render suspend
2063    if (!sd->ewk_view_disable_render)
2064      sd->ewk_view_disable_render = (Eina_Bool (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_disable_render");
2065    sd->ewk_view_disable_render(webview);
2066
2067    // plugin suspend
2068    if (!sd->ewk_view_setting_enable_plugins_get)
2069      sd->ewk_view_setting_enable_plugins_get = (Eina_Bool (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_setting_enable_plugins_get");
2070    if (sd->ewk_view_setting_enable_plugins_get(webview))
2071      {
2072         if (!sd->ewk_view_pause_and_or_hide_plugins)
2073           sd->ewk_view_pause_and_or_hide_plugins = (void (*)(Evas_Object *, Eina_Bool, Eina_Bool))dlsym(ewk_handle, "ewk_view_pause_and_or_hide_plugins");
2074         sd->ewk_view_pause_and_or_hide_plugins(webview, EINA_FALSE, hidePlugin);
2075      }
2076
2077    // cancel pre-render
2078    if (sd->tiled)
2079      {
2080         if (!sd->ewk_view_pre_render_cancel)
2081           sd->ewk_view_pre_render_cancel = (void (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_pre_render_cancel");
2082         sd->ewk_view_pre_render_cancel(webview);
2083      }
2084
2085    // network suspend
2086    if (!sd->ewk_view_suspend_request)
2087      sd->ewk_view_suspend_request = (Eina_Bool (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_suspend_request");
2088    sd->ewk_view_suspend_request(webview); // suspend network loading
2089
2090 }
2091
2092 static void
2093 _resume_all(Smart_Data *sd, Eina_Bool hidePlugin)
2094 {
2095    Evas_Object *webview = sd->base.self;
2096
2097    // js resume
2098    if (!sd->ewk_view_javascript_resume)
2099      sd->ewk_view_javascript_resume = (void (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_javascript_resume");
2100    sd->ewk_view_javascript_resume(webview);
2101
2102    // render resume
2103    if (sd->tiled)
2104      {
2105         if (!sd->ewk_view_enable_render)
2106           sd->ewk_view_enable_render = (Eina_Bool (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_enable_render");
2107         sd->ewk_view_enable_render(webview);
2108      }
2109
2110    // plugin resume
2111    if (!sd->ewk_view_pause_and_or_hide_plugins)
2112      sd->ewk_view_pause_and_or_hide_plugins = (void (*)(Evas_Object *, Eina_Bool, Eina_Bool))dlsym(ewk_handle, "ewk_view_pause_and_or_hide_plugins");
2113    sd->ewk_view_pause_and_or_hide_plugins(webview, EINA_FALSE, hidePlugin);
2114
2115    // network resume
2116    if (!sd->ewk_view_resume_request)
2117      sd->ewk_view_resume_request = (Eina_Bool (*)(Evas_Object *))dlsym(ewk_handle, "ewk_view_resume_request");
2118    sd->ewk_view_resume_request(webview);
2119 }
2120
2121 static void
2122 _zoom_start(Smart_Data* sd, int centerX, int centerY, int distance)
2123 {
2124    DBG("%s\n", __func__);
2125    sd->zoom.basis.x = centerX;
2126    sd->zoom.basis.y = centerY;
2127    sd->zoom.finger_distance = distance;
2128    sd->zoom.zooming_level = 0;
2129    sd->on_zooming = EINA_TRUE;
2130    if (!sd->ewk_view_zoom_get)
2131      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
2132    sd->zoom.zoom_rate_at_start = sd->ewk_view_zoom_get(sd->base.self);
2133    sd->zoom.zooming_rate = sd->zoom.zoom_rate_at_start;
2134
2135    _suspend_all(sd, EINA_TRUE);
2136
2137    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
2138      _text_selection_hide(sd);
2139 }
2140
2141 static void
2142 _zoom_move(Smart_Data* sd, int centerX, int centerY, int distance)
2143 {
2144    if (sd->on_zooming == EINA_FALSE) return;
2145    //DBG("%s\n", __func__);
2146
2147    int zoom_distance = distance - sd->zoom.finger_distance;
2148
2149    if (zoom_distance != sd->zoom.zooming_level)
2150      {
2151         float zoom_ratio;
2152
2153         if (sd->use_zoom_bouncing)
2154           {
2155              float min_zoom_rate = sd->zoom.min_zoom_rate * ZOOM_OUT_BOUNCING;
2156              if (min_zoom_rate <= 0) min_zoom_rate = MIN_ZOOM_RATIO;
2157              float max_zoom_rate = sd->zoom.max_zoom_rate * ZOOM_IN_BOUNCING;
2158
2159              if (sd->zoom.zooming_rate < sd->zoom.min_zoom_rate)
2160                {
2161                   float step = (sd->zoom.min_zoom_rate - min_zoom_rate) / (float)BOUNCING_DISTANCE;
2162                   zoom_ratio = sd->zoom.zooming_rate + (zoom_distance - sd->zoom.zooming_level) * step;
2163                }
2164              else if (sd->zoom.zooming_rate > sd->zoom.max_zoom_rate)
2165                {
2166                   float step = (max_zoom_rate - sd->zoom.max_zoom_rate) / (float)BOUNCING_DISTANCE;
2167                   zoom_ratio = sd->zoom.zooming_rate + (zoom_distance - sd->zoom.zooming_level) * step;
2168                }
2169              else
2170                {
2171                   zoom_ratio = sd->zoom.zoom_rate_at_start + zoom_distance * ZOOM_STEP_PER_PIXEL;
2172                }
2173
2174              if (zoom_ratio < min_zoom_rate)
2175                zoom_ratio = min_zoom_rate;
2176              if (zoom_ratio > max_zoom_rate)
2177                zoom_ratio = max_zoom_rate;
2178           }
2179         else
2180           {
2181              zoom_ratio = sd->zoom.zoom_rate_at_start + zoom_distance * ZOOM_STEP_PER_PIXEL;
2182              if (zoom_ratio < sd->zoom.min_zoom_rate)
2183                zoom_ratio = sd->zoom.min_zoom_rate;
2184              if (zoom_ratio > sd->zoom.max_zoom_rate)
2185                zoom_ratio = sd->zoom.max_zoom_rate;
2186           }
2187         sd->zoom.zooming_level = zoom_distance;
2188         sd->zoom.zooming_rate = zoom_ratio;
2189
2190         //printf("new zoom : %f, (%d, %d)\n", zoom_ratio, centerX, centerY);
2191         if (!sd->ewk_view_zoom_weak_set)
2192           sd->ewk_view_zoom_weak_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_weak_set");
2193         sd->ewk_view_zoom_weak_set(sd->base.self, zoom_ratio, sd->zoom.basis.x, sd->zoom.basis.y);
2194         DBG("<< zoom weak set [%f] >>\n", zoom_ratio);
2195      }
2196 }
2197
2198 static void
2199 _zoom_stop(Smart_Data* sd)
2200 {
2201    sd->on_zooming = EINA_FALSE;
2202    DBG("%s ( %d )\n", __func__, sd->zoom.zooming_level);
2203    if (sd->zoom.zooming_level == 0) return;
2204
2205    sd->zoom.zoom_rate_to_set = sd->zoom.zooming_rate;
2206    if (sd->zoom.zoom_rate_to_set < sd->zoom.min_zoom_rate)
2207      sd->zoom.zoom_rate_to_set = sd->zoom.min_zoom_rate;
2208    if (sd->zoom.zoom_rate_to_set > sd->zoom.max_zoom_rate)
2209      sd->zoom.zoom_rate_to_set = sd->zoom.max_zoom_rate;
2210    if (sd->use_zoom_bouncing
2211          && (sd->zoom.zoom_rate_to_set != sd->zoom.zooming_rate))
2212      {
2213         sd->zoom.zoom_rate_at_start = sd->zoom.zooming_rate;
2214         smart_zoom_index = N_COSINE - 1;
2215         ecore_animator_frametime_set(1.0 / ZOOM_FRAMERATE);
2216         sd->smart_zoom_animator = ecore_animator_add(_smart_zoom_animator, sd);
2217      }
2218    else
2219      {
2220         if (!sd->ewk_view_zoom_set)
2221           sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
2222         sd->ewk_view_zoom_set(sd->base.self, sd->zoom.zoom_rate_to_set, sd->zoom.basis.x, sd->zoom.basis.y);
2223      }
2224    DBG("<< zoom set [%f] >>\n", sd->zoom.zooming_rate);
2225
2226    _resume_all(sd, EINA_FALSE);
2227
2228    if (sd->tiled)
2229      {
2230         if (!sd->ewk_view_tiled_unused_cache_get)
2231           sd->ewk_view_tiled_unused_cache_get = (Ewk_Tile_Unused_Cache *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_get");
2232         Ewk_Tile_Unused_Cache* ewk_tile_cache = sd->ewk_view_tiled_unused_cache_get(sd->base.self);
2233         if (!sd->ewk_tile_unused_cache_auto_flush)
2234           sd->ewk_tile_unused_cache_auto_flush = (void (*)(Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_auto_flush");
2235         sd->ewk_tile_unused_cache_auto_flush(ewk_tile_cache);
2236         _directional_pre_render(sd->base.self, 0, 0);
2237      }
2238
2239    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
2240      {
2241         if (!sd->ewk_view_frame_main_get)
2242           sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2243         if (!sd->ewk_frame_selection_handlers_get)
2244           sd->ewk_frame_selection_handlers_get = (Eina_Bool (*)(Evas_Object *, int *, int *, int *, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_selection_handlers_get");
2245         int tx, ty, th, bx, by, bh;
2246         sd->ewk_frame_selection_handlers_get(sd->ewk_view_frame_main_get(sd->base.self), &tx, &ty, &th, &bx, &by, &bh);
2247         _coords_ewk_to_evas(sd->base.self, tx, ty, &tx, &ty);
2248         _coords_ewk_to_evas(sd->base.self, bx, by, &bx, &by);
2249         _text_selection_show();
2250         _text_selection_set_front_info(sd, tx, ty, th);
2251         _text_selection_set_back_info(sd, bx, by, bh);
2252      }
2253 }
2254
2255 static void
2256 _adjust_to_contents_boundary(Evas_Object* obj, int* to_x, int* to_y,
2257       int from_x, int from_y, float new_zoom_rate)
2258 {
2259    INTERNAL_ENTRY;
2260    // get view's geometry
2261    int view_x, view_y, view_w, view_h;
2262    evas_object_geometry_get(obj, &view_x, &view_y, &view_w, &view_h);
2263
2264    // get contentsSize
2265    if (!sd->ewk_view_frame_main_get)
2266      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2267    if (!sd->ewk_frame_contents_size_get)
2268      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
2269
2270    int contents_w, contents_h;
2271    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(obj), &contents_w, &contents_h);
2272    if (!sd->ewk_view_zoom_get)
2273      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
2274    float current_zoom_rate = sd->ewk_view_zoom_get(obj);
2275    if (!sd->ewk_view_zoom_cairo_scaling_get)
2276      sd->ewk_view_zoom_cairo_scaling_get = (Eina_Bool (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_cairo_scaling_get");
2277    if (sd->ewk_view_zoom_cairo_scaling_get(obj))
2278      {
2279         contents_w *= current_zoom_rate;
2280         contents_h *= current_zoom_rate;
2281      }
2282
2283    // check boundary - should not exceed the left, right, top and bottom of contents after zoom
2284    float zoom_step = new_zoom_rate / current_zoom_rate;
2285    int ewk_from_x, ewk_from_y;
2286    _coords_evas_to_ewk(obj, from_x, from_y, &ewk_from_x, &ewk_from_y);
2287    int contents_left = ewk_from_x * zoom_step; // left contents size of from
2288    int contents_right = contents_w * zoom_step - contents_left; // right contents size of from
2289    int screen_left = (*to_x) - view_x;
2290    int screen_right = view_w - screen_left;
2291    if (contents_left < screen_left)
2292      (*to_x) -= (screen_left - contents_left);
2293    else if (contents_right < screen_right)
2294      (*to_x) += (screen_right - contents_right);
2295    int contents_top = ewk_from_y * zoom_step; // top contents size of from
2296    int contents_bottom = contents_h * zoom_step - contents_top; // bottom contents size of from
2297    int screen_top = (*to_y) - view_y;
2298    int screen_bottom = view_h - screen_top;
2299    if (contents_top < screen_top)
2300      (*to_y) -= (screen_top - contents_top);
2301    else if (contents_bottom < screen_bottom)
2302      (*to_y) += (screen_bottom - contents_bottom);
2303 }
2304
2305 static int
2306 _smart_zoom_animator(void* data)
2307 {
2308    Smart_Data* sd = (Smart_Data*)data;
2309
2310    if (!sd->ewk_view_frame_main_get)
2311      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2312
2313    // stop
2314    if (smart_zoom_index < 0)
2315      {
2316         if (!sd->ewk_view_zoom_set)
2317           sd->ewk_view_zoom_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_set");
2318         sd->ewk_view_zoom_set(sd->base.self, sd->zoom.zoom_rate_to_set,
2319               sd->zoom.basis.x, sd->zoom.basis.y);
2320         if (sd->smart_zoom_animator)
2321           {
2322              ecore_animator_del(sd->smart_zoom_animator);
2323              sd->smart_zoom_animator = NULL;
2324           }
2325
2326         _elm_smart_touch_start(sd->touch_obj);
2327
2328         _resume_all(sd, EINA_FALSE);
2329
2330         if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
2331           {
2332              if (!sd->ewk_frame_selection_handlers_get)
2333                sd->ewk_frame_selection_handlers_get = (Eina_Bool (*)(Evas_Object *, int *, int *, int *, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_selection_handlers_get");
2334              int tx, ty, th, bx, by, bh;
2335              sd->ewk_frame_selection_handlers_get(sd->ewk_view_frame_main_get(sd->base.self),
2336                    &tx, &ty, &th, &bx, &by, &bh);
2337              _coords_ewk_to_evas(sd->base.self, tx, ty, &tx, &ty);
2338              _coords_ewk_to_evas(sd->base.self, bx, by, &bx, &by);
2339              _text_selection_show();
2340              _text_selection_set_front_info(sd, tx, ty, th);
2341              _text_selection_set_back_info(sd, bx, by, bh);
2342           }
2343
2344         return ECORE_CALLBACK_CANCEL;
2345      }
2346
2347    if (sd->zoom.zoom_rate_at_start != sd->zoom.zoom_rate_to_set)
2348      {
2349         // weak zoom
2350         float zoom_rate = sd->zoom.zoom_rate_at_start
2351            + ((sd->zoom.zoom_rate_to_set - sd->zoom.zoom_rate_at_start) * cosine[smart_zoom_index]);
2352         if (!sd->ewk_view_zoom_weak_set)
2353           sd->ewk_view_zoom_weak_set = (Eina_Bool (*)(Evas_Object *, float, Evas_Coord, Evas_Coord))dlsym(ewk_handle, "ewk_view_zoom_weak_set");
2354         if (zoom_rate <= sd->zoom.min_zoom_rate)
2355           {
2356              if (!sd->ewk_frame_scroll_pos_get)
2357                sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
2358              if (!sd->ewk_view_zoom_get)
2359                sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
2360              int scroll_x, scroll_y;
2361              sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(sd->base.self), &scroll_x, &scroll_y);
2362              float current_zoom_rate = sd->ewk_view_zoom_get(sd->base.self);
2363              int center_x = (scroll_x * sd->zoom.zoom_rate_to_set * current_zoom_rate)
2364                             / (current_zoom_rate - sd->zoom.zoom_rate_to_set);
2365              int center_y = (scroll_y * sd->zoom.zoom_rate_to_set * current_zoom_rate)
2366                             / (current_zoom_rate - sd->zoom.zoom_rate_to_set);
2367
2368              int basis_x = sd->zoom.basis.x + (center_x - sd->zoom.basis.x) * cosine[smart_zoom_index];
2369              int basis_y = sd->zoom.basis.y + (center_y - sd->zoom.basis.y) * cosine[smart_zoom_index];
2370              sd->ewk_view_zoom_weak_set(sd->base.self, zoom_rate, basis_x, basis_y);
2371              smart_zoom_index--; // in order to make zoom bouncing more faster
2372           }
2373         if (zoom_rate >= sd->zoom.max_zoom_rate)
2374           {
2375              sd->ewk_view_zoom_weak_set(sd->base.self, zoom_rate, sd->zoom.basis.x, sd->zoom.basis.y);
2376              smart_zoom_index--; // in order to make zoom bouncing more faster
2377           }
2378         else
2379           {
2380              sd->ewk_view_zoom_weak_set(sd->base.self, zoom_rate, sd->zoom.basis.x, sd->zoom.basis.y);
2381           }
2382      } else {
2383           if (!sd->ewk_frame_scroll_pos_get)
2384             sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
2385           // save old scroll positions
2386           int current_scroll_x, current_scroll_y;
2387           sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(sd->base.self), &current_scroll_x, &current_scroll_y);
2388
2389           // get to set position
2390           int to_set_x = sd->zoom.scroll_at_start.x
2391              + (sd->zoom.scroll_to_set.x - sd->zoom.scroll_at_start.x) * cosine[smart_zoom_index];
2392           int to_set_y = sd->zoom.scroll_at_start.y
2393              + (sd->zoom.scroll_to_set.y - sd->zoom.scroll_at_start.y) * cosine[smart_zoom_index];
2394
2395           if (!sd->ewk_frame_scroll_add)
2396             sd->ewk_frame_scroll_add = (Eina_Bool (*)(Evas_Object *, int, int))dlsym(ewk_handle, "ewk_frame_scroll_add");
2397           // scroll
2398           sd->ewk_frame_scroll_add(sd->ewk_view_frame_main_get(sd->base.self),
2399                 to_set_x - current_scroll_x, to_set_y - current_scroll_y);
2400      }
2401    smart_zoom_index--;
2402
2403    return ECORE_CALLBACK_RENEW;
2404 }
2405
2406 static void
2407 _smart_cb_pinch_zoom_start(void* data, Evas_Object* webview, void* event_info)
2408 {
2409    //DBG("%s\n", __func__);
2410    Smart_Data *sd = (Smart_Data *)data;
2411    if (!sd) return;
2412
2413    Evas_Point* arr = (Evas_Point*) event_info;
2414    int centerX = (arr[0].x + arr[1].x) / 2;
2415    int centerY = (arr[0].y + arr[1].y) / 2;
2416    int dx = arr[0].x - arr[1].x;
2417    int dy = arr[0].y - arr[1].y;
2418    int distance = sqrt((double)(dx * dx + dy * dy));
2419    _zoom_start(sd, centerX, centerY, distance);
2420 }
2421
2422 static void
2423 _smart_cb_pinch_zoom_move(void* data, Evas_Object* webview, void* event_info)
2424 {
2425    //DBG("%s\n", __func__);
2426    Smart_Data *sd = (Smart_Data *)data;
2427    if (!sd) return;
2428
2429    Evas_Point* arr = (Evas_Point*) event_info;
2430    int centerX = (arr[0].x + arr[1].x) / 2;
2431    int centerY = (arr[0].y + arr[1].y) / 2;
2432    int dx = arr[0].x - arr[1].x;
2433    int dy = arr[0].y - arr[1].y;
2434    int distance = sqrt((double)(dx * dx + dy * dy));
2435    _zoom_move(sd, centerX, centerY, distance);
2436 }
2437
2438 static void
2439 _smart_cb_pinch_zoom_stop(void* data, Evas_Object* webview, void* event_info)
2440 {
2441    //DBG("%s\n", __func__);
2442    Smart_Data *sd = (Smart_Data *)data;
2443    if (!sd) return;
2444
2445    _zoom_stop(sd);
2446    _minimap_update(sd->minimap.content, sd, sd->thumbnail, sd->minimap.cw, sd->minimap.ch);
2447
2448    if (sd->tiled)
2449      {
2450         if (!sd->ewk_view_tiled_unused_cache_get)
2451           sd->ewk_view_tiled_unused_cache_get = (Ewk_Tile_Unused_Cache *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_tiled_unused_cache_get");
2452         Ewk_Tile_Unused_Cache *tuc = sd->ewk_view_tiled_unused_cache_get(webview);
2453         if (!sd->ewk_tile_unused_cache_used_get)
2454           sd->ewk_tile_unused_cache_used_get = (size_t (*)(const Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_used_get");
2455         size_t used = sd->ewk_tile_unused_cache_used_get(tuc);
2456         if (!sd->ewk_tile_unused_cache_max_get)
2457           sd->ewk_tile_unused_cache_max_get = (size_t (*)(const Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_max_get");
2458         size_t max = sd->ewk_tile_unused_cache_max_get(tuc);
2459         DBG("[%s] max = %d  used = %d \n", __func__, max, used);
2460         if (used > max)
2461           {
2462              if (!sd->ewk_tile_unused_cache_auto_flush)
2463                sd->ewk_tile_unused_cache_auto_flush = (void (*)(Ewk_Tile_Unused_Cache *))dlsym(ewk_handle, "ewk_tile_unused_cache_auto_flush");
2464              sd->ewk_tile_unused_cache_auto_flush(tuc);
2465           }
2466      }
2467 }
2468
2469 static void
2470 _smart_cb_vertical_zoom_start(void* data, Evas_Object* webview, void* event_info)
2471 {
2472    DBG("%s\n", __func__);
2473    Smart_Data *sd = (Smart_Data *)data;
2474    if (!sd) return;
2475
2476    Evas_Point* arr = (Evas_Point*) event_info;
2477    int centerX = (arr[0].x + arr[1].x) / 2;
2478    int centerY = (arr[0].y + arr[1].y) / 2;
2479    //int dx = arr[0].x - arr[1].x;
2480    //int dy = arr[0].y - arr[1].y;
2481    //int distance = sqrt((double)(dx * dx + dy * dy));
2482    _zoom_start(sd, centerX, centerY, centerY);
2483 }
2484
2485 static void
2486 _smart_cb_vertical_zoom_move(void* data, Evas_Object* webview, void* event_info)
2487 {
2488    DBG("%s\n", __func__);
2489    Smart_Data *sd = (Smart_Data *)data;
2490    if (!sd) return;
2491
2492    Evas_Point* arr = (Evas_Point*) event_info;
2493    int centerX = (arr[0].x + arr[1].x) / 2;
2494    int centerY = (arr[0].y + arr[1].y) / 2;
2495    //int dx = arr[0].x - arr[1].x;
2496    //int dy = arr[0].y - arr[1].y;
2497    //int distance = centerY - sd->zoom.cy;
2498    _zoom_move(sd, centerX, centerY, centerY);
2499 }
2500
2501 static void
2502 _smart_cb_vertical_zoom_stop(void* data, Evas_Object* webview, void* event_info)
2503 {
2504    DBG("%s\n", __func__);
2505    Smart_Data *sd = (Smart_Data *)data;
2506    if (!sd) return;
2507
2508    _zoom_stop(sd);
2509    _minimap_update(sd->minimap.content, sd, sd->thumbnail, sd->minimap.cw, sd->minimap.ch);
2510 }
2511
2512 static void
2513 _smart_cb_smart_zoom(void* data, Evas_Object* webview, void* event_info)
2514 {
2515    DBG("%s\n", __func__);
2516    Smart_Data *sd = (Smart_Data *)data;
2517    if (!sd) return;
2518    Evas_Point* point = (Evas_Point*)event_info;
2519
2520    if (sd->events_feed == EINA_TRUE) return;
2521
2522    if (!sd->ewk_view_frame_main_get)
2523      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2524
2525    _elm_smart_touch_stop(sd->touch_obj);
2526
2527    // get rect
2528    int ewk_x = 0, ewk_y = 0;
2529    Eina_Rectangle rect;
2530    _coords_evas_to_ewk(webview, point->x, point->y, &ewk_x, &ewk_y);
2531    if (!sd->ewk_view_get_smart_zoom_rect)
2532      sd->ewk_view_get_smart_zoom_rect = (Eina_Bool (*)(Evas_Object *, int, int, const Evas_Event_Mouse_Up *, Eina_Rectangle *))dlsym(ewk_handle, "ewk_view_get_smart_zoom_rect");
2533    sd->ewk_view_get_smart_zoom_rect(webview, ewk_x, ewk_y, &sd->mouse_up_copy, &rect);
2534
2535    // calculate zoom_rate and center of rect
2536    int view_x, view_y, view_w, view_h;
2537    evas_object_geometry_get(webview, &view_x, &view_y, &view_w, &view_h);
2538    if (!sd->ewk_view_zoom_get)
2539      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
2540    float current_zoom_rate = sd->ewk_view_zoom_get(webview);
2541    float zoom_rate;
2542    int rect_center_x, rect_center_y;
2543    if (rect.w)
2544      {
2545         zoom_rate = current_zoom_rate * (float)view_w / (float)rect.w;
2546         _coords_ewk_to_evas(webview, rect.x + (rect.w >> 1), rect.y + (rect.h >> 1), &rect_center_x, &rect_center_y);
2547         if ((rect.h / current_zoom_rate) * zoom_rate > view_h)
2548           {
2549              rect_center_y = point->y;
2550           }
2551         // check zoom rate
2552         if (zoom_rate < sd->zoom.min_zoom_rate)
2553           zoom_rate = sd->zoom.min_zoom_rate;
2554         if (zoom_rate > sd->zoom.max_zoom_rate)
2555           zoom_rate = sd->zoom.max_zoom_rate;
2556         if (zoom_rate == current_zoom_rate)
2557           zoom_rate = sd->zoom.min_zoom_rate;
2558      } else {
2559           zoom_rate = sd->zoom.min_zoom_rate;
2560           rect_center_x = point->x;
2561           rect_center_y = point->y;
2562      }
2563
2564    // set zooming data
2565    float zoom_step = zoom_rate / current_zoom_rate;
2566    int center_x = view_x + (view_w >> 1);
2567    int center_y = view_y + (view_h >> 1);
2568
2569    _adjust_to_contents_boundary(webview, &center_x, &center_y, rect_center_x, rect_center_y, zoom_rate);
2570
2571    // set data for smart zoom
2572    sd->zoom.basis.x = (center_x - zoom_step * rect_center_x) / (1 - zoom_step);
2573    sd->zoom.basis.y = (center_y - zoom_step * rect_center_y) / (1 - zoom_step) - view_y;
2574    sd->zoom.zoom_rate_at_start = current_zoom_rate;
2575    sd->zoom.zoom_rate_to_set = zoom_rate;
2576    smart_zoom_index = N_COSINE - 1;
2577
2578    _suspend_all(sd, EINA_TRUE);
2579
2580    // run animator
2581    ecore_animator_frametime_set(1.0 / ZOOM_FRAMERATE);
2582    sd->smart_zoom_animator = ecore_animator_add(_smart_zoom_animator, sd);
2583
2584    // hide textSelection handlers during zooming
2585    if (sd->use_text_selection == EINA_TRUE && sd->text_selection_on == EINA_TRUE)
2586      _text_selection_hide(sd);
2587 }
2588
2589 static void
2590 _zoom_to_rect(Smart_Data *sd, int x, int y)
2591 {
2592    DBG("%s\n", __func__);
2593    Evas_Object *webview = sd->base.self;
2594
2595    if (!sd->ewk_view_frame_main_get)
2596      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2597
2598    // performing a hit test
2599    _coords_evas_to_ewk(webview, x, y, &x, &y);
2600    if (!sd->ewk_frame_hit_test_new)
2601      sd->ewk_frame_hit_test_new = (Ewk_Hit_Test * (*)(const Evas_Object *, int, int))dlsym(ewk_handle, "ewk_frame_hit_test_new");
2602    Ewk_Hit_Test *hit_test = sd->ewk_frame_hit_test_new(sd->ewk_view_frame_main_get(webview), x, y);
2603
2604    // calculate zoom_rate and center of rect
2605    if (hit_test->bounding_box.w && hit_test->bounding_box.h)
2606      {
2607         // set zooming data
2608         float zoom_rate = INPUT_ZOOM_RATIO;
2609         if (!sd->ewk_view_zoom_get)
2610           sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
2611         float current_zoom_rate = sd->ewk_view_zoom_get(webview);
2612         float zoom_step = zoom_rate / current_zoom_rate;
2613
2614         // get position to move from
2615         int view_x, view_y, view_w, view_h;
2616         evas_object_geometry_get(webview, &view_x, &view_y, &view_w, &view_h);
2617         int from_x, from_y;
2618         _coords_ewk_to_evas(webview, hit_test->bounding_box.x, hit_test->bounding_box.y, &from_x, &from_y);
2619         from_x = from_x + ((view_w - INPUT_LOCATION_X) / 2) / zoom_step;
2620         from_y = from_y + hit_test->bounding_box.h / 2;
2621
2622         // get position to move to
2623         int to_x = view_x + INPUT_LOCATION_X + (view_w - INPUT_LOCATION_X) / 2;
2624         int to_y = view_y + INPUT_LOCATION_Y + (hit_test->bounding_box.h / 2) * zoom_step;
2625
2626         // adjust to contents
2627         _adjust_to_contents_boundary(webview, &to_x, &to_y, from_x, from_y, zoom_rate);
2628
2629         // set data for smart zoom
2630         sd->zoom.basis.x = (to_x - zoom_step * from_x) / (1 - zoom_step);
2631         sd->zoom.basis.y = (to_y - zoom_step * from_y) / (1 - zoom_step) - view_y;
2632         sd->zoom.zoom_rate_at_start = current_zoom_rate;
2633         sd->zoom.zoom_rate_to_set = zoom_rate;
2634         if (!sd->ewk_frame_scroll_pos_get)
2635           sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
2636         sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(webview),
2637               &sd->zoom.scroll_at_start.x, &sd->zoom.scroll_at_start.y);
2638         sd->zoom.scroll_to_set.x = sd->zoom.scroll_at_start.x + (from_x - to_x);
2639         sd->zoom.scroll_to_set.y = sd->zoom.scroll_at_start.y + (from_y - to_y);
2640         smart_zoom_index = N_COSINE - 1;
2641
2642         _suspend_all(sd, EINA_TRUE);
2643
2644         // run animator
2645         ecore_animator_frametime_set(1.0 / ZOOM_FRAMERATE);
2646         sd->smart_zoom_animator = ecore_animator_add(_smart_zoom_animator, sd);
2647      }
2648
2649    if (!sd->ewk_frame_hit_test_free)
2650      sd->ewk_frame_hit_test_free = (void (*)(Ewk_Hit_Test *))dlsym(ewk_handle, "ewk_frame_hit_test_free");
2651    sd->ewk_frame_hit_test_free(hit_test);
2652 }
2653
2654 // text-selection
2655 #define BAR_WIDTH            4
2656 #define BAR_HEIGHT           10
2657 #define HANDLE_WIDTH         60
2658 #define HANDLE_HEIGHT        60
2659 #define HANDLE_PRESS_RANGE   50
2660 #define HANDLE_MIDDLE_LENGTH 60
2661
2662 static Evas_Object* front_bar_icon;
2663 static Evas_Object* front_handle_icon;
2664 static Evas_Object* back_bar_icon;
2665 static Evas_Object* back_handle_icon;
2666
2667 static Eina_Bool initialized = EINA_FALSE;
2668
2669 static void
2670 _text_selection_init(Evas_Object* parent)
2671 {
2672    DBG("<< %s >>\n", __FUNCTION__);
2673
2674    if (initialized)
2675      return;
2676
2677    // front bar
2678    front_bar_icon = (Evas_Object*)elm_icon_add(parent);
2679    elm_icon_standard_set(front_bar_icon, "webview/ts_bar");
2680    elm_icon_scale_set(front_bar_icon, true, true);
2681    evas_object_pass_events_set(front_bar_icon, true);
2682
2683    // front handle
2684    front_handle_icon = (Evas_Object*)elm_icon_add(parent);
2685    elm_icon_standard_set(front_handle_icon, "webview/ts_handle_front");
2686    elm_icon_scale_set(front_handle_icon, false, false);
2687    evas_object_pass_events_set(front_handle_icon, true);
2688
2689    // back bar
2690    back_bar_icon = (Evas_Object*)elm_icon_add(parent);
2691    elm_icon_standard_set(back_bar_icon, "webview/ts_bar");
2692    elm_icon_scale_set(back_bar_icon, true, true);
2693    evas_object_pass_events_set(back_bar_icon, true);
2694
2695    // back handle
2696    back_handle_icon = (Evas_Object*)elm_icon_add(parent);
2697    elm_icon_standard_set(back_handle_icon, "webview/ts_handle_back");
2698    elm_icon_scale_set(back_handle_icon, false, false);
2699    evas_object_pass_events_set(back_handle_icon, true);
2700
2701    initialized = EINA_TRUE;
2702 }
2703
2704 static void
2705 _text_selection_show(void)
2706 {
2707    evas_object_show(front_bar_icon);
2708    evas_object_show(front_handle_icon);
2709    evas_object_show(back_bar_icon);
2710    evas_object_show(back_handle_icon);
2711 }
2712
2713 static void
2714 _text_selection_hide(Smart_Data *sd)
2715 {
2716    evas_object_hide(front_bar_icon);
2717    evas_object_hide(front_handle_icon);
2718    evas_object_hide(back_bar_icon);
2719    evas_object_hide(back_handle_icon);
2720
2721    sd->text_selection.front.x = -1;
2722    sd->text_selection.front.y = -1;
2723    sd->text_selection.front.h = -1;
2724    sd->text_selection.front_handle.x = -1;
2725    sd->text_selection.front_handle.y = -1;
2726    sd->text_selection.back.x = -1;
2727    sd->text_selection.back.y = -1;
2728    sd->text_selection.back.h = -1;
2729    sd->text_selection.back_handle.x = -1;
2730    sd->text_selection.back_handle.y = -1;
2731 }
2732
2733 static void
2734 _text_selection_set_front_info(Smart_Data *sd, int x, int y, int height)
2735 {
2736    Evas_Object *webview = sd->base.self;
2737
2738    Evas_Coord_Rectangle* front = &(sd->text_selection.front);
2739    Evas_Point* front_handle = &(sd->text_selection.front_handle);
2740
2741    front->h = height;
2742    int front_bar_height = height + HANDLE_MIDDLE_LENGTH + HANDLE_HEIGHT;
2743
2744    // set size
2745    evas_object_resize(front_bar_icon, BAR_WIDTH, front_bar_height);
2746    evas_object_resize(front_handle_icon, HANDLE_WIDTH, HANDLE_HEIGHT);
2747
2748    // set location
2749    front_handle->x = x - (HANDLE_WIDTH / 2);
2750    int win_y, win_height, win_bottom;
2751    evas_object_geometry_get(webview, NULL, &win_y, NULL, &win_height);
2752    win_bottom = win_y + win_height;
2753    if ((front_handle->y == -1 && (y + front_bar_height > win_bottom))
2754          || ((front_handle->y < front->y) && (y + front->h - front_bar_height > win_y))
2755          || ((front_handle->y > front->y) && (y + front_bar_height > win_bottom)))
2756      { // upper handle
2757         front_handle->y = y + front->h - front_bar_height + (HANDLE_HEIGHT / 2);
2758         evas_object_move(front_bar_icon, x, y + front->h - front_bar_height);
2759         evas_object_move(front_handle_icon, x - HANDLE_WIDTH, y + front->h - front_bar_height);
2760
2761      }
2762    else
2763      { // lower handle
2764         front_handle->y = y + front_bar_height - (HANDLE_HEIGHT / 2);
2765         evas_object_move(front_bar_icon, x, y);
2766         evas_object_move(front_handle_icon, x - HANDLE_WIDTH, front_handle->y - (HANDLE_HEIGHT / 2));
2767      }
2768
2769    front->x = x;
2770    front->y = y;
2771 }
2772
2773 static void
2774 _text_selection_set_back_info(Smart_Data *sd, int x, int y, int height)
2775 {
2776    Evas_Object *webview = sd->base.self;
2777
2778    Evas_Coord_Rectangle* back = &(sd->text_selection.back);
2779    Evas_Point* back_handle = &(sd->text_selection.back_handle);
2780
2781    back->h = height;
2782    int back_bar_height = height + HANDLE_MIDDLE_LENGTH + HANDLE_HEIGHT;
2783
2784    // set size
2785    evas_object_resize(back_bar_icon, BAR_WIDTH, back_bar_height);
2786    evas_object_resize(back_handle_icon, HANDLE_WIDTH, HANDLE_HEIGHT);
2787
2788    // set location
2789    back_handle->x = x + (HANDLE_WIDTH / 2);
2790    int win_y, win_height, win_bottom;
2791    evas_object_geometry_get(webview, NULL, &win_y, NULL, &win_height);
2792    win_bottom = win_y + win_height;
2793    if ((back_handle->y == -1 && (y - back->h + back_bar_height > win_bottom))
2794          || ((back_handle->y < back->y) && (y - back_bar_height > win_y))
2795          || ((back_handle->y > back->y) && (y - back->h + back_bar_height > win_bottom))) { // upper handle
2796         back_handle->y = y - back->h - HANDLE_MIDDLE_LENGTH - (HANDLE_HEIGHT / 2);
2797         evas_object_move(back_bar_icon, x - BAR_WIDTH, y - back_bar_height);
2798         evas_object_move(back_handle_icon, x, back_handle->y - (HANDLE_HEIGHT / 2));
2799
2800    } else {
2801         back_handle->y = y + HANDLE_MIDDLE_LENGTH + (HANDLE_HEIGHT / 2);
2802         evas_object_move(back_bar_icon, x - BAR_WIDTH, y - back->h);
2803         evas_object_move(back_handle_icon, x, back_handle->y - (HANDLE_HEIGHT / 2));
2804    }
2805
2806    back->x = x;
2807    back->y = y;
2808 }
2809
2810 static Eina_Bool
2811 _text_selection_handle_pressed(Smart_Data *sd, int x, int y)
2812 {
2813    Evas_Point front_handle = sd->text_selection.front_handle;
2814    Evas_Point back_handle = sd->text_selection.back_handle;
2815
2816    // check front handle
2817    if (x > (front_handle.x - HANDLE_PRESS_RANGE) && x < (front_handle.x + HANDLE_PRESS_RANGE)
2818          && y > (front_handle.y - HANDLE_PRESS_RANGE) && y < (front_handle.y + HANDLE_PRESS_RANGE))
2819      sd->text_selection.front_handle_moving = EINA_TRUE;
2820
2821    // check back handle
2822    if (x > (back_handle.x - HANDLE_PRESS_RANGE) && x < (back_handle.x + HANDLE_PRESS_RANGE)
2823          && y > (back_handle.y - HANDLE_PRESS_RANGE) && y < (back_handle.y + HANDLE_PRESS_RANGE))
2824      {
2825         if (sd->text_selection.front_handle_moving == EINA_TRUE)
2826           {
2827              if (abs(x - front_handle.x) + abs(y - front_handle.y)
2828                    > abs(x - back_handle.x) + abs(y - back_handle.y))
2829                {
2830                   sd->text_selection.front_handle_moving = EINA_FALSE;
2831                   sd->text_selection.back_handle_moving = EINA_TRUE;
2832                }
2833           }
2834         else
2835           {
2836              sd->text_selection.back_handle_moving = EINA_TRUE;
2837           }
2838      }
2839
2840    return (sd->text_selection.front_handle_moving || sd->text_selection.back_handle_moving);
2841 }
2842
2843 static void
2844 _text_selection_update_position(Smart_Data *sd, int x, int y)
2845 {
2846    Evas_Object *webview = sd->base.self;
2847
2848    Evas_Coord_Rectangle* front = &(sd->text_selection.front);
2849    Evas_Coord_Rectangle* back = &(sd->text_selection.back);
2850
2851    if (!sd->ewk_view_frame_main_get)
2852      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2853
2854    // set selected region with front handle
2855    if (sd->text_selection.front_handle_moving == EINA_TRUE)
2856      {
2857         x = x + (HANDLE_WIDTH >> 1);
2858         if (sd->text_selection.front_handle.y < sd->text_selection.front.y)
2859           y = y + (HANDLE_HEIGHT >> 1) + HANDLE_MIDDLE_LENGTH;
2860         else
2861           y = y - front->h - HANDLE_MIDDLE_LENGTH - (HANDLE_HEIGHT >> 1);
2862
2863         if (y > back->y)
2864           y = back->y - back->h / 2;
2865
2866         if (!sd->ewk_frame_selection_left_set)
2867           sd->ewk_frame_selection_left_set = (Eina_Bool (*)(Evas_Object *, int, int, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_selection_left_set");
2868         int ewkX, ewkY;
2869         _coords_evas_to_ewk(webview, x, y, &ewkX, &ewkY);
2870         if (sd->ewk_frame_selection_left_set(sd->ewk_view_frame_main_get(webview), ewkX, ewkY,
2871                  &front->x, &front->y, &front->h)) {
2872              _coords_ewk_to_evas(webview, front->x, front->y, &front->x, &front->y);
2873              _text_selection_set_front_info(sd, front->x, front->y, front->h);
2874         }
2875
2876         // set selected region with back handle
2877      }
2878    else if (sd->text_selection.back_handle_moving)
2879      {
2880         x = x - (HANDLE_WIDTH >> 1);
2881         if (sd->text_selection.back_handle.y < sd->text_selection.back.y)
2882           y = y + (HANDLE_HEIGHT >> 1) + HANDLE_MIDDLE_LENGTH;
2883         else
2884           y = y - back->h - HANDLE_MIDDLE_LENGTH - (HANDLE_HEIGHT >> 1);
2885
2886         if (y < front->y)
2887           y = front->y + front->h / 2;
2888
2889         if (!sd->ewk_frame_selection_right_set)
2890           sd->ewk_frame_selection_right_set = (Eina_Bool (*)(Evas_Object *, int, int, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_selection_right_set");
2891         int ewkX, ewkY;
2892         _coords_evas_to_ewk(webview, x, y, &ewkX, &ewkY);
2893         if (sd->ewk_frame_selection_right_set(sd->ewk_view_frame_main_get(webview), ewkX, ewkY,
2894                  &back->x, &back->y, &back->h)) {
2895              _coords_ewk_to_evas(webview, back->x, back->y, &back->x, &back->y);
2896              _text_selection_set_back_info(sd, back->x, back->y, back->h);
2897         }
2898      }
2899 }
2900
2901 static void
2902 _text_selection_move_by(Smart_Data *sd, int dx, int dy)
2903 {
2904    _text_selection_set_front_info(sd, sd->text_selection.front.x + dx,
2905          sd->text_selection.front.y + dy,
2906          sd->text_selection.front.h);
2907    _text_selection_set_back_info(sd, sd->text_selection.back.x + dx,
2908          sd->text_selection.back.y + dy,
2909          sd->text_selection.back.h);
2910 }
2911 // minimap
2912 static void
2913 _minimap_update_detail(Evas_Object* minimap, Smart_Data *sd, cairo_surface_t* src, int srcW, int srcH, Eina_Rectangle* visibleRect)
2914 {
2915    void* pixels;
2916    cairo_t* cr;
2917    cairo_surface_t* dest;
2918    cairo_status_t status;
2919
2920    if (!sd->cairo_surface_status)
2921      sd->cairo_surface_status = (cairo_status_t (*)(cairo_surface_t *))dlsym(cairo_handle, "cairo_surface_status");
2922    if (!sd->cairo_image_surface_create_for_data)
2923      sd->cairo_image_surface_create_for_data = (cairo_surface_t * (*)(unsigned char *, cairo_format_t, int, int, int))dlsym(cairo_handle, "cairo_image_surface_create_for_data");
2924
2925    //TODO: check which one is faster
2926    //      1) reuse minimap
2927    //      2) recreate evas_object and set pixel
2928    evas_object_image_size_set(minimap, srcW, srcH);
2929    evas_object_image_fill_set(minimap, 0, 0, srcW, srcH);
2930    evas_object_resize(minimap, srcW, srcH);
2931
2932    pixels = evas_object_image_data_get(minimap, 1);
2933    dest = sd->cairo_image_surface_create_for_data(
2934          (unsigned char*)pixels, CAIRO_FORMAT_RGB24, srcW, srcH, srcW * 4);
2935    status = sd->cairo_surface_status(dest);
2936    if (status != CAIRO_STATUS_SUCCESS)
2937      {
2938         printf("[%s] fail to create cairo surface\n", __func__);
2939         goto error_cairo_surface;
2940      }
2941
2942    if (!sd->cairo_create)
2943      sd->cairo_create = (cairo_t * (*)(cairo_surface_t *))dlsym(cairo_handle, "cairo_create");
2944    cr = sd->cairo_create(dest);
2945    status = sd->cairo_surface_status(dest);
2946    if (status != CAIRO_STATUS_SUCCESS)
2947      {
2948         printf("[%s] fail to create cairo\n", __func__);
2949         goto error_cairo;
2950      }
2951
2952    if (!sd->cairo_set_source_surface)
2953      sd->cairo_set_source_surface = (void (*)(cairo_t *, cairo_surface_t *, double, double))dlsym(cairo_handle, "cairo_set_source_surface");
2954    if (!sd->cairo_paint)
2955      sd->cairo_paint = (void (*)(cairo_t *))dlsym(cairo_handle, "cairo_paint");
2956    if (!sd->cairo_set_source_rgb)
2957      sd->cairo_set_source_rgb = (void (*)(cairo_t *, double, double, double))dlsym(cairo_handle, "cairo_set_source_rgb");
2958    if (!sd->cairo_rectangle)
2959      sd->cairo_rectangle = (void (*)(cairo_t *, double, double, double, double))dlsym(cairo_handle, "cairo_rectangle");
2960    if (!sd->cairo_set_line_width)
2961      sd->cairo_set_line_width = (void (*)(cairo_t *, double))dlsym(cairo_handle, "cairo_set_line_width");
2962    if (!sd->cairo_stroke)
2963      sd->cairo_stroke = (void (*)(cairo_t *cr))dlsym(cairo_handle, "cairo_stroke"); 
2964    if (!sd->cairo_set_antialias)
2965      sd->cairo_set_antialias = (void (*)(cairo_t *, cairo_antialias_t))dlsym(cairo_handle, "cairo_set_antialias");
2966
2967    sd->cairo_set_antialias(cr, CAIRO_ANTIALIAS_NONE);
2968    sd->cairo_set_source_surface(cr, src, 0, 0);
2969    sd->cairo_paint(cr);
2970    sd->cairo_set_source_rgb(cr, 0, 0, 255);
2971    sd->cairo_set_line_width(cr, 1);
2972    sd->cairo_rectangle(cr,
2973          visibleRect->x, visibleRect->y, visibleRect->w, visibleRect->h);
2974    sd->cairo_stroke(cr);
2975
2976    if (!sd->cairo_destroy)
2977      sd->cairo_destroy = (void (*)(cairo_t *))dlsym(cairo_handle, "cairo_destroy");
2978    sd->cairo_destroy(cr);
2979
2980 error_cairo:
2981    if (!sd->cairo_surface_destroy)
2982      sd->cairo_surface_destroy = (void (*)(cairo_surface_t *))dlsym(cairo_handle, "cairo_surface_destroy");
2983    sd->cairo_surface_destroy(dest);
2984 error_cairo_surface:
2985    evas_object_image_data_set(minimap, pixels);
2986    return;
2987 }
2988
2989 static void
2990 _minimap_update(Evas_Object* minimap, Smart_Data *sd, cairo_surface_t* src, int minimapW, int minimapH)
2991 {
2992    if (minimap == NULL || src == NULL) return;
2993    Evas_Object *webview = sd->base.self;
2994
2995    if (!sd->ewk_view_frame_main_get)
2996      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
2997
2998    if (!sd->ewk_frame_contents_size_get)
2999      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
3000    int cw, ch;
3001    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(webview), &cw, &ch);
3002    if (cw == 0 || ch == 0) return;
3003
3004    if (!sd->ewk_frame_visible_content_geometry_get)
3005      sd->ewk_frame_visible_content_geometry_get = (Eina_Bool (*)(const Evas_Object *, Eina_Bool, int *, int *, int *, int *))dlsym(ewk_handle, "ewk_frame_visible_content_geometry_get");
3006    int x, y, w, h;
3007    sd->ewk_frame_visible_content_geometry_get(
3008          sd->ewk_view_frame_main_get(webview), EINA_FALSE,
3009          &x, &y, &w, &h);
3010    DBG("visible area : %d, %d, %d, %d\n", x, y, w, h);
3011
3012    Eina_Rectangle rect = {
3013         x * minimapW / cw, y * minimapH / ch,
3014         w * minimapW / cw, h * minimapH / ch};
3015    _minimap_update_detail(minimap, sd, src, minimapW, minimapH, &rect);
3016 }
3017
3018 static cairo_surface_t*
3019 _image_clone_get(Smart_Data *sd, int* minimap_w, int* minimap_h)
3020 {
3021    DBG("%s is called\n", __func__);
3022    Evas_Object *webview = sd->base.self;
3023    EWK_VIEW_PRIV_GET_OR_RETURN(sd, priv, NULL);
3024
3025    if (!sd->ewk_view_frame_main_get)
3026      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
3027
3028    if (!sd->ewk_frame_contents_size_get)
3029      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
3030    int w, h;
3031    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(webview), &w, &h);
3032    printf(" W : %d / H : %d\n", w, h);
3033
3034    float x_scale = MINIMAP_WIDTH / (float)w;
3035    float y_scale = MINIMAP_HEIGHT / (float)h;
3036    float scale_factor;
3037    if (x_scale < y_scale)
3038      {
3039         scale_factor = x_scale;
3040         *minimap_w = MINIMAP_WIDTH;
3041         *minimap_h = h * scale_factor;
3042      }
3043    else
3044      {
3045         scale_factor = y_scale;
3046         *minimap_w = w * scale_factor;
3047         *minimap_h = MINIMAP_HEIGHT;
3048      }
3049    printf(" minimap w,h : (%d, %d)\n", *minimap_w, *minimap_h);
3050
3051    if (!sd->ewk_view_paint_contents)
3052      sd->ewk_view_paint_contents = (Eina_Bool (*)(Ewk_View_Private_Data *, cairo_t *, const Eina_Rectangle *))dlsym(ewk_handle, "ewk_view_paint_contents");
3053    if (!sd->cairo_image_surface_create)
3054      sd->cairo_image_surface_create = (cairo_surface_t * (*)(cairo_format_t, int, int))dlsym(cairo_handle, "cairo_image_surface_create");
3055    if (!sd->cairo_create)
3056      sd->cairo_create = (cairo_t * (*)(cairo_surface_t *))dlsym(cairo_handle, "cairo_create");
3057    if (!sd->cairo_destroy)
3058      sd->cairo_destroy = (void (*)(cairo_t *))dlsym(cairo_handle, "cairo_destroy");
3059    if (!sd->cairo_scale)
3060      sd->cairo_scale = (void (*)(cairo_t *, double, double))dlsym(cairo_handle, "cairo_scale");
3061    if (!sd->cairo_surface_write_to_png)
3062      sd->cairo_surface_write_to_png = (cairo_status_t (*)(cairo_surface_t *, const char *))dlsym(cairo_handle, "cairo_surface_write_to_png");
3063    if (!sd->cairo_set_antialias)
3064      sd->cairo_set_antialias = (void (*)(cairo_t *, cairo_antialias_t))dlsym(cairo_handle, "cairo_set_antialias");
3065
3066    cairo_surface_t* ret = sd->cairo_image_surface_create(CAIRO_FORMAT_RGB24, *minimap_w, *minimap_h);
3067    cairo_t* cr = sd->cairo_create(ret);
3068    sd->cairo_set_antialias(cr, CAIRO_ANTIALIAS_NONE);
3069    sd->cairo_scale(cr, scale_factor, scale_factor);
3070    Eina_Rectangle rect = {0, 0, w, h};
3071    sd->ewk_view_paint_contents(priv, cr, &rect);
3072    sd->cairo_destroy(cr);
3073    sd->cairo_surface_write_to_png(ret, "/home/root/test.png");
3074
3075    return ret;
3076 }
3077
3078 // coord
3079 static void
3080 _unzoom_position(Evas_Object* obj, int x, int y, int* ux, int* uy)
3081 {
3082    INTERNAL_ENTRY;
3083    int viewY;
3084    evas_object_geometry_get(obj, NULL, &viewY, NULL, NULL);
3085
3086    if (!sd->ewk_view_zoom_get)
3087      sd->ewk_view_zoom_get = (float (*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_zoom_get");
3088    float zoomRatio = sd->ewk_view_zoom_get(obj);
3089    if (zoomRatio)
3090      {
3091         *ux = x / zoomRatio;
3092         *uy = (y - viewY) / zoomRatio;
3093      }
3094 }
3095
3096 static void
3097 _coords_evas_to_ewk(Evas_Object* obj, int x, int y, int* ux, int* uy)
3098 {
3099    INTERNAL_ENTRY;
3100
3101    if (!sd->ewk_view_frame_main_get)
3102      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
3103
3104    if (!sd->ewk_frame_scroll_pos_get)
3105      sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
3106
3107    int scrollX, scrollY, viewY;
3108    sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(obj), &scrollX, &scrollY);
3109    evas_object_geometry_get(obj, NULL, &viewY, NULL, NULL);
3110    *ux = x + scrollX;
3111    *uy = y + scrollY - viewY;
3112 }
3113
3114 static void
3115 _coords_ewk_to_evas(Evas_Object* obj, int x, int y, int* ux, int* uy)
3116 {
3117    INTERNAL_ENTRY;
3118
3119    if (!sd->ewk_view_frame_main_get)
3120      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
3121
3122    if (!sd->ewk_frame_scroll_pos_get)
3123      sd->ewk_frame_scroll_pos_get = (Eina_Bool (*)(const Evas_Object *, int *, int *))dlsym(ewk_handle, "ewk_frame_scroll_pos_get");
3124
3125    int scrollX, scrollY, viewY;
3126    sd->ewk_frame_scroll_pos_get(sd->ewk_view_frame_main_get(obj), &scrollX, &scrollY);
3127    evas_object_geometry_get(obj, NULL, &viewY, NULL, NULL);
3128    *ux = x - scrollX;
3129    *uy = y - scrollY + viewY;
3130 }
3131 static void
3132 _update_min_zoom_rate(Evas_Object *obj)
3133 {
3134    INTERNAL_ENTRY;
3135
3136    if (!sd->ewk_view_frame_main_get)
3137      sd->ewk_view_frame_main_get = (Evas_Object *(*)(const Evas_Object *))dlsym(ewk_handle, "ewk_view_frame_main_get");
3138    if (!sd->ewk_frame_contents_size_get)
3139      sd->ewk_frame_contents_size_get = (Eina_Bool (*)(const Evas_Object *, Evas_Coord *, Evas_Coord *))dlsym(ewk_handle, "ewk_frame_contents_size_get");
3140    if (!sd->ewk_view_zoom_range_set)
3141      sd->ewk_view_zoom_range_set = (Eina_Bool (*)(Evas_Object *, float, float))dlsym(ewk_handle, "ewk_view_zoom_range_set");
3142
3143    int content_w, object_w;
3144    evas_object_geometry_get(obj, NULL, NULL, &object_w, NULL);
3145    sd->ewk_frame_contents_size_get(sd->ewk_view_frame_main_get(obj), &content_w, NULL);
3146    if (content_w)
3147      sd->zoom.min_zoom_rate = (float)object_w / (float)content_w;
3148
3149    if (sd->use_zoom_bouncing)
3150      {
3151         float min_zoom_rate = sd->zoom.min_zoom_rate * ZOOM_OUT_BOUNCING;
3152         if (min_zoom_rate <= 0) min_zoom_rate = MIN_ZOOM_RATIO;
3153         float max_zoom_rate = sd->zoom.max_zoom_rate * ZOOM_IN_BOUNCING;
3154         sd->ewk_view_zoom_range_set(obj, min_zoom_rate, max_zoom_rate);
3155      }
3156    else
3157      {
3158         sd->ewk_view_zoom_range_set(obj, sd->zoom.min_zoom_rate, sd->zoom.max_zoom_rate);
3159      }
3160 }
3161 static void
3162 _geolocation_permission_callback(void *geolocation_obj, const char* url)
3163 {
3164    printf("\n\n<< %s >>\n\n", __func__);
3165    INTERNAL_ENTRY;
3166
3167    Evas_Object *popup;
3168    int length;
3169    char msg1[] = "The page at ";
3170    char msg2[] = "<br>wants to know where you are.<br>Do you want to share location?";
3171    char *msg = NULL;
3172    int result;
3173
3174    length = strlen(msg1) + strlen(url) + strlen(msg2);
3175    msg = calloc(length + 1, sizeof(char));
3176    strncpy(msg, msg1, strlen(msg1));
3177    strncat(msg, url, strlen(url));
3178    strncat(msg, msg2, strlen(msg2));
3179    msg[length] = '\0';
3180
3181    popup = elm_popup_add(obj);
3182    evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
3183    elm_popup_desc_set(popup, msg);
3184    elm_popup_buttons_add(popup, 2, "Share", ELM_POPUP_RESPONSE_OK,
3185                                    "Don't Share", ELM_POPUP_RESPONSE_CANCEL, NULL);
3186    result = elm_popup_run(popup); // modal dialog
3187    switch (result)
3188      {
3189       case ELM_POPUP_RESPONSE_OK:
3190          if (!sd->ewk_set_geolocation_sharing_allowed)
3191            sd->ewk_set_geolocation_sharing_allowed = (void (*)(void *, Eina_Bool))dlsym(ewk_handle, "ewk_set_geolocation_sharing_allowed");
3192          sd->ewk_set_geolocation_sharing_allowed(geolocation_obj, EINA_TRUE);
3193          break;
3194
3195       case ELM_POPUP_RESPONSE_CANCEL:
3196          if (!sd->ewk_set_geolocation_sharing_allowed)
3197            sd->ewk_set_geolocation_sharing_allowed = (void (*)(void *, Eina_Bool))dlsym(ewk_handle, "ewk_set_geolocation_sharing_allowed");
3198          sd->ewk_set_geolocation_sharing_allowed(geolocation_obj, EINA_FALSE);
3199          break;
3200
3201       default:
3202          break;
3203      }
3204    if (msg)
3205      free(msg);
3206 }
3207 #endif