MIN/MAX macros -> elm_priv.h
[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    evas_object_geometry_get(obj, NULL, NULL, &win_w, &win_h);
1782    if (h) nx = win_w >= screen_w ? 0 : (screen_w / 2) - (win_w / 2);
1783    else nx = win->screen.x;
1784    if (v) ny = win_h >= screen_h ? 0 : (screen_h / 2) - (win_h / 2);
1785    else ny = win->screen.y;
1786    evas_object_move(obj, nx, ny);
1787 }
1788
1789 EAPI void
1790 elm_win_borderless_set(Evas_Object *obj, Eina_Bool borderless)
1791 {
1792    Elm_Win *win;
1793    ELM_CHECK_WIDTYPE(obj, widtype);
1794    win = elm_widget_data_get(obj);
1795    if (!win) return;
1796    ecore_evas_borderless_set(win->ee, borderless);
1797 #ifdef HAVE_ELEMENTARY_X
1798    _elm_win_xwin_update(win);
1799 #endif
1800 }
1801
1802 EAPI Eina_Bool
1803 elm_win_borderless_get(const Evas_Object *obj)
1804 {
1805    Elm_Win *win;
1806    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1807    win = elm_widget_data_get(obj);
1808    if (!win) return EINA_FALSE;
1809    return ecore_evas_borderless_get(win->ee);
1810 }
1811
1812 EAPI void
1813 elm_win_shaped_set(Evas_Object *obj, Eina_Bool shaped)
1814 {
1815    Elm_Win *win;
1816    ELM_CHECK_WIDTYPE(obj, widtype);
1817    win = elm_widget_data_get(obj);
1818    if (!win) return;
1819    ecore_evas_shaped_set(win->ee, shaped);
1820 #ifdef HAVE_ELEMENTARY_X
1821    _elm_win_xwin_update(win);
1822 #endif
1823 }
1824
1825 EAPI Eina_Bool
1826 elm_win_shaped_get(const Evas_Object *obj)
1827 {
1828    Elm_Win *win;
1829    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1830    win = elm_widget_data_get(obj);
1831    if (!win) return EINA_FALSE;
1832    return ecore_evas_shaped_get(win->ee);
1833 }
1834
1835 EAPI void
1836 elm_win_alpha_set(Evas_Object *obj, Eina_Bool alpha)
1837 {
1838    Elm_Win *win;
1839    ELM_CHECK_WIDTYPE(obj, widtype);
1840    win = elm_widget_data_get(obj);
1841    if (!win) return;
1842    if (win->frame_obj)
1843      {
1844      }
1845    else if (win->img_obj)
1846      {
1847         evas_object_image_alpha_set(win->img_obj, alpha);
1848         ecore_evas_alpha_set(win->ee, alpha);
1849      }
1850    else
1851      {
1852 #ifdef HAVE_ELEMENTARY_X
1853         if (win->xwin)
1854           {
1855              if (alpha)
1856                {
1857                   if (!_elm_config->compositing)
1858                      elm_win_shaped_set(obj, alpha);
1859                   else
1860                      ecore_evas_alpha_set(win->ee, alpha);
1861                }
1862              else
1863                 ecore_evas_alpha_set(win->ee, alpha);
1864              _elm_win_xwin_update(win);
1865           }
1866         else
1867 #endif
1868            ecore_evas_alpha_set(win->ee, alpha);
1869      }
1870 }
1871
1872 EAPI Eina_Bool
1873 elm_win_alpha_get(const Evas_Object *obj)
1874 {
1875    Elm_Win *win;
1876    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1877    win = elm_widget_data_get(obj);
1878    if (!win) return EINA_FALSE;
1879    if (win->frame_obj)
1880      {
1881      }
1882    else if (win->img_obj)
1883      {
1884         return evas_object_image_alpha_get(win->img_obj);
1885      }
1886    return ecore_evas_alpha_get(win->ee);
1887 }
1888
1889 EAPI void
1890 elm_win_transparent_set(Evas_Object *obj, Eina_Bool transparent)
1891 {
1892    Elm_Win *win;
1893    ELM_CHECK_WIDTYPE(obj, widtype);
1894    win = elm_widget_data_get(obj);
1895    if (!win) return;
1896
1897    if (win->frame_obj)
1898      {
1899      }
1900    else if (win->img_obj)
1901      {
1902         evas_object_image_alpha_set(win->img_obj, transparent);
1903      }
1904    else
1905      {
1906 #ifdef HAVE_ELEMENTARY_X
1907         if (win->xwin)
1908           {
1909              ecore_evas_transparent_set(win->ee, transparent);
1910              _elm_win_xwin_update(win);
1911           }
1912         else
1913 #endif
1914            ecore_evas_transparent_set(win->ee, transparent);
1915      }
1916 }
1917
1918 EAPI Eina_Bool
1919 elm_win_transparent_get(const Evas_Object *obj)
1920 {
1921    Elm_Win *win;
1922    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1923    win = elm_widget_data_get(obj);
1924    if (!win) return EINA_FALSE;
1925
1926    return ecore_evas_transparent_get(win->ee);
1927 }
1928
1929 EAPI void
1930 elm_win_override_set(Evas_Object *obj, Eina_Bool override)
1931 {
1932    Elm_Win *win;
1933    ELM_CHECK_WIDTYPE(obj, widtype);
1934    win = elm_widget_data_get(obj);
1935    if (!win) return;
1936    ecore_evas_override_set(win->ee, override);
1937 #ifdef HAVE_ELEMENTARY_X
1938    _elm_win_xwin_update(win);
1939 #endif
1940 }
1941
1942 EAPI Eina_Bool
1943 elm_win_override_get(const Evas_Object *obj)
1944 {
1945    Elm_Win *win;
1946    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1947    win = elm_widget_data_get(obj);
1948    if (!win) return EINA_FALSE;
1949    return ecore_evas_override_get(win->ee);
1950 }
1951
1952 EAPI void
1953 elm_win_fullscreen_set(Evas_Object *obj, Eina_Bool fullscreen)
1954 {
1955    Elm_Win *win;
1956    ELM_CHECK_WIDTYPE(obj, widtype);
1957    win = elm_widget_data_get(obj);
1958    if (!win) return;
1959
1960    // YYY: handle if win->img_obj
1961 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1962    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
1963        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1964      {
1965         // these engines... can ONLY be fullscreen
1966         return;
1967      }
1968    else
1969      {
1970         ecore_evas_fullscreen_set(win->ee, fullscreen);
1971 #ifdef HAVE_ELEMENTARY_X
1972         _elm_win_xwin_update(win);
1973 #endif
1974      }
1975 #undef ENGINE_COMPARE
1976 }
1977
1978 EAPI Eina_Bool
1979 elm_win_fullscreen_get(const Evas_Object *obj)
1980 {
1981    Elm_Win *win;
1982    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1983    win = elm_widget_data_get(obj);
1984    if (!win) return EINA_FALSE;
1985
1986 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1987    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
1988        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1989      {
1990         // these engines... can ONLY be fullscreen
1991         return EINA_TRUE;
1992      }
1993    else
1994      {
1995         return ecore_evas_fullscreen_get(win->ee);
1996      }
1997 #undef ENGINE_COMPARE
1998 }
1999
2000 EAPI void
2001 elm_win_maximized_set(Evas_Object *obj, Eina_Bool maximized)
2002 {
2003    Elm_Win *win;
2004    ELM_CHECK_WIDTYPE(obj, widtype);
2005    win = elm_widget_data_get(obj);
2006    if (!win) return;
2007    // YYY: handle if win->img_obj
2008    ecore_evas_maximized_set(win->ee, maximized);
2009 #ifdef HAVE_ELEMENTARY_X
2010    _elm_win_xwin_update(win);
2011 #endif
2012 }
2013
2014 EAPI Eina_Bool
2015 elm_win_maximized_get(const Evas_Object *obj)
2016 {
2017    Elm_Win *win;
2018    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2019    win = elm_widget_data_get(obj);
2020    if (!win) return EINA_FALSE;
2021    return ecore_evas_maximized_get(win->ee);
2022 }
2023
2024 EAPI void
2025 elm_win_iconified_set(Evas_Object *obj, Eina_Bool iconified)
2026 {
2027    Elm_Win *win;
2028    ELM_CHECK_WIDTYPE(obj, widtype);
2029    win = elm_widget_data_get(obj);
2030    if (!win) return;
2031    ecore_evas_iconified_set(win->ee, iconified);
2032 #ifdef HAVE_ELEMENTARY_X
2033    _elm_win_xwin_update(win);
2034 #endif
2035 }
2036
2037 EAPI Eina_Bool
2038 elm_win_iconified_get(const Evas_Object *obj)
2039 {
2040    Elm_Win *win;
2041    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2042    win = elm_widget_data_get(obj);
2043    if (!win) return EINA_FALSE;
2044    return ecore_evas_iconified_get(win->ee);
2045 }
2046
2047 EAPI void
2048 elm_win_layer_set(Evas_Object *obj, int layer)
2049 {
2050    Elm_Win *win;
2051    ELM_CHECK_WIDTYPE(obj, widtype);
2052    win = elm_widget_data_get(obj);
2053    if (!win) return;
2054    ecore_evas_layer_set(win->ee, layer);
2055 #ifdef HAVE_ELEMENTARY_X
2056    _elm_win_xwin_update(win);
2057 #endif
2058 }
2059
2060 EAPI int
2061 elm_win_layer_get(const Evas_Object *obj)
2062 {
2063    Elm_Win *win;
2064    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2065    win = elm_widget_data_get(obj);
2066    if (!win) return -1;
2067    return ecore_evas_layer_get(win->ee);
2068 }
2069
2070 EAPI void
2071 elm_win_rotation_set(Evas_Object *obj, int rotation)
2072 {
2073    Elm_Win *win;
2074    ELM_CHECK_WIDTYPE(obj, widtype);
2075    win = elm_widget_data_get(obj);
2076    if (!win) return;
2077    if (win->rot == rotation) return;
2078    win->rot = rotation;
2079    ecore_evas_rotation_set(win->ee, rotation);
2080    evas_object_size_hint_min_set(obj, -1, -1);
2081    evas_object_size_hint_max_set(obj, -1, -1);
2082    _elm_win_eval_subobjs(obj);
2083 #ifdef HAVE_ELEMENTARY_X
2084    _elm_win_xwin_update(win);
2085 #endif
2086 }
2087
2088 EAPI void
2089 elm_win_rotation_with_resize_set(Evas_Object *obj, int rotation)
2090 {
2091    Elm_Win *win;
2092    ELM_CHECK_WIDTYPE(obj, widtype);
2093    win = elm_widget_data_get(obj);
2094    if (!win) return;
2095    if (win->rot == rotation) return;
2096    win->rot = rotation;
2097    ecore_evas_rotation_with_resize_set(win->ee, rotation);
2098    evas_object_size_hint_min_set(obj, -1, -1);
2099    evas_object_size_hint_max_set(obj, -1, -1);
2100    _elm_win_eval_subobjs(obj);
2101 #ifdef HAVE_ELEMENTARY_X
2102    _elm_win_xwin_update(win);
2103 #endif
2104 }
2105
2106 EAPI int
2107 elm_win_rotation_get(const Evas_Object *obj)
2108 {
2109    Elm_Win *win;
2110    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2111    win = elm_widget_data_get(obj);
2112    if (!win) return -1;
2113    return win->rot;
2114 }
2115
2116 EAPI void
2117 elm_win_sticky_set(Evas_Object *obj, Eina_Bool sticky)
2118 {
2119    Elm_Win *win;
2120    ELM_CHECK_WIDTYPE(obj, widtype);
2121    win = elm_widget_data_get(obj);
2122    if (!win) return;
2123    ecore_evas_sticky_set(win->ee, sticky);
2124 #ifdef HAVE_ELEMENTARY_X
2125    _elm_win_xwin_update(win);
2126 #endif
2127 }
2128
2129 EAPI Eina_Bool
2130 elm_win_sticky_get(const Evas_Object *obj)
2131 {
2132    Elm_Win *win;
2133    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2134    win = elm_widget_data_get(obj);
2135    if (!win) return EINA_FALSE;
2136    return ecore_evas_sticky_get(win->ee);
2137 }
2138
2139 EAPI void
2140 elm_win_keyboard_mode_set(Evas_Object *obj, Elm_Win_Keyboard_Mode mode)
2141 {
2142    Elm_Win *win;
2143    ELM_CHECK_WIDTYPE(obj, widtype);
2144    win = elm_widget_data_get(obj);
2145    if (!win) return;
2146    if (mode == win->kbdmode) return;
2147 #ifdef HAVE_ELEMENTARY_X
2148    _elm_win_xwindow_get(win);
2149 #endif
2150    win->kbdmode = mode;
2151 #ifdef HAVE_ELEMENTARY_X
2152    if (win->xwin)
2153      ecore_x_e_virtual_keyboard_state_set
2154         (win->xwin, (Ecore_X_Virtual_Keyboard_State)win->kbdmode);
2155 #endif
2156 }
2157
2158 EAPI Elm_Win_Keyboard_Mode
2159 elm_win_keyboard_mode_get(const Evas_Object *obj)
2160 {
2161    Elm_Win *win;
2162    ELM_CHECK_WIDTYPE(obj, widtype) ELM_WIN_KEYBOARD_UNKNOWN;
2163    win = elm_widget_data_get(obj);
2164    if (!win) return ELM_WIN_KEYBOARD_UNKNOWN;
2165    return win->kbdmode;
2166 }
2167
2168 EAPI void
2169 elm_win_keyboard_win_set(Evas_Object *obj, Eina_Bool is_keyboard)
2170 {
2171    Elm_Win *win;
2172    ELM_CHECK_WIDTYPE(obj, widtype);
2173    win = elm_widget_data_get(obj);
2174    if (!win) return;
2175 #ifdef HAVE_ELEMENTARY_X
2176    _elm_win_xwindow_get(win);
2177    if (win->xwin)
2178      ecore_x_e_virtual_keyboard_set(win->xwin, is_keyboard);
2179 #else
2180    (void) is_keyboard;
2181 #endif
2182 }
2183
2184 EAPI Eina_Bool
2185 elm_win_keyboard_win_get(const Evas_Object *obj)
2186 {
2187    Elm_Win *win;
2188    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2189    win = elm_widget_data_get(obj);
2190    if (!win) return EINA_FALSE;
2191 #ifdef HAVE_ELEMENTARY_X
2192    _elm_win_xwindow_get(win);
2193    if (win->xwin)
2194      return ecore_x_e_virtual_keyboard_get(win->xwin);
2195 #endif
2196    return EINA_FALSE;
2197 }
2198
2199 EAPI void
2200 elm_win_screen_position_get(const Evas_Object *obj, int *x, int *y)
2201 {
2202    Elm_Win *win;
2203    ELM_CHECK_WIDTYPE(obj, widtype);
2204    win = elm_widget_data_get(obj);
2205    if (!win) return;
2206    if (x) *x = win->screen.x;
2207    if (y) *y = win->screen.y;
2208 }
2209
2210 EAPI Eina_Bool
2211 elm_win_focus_get(const Evas_Object *obj)
2212 {
2213    Elm_Win *win;
2214    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2215    win = elm_widget_data_get(obj);
2216    if (!win) return EINA_FALSE;
2217    return ecore_evas_focus_get(win->ee);
2218 }
2219
2220 EAPI void
2221 elm_win_screen_constrain_set(Evas_Object *obj, Eina_Bool constrain)
2222 {
2223    Elm_Win *win;
2224    ELM_CHECK_WIDTYPE(obj, widtype);
2225    win = elm_widget_data_get(obj);
2226    if (!win) return;
2227    win->constrain = !!constrain;
2228 }
2229
2230 EAPI Eina_Bool
2231 elm_win_screen_constrain_get(Evas_Object *obj)
2232 {
2233    Elm_Win *win;
2234    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2235    win = elm_widget_data_get(obj);
2236    if (!win) return EINA_FALSE;
2237    return win->constrain;
2238 }
2239
2240 EAPI void
2241 elm_win_screen_size_get(const Evas_Object *obj, int *x, int *y, int *w, int *h)
2242 {
2243    Elm_Win *win;
2244    ELM_CHECK_WIDTYPE(obj, widtype);
2245    win = elm_widget_data_get(obj);
2246    if (!win) return;
2247    ecore_evas_screen_geometry_get(win->ee, x, y, w, h);
2248 }
2249
2250 EAPI void
2251 elm_win_conformant_set(Evas_Object *obj, Eina_Bool conformant)
2252 {
2253    Elm_Win *win;
2254    ELM_CHECK_WIDTYPE(obj, widtype);
2255    win = elm_widget_data_get(obj);
2256    if (!win) return;
2257 #ifdef HAVE_ELEMENTARY_X
2258    _elm_win_xwindow_get(win);
2259    if (win->xwin)
2260      ecore_x_e_illume_conformant_set(win->xwin, conformant);
2261 #else
2262    (void) conformant;
2263 #endif
2264 }
2265
2266 EAPI Eina_Bool
2267 elm_win_conformant_get(const Evas_Object *obj)
2268 {
2269    Elm_Win *win;
2270    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2271    win = elm_widget_data_get(obj);
2272    if (!win) return EINA_FALSE;
2273 #ifdef HAVE_ELEMENTARY_X
2274    _elm_win_xwindow_get(win);
2275    if (win->xwin)
2276      return ecore_x_e_illume_conformant_get(win->xwin);
2277 #endif
2278    return EINA_FALSE;
2279 }
2280
2281 EAPI void
2282 elm_win_quickpanel_set(Evas_Object *obj, Eina_Bool quickpanel)
2283 {
2284    Elm_Win *win;
2285    ELM_CHECK_WIDTYPE(obj, widtype);
2286    win = elm_widget_data_get(obj);
2287    if (!win) return;
2288 #ifdef HAVE_ELEMENTARY_X
2289    _elm_win_xwindow_get(win);
2290    if (win->xwin)
2291      {
2292         ecore_x_e_illume_quickpanel_set(win->xwin, quickpanel);
2293         if (quickpanel)
2294           {
2295              Ecore_X_Window_State states[2];
2296
2297              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2298              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2299              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2300              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2301           }
2302      }
2303 #else
2304    (void) quickpanel;
2305 #endif
2306 }
2307
2308 EAPI Eina_Bool
2309 elm_win_quickpanel_get(const Evas_Object *obj)
2310 {
2311    Elm_Win *win;
2312    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2313    win = elm_widget_data_get(obj);
2314    if (!win) return EINA_FALSE;
2315 #ifdef HAVE_ELEMENTARY_X
2316    _elm_win_xwindow_get(win);
2317    if (win->xwin)
2318      return ecore_x_e_illume_quickpanel_get(win->xwin);
2319 #endif
2320    return EINA_FALSE;
2321 }
2322
2323 EAPI void
2324 elm_win_quickpanel_priority_major_set(Evas_Object *obj, int priority)
2325 {
2326    Elm_Win *win;
2327    ELM_CHECK_WIDTYPE(obj, widtype);
2328    win = elm_widget_data_get(obj);
2329    if (!win) return;
2330 #ifdef HAVE_ELEMENTARY_X
2331    _elm_win_xwindow_get(win);
2332    if (win->xwin)
2333      ecore_x_e_illume_quickpanel_priority_major_set(win->xwin, priority);
2334 #else
2335    (void) priority;
2336 #endif
2337 }
2338
2339 EAPI int
2340 elm_win_quickpanel_priority_major_get(const Evas_Object *obj)
2341 {
2342    Elm_Win *win;
2343    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2344    win = elm_widget_data_get(obj);
2345    if (!win) return -1;
2346 #ifdef HAVE_ELEMENTARY_X
2347    _elm_win_xwindow_get(win);
2348    if (win->xwin)
2349      return ecore_x_e_illume_quickpanel_priority_major_get(win->xwin);
2350 #endif
2351    return -1;
2352 }
2353
2354 EAPI void
2355 elm_win_quickpanel_priority_minor_set(Evas_Object *obj, int priority)
2356 {
2357    Elm_Win *win;
2358    ELM_CHECK_WIDTYPE(obj, widtype);
2359    win = elm_widget_data_get(obj);
2360    if (!win) return;
2361 #ifdef HAVE_ELEMENTARY_X
2362    _elm_win_xwindow_get(win);
2363    if (win->xwin)
2364      ecore_x_e_illume_quickpanel_priority_minor_set(win->xwin, priority);
2365 #else
2366    (void) priority;
2367 #endif
2368 }
2369
2370 EAPI int
2371 elm_win_quickpanel_priority_minor_get(const Evas_Object *obj)
2372 {
2373    Elm_Win *win;
2374    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2375    win = elm_widget_data_get(obj);
2376    if (!win) return -1;
2377 #ifdef HAVE_ELEMENTARY_X
2378    _elm_win_xwindow_get(win);
2379    if (win->xwin)
2380      return ecore_x_e_illume_quickpanel_priority_minor_get(win->xwin);
2381 #endif
2382    return -1;
2383 }
2384
2385 EAPI void
2386 elm_win_quickpanel_zone_set(Evas_Object *obj, int zone)
2387 {
2388    Elm_Win *win;
2389    ELM_CHECK_WIDTYPE(obj, widtype);
2390    win = elm_widget_data_get(obj);
2391    if (!win) return;
2392 #ifdef HAVE_ELEMENTARY_X
2393    _elm_win_xwindow_get(win);
2394    if (win->xwin)
2395      ecore_x_e_illume_quickpanel_zone_set(win->xwin, zone);
2396 #else
2397    (void) zone;
2398 #endif
2399 }
2400
2401 EAPI int
2402 elm_win_quickpanel_zone_get(const Evas_Object *obj)
2403 {
2404    Elm_Win *win;
2405    ELM_CHECK_WIDTYPE(obj, widtype) 0;
2406    win = elm_widget_data_get(obj);
2407    if (!win) return 0;
2408 #ifdef HAVE_ELEMENTARY_X
2409    _elm_win_xwindow_get(win);
2410    if (win->xwin)
2411      return ecore_x_e_illume_quickpanel_zone_get(win->xwin);
2412 #endif
2413    return 0;
2414 }
2415
2416 EAPI void
2417 elm_win_prop_focus_skip_set(Evas_Object *obj, Eina_Bool skip)
2418 {
2419    Elm_Win *win;
2420    ELM_CHECK_WIDTYPE(obj, widtype);
2421    win = elm_widget_data_get(obj);
2422    if (!win) return;
2423 #ifdef HAVE_ELEMENTARY_X
2424    _elm_win_xwindow_get(win);
2425    if (skip)
2426      {
2427         if (win->xwin)
2428           {
2429              Ecore_X_Window_State states[2];
2430
2431              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2432              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2433              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2434              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2435           }
2436      }
2437 #else
2438    (void) skip;
2439 #endif
2440 }
2441
2442 EAPI void
2443 elm_win_illume_command_send(Evas_Object *obj, Elm_Illume_Command command, void *params __UNUSED__)
2444 {
2445    Elm_Win *win;
2446    ELM_CHECK_WIDTYPE(obj, widtype);
2447    win = elm_widget_data_get(obj);
2448    if (!win) return;
2449 #ifdef HAVE_ELEMENTARY_X
2450    _elm_win_xwindow_get(win);
2451    if (win->xwin)
2452      {
2453         switch (command)
2454           {
2455            case ELM_ILLUME_COMMAND_FOCUS_BACK:
2456               ecore_x_e_illume_focus_back_send(win->xwin);
2457               break;
2458            case ELM_ILLUME_COMMAND_FOCUS_FORWARD:
2459               ecore_x_e_illume_focus_forward_send(win->xwin);
2460               break;
2461            case ELM_ILLUME_COMMAND_FOCUS_HOME:
2462               ecore_x_e_illume_focus_home_send(win->xwin);
2463               break;
2464            case ELM_ILLUME_COMMAND_CLOSE:
2465               ecore_x_e_illume_close_send(win->xwin);
2466               break;
2467            default:
2468               break;
2469           }
2470      }
2471 #else
2472    (void) command;
2473 #endif
2474 }
2475
2476 EAPI Evas_Object *
2477 elm_win_inlined_image_object_get(Evas_Object *obj)
2478 {
2479    Elm_Win *win;
2480    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2481    win = elm_widget_data_get(obj);
2482    if (!win) return NULL;
2483    return win->img_obj;
2484 }
2485
2486 EAPI void
2487 elm_win_focus_highlight_enabled_set(Evas_Object *obj, Eina_Bool enabled)
2488 {
2489    Elm_Win *win;
2490
2491    ELM_CHECK_WIDTYPE(obj, widtype);
2492
2493    win = elm_widget_data_get(obj);
2494    enabled = !!enabled;
2495    if (win->focus_highlight.enabled == enabled)
2496      return;
2497
2498    win->focus_highlight.enabled = enabled;
2499
2500    if (win->focus_highlight.enabled)
2501      _elm_win_focus_highlight_init(win);
2502    else
2503      _elm_win_focus_highlight_shutdown(win);
2504 }
2505
2506 EAPI Eina_Bool
2507 elm_win_focus_highlight_enabled_get(const Evas_Object *obj)
2508 {
2509    Elm_Win *win;
2510
2511    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2512
2513    win = elm_widget_data_get(obj);
2514    return win->focus_highlight.enabled;
2515 }
2516
2517 EAPI void
2518 elm_win_focus_highlight_style_set(Evas_Object *obj, const char *style)
2519 {
2520    Elm_Win *win;
2521
2522    ELM_CHECK_WIDTYPE(obj, widtype);
2523
2524    win = elm_widget_data_get(obj);
2525    eina_stringshare_replace(&win->focus_highlight.style, style);
2526    win->focus_highlight.changed_theme = EINA_TRUE;
2527    _elm_win_focus_highlight_reconfigure_job_start(win);
2528 }
2529
2530 EAPI const char *
2531 elm_win_focus_highlight_style_get(const Evas_Object *obj)
2532 {
2533    Elm_Win *win;
2534
2535    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2536
2537    win = elm_widget_data_get(obj);
2538    return win->focus_highlight.style;
2539 }
2540
2541 typedef struct _Widget_Data Widget_Data;
2542
2543 struct _Widget_Data
2544 {
2545    Evas_Object *frm;
2546    Evas_Object *content;
2547 };
2548
2549 static void _del_hook(Evas_Object *obj);
2550 static void _theme_hook(Evas_Object *obj);
2551 static void _sizing_eval(Evas_Object *obj);
2552 static void _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
2553 static void _sub_del(void *data, Evas_Object *obj, void *event_info);
2554
2555 static const char *widtype2 = NULL;
2556
2557 static void
2558 _del_hook(Evas_Object *obj)
2559 {
2560    Widget_Data *wd = elm_widget_data_get(obj);
2561    if (!wd) return;
2562    free(wd);
2563 }
2564
2565 static void
2566 _theme_hook(Evas_Object *obj)
2567 {
2568    Widget_Data *wd = elm_widget_data_get(obj);
2569    _elm_theme_object_set(obj, wd->frm, "win", "inwin", elm_widget_style_get(obj));
2570    if (wd->content)
2571      edje_object_part_swallow(wd->frm, "elm.swallow.content", wd->content);
2572    _sizing_eval(obj);
2573
2574    evas_object_smart_callback_call(obj, SIG_THEME_CHANGED, NULL);
2575 }
2576
2577 static Eina_Bool
2578 _elm_inwin_focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next)
2579 {
2580    Widget_Data *wd = elm_widget_data_get(obj);
2581
2582    if (!wd)
2583      return EINA_FALSE;
2584
2585    /* Try Focus cycle in subitem */
2586    if (wd->content)
2587      {
2588         elm_widget_focus_next_get(wd->content, dir, next);
2589         if (*next)
2590           return EINA_TRUE;
2591      }
2592
2593    *next = (Evas_Object *)obj;
2594    return EINA_FALSE;
2595 }
2596
2597 static void
2598 _sizing_eval(Evas_Object *obj)
2599 {
2600    Widget_Data *wd = elm_widget_data_get(obj);
2601    Evas_Coord minw = -1, minh = -1;
2602
2603    evas_object_size_hint_min_get(wd->content, &minw, &minh);
2604    edje_object_size_min_calc(wd->frm, &minw, &minh);
2605    evas_object_size_hint_min_set(obj, minw, minh);
2606    evas_object_size_hint_max_set(obj, -1, -1);
2607 }
2608
2609 static void
2610 _changed_size_hints(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
2611 {
2612    _sizing_eval(data);
2613 }
2614
2615 static void
2616 _sub_del(void *data __UNUSED__, Evas_Object *obj, void *event_info)
2617 {
2618    Widget_Data *wd = elm_widget_data_get(obj);
2619    Evas_Object *sub = event_info;
2620    if (sub == wd->content)
2621      {
2622         evas_object_event_callback_del_full
2623            (sub, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _changed_size_hints, obj);
2624         wd->content = NULL;
2625         _sizing_eval(obj);
2626      }
2627 }
2628
2629 EAPI Evas_Object *
2630 elm_win_inwin_add(Evas_Object *obj)
2631 {
2632    Evas_Object *obj2;
2633    Widget_Data *wd;
2634    Elm_Win *win;
2635
2636    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2637    win = elm_widget_data_get(obj);
2638    if (!win) return NULL;
2639    wd = ELM_NEW(Widget_Data);
2640    obj2 = elm_widget_add(win->evas);
2641    elm_widget_type_set(obj2, "inwin");
2642    ELM_SET_WIDTYPE(widtype2, "inwin");
2643    elm_widget_sub_object_add(obj, obj2);
2644    evas_object_size_hint_weight_set(obj2, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
2645    evas_object_size_hint_align_set(obj2, EVAS_HINT_FILL, EVAS_HINT_FILL);
2646    elm_win_resize_object_add(obj, obj2);
2647
2648    elm_widget_data_set(obj2, wd);
2649    elm_widget_del_hook_set(obj2, _del_hook);
2650    elm_widget_theme_hook_set(obj2, _theme_hook);
2651    elm_widget_focus_next_hook_set(obj2, _elm_inwin_focus_next_hook);
2652    elm_widget_can_focus_set(obj2, EINA_TRUE);
2653    elm_widget_highlight_ignore_set(obj2, EINA_TRUE);
2654
2655    wd->frm = edje_object_add(win->evas);
2656    _elm_theme_object_set(obj, wd->frm, "win", "inwin", "default");
2657    elm_widget_resize_object_set(obj2, wd->frm);
2658
2659    evas_object_smart_callback_add(obj2, "sub-object-del", _sub_del, obj2);
2660
2661    _sizing_eval(obj2);
2662    return obj2;
2663 }
2664
2665 EAPI void
2666 elm_win_inwin_activate(Evas_Object *obj)
2667 {
2668    ELM_CHECK_WIDTYPE(obj, widtype2);
2669    Widget_Data *wd = elm_widget_data_get(obj);
2670    if (!wd) return;
2671    evas_object_raise(obj);
2672    evas_object_show(obj);
2673    edje_object_signal_emit(wd->frm, "elm,action,show", "elm");
2674    elm_object_focus_set(obj, EINA_TRUE);
2675 }
2676
2677 EAPI void
2678 elm_win_inwin_content_set(Evas_Object *obj, Evas_Object *content)
2679 {
2680    ELM_CHECK_WIDTYPE(obj, widtype2);
2681    Widget_Data *wd = elm_widget_data_get(obj);
2682    if (!wd) return;
2683    if (wd->content == content) return;
2684    if (wd->content) evas_object_del(wd->content);
2685    wd->content = content;
2686    if (content)
2687      {
2688         elm_widget_sub_object_add(obj, content);
2689         evas_object_event_callback_add(content, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
2690                                        _changed_size_hints, obj);
2691         edje_object_part_swallow(wd->frm, "elm.swallow.content", content);
2692      }
2693    _sizing_eval(obj);
2694 }
2695
2696 EAPI Evas_Object *
2697 elm_win_inwin_content_get(const Evas_Object *obj)
2698 {
2699    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2700    Widget_Data *wd = elm_widget_data_get(obj);
2701    if (!wd) return NULL;
2702    return wd->content;
2703 }
2704
2705 EAPI Evas_Object *
2706 elm_win_inwin_content_unset(Evas_Object *obj)
2707 {
2708    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2709    Widget_Data *wd = elm_widget_data_get(obj);
2710    if (!wd) return NULL;
2711    if (!wd->content) return NULL;
2712    Evas_Object *content = wd->content;
2713    elm_widget_sub_object_del(obj, wd->content);
2714    evas_object_event_callback_del_full(wd->content,
2715                                        EVAS_CALLBACK_CHANGED_SIZE_HINTS,
2716                                        _changed_size_hints, obj);
2717    edje_object_part_unswallow(wd->frm, wd->content);
2718    wd->content = NULL;
2719    return content;
2720 }
2721
2722 /* windowing spcific calls - shall we do this differently? */
2723
2724 static Ecore_X_Window
2725 _elm_ee_win_get(const Evas_Object *obj)
2726 {
2727    if (!obj) return 0;
2728 #ifdef HAVE_ELEMENTARY_X
2729    Ecore_Evas *ee = ecore_evas_ecore_evas_get(evas_object_evas_get(obj));
2730    if (ee) return (Ecore_X_Window)ecore_evas_window_get(ee);
2731 #endif
2732    return 0;
2733 }
2734
2735 EAPI Ecore_X_Window
2736 elm_win_xwindow_get(const Evas_Object *obj)
2737 {
2738    Elm_Win *win;
2739    const char *type;
2740
2741    if (!obj) return 0;
2742    type = elm_widget_type_get(obj);
2743    if ((!type) || (type != widtype)) return _elm_ee_win_get(obj);
2744    win = elm_widget_data_get(obj);
2745    if (!win) return 0;
2746 #ifdef HAVE_ELEMENTARY_X
2747    if (win->xwin) return win->xwin;
2748    if (win->parent) return elm_win_xwindow_get(win->parent);
2749 #endif
2750    return 0;
2751 }