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