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