Ecore_Evas (Wayland): Quiet down the output by disabling LOGFN.
[profile/ivi/ecore.git] / src / lib / ecore_evas / ecore_evas_wayland_shm.c
1 #ifdef HAVE_CONFIG_H
2 # include "config.h"
3 #endif
4
5 //#define LOGFNS 1
6
7 #ifdef LOGFNS
8 # include <stdio.h>
9 # define LOGFN(fl, ln, fn) \
10    printf("-ECORE_EVAS-WL: %25s: %5i - %s\n", fl, ln, fn);
11 #else
12 # define LOGFN(fl, ln, fn)
13 #endif
14
15 #ifdef BUILD_ECORE_EVAS_WAYLAND_SHM
16 # include <stdlib.h>
17 # include <string.h>
18 # include <unistd.h>
19 # include <sys/types.h>
20 # include <sys/mman.h>
21 #endif
22
23 # include <Eina.h>
24 # include <Evas.h>
25 # include <Ecore.h>
26
27 # include "ecore_evas_private.h"
28 # include "Ecore_Evas.h"
29
30 #ifdef BUILD_ECORE_EVAS_WAYLAND_SHM
31 # include <Evas_Engine_Wayland_Shm.h>
32 # include <Ecore_Wayland.h>
33
34 /* local structures */
35 typedef struct _EE_Wl_Smart_Data EE_Wl_Smart_Data;
36 struct _EE_Wl_Smart_Data 
37 {
38    Evas_Object *frame;
39    Evas_Object *text;
40    Evas_Coord x, y, w, h;
41 };
42
43 /* local function prototypes */
44 static int _ecore_evas_wl_init(void);
45 static int _ecore_evas_wl_shutdown(void);
46 static void _ecore_evas_wl_pre_free(Ecore_Evas *ee);
47 static void _ecore_evas_wl_free(Ecore_Evas *ee);
48 static void _ecore_evas_wl_callback_resize_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee));
49 static void _ecore_evas_wl_callback_move_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee));
50 static void _ecore_evas_wl_callback_delete_request_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee));
51 static void _ecore_evas_wl_callback_focus_in_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee));
52 static void _ecore_evas_wl_callback_focus_out_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee));
53 static void _ecore_evas_wl_move(Ecore_Evas *ee, int x, int y);
54 static void _ecore_evas_wl_resize(Ecore_Evas *ee, int w, int h);
55 static void _ecore_evas_wl_show(Ecore_Evas *ee);
56 static void _ecore_evas_wl_hide(Ecore_Evas *ee);
57 static void _ecore_evas_wl_title_set(Ecore_Evas *ee, const char *t);
58 static void _ecore_evas_wl_name_class_set(Ecore_Evas *ee, const char *n, const char *c);
59 static void _ecore_evas_wl_size_min_set(Ecore_Evas *ee, int w, int h);
60 static void _ecore_evas_wl_size_max_set(Ecore_Evas *ee, int w, int h);
61 static void _ecore_evas_wl_size_base_set(Ecore_Evas *ee, int w, int h);
62 static void _ecore_evas_wl_size_step_set(Ecore_Evas *ee, int w, int h);
63 static void _ecore_evas_wl_object_cursor_set(Ecore_Evas *ee, Evas_Object  *obj, int layer, int hot_x, int hot_y);
64 static void _ecore_evas_wl_object_cursor_del(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event __UNUSED__);
65 static int _ecore_evas_wl_render(Ecore_Evas *ee);
66 static void _ecore_evas_wl_screen_geometry_get(const Ecore_Evas *ee __UNUSED__, int *x, int *y, int *w, int *h);
67 static void _ecore_evas_wl_buffer_new(Ecore_Evas *ee, void **dest);
68
69 static Eina_Bool _ecore_evas_wl_event_mouse_down(void *data __UNUSED__, int type __UNUSED__, void *event);
70 static Eina_Bool _ecore_evas_wl_event_mouse_up(void *data __UNUSED__, int type __UNUSED__, void *event);
71 static Eina_Bool _ecore_evas_wl_event_mouse_move(void *data __UNUSED__, int type __UNUSED__, void *event);
72 static Eina_Bool _ecore_evas_wl_event_mouse_wheel(void *data __UNUSED__, int type __UNUSED__, void *event);
73 static Eina_Bool _ecore_evas_wl_event_mouse_in(void *data __UNUSED__, int type __UNUSED__, void *event);
74 static Eina_Bool _ecore_evas_wl_event_mouse_out(void *data __UNUSED__, int type __UNUSED__, void *event);
75 static Eina_Bool _ecore_evas_wl_event_focus_in(void *data __UNUSED__, int type __UNUSED__, void *event);
76 static Eina_Bool _ecore_evas_wl_event_focus_out(void *data __UNUSED__, int type __UNUSED__, void *event);
77
78 /* SMART stuff for frame */
79 static Evas_Smart *_ecore_evas_wl_smart = NULL;
80
81 static void _ecore_evas_wl_smart_init(void);
82 static void _ecore_evas_wl_smart_add(Evas_Object *obj);
83 static void _ecore_evas_wl_smart_del(Evas_Object *obj);
84 static void _ecore_evas_wl_smart_resize(Evas_Object *obj, Evas_Coord w, Evas_Coord h);
85 static void _ecore_evas_wl_smart_show(Evas_Object *obj);
86 static void _ecore_evas_wl_smart_hide(Evas_Object *obj);
87
88 static Evas_Object *_ecore_evas_wl_frame_add(Evas *evas);
89
90 /* local variables */
91 static int _ecore_evas_wl_init_count = 0;
92 static Ecore_Event_Handler *_ecore_evas_wl_event_handlers[8];
93
94 static Ecore_Evas_Engine_Func _ecore_wl_engine_func = 
95 {
96    _ecore_evas_wl_free, 
97    _ecore_evas_wl_callback_resize_set, 
98    _ecore_evas_wl_callback_move_set, 
99    NULL, // callback show set
100    NULL, // callback hide set
101    _ecore_evas_wl_callback_delete_request_set, 
102    NULL, // callback destroy set
103    _ecore_evas_wl_callback_focus_in_set, 
104    _ecore_evas_wl_callback_focus_out_set, 
105    NULL, // callback mouse in set
106    NULL, // callback mouse out set
107    NULL, // callback sticky set
108    NULL, // callback unsticky set
109    NULL, // callback pre render set
110    NULL, // callback post render set
111    _ecore_evas_wl_move, 
112    NULL, // func managed move
113    _ecore_evas_wl_resize, 
114    NULL, // func move_resize
115    NULL, // func rotation set
116    NULL, // func shaped set
117    _ecore_evas_wl_show, 
118    _ecore_evas_wl_hide, 
119    NULL, // func raise
120    NULL, // func lower
121    NULL, // func activate
122    _ecore_evas_wl_title_set, 
123    _ecore_evas_wl_name_class_set, 
124    _ecore_evas_wl_size_min_set, 
125    _ecore_evas_wl_size_max_set, 
126    _ecore_evas_wl_size_base_set, 
127    _ecore_evas_wl_size_step_set, 
128    _ecore_evas_wl_object_cursor_set, 
129    NULL, // func layer set
130    NULL, // func focus set
131    NULL, // func iconified set
132    NULL, // func borderless set
133    NULL, // func override set
134    NULL, // func maximized set
135    NULL, // func fullscreen set
136    NULL, // func avoid_damage set
137    NULL, // func withdrawn set
138    NULL, // func sticky set
139    NULL, // func ignore_events set
140    NULL, // func alpha set
141    NULL, // func transparent set
142    _ecore_evas_wl_render, 
143    _ecore_evas_wl_screen_geometry_get
144 };
145
146 /* external variables */
147 #endif
148
149 #ifdef BUILD_ECORE_EVAS_WAYLAND_SHM
150 EAPI Ecore_Evas *
151 ecore_evas_wayland_shm_new(const char *disp_name, int x, int y, int w, int h, int frame)
152 {
153    Evas_Engine_Info_Wayland_Shm *einfo;
154    Ecore_Evas *ee;
155    int method = 0;
156    static int _win_id = 1;
157
158    LOGFN(__FILE__, __LINE__, __FUNCTION__);
159
160    if (!(method = evas_render_method_lookup("wayland_shm"))) 
161      {
162         ERR("Render method lookup failed.");
163         return NULL;
164      }
165
166    if (!(ecore_wl_init(disp_name))) 
167      {
168         ERR("Failed to initialize Ecore Wayland.");
169         return NULL;
170      }
171
172    if (!(ee = calloc(1, sizeof(Ecore_Evas)))) 
173      {
174         ERR("Failed to allocate Ecore_Evas.");
175         ecore_wl_shutdown();
176         return NULL;
177      }
178
179    ECORE_MAGIC_SET(ee, ECORE_MAGIC_EVAS);
180
181    _ecore_evas_wl_init();
182
183    ee->engine.func = (Ecore_Evas_Engine_Func *)&_ecore_wl_engine_func;
184
185    ee->driver = "wayland_shm";
186    if (disp_name) ee->name = strdup(disp_name);
187
188    if (w < 1) w = 1;
189    if (h < 1) h = 1;
190
191    ee->req.x = ee->x = x;
192    ee->req.y = ee->y = y;
193    ee->req.w = ee->w = w;
194    ee->req.h = ee->h = h;
195    ee->rotation = 0;
196    ee->prop.max.w = ee->prop.max.h = 32767;
197    ee->prop.layer = 4;
198    ee->prop.request_pos = 0;
199    ee->prop.sticky = 0;
200    ee->prop.draw_frame = frame;
201    ee->prop.window = _win_id++;
202
203    ee->evas = evas_new();
204    evas_data_attach_set(ee->evas, ee);
205    evas_output_method_set(ee->evas, method);
206    evas_output_size_set(ee->evas, ee->w, ee->h);
207    evas_output_viewport_set(ee->evas, 0, 0, ee->w, ee->h);
208
209    if (ee->prop.draw_frame) 
210      evas_output_framespace_set(ee->evas, 4, 18, 8, 22);
211
212    if ((einfo = (Evas_Engine_Info_Wayland_Shm *)evas_engine_info_get(ee->evas))) 
213      {
214         einfo->info.rotation = ee->rotation;
215         einfo->info.debug = EINA_FALSE;
216         if (!evas_engine_info_set(ee->evas, (Evas_Engine_Info *)einfo)) 
217           {
218              ERR("Failed to set Evas Engine Info for '%s'.", ee->driver);
219              ecore_evas_free(ee);
220              return NULL;
221           }
222      }
223    else 
224      {
225         ERR("Failed to get Evas Engine Info for '%s'.", ee->driver);
226         ecore_evas_free(ee);
227         return NULL;
228      }
229
230    /* NB: we need to be notified before 'free' so we can munmap the evas 
231     * engine destination */
232    ecore_evas_callback_pre_free_set(ee, _ecore_evas_wl_pre_free);
233
234    if (ee->prop.draw_frame) 
235      {
236         ee->engine.wl.frame = _ecore_evas_wl_frame_add(ee->evas);
237         evas_object_is_frame_object_set(ee->engine.wl.frame, EINA_TRUE);
238         evas_object_move(ee->engine.wl.frame, 0, 0);
239      }
240
241    ecore_evas_input_event_register(ee);
242    _ecore_evas_register(ee);
243
244    ecore_event_window_register(ee->prop.window, ee, ee->evas, 
245                                (Ecore_Event_Mouse_Move_Cb)_ecore_evas_mouse_move_process, 
246                                (Ecore_Event_Multi_Move_Cb)_ecore_evas_mouse_multi_move_process, 
247                                (Ecore_Event_Multi_Down_Cb)_ecore_evas_mouse_multi_down_process, 
248                                (Ecore_Event_Multi_Up_Cb)_ecore_evas_mouse_multi_up_process);
249
250    evas_event_feed_mouse_in(ee->evas, (unsigned int)((unsigned long long)(ecore_time_get() * 1000.0) & 0xffffffff), NULL);
251
252    return ee;
253 }
254
255 /* local functions */
256 static int 
257 _ecore_evas_wl_init(void)
258 {
259    LOGFN(__FILE__, __LINE__, __FUNCTION__);
260
261    if (++_ecore_evas_wl_init_count != 1)
262      return _ecore_evas_wl_init_count;
263
264    _ecore_evas_wl_event_handlers[0] = 
265      ecore_event_handler_add(ECORE_EVENT_MOUSE_BUTTON_DOWN, 
266                              _ecore_evas_wl_event_mouse_down, NULL);
267    _ecore_evas_wl_event_handlers[1] = 
268      ecore_event_handler_add(ECORE_EVENT_MOUSE_BUTTON_UP, 
269                              _ecore_evas_wl_event_mouse_up, NULL);
270    _ecore_evas_wl_event_handlers[2] = 
271      ecore_event_handler_add(ECORE_EVENT_MOUSE_MOVE, 
272                              _ecore_evas_wl_event_mouse_move, NULL);
273    _ecore_evas_wl_event_handlers[3] = 
274      ecore_event_handler_add(ECORE_EVENT_MOUSE_WHEEL, 
275                              _ecore_evas_wl_event_mouse_wheel, NULL);
276    _ecore_evas_wl_event_handlers[4] = 
277      ecore_event_handler_add(ECORE_WL_EVENT_MOUSE_IN, 
278                              _ecore_evas_wl_event_mouse_in, NULL);
279    _ecore_evas_wl_event_handlers[5] = 
280      ecore_event_handler_add(ECORE_WL_EVENT_MOUSE_OUT, 
281                              _ecore_evas_wl_event_mouse_out, NULL);
282    _ecore_evas_wl_event_handlers[6] = 
283      ecore_event_handler_add(ECORE_WL_EVENT_FOCUS_IN, 
284                              _ecore_evas_wl_event_focus_in, NULL);
285    _ecore_evas_wl_event_handlers[7] = 
286      ecore_event_handler_add(ECORE_WL_EVENT_FOCUS_OUT, 
287                              _ecore_evas_wl_event_focus_out, NULL);
288
289    ecore_event_evas_init();
290
291    return _ecore_evas_wl_init_count;
292 }
293
294 static int 
295 _ecore_evas_wl_shutdown(void)
296 {
297    unsigned int i = 0;
298
299    LOGFN(__FILE__, __LINE__, __FUNCTION__);
300
301    if (--_ecore_evas_wl_init_count != 0)
302      return _ecore_evas_wl_init_count;
303
304    for (i = 0; i < sizeof(_ecore_evas_wl_event_handlers) / sizeof(Ecore_Event_Handler *); i++) 
305      {
306         if (_ecore_evas_wl_event_handlers[i])
307           ecore_event_handler_del(_ecore_evas_wl_event_handlers[i]);
308      }
309
310    ecore_event_evas_shutdown();
311
312    return _ecore_evas_wl_init_count;
313 }
314
315 static void 
316 _ecore_evas_wl_pre_free(Ecore_Evas *ee)
317 {
318    Evas_Engine_Info_Wayland_Shm *einfo;
319
320    LOGFN(__FILE__, __LINE__, __FUNCTION__);
321
322    /* destroy frame */
323    if (ee->engine.wl.frame) evas_object_del(ee->engine.wl.frame);
324
325    /* get engine info */
326    einfo = (Evas_Engine_Info_Wayland_Shm *)evas_engine_info_get(ee->evas);
327    if ((einfo) && (einfo->info.dest))
328      {
329         int ret = 0;
330
331         /* munmap previous engine destination */
332         ret = munmap(einfo->info.dest, ((ee->w * sizeof(int)) * ee->h));
333      }
334 }
335
336 static void 
337 _ecore_evas_wl_free(Ecore_Evas *ee)
338 {
339    LOGFN(__FILE__, __LINE__, __FUNCTION__);
340
341    /* destroy buffer */
342    if (ee->engine.wl.buffer) wl_buffer_destroy(ee->engine.wl.buffer);
343    ee->engine.wl.buffer = NULL;
344
345    /* destroy shell surface */
346    if (ee->engine.wl.shell_surface)
347      wl_shell_surface_destroy(ee->engine.wl.shell_surface);
348    ee->engine.wl.shell_surface = NULL;
349
350    /* destroy surface */
351    if (ee->engine.wl.surface) wl_surface_destroy(ee->engine.wl.surface);
352    ee->engine.wl.surface = NULL;
353
354    ecore_event_window_unregister(ee->prop.window);
355
356    _ecore_evas_wl_shutdown();
357    ecore_wl_shutdown();
358 }
359
360 static void 
361 _ecore_evas_wl_callback_resize_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee))
362 {
363    LOGFN(__FILE__, __LINE__, __FUNCTION__);
364
365    if (!ee) return;
366    ee->func.fn_resize = func;
367 }
368
369 static void 
370 _ecore_evas_wl_callback_move_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee))
371 {
372    LOGFN(__FILE__, __LINE__, __FUNCTION__);
373
374    if (!ee) return;
375    ee->func.fn_move = func;
376 }
377
378 static void 
379 _ecore_evas_wl_callback_delete_request_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee))
380 {
381    LOGFN(__FILE__, __LINE__, __FUNCTION__);
382
383    if (!ee) return;
384    ee->func.fn_delete_request = func;
385 }
386
387 static void 
388 _ecore_evas_wl_callback_focus_in_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee))
389 {
390    LOGFN(__FILE__, __LINE__, __FUNCTION__);
391
392    if (!ee) return;
393    ee->func.fn_focus_in = func;
394 }
395
396 static void 
397 _ecore_evas_wl_callback_focus_out_set(Ecore_Evas *ee, void (*func)(Ecore_Evas *ee))
398 {
399    LOGFN(__FILE__, __LINE__, __FUNCTION__);
400
401    if (!ee) return;
402    ee->func.fn_focus_out = func;
403 }
404
405 static void 
406 _ecore_evas_wl_move(Ecore_Evas *ee, int x, int y) 
407 {
408    LOGFN(__FILE__, __LINE__, __FUNCTION__);
409
410    if (!ee) return;
411    if ((ee->x == x) && (ee->y == y)) return;
412    ee->req.x = x;
413    ee->req.y = y;
414
415    ee->x = x;
416    ee->y = y;
417    /* FIXME: Forward this to Wayland */
418 }
419
420 static void 
421 _ecore_evas_wl_resize(Ecore_Evas *ee, int w, int h)
422 {
423    Evas_Engine_Info_Wayland_Shm *einfo;
424
425    LOGFN(__FILE__, __LINE__, __FUNCTION__);
426
427    if (!ee) return;
428    if (w < 1) w = 1;
429    if (h < 1) h = 1;
430    if ((ee->w == w) && (ee->h == h)) return;
431    ee->req.w = w;
432    ee->req.h = h;
433
434    /* get engine info */
435    einfo = (Evas_Engine_Info_Wayland_Shm *)evas_engine_info_get(ee->evas);
436    if (einfo->info.dest)
437      {
438         int ret = 0;
439
440         /* munmap previous engine destination */
441         ret = munmap(einfo->info.dest, ((ee->w * sizeof(int)) * ee->h));
442      }
443
444    /* free old buffer */
445    if (ee->engine.wl.buffer) wl_buffer_destroy(ee->engine.wl.buffer);
446    ee->engine.wl.buffer = NULL;
447
448    ee->w = w;
449    ee->h = h;
450
451    /* create buffer @ new size (also mmaps the new destination) */
452    _ecore_evas_wl_buffer_new(ee, &einfo->info.dest);
453
454    /* change evas output & viewport sizes */
455    evas_output_size_set(ee->evas, ee->w, ee->h);
456    evas_output_viewport_set(ee->evas, 0, 0, ee->w, ee->h);
457    if (ee->engine.wl.frame)
458      evas_object_resize(ee->engine.wl.frame, ee->w, ee->h);
459
460    /* set new engine destination */
461    evas_engine_info_set(ee->evas, (Evas_Engine_Info *)einfo);
462
463    /* flush new buffer fd */
464    ecore_wl_flush();
465
466    /* damage buffer */
467    wl_buffer_damage(ee->engine.wl.buffer, 0, 0, ee->w, ee->h);
468
469    if (ee->visible) 
470      {
471         /* if visible, attach to surface */
472         wl_surface_attach(ee->engine.wl.surface, ee->engine.wl.buffer, 0, 0);
473
474         /* damage surface */
475         wl_surface_damage(ee->engine.wl.surface, 0, 0, ee->w, ee->h);
476      }
477
478    if (ee->func.fn_resize) ee->func.fn_resize(ee);
479 }
480
481 static void 
482 _ecore_evas_wl_show(Ecore_Evas *ee)
483 {
484    Evas_Engine_Info_Wayland_Shm *einfo;
485
486    LOGFN(__FILE__, __LINE__, __FUNCTION__);
487
488    if (!ee) return;
489    if (ee->visible) return;
490
491    /* get engine info */
492    einfo = (Evas_Engine_Info_Wayland_Shm *)evas_engine_info_get(ee->evas);
493
494    /* create new surface */
495    ee->engine.wl.surface = 
496      wl_compositor_create_surface(ecore_wl_compositor_get());
497    wl_surface_set_user_data(ee->engine.wl.surface, (void *)ee->prop.window);
498
499    /* get new shell surface */
500    ee->engine.wl.shell_surface = 
501      wl_shell_get_shell_surface(ecore_wl_shell_get(), ee->engine.wl.surface);
502
503    /* set toplevel */
504    wl_shell_surface_set_toplevel(ee->engine.wl.shell_surface);
505
506    /* create buffer @ new size (also mmaps the new destination) */
507    _ecore_evas_wl_buffer_new(ee, &einfo->info.dest);
508
509    if (ee->engine.wl.frame) 
510      {
511         evas_object_show(ee->engine.wl.frame);
512         evas_object_resize(ee->engine.wl.frame, ee->w, ee->h);
513      }
514
515    /* set new engine destination */
516    evas_engine_info_set(ee->evas, (Evas_Engine_Info *)einfo);
517
518    /* flush new buffer fd */
519    ecore_wl_flush();
520
521    /* attach buffer to surface */
522    wl_surface_attach(ee->engine.wl.surface, ee->engine.wl.buffer, 0, 0);
523
524    /* NB: No need to do a damage here. If we do, we end up w/ screen 
525     * artifacts in the compositor */
526    /* wl_surface_damage(ee->engine.wl.surface, 0, 0, ee->w, ee->h); */
527
528    ee->visible = 1;
529    if (ee->func.fn_show) ee->func.fn_show(ee);
530 }
531
532 static void 
533 _ecore_evas_wl_hide(Ecore_Evas *ee) 
534 {
535    Evas_Engine_Info_Wayland_Shm *einfo;
536
537    LOGFN(__FILE__, __LINE__, __FUNCTION__);
538
539    if (!ee) return;
540    if (!ee->visible) return;
541
542    /* get engine info */
543    einfo = (Evas_Engine_Info_Wayland_Shm *)evas_engine_info_get(ee->evas);
544    if (einfo->info.dest)
545      {
546         int ret = 0;
547
548         /* munmap previous engine destination */
549         ret = munmap(einfo->info.dest, ((ee->w * sizeof(int)) * ee->h));
550         einfo->info.dest = NULL;
551      }
552
553    /* free old buffer */
554    if (ee->engine.wl.buffer) wl_buffer_destroy(ee->engine.wl.buffer);
555    ee->engine.wl.buffer = NULL;
556
557    /* set new engine destination */
558    evas_engine_info_set(ee->evas, (Evas_Engine_Info *)einfo);
559
560    /* destroy shell surface */
561    if (ee->engine.wl.shell_surface) 
562      wl_shell_surface_destroy(ee->engine.wl.shell_surface);
563    ee->engine.wl.shell_surface = NULL;
564
565    /* destroy surface */
566    if (ee->engine.wl.surface) wl_surface_destroy(ee->engine.wl.surface);
567    ee->engine.wl.surface = NULL;
568
569    ee->visible = 0;
570    ee->should_be_visible = 0;
571    if (ee->func.fn_hide) ee->func.fn_hide(ee);
572 }
573
574 static void 
575 _ecore_evas_wl_title_set(Ecore_Evas *ee, const char *t) 
576 {
577    LOGFN(__FILE__, __LINE__, __FUNCTION__);
578
579    if (!ee) return;
580    if (ee->prop.title) free(ee->prop.title);
581    ee->prop.title = NULL;
582    if (t) ee->prop.title = strdup(t);
583
584    if ((ee->prop.draw_frame) && (ee->engine.wl.frame)) 
585      {
586         EE_Wl_Smart_Data *sd;
587
588         if (!(sd = evas_object_smart_data_get(ee->engine.wl.frame))) return;
589         evas_object_text_text_set(sd->text, ee->prop.title);
590      }
591 }
592
593 static void 
594 _ecore_evas_wl_name_class_set(Ecore_Evas *ee, const char *n, const char *c) 
595 {
596    LOGFN(__FILE__, __LINE__, __FUNCTION__);
597
598    if (!ee) return;
599    if (ee->prop.name) free(ee->prop.name);
600    if (ee->prop.clas) free(ee->prop.clas);
601    ee->prop.name = NULL;
602    ee->prop.clas = NULL;
603    if (n) ee->prop.name = strdup(n);
604    if (c) ee->prop.clas = strdup(c);
605    /* FIXME: Forward these changes to Wayland somehow */
606 }
607
608 static void 
609 _ecore_evas_wl_size_min_set(Ecore_Evas *ee, int w, int h) 
610 {
611    LOGFN(__FILE__, __LINE__, __FUNCTION__);
612
613    if (!ee) return;
614    if (w < 0) w = 0;
615    if (h < 0) h = 0;
616    if ((ee->prop.min.w == w) && (ee->prop.min.h == h)) return;
617    ee->prop.min.w = w;
618    ee->prop.min.h = h;
619 }
620
621 static void 
622 _ecore_evas_wl_size_max_set(Ecore_Evas *ee, int w, int h) 
623 {
624    LOGFN(__FILE__, __LINE__, __FUNCTION__);
625
626    if (!ee) return;
627    if (w < 0) w = 0;
628    if (h < 0) h = 0;
629    if ((ee->prop.max.w == w) && (ee->prop.max.h == h)) return;
630    ee->prop.max.w = w;
631    ee->prop.max.h = h;
632 }
633
634 static void 
635 _ecore_evas_wl_size_base_set(Ecore_Evas *ee, int w, int h) 
636 {
637    LOGFN(__FILE__, __LINE__, __FUNCTION__);
638
639    if (!ee) return;
640    if (w < 0) w = 0;
641    if (h < 0) h = 0;
642    if ((ee->prop.base.w == w) && (ee->prop.base.h == h)) return;
643    ee->prop.base.w = w;
644    ee->prop.base.h = h;
645 }
646
647 static void 
648 _ecore_evas_wl_size_step_set(Ecore_Evas *ee, int w, int h) 
649 {
650    LOGFN(__FILE__, __LINE__, __FUNCTION__);
651
652    if (!ee) return;
653    if (w < 0) w = 0;
654    if (h < 0) h = 0;
655    if ((ee->prop.step.w == w) && (ee->prop.step.h == h)) return;
656    ee->prop.step.w = w;
657    ee->prop.step.h = h;
658 }
659
660 static void 
661 _ecore_evas_wl_object_cursor_set(Ecore_Evas *ee, Evas_Object  *obj, int layer, int hot_x, int hot_y) 
662 {
663    int x = 0, y = 0;
664
665    LOGFN(__FILE__, __LINE__, __FUNCTION__);
666
667    if (!ee) return;
668    if (ee->prop.cursor.object) evas_object_del(ee->prop.cursor.object);
669    ee->prop.cursor.object = NULL;
670
671    if (!obj) 
672      {
673         ee->prop.cursor.layer = 0;
674         ee->prop.cursor.hot.x = 0;
675         ee->prop.cursor.hot.y = 0;
676         return;
677      }
678
679    ee->prop.cursor.object = obj;
680    ee->prop.cursor.layer = layer;
681    ee->prop.cursor.hot.x = hot_x;
682    ee->prop.cursor.hot.y = hot_y;
683
684    evas_pointer_output_xy_get(ee->evas, &x, &y);
685    evas_object_layer_set(ee->prop.cursor.object, ee->prop.cursor.layer);
686    evas_object_move(ee->prop.cursor.object, 
687                     x - ee->prop.cursor.hot.x, y - ee->prop.cursor.hot.y);
688    evas_object_pass_events_set(ee->prop.cursor.object, 1);
689    if (evas_pointer_inside_get(ee->evas))
690      evas_object_show(ee->prop.cursor.object);
691
692    evas_object_event_callback_add(obj, EVAS_CALLBACK_DEL, 
693                                   _ecore_evas_wl_object_cursor_del, ee);
694 }
695
696 static void 
697 _ecore_evas_wl_object_cursor_del(void *data, Evas *evas __UNUSED__, Evas_Object *obj __UNUSED__, void *event __UNUSED__) 
698 {
699    Ecore_Evas *ee;
700
701    if (!(ee = data)) return;
702    ee->prop.cursor.object = NULL;
703 }
704
705 static int 
706 _ecore_evas_wl_render(Ecore_Evas *ee)
707 {
708    int rend = 0;
709
710    if (!ee) return 0;
711    if (!ee->visible) 
712      evas_norender(ee->evas);
713    else 
714      {
715         Eina_List *ll = NULL, *updates = NULL;
716         Ecore_Evas *ee2 = NULL;
717
718         if (ee->func.fn_pre_render) ee->func.fn_pre_render(ee);
719
720         EINA_LIST_FOREACH(ee->sub_ecore_evas, ll, ee2) 
721           {
722              if (ee2->func.fn_pre_render) ee2->func.fn_pre_render(ee2);
723              if (ee2->engine.func->fn_render)
724                rend |= ee2->engine.func->fn_render(ee2);
725              if (ee2->func.fn_post_render) ee2->func.fn_post_render(ee2);
726           }
727
728         if ((updates = evas_render_updates(ee->evas))) 
729           {
730              Eina_List *l = NULL;
731              Eina_Rectangle *r;
732
733              EINA_LIST_FOREACH(updates, l, r) 
734                {
735                   if (ee->engine.wl.buffer)
736                     wl_buffer_damage(ee->engine.wl.buffer, 
737                                      r->x, r->y, r->w, r->h);
738
739                   if (ee->engine.wl.surface)
740                     wl_surface_damage(ee->engine.wl.surface, 
741                                       r->x, r->y, r->w, r->h);
742                }
743
744              evas_render_updates_free(updates);
745              _ecore_evas_idle_timeout_update(ee);
746              rend = 1;
747           }
748
749         if (ee->func.fn_post_render) ee->func.fn_post_render(ee);
750      }
751
752    return rend;
753 }
754
755 static void 
756 _ecore_evas_wl_screen_geometry_get(const Ecore_Evas *ee __UNUSED__, int *x, int *y, int *w, int *h)
757 {
758    LOGFN(__FILE__, __LINE__, __FUNCTION__);
759
760    if (x) *x = 0;
761    if (y) *y = 0;
762    ecore_wl_screen_size_get(w, h);
763 }
764
765 static Eina_Bool 
766 _ecore_evas_wl_event_mouse_down(void *data __UNUSED__, int type __UNUSED__, void *event)
767 {
768    Ecore_Evas *ee;
769    Ecore_Event_Mouse_Button *ev;
770
771    LOGFN(__FILE__, __LINE__, __FUNCTION__);
772
773    ev = event;
774    ee = ecore_event_window_match(ev->window);
775    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
776    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
777    evas_event_feed_mouse_down(ee->evas, ev->buttons, ev->modifiers, 
778                               ev->timestamp, NULL);
779    return ECORE_CALLBACK_PASS_ON;
780 }
781
782 static Eina_Bool 
783 _ecore_evas_wl_event_mouse_up(void *data __UNUSED__, int type __UNUSED__, void *event)
784 {
785    Ecore_Evas *ee;
786    Ecore_Event_Mouse_Button *ev;
787
788    LOGFN(__FILE__, __LINE__, __FUNCTION__);
789
790    ev = event;
791    ee = ecore_event_window_match(ev->window);
792    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
793    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
794    evas_event_feed_mouse_up(ee->evas, ev->buttons, ev->modifiers, 
795                             ev->timestamp, NULL);
796    return ECORE_CALLBACK_PASS_ON;
797 }
798
799 static Eina_Bool 
800 _ecore_evas_wl_event_mouse_move(void *data __UNUSED__, int type __UNUSED__, void *event)
801 {
802    Ecore_Evas *ee;
803    Ecore_Event_Mouse_Move *ev;
804
805    ev = event;
806    ee = ecore_event_window_match(ev->window);
807    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
808    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
809    ee->mouse.x = ev->x;
810    ee->mouse.y = ev->y;
811    evas_event_feed_mouse_move(ee->evas, ev->x, ev->y, ev->timestamp, NULL);
812    _ecore_evas_mouse_move_process(ee, ev->x, ev->y, ev->timestamp);
813    return ECORE_CALLBACK_PASS_ON;
814 }
815
816 static Eina_Bool 
817 _ecore_evas_wl_event_mouse_wheel(void *data __UNUSED__, int type __UNUSED__, void *event)
818 {
819    Ecore_Evas *ee;
820    Ecore_Event_Mouse_Wheel *ev;
821
822    LOGFN(__FILE__, __LINE__, __FUNCTION__);
823
824    ev = event;
825    ee = ecore_event_window_match(ev->window);
826    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
827    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
828    evas_event_feed_mouse_wheel(ee->evas, ev->direction, ev->z, 
829                                ev->timestamp, NULL);
830    return ECORE_CALLBACK_PASS_ON;
831 }
832
833 static Eina_Bool 
834 _ecore_evas_wl_event_mouse_in(void *data __UNUSED__, int type __UNUSED__, void *event)
835 {
836    Ecore_Evas *ee;
837    Ecore_Wl_Event_Mouse_In *ev;
838
839    LOGFN(__FILE__, __LINE__, __FUNCTION__);
840
841    ev = event;
842    ee = ecore_event_window_match(ev->window);
843    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
844    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
845    if (ee->func.fn_mouse_in) ee->func.fn_mouse_in(ee);
846    ecore_event_evas_modifier_lock_update(ee->evas, ev->modifiers);
847    evas_event_feed_mouse_in(ee->evas, ev->time, NULL);
848    _ecore_evas_mouse_move_process(ee, ev->x, ev->y, ev->time);
849    return ECORE_CALLBACK_PASS_ON;
850 }
851
852 static Eina_Bool 
853 _ecore_evas_wl_event_mouse_out(void *data __UNUSED__, int type __UNUSED__, void *event)
854 {
855    Ecore_Evas *ee;
856    Ecore_Wl_Event_Mouse_Out *ev;
857
858    LOGFN(__FILE__, __LINE__, __FUNCTION__);
859
860    ev = event;
861    ee = ecore_event_window_match(ev->window);
862    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
863    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
864    ecore_event_evas_modifier_lock_update(ee->evas, ev->modifiers);
865    _ecore_evas_mouse_move_process(ee, ev->x, ev->y, ev->time);
866    evas_event_feed_mouse_out(ee->evas, ev->time, NULL);
867    if (ee->func.fn_mouse_out) ee->func.fn_mouse_out(ee);
868    if (ee->prop.cursor.object) evas_object_hide(ee->prop.cursor.object);
869    return ECORE_CALLBACK_PASS_ON;
870 }
871
872 static Eina_Bool 
873 _ecore_evas_wl_event_focus_in(void *data __UNUSED__, int type __UNUSED__, void *event)
874 {
875    Ecore_Evas *ee;
876    Ecore_Wl_Event_Focus_In *ev;
877
878    LOGFN(__FILE__, __LINE__, __FUNCTION__);
879
880    ev = event;
881    ee = ecore_event_window_match(ev->window);
882    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
883    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
884    ee->prop.focused = 1;
885    evas_focus_in(ee->evas);
886    if (ee->func.fn_focus_in) ee->func.fn_focus_in(ee);
887    return ECORE_CALLBACK_PASS_ON;
888 }
889
890 static Eina_Bool 
891 _ecore_evas_wl_event_focus_out(void *data __UNUSED__, int type __UNUSED__, void *event)
892 {
893    Ecore_Evas *ee;
894    Ecore_Wl_Event_Focus_Out *ev;
895
896    LOGFN(__FILE__, __LINE__, __FUNCTION__);
897
898    ev = event;
899    ee = ecore_event_window_match(ev->window);
900    if ((!ee) || (ee->ignore_events)) return ECORE_CALLBACK_PASS_ON;
901    if (ev->window != ee->prop.window) return ECORE_CALLBACK_PASS_ON;
902    evas_focus_out(ee->evas);
903    ee->prop.focused = 0;
904    if (ee->func.fn_focus_out) ee->func.fn_focus_out(ee);
905    return ECORE_CALLBACK_PASS_ON;
906 }
907
908 static void 
909 _ecore_evas_wl_buffer_new(Ecore_Evas *ee, void **dest)
910 {
911    static unsigned int format;
912    char tmp[PATH_MAX];
913    int fd = -1, stride = 0, size = 0;
914    void *ret;
915
916    LOGFN(__FILE__, __LINE__, __FUNCTION__);
917
918    if (dest) *dest = NULL;
919
920    if (!format) format = ecore_wl_format_get();
921
922    strcpy(tmp, "/tmp/ecore-wayland_shm-XXXXXX");
923    if ((fd = mkstemp(tmp)) < 0) 
924      {
925         ERR("Could not create temporary file.");
926         return;
927      }
928
929    stride = (ee->w * sizeof(int));
930    size = (stride * ee->h);
931    if (ftruncate(fd, size) < 0) 
932      {
933         ERR("Could not truncate temporary file.");
934         close(fd);
935         return;
936      }
937
938    ret = mmap(NULL, size, (PROT_READ | PROT_WRITE), MAP_SHARED, fd, 0);
939    unlink(tmp);
940
941    if (ret == MAP_FAILED) 
942      {
943         ERR("mmap of temporary file failed.");
944         close(fd);
945         return;
946      }
947
948    if (dest) *dest = ret;
949
950    ee->engine.wl.buffer = 
951      wl_shm_create_buffer(ecore_wl_shm_get(), fd, ee->w, ee->h, stride, format);
952
953    close(fd);
954 }
955
956 static void 
957 _ecore_evas_wl_smart_init(void) 
958 {
959    if (_ecore_evas_wl_smart) return;
960      {
961         static const Evas_Smart_Class sc = 
962           {
963              "ecore_evas_wl_frame", EVAS_SMART_CLASS_VERSION, 
964              _ecore_evas_wl_smart_add, 
965              _ecore_evas_wl_smart_del, 
966              NULL, 
967              _ecore_evas_wl_smart_resize, 
968              _ecore_evas_wl_smart_show, 
969              _ecore_evas_wl_smart_hide, 
970              NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
971           };
972         _ecore_evas_wl_smart = evas_smart_class_new(&sc);
973      }
974 }
975
976 static void 
977 _ecore_evas_wl_smart_add(Evas_Object *obj) 
978 {
979    EE_Wl_Smart_Data *sd;
980    Evas *evas;
981
982    LOGFN(__FILE__, __LINE__, __FUNCTION__);
983
984    if (!(sd = calloc(1, sizeof(EE_Wl_Smart_Data)))) return;
985
986    evas = evas_object_evas_get(obj);
987
988    sd->x = 0;
989    sd->y = 0;
990    sd->w = 1;
991    sd->h = 1;
992
993    sd->frame = evas_object_rectangle_add(evas);
994    evas_object_is_frame_object_set(sd->frame, EINA_TRUE);
995    evas_object_color_set(sd->frame, 249, 249, 249, 255);
996    evas_object_smart_member_add(sd->frame, obj);
997
998    sd->text = evas_object_text_add(evas);
999    evas_object_color_set(sd->text, 0, 0, 0, 255);
1000    evas_object_text_style_set(sd->text, EVAS_TEXT_STYLE_PLAIN);
1001    evas_object_text_font_set(sd->text, "Sans", 10);
1002    evas_object_text_text_set(sd->text, "Smart Test");
1003
1004    evas_object_smart_data_set(obj, sd);
1005 }
1006
1007 static void 
1008 _ecore_evas_wl_smart_del(Evas_Object *obj) 
1009 {
1010    EE_Wl_Smart_Data *sd;
1011
1012    LOGFN(__FILE__, __LINE__, __FUNCTION__);
1013
1014    if (!(sd = evas_object_smart_data_get(obj))) return;
1015    evas_object_del(sd->text);
1016    evas_object_del(sd->frame);
1017    free(sd);
1018 }
1019
1020 static void 
1021 _ecore_evas_wl_smart_resize(Evas_Object *obj, Evas_Coord w, Evas_Coord h) 
1022 {
1023    EE_Wl_Smart_Data *sd;
1024
1025    LOGFN(__FILE__, __LINE__, __FUNCTION__);
1026
1027    if (!(sd = evas_object_smart_data_get(obj))) return;
1028    if ((sd->w == w) && (sd->h == h)) return;
1029    sd->w = w;
1030    sd->h = h;
1031    evas_object_resize(sd->frame, w, h);
1032 }
1033
1034 static void 
1035 _ecore_evas_wl_smart_show(Evas_Object *obj) 
1036 {
1037    EE_Wl_Smart_Data *sd;
1038
1039    LOGFN(__FILE__, __LINE__, __FUNCTION__);
1040
1041    if (!(sd = evas_object_smart_data_get(obj))) return;
1042    evas_object_show(sd->frame);
1043    evas_object_show(sd->text);
1044 }
1045
1046 static void 
1047 _ecore_evas_wl_smart_hide(Evas_Object *obj) 
1048 {
1049    EE_Wl_Smart_Data *sd;
1050
1051    LOGFN(__FILE__, __LINE__, __FUNCTION__);
1052
1053    if (!(sd = evas_object_smart_data_get(obj))) return;
1054    evas_object_hide(sd->text);
1055    evas_object_hide(sd->frame);
1056 }
1057
1058 static Evas_Object *
1059 _ecore_evas_wl_frame_add(Evas *evas) 
1060 {
1061    LOGFN(__FILE__, __LINE__, __FUNCTION__);
1062
1063    _ecore_evas_wl_smart_init();
1064    return evas_object_smart_add(evas, _ecore_evas_wl_smart);
1065 }
1066
1067 #else
1068 EAPI Ecore_Evas *
1069 ecore_evas_wayland_shm_new(const char *disp_name __UNUSED__, int x __UNUSED__, int y __UNUSED__, int w __UNUSED__, int h __UNUSED__, int frame __UNUSED__)
1070 {
1071    return NULL;
1072 }
1073 #endif