use ecore-evas and eevas image calls consistently for inlined wins
[framework/uifw/elementary.git] / src / lib / elm_win.c
1 #include <Elementary.h>
2 #include "elm_priv.h"
3
4 typedef struct _Elm_Win Elm_Win;
5
6 struct _Elm_Win
7 {
8    Ecore_Evas *ee;
9    Evas *evas;
10    Evas_Object *parent, *win_obj, *img_obj, *frame_obj;
11    Eina_List *subobjs;
12 #ifdef HAVE_ELEMENTARY_X
13    Ecore_X_Window xwin;
14    Ecore_Event_Handler *client_message_handler;
15 #endif
16    Ecore_Job *deferred_resize_job;
17    Ecore_Job *deferred_child_eval_job;
18
19    Elm_Win_Type type;
20    Elm_Win_Keyboard_Mode kbdmode;
21    struct {
22       const char *info;
23       Ecore_Timer *timer;
24       int repeat_count;
25       int shot_counter;
26    } shot;
27    Eina_Bool autodel : 1;
28    int *autodel_clear, rot;
29    int show_count;
30    struct {
31       int x, y;
32    } screen;
33
34    struct {
35       Evas_Object *top;
36
37       struct {
38          Evas_Object *target;
39          Eina_Bool visible : 1;
40          Eina_Bool handled : 1;
41       } cur, prev;
42
43       const char *style;
44       Ecore_Job *reconf_job;
45
46       Eina_Bool enabled : 1;
47       Eina_Bool changed_theme : 1;
48       Eina_Bool top_animate : 1;
49       Eina_Bool geometry_changed : 1;
50    } focus_highlight;
51 };
52
53 static const char *widtype = NULL;
54 static void _elm_win_obj_callback_del(void *data, Evas *e, Evas_Object *obj, void *event_info);
55 static void _elm_win_obj_callback_img_obj_del(void *data, Evas *e, Evas_Object *obj, void *event_info);
56 static void _elm_win_obj_callback_parent_del(void *data, Evas *e, Evas_Object *obj, void *event_info);
57 static void _elm_win_obj_intercept_move(void *data, Evas_Object *obj, Evas_Coord x, Evas_Coord y);
58 static void _elm_win_obj_intercept_show(void *data, Evas_Object *obj);
59 static void _elm_win_move(Ecore_Evas *ee);
60 static void _elm_win_resize(Ecore_Evas *ee);
61 static void _elm_win_delete_request(Ecore_Evas *ee);
62 static void _elm_win_resize_job(void *data);
63 #ifdef HAVE_ELEMENTARY_X
64 static void _elm_win_xwin_update(Elm_Win *win);
65 #endif
66 static void _elm_win_eval_subobjs(Evas_Object *obj);
67 static void _elm_win_subobj_callback_del(void *data, Evas *e, Evas_Object *obj, void *event_info);
68 static void _elm_win_subobj_callback_changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
69 static void _elm_win_focus_highlight_init(Elm_Win *win);
70 static void _elm_win_focus_highlight_shutdown(Elm_Win *win);
71 static void _elm_win_focus_highlight_visible_set(Elm_Win *win, Eina_Bool visible);
72 static void _elm_win_focus_highlight_reconfigure_job_start(Elm_Win *win);
73 static void _elm_win_focus_highlight_reconfigure_job_stop(Elm_Win *win);
74 static void _elm_win_focus_highlight_anim_end(void *data, Evas_Object *obj, const char *emission, const char *source);
75 static void _elm_win_focus_highlight_reconfigure(Elm_Win *win);
76
77 static const char SIG_DELETE_REQUEST[] = "delete,request";
78 static const char SIG_FOCUS_OUT[] = "focus,out";
79 static const char SIG_FOCUS_IN[] = "focus,in";
80 static const char SIG_MOVED[] = "moved";
81 static const char SIG_THEME_CHANGED[] = "theme,changed";
82
83 static const Evas_Smart_Cb_Description _signals[] = {
84    {SIG_DELETE_REQUEST, ""},
85    {SIG_FOCUS_OUT, ""},
86    {SIG_FOCUS_IN, ""},
87    {SIG_MOVED, ""},
88    {NULL, NULL}
89 };
90
91
92
93 Eina_List *_elm_win_list = NULL;
94 int _elm_win_deferred_free = 0;
95
96 // exmaple shot spec (wait 0.1 sec then save as my-window.png):
97 // ELM_ENGINE="shot:delay=0.1:file=my-window.png"
98
99 static double
100 _shot_delay_get(Elm_Win *win)
101 {
102    char *p, *pd;
103    char *d = strdup(win->shot.info);
104
105    if (!d) return 0.5;
106    for (p = (char *)win->shot.info; *p; p++)
107      {
108         if (!strncmp(p, "delay=", 6))
109           {
110              double v;
111
112              for (pd = d, p += 6; (*p) && (*p != ':'); p++, pd++)
113                {
114                   *pd = *p;
115                }
116              *pd = 0;
117              v = atof(d);
118              free(d);
119              return v;
120           }
121      }
122    free(d);
123    return 0.5;
124 }
125
126 static char *
127 _shot_file_get(Elm_Win *win)
128 {
129    char *p;
130    char *tmp = strdup(win->shot.info);
131    char *repname = NULL;
132
133    if (!tmp) return NULL;
134
135    for (p = (char *)win->shot.info; *p; p++)
136      {
137         if (!strncmp(p, "file=", 5))
138           {
139              strcpy(tmp, p + 5);
140              if (!win->shot.repeat_count) return tmp;
141              else
142                {
143                   char *dotptr = strrchr(tmp, '.');
144                   if (dotptr)
145                     {
146                        repname = malloc(sizeof(char)*(strlen(tmp) + 16));
147                        strncpy(repname, tmp, dotptr - tmp);
148                        sprintf(repname + (dotptr - tmp), "%03i",
149                                win->shot.shot_counter + 1);
150                        strcat(repname, dotptr);
151                        return repname;
152                     }
153                }
154           }
155      }
156    free(tmp);
157    if (!win->shot.repeat_count) return strdup("out.png");
158    else
159      {
160         repname = malloc(sizeof(char) * 24);
161         sprintf(repname, "out%03i.png", win->shot.shot_counter + 1);
162         return repname;
163      }
164 }
165
166 static int
167 _shot_repeat_count_get(Elm_Win *win)
168 {
169
170    char *p, *pd;
171    char *d = strdup(win->shot.info);
172
173    if (!d) return 0;
174    for (p = (char *)win->shot.info; *p; p++)
175      {
176         if (!strncmp(p, "repeat=", 7))
177           {
178              int v;
179
180              for (pd = d, p += 7; (*p) && (*p != ':'); p++, pd++)
181                {
182                   *pd = *p;
183                }
184              *pd = 0;
185              v = atoi(d);
186              if (v < 0) v = 0;
187              if (v > 1000) v = 999;
188              free(d);
189              return v;
190           }
191      }
192    free(d);
193    return 0;
194 }
195
196 static char *
197 _shot_key_get(Elm_Win *win __UNUSED__)
198 {
199    return NULL;
200 }
201
202 static char *
203 _shot_flags_get(Elm_Win *win __UNUSED__)
204 {
205    return NULL;
206 }
207
208 static void
209 _shot_do(Elm_Win *win)
210 {
211    Ecore_Evas *ee;
212    Evas_Object *o;
213    unsigned int *pixels;
214    int w, h;
215    char *file, *key, *flags;
216
217    ecore_evas_manual_render(win->ee);
218    pixels = (void *)ecore_evas_buffer_pixels_get(win->ee);
219    if (!pixels) return;
220    ecore_evas_geometry_get(win->ee, NULL, NULL, &w, &h);
221    if ((w < 1) || (h < 1)) return;
222    file = _shot_file_get(win);
223    if (!file) return;
224    key = _shot_key_get(win);
225    flags = _shot_flags_get(win);
226    ee = ecore_evas_buffer_new(1, 1);
227    o = evas_object_image_add(ecore_evas_get(ee));
228    evas_object_image_alpha_set(o, ecore_evas_alpha_get(win->ee));
229    evas_object_image_size_set(o, w, h);
230    evas_object_image_data_set(o, pixels);
231    if (!evas_object_image_save(o, file, key, flags))
232      {
233         ERR("Cannot save window to '%s' (key '%s', flags '%s')",
234             file, key, flags);
235      }
236    free(file);
237    if (key) free(key);
238    if (flags) free(flags);
239    ecore_evas_free(ee);
240    if (win->shot.repeat_count) win->shot.shot_counter++;
241 }
242
243 static Eina_Bool
244 _shot_delay(void *data)
245 {
246    Elm_Win *win = data;
247    _shot_do(win);
248    if (win->shot.repeat_count)
249      {
250         int remainshot = (win->shot.repeat_count - win->shot.shot_counter);
251         if (remainshot > 0) return EINA_TRUE;
252      }
253    win->shot.timer = NULL;
254    elm_exit();
255    return EINA_FALSE;
256 }
257
258 static void
259 _shot_init(Elm_Win *win)
260 {
261    if (!win->shot.info) return;
262    win->shot.repeat_count = _shot_repeat_count_get(win);
263    win->shot.shot_counter = 0;
264 }
265
266 static void
267 _shot_handle(Elm_Win *win)
268 {
269    if (!win->shot.info) return;
270    win->shot.timer = ecore_timer_add(_shot_delay_get(win), _shot_delay, win);
271 }
272
273 static void
274 _elm_win_move(Ecore_Evas *ee)
275 {
276    Evas_Object *obj = ecore_evas_object_associate_get(ee);
277    Elm_Win *win;
278    int x, y;
279
280    if (!obj) return;
281    win = elm_widget_data_get(obj);
282    if (!win) return;
283    ecore_evas_geometry_get(ee, &x, &y, NULL, NULL);
284    win->screen.x = x;
285    win->screen.y = y;
286    evas_object_smart_callback_call(win->win_obj, SIG_MOVED, NULL);
287 }
288
289 static void
290 _elm_win_resize(Ecore_Evas *ee)
291 {
292    Evas_Object *obj = ecore_evas_object_associate_get(ee);
293    Elm_Win *win;
294
295    if (!obj) return;
296    win = elm_widget_data_get(obj);
297    if (!win) return;
298    if (win->deferred_resize_job) ecore_job_del(win->deferred_resize_job);
299    win->deferred_resize_job = ecore_job_add(_elm_win_resize_job, win);
300 }
301
302 static void
303 _elm_win_focus_in(Ecore_Evas *ee)
304 {
305    Evas_Object *obj = ecore_evas_object_associate_get(ee);
306    Elm_Win *win;
307
308    if (!obj) return;
309    win = elm_widget_data_get(obj);
310    if (!win) return;
311    _elm_widget_top_win_focused_set(win->win_obj, EINA_TRUE);
312    if (win->show_count == 1)
313      {
314         elm_object_focus_set(win->win_obj, EINA_TRUE);
315         win->show_count++;
316      }
317    else
318      elm_widget_focus_restore(win->win_obj);
319    evas_object_smart_callback_call(win->win_obj, SIG_FOCUS_IN, NULL);
320    win->focus_highlight.cur.visible = EINA_TRUE;
321    _elm_win_focus_highlight_reconfigure_job_start(win);
322    if (win->frame_obj)
323      {
324      }
325    else if (win->img_obj)
326      {
327         /* do nothing */
328      }
329 }
330
331 static void
332 _elm_win_focus_out(Ecore_Evas *ee)
333 {
334    Evas_Object *obj = ecore_evas_object_associate_get(ee);
335    Elm_Win *win;
336
337    if (!obj) return;
338    win = elm_widget_data_get(obj);
339    if (!win) return;
340    elm_object_focus_set(win->win_obj, EINA_FALSE);
341    _elm_widget_top_win_focused_set(win->win_obj, EINA_FALSE);
342    evas_object_smart_callback_call(win->win_obj, SIG_FOCUS_OUT, NULL);
343    win->focus_highlight.cur.visible = EINA_FALSE;
344    _elm_win_focus_highlight_reconfigure_job_start(win);
345    if (win->frame_obj)
346      {
347      }
348    else if (win->img_obj)
349      {
350         /* do nothing */
351      }
352 }
353
354 static Eina_Bool
355 _elm_win_focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next)
356 {
357    Elm_Win *wd = elm_widget_data_get(obj);
358    const Eina_List *items;
359    void *(*list_data_get) (const Eina_List *list);
360
361    if (!wd)
362      return EINA_FALSE;
363
364    /* Focus chain */
365    if (wd->subobjs)
366      {
367         if (!(items = elm_widget_focus_custom_chain_get(obj)))
368           {
369              items = wd->subobjs;
370              if (!items)
371                return EINA_FALSE;
372           }
373         list_data_get = eina_list_data_get;
374
375         elm_widget_focus_list_next_get(obj, items, list_data_get, dir, next);
376
377         if (*next)
378           return EINA_TRUE;
379      }
380
381    *next = (Evas_Object *)obj;
382    return EINA_FALSE;
383 }
384
385 static void
386 _elm_win_on_focus_hook(void *data __UNUSED__, Evas_Object *obj)
387 {
388    Elm_Win *win = elm_widget_data_get(obj);
389    if (!win) return;
390
391    if (win->img_obj)
392       evas_object_focus_set(win->img_obj, elm_widget_focus_get(obj));
393    else
394       evas_object_focus_set(obj, elm_widget_focus_get(obj));
395 }
396
397 static Eina_Bool
398 _elm_win_event_cb(Evas_Object *obj, Evas_Object *src __UNUSED__, Evas_Callback_Type type, void *event_info)
399 {
400    if (type == EVAS_CALLBACK_KEY_DOWN)
401      {
402         Evas_Event_Key_Down *ev = event_info;
403         if (!strcmp(ev->keyname, "Tab"))
404           {
405              if (evas_key_modifier_is_set(ev->modifiers, "Shift"))
406                elm_widget_focus_cycle(obj, ELM_FOCUS_PREVIOUS);
407              else
408                elm_widget_focus_cycle(obj, ELM_FOCUS_NEXT);
409              ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
410              return EINA_TRUE;
411           }
412         else if ((!strcmp(ev->keyname, "Left")) ||
413                  (!strcmp(ev->keyname, "KP_Left")))
414           {
415              //TODO : woohyun jung
416           }
417         else if ((!strcmp(ev->keyname, "Right")) ||
418                  (!strcmp(ev->keyname, "KP_Right")))
419           {
420              //TODO : woohyun jung
421           }
422         else if ((!strcmp(ev->keyname, "Up")) ||
423                  (!strcmp(ev->keyname, "KP_Up")))
424           {
425              //TODO : woohyun jung
426           }
427         else if ((!strcmp(ev->keyname, "Down")) ||
428                  (!strcmp(ev->keyname, "KP_Down")))
429           {
430              //TODO : woohyun jung
431           }
432      }
433
434    return EINA_FALSE;
435 }
436
437 static void
438 _deferred_ecore_evas_free(void *data)
439 {
440    ecore_evas_free(data);
441    _elm_win_deferred_free--;
442 }
443
444 static void
445 _elm_win_obj_callback_show(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
446 {
447    Elm_Win *win = data;
448
449    if (!win->show_count) win->show_count++;
450    if (win->shot.info) _shot_handle(win);
451 }
452
453 static void
454 _elm_win_obj_callback_hide(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
455 {
456    Elm_Win *win = data;
457
458    if (win->frame_obj)
459      {
460      }
461    else if (win->img_obj)
462      {
463         evas_object_hide(win->img_obj);
464      }
465 }
466
467 static void
468 _elm_win_obj_callback_del(void *data, Evas *e, Evas_Object *obj, void *event_info __UNUSED__)
469 {
470    Elm_Win *win = data;
471    Evas_Object *child;
472
473    if (win->parent)
474      {
475         evas_object_event_callback_del_full(win->parent, EVAS_CALLBACK_DEL,
476                                             _elm_win_obj_callback_parent_del, win);
477         win->parent = NULL;
478      }
479    if (win->autodel_clear) *(win->autodel_clear) = -1;
480    _elm_win_list = eina_list_remove(_elm_win_list, win->win_obj);
481    while (win->subobjs) elm_win_resize_object_del(obj, win->subobjs->data);
482    if (win->ee)
483      {
484         ecore_evas_callback_delete_request_set(win->ee, NULL);
485         ecore_evas_callback_resize_set(win->ee, NULL);
486      }
487    if (win->deferred_resize_job) ecore_job_del(win->deferred_resize_job);
488    if (win->deferred_child_eval_job) ecore_job_del(win->deferred_child_eval_job);
489    if (win->shot.info) eina_stringshare_del(win->shot.info);
490    if (win->shot.timer) ecore_timer_del(win->shot.timer);
491    evas_object_event_callback_del_full(win->win_obj, EVAS_CALLBACK_DEL,
492                                        _elm_win_obj_callback_del, win);
493    while (((child = evas_object_bottom_get(win->evas))) &&
494           (child != obj))
495      {
496         evas_object_del(child);
497      }
498    while (((child = evas_object_top_get(win->evas))) &&
499           (child != obj))
500      {
501         evas_object_del(child);
502      }
503 #ifdef HAVE_ELEMENTARY_X
504    if (win->client_message_handler)
505      ecore_event_handler_del(win->client_message_handler);
506 #endif
507    // FIXME: Why are we flushing edje on every window destroy ??
508    //   edje_file_cache_flush();
509    //   edje_collection_cache_flush();
510    //   evas_image_cache_flush(win->evas);
511    //   evas_font_cache_flush(win->evas);
512    // FIXME: we are in the del handler for the object and delete the canvas
513    // that lives under it from the handler... nasty. deferring doesn't help either
514
515    if (win->img_obj)
516      {
517         win->img_obj = NULL;
518      }
519    else
520      {
521         if (win->ee)
522           {
523              ecore_job_add(_deferred_ecore_evas_free, win->ee);
524              _elm_win_deferred_free++;
525           }
526      }
527
528    _elm_win_focus_highlight_shutdown(win);
529    eina_stringshare_del(win->focus_highlight.style);
530
531    free(win);
532
533    if ((!_elm_win_list) &&
534        (elm_policy_get(ELM_POLICY_QUIT) == ELM_POLICY_QUIT_LAST_WINDOW_CLOSED))
535      {
536         edje_file_cache_flush();
537         edje_collection_cache_flush();
538         evas_image_cache_flush(e);
539         evas_font_cache_flush(e);
540         elm_exit();
541      }
542 }
543
544 static void
545 _elm_win_obj_callback_img_obj_del(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
546 {
547    Elm_Win *win = data;
548    if (!win->img_obj) return;
549    evas_object_event_callback_del_full
550       (win->img_obj, EVAS_CALLBACK_DEL, _elm_win_obj_callback_img_obj_del, win);
551    evas_object_del(win->img_obj);
552 }
553
554 static void
555 _elm_win_obj_callback_parent_del(void *data, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
556 {
557    Elm_Win *win = data;
558    if (obj == win->parent) win->parent = NULL;
559 }
560
561 static void
562 _elm_win_obj_intercept_move(void *data, Evas_Object *obj, Evas_Coord x, Evas_Coord y)
563 {
564    Elm_Win *win = data;
565
566    if (win->img_obj)
567      {
568         if ((x != win->screen.x) || (y != win->screen.y))
569           {
570              win->screen.x = x;
571              win->screen.y = y;
572              evas_object_smart_callback_call(win->win_obj, SIG_MOVED, NULL);
573           }
574      }
575    else
576      {
577         evas_object_move(obj, x, y);
578      }
579 }
580
581 static void
582 _elm_win_obj_intercept_show(void *data, Evas_Object *obj)
583 {
584    Elm_Win *win = data;
585    // this is called to make sure all smart containers have calculated their
586    // sizes BEFORE we show the window to make sure it initially appears at
587    // our desired size (ie min size is known first)
588    evas_smart_objects_calculate(evas_object_evas_get(obj));
589    evas_object_show(obj);
590    if (win->frame_obj)
591      {
592      }
593    else if (win->img_obj)
594      {
595         evas_object_show(win->img_obj);
596      }
597 }
598
599 static void
600 _elm_win_obj_callback_move(void *data, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
601 {
602    Elm_Win *win = data;
603
604    if (ecore_evas_override_get(win->ee))
605      {
606         Evas_Coord x, y;
607
608         evas_object_geometry_get(obj, &x, &y, NULL, NULL);
609         win->screen.x = x;
610         win->screen.y = y;
611         evas_object_smart_callback_call(win->win_obj, SIG_MOVED, NULL);
612      }
613    if (win->frame_obj)
614      {
615      }
616    else if (win->img_obj)
617      {
618         Evas_Coord x, y;
619
620         evas_object_geometry_get(obj, &x, &y, NULL, NULL);
621         win->screen.x = x;
622         win->screen.y = y;
623 //        evas_object_move(win->img_obj, x, y);
624      }
625 }
626
627 static void
628 _elm_win_obj_callback_resize(void *data, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
629 {
630    Elm_Win *win = data;
631
632    if (win->frame_obj)
633      {
634      }
635    else if (win->img_obj)
636      {
637         Evas_Coord w = 1, h = 1;
638
639         evas_object_geometry_get(obj, NULL, NULL, &w, &h);
640         if (w < 1) w = 1;
641         if (h < 1) h = 1;
642         evas_object_image_size_set(win->img_obj, w, h);
643      }
644 }
645
646 static void
647 _elm_win_delete_request(Ecore_Evas *ee)
648 {
649    Evas_Object *obj = ecore_evas_object_associate_get(ee);
650    Elm_Win *win;
651    if (strcmp(elm_widget_type_get(obj), "win")) return;
652
653    win = elm_widget_data_get(obj);
654    if (!win) return;
655    int autodel = win->autodel;
656    win->autodel_clear = &autodel;
657    evas_object_ref(win->win_obj);
658    evas_object_smart_callback_call(win->win_obj, SIG_DELETE_REQUEST, NULL);
659    // FIXME: if above callback deletes - then the below will be invalid
660    if (autodel) evas_object_del(win->win_obj);
661    else win->autodel_clear = NULL;
662    evas_object_unref(win->win_obj);
663 }
664
665 static void
666 _elm_win_resize_job(void *data)
667 {
668    Elm_Win *win = data;
669    const Eina_List *l;
670    Evas_Object *obj;
671    int w, h;
672
673    win->deferred_resize_job = NULL;
674    ecore_evas_geometry_get(win->ee, NULL, NULL, &w, &h);
675    evas_object_resize(win->win_obj, w, h);
676    if (win->frame_obj)
677      {
678      }
679    else if (win->img_obj)
680      {
681      }
682    EINA_LIST_FOREACH(win->subobjs, l, obj)
683      {
684         evas_object_move(obj, 0, 0);
685         evas_object_resize(obj, w, h);
686      }
687 }
688
689 #ifdef HAVE_ELEMENTARY_X
690 static void
691 _elm_win_xwindow_get(Elm_Win *win)
692 {
693    win->xwin = 0;
694
695 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
696    if (ENGINE_COMPARE(ELM_SOFTWARE_X11))
697      {
698        if (win->ee) win->xwin = ecore_evas_software_x11_window_get(win->ee);
699      }
700    else if (ENGINE_COMPARE(ELM_SOFTWARE_X11) ||
701             ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
702             ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE) ||
703             ENGINE_COMPARE(ELM_SOFTWARE_SDL) ||
704             ENGINE_COMPARE(ELM_SOFTWARE_16_SDL) ||
705             ENGINE_COMPARE(ELM_OPENGL_SDL))
706      {
707      }
708    else if (ENGINE_COMPARE(ELM_SOFTWARE_16_X11))
709      {
710         if (win->ee) win->xwin = ecore_evas_software_x11_16_window_get(win->ee);
711      }
712    else if (ENGINE_COMPARE(ELM_SOFTWARE_8_X11))
713      {
714         if (win->ee) win->xwin = ecore_evas_software_x11_8_window_get(win->ee);
715      }
716 /* killed
717    else if (ENGINE_COMPARE(ELM_XRENDER_X11))
718      {
719         if (win->ee) win->xwin = ecore_evas_xrender_x11_window_get(win->ee);
720      }
721  */
722    else if (ENGINE_COMPARE(ELM_OPENGL_X11))
723      {
724         if (win->ee) win->xwin = ecore_evas_gl_x11_window_get(win->ee);
725      }
726    else if (ENGINE_COMPARE(ELM_SOFTWARE_WIN32))
727      {
728         if (win->ee) win->xwin = (long)ecore_evas_win32_window_get(win->ee);
729      }
730 #undef ENGINE_COMPARE
731 }
732 #endif
733
734 #ifdef HAVE_ELEMENTARY_X
735 static void
736 _elm_win_xwin_update(Elm_Win *win)
737 {
738    _elm_win_xwindow_get(win);
739    if (win->parent)
740      {
741         Elm_Win *win2;
742
743         win2 = elm_widget_data_get(win->parent);
744         if (win2)
745           {
746              if (win->xwin)
747                ecore_x_icccm_transient_for_set(win->xwin, win2->xwin);
748           }
749      }
750
751    if (!win->xwin) return; /* nothing more to do */
752
753    switch (win->type)
754      {
755       case ELM_WIN_BASIC:
756          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_NORMAL);
757          break;
758       case ELM_WIN_DIALOG_BASIC:
759          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_DIALOG);
760          break;
761       case ELM_WIN_DESKTOP:
762          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_DESKTOP);
763          break;
764       case ELM_WIN_DOCK:
765          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_DOCK);
766          break;
767       case ELM_WIN_TOOLBAR:
768          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_TOOLBAR);
769          break;
770       case ELM_WIN_MENU:
771          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_MENU);
772          break;
773       case ELM_WIN_UTILITY:
774          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_UTILITY);
775          break;
776       case ELM_WIN_SPLASH:
777          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_SPLASH);
778          break;
779       case ELM_WIN_DROPDOWN_MENU:
780          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_DROPDOWN_MENU);
781          break;
782       case ELM_WIN_POPUP_MENU:
783          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_POPUP_MENU);
784          break;
785       case ELM_WIN_TOOLTIP:
786          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_TOOLTIP);
787          break;
788       case ELM_WIN_NOTIFICATION:
789          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_NOTIFICATION);
790          break;
791       case ELM_WIN_COMBO:
792          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_COMBO);
793          break;
794       case ELM_WIN_DND:
795          ecore_x_netwm_window_type_set(win->xwin, ECORE_X_WINDOW_TYPE_DND);
796          break;
797       default:
798          break;
799      }
800    ecore_x_e_virtual_keyboard_state_set
801       (win->xwin, (Ecore_X_Virtual_Keyboard_State)win->kbdmode);
802 }
803 #endif
804
805 static void
806 _elm_win_eval_subobjs(Evas_Object *obj)
807 {
808    const Eina_List *l;
809    const Evas_Object *child;
810
811    Elm_Win *win = elm_widget_data_get(obj);
812    Evas_Coord w, h, minw = -1, minh = -1, maxw = -1, maxh = -1;
813    int xx = 1, xy = 1;
814    double wx, wy;
815
816    EINA_LIST_FOREACH(win->subobjs, l, child)
817      {
818         evas_object_size_hint_weight_get(child, &wx, &wy);
819         if (wx == 0.0) xx = 0;
820         if (wy == 0.0) xy = 0;
821
822         evas_object_size_hint_min_get(child, &w, &h);
823         if (w < 1) w = 1;
824         if (h < 1) h = 1;
825         if (w > minw) minw = w;
826         if (h > minh) minh = h;
827
828         evas_object_size_hint_max_get(child, &w, &h);
829         if (w < 1) w = -1;
830         if (h < 1) h = -1;
831         if (maxw == -1) maxw = w;
832         else if ((w > 0) && (w < maxw)) maxw = w;
833         if (maxh == -1) maxh = h;
834         else if ((h > 0) && (h < maxh)) maxh = h;
835      }
836    if (!xx) maxw = minw;
837    else maxw = 32767;
838    if (!xy) maxh = minh;
839    else maxh = 32767;
840    evas_object_size_hint_min_set(obj, minw, minh);
841    evas_object_size_hint_max_set(obj, maxw, maxh);
842    evas_object_geometry_get(obj, NULL, NULL, &w, &h);
843    if (w < minw) w = minw;
844    if (h < minh) h = minh;
845    if ((maxw >= 0) && (w > maxw)) w = maxw;
846    if ((maxh >= 0) && (h > maxh)) h = maxh;
847    evas_object_resize(obj, w, h);
848 }
849
850 static void
851 _elm_win_subobj_callback_del(void *data, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
852 {
853    Elm_Win *win = elm_widget_data_get(data);
854    win->subobjs = eina_list_remove(win->subobjs, obj);
855    _elm_win_eval_subobjs(win->win_obj);
856 }
857
858 static void
859 _elm_win_subobj_callback_changed_size_hints(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
860 {
861    _elm_win_eval_subobjs(data);
862 }
863
864 void
865 _elm_win_shutdown(void)
866 {
867    while (_elm_win_list)
868      evas_object_del(_elm_win_list->data);
869 }
870
871 void
872 _elm_win_rescale(Elm_Theme *th, Eina_Bool use_theme)
873 {
874    const Eina_List *l;
875    Evas_Object *obj;
876
877    if (!use_theme)
878      {
879         EINA_LIST_FOREACH(_elm_win_list, l, obj)
880           elm_widget_theme(obj);
881      }
882    else
883      {
884         EINA_LIST_FOREACH(_elm_win_list, l, obj)
885           elm_widget_theme_specific(obj, th, EINA_FALSE);
886      }
887 }
888
889 #ifdef HAVE_ELEMENTARY_X
890 static Eina_Bool
891 _elm_win_client_message(void *data, int type __UNUSED__, void *event)
892 {
893    Elm_Win *win = data;
894    Ecore_X_Event_Client_Message *e = event;
895
896    if (e->format != 32) return ECORE_CALLBACK_PASS_ON;
897    if (e->message_type == ECORE_X_ATOM_E_COMP_FLUSH)
898      {
899         if ((unsigned)e->data.l[0] == win->xwin)
900           {
901              Evas *evas = evas_object_evas_get(win->win_obj);
902              if (evas)
903                {
904                   edje_file_cache_flush();
905                   edje_collection_cache_flush();
906                   evas_image_cache_flush(evas);
907                   evas_font_cache_flush(evas);
908                }
909           }
910      }
911    else if (e->message_type == ECORE_X_ATOM_E_COMP_DUMP)
912      {
913         if ((unsigned)e->data.l[0] == win->xwin)
914           {
915              Evas *evas = evas_object_evas_get(win->win_obj);
916              if (evas)
917                {
918                   edje_file_cache_flush();
919                   edje_collection_cache_flush();
920                   evas_image_cache_flush(evas);
921                   evas_font_cache_flush(evas);
922                   evas_render_dump(evas);
923                }
924           }
925      }
926    return ECORE_CALLBACK_PASS_ON;
927 }
928 #endif
929
930 static void
931 _elm_win_focus_target_move(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
932 {
933    Elm_Win *win = data;
934
935    win->focus_highlight.geometry_changed = EINA_TRUE;
936    _elm_win_focus_highlight_reconfigure_job_start(win);
937 }
938
939 static void
940 _elm_win_focus_target_resize(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
941 {
942    Elm_Win *win = data;
943
944    win->focus_highlight.geometry_changed = EINA_TRUE;
945    _elm_win_focus_highlight_reconfigure_job_start(win);
946 }
947
948 static void
949 _elm_win_focus_target_del(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
950 {
951    Elm_Win *win = data;
952
953    win->focus_highlight.cur.target = NULL;
954
955    _elm_win_focus_highlight_reconfigure_job_start(win);
956 }
957
958 static void
959 _elm_win_focus_target_callbacks_add(Elm_Win *win)
960 {
961    Evas_Object *obj = win->focus_highlight.cur.target;
962
963    evas_object_event_callback_add(obj, EVAS_CALLBACK_MOVE,
964                                   _elm_win_focus_target_move, win);
965    evas_object_event_callback_add(obj, EVAS_CALLBACK_RESIZE,
966                                   _elm_win_focus_target_resize, win);
967    evas_object_event_callback_add(obj, EVAS_CALLBACK_DEL,
968                                   _elm_win_focus_target_del, win);
969 }
970
971 static void
972 _elm_win_focus_target_callbacks_del(Elm_Win *win)
973 {
974    Evas_Object *obj = win->focus_highlight.cur.target;
975
976    evas_object_event_callback_del_full(obj, EVAS_CALLBACK_MOVE,
977                                        _elm_win_focus_target_move, win);
978    evas_object_event_callback_del_full(obj, EVAS_CALLBACK_RESIZE,
979                                        _elm_win_focus_target_resize, win);
980    evas_object_event_callback_del_full(obj, EVAS_CALLBACK_DEL,
981                                        _elm_win_focus_target_del, win);
982 }
983
984 static Evas_Object *
985 _elm_win_focus_target_get(Evas_Object *obj)
986 {
987    Evas_Object *o = obj;
988
989    do
990      {
991         if (elm_widget_is(o))
992           {
993              if (!elm_widget_highlight_ignore_get(o))
994                break;
995              o = elm_widget_parent_get(o);
996              if (!o)
997                o = evas_object_smart_parent_get(o);
998           }
999         else
1000           {
1001              o = elm_widget_parent_widget_get(o);
1002              if (!o)
1003                o = evas_object_smart_parent_get(o);
1004           }
1005      }
1006    while (o);
1007
1008    return o;
1009 }
1010
1011 static void
1012 _elm_win_object_focus_in(void *data, Evas *e __UNUSED__, void *event_info)
1013 {
1014    Evas_Object *obj = event_info, *target;
1015    Elm_Win *win = data;
1016
1017    if (win->focus_highlight.cur.target == obj)
1018      return;
1019
1020    target = _elm_win_focus_target_get(obj);
1021    win->focus_highlight.cur.target = target;
1022    if (elm_widget_highlight_in_theme_get(target))
1023      win->focus_highlight.cur.handled = EINA_TRUE;
1024    else
1025      _elm_win_focus_target_callbacks_add(win);
1026
1027    _elm_win_focus_highlight_reconfigure_job_start(win);
1028 }
1029
1030 static void
1031 _elm_win_object_focus_out(void *data, Evas *e __UNUSED__, void *event_info __UNUSED__)
1032 {
1033    Elm_Win *win = data;
1034
1035    if (!win->focus_highlight.cur.target)
1036      return;
1037
1038    if (!win->focus_highlight.cur.handled)
1039      _elm_win_focus_target_callbacks_del(win);
1040    win->focus_highlight.cur.target = NULL;
1041    win->focus_highlight.cur.handled = EINA_FALSE;
1042
1043    _elm_win_focus_highlight_reconfigure_job_start(win);
1044 }
1045
1046 static void
1047 _elm_win_focus_highlight_hide(void *data __UNUSED__, Evas_Object *obj, const char *emission __UNUSED__, const char *source __UNUSED__)
1048 {
1049    evas_object_hide(obj);
1050 }
1051
1052 static void
1053 _elm_win_focus_highlight_init(Elm_Win *win)
1054 {
1055    evas_event_callback_add(win->evas, EVAS_CALLBACK_CANVAS_OBJECT_FOCUS_IN,
1056                            _elm_win_object_focus_in, win);
1057    evas_event_callback_add(win->evas,
1058                            EVAS_CALLBACK_CANVAS_OBJECT_FOCUS_OUT,
1059                            _elm_win_object_focus_out, win);
1060
1061    win->focus_highlight.cur.target = evas_focus_get(win->evas);
1062
1063    win->focus_highlight.top = edje_object_add(win->evas);
1064    win->focus_highlight.changed_theme = EINA_TRUE;
1065    edje_object_signal_callback_add(win->focus_highlight.top,
1066                                    "elm,action,focus,hide,end", "",
1067                                    _elm_win_focus_highlight_hide, NULL);
1068    edje_object_signal_callback_add(win->focus_highlight.top,
1069                                    "elm,action,focus,anim,end", "",
1070                                    _elm_win_focus_highlight_anim_end, win);
1071    _elm_win_focus_highlight_reconfigure_job_start(win);
1072 }
1073
1074 static void
1075 _elm_win_focus_highlight_shutdown(Elm_Win *win)
1076 {
1077    _elm_win_focus_highlight_reconfigure_job_stop(win);
1078    if (win->focus_highlight.cur.target)
1079      {
1080         _elm_win_focus_target_callbacks_del(win);
1081         win->focus_highlight.cur.target = NULL;
1082      }
1083    if (win->focus_highlight.top)
1084      {
1085         evas_object_del(win->focus_highlight.top);
1086         win->focus_highlight.top = NULL;
1087      }
1088
1089    evas_event_callback_del_full(win->evas,
1090                                 EVAS_CALLBACK_CANVAS_OBJECT_FOCUS_IN,
1091                                 _elm_win_object_focus_in, win);
1092    evas_event_callback_del_full(win->evas,
1093                                 EVAS_CALLBACK_CANVAS_OBJECT_FOCUS_OUT,
1094                                 _elm_win_object_focus_out, win);
1095 }
1096
1097 static void
1098 _elm_win_focus_highlight_visible_set(Elm_Win *win, Eina_Bool visible)
1099 {
1100    Evas_Object *top;
1101
1102    top = win->focus_highlight.top;
1103    if (visible)
1104      {
1105         if (top)
1106           {
1107              evas_object_show(top);
1108              edje_object_signal_emit(top, "elm,action,focus,show", "elm");
1109           }
1110      }
1111    else
1112      {
1113         if (top)
1114           edje_object_signal_emit(top, "elm,action,focus,hide", "elm");
1115      }
1116 }
1117
1118 static void
1119 _elm_win_focus_highlight_reconfigure_job(void *data)
1120 {
1121    _elm_win_focus_highlight_reconfigure((Elm_Win *)data);
1122 }
1123
1124 static void
1125 _elm_win_focus_highlight_reconfigure_job_start(Elm_Win *win)
1126 {
1127    if (win->focus_highlight.reconf_job)
1128      ecore_job_del(win->focus_highlight.reconf_job);
1129    win->focus_highlight.reconf_job = ecore_job_add(
1130       _elm_win_focus_highlight_reconfigure_job, win);
1131 }
1132
1133 static void
1134 _elm_win_focus_highlight_reconfigure_job_stop(Elm_Win *win)
1135 {
1136    if (win->focus_highlight.reconf_job)
1137      ecore_job_del(win->focus_highlight.reconf_job);
1138    win->focus_highlight.reconf_job = NULL;
1139 }
1140
1141 static void
1142 _elm_win_focus_highlight_simple_setup(Elm_Win *win, Evas_Object *obj)
1143 {
1144    Evas_Object *clip, *target = win->focus_highlight.cur.target;
1145    Evas_Coord x, y, w, h;
1146
1147    clip = evas_object_clip_get(target);
1148    evas_object_geometry_get(target, &x, &y, &w, &h);
1149
1150    evas_object_move(obj, x, y);
1151    evas_object_resize(obj, w, h);
1152    evas_object_clip_set(obj, clip);
1153 }
1154
1155 static void
1156 _elm_win_focus_highlight_anim_setup(Elm_Win *win, Evas_Object *obj)
1157 {
1158    Evas_Coord tx, ty, tw, th;
1159    Evas_Coord w, h, px, py, pw, ph;
1160    Edje_Message_Int_Set *m;
1161    Evas_Object *previous = win->focus_highlight.prev.target;
1162    Evas_Object *target = win->focus_highlight.cur.target;
1163
1164    evas_object_geometry_get(win->win_obj, NULL, NULL, &w, &h);
1165    evas_object_geometry_get(target, &tx, &ty, &tw, &th);
1166    evas_object_geometry_get(previous, &px, &py, &pw, &ph);
1167    evas_object_move(obj, 0, 0);
1168    evas_object_resize(obj, tw, th);
1169    evas_object_clip_unset(obj);
1170
1171    m = alloca(sizeof(*m) + (sizeof(int) * 8));
1172    m->count = 8;
1173    m->val[0] = px;
1174    m->val[1] = py;
1175    m->val[2] = pw;
1176    m->val[3] = ph;
1177    m->val[4] = tx;
1178    m->val[5] = ty;
1179    m->val[6] = tw;
1180    m->val[7] = th;
1181    edje_object_message_send(obj, EDJE_MESSAGE_INT_SET, 1, m);
1182 }
1183
1184 static void
1185 _elm_win_focus_highlight_anim_end(void *data, Evas_Object *obj, const char *emission __UNUSED__, const char *source __UNUSED__)
1186 {
1187    Elm_Win *win = data;
1188    _elm_win_focus_highlight_simple_setup(win, obj);
1189 }
1190
1191 static void
1192 _elm_win_focus_highlight_reconfigure(Elm_Win *win)
1193 {
1194    Evas_Object *target = win->focus_highlight.cur.target;
1195    Evas_Object *previous = win->focus_highlight.prev.target;
1196    Evas_Object *top = win->focus_highlight.top;
1197    Eina_Bool visible_changed;
1198    Eina_Bool common_visible;
1199    const char *sig = NULL;
1200
1201    _elm_win_focus_highlight_reconfigure_job_stop(win);
1202
1203    visible_changed = (win->focus_highlight.cur.visible !=
1204                       win->focus_highlight.prev.visible);
1205
1206    if ((target == previous) && (!visible_changed) &&
1207        (!win->focus_highlight.geometry_changed))
1208      return;
1209
1210    if ((previous) && (win->focus_highlight.prev.handled))
1211      elm_widget_signal_emit(previous, "elm,action,focus_highlight,hide", "elm");
1212
1213    if (!target)
1214      common_visible = EINA_FALSE;
1215    else if (win->focus_highlight.cur.handled)
1216      {
1217         common_visible = EINA_FALSE;
1218         if (win->focus_highlight.cur.visible)
1219           sig = "elm,action,focus_highlight,show";
1220         else
1221           sig = "elm,action,focus_highlight,hide";
1222      }
1223    else
1224      common_visible = win->focus_highlight.cur.visible;
1225
1226    _elm_win_focus_highlight_visible_set(win, common_visible);
1227    if (sig)
1228      elm_widget_signal_emit(target, sig, "elm");
1229
1230    if ((!target) || (!common_visible) || (win->focus_highlight.cur.handled))
1231      goto the_end;
1232
1233    if (win->focus_highlight.changed_theme)
1234      {
1235         const char *str;
1236         if (win->focus_highlight.style)
1237           str = win->focus_highlight.style;
1238         else
1239           str = "default";
1240         _elm_theme_object_set(win->win_obj, top, "focus_highlight", "top",
1241                               str);
1242         win->focus_highlight.changed_theme = EINA_FALSE;
1243
1244         if (_elm_config->focus_highlight_animate)
1245           {
1246              str = edje_object_data_get(win->focus_highlight.top, "animate");
1247              win->focus_highlight.top_animate = ((str) && (!strcmp(str, "on")));
1248           }
1249      }
1250
1251    if ((win->focus_highlight.top_animate) && (previous) &&
1252        (!win->focus_highlight.prev.handled))
1253      _elm_win_focus_highlight_anim_setup(win, top);
1254    else
1255      _elm_win_focus_highlight_simple_setup(win, top);
1256    evas_object_raise(top);
1257
1258 the_end:
1259    win->focus_highlight.geometry_changed = EINA_FALSE;
1260    win->focus_highlight.prev = win->focus_highlight.cur;
1261 }
1262
1263 #ifdef ELM_DEBUG
1264 static void
1265 _debug_key_down(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info)
1266 {
1267    Evas_Event_Key_Down *ev = event_info;
1268
1269    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD)
1270      return;
1271
1272
1273    if ((strcmp(ev->keyname, "F12")) ||
1274        (!evas_key_modifier_is_set(ev->modifiers, "Control")))
1275      return;
1276
1277    printf("Tree graph generated.\n");
1278    elm_object_tree_dot_dump(obj, "./dump.dot");
1279 }
1280 #endif
1281
1282 static void
1283 _win_img_hide(void        *data,
1284               Evas        *e __UNUSED__,
1285               Evas_Object *obj __UNUSED__,
1286               void        *event_info __UNUSED__)
1287 {
1288    Elm_Win *win = data;
1289
1290    elm_widget_focus_hide_handle(win->win_obj);
1291 }
1292
1293 static void
1294 _win_img_mouse_up(void        *data,
1295                   Evas        *e __UNUSED__,
1296                   Evas_Object *obj __UNUSED__,
1297                   void        *event_info)
1298 {
1299    Elm_Win *win = data;
1300    Evas_Event_Mouse_Up *ev = event_info;
1301    if (!(ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD))
1302       elm_widget_focus_mouse_up_handle(win->win_obj);
1303 }
1304
1305 static void
1306 _win_img_focus_in(void        *data,
1307                   Evas        *e __UNUSED__,
1308                   Evas_Object *obj __UNUSED__,
1309                   void        *event_info __UNUSED__)
1310 {
1311    Elm_Win *win = data;
1312    elm_widget_focus_steal(win->win_obj);
1313 }
1314
1315 static void
1316 _win_img_focus_out(void        *data,
1317                    Evas        *e __UNUSED__,
1318                    Evas_Object *obj __UNUSED__,
1319                    void        *event_info __UNUSED__)
1320 {
1321    Elm_Win *win = data;
1322    elm_widget_focused_object_clear(win->win_obj);
1323 }
1324
1325 static void
1326 _win_inlined_image_set(Elm_Win *win)
1327 {
1328    evas_object_image_alpha_set(win->img_obj, EINA_FALSE);
1329    evas_object_image_filled_set(win->img_obj, EINA_TRUE);
1330    evas_object_event_callback_add(win->img_obj, EVAS_CALLBACK_DEL,
1331                                   _elm_win_obj_callback_img_obj_del, win);
1332
1333    evas_object_event_callback_add(win->img_obj, EVAS_CALLBACK_HIDE,
1334                                   _win_img_hide, win);
1335    evas_object_event_callback_add(win->img_obj, EVAS_CALLBACK_MOUSE_UP,
1336                                   _win_img_mouse_up, win);
1337    evas_object_event_callback_add(win->img_obj, EVAS_CALLBACK_FOCUS_IN,
1338                                   _win_img_focus_in, win);
1339    evas_object_event_callback_add(win->img_obj, EVAS_CALLBACK_FOCUS_OUT,
1340                                   _win_img_focus_out, win);
1341 }
1342
1343 EAPI Evas_Object *
1344 elm_win_add(Evas_Object *parent, const char *name, Elm_Win_Type type)
1345 {
1346    Elm_Win *win;
1347    const Eina_List *l;
1348    const char *fontpath;
1349
1350    win = ELM_NEW(Elm_Win);
1351
1352 #define FALLBACK_TRY(engine)                                              \
1353    if (!win->ee)                                                          \
1354    do {                                                                   \
1355         CRITICAL(engine " engine creation failed. Trying software X11."); \
1356         elm_engine_set(ELM_SOFTWARE_X11);                                 \
1357         win->ee = ecore_evas_software_x11_new(NULL, 0, 0, 0, 1, 1);       \
1358    } while (0)
1359 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1360
1361    switch (type)
1362      {
1363       case ELM_WIN_INLINED_IMAGE:
1364         if (!parent) break;
1365         {
1366            Evas *e = evas_object_evas_get(parent);
1367            Ecore_Evas *ee;
1368            if (!e) break;
1369            ee = ecore_evas_ecore_evas_get(e);
1370            if (!ee) break;
1371            win->img_obj = ecore_evas_object_image_new(ee);
1372            if (!win->img_obj) break;
1373            win->ee = ecore_evas_object_ecore_evas_get(win->img_obj);
1374            if (win->ee)
1375              {
1376                 _win_inlined_image_set(win);
1377                 break;
1378              }
1379            evas_object_del(win->img_obj);
1380            win->img_obj = NULL;
1381         }
1382         break;
1383       default:
1384         if (ENGINE_COMPARE(ELM_SOFTWARE_X11))
1385           {
1386              win->ee = ecore_evas_software_x11_new(NULL, 0, 0, 0, 1, 1);
1387 #ifdef HAVE_ELEMENTARY_X
1388              win->client_message_handler = ecore_event_handler_add
1389                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1390 #endif
1391           }
1392         else if (ENGINE_COMPARE(ELM_SOFTWARE_FB))
1393           {
1394              win->ee = ecore_evas_fb_new(NULL, 0, 1, 1);
1395              FALLBACK_TRY("Sofware FB");
1396           }
1397         else if (ENGINE_COMPARE(ELM_SOFTWARE_DIRECTFB))
1398           {
1399              win->ee = ecore_evas_directfb_new(NULL, 1, 0, 0, 1, 1);
1400              FALLBACK_TRY("Sofware DirectFB");
1401           }
1402         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_X11))
1403           {
1404              win->ee = ecore_evas_software_x11_16_new(NULL, 0, 0, 0, 1, 1);
1405              FALLBACK_TRY("Sofware-16");
1406 #ifdef HAVE_ELEMENTARY_X
1407              win->client_message_handler = ecore_event_handler_add
1408                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1409 #endif
1410      }
1411         else if (ENGINE_COMPARE(ELM_SOFTWARE_8_X11))
1412           {
1413              win->ee = ecore_evas_software_x11_8_new(NULL, 0, 0, 0, 1, 1);
1414              FALLBACK_TRY("Sofware-8");
1415 #ifdef HAVE_ELEMENTARY_X
1416              win->client_message_handler = ecore_event_handler_add
1417                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1418 #endif
1419           }
1420 /* killed
1421         else if (ENGINE_COMPARE(ELM_XRENDER_X11))
1422           {
1423              win->ee = ecore_evas_xrender_x11_new(NULL, 0, 0, 0, 1, 1);
1424              FALLBACK_TRY("XRender");
1425 #ifdef HAVE_ELEMENTARY_X
1426              win->client_message_handler = ecore_event_handler_add
1427                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1428 #endif
1429           }
1430  */
1431         else if (ENGINE_COMPARE(ELM_OPENGL_X11))
1432           {
1433              int opt[10];
1434              int opt_i = 0;
1435
1436              if (_elm_config->vsync)
1437                {
1438                   opt[opt_i] = ECORE_EVAS_GL_X11_OPT_VSYNC;
1439                   opt_i++;
1440                   opt[opt_i] = 1;
1441                   opt_i++;
1442                }
1443              if (opt_i > 0)
1444                 win->ee = ecore_evas_gl_x11_options_new(NULL, 0, 0, 0, 1, 1, opt);
1445              else
1446                 win->ee = ecore_evas_gl_x11_new(NULL, 0, 0, 0, 1, 1);
1447              FALLBACK_TRY("OpenGL");
1448 #ifdef HAVE_ELEMENTARY_X
1449              win->client_message_handler = ecore_event_handler_add
1450                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1451 #endif
1452           }
1453         else if (ENGINE_COMPARE(ELM_SOFTWARE_WIN32))
1454           {
1455              win->ee = ecore_evas_software_gdi_new(NULL, 0, 0, 1, 1);
1456              FALLBACK_TRY("Sofware Win32");
1457           }
1458         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1459           {
1460              win->ee = ecore_evas_software_wince_gdi_new(NULL, 0, 0, 1, 1);
1461              FALLBACK_TRY("Sofware-16-WinCE");
1462           }
1463         else if (ENGINE_COMPARE(ELM_SOFTWARE_SDL))
1464           {
1465              win->ee = ecore_evas_sdl_new(NULL, 0, 0, 0, 0, 0, 1);
1466              FALLBACK_TRY("Sofware SDL");
1467           }
1468         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_SDL))
1469           {
1470              win->ee = ecore_evas_sdl16_new(NULL, 0, 0, 0, 0, 0, 1);
1471              FALLBACK_TRY("Sofware-16-SDL");
1472           }
1473         else if (ENGINE_COMPARE(ELM_OPENGL_SDL))
1474           {
1475              win->ee = ecore_evas_gl_sdl_new(NULL, 1, 1, 0, 0);
1476              FALLBACK_TRY("OpenGL SDL");
1477           }
1478         else if (ENGINE_COMPARE(ELM_BUFFER))
1479           {
1480              win->ee = ecore_evas_buffer_new(1, 1);
1481           }
1482         else if (ENGINE_COMPARE(ELM_EWS))
1483           {
1484              win->ee = ecore_evas_ews_new(0, 0, 1, 1);
1485           }
1486         else if (!strncmp(_elm_config->engine, "shot:", 5))
1487           {
1488              win->ee = ecore_evas_buffer_new(1, 1);
1489              ecore_evas_manual_render_set(win->ee, EINA_TRUE);
1490              win->shot.info = eina_stringshare_add(_elm_config->engine + 5);
1491              _shot_init(win);
1492           }
1493 #undef FALLBACK_TRY
1494         break;
1495      }
1496
1497    if (!win->ee)
1498      {
1499         ERR("Cannot create window.");
1500         free(win);
1501         return NULL;
1502      }
1503 #ifdef HAVE_ELEMENTARY_X
1504    _elm_win_xwindow_get(win);
1505 #endif
1506    if ((_elm_config->bgpixmap) && (!_elm_config->compositing))
1507      ecore_evas_avoid_damage_set(win->ee, ECORE_EVAS_AVOID_DAMAGE_EXPOSE);
1508    // bg pixmap done by x - has other issues like can be redrawn by x before it
1509    // is filled/ready by app
1510    //     ecore_evas_avoid_damage_set(win->ee, ECORE_EVAS_AVOID_DAMAGE_BUILT_IN);
1511
1512    win->type = type;
1513    win->parent = parent;
1514    if (win->parent)
1515      evas_object_event_callback_add(win->parent, EVAS_CALLBACK_DEL,
1516                                     _elm_win_obj_callback_parent_del, win);
1517
1518    win->evas = ecore_evas_get(win->ee);
1519    win->win_obj = elm_widget_add(win->evas);
1520    elm_widget_type_set(win->win_obj, "win");
1521    ELM_SET_WIDTYPE(widtype, "win");
1522    elm_widget_data_set(win->win_obj, win);
1523    elm_widget_event_hook_set(win->win_obj, _elm_win_event_cb);
1524    elm_widget_on_focus_hook_set(win->win_obj, _elm_win_on_focus_hook, NULL);
1525    elm_widget_can_focus_set(win->win_obj, EINA_TRUE);
1526    elm_widget_highlight_ignore_set(win->win_obj, EINA_TRUE);
1527    elm_widget_focus_next_hook_set(win->win_obj, _elm_win_focus_next_hook);
1528    evas_object_color_set(win->win_obj, 0, 0, 0, 0);
1529    evas_object_move(win->win_obj, 0, 0);
1530    evas_object_resize(win->win_obj, 1, 1);
1531    evas_object_layer_set(win->win_obj, 50);
1532    evas_object_pass_events_set(win->win_obj, EINA_TRUE);
1533
1534    if (type == ELM_WIN_INLINED_IMAGE)
1535      elm_widget_parent2_set(win->win_obj, parent);
1536    ecore_evas_object_associate(win->ee, win->win_obj,
1537                                ECORE_EVAS_OBJECT_ASSOCIATE_BASE |
1538                                ECORE_EVAS_OBJECT_ASSOCIATE_STACK |
1539                                ECORE_EVAS_OBJECT_ASSOCIATE_LAYER);
1540    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_SHOW,
1541                                   _elm_win_obj_callback_show, win);
1542    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_HIDE,
1543                                   _elm_win_obj_callback_hide, win);
1544    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_DEL,
1545                                   _elm_win_obj_callback_del, win);
1546    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_MOVE,
1547                                   _elm_win_obj_callback_move, win);
1548    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_RESIZE,
1549                                   _elm_win_obj_callback_resize, win);
1550    if (win->img_obj)
1551      evas_object_intercept_move_callback_add(win->win_obj,
1552                                              _elm_win_obj_intercept_move, win);
1553    evas_object_intercept_show_callback_add(win->win_obj,
1554                                            _elm_win_obj_intercept_show, win);
1555
1556    ecore_evas_name_class_set(win->ee, name, _elm_appname);
1557    ecore_evas_callback_delete_request_set(win->ee, _elm_win_delete_request);
1558    ecore_evas_callback_resize_set(win->ee, _elm_win_resize);
1559    ecore_evas_callback_focus_in_set(win->ee, _elm_win_focus_in);
1560    ecore_evas_callback_focus_out_set(win->ee, _elm_win_focus_out);
1561    ecore_evas_callback_move_set(win->ee, _elm_win_move);
1562    evas_image_cache_set(win->evas, (_elm_config->image_cache * 1024));
1563    evas_font_cache_set(win->evas, (_elm_config->font_cache * 1024));
1564    EINA_LIST_FOREACH(_elm_config->font_dirs, l, fontpath)
1565      evas_font_path_append(win->evas, fontpath);
1566    if (!_elm_config->font_hinting)
1567      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_NONE);
1568    else if (_elm_config->font_hinting == 1)
1569      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_AUTO);
1570    else if (_elm_config->font_hinting == 2)
1571      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_BYTECODE);
1572
1573 #ifdef HAVE_ELEMENTARY_X
1574    _elm_win_xwin_update(win);
1575 #endif
1576
1577    _elm_win_list = eina_list_append(_elm_win_list, win->win_obj);
1578
1579    if (ENGINE_COMPARE(ELM_SOFTWARE_FB))
1580      {
1581         ecore_evas_fullscreen_set(win->ee, 1);
1582      }
1583 #undef ENGINE_COMPARE
1584
1585    if (_elm_config->focus_highlight_enable)
1586      elm_win_focus_highlight_enabled_set(win->win_obj, EINA_TRUE);
1587
1588 #ifdef ELM_DEBUG
1589    Evas_Modifier_Mask mask = evas_key_modifier_mask_get(win->evas, "Control");
1590    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_KEY_DOWN,
1591                                   _debug_key_down, win);
1592
1593    Eina_Bool ret = evas_object_key_grab(win->win_obj, "F12", mask, 0,
1594                                         EINA_TRUE);
1595    printf("Ctrl+F12 key combination exclusive for dot tree generation\n");
1596 #endif
1597
1598    evas_object_smart_callbacks_descriptions_set(win->win_obj, _signals);
1599
1600    return win->win_obj;
1601 }
1602
1603 EAPI void
1604 elm_win_resize_object_add(Evas_Object *obj, Evas_Object *subobj)
1605 {
1606    Evas_Coord w, h;
1607    Elm_Win *win;
1608    ELM_CHECK_WIDTYPE(obj, widtype);
1609    win = elm_widget_data_get(obj);
1610    if (!win) return;
1611    if (eina_list_data_find(win->subobjs, subobj)) return;
1612    win->subobjs = eina_list_append(win->subobjs, subobj);
1613    elm_widget_sub_object_add(obj, subobj);
1614    evas_object_event_callback_add(subobj, EVAS_CALLBACK_DEL,
1615                                   _elm_win_subobj_callback_del, obj);
1616    evas_object_event_callback_add(subobj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1617                                   _elm_win_subobj_callback_changed_size_hints,
1618                                   obj);
1619    ecore_evas_geometry_get(win->ee, NULL, NULL, &w, &h);
1620    evas_object_move(subobj, 0, 0);
1621    evas_object_resize(subobj, w, h);
1622    _elm_win_eval_subobjs(obj);
1623 }
1624
1625 EAPI void
1626 elm_win_resize_object_del(Evas_Object *obj, Evas_Object *subobj)
1627 {
1628    Elm_Win *win;
1629    ELM_CHECK_WIDTYPE(obj, widtype);
1630    win = elm_widget_data_get(obj);
1631    if (!win) return;
1632    evas_object_event_callback_del_full(subobj,
1633                                        EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1634                                        _elm_win_subobj_callback_changed_size_hints,
1635                                        obj);
1636    evas_object_event_callback_del_full(subobj, EVAS_CALLBACK_DEL,
1637                                        _elm_win_subobj_callback_del, obj);
1638    win->subobjs = eina_list_remove(win->subobjs, subobj);
1639    elm_widget_sub_object_del(obj, subobj);
1640    _elm_win_eval_subobjs(obj);
1641 }
1642
1643 EAPI void
1644 elm_win_title_set(Evas_Object *obj, const char *title)
1645 {
1646    Elm_Win *win;
1647    ELM_CHECK_WIDTYPE(obj, widtype);
1648    win = elm_widget_data_get(obj);
1649    if (!win) return;
1650    ecore_evas_title_set(win->ee, title);
1651 }
1652
1653 EAPI const char *
1654 elm_win_title_get(const Evas_Object *obj)
1655 {
1656    Elm_Win *win;
1657    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
1658    win = elm_widget_data_get(obj);
1659    if (!win) return NULL;
1660    return ecore_evas_title_get(win->ee);
1661 }
1662
1663 EAPI void
1664 elm_win_autodel_set(Evas_Object *obj, Eina_Bool autodel)
1665 {
1666    Elm_Win *win;
1667    ELM_CHECK_WIDTYPE(obj, widtype);
1668    win = elm_widget_data_get(obj);
1669    if (!win) return;
1670    win->autodel = autodel;
1671 }
1672
1673 EAPI Eina_Bool
1674 elm_win_autodel_get(const Evas_Object *obj)
1675 {
1676    Elm_Win *win;
1677    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1678    win = elm_widget_data_get(obj);
1679    if (!win) return EINA_FALSE;
1680    return win->autodel;
1681 }
1682
1683 EAPI void
1684 elm_win_activate(Evas_Object *obj)
1685 {
1686    Elm_Win *win;
1687    ELM_CHECK_WIDTYPE(obj, widtype);
1688    win = elm_widget_data_get(obj);
1689    if (!win) return;
1690    ecore_evas_activate(win->ee);
1691 }
1692
1693 EAPI void
1694 elm_win_lower(Evas_Object *obj)
1695 {
1696    Elm_Win *win;
1697    ELM_CHECK_WIDTYPE(obj, widtype);
1698    win = elm_widget_data_get(obj);
1699    if (!win) return;
1700    ecore_evas_lower(win->ee);
1701 }
1702
1703 EAPI void
1704 elm_win_raise(Evas_Object *obj)
1705 {
1706    Elm_Win *win;
1707    ELM_CHECK_WIDTYPE(obj, widtype);
1708    win = elm_widget_data_get(obj);
1709    if (!win) return;
1710    ecore_evas_raise(win->ee);
1711 }
1712
1713 EAPI void
1714 elm_win_borderless_set(Evas_Object *obj, Eina_Bool borderless)
1715 {
1716    Elm_Win *win;
1717    ELM_CHECK_WIDTYPE(obj, widtype);
1718    win = elm_widget_data_get(obj);
1719    if (!win) return;
1720    ecore_evas_borderless_set(win->ee, borderless);
1721 #ifdef HAVE_ELEMENTARY_X
1722    _elm_win_xwin_update(win);
1723 #endif
1724 }
1725
1726 EAPI Eina_Bool
1727 elm_win_borderless_get(const Evas_Object *obj)
1728 {
1729    Elm_Win *win;
1730    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1731    win = elm_widget_data_get(obj);
1732    if (!win) return EINA_FALSE;
1733    return ecore_evas_borderless_get(win->ee);
1734 }
1735
1736 EAPI void
1737 elm_win_shaped_set(Evas_Object *obj, Eina_Bool shaped)
1738 {
1739    Elm_Win *win;
1740    ELM_CHECK_WIDTYPE(obj, widtype);
1741    win = elm_widget_data_get(obj);
1742    if (!win) return;
1743    ecore_evas_shaped_set(win->ee, shaped);
1744 #ifdef HAVE_ELEMENTARY_X
1745    _elm_win_xwin_update(win);
1746 #endif
1747 }
1748
1749 EAPI Eina_Bool
1750 elm_win_shaped_get(const Evas_Object *obj)
1751 {
1752    Elm_Win *win;
1753    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1754    win = elm_widget_data_get(obj);
1755    if (!win) return EINA_FALSE;
1756    return ecore_evas_shaped_get(win->ee);
1757 }
1758
1759 EAPI void
1760 elm_win_alpha_set(Evas_Object *obj, Eina_Bool alpha)
1761 {
1762    Elm_Win *win;
1763    ELM_CHECK_WIDTYPE(obj, widtype);
1764    win = elm_widget_data_get(obj);
1765    if (!win) return;
1766    if (win->frame_obj)
1767      {
1768      }
1769    else if (win->img_obj)
1770      {
1771         evas_object_image_alpha_set(win->img_obj, alpha);
1772         ecore_evas_alpha_set(win->ee, alpha);
1773      }
1774    else
1775      {
1776 #ifdef HAVE_ELEMENTARY_X
1777         if (win->xwin)
1778           {
1779              if (alpha)
1780                {
1781                   if (!_elm_config->compositing)
1782                      elm_win_shaped_set(obj, alpha);
1783                   else
1784                      ecore_evas_alpha_set(win->ee, alpha);
1785                }
1786              else
1787                 ecore_evas_alpha_set(win->ee, alpha);
1788              _elm_win_xwin_update(win);
1789           }
1790         else
1791 #endif
1792            ecore_evas_alpha_set(win->ee, alpha);
1793      }
1794 }
1795
1796 EAPI Eina_Bool
1797 elm_win_alpha_get(const Evas_Object *obj)
1798 {
1799    Elm_Win *win;
1800    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1801    win = elm_widget_data_get(obj);
1802    if (!win) return EINA_FALSE;
1803    if (win->frame_obj)
1804      {
1805      }
1806    else if (win->img_obj)
1807      {
1808         return evas_object_image_alpha_get(win->img_obj);
1809      }
1810    return ecore_evas_alpha_get(win->ee);
1811 }
1812
1813 EAPI void
1814 elm_win_transparent_set(Evas_Object *obj, Eina_Bool transparent)
1815 {
1816    Elm_Win *win;
1817    ELM_CHECK_WIDTYPE(obj, widtype);
1818    win = elm_widget_data_get(obj);
1819    if (!win) return;
1820
1821    if (win->frame_obj)
1822      {
1823      }
1824    else if (win->img_obj)
1825      {
1826         evas_object_image_alpha_set(win->img_obj, transparent);
1827      }
1828    else
1829      {
1830 #ifdef HAVE_ELEMENTARY_X
1831         if (win->xwin)
1832           {
1833              ecore_evas_transparent_set(win->ee, transparent);
1834              _elm_win_xwin_update(win);
1835           }
1836         else
1837 #endif
1838            ecore_evas_transparent_set(win->ee, transparent);
1839      }
1840 }
1841
1842 EAPI Eina_Bool
1843 elm_win_transparent_get(const Evas_Object *obj)
1844 {
1845    Elm_Win *win;
1846    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1847    win = elm_widget_data_get(obj);
1848    if (!win) return EINA_FALSE;
1849
1850    return ecore_evas_transparent_get(win->ee);
1851 }
1852
1853 EAPI void
1854 elm_win_override_set(Evas_Object *obj, Eina_Bool override)
1855 {
1856    Elm_Win *win;
1857    ELM_CHECK_WIDTYPE(obj, widtype);
1858    win = elm_widget_data_get(obj);
1859    if (!win) return;
1860    ecore_evas_override_set(win->ee, override);
1861 #ifdef HAVE_ELEMENTARY_X
1862    _elm_win_xwin_update(win);
1863 #endif
1864 }
1865
1866 EAPI Eina_Bool
1867 elm_win_override_get(const Evas_Object *obj)
1868 {
1869    Elm_Win *win;
1870    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1871    win = elm_widget_data_get(obj);
1872    if (!win) return EINA_FALSE;
1873    return ecore_evas_override_get(win->ee);
1874 }
1875
1876 EAPI void
1877 elm_win_fullscreen_set(Evas_Object *obj, Eina_Bool fullscreen)
1878 {
1879    Elm_Win *win;
1880    ELM_CHECK_WIDTYPE(obj, widtype);
1881    win = elm_widget_data_get(obj);
1882    if (!win) return;
1883
1884    // YYY: handle if win->img_obj
1885 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1886    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
1887        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1888      {
1889         // these engines... can ONLY be fullscreen
1890         return;
1891      }
1892    else
1893      {
1894         ecore_evas_fullscreen_set(win->ee, fullscreen);
1895 #ifdef HAVE_ELEMENTARY_X
1896         _elm_win_xwin_update(win);
1897 #endif
1898      }
1899 #undef ENGINE_COMPARE
1900 }
1901
1902 EAPI Eina_Bool
1903 elm_win_fullscreen_get(const Evas_Object *obj)
1904 {
1905    Elm_Win *win;
1906    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1907    win = elm_widget_data_get(obj);
1908    if (!win) return EINA_FALSE;
1909
1910 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1911    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
1912        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1913      {
1914         // these engines... can ONLY be fullscreen
1915         return EINA_TRUE;
1916      }
1917    else
1918      {
1919         return ecore_evas_fullscreen_get(win->ee);
1920      }
1921 #undef ENGINE_COMPARE
1922 }
1923
1924 EAPI void
1925 elm_win_maximized_set(Evas_Object *obj, Eina_Bool maximized)
1926 {
1927    Elm_Win *win;
1928    ELM_CHECK_WIDTYPE(obj, widtype);
1929    win = elm_widget_data_get(obj);
1930    if (!win) return;
1931    // YYY: handle if win->img_obj
1932    ecore_evas_maximized_set(win->ee, maximized);
1933 #ifdef HAVE_ELEMENTARY_X
1934    _elm_win_xwin_update(win);
1935 #endif
1936 }
1937
1938 EAPI Eina_Bool
1939 elm_win_maximized_get(const Evas_Object *obj)
1940 {
1941    Elm_Win *win;
1942    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1943    win = elm_widget_data_get(obj);
1944    if (!win) return EINA_FALSE;
1945    return ecore_evas_maximized_get(win->ee);
1946 }
1947
1948 EAPI void
1949 elm_win_iconified_set(Evas_Object *obj, Eina_Bool iconified)
1950 {
1951    Elm_Win *win;
1952    ELM_CHECK_WIDTYPE(obj, widtype);
1953    win = elm_widget_data_get(obj);
1954    if (!win) return;
1955    ecore_evas_iconified_set(win->ee, iconified);
1956 #ifdef HAVE_ELEMENTARY_X
1957    _elm_win_xwin_update(win);
1958 #endif
1959 }
1960
1961 EAPI Eina_Bool
1962 elm_win_iconified_get(const Evas_Object *obj)
1963 {
1964    Elm_Win *win;
1965    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1966    win = elm_widget_data_get(obj);
1967    if (!win) return EINA_FALSE;
1968    return ecore_evas_iconified_get(win->ee);
1969 }
1970
1971 EAPI void
1972 elm_win_layer_set(Evas_Object *obj, int layer)
1973 {
1974    Elm_Win *win;
1975    ELM_CHECK_WIDTYPE(obj, widtype);
1976    win = elm_widget_data_get(obj);
1977    if (!win) return;
1978    ecore_evas_layer_set(win->ee, layer);
1979 #ifdef HAVE_ELEMENTARY_X
1980    _elm_win_xwin_update(win);
1981 #endif
1982 }
1983
1984 EAPI int
1985 elm_win_layer_get(const Evas_Object *obj)
1986 {
1987    Elm_Win *win;
1988    ELM_CHECK_WIDTYPE(obj, widtype) -1;
1989    win = elm_widget_data_get(obj);
1990    if (!win) return -1;
1991    return ecore_evas_layer_get(win->ee);
1992 }
1993
1994 EAPI void
1995 elm_win_rotation_set(Evas_Object *obj, int rotation)
1996 {
1997    Elm_Win *win;
1998    ELM_CHECK_WIDTYPE(obj, widtype);
1999    win = elm_widget_data_get(obj);
2000    if (!win) return;
2001    if (win->rot == rotation) return;
2002    win->rot = rotation;
2003    ecore_evas_rotation_set(win->ee, rotation);
2004    evas_object_size_hint_min_set(obj, -1, -1);
2005    evas_object_size_hint_max_set(obj, -1, -1);
2006    _elm_win_eval_subobjs(obj);
2007 #ifdef HAVE_ELEMENTARY_X
2008    _elm_win_xwin_update(win);
2009 #endif
2010 }
2011
2012 EAPI void
2013 elm_win_rotation_with_resize_set(Evas_Object *obj, int rotation)
2014 {
2015    Elm_Win *win;
2016    ELM_CHECK_WIDTYPE(obj, widtype);
2017    win = elm_widget_data_get(obj);
2018    if (!win) return;
2019    if (win->rot == rotation) return;
2020    win->rot = rotation;
2021    ecore_evas_rotation_with_resize_set(win->ee, rotation);
2022    evas_object_size_hint_min_set(obj, -1, -1);
2023    evas_object_size_hint_max_set(obj, -1, -1);
2024    _elm_win_eval_subobjs(obj);
2025 #ifdef HAVE_ELEMENTARY_X
2026    _elm_win_xwin_update(win);
2027 #endif
2028 }
2029
2030 EAPI int
2031 elm_win_rotation_get(const Evas_Object *obj)
2032 {
2033    Elm_Win *win;
2034    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2035    win = elm_widget_data_get(obj);
2036    if (!win) return -1;
2037    return win->rot;
2038 }
2039
2040 EAPI void
2041 elm_win_sticky_set(Evas_Object *obj, Eina_Bool sticky)
2042 {
2043    Elm_Win *win;
2044    ELM_CHECK_WIDTYPE(obj, widtype);
2045    win = elm_widget_data_get(obj);
2046    if (!win) return;
2047    ecore_evas_sticky_set(win->ee, sticky);
2048 #ifdef HAVE_ELEMENTARY_X
2049    _elm_win_xwin_update(win);
2050 #endif
2051 }
2052
2053 EAPI Eina_Bool
2054 elm_win_sticky_get(const Evas_Object *obj)
2055 {
2056    Elm_Win *win;
2057    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2058    win = elm_widget_data_get(obj);
2059    if (!win) return EINA_FALSE;
2060    return ecore_evas_sticky_get(win->ee);
2061 }
2062
2063 EAPI void
2064 elm_win_keyboard_mode_set(Evas_Object *obj, Elm_Win_Keyboard_Mode mode)
2065 {
2066    Elm_Win *win;
2067    ELM_CHECK_WIDTYPE(obj, widtype);
2068    win = elm_widget_data_get(obj);
2069    if (!win) return;
2070    if (mode == win->kbdmode) return;
2071 #ifdef HAVE_ELEMENTARY_X
2072    _elm_win_xwindow_get(win);
2073 #endif
2074    win->kbdmode = mode;
2075 #ifdef HAVE_ELEMENTARY_X
2076    if (win->xwin)
2077      ecore_x_e_virtual_keyboard_state_set
2078         (win->xwin, (Ecore_X_Virtual_Keyboard_State)win->kbdmode);
2079 #endif
2080 }
2081
2082 EAPI Elm_Win_Keyboard_Mode
2083 elm_win_keyboard_mode_get(const Evas_Object *obj)
2084 {
2085    Elm_Win *win;
2086    ELM_CHECK_WIDTYPE(obj, widtype) ELM_WIN_KEYBOARD_UNKNOWN;
2087    win = elm_widget_data_get(obj);
2088    if (!win) return ELM_WIN_KEYBOARD_UNKNOWN;
2089    return win->kbdmode;
2090 }
2091
2092 EAPI void
2093 elm_win_keyboard_win_set(Evas_Object *obj, Eina_Bool is_keyboard)
2094 {
2095    Elm_Win *win;
2096    ELM_CHECK_WIDTYPE(obj, widtype);
2097    win = elm_widget_data_get(obj);
2098    if (!win) return;
2099 #ifdef HAVE_ELEMENTARY_X
2100    _elm_win_xwindow_get(win);
2101    if (win->xwin)
2102      ecore_x_e_virtual_keyboard_set(win->xwin, is_keyboard);
2103 #endif
2104 }
2105
2106 EAPI Eina_Bool
2107 elm_win_keyboard_win_get(const Evas_Object *obj)
2108 {
2109    Elm_Win *win;
2110    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2111    win = elm_widget_data_get(obj);
2112    if (!win) return EINA_FALSE;
2113 #ifdef HAVE_ELEMENTARY_X
2114    _elm_win_xwindow_get(win);
2115    if (win->xwin)
2116      return ecore_x_e_virtual_keyboard_get(win->xwin);
2117 #endif
2118    return EINA_FALSE;
2119 }
2120
2121 EAPI void
2122 elm_win_screen_position_get(const Evas_Object *obj, int *x, int *y)
2123 {
2124    Elm_Win *win;
2125    ELM_CHECK_WIDTYPE(obj, widtype);
2126    win = elm_widget_data_get(obj);
2127    if (!win) return;
2128    if (x) *x = win->screen.x;
2129    if (y) *y = win->screen.y;
2130 }
2131
2132 EAPI void
2133 elm_win_conformant_set(Evas_Object *obj, Eina_Bool conformant)
2134 {
2135    Elm_Win *win;
2136    ELM_CHECK_WIDTYPE(obj, widtype);
2137    win = elm_widget_data_get(obj);
2138    if (!win) return;
2139 #ifdef HAVE_ELEMENTARY_X
2140    _elm_win_xwindow_get(win);
2141    if (win->xwin)
2142      ecore_x_e_illume_conformant_set(win->xwin, conformant);
2143 #endif
2144 }
2145
2146 EAPI Eina_Bool
2147 elm_win_conformant_get(const Evas_Object *obj)
2148 {
2149    Elm_Win *win;
2150    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2151    win = elm_widget_data_get(obj);
2152    if (!win) return EINA_FALSE;
2153 #ifdef HAVE_ELEMENTARY_X
2154    _elm_win_xwindow_get(win);
2155    if (win->xwin)
2156      return ecore_x_e_illume_conformant_get(win->xwin);
2157 #endif
2158    return EINA_FALSE;
2159 }
2160
2161 EAPI void
2162 elm_win_quickpanel_set(Evas_Object *obj, Eina_Bool quickpanel)
2163 {
2164    Elm_Win *win;
2165    ELM_CHECK_WIDTYPE(obj, widtype);
2166    win = elm_widget_data_get(obj);
2167    if (!win) return;
2168 #ifdef HAVE_ELEMENTARY_X
2169    _elm_win_xwindow_get(win);
2170    if (win->xwin)
2171      {
2172         ecore_x_e_illume_quickpanel_set(win->xwin, quickpanel);
2173         if (quickpanel)
2174           {
2175              Ecore_X_Window_State states[2];
2176
2177              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2178              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2179              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2180              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2181           }
2182      }
2183 #endif
2184 }
2185
2186 EAPI Eina_Bool
2187 elm_win_quickpanel_get(const Evas_Object *obj)
2188 {
2189    Elm_Win *win;
2190    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2191    win = elm_widget_data_get(obj);
2192    if (!win) return EINA_FALSE;
2193 #ifdef HAVE_ELEMENTARY_X
2194    _elm_win_xwindow_get(win);
2195    if (win->xwin)
2196      return ecore_x_e_illume_quickpanel_get(win->xwin);
2197 #endif
2198    return EINA_FALSE;
2199 }
2200
2201 EAPI void
2202 elm_win_quickpanel_priority_major_set(Evas_Object *obj, int priority)
2203 {
2204    Elm_Win *win;
2205    ELM_CHECK_WIDTYPE(obj, widtype);
2206    win = elm_widget_data_get(obj);
2207    if (!win) return;
2208 #ifdef HAVE_ELEMENTARY_X
2209    _elm_win_xwindow_get(win);
2210    if (win->xwin)
2211      ecore_x_e_illume_quickpanel_priority_major_set(win->xwin, priority);
2212 #endif
2213 }
2214
2215 EAPI int
2216 elm_win_quickpanel_priority_major_get(const Evas_Object *obj)
2217 {
2218    Elm_Win *win;
2219    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2220    win = elm_widget_data_get(obj);
2221    if (!win) return -1;
2222 #ifdef HAVE_ELEMENTARY_X
2223    _elm_win_xwindow_get(win);
2224    if (win->xwin)
2225      return ecore_x_e_illume_quickpanel_priority_major_get(win->xwin);
2226 #endif
2227    return -1;
2228 }
2229
2230 EAPI void
2231 elm_win_quickpanel_priority_minor_set(Evas_Object *obj, int priority)
2232 {
2233    Elm_Win *win;
2234    ELM_CHECK_WIDTYPE(obj, widtype);
2235    win = elm_widget_data_get(obj);
2236    if (!win) return;
2237 #ifdef HAVE_ELEMENTARY_X
2238    _elm_win_xwindow_get(win);
2239    if (win->xwin)
2240      ecore_x_e_illume_quickpanel_priority_minor_set(win->xwin, priority);
2241 #endif
2242 }
2243
2244 EAPI int
2245 elm_win_quickpanel_priority_minor_get(const Evas_Object *obj)
2246 {
2247    Elm_Win *win;
2248    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2249    win = elm_widget_data_get(obj);
2250    if (!win) return -1;
2251 #ifdef HAVE_ELEMENTARY_X
2252    _elm_win_xwindow_get(win);
2253    if (win->xwin)
2254      return ecore_x_e_illume_quickpanel_priority_minor_get(win->xwin);
2255 #endif
2256    return -1;
2257 }
2258
2259 EAPI void
2260 elm_win_quickpanel_zone_set(Evas_Object *obj, int zone)
2261 {
2262    Elm_Win *win;
2263    ELM_CHECK_WIDTYPE(obj, widtype);
2264    win = elm_widget_data_get(obj);
2265    if (!win) return;
2266 #ifdef HAVE_ELEMENTARY_X
2267    _elm_win_xwindow_get(win);
2268    if (win->xwin)
2269      ecore_x_e_illume_quickpanel_zone_set(win->xwin, zone);
2270 #endif
2271 }
2272
2273 EAPI int
2274 elm_win_quickpanel_zone_get(const Evas_Object *obj)
2275 {
2276    Elm_Win *win;
2277    ELM_CHECK_WIDTYPE(obj, widtype) 0;
2278    win = elm_widget_data_get(obj);
2279    if (!win) return 0;
2280 #ifdef HAVE_ELEMENTARY_X
2281    _elm_win_xwindow_get(win);
2282    if (win->xwin)
2283      return ecore_x_e_illume_quickpanel_zone_get(win->xwin);
2284 #endif
2285    return 0;
2286 }
2287
2288 EAPI void
2289 elm_win_prop_focus_skip_set(Evas_Object *obj, Eina_Bool skip)
2290 {
2291    Elm_Win *win;
2292    ELM_CHECK_WIDTYPE(obj, widtype);
2293    win = elm_widget_data_get(obj);
2294    if (!win) return;
2295 #ifdef HAVE_ELEMENTARY_X
2296    _elm_win_xwindow_get(win);
2297    if (skip)
2298      {
2299         if (win->xwin)
2300           {
2301              Ecore_X_Window_State states[2];
2302
2303              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2304              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2305              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2306              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2307           }
2308      }
2309 #endif
2310 }
2311
2312 EAPI void
2313 elm_win_illume_command_send(Evas_Object *obj, Elm_Illume_Command command, void *params __UNUSED__)
2314 {
2315    Elm_Win *win;
2316    ELM_CHECK_WIDTYPE(obj, widtype);
2317    win = elm_widget_data_get(obj);
2318    if (!win) return;
2319 #ifdef HAVE_ELEMENTARY_X
2320    _elm_win_xwindow_get(win);
2321    if (win->xwin)
2322      {
2323         switch (command)
2324           {
2325            case ELM_ILLUME_COMMAND_FOCUS_BACK:
2326               ecore_x_e_illume_focus_back_send(win->xwin);
2327               break;
2328            case ELM_ILLUME_COMMAND_FOCUS_FORWARD:
2329               ecore_x_e_illume_focus_forward_send(win->xwin);
2330               break;
2331            case ELM_ILLUME_COMMAND_FOCUS_HOME:
2332               ecore_x_e_illume_focus_home_send(win->xwin);
2333               break;
2334            case ELM_ILLUME_COMMAND_CLOSE:
2335               ecore_x_e_illume_close_send(win->xwin);
2336               break;
2337            default:
2338               break;
2339           }
2340      }
2341 #endif
2342 }
2343
2344 EAPI Evas_Object *
2345 elm_win_inlined_image_object_get(Evas_Object *obj)
2346 {
2347    Elm_Win *win;
2348    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2349    win = elm_widget_data_get(obj);
2350    if (!win) return NULL;
2351    return win->img_obj;
2352 }
2353
2354 EAPI void
2355 elm_win_focus_highlight_enabled_set(Evas_Object *obj, Eina_Bool enabled)
2356 {
2357    Elm_Win *win;
2358
2359    ELM_CHECK_WIDTYPE(obj, widtype);
2360
2361    win = elm_widget_data_get(obj);
2362    enabled = !!enabled;
2363    if (win->focus_highlight.enabled == enabled)
2364      return;
2365
2366    win->focus_highlight.enabled = enabled;
2367
2368    if (win->focus_highlight.enabled)
2369      _elm_win_focus_highlight_init(win);
2370    else
2371      _elm_win_focus_highlight_shutdown(win);
2372 }
2373
2374 EAPI Eina_Bool
2375 elm_win_focus_highlight_enabled_get(const Evas_Object *obj)
2376 {
2377    Elm_Win *win;
2378
2379    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2380
2381    win = elm_widget_data_get(obj);
2382    return win->focus_highlight.enabled;
2383 }
2384
2385 EAPI void
2386 elm_win_focus_highlight_style_set(Evas_Object *obj, const char *style)
2387 {
2388    Elm_Win *win;
2389
2390    ELM_CHECK_WIDTYPE(obj, widtype);
2391
2392    win = elm_widget_data_get(obj);
2393    eina_stringshare_replace(&win->focus_highlight.style, style);
2394    win->focus_highlight.changed_theme = EINA_TRUE;
2395    _elm_win_focus_highlight_reconfigure_job_start(win);
2396 }
2397
2398 EAPI const char *
2399 elm_win_focus_highlight_style_get(const Evas_Object *obj)
2400 {
2401    Elm_Win *win;
2402
2403    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2404
2405    win = elm_widget_data_get(obj);
2406    return win->focus_highlight.style;
2407 }
2408
2409 typedef struct _Widget_Data Widget_Data;
2410
2411 struct _Widget_Data
2412 {
2413    Evas_Object *frm;
2414    Evas_Object *content;
2415 };
2416
2417 static void _del_hook(Evas_Object *obj);
2418 static void _theme_hook(Evas_Object *obj);
2419 static void _sizing_eval(Evas_Object *obj);
2420 static void _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
2421 static void _sub_del(void *data, Evas_Object *obj, void *event_info);
2422
2423 static const char *widtype2 = NULL;
2424
2425 static void
2426 _del_hook(Evas_Object *obj)
2427 {
2428    Widget_Data *wd = elm_widget_data_get(obj);
2429    if (!wd) return;
2430    free(wd);
2431 }
2432
2433 static void
2434 _theme_hook(Evas_Object *obj)
2435 {
2436    Widget_Data *wd = elm_widget_data_get(obj);
2437    _elm_theme_object_set(obj, wd->frm, "win", "inwin", elm_widget_style_get(obj));
2438    if (wd->content)
2439      edje_object_part_swallow(wd->frm, "elm.swallow.content", wd->content);
2440    _sizing_eval(obj);
2441
2442    evas_object_smart_callback_call(obj, SIG_THEME_CHANGED, NULL);
2443 }
2444
2445 static Eina_Bool
2446 _elm_inwin_focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next)
2447 {
2448    Widget_Data *wd = elm_widget_data_get(obj);
2449
2450    if (!wd)
2451      return EINA_FALSE;
2452
2453    /* Try Focus cycle in subitem */
2454    if (wd->content)
2455      {
2456         elm_widget_focus_next_get(wd->content, dir, next);
2457         if (*next)
2458           return EINA_TRUE;
2459      }
2460
2461    *next = (Evas_Object *)obj;
2462    return EINA_FALSE;
2463 }
2464
2465 static void
2466 _sizing_eval(Evas_Object *obj)
2467 {
2468    Widget_Data *wd = elm_widget_data_get(obj);
2469    Evas_Coord minw = -1, minh = -1;
2470
2471    evas_object_size_hint_min_get(wd->content, &minw, &minh);
2472    edje_object_size_min_calc(wd->frm, &minw, &minh);
2473    evas_object_size_hint_min_set(obj, minw, minh);
2474    evas_object_size_hint_max_set(obj, -1, -1);
2475 }
2476
2477 static void
2478 _changed_size_hints(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
2479 {
2480    _sizing_eval(data);
2481 }
2482
2483 static void
2484 _sub_del(void *data __UNUSED__, Evas_Object *obj, void *event_info)
2485 {
2486    Widget_Data *wd = elm_widget_data_get(obj);
2487    Evas_Object *sub = event_info;
2488    if (sub == wd->content)
2489      {
2490         evas_object_event_callback_del_full
2491            (sub, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _changed_size_hints, obj);
2492         wd->content = NULL;
2493         _sizing_eval(obj);
2494      }
2495 }
2496
2497 EAPI Evas_Object *
2498 elm_win_inwin_add(Evas_Object *obj)
2499 {
2500    Evas_Object *obj2;
2501    Widget_Data *wd;
2502    Elm_Win *win;
2503
2504    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2505    win = elm_widget_data_get(obj);
2506    if (!win) return NULL;
2507    wd = ELM_NEW(Widget_Data);
2508    obj2 = elm_widget_add(win->evas);
2509    elm_widget_type_set(obj2, "inwin");
2510    ELM_SET_WIDTYPE(widtype2, "inwin");
2511    elm_widget_sub_object_add(obj, obj2);
2512    evas_object_size_hint_weight_set(obj2, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
2513    evas_object_size_hint_align_set(obj2, EVAS_HINT_FILL, EVAS_HINT_FILL);
2514    elm_win_resize_object_add(obj, obj2);
2515
2516    elm_widget_data_set(obj2, wd);
2517    elm_widget_del_hook_set(obj2, _del_hook);
2518    elm_widget_theme_hook_set(obj2, _theme_hook);
2519    elm_widget_focus_next_hook_set(obj2, _elm_inwin_focus_next_hook);
2520    elm_widget_can_focus_set(obj2, EINA_TRUE);
2521    elm_widget_highlight_ignore_set(obj2, EINA_TRUE);
2522
2523    wd->frm = edje_object_add(win->evas);
2524    _elm_theme_object_set(obj, wd->frm, "win", "inwin", "default");
2525    elm_widget_resize_object_set(obj2, wd->frm);
2526
2527    evas_object_smart_callback_add(obj2, "sub-object-del", _sub_del, obj2);
2528
2529    _sizing_eval(obj2);
2530    return obj2;
2531 }
2532
2533 EAPI void
2534 elm_win_inwin_activate(Evas_Object *obj)
2535 {
2536    ELM_CHECK_WIDTYPE(obj, widtype2);
2537    Widget_Data *wd = elm_widget_data_get(obj);
2538    if (!wd) return;
2539    evas_object_raise(obj);
2540    evas_object_show(obj);
2541    edje_object_signal_emit(wd->frm, "elm,action,show", "elm");
2542    elm_object_focus_set(obj, EINA_TRUE);
2543 }
2544
2545 EAPI void
2546 elm_win_inwin_content_set(Evas_Object *obj, Evas_Object *content)
2547 {
2548    ELM_CHECK_WIDTYPE(obj, widtype2);
2549    Widget_Data *wd = elm_widget_data_get(obj);
2550    if (!wd) return;
2551    if (wd->content == content) return;
2552    if (wd->content) evas_object_del(wd->content);
2553    wd->content = content;
2554    if (content)
2555      {
2556         elm_widget_sub_object_add(obj, content);
2557         evas_object_event_callback_add(content, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
2558                                        _changed_size_hints, obj);
2559         edje_object_part_swallow(wd->frm, "elm.swallow.content", content);
2560      }
2561    _sizing_eval(obj);
2562 }
2563
2564 EAPI Evas_Object *
2565 elm_win_inwin_content_get(const Evas_Object *obj)
2566 {
2567    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2568    Widget_Data *wd = elm_widget_data_get(obj);
2569    if (!wd) return NULL;
2570    return wd->content;
2571 }
2572
2573 EAPI Evas_Object *
2574 elm_win_inwin_content_unset(Evas_Object *obj)
2575 {
2576    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2577    Widget_Data *wd = elm_widget_data_get(obj);
2578    if (!wd) return NULL;
2579    if (!wd->content) return NULL;
2580    Evas_Object *content = wd->content;
2581    elm_widget_sub_object_del(obj, wd->content);
2582    edje_object_part_unswallow(wd->frm, wd->content);
2583    wd->content = NULL;
2584    return content;
2585 }
2586
2587 /* windowing spcific calls - shall we do this differently? */
2588
2589 static Ecore_X_Window
2590 _elm_ee_win_get(const Evas_Object *obj)
2591 {
2592    if (!obj) return 0;
2593 #ifdef HAVE_ELEMENTARY_X
2594    Ecore_Evas *ee = ecore_evas_ecore_evas_get(evas_object_evas_get(obj));
2595    if (ee) return (Ecore_X_Window)ecore_evas_window_get(ee);
2596 #endif
2597    return 0;
2598 }
2599
2600 EAPI Ecore_X_Window
2601 elm_win_xwindow_get(const Evas_Object *obj)
2602 {
2603    Elm_Win *win;
2604    const char *type;
2605
2606    if (!obj) return 0;
2607    type = elm_widget_type_get(obj);
2608    if (!type) return 0;
2609    if (type != widtype) return _elm_ee_win_get(obj);
2610 #ifdef HAVE_ELEMENTARY_X
2611    win = elm_widget_data_get(obj);
2612    if (!win) return 0;
2613    if (win->xwin) return win->xwin;
2614    if (win->parent) return elm_win_xwindow_get(win->parent);
2615 #endif
2616    return 0;
2617    win = NULL;
2618 }