fix super annoying elm_win bug that went something like this:
[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 (win->constrain)
643           {
644              int sw, sh;
645              ecore_evas_screen_geometry_get(win->ee, NULL, NULL, &sw, &sh);
646              w = MIN(w, sw);
647              h = MIN(h, sh);
648           }
649         if (w < 1) w = 1;
650         if (h < 1) h = 1;
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 static void
1371 _subobj_del(Evas_Object *obj, Evas_Object *subobj, void *event_info __UNUSED__)
1372 {
1373    Elm_Win *win = elm_widget_data_get(obj);
1374    evas_object_event_callback_del_full(subobj,
1375                                        EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1376                                        _elm_win_subobj_callback_changed_size_hints,
1377                                        obj);
1378    evas_object_event_callback_del_full(subobj, EVAS_CALLBACK_DEL,
1379                                        _elm_win_subobj_callback_del, obj);
1380    win->subobjs = eina_list_remove(win->subobjs, subobj);
1381    evas_object_smart_callback_del(subobj, "win-resize-del-private-dont-use-this. seriously", (Evas_Smart_Cb)_subobj_del);
1382    _elm_win_eval_subobjs(obj);
1383 }
1384
1385 EAPI Evas_Object *
1386 elm_win_add(Evas_Object *parent, const char *name, Elm_Win_Type type)
1387 {
1388    Elm_Win *win;
1389    const Eina_List *l;
1390    const char *fontpath;
1391
1392    win = ELM_NEW(Elm_Win);
1393
1394 #define FALLBACK_TRY(engine)                                            \
1395    if (!win->ee)                                                        \
1396       do {                                                              \
1397          CRITICAL(engine " engine creation failed. Trying default.");   \
1398          win->ee = ecore_evas_new(NULL, 0, 0, 1, 1, NULL);              \
1399          if (win->ee)                                                   \
1400             elm_engine_set(ecore_evas_engine_name_get(win->ee));        \
1401    } while (0)
1402 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1403
1404    switch (type)
1405      {
1406       case ELM_WIN_INLINED_IMAGE:
1407         if (!parent) break;
1408         {
1409            Evas *e = evas_object_evas_get(parent);
1410            Ecore_Evas *ee;
1411            if (!e) break;
1412            ee = ecore_evas_ecore_evas_get(e);
1413            if (!ee) break;
1414            win->img_obj = ecore_evas_object_image_new(ee);
1415            if (!win->img_obj) break;
1416            win->ee = ecore_evas_object_ecore_evas_get(win->img_obj);
1417            if (win->ee)
1418              {
1419                 _win_inlined_image_set(win);
1420                 break;
1421              }
1422            evas_object_del(win->img_obj);
1423            win->img_obj = NULL;
1424         }
1425         break;
1426       default:
1427         if (ENGINE_COMPARE(ELM_SOFTWARE_X11))
1428           {
1429              win->ee = ecore_evas_software_x11_new(NULL, 0, 0, 0, 1, 1);
1430 #ifdef HAVE_ELEMENTARY_X
1431              win->client_message_handler = ecore_event_handler_add
1432                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1433 #endif
1434              FALLBACK_TRY("Sofware X11");
1435           }
1436         else if (ENGINE_COMPARE(ELM_SOFTWARE_FB))
1437           {
1438              win->ee = ecore_evas_fb_new(NULL, 0, 1, 1);
1439              FALLBACK_TRY("Sofware FB");
1440           }
1441         else if (ENGINE_COMPARE(ELM_SOFTWARE_DIRECTFB))
1442           {
1443              win->ee = ecore_evas_directfb_new(NULL, 1, 0, 0, 1, 1);
1444              FALLBACK_TRY("Sofware DirectFB");
1445           }
1446         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_X11))
1447           {
1448              win->ee = ecore_evas_software_x11_16_new(NULL, 0, 0, 0, 1, 1);
1449              FALLBACK_TRY("Sofware-16");
1450 #ifdef HAVE_ELEMENTARY_X
1451              win->client_message_handler = ecore_event_handler_add
1452                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1453 #endif
1454      }
1455         else if (ENGINE_COMPARE(ELM_SOFTWARE_8_X11))
1456           {
1457              win->ee = ecore_evas_software_x11_8_new(NULL, 0, 0, 0, 1, 1);
1458              FALLBACK_TRY("Sofware-8");
1459 #ifdef HAVE_ELEMENTARY_X
1460              win->client_message_handler = ecore_event_handler_add
1461                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1462 #endif
1463           }
1464 /* killed
1465         else if (ENGINE_COMPARE(ELM_XRENDER_X11))
1466           {
1467              win->ee = ecore_evas_xrender_x11_new(NULL, 0, 0, 0, 1, 1);
1468              FALLBACK_TRY("XRender");
1469 #ifdef HAVE_ELEMENTARY_X
1470              win->client_message_handler = ecore_event_handler_add
1471                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1472 #endif
1473           }
1474  */
1475         else if (ENGINE_COMPARE(ELM_OPENGL_X11))
1476           {
1477              int opt[10];
1478              int opt_i = 0;
1479
1480              if (_elm_config->vsync)
1481                {
1482                   opt[opt_i] = ECORE_EVAS_GL_X11_OPT_VSYNC;
1483                   opt_i++;
1484                   opt[opt_i] = 1;
1485                   opt_i++;
1486                }
1487              if (opt_i > 0)
1488                 win->ee = ecore_evas_gl_x11_options_new(NULL, 0, 0, 0, 1, 1, opt);
1489              else
1490                 win->ee = ecore_evas_gl_x11_new(NULL, 0, 0, 0, 1, 1);
1491              FALLBACK_TRY("OpenGL");
1492 #ifdef HAVE_ELEMENTARY_X
1493              win->client_message_handler = ecore_event_handler_add
1494                 (ECORE_X_EVENT_CLIENT_MESSAGE, _elm_win_client_message, win);
1495 #endif
1496           }
1497         else if (ENGINE_COMPARE(ELM_SOFTWARE_WIN32))
1498           {
1499              win->ee = ecore_evas_software_gdi_new(NULL, 0, 0, 1, 1);
1500              FALLBACK_TRY("Sofware Win32");
1501           }
1502         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1503           {
1504              win->ee = ecore_evas_software_wince_gdi_new(NULL, 0, 0, 1, 1);
1505              FALLBACK_TRY("Sofware-16-WinCE");
1506           }
1507         else if (ENGINE_COMPARE(ELM_SOFTWARE_PSL1GHT))
1508           {
1509              win->ee = ecore_evas_psl1ght_new(NULL, 1, 1);
1510              FALLBACK_TRY("PSL1GHT");
1511           }
1512         else if (ENGINE_COMPARE(ELM_SOFTWARE_SDL))
1513           {
1514              win->ee = ecore_evas_sdl_new(NULL, 0, 0, 0, 0, 0, 1);
1515              FALLBACK_TRY("Sofware SDL");
1516           }
1517         else if (ENGINE_COMPARE(ELM_SOFTWARE_16_SDL))
1518           {
1519              win->ee = ecore_evas_sdl16_new(NULL, 0, 0, 0, 0, 0, 1);
1520              FALLBACK_TRY("Sofware-16-SDL");
1521           }
1522         else if (ENGINE_COMPARE(ELM_OPENGL_SDL))
1523           {
1524              win->ee = ecore_evas_gl_sdl_new(NULL, 1, 1, 0, 0);
1525              FALLBACK_TRY("OpenGL SDL");
1526           }
1527         else if (ENGINE_COMPARE(ELM_OPENGL_COCOA))
1528           {
1529              win->ee = ecore_evas_cocoa_new(NULL, 1, 1, 0, 0);
1530              FALLBACK_TRY("OpenGL Cocoa");
1531           }
1532         else if (ENGINE_COMPARE(ELM_BUFFER))
1533           {
1534              win->ee = ecore_evas_buffer_new(1, 1);
1535           }
1536         else if (ENGINE_COMPARE(ELM_EWS))
1537           {
1538              win->ee = ecore_evas_ews_new(0, 0, 1, 1);
1539           }
1540         else if (!strncmp(_elm_config->engine, "shot:", 5))
1541           {
1542              win->ee = ecore_evas_buffer_new(1, 1);
1543              ecore_evas_manual_render_set(win->ee, EINA_TRUE);
1544              win->shot.info = eina_stringshare_add(_elm_config->engine + 5);
1545              _shot_init(win);
1546           }
1547 #undef FALLBACK_TRY
1548         break;
1549      }
1550
1551    if (!win->ee)
1552      {
1553         ERR("Cannot create window.");
1554         free(win);
1555         return NULL;
1556      }
1557 #ifdef HAVE_ELEMENTARY_X
1558    _elm_win_xwindow_get(win);
1559 #endif
1560    if ((_elm_config->bgpixmap) && (!_elm_config->compositing))
1561      ecore_evas_avoid_damage_set(win->ee, ECORE_EVAS_AVOID_DAMAGE_EXPOSE);
1562    // bg pixmap done by x - has other issues like can be redrawn by x before it
1563    // is filled/ready by app
1564    //     ecore_evas_avoid_damage_set(win->ee, ECORE_EVAS_AVOID_DAMAGE_BUILT_IN);
1565
1566    win->type = type;
1567    win->parent = parent;
1568    if (win->parent)
1569      evas_object_event_callback_add(win->parent, EVAS_CALLBACK_DEL,
1570                                     _elm_win_obj_callback_parent_del, win);
1571
1572    win->evas = ecore_evas_get(win->ee);
1573    win->win_obj = elm_widget_add(win->evas);
1574    elm_widget_type_set(win->win_obj, "win");
1575    ELM_SET_WIDTYPE(widtype, "win");
1576    elm_widget_data_set(win->win_obj, win);
1577    elm_widget_event_hook_set(win->win_obj, _elm_win_event_cb);
1578    elm_widget_on_focus_hook_set(win->win_obj, _elm_win_on_focus_hook, NULL);
1579    elm_widget_can_focus_set(win->win_obj, EINA_TRUE);
1580    elm_widget_highlight_ignore_set(win->win_obj, EINA_TRUE);
1581    elm_widget_focus_next_hook_set(win->win_obj, _elm_win_focus_next_hook);
1582    evas_object_color_set(win->win_obj, 0, 0, 0, 0);
1583    evas_object_move(win->win_obj, 0, 0);
1584    evas_object_resize(win->win_obj, 1, 1);
1585    evas_object_layer_set(win->win_obj, 50);
1586    evas_object_pass_events_set(win->win_obj, EINA_TRUE);
1587
1588    if (type == ELM_WIN_INLINED_IMAGE)
1589      elm_widget_parent2_set(win->win_obj, parent);
1590    ecore_evas_object_associate(win->ee, win->win_obj,
1591                                ECORE_EVAS_OBJECT_ASSOCIATE_BASE |
1592                                ECORE_EVAS_OBJECT_ASSOCIATE_STACK |
1593                                ECORE_EVAS_OBJECT_ASSOCIATE_LAYER);
1594    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_SHOW,
1595                                   _elm_win_obj_callback_show, win);
1596    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_HIDE,
1597                                   _elm_win_obj_callback_hide, win);
1598    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_DEL,
1599                                   _elm_win_obj_callback_del, win);
1600    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_MOVE,
1601                                   _elm_win_obj_callback_move, win);
1602    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_RESIZE,
1603                                   _elm_win_obj_callback_resize, win);
1604    if (win->img_obj)
1605      evas_object_intercept_move_callback_add(win->win_obj,
1606                                              _elm_win_obj_intercept_move, win);
1607    evas_object_intercept_show_callback_add(win->win_obj,
1608                                            _elm_win_obj_intercept_show, win);
1609
1610    ecore_evas_name_class_set(win->ee, name, _elm_appname);
1611    ecore_evas_callback_delete_request_set(win->ee, _elm_win_delete_request);
1612    ecore_evas_callback_resize_set(win->ee, _elm_win_resize);
1613    ecore_evas_callback_focus_in_set(win->ee, _elm_win_focus_in);
1614    ecore_evas_callback_focus_out_set(win->ee, _elm_win_focus_out);
1615    ecore_evas_callback_move_set(win->ee, _elm_win_move);
1616    evas_image_cache_set(win->evas, (_elm_config->image_cache * 1024));
1617    evas_font_cache_set(win->evas, (_elm_config->font_cache * 1024));
1618    EINA_LIST_FOREACH(_elm_config->font_dirs, l, fontpath)
1619      evas_font_path_append(win->evas, fontpath);
1620    if (!_elm_config->font_hinting)
1621      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_NONE);
1622    else if (_elm_config->font_hinting == 1)
1623      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_AUTO);
1624    else if (_elm_config->font_hinting == 2)
1625      evas_font_hinting_set(win->evas, EVAS_FONT_HINTING_BYTECODE);
1626
1627 #ifdef HAVE_ELEMENTARY_X
1628    _elm_win_xwin_update(win);
1629 #endif
1630
1631    _elm_win_list = eina_list_append(_elm_win_list, win->win_obj);
1632
1633    if (ENGINE_COMPARE(ELM_SOFTWARE_FB))
1634      {
1635         ecore_evas_fullscreen_set(win->ee, 1);
1636      }
1637 #undef ENGINE_COMPARE
1638
1639    if (_elm_config->focus_highlight_enable)
1640      elm_win_focus_highlight_enabled_set(win->win_obj, EINA_TRUE);
1641
1642 #ifdef ELM_DEBUG
1643    Evas_Modifier_Mask mask = evas_key_modifier_mask_get(win->evas, "Control");
1644    evas_object_event_callback_add(win->win_obj, EVAS_CALLBACK_KEY_DOWN,
1645                                   _debug_key_down, win);
1646
1647    Eina_Bool ret = evas_object_key_grab(win->win_obj, "F12", mask, 0,
1648                                         EINA_TRUE);
1649    printf("Ctrl+F12 key combination exclusive for dot tree generation\n");
1650 #endif
1651
1652    evas_object_smart_callbacks_descriptions_set(win->win_obj, _signals);
1653
1654    return win->win_obj;
1655 }
1656
1657 EAPI Evas_Object *
1658 elm_win_util_standard_add(const char *name, const char *title)
1659 {
1660    Evas_Object *win, *bg;
1661
1662    win = elm_win_add(NULL, name, ELM_WIN_BASIC);
1663    if (!win) return NULL;
1664    elm_win_title_set(win, title);
1665    bg = elm_bg_add(win);
1666    if (!bg)
1667      {
1668         evas_object_del(win);
1669         return NULL;
1670      }
1671    evas_object_size_hint_weight_set(bg, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
1672    elm_win_resize_object_add(win, bg);
1673    evas_object_show(bg);
1674    return win;
1675 }
1676
1677 EAPI void
1678 elm_win_resize_object_add(Evas_Object *obj, Evas_Object *subobj)
1679 {
1680    Evas_Coord w, h;
1681    Elm_Win *win;
1682    ELM_CHECK_WIDTYPE(obj, widtype);
1683    win = elm_widget_data_get(obj);
1684    if (!win) return;
1685    if (eina_list_data_find(win->subobjs, subobj)) return;
1686    win->subobjs = eina_list_append(win->subobjs, subobj);
1687    elm_widget_sub_object_add(obj, subobj);
1688    evas_object_event_callback_add(subobj, EVAS_CALLBACK_DEL,
1689                                   _elm_win_subobj_callback_del, obj);
1690    evas_object_event_callback_add(subobj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1691                                   _elm_win_subobj_callback_changed_size_hints,
1692                                   obj);
1693    evas_object_geometry_get(obj, NULL, NULL, &w, &h);
1694    evas_object_move(subobj, 0, 0);
1695    evas_object_resize(subobj, w, h);
1696    evas_object_smart_callback_add(subobj, "win-resize-del-private-dont-use-this. seriously", (Evas_Smart_Cb)_subobj_del, obj);
1697    _elm_win_eval_subobjs(obj);
1698 }
1699
1700 EAPI void
1701 elm_win_resize_object_del(Evas_Object *obj, Evas_Object *subobj)
1702 {
1703    Elm_Win *win;
1704    ELM_CHECK_WIDTYPE(obj, widtype);
1705    win = elm_widget_data_get(obj);
1706    if (!win) return;
1707    evas_object_event_callback_del_full(subobj,
1708                                        EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1709                                        _elm_win_subobj_callback_changed_size_hints,
1710                                        obj);
1711    evas_object_event_callback_del_full(subobj, EVAS_CALLBACK_DEL,
1712                                        _elm_win_subobj_callback_del, obj);
1713    win->subobjs = eina_list_remove(win->subobjs, subobj);
1714    elm_widget_sub_object_del(obj, subobj);
1715    _elm_win_eval_subobjs(obj);
1716 }
1717
1718 EAPI void
1719 elm_win_title_set(Evas_Object *obj, const char *title)
1720 {
1721    Elm_Win *win;
1722    ELM_CHECK_WIDTYPE(obj, widtype);
1723    win = elm_widget_data_get(obj);
1724    if (!win) return;
1725    ecore_evas_title_set(win->ee, title);
1726 }
1727
1728 EAPI const char *
1729 elm_win_title_get(const Evas_Object *obj)
1730 {
1731    Elm_Win *win;
1732    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
1733    win = elm_widget_data_get(obj);
1734    if (!win) return NULL;
1735    return ecore_evas_title_get(win->ee);
1736 }
1737
1738 EAPI void
1739 elm_win_autodel_set(Evas_Object *obj, Eina_Bool autodel)
1740 {
1741    Elm_Win *win;
1742    ELM_CHECK_WIDTYPE(obj, widtype);
1743    win = elm_widget_data_get(obj);
1744    if (!win) return;
1745    win->autodel = autodel;
1746 }
1747
1748 EAPI Eina_Bool
1749 elm_win_autodel_get(const Evas_Object *obj)
1750 {
1751    Elm_Win *win;
1752    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1753    win = elm_widget_data_get(obj);
1754    if (!win) return EINA_FALSE;
1755    return win->autodel;
1756 }
1757
1758 EAPI void
1759 elm_win_activate(Evas_Object *obj)
1760 {
1761    Elm_Win *win;
1762    ELM_CHECK_WIDTYPE(obj, widtype);
1763    win = elm_widget_data_get(obj);
1764    if (!win) return;
1765    ecore_evas_activate(win->ee);
1766 }
1767
1768 EAPI void
1769 elm_win_lower(Evas_Object *obj)
1770 {
1771    Elm_Win *win;
1772    ELM_CHECK_WIDTYPE(obj, widtype);
1773    win = elm_widget_data_get(obj);
1774    if (!win) return;
1775    ecore_evas_lower(win->ee);
1776 }
1777
1778 EAPI void
1779 elm_win_raise(Evas_Object *obj)
1780 {
1781    Elm_Win *win;
1782    ELM_CHECK_WIDTYPE(obj, widtype);
1783    win = elm_widget_data_get(obj);
1784    if (!win) return;
1785    ecore_evas_raise(win->ee);
1786 }
1787
1788 EAPI void
1789 elm_win_center(Evas_Object *obj, Eina_Bool h, Eina_Bool v)
1790 {
1791    Elm_Win *win;
1792    int win_w, win_h, screen_w, screen_h, nx, ny;
1793    ELM_CHECK_WIDTYPE(obj, widtype);
1794    win = elm_widget_data_get(obj);
1795    if (!win) return;
1796    ecore_evas_screen_geometry_get(win->ee, NULL, NULL, &screen_w, &screen_h);
1797    if ((!screen_w) || (!screen_h)) return;
1798    evas_object_geometry_get(obj, NULL, NULL, &win_w, &win_h);
1799    if ((!win_w) || (!win_h)) return;
1800    if (h) nx = win_w >= screen_w ? 0 : (screen_w / 2) - (win_w / 2);
1801    else nx = win->screen.x;
1802    if (v) ny = win_h >= screen_h ? 0 : (screen_h / 2) - (win_h / 2);
1803    else ny = win->screen.y;
1804    if (nx < 0) nx = 0;
1805    if (ny < 0) ny = 0;
1806    evas_object_move(obj, nx, ny);
1807 }
1808
1809 EAPI void
1810 elm_win_borderless_set(Evas_Object *obj, Eina_Bool borderless)
1811 {
1812    Elm_Win *win;
1813    ELM_CHECK_WIDTYPE(obj, widtype);
1814    win = elm_widget_data_get(obj);
1815    if (!win) return;
1816    ecore_evas_borderless_set(win->ee, borderless);
1817 #ifdef HAVE_ELEMENTARY_X
1818    _elm_win_xwin_update(win);
1819 #endif
1820 }
1821
1822 EAPI Eina_Bool
1823 elm_win_borderless_get(const Evas_Object *obj)
1824 {
1825    Elm_Win *win;
1826    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1827    win = elm_widget_data_get(obj);
1828    if (!win) return EINA_FALSE;
1829    return ecore_evas_borderless_get(win->ee);
1830 }
1831
1832 EAPI void
1833 elm_win_shaped_set(Evas_Object *obj, Eina_Bool shaped)
1834 {
1835    Elm_Win *win;
1836    ELM_CHECK_WIDTYPE(obj, widtype);
1837    win = elm_widget_data_get(obj);
1838    if (!win) return;
1839    ecore_evas_shaped_set(win->ee, shaped);
1840 #ifdef HAVE_ELEMENTARY_X
1841    _elm_win_xwin_update(win);
1842 #endif
1843 }
1844
1845 EAPI Eina_Bool
1846 elm_win_shaped_get(const Evas_Object *obj)
1847 {
1848    Elm_Win *win;
1849    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1850    win = elm_widget_data_get(obj);
1851    if (!win) return EINA_FALSE;
1852    return ecore_evas_shaped_get(win->ee);
1853 }
1854
1855 EAPI void
1856 elm_win_alpha_set(Evas_Object *obj, Eina_Bool alpha)
1857 {
1858    Elm_Win *win;
1859    ELM_CHECK_WIDTYPE(obj, widtype);
1860    win = elm_widget_data_get(obj);
1861    if (!win) return;
1862    if (win->frame_obj)
1863      {
1864      }
1865    else if (win->img_obj)
1866      {
1867         evas_object_image_alpha_set(win->img_obj, alpha);
1868         ecore_evas_alpha_set(win->ee, alpha);
1869      }
1870    else
1871      {
1872 #ifdef HAVE_ELEMENTARY_X
1873         if (win->xwin)
1874           {
1875              if (alpha)
1876                {
1877                   if (!_elm_config->compositing)
1878                      elm_win_shaped_set(obj, alpha);
1879                   else
1880                      ecore_evas_alpha_set(win->ee, alpha);
1881                }
1882              else
1883                 ecore_evas_alpha_set(win->ee, alpha);
1884              _elm_win_xwin_update(win);
1885           }
1886         else
1887 #endif
1888            ecore_evas_alpha_set(win->ee, alpha);
1889      }
1890 }
1891
1892 EAPI Eina_Bool
1893 elm_win_alpha_get(const Evas_Object *obj)
1894 {
1895    Elm_Win *win;
1896    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1897    win = elm_widget_data_get(obj);
1898    if (!win) return EINA_FALSE;
1899    if (win->frame_obj)
1900      {
1901      }
1902    else if (win->img_obj)
1903      {
1904         return evas_object_image_alpha_get(win->img_obj);
1905      }
1906    return ecore_evas_alpha_get(win->ee);
1907 }
1908
1909 EAPI void
1910 elm_win_transparent_set(Evas_Object *obj, Eina_Bool transparent)
1911 {
1912    Elm_Win *win;
1913    ELM_CHECK_WIDTYPE(obj, widtype);
1914    win = elm_widget_data_get(obj);
1915    if (!win) return;
1916
1917    if (win->frame_obj)
1918      {
1919      }
1920    else if (win->img_obj)
1921      {
1922         evas_object_image_alpha_set(win->img_obj, transparent);
1923      }
1924    else
1925      {
1926 #ifdef HAVE_ELEMENTARY_X
1927         if (win->xwin)
1928           {
1929              ecore_evas_transparent_set(win->ee, transparent);
1930              _elm_win_xwin_update(win);
1931           }
1932         else
1933 #endif
1934            ecore_evas_transparent_set(win->ee, transparent);
1935      }
1936 }
1937
1938 EAPI Eina_Bool
1939 elm_win_transparent_get(const Evas_Object *obj)
1940 {
1941    Elm_Win *win;
1942    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1943    win = elm_widget_data_get(obj);
1944    if (!win) return EINA_FALSE;
1945
1946    return ecore_evas_transparent_get(win->ee);
1947 }
1948
1949 EAPI void
1950 elm_win_override_set(Evas_Object *obj, Eina_Bool override)
1951 {
1952    Elm_Win *win;
1953    ELM_CHECK_WIDTYPE(obj, widtype);
1954    win = elm_widget_data_get(obj);
1955    if (!win) return;
1956    ecore_evas_override_set(win->ee, override);
1957 #ifdef HAVE_ELEMENTARY_X
1958    _elm_win_xwin_update(win);
1959 #endif
1960 }
1961
1962 EAPI Eina_Bool
1963 elm_win_override_get(const Evas_Object *obj)
1964 {
1965    Elm_Win *win;
1966    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1967    win = elm_widget_data_get(obj);
1968    if (!win) return EINA_FALSE;
1969    return ecore_evas_override_get(win->ee);
1970 }
1971
1972 EAPI void
1973 elm_win_fullscreen_set(Evas_Object *obj, Eina_Bool fullscreen)
1974 {
1975    Elm_Win *win;
1976    ELM_CHECK_WIDTYPE(obj, widtype);
1977    win = elm_widget_data_get(obj);
1978    if (!win) return;
1979
1980    // YYY: handle if win->img_obj
1981 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
1982    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
1983        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
1984      {
1985         // these engines... can ONLY be fullscreen
1986         return;
1987      }
1988    else
1989      {
1990         ecore_evas_fullscreen_set(win->ee, fullscreen);
1991 #ifdef HAVE_ELEMENTARY_X
1992         _elm_win_xwin_update(win);
1993 #endif
1994      }
1995 #undef ENGINE_COMPARE
1996 }
1997
1998 EAPI Eina_Bool
1999 elm_win_fullscreen_get(const Evas_Object *obj)
2000 {
2001    Elm_Win *win;
2002    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2003    win = elm_widget_data_get(obj);
2004    if (!win) return EINA_FALSE;
2005
2006 #define ENGINE_COMPARE(name) (!strcmp(_elm_config->engine, name))
2007    if (ENGINE_COMPARE(ELM_SOFTWARE_FB) ||
2008        ENGINE_COMPARE(ELM_SOFTWARE_16_WINCE))
2009      {
2010         // these engines... can ONLY be fullscreen
2011         return EINA_TRUE;
2012      }
2013    else
2014      {
2015         return ecore_evas_fullscreen_get(win->ee);
2016      }
2017 #undef ENGINE_COMPARE
2018 }
2019
2020 EAPI void
2021 elm_win_maximized_set(Evas_Object *obj, Eina_Bool maximized)
2022 {
2023    Elm_Win *win;
2024    ELM_CHECK_WIDTYPE(obj, widtype);
2025    win = elm_widget_data_get(obj);
2026    if (!win) return;
2027    // YYY: handle if win->img_obj
2028    ecore_evas_maximized_set(win->ee, maximized);
2029 #ifdef HAVE_ELEMENTARY_X
2030    _elm_win_xwin_update(win);
2031 #endif
2032 }
2033
2034 EAPI Eina_Bool
2035 elm_win_maximized_get(const Evas_Object *obj)
2036 {
2037    Elm_Win *win;
2038    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2039    win = elm_widget_data_get(obj);
2040    if (!win) return EINA_FALSE;
2041    return ecore_evas_maximized_get(win->ee);
2042 }
2043
2044 EAPI void
2045 elm_win_iconified_set(Evas_Object *obj, Eina_Bool iconified)
2046 {
2047    Elm_Win *win;
2048    ELM_CHECK_WIDTYPE(obj, widtype);
2049    win = elm_widget_data_get(obj);
2050    if (!win) return;
2051    ecore_evas_iconified_set(win->ee, iconified);
2052 #ifdef HAVE_ELEMENTARY_X
2053    _elm_win_xwin_update(win);
2054 #endif
2055 }
2056
2057 EAPI Eina_Bool
2058 elm_win_iconified_get(const Evas_Object *obj)
2059 {
2060    Elm_Win *win;
2061    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2062    win = elm_widget_data_get(obj);
2063    if (!win) return EINA_FALSE;
2064    return ecore_evas_iconified_get(win->ee);
2065 }
2066
2067 EAPI void
2068 elm_win_layer_set(Evas_Object *obj, int layer)
2069 {
2070    Elm_Win *win;
2071    ELM_CHECK_WIDTYPE(obj, widtype);
2072    win = elm_widget_data_get(obj);
2073    if (!win) return;
2074    ecore_evas_layer_set(win->ee, layer);
2075 #ifdef HAVE_ELEMENTARY_X
2076    _elm_win_xwin_update(win);
2077 #endif
2078 }
2079
2080 EAPI int
2081 elm_win_layer_get(const Evas_Object *obj)
2082 {
2083    Elm_Win *win;
2084    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2085    win = elm_widget_data_get(obj);
2086    if (!win) return -1;
2087    return ecore_evas_layer_get(win->ee);
2088 }
2089
2090 EAPI void
2091 elm_win_rotation_set(Evas_Object *obj, int rotation)
2092 {
2093    Elm_Win *win;
2094    ELM_CHECK_WIDTYPE(obj, widtype);
2095    win = elm_widget_data_get(obj);
2096    if (!win) return;
2097    if (win->rot == rotation) return;
2098    win->rot = rotation;
2099    ecore_evas_rotation_set(win->ee, rotation);
2100    evas_object_size_hint_min_set(obj, -1, -1);
2101    evas_object_size_hint_max_set(obj, -1, -1);
2102    _elm_win_eval_subobjs(obj);
2103 #ifdef HAVE_ELEMENTARY_X
2104    _elm_win_xwin_update(win);
2105 #endif
2106 }
2107
2108 EAPI void
2109 elm_win_rotation_with_resize_set(Evas_Object *obj, int rotation)
2110 {
2111    Elm_Win *win;
2112    ELM_CHECK_WIDTYPE(obj, widtype);
2113    win = elm_widget_data_get(obj);
2114    if (!win) return;
2115    if (win->rot == rotation) return;
2116    win->rot = rotation;
2117    ecore_evas_rotation_with_resize_set(win->ee, rotation);
2118    evas_object_size_hint_min_set(obj, -1, -1);
2119    evas_object_size_hint_max_set(obj, -1, -1);
2120    _elm_win_eval_subobjs(obj);
2121 #ifdef HAVE_ELEMENTARY_X
2122    _elm_win_xwin_update(win);
2123 #endif
2124 }
2125
2126 EAPI int
2127 elm_win_rotation_get(const Evas_Object *obj)
2128 {
2129    Elm_Win *win;
2130    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2131    win = elm_widget_data_get(obj);
2132    if (!win) return -1;
2133    return win->rot;
2134 }
2135
2136 EAPI void
2137 elm_win_sticky_set(Evas_Object *obj, Eina_Bool sticky)
2138 {
2139    Elm_Win *win;
2140    ELM_CHECK_WIDTYPE(obj, widtype);
2141    win = elm_widget_data_get(obj);
2142    if (!win) return;
2143    ecore_evas_sticky_set(win->ee, sticky);
2144 #ifdef HAVE_ELEMENTARY_X
2145    _elm_win_xwin_update(win);
2146 #endif
2147 }
2148
2149 EAPI Eina_Bool
2150 elm_win_sticky_get(const Evas_Object *obj)
2151 {
2152    Elm_Win *win;
2153    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2154    win = elm_widget_data_get(obj);
2155    if (!win) return EINA_FALSE;
2156    return ecore_evas_sticky_get(win->ee);
2157 }
2158
2159 EAPI void
2160 elm_win_keyboard_mode_set(Evas_Object *obj, Elm_Win_Keyboard_Mode mode)
2161 {
2162    Elm_Win *win;
2163    ELM_CHECK_WIDTYPE(obj, widtype);
2164    win = elm_widget_data_get(obj);
2165    if (!win) return;
2166    if (mode == win->kbdmode) return;
2167 #ifdef HAVE_ELEMENTARY_X
2168    _elm_win_xwindow_get(win);
2169 #endif
2170    win->kbdmode = mode;
2171 #ifdef HAVE_ELEMENTARY_X
2172    if (win->xwin)
2173      ecore_x_e_virtual_keyboard_state_set
2174         (win->xwin, (Ecore_X_Virtual_Keyboard_State)win->kbdmode);
2175 #endif
2176 }
2177
2178 EAPI Elm_Win_Keyboard_Mode
2179 elm_win_keyboard_mode_get(const Evas_Object *obj)
2180 {
2181    Elm_Win *win;
2182    ELM_CHECK_WIDTYPE(obj, widtype) ELM_WIN_KEYBOARD_UNKNOWN;
2183    win = elm_widget_data_get(obj);
2184    if (!win) return ELM_WIN_KEYBOARD_UNKNOWN;
2185    return win->kbdmode;
2186 }
2187
2188 EAPI void
2189 elm_win_keyboard_win_set(Evas_Object *obj, Eina_Bool is_keyboard)
2190 {
2191    Elm_Win *win;
2192    ELM_CHECK_WIDTYPE(obj, widtype);
2193    win = elm_widget_data_get(obj);
2194    if (!win) return;
2195 #ifdef HAVE_ELEMENTARY_X
2196    _elm_win_xwindow_get(win);
2197    if (win->xwin)
2198      ecore_x_e_virtual_keyboard_set(win->xwin, is_keyboard);
2199 #else
2200    (void) is_keyboard;
2201 #endif
2202 }
2203
2204 EAPI Eina_Bool
2205 elm_win_keyboard_win_get(const Evas_Object *obj)
2206 {
2207    Elm_Win *win;
2208    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2209    win = elm_widget_data_get(obj);
2210    if (!win) return EINA_FALSE;
2211 #ifdef HAVE_ELEMENTARY_X
2212    _elm_win_xwindow_get(win);
2213    if (win->xwin)
2214      return ecore_x_e_virtual_keyboard_get(win->xwin);
2215 #endif
2216    return EINA_FALSE;
2217 }
2218
2219 EAPI void
2220 elm_win_screen_position_get(const Evas_Object *obj, int *x, int *y)
2221 {
2222    Elm_Win *win;
2223    ELM_CHECK_WIDTYPE(obj, widtype);
2224    win = elm_widget_data_get(obj);
2225    if (!win) return;
2226    if (x) *x = win->screen.x;
2227    if (y) *y = win->screen.y;
2228 }
2229
2230 EAPI Eina_Bool
2231 elm_win_focus_get(const 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 ecore_evas_focus_get(win->ee);
2238 }
2239
2240 EAPI void
2241 elm_win_screen_constrain_set(Evas_Object *obj, Eina_Bool constrain)
2242 {
2243    Elm_Win *win;
2244    ELM_CHECK_WIDTYPE(obj, widtype);
2245    win = elm_widget_data_get(obj);
2246    if (!win) return;
2247    win->constrain = !!constrain;
2248 }
2249
2250 EAPI Eina_Bool
2251 elm_win_screen_constrain_get(Evas_Object *obj)
2252 {
2253    Elm_Win *win;
2254    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2255    win = elm_widget_data_get(obj);
2256    if (!win) return EINA_FALSE;
2257    return win->constrain;
2258 }
2259
2260 EAPI void
2261 elm_win_screen_size_get(const Evas_Object *obj, int *x, int *y, int *w, int *h)
2262 {
2263    Elm_Win *win;
2264    ELM_CHECK_WIDTYPE(obj, widtype);
2265    win = elm_widget_data_get(obj);
2266    if (!win) return;
2267    ecore_evas_screen_geometry_get(win->ee, x, y, w, h);
2268 }
2269
2270 EAPI void
2271 elm_win_conformant_set(Evas_Object *obj, Eina_Bool conformant)
2272 {
2273    Elm_Win *win;
2274    ELM_CHECK_WIDTYPE(obj, widtype);
2275    win = elm_widget_data_get(obj);
2276    if (!win) return;
2277 #ifdef HAVE_ELEMENTARY_X
2278    _elm_win_xwindow_get(win);
2279    if (win->xwin)
2280      ecore_x_e_illume_conformant_set(win->xwin, conformant);
2281 #else
2282    (void) conformant;
2283 #endif
2284 }
2285
2286 EAPI Eina_Bool
2287 elm_win_conformant_get(const Evas_Object *obj)
2288 {
2289    Elm_Win *win;
2290    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2291    win = elm_widget_data_get(obj);
2292    if (!win) return EINA_FALSE;
2293 #ifdef HAVE_ELEMENTARY_X
2294    _elm_win_xwindow_get(win);
2295    if (win->xwin)
2296      return ecore_x_e_illume_conformant_get(win->xwin);
2297 #endif
2298    return EINA_FALSE;
2299 }
2300
2301 EAPI void
2302 elm_win_quickpanel_set(Evas_Object *obj, Eina_Bool quickpanel)
2303 {
2304    Elm_Win *win;
2305    ELM_CHECK_WIDTYPE(obj, widtype);
2306    win = elm_widget_data_get(obj);
2307    if (!win) return;
2308 #ifdef HAVE_ELEMENTARY_X
2309    _elm_win_xwindow_get(win);
2310    if (win->xwin)
2311      {
2312         ecore_x_e_illume_quickpanel_set(win->xwin, quickpanel);
2313         if (quickpanel)
2314           {
2315              Ecore_X_Window_State states[2];
2316
2317              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2318              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2319              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2320              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2321           }
2322      }
2323 #else
2324    (void) quickpanel;
2325 #endif
2326 }
2327
2328 EAPI Eina_Bool
2329 elm_win_quickpanel_get(const Evas_Object *obj)
2330 {
2331    Elm_Win *win;
2332    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2333    win = elm_widget_data_get(obj);
2334    if (!win) return EINA_FALSE;
2335 #ifdef HAVE_ELEMENTARY_X
2336    _elm_win_xwindow_get(win);
2337    if (win->xwin)
2338      return ecore_x_e_illume_quickpanel_get(win->xwin);
2339 #endif
2340    return EINA_FALSE;
2341 }
2342
2343 EAPI void
2344 elm_win_quickpanel_priority_major_set(Evas_Object *obj, int priority)
2345 {
2346    Elm_Win *win;
2347    ELM_CHECK_WIDTYPE(obj, widtype);
2348    win = elm_widget_data_get(obj);
2349    if (!win) return;
2350 #ifdef HAVE_ELEMENTARY_X
2351    _elm_win_xwindow_get(win);
2352    if (win->xwin)
2353      ecore_x_e_illume_quickpanel_priority_major_set(win->xwin, priority);
2354 #else
2355    (void) priority;
2356 #endif
2357 }
2358
2359 EAPI int
2360 elm_win_quickpanel_priority_major_get(const Evas_Object *obj)
2361 {
2362    Elm_Win *win;
2363    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2364    win = elm_widget_data_get(obj);
2365    if (!win) return -1;
2366 #ifdef HAVE_ELEMENTARY_X
2367    _elm_win_xwindow_get(win);
2368    if (win->xwin)
2369      return ecore_x_e_illume_quickpanel_priority_major_get(win->xwin);
2370 #endif
2371    return -1;
2372 }
2373
2374 EAPI void
2375 elm_win_quickpanel_priority_minor_set(Evas_Object *obj, int priority)
2376 {
2377    Elm_Win *win;
2378    ELM_CHECK_WIDTYPE(obj, widtype);
2379    win = elm_widget_data_get(obj);
2380    if (!win) return;
2381 #ifdef HAVE_ELEMENTARY_X
2382    _elm_win_xwindow_get(win);
2383    if (win->xwin)
2384      ecore_x_e_illume_quickpanel_priority_minor_set(win->xwin, priority);
2385 #else
2386    (void) priority;
2387 #endif
2388 }
2389
2390 EAPI int
2391 elm_win_quickpanel_priority_minor_get(const Evas_Object *obj)
2392 {
2393    Elm_Win *win;
2394    ELM_CHECK_WIDTYPE(obj, widtype) -1;
2395    win = elm_widget_data_get(obj);
2396    if (!win) return -1;
2397 #ifdef HAVE_ELEMENTARY_X
2398    _elm_win_xwindow_get(win);
2399    if (win->xwin)
2400      return ecore_x_e_illume_quickpanel_priority_minor_get(win->xwin);
2401 #endif
2402    return -1;
2403 }
2404
2405 EAPI void
2406 elm_win_quickpanel_zone_set(Evas_Object *obj, int zone)
2407 {
2408    Elm_Win *win;
2409    ELM_CHECK_WIDTYPE(obj, widtype);
2410    win = elm_widget_data_get(obj);
2411    if (!win) return;
2412 #ifdef HAVE_ELEMENTARY_X
2413    _elm_win_xwindow_get(win);
2414    if (win->xwin)
2415      ecore_x_e_illume_quickpanel_zone_set(win->xwin, zone);
2416 #else
2417    (void) zone;
2418 #endif
2419 }
2420
2421 EAPI int
2422 elm_win_quickpanel_zone_get(const Evas_Object *obj)
2423 {
2424    Elm_Win *win;
2425    ELM_CHECK_WIDTYPE(obj, widtype) 0;
2426    win = elm_widget_data_get(obj);
2427    if (!win) return 0;
2428 #ifdef HAVE_ELEMENTARY_X
2429    _elm_win_xwindow_get(win);
2430    if (win->xwin)
2431      return ecore_x_e_illume_quickpanel_zone_get(win->xwin);
2432 #endif
2433    return 0;
2434 }
2435
2436 EAPI void
2437 elm_win_prop_focus_skip_set(Evas_Object *obj, Eina_Bool skip)
2438 {
2439    Elm_Win *win;
2440    ELM_CHECK_WIDTYPE(obj, widtype);
2441    win = elm_widget_data_get(obj);
2442    if (!win) return;
2443 #ifdef HAVE_ELEMENTARY_X
2444    _elm_win_xwindow_get(win);
2445    if (skip)
2446      {
2447         if (win->xwin)
2448           {
2449              Ecore_X_Window_State states[2];
2450
2451              ecore_x_icccm_hints_set(win->xwin, 0, 0, 0, 0, 0, 0, 0);
2452              states[0] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
2453              states[1] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
2454              ecore_x_netwm_window_state_set(win->xwin, states, 2);
2455           }
2456      }
2457 #else
2458    (void) skip;
2459 #endif
2460 }
2461
2462 EAPI void
2463 elm_win_illume_command_send(Evas_Object *obj, Elm_Illume_Command command, void *params __UNUSED__)
2464 {
2465    Elm_Win *win;
2466    ELM_CHECK_WIDTYPE(obj, widtype);
2467    win = elm_widget_data_get(obj);
2468    if (!win) return;
2469 #ifdef HAVE_ELEMENTARY_X
2470    _elm_win_xwindow_get(win);
2471    if (win->xwin)
2472      {
2473         switch (command)
2474           {
2475            case ELM_ILLUME_COMMAND_FOCUS_BACK:
2476               ecore_x_e_illume_focus_back_send(win->xwin);
2477               break;
2478            case ELM_ILLUME_COMMAND_FOCUS_FORWARD:
2479               ecore_x_e_illume_focus_forward_send(win->xwin);
2480               break;
2481            case ELM_ILLUME_COMMAND_FOCUS_HOME:
2482               ecore_x_e_illume_focus_home_send(win->xwin);
2483               break;
2484            case ELM_ILLUME_COMMAND_CLOSE:
2485               ecore_x_e_illume_close_send(win->xwin);
2486               break;
2487            default:
2488               break;
2489           }
2490      }
2491 #else
2492    (void) command;
2493 #endif
2494 }
2495
2496 EAPI Evas_Object *
2497 elm_win_inlined_image_object_get(Evas_Object *obj)
2498 {
2499    Elm_Win *win;
2500    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2501    win = elm_widget_data_get(obj);
2502    if (!win) return NULL;
2503    return win->img_obj;
2504 }
2505
2506 EAPI void
2507 elm_win_focus_highlight_enabled_set(Evas_Object *obj, Eina_Bool enabled)
2508 {
2509    Elm_Win *win;
2510
2511    ELM_CHECK_WIDTYPE(obj, widtype);
2512
2513    win = elm_widget_data_get(obj);
2514    enabled = !!enabled;
2515    if (win->focus_highlight.enabled == enabled)
2516      return;
2517
2518    win->focus_highlight.enabled = enabled;
2519
2520    if (win->focus_highlight.enabled)
2521      _elm_win_focus_highlight_init(win);
2522    else
2523      _elm_win_focus_highlight_shutdown(win);
2524 }
2525
2526 EAPI Eina_Bool
2527 elm_win_focus_highlight_enabled_get(const Evas_Object *obj)
2528 {
2529    Elm_Win *win;
2530
2531    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
2532
2533    win = elm_widget_data_get(obj);
2534    return win->focus_highlight.enabled;
2535 }
2536
2537 EAPI void
2538 elm_win_focus_highlight_style_set(Evas_Object *obj, const char *style)
2539 {
2540    Elm_Win *win;
2541
2542    ELM_CHECK_WIDTYPE(obj, widtype);
2543
2544    win = elm_widget_data_get(obj);
2545    eina_stringshare_replace(&win->focus_highlight.style, style);
2546    win->focus_highlight.changed_theme = EINA_TRUE;
2547    _elm_win_focus_highlight_reconfigure_job_start(win);
2548 }
2549
2550 EAPI const char *
2551 elm_win_focus_highlight_style_get(const Evas_Object *obj)
2552 {
2553    Elm_Win *win;
2554
2555    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2556
2557    win = elm_widget_data_get(obj);
2558    return win->focus_highlight.style;
2559 }
2560
2561 typedef struct _Widget_Data Widget_Data;
2562
2563 struct _Widget_Data
2564 {
2565    Evas_Object *frm;
2566    Evas_Object *content;
2567 };
2568
2569 static void _del_hook(Evas_Object *obj);
2570 static void _theme_hook(Evas_Object *obj);
2571 static void _sizing_eval(Evas_Object *obj);
2572 static void _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
2573 static void _sub_del(void *data, Evas_Object *obj, void *event_info);
2574
2575 static const char *widtype2 = NULL;
2576
2577 static void
2578 _del_hook(Evas_Object *obj)
2579 {
2580    Widget_Data *wd = elm_widget_data_get(obj);
2581    if (!wd) return;
2582    free(wd);
2583 }
2584
2585 static void
2586 _theme_hook(Evas_Object *obj)
2587 {
2588    Widget_Data *wd = elm_widget_data_get(obj);
2589    _elm_theme_object_set(obj, wd->frm, "win", "inwin", elm_widget_style_get(obj));
2590    if (wd->content)
2591      edje_object_part_swallow(wd->frm, "elm.swallow.content", wd->content);
2592    _sizing_eval(obj);
2593
2594    evas_object_smart_callback_call(obj, SIG_THEME_CHANGED, NULL);
2595 }
2596
2597 static Eina_Bool
2598 _elm_inwin_focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next)
2599 {
2600    Widget_Data *wd = elm_widget_data_get(obj);
2601
2602    if (!wd)
2603      return EINA_FALSE;
2604
2605    /* Try Focus cycle in subitem */
2606    if (wd->content)
2607      {
2608         elm_widget_focus_next_get(wd->content, dir, next);
2609         if (*next)
2610           return EINA_TRUE;
2611      }
2612
2613    *next = (Evas_Object *)obj;
2614    return EINA_FALSE;
2615 }
2616
2617 static void
2618 _sizing_eval(Evas_Object *obj)
2619 {
2620    Widget_Data *wd = elm_widget_data_get(obj);
2621    Evas_Coord minw = -1, minh = -1;
2622
2623    evas_object_size_hint_min_get(wd->content, &minw, &minh);
2624    edje_object_size_min_calc(wd->frm, &minw, &minh);
2625    evas_object_size_hint_min_set(obj, minw, minh);
2626    evas_object_size_hint_max_set(obj, -1, -1);
2627 }
2628
2629 static void
2630 _changed_size_hints(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
2631 {
2632    _sizing_eval(data);
2633 }
2634
2635 static void
2636 _sub_del(void *data __UNUSED__, Evas_Object *obj, void *event_info)
2637 {
2638    Widget_Data *wd = elm_widget_data_get(obj);
2639    Evas_Object *sub = event_info;
2640    if (sub == wd->content)
2641      {
2642         evas_object_event_callback_del_full
2643            (sub, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _changed_size_hints, obj);
2644         wd->content = NULL;
2645         _sizing_eval(obj);
2646      }
2647 }
2648
2649 EAPI Evas_Object *
2650 elm_win_inwin_add(Evas_Object *obj)
2651 {
2652    Evas_Object *obj2;
2653    Widget_Data *wd;
2654    Elm_Win *win;
2655
2656    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
2657    win = elm_widget_data_get(obj);
2658    if (!win) return NULL;
2659    wd = ELM_NEW(Widget_Data);
2660    obj2 = elm_widget_add(win->evas);
2661    elm_widget_type_set(obj2, "inwin");
2662    ELM_SET_WIDTYPE(widtype2, "inwin");
2663    elm_widget_sub_object_add(obj, obj2);
2664    evas_object_size_hint_weight_set(obj2, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
2665    evas_object_size_hint_align_set(obj2, EVAS_HINT_FILL, EVAS_HINT_FILL);
2666    elm_win_resize_object_add(obj, obj2);
2667
2668    elm_widget_data_set(obj2, wd);
2669    elm_widget_del_hook_set(obj2, _del_hook);
2670    elm_widget_theme_hook_set(obj2, _theme_hook);
2671    elm_widget_focus_next_hook_set(obj2, _elm_inwin_focus_next_hook);
2672    elm_widget_can_focus_set(obj2, EINA_TRUE);
2673    elm_widget_highlight_ignore_set(obj2, EINA_TRUE);
2674
2675    wd->frm = edje_object_add(win->evas);
2676    _elm_theme_object_set(obj, wd->frm, "win", "inwin", "default");
2677    elm_widget_resize_object_set(obj2, wd->frm);
2678
2679    evas_object_smart_callback_add(obj2, "sub-object-del", _sub_del, obj2);
2680
2681    _sizing_eval(obj2);
2682    return obj2;
2683 }
2684
2685 EAPI void
2686 elm_win_inwin_activate(Evas_Object *obj)
2687 {
2688    ELM_CHECK_WIDTYPE(obj, widtype2);
2689    Widget_Data *wd = elm_widget_data_get(obj);
2690    if (!wd) return;
2691    evas_object_raise(obj);
2692    evas_object_show(obj);
2693    edje_object_signal_emit(wd->frm, "elm,action,show", "elm");
2694    elm_object_focus_set(obj, EINA_TRUE);
2695 }
2696
2697 EAPI void
2698 elm_win_inwin_content_set(Evas_Object *obj, Evas_Object *content)
2699 {
2700    ELM_CHECK_WIDTYPE(obj, widtype2);
2701    Widget_Data *wd = elm_widget_data_get(obj);
2702    if (!wd) return;
2703    if (wd->content == content) return;
2704    if (wd->content) evas_object_del(wd->content);
2705    wd->content = content;
2706    if (content)
2707      {
2708         elm_widget_sub_object_add(obj, content);
2709         evas_object_event_callback_add(content, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
2710                                        _changed_size_hints, obj);
2711         edje_object_part_swallow(wd->frm, "elm.swallow.content", content);
2712      }
2713    _sizing_eval(obj);
2714 }
2715
2716 EAPI Evas_Object *
2717 elm_win_inwin_content_get(const Evas_Object *obj)
2718 {
2719    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2720    Widget_Data *wd = elm_widget_data_get(obj);
2721    if (!wd) return NULL;
2722    return wd->content;
2723 }
2724
2725 EAPI Evas_Object *
2726 elm_win_inwin_content_unset(Evas_Object *obj)
2727 {
2728    ELM_CHECK_WIDTYPE(obj, widtype2) NULL;
2729    Widget_Data *wd = elm_widget_data_get(obj);
2730    if (!wd) return NULL;
2731    if (!wd->content) return NULL;
2732    Evas_Object *content = wd->content;
2733    elm_widget_sub_object_del(obj, wd->content);
2734    evas_object_event_callback_del_full(wd->content,
2735                                        EVAS_CALLBACK_CHANGED_SIZE_HINTS,
2736                                        _changed_size_hints, obj);
2737    edje_object_part_unswallow(wd->frm, wd->content);
2738    wd->content = NULL;
2739    return content;
2740 }
2741
2742 /* windowing spcific calls - shall we do this differently? */
2743
2744 static Ecore_X_Window
2745 _elm_ee_win_get(const Evas_Object *obj)
2746 {
2747    if (!obj) return 0;
2748 #ifdef HAVE_ELEMENTARY_X
2749    Ecore_Evas *ee = ecore_evas_ecore_evas_get(evas_object_evas_get(obj));
2750    if (ee) return (Ecore_X_Window)ecore_evas_window_get(ee);
2751 #endif
2752    return 0;
2753 }
2754
2755 EAPI Ecore_X_Window
2756 elm_win_xwindow_get(const Evas_Object *obj)
2757 {
2758    Elm_Win *win;
2759    const char *type;
2760
2761    if (!obj) return 0;
2762    type = elm_widget_type_get(obj);
2763    if ((!type) || (type != widtype)) return _elm_ee_win_get(obj);
2764    win = elm_widget_data_get(obj);
2765    if (!win) return 0;
2766 #ifdef HAVE_ELEMENTARY_X
2767    if (win->xwin) return win->xwin;
2768    if (win->parent) return elm_win_xwindow_get(win->parent);
2769 #endif
2770    return 0;
2771 }