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