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