3 EAPI Ecore_X_Atom ATM__QTOPIA_SOFT_MENU = 0;
4 EAPI Ecore_X_Atom ATM__QTOPIA_SOFT_MENUS = 0;
5 EAPI Ecore_X_Atom ATM_GNOME_SM_PROXY = 0;
6 EAPI Ecore_X_Atom ATM_ENLIGHTENMENT_COMMS = 0;
7 EAPI Ecore_X_Atom ATM_ENLIGHTENMENT_VERSION = 0;
8 EAPI Ecore_X_Atom ATM_ENLIGHTENMENT_SCALE = 0;
13 Ecore_X_Window *roots = NULL;
15 const char *atom_names[] = {
19 "ENLIGHTENMENT_COMMS",
20 "ENLIGHTENMENT_VERSION",
23 Ecore_X_Atom atoms[6];
25 ecore_x_atoms_get(atom_names, 6, atoms);
26 ATM__QTOPIA_SOFT_MENU = atoms[0];
27 ATM__QTOPIA_SOFT_MENUS = atoms[1];
28 ATM_GNOME_SM_PROXY = atoms[2];
29 ATM_ENLIGHTENMENT_COMMS = atoms[3];
30 ATM_ENLIGHTENMENT_VERSION = atoms[4];
31 ATM_ENLIGHTENMENT_SCALE = atoms[5];
33 roots = ecore_x_window_root_list(&num);
36 Ecore_X_Atom supported[43];
40 /* Set what hints we support */
41 /* Root Window Properties (and Related Messages) */
42 supported[supported_num++] = ECORE_X_ATOM_NET_CLIENT_LIST;
43 supported[supported_num++] = ECORE_X_ATOM_NET_CLIENT_LIST_STACKING;
44 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_NUMBER_OF_DESKTOPS, 1);*/
45 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_DESKTOP_GEOMETRY, 1);*/
46 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_DESKTOP_VIEWPORT, 1);*/
47 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_CURRENT_DESKTOP, 1);*/
48 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_DESKTOP_NAMES, 1);*/
49 supported[supported_num++] = ECORE_X_ATOM_NET_ACTIVE_WINDOW;
50 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WORKAREA, 1);*/
51 supported[supported_num++] = ECORE_X_ATOM_NET_SUPPORTING_WM_CHECK;
52 supported[supported_num++] = ECORE_X_ATOM_NET_VIRTUAL_ROOTS;
53 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_DESKTOP_LAYOUT, 1);*/
54 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_SHOWING_DESKTOP, 1);*/
56 /* Other Root Window Messages */
57 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_CLOSE_WINDOW, 1);*/
58 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_MOVERESIZE_WINDOW, 1);*/
59 supported[supported_num++] = ECORE_X_ATOM_NET_WM_MOVERESIZE;
60 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_RESTACK_WINDOW, 1);*/
61 supported[supported_num++] = ECORE_X_ATOM_NET_REQUEST_FRAME_EXTENTS;
63 /* Application Window Properties */
64 supported[supported_num++] = ECORE_X_ATOM_NET_WM_NAME;
65 supported[supported_num++] = ECORE_X_ATOM_NET_WM_VISIBLE_NAME;
66 supported[supported_num++] = ECORE_X_ATOM_NET_WM_ICON_NAME;
67 supported[supported_num++] = ECORE_X_ATOM_NET_WM_VISIBLE_ICON_NAME;
68 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_DESKTOP, 1);*/
69 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE;
70 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_DESKTOP;
71 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_DOCK;
72 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_TOOLBAR;
73 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_MENU;
74 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_UTILITY;
75 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_SPLASH;
76 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_DIALOG;
77 supported[supported_num++] = ECORE_X_ATOM_NET_WM_WINDOW_TYPE_NORMAL;
78 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE;
79 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_MODAL;
80 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_STICKY;
81 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_MAXIMIZED_VERT;
82 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_MAXIMIZED_HORZ;
83 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_SHADED;
84 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_SKIP_TASKBAR;
85 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_SKIP_PAGER;
86 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_HIDDEN;
87 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_FULLSCREEN;
88 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_ABOVE;
89 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STATE_BELOW;
90 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_STATE_DEMANDS_ATTENTION, 1);*/
91 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ALLOWED_ACTIONS, 1);*/
92 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_MOVE, 1);*/
93 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_RESIZE, 1);*/
94 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_MINIMIZE, 1);*/
95 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_SHADE, 1);*/
96 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_STICK, 1);*/
97 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_MAXIMIZE_HORZ, 1);*/
98 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_MAXIMIZE_VERT, 1);*/
99 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_FULLSCREEN, 1);*/
100 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_CHANGE_DESKTOP, 1);*/
101 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ACTION_CLOSE, 1);*/
102 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STRUT;
103 supported[supported_num++] = ECORE_X_ATOM_NET_WM_STRUT_PARTIAL;
104 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_ICON_GEOMETRY, 1);*/
105 supported[supported_num++] = ECORE_X_ATOM_NET_WM_ICON;
106 supported[supported_num++] = ECORE_X_ATOM_NET_WM_PID;
107 /*ecore_x_netwm_supported(roots[supported_num], ECORE_X_ATOM_NET_WM_HANDLED_ICONS, 1);*/
108 supported[supported_num++] = ECORE_X_ATOM_NET_WM_USER_TIME;
109 supported[supported_num++] = ECORE_X_ATOM_NET_FRAME_EXTENTS;
110 supported[supported_num++] = ECORE_X_ATOM_NET_WM_PING;
111 supported[supported_num++] = ECORE_X_ATOM_NET_WM_SYNC_REQUEST;
112 supported[supported_num++] = ECORE_X_ATOM_NET_WM_SYNC_REQUEST_COUNTER;
113 supported[supported_num++] = ECORE_X_ATOM_E_VIDEO_PARENT;
114 supported[supported_num++] = ECORE_X_ATOM_E_VIDEO_POSITION;
116 for (i = 0; i < num; i++)
118 Ecore_X_Window win, twin;
123 /* check for previous netwm wm and wait for it to die */
124 ts = ecore_time_get();
125 nwins = ecore_x_window_prop_window_get(roots[i],
126 ECORE_X_ATOM_NET_SUPPORTING_WM_CHECK,
132 nwins = ecore_x_window_prop_window_get(win,
133 ECORE_X_ATOM_NET_SUPPORTING_WM_CHECK,
135 if (nwins < 1) break;
136 if (twin != win) break;
137 if (ecore_x_netwm_name_get(win, &name))
141 if (strcmp(name, "Enlightenment"))
150 if ((ecore_time_get() - ts) > 2.0)
152 e_error_message_show(_("A previous instance of Enlightenment is still active\n"
153 "on this screen. Aborting startup.\n"));
159 /* TODO: Remember this window and close it on shutdown */
160 win = ecore_x_window_new(roots[i], -200, -200, 5, 5);
162 * I don't FUCKING believe it. if we PRETEND we are Kwin - java is happy.
163 * why? it expects a double reparenting wm then. java insists on finding this
164 * out when it should be irrelevant! stupid code! I can't believe the time we
165 * just wasted hunting a bug that wasn't and that is due to sheer stupid
166 * coding (in java's awt layer that swing also uses).
168 /* Now for more stupidity... Openoffice.org will change its look and feel
169 * depending on what wm it thinks there is... so if we pretend to be Kwin...
170 * it tries to use kde preferences, if found.
172 /* I have disabled this now by pretending to be E16 with e16 comms. this
173 * means java plays nice and uses our FRAME property.. but we had to do other
174 * evil stuff as java EXPECTS all this at REPARENT time... i've deferred
175 * reparenting... i hate java!
177 /* ecore_x_netwm_wm_identify(roots[i], win, "KWin");*/
178 ecore_x_netwm_wm_identify(roots[i], win, "Enlightenment");
179 /* this makes openoffice.org read gtk settings so it doesn't look like shit */
180 e_hints_openoffice_gnome_fake(roots[i]);
182 ecore_x_netwm_supported_set(roots[i], supported, supported_num);
184 /* fake mwm, this might crash some ol' motif apps, if
185 they still exist, but at least it makes borderless
186 feature of Eterm and urxvt work... */
187 ecore_x_atom_get("_MOTIF_WM_INFO");
194 * This is here so we don't have to pretend to be Kwin anymore - we pretend
195 * to do old e16 style ipc. in fact we just ignore it... but set up the
199 e_hints_e16_comms_pretend(E_Manager *man)
204 win = ecore_x_window_input_new(man->root, -100, -100, 1, 1);
206 /* to help detect this is NOT e16 */
207 snprintf(buf, sizeof(buf), "Enlightenment %s", VERSION);
208 ecore_x_window_prop_property_set(win, ATM_ENLIGHTENMENT_VERSION, ECORE_X_ATOM_STRING, 8, buf, strlen(buf));
209 ecore_x_window_prop_property_set(man->root, ATM_ENLIGHTENMENT_VERSION, ECORE_X_ATOM_STRING, 8, buf, strlen(buf));
211 snprintf(buf, sizeof(buf), "WINID %8x", (int)win);
212 ecore_x_window_prop_property_set(win, ATM_ENLIGHTENMENT_COMMS, ECORE_X_ATOM_STRING, 8, buf, 14);
214 ecore_x_window_prop_property_set(man->root, ATM_ENLIGHTENMENT_COMMS, ECORE_X_ATOM_STRING, 8, buf, 14);
218 e_hints_manager_init(E_Manager *man)
220 /* Set desktop count, desktop names and workarea */
222 unsigned int *areas = NULL;
225 Ecore_X_Window *vroots = NULL;
226 /* FIXME: Desktop names not yet implemented */
229 e_hints_e16_comms_pretend(man);
231 num = eina_list_count(man->containers);
233 vroots = calloc(num, sizeof(Ecore_X_Window));
236 /* names = calloc(num, sizeof(char *));*/
238 areas = calloc(4 * num, sizeof(unsigned int));
245 EINA_LIST_FOREACH(man->containers, cl, c)
248 areas[4 * i + 1] = c->y;
249 areas[4 * i + 2] = c->w;
250 areas[4 * i + 3] = c->h;
251 vroots[i++] = c->win;
255 ecore_x_netwm_desk_count_set(man->root, num);
256 /* No need for workarea without desktops */
257 ecore_x_netwm_desk_workareas_set(man->root, num, areas);
264 /* FIXME, this should set the list in map order, not stack order */
266 e_hints_client_list_set(void)
268 Eina_List *ml = NULL, *cl = NULL;
269 unsigned int i = 0, num = 0;
274 Ecore_X_Window *clients = NULL;
276 /* Get client count by adding client lists on all containers */
277 EINA_LIST_FOREACH(e_manager_list(), ml, m)
279 EINA_LIST_FOREACH(m->containers, cl, c)
281 num += e_container_borders_count(c);
285 clients = calloc(num, sizeof(Ecore_X_Window));
289 /* Fetch window IDs and add to array */
292 EINA_LIST_FOREACH(e_manager_list(), ml, m)
295 EINA_LIST_FOREACH(m->containers, cl, c)
297 bl = e_container_border_list_first(c);
298 while ((b = e_container_border_list_next(bl)))
299 clients[i++] = b->client.win;
300 e_container_border_list_free(bl);
304 ecore_x_netwm_client_list_stacking_set(m->root, clients, i);
305 ecore_x_netwm_client_list_set(m->root, clients, i);
309 ecore_x_netwm_client_list_set(m->root, NULL, 0);
310 ecore_x_netwm_client_list_stacking_set(m->root, NULL, 0);
316 EINA_LIST_FOREACH(e_manager_list(), ml, m)
318 ecore_x_netwm_client_list_set(m->root, NULL, 0);
319 ecore_x_netwm_client_list_stacking_set(m->root, NULL, 0);
325 /* Client list is already in stacking order, so this function is nearly
326 * identical to the previous one */
328 e_hints_client_stacking_set(void)
330 Eina_List *ml = NULL, *cl = NULL;
331 unsigned int i = 0, num = 0;
336 Ecore_X_Window *clients = NULL;
338 /* Get client count */
339 EINA_LIST_FOREACH(e_manager_list(), ml, m)
341 EINA_LIST_FOREACH(m->containers, cl, c)
343 num += e_container_borders_count(c);
349 clients = calloc(num, sizeof(Ecore_X_Window));
350 if (!clients) return;
352 EINA_LIST_FOREACH(e_manager_list(), ml, m)
354 EINA_LIST_FOREACH(m->containers, cl, c)
356 bl = e_container_border_list_first(c);
357 while ((b = e_container_border_list_next(bl)))
361 e_error_message_show("e_hints.c: e_hints_client_stacking_set()\n"
363 "Window list size greater than window count.\n"
364 "This is really bad.\n"
365 "Please report this.\n");
368 clients[i++] = b->client.win;
370 e_container_border_list_free(bl);
375 e_error_message_show("e_hints.c: e_hints_client_stacking_set()\n"
377 "Window list size less than window count.\n"
378 "This is strange, but not harmful.\n"
379 "Please report this.\n");
381 EINA_LIST_FOREACH(e_manager_list(), ml, m)
383 ecore_x_netwm_client_list_stacking_set(m->root, clients, num);
389 EINA_LIST_FOREACH(e_manager_list(), ml, m)
391 ecore_x_netwm_client_list_stacking_set(m->root, NULL, 0);
397 e_hints_active_window_set(E_Manager *man,
402 ecore_x_netwm_client_active_set(man->root, bd->client.win);
404 ecore_x_netwm_client_active_set(man->root, 0);
408 e_hints_window_init(E_Border *bd)
410 E_Remember *rem = NULL;
415 if (bd->client.icccm.state == ECORE_X_WINDOW_STATE_HINT_NONE)
417 if (bd->client.netwm.state.hidden)
418 bd->client.icccm.state = ECORE_X_WINDOW_STATE_HINT_ICONIC;
420 bd->client.icccm.state = ECORE_X_WINDOW_STATE_HINT_NORMAL;
423 if ((rem) && (rem->prop.layer))
425 bd->layer = rem->prop.layer;
426 e_border_layer_set(bd, bd->layer);
430 if (!bd->lock_client_stacking)
432 if (bd->client.netwm.type == ECORE_X_WINDOW_TYPE_DESKTOP)
433 e_border_layer_set(bd, 0);
434 else if (bd->client.netwm.state.stacking == E_STACKING_BELOW)
435 e_border_layer_set(bd, 50);
436 else if (bd->client.netwm.state.stacking == E_STACKING_ABOVE)
437 e_border_layer_set(bd, 150);
438 else if (bd->client.netwm.type == ECORE_X_WINDOW_TYPE_DOCK)
439 e_border_layer_set(bd, 150);
440 #ifdef _F_NOTIFICATION_LAYER_POLICY_
441 else if (bd->client.netwm.type == ECORE_X_WINDOW_TYPE_NOTIFICATION)
442 e_border_layer_set(bd, 240);
445 e_border_layer_set(bd, 100);
451 if ((bd->parent) && (e_config->transient.layer))
452 e_border_layer_set(bd, bd->parent->layer);
455 /* Ignore this, E has incompatible desktop setup */
456 if (ecore_x_netwm_desktop_get(bd->client.win, &bd->client.netwm.desktop))
458 if (bd->client.netwm.desktop == 0xffffffff)
462 else if (bd->client.netwm.desktop < (bd->zone->desk_x_count * bd->zone->desk_y_count))
466 desk = e_desk_at_pos_get(bd->zone, bd->client.netwm.desktop);
468 e_border_desk_set(bd, desk);
472 /* Update netwm desktop with current desktop */
473 e_hints_window_desktop_set(bd);
478 /* Update netwm desktop with current desktop */
479 e_hints_window_desktop_set(bd);
486 if ((ecore_x_netwm_startup_id_get(bd->client.win, &str) && (str)) ||
487 ((bd->client.icccm.client_leader > 0) &&
488 ecore_x_netwm_startup_id_get(bd->client.icccm.client_leader, &str) && (str))
491 if (!strncmp(str, "E_START|", 8))
496 if (id > 0) bd->client.netwm.startup_id = id;
501 /* It's ok not to have fetch flag, should only be set on startup
502 * and not changed. */
503 if (!ecore_x_netwm_pid_get(bd->client.win, &bd->client.netwm.pid))
505 if (bd->client.icccm.client_leader)
507 if (!ecore_x_netwm_pid_get(bd->client.icccm.client_leader, &bd->client.netwm.pid))
508 bd->client.netwm.pid = -1;
511 bd->client.netwm.pid = -1;
514 if (bd->client.netwm.state.sticky)
516 if (!bd->lock_client_sticky)
519 e_hints_window_sticky_set(bd, 0);
521 if (bd->client.netwm.state.shaded)
523 if (!bd->lock_client_shade)
524 e_border_shade(bd, e_hints_window_shade_direction_get(bd));
526 e_hints_window_shaded_set(bd, 0);
528 if ((bd->client.netwm.state.maximized_v) && (bd->client.netwm.state.maximized_h))
530 if (!bd->lock_client_maximize)
532 e_hints_window_size_get(bd);
533 e_border_maximize(bd, e_config->maximize_policy);
536 e_hints_window_maximized_set(bd, 0, 0);
538 else if (bd->client.netwm.state.maximized_h)
540 if (!bd->lock_client_maximize)
542 e_hints_window_size_get(bd);
543 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_HORIZONTAL);
546 e_hints_window_maximized_set(bd, 0, 0);
548 else if (bd->client.netwm.state.maximized_v)
550 if (!bd->lock_client_maximize)
552 e_hints_window_size_get(bd);
553 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_VERTICAL);
556 e_hints_window_maximized_set(bd, 0, 0);
558 if (bd->client.netwm.state.fullscreen)
560 if (!bd->lock_client_fullscreen)
562 e_hints_window_size_get(bd);
563 e_border_fullscreen(bd, e_config->fullscreen_policy);
566 e_hints_window_fullscreen_set(bd, 0);
568 if ((bd->client.icccm.state == ECORE_X_WINDOW_STATE_HINT_ICONIC) &&
569 (bd->client.netwm.state.hidden))
571 if (!bd->lock_client_iconify)
572 e_border_iconify(bd);
574 e_hints_window_visible_set(bd);
576 else if ((bd->parent) && (e_config->transient.iconify) && (bd->parent->iconic))
577 e_border_iconify(bd);
578 /* If a window isn't iconic, and is one the current desk,
580 else if (bd->desk == e_desk_current_get(bd->zone))
584 if (bd->client.e.state.centered)
589 /* Update stacking */
590 e_hints_client_list_set();
591 e_hints_client_stacking_set();
595 e_hints_window_state_set(E_Border *bd)
597 Ecore_X_Window_State state[10];
600 if (bd->client.netwm.state.modal)
601 state[num++] = ECORE_X_WINDOW_STATE_MODAL;
602 if (bd->client.netwm.state.sticky)
603 state[num++] = ECORE_X_WINDOW_STATE_STICKY;
604 if (bd->client.netwm.state.maximized_v)
605 state[num++] = ECORE_X_WINDOW_STATE_MAXIMIZED_VERT;
606 if (bd->client.netwm.state.maximized_h)
607 state[num++] = ECORE_X_WINDOW_STATE_MAXIMIZED_HORZ;
608 if (bd->client.netwm.state.shaded)
609 state[num++] = ECORE_X_WINDOW_STATE_SHADED;
610 if (bd->client.netwm.state.skip_taskbar)
611 state[num++] = ECORE_X_WINDOW_STATE_SKIP_TASKBAR;
612 if (bd->client.netwm.state.skip_pager)
613 state[num++] = ECORE_X_WINDOW_STATE_SKIP_PAGER;
614 if (bd->client.netwm.state.hidden)
615 state[num++] = ECORE_X_WINDOW_STATE_HIDDEN;
616 if (bd->client.netwm.state.fullscreen)
617 state[num++] = ECORE_X_WINDOW_STATE_FULLSCREEN;
619 switch (bd->client.netwm.state.stacking)
621 case E_STACKING_ABOVE:
622 state[num++] = ECORE_X_WINDOW_STATE_ABOVE;
625 case E_STACKING_BELOW:
626 state[num++] = ECORE_X_WINDOW_STATE_BELOW;
629 case E_STACKING_NONE:
633 ecore_x_netwm_window_state_set(bd->client.win, state, num);
637 e_hints_allowed_action_set(E_Border *bd)
639 Ecore_X_Action action[10];
642 if (bd->client.netwm.action.move)
643 action[num++] = ECORE_X_ACTION_MOVE;
644 if (bd->client.netwm.action.resize)
645 action[num++] = ECORE_X_ACTION_RESIZE;
646 if (bd->client.netwm.action.minimize)
647 action[num++] = ECORE_X_ACTION_MINIMIZE;
648 if (bd->client.netwm.action.shade)
649 action[num++] = ECORE_X_ACTION_SHADE;
650 if (bd->client.netwm.action.stick)
651 action[num++] = ECORE_X_ACTION_STICK;
652 if (bd->client.netwm.action.maximized_h)
653 action[num++] = ECORE_X_ACTION_MAXIMIZE_HORZ;
654 if (bd->client.netwm.action.maximized_v)
655 action[num++] = ECORE_X_ACTION_MAXIMIZE_VERT;
656 if (bd->client.netwm.action.fullscreen)
657 action[num++] = ECORE_X_ACTION_FULLSCREEN;
658 if (bd->client.netwm.action.change_desktop)
659 action[num++] = ECORE_X_ACTION_CHANGE_DESKTOP;
660 if (bd->client.netwm.action.close)
661 action[num++] = ECORE_X_ACTION_CLOSE;
663 ecore_x_netwm_allowed_action_set(bd->client.win, action, num);
667 e_hints_window_type_set(E_Border *bd)
669 ecore_x_netwm_window_type_set(bd->client.win, bd->client.netwm.type);
673 e_hints_window_type_get(E_Border *bd)
675 Ecore_X_Window_Type *types = NULL;
678 num = ecore_x_netwm_window_types_get(bd->client.win, &types);
679 if (bd->client.netwm.extra_types)
681 free(bd->client.netwm.extra_types);
682 bd->client.netwm.extra_types = NULL;
683 bd->client.netwm.extra_types_num = 0;
686 bd->client.netwm.type = ECORE_X_WINDOW_TYPE_UNKNOWN;
690 bd->client.netwm.type = types[j];
693 (bd->client.netwm.type == ECORE_X_WINDOW_TYPE_UNKNOWN))
696 bd->client.netwm.type = types[j];
700 bd->client.netwm.extra_types =
701 malloc((num - j) * sizeof(Ecore_X_Window_Type));
702 if (bd->client.netwm.extra_types)
704 for (i = j + 1; i < num; i++)
705 bd->client.netwm.extra_types[i - (j + 1)] = types[i];
706 bd->client.netwm.extra_types_num = num - j;
714 e_hints_window_state_update(E_Border *bd,
715 Ecore_X_Window_State state,
716 Ecore_X_Window_State_Action action)
720 case ECORE_X_WINDOW_STATE_ICONIFIED:
721 if (action != ECORE_X_WINDOW_STATE_ACTION_ADD) return;
722 if (bd->client.icccm.state == ECORE_X_WINDOW_STATE_HINT_ICONIC) return;
723 if (bd->lock_client_iconify) return;
724 e_border_iconify(bd);
727 case ECORE_X_WINDOW_STATE_MODAL:
730 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
731 if (bd->client.netwm.state.modal)
733 bd->client.netwm.state.modal = 0;
734 bd->client.netwm.update.state = 1;
739 case ECORE_X_WINDOW_STATE_ACTION_ADD:
740 if (!bd->client.netwm.state.modal)
742 bd->client.netwm.state.modal = 1;
743 bd->client.netwm.update.state = 1;
748 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
749 bd->client.netwm.state.modal = !bd->client.netwm.state.modal;
750 bd->client.netwm.update.state = 1;
756 case ECORE_X_WINDOW_STATE_STICKY:
757 if (bd->lock_client_sticky) return;
760 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
761 e_border_unstick(bd);
764 case ECORE_X_WINDOW_STATE_ACTION_ADD:
768 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
770 e_border_unstick(bd);
777 case ECORE_X_WINDOW_STATE_MAXIMIZED_VERT:
778 if (bd->lock_client_maximize) return;
781 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
782 if (bd->maximized & E_MAXIMIZE_VERTICAL)
783 e_border_unmaximize(bd, E_MAXIMIZE_VERTICAL);
786 case ECORE_X_WINDOW_STATE_ACTION_ADD:
787 if (!(bd->maximized & E_MAXIMIZE_VERTICAL))
788 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_VERTICAL);
791 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
792 if (bd->maximized & E_MAXIMIZE_VERTICAL)
793 e_border_unmaximize(bd, E_MAXIMIZE_VERTICAL);
795 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_VERTICAL);
800 case ECORE_X_WINDOW_STATE_MAXIMIZED_HORZ:
801 if (bd->lock_client_maximize) return;
804 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
805 if (bd->maximized & E_MAXIMIZE_HORIZONTAL)
806 e_border_unmaximize(bd, E_MAXIMIZE_HORIZONTAL);
809 case ECORE_X_WINDOW_STATE_ACTION_ADD:
810 if (!(bd->maximized & E_MAXIMIZE_HORIZONTAL))
811 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_HORIZONTAL);
814 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
815 if (bd->maximized & E_MAXIMIZE_HORIZONTAL)
816 e_border_unmaximize(bd, E_MAXIMIZE_HORIZONTAL);
818 e_border_maximize(bd, (e_config->maximize_policy & E_MAXIMIZE_TYPE) | E_MAXIMIZE_HORIZONTAL);
823 case ECORE_X_WINDOW_STATE_SHADED:
824 if (bd->lock_client_shade) return;
827 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
828 e_border_unshade(bd, e_hints_window_shade_direction_get(bd));
831 case ECORE_X_WINDOW_STATE_ACTION_ADD:
832 e_border_shade(bd, e_hints_window_shade_direction_get(bd));
835 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
837 e_border_unshade(bd, e_hints_window_shade_direction_get(bd));
839 e_border_shade(bd, e_hints_window_shade_direction_get(bd));
844 case ECORE_X_WINDOW_STATE_SKIP_TASKBAR:
847 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
848 if (bd->client.netwm.state.skip_taskbar)
850 bd->client.netwm.state.skip_taskbar = 0;
851 bd->client.netwm.update.state = 1;
856 case ECORE_X_WINDOW_STATE_ACTION_ADD:
857 if (!bd->client.netwm.state.skip_taskbar)
859 bd->client.netwm.state.skip_taskbar = 1;
860 bd->client.netwm.update.state = 1;
865 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
866 bd->client.netwm.state.skip_taskbar = !bd->client.netwm.state.skip_taskbar;
867 bd->client.netwm.update.state = 1;
873 case ECORE_X_WINDOW_STATE_SKIP_PAGER:
876 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
877 if (bd->client.netwm.state.skip_pager)
879 bd->client.netwm.state.skip_pager = 0;
880 bd->client.netwm.update.state = 1;
885 case ECORE_X_WINDOW_STATE_ACTION_ADD:
886 if (!bd->client.netwm.state.skip_pager)
888 bd->client.netwm.state.skip_pager = 1;
889 bd->client.netwm.update.state = 1;
894 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
895 bd->client.netwm.state.skip_pager = !bd->client.netwm.state.skip_pager;
896 bd->client.netwm.update.state = 1;
902 case ECORE_X_WINDOW_STATE_HIDDEN:
906 case ECORE_X_WINDOW_STATE_FULLSCREEN:
907 if (bd->lock_client_fullscreen) return;
910 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
911 e_border_unfullscreen(bd);
914 case ECORE_X_WINDOW_STATE_ACTION_ADD:
915 e_border_fullscreen(bd, e_config->fullscreen_policy);
918 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
920 e_border_unfullscreen(bd);
922 e_border_fullscreen(bd, e_config->fullscreen_policy);
927 case ECORE_X_WINDOW_STATE_ABOVE:
928 if (bd->lock_client_stacking) return;
929 /* FIXME: Should this require that BELOW is set to 0 first, or just
933 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
934 e_border_layer_set(bd, 100);
935 e_hints_window_stacking_set(bd, E_STACKING_NONE);
938 case ECORE_X_WINDOW_STATE_ACTION_ADD:
939 e_hints_window_stacking_set(bd, E_STACKING_ABOVE);
940 e_border_layer_set(bd, 150);
943 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
944 if (bd->layer == 150)
946 e_hints_window_stacking_set(bd, E_STACKING_NONE);
947 e_border_layer_set(bd, 100);
951 e_hints_window_stacking_set(bd, E_STACKING_ABOVE);
952 e_border_layer_set(bd, 150);
958 case ECORE_X_WINDOW_STATE_BELOW:
959 if (bd->lock_client_stacking) return;
960 /* FIXME: Should this require that ABOVE is set to 0 first, or just
964 case ECORE_X_WINDOW_STATE_ACTION_REMOVE:
965 e_hints_window_stacking_set(bd, E_STACKING_NONE);
966 e_border_layer_set(bd, 100);
969 case ECORE_X_WINDOW_STATE_ACTION_ADD:
970 e_hints_window_stacking_set(bd, E_STACKING_BELOW);
971 e_border_layer_set(bd, 50);
974 case ECORE_X_WINDOW_STATE_ACTION_TOGGLE:
977 e_hints_window_stacking_set(bd, E_STACKING_NONE);
978 e_border_layer_set(bd, 100);
982 e_hints_window_stacking_set(bd, E_STACKING_BELOW);
983 e_border_layer_set(bd, 50);
989 case ECORE_X_WINDOW_STATE_DEMANDS_ATTENTION:
993 case ECORE_X_WINDOW_STATE_UNKNOWN:
1000 e_hints_window_state_get(E_Border *bd)
1002 unsigned int i, num;
1003 Ecore_X_Window_State *state;
1005 bd->client.netwm.state.modal = 0;
1006 bd->client.netwm.state.sticky = 0;
1007 bd->client.netwm.state.maximized_v = 0;
1008 bd->client.netwm.state.maximized_h = 0;
1009 bd->client.netwm.state.shaded = 0;
1010 bd->client.netwm.state.skip_taskbar = 0;
1011 bd->client.netwm.state.skip_pager = 0;
1012 bd->client.netwm.state.hidden = 0;
1013 bd->client.netwm.state.fullscreen = 0;
1014 bd->client.netwm.state.stacking = 0;
1016 ecore_x_netwm_window_state_get(bd->client.win, &state, &num);
1019 for (i = 0; i < num; i++)
1023 case ECORE_X_WINDOW_STATE_ICONIFIED:
1027 case ECORE_X_WINDOW_STATE_MODAL:
1028 bd->client.netwm.state.modal = 1;
1031 case ECORE_X_WINDOW_STATE_STICKY:
1032 bd->client.netwm.state.sticky = 1;
1035 case ECORE_X_WINDOW_STATE_MAXIMIZED_VERT:
1036 bd->client.netwm.state.maximized_v = 1;
1039 case ECORE_X_WINDOW_STATE_MAXIMIZED_HORZ:
1040 bd->client.netwm.state.maximized_h = 1;
1043 case ECORE_X_WINDOW_STATE_SHADED:
1044 bd->client.netwm.state.shaded = 1;
1047 case ECORE_X_WINDOW_STATE_SKIP_TASKBAR:
1048 bd->client.netwm.state.skip_taskbar = 1;
1051 case ECORE_X_WINDOW_STATE_SKIP_PAGER:
1052 bd->client.netwm.state.skip_pager = 1;
1055 case ECORE_X_WINDOW_STATE_HIDDEN:
1056 bd->client.netwm.state.hidden = 1;
1059 case ECORE_X_WINDOW_STATE_FULLSCREEN:
1060 bd->client.netwm.state.fullscreen = 1;
1063 case ECORE_X_WINDOW_STATE_ABOVE:
1064 bd->client.netwm.state.stacking = E_STACKING_ABOVE;
1067 case ECORE_X_WINDOW_STATE_BELOW:
1068 bd->client.netwm.state.stacking = E_STACKING_BELOW;
1071 case ECORE_X_WINDOW_STATE_DEMANDS_ATTENTION:
1075 case ECORE_X_WINDOW_STATE_UNKNOWN:
1085 e_hints_allowed_action_update(E_Border *bd __UNUSED__,
1086 Ecore_X_Action action)
1090 case ECORE_X_ACTION_MOVE:
1093 case ECORE_X_ACTION_RESIZE:
1096 case ECORE_X_ACTION_MINIMIZE:
1099 case ECORE_X_ACTION_SHADE:
1102 case ECORE_X_ACTION_STICK:
1105 case ECORE_X_ACTION_MAXIMIZE_HORZ:
1108 case ECORE_X_ACTION_MAXIMIZE_VERT:
1111 case ECORE_X_ACTION_FULLSCREEN:
1114 case ECORE_X_ACTION_CHANGE_DESKTOP:
1117 case ECORE_X_ACTION_CLOSE:
1120 case ECORE_X_ACTION_ABOVE:
1123 case ECORE_X_ACTION_BELOW:
1129 e_hints_allowed_action_get(E_Border *bd)
1131 Ecore_X_Action *action;
1135 bd->client.netwm.action.move = 0;
1136 bd->client.netwm.action.resize = 0;
1137 bd->client.netwm.action.minimize = 0;
1138 bd->client.netwm.action.shade = 0;
1139 bd->client.netwm.action.stick = 0;
1140 bd->client.netwm.action.maximized_h = 0;
1141 bd->client.netwm.action.maximized_v = 0;
1142 bd->client.netwm.action.fullscreen = 0;
1143 bd->client.netwm.action.change_desktop = 0;
1144 bd->client.netwm.action.close = 0;
1146 ecore_x_netwm_allowed_action_get(bd->client.win, &action, &num);
1149 for (i = 0; i < num; i++)
1153 case ECORE_X_ACTION_MOVE:
1154 bd->client.netwm.action.move = 1;
1157 case ECORE_X_ACTION_RESIZE:
1158 bd->client.netwm.action.resize = 1;
1161 case ECORE_X_ACTION_MINIMIZE:
1162 bd->client.netwm.action.minimize = 1;
1165 case ECORE_X_ACTION_SHADE:
1166 bd->client.netwm.action.shade = 1;
1169 case ECORE_X_ACTION_STICK:
1170 bd->client.netwm.action.stick = 1;
1173 case ECORE_X_ACTION_MAXIMIZE_HORZ:
1174 bd->client.netwm.action.maximized_h = 1;
1177 case ECORE_X_ACTION_MAXIMIZE_VERT:
1178 bd->client.netwm.action.maximized_v = 1;
1181 case ECORE_X_ACTION_FULLSCREEN:
1182 bd->client.netwm.action.fullscreen = 1;
1185 case ECORE_X_ACTION_CHANGE_DESKTOP:
1186 bd->client.netwm.action.change_desktop = 1;
1189 case ECORE_X_ACTION_CLOSE:
1190 bd->client.netwm.action.close = 1;
1193 case ECORE_X_ACTION_ABOVE:
1196 case ECORE_X_ACTION_BELOW:
1205 e_hints_window_visible_set(E_Border *bd)
1207 if (bd->client.icccm.state != ECORE_X_WINDOW_STATE_HINT_NORMAL)
1209 ecore_x_icccm_state_set(bd->client.win, ECORE_X_WINDOW_STATE_HINT_NORMAL);
1210 bd->client.icccm.state = ECORE_X_WINDOW_STATE_HINT_NORMAL;
1212 if (bd->client.netwm.state.hidden)
1214 bd->client.netwm.update.state = 1;
1215 bd->client.netwm.state.hidden = 0;
1221 e_hints_window_iconic_set(E_Border *bd)
1223 if (bd->client.icccm.state != ECORE_X_WINDOW_STATE_HINT_ICONIC)
1225 ecore_x_icccm_state_set(bd->client.win, ECORE_X_WINDOW_STATE_HINT_ICONIC);
1226 bd->client.icccm.state = ECORE_X_WINDOW_STATE_HINT_ICONIC;
1228 if (!bd->client.netwm.state.hidden)
1230 bd->client.netwm.update.state = 1;
1231 bd->client.netwm.state.hidden = 1;
1237 e_hints_window_hidden_set(E_Border *bd)
1239 if (bd->client.icccm.state != ECORE_X_WINDOW_STATE_HINT_WITHDRAWN)
1241 ecore_x_icccm_state_set(bd->client.win, ECORE_X_WINDOW_STATE_HINT_WITHDRAWN);
1242 bd->client.icccm.state = ECORE_X_WINDOW_STATE_HINT_WITHDRAWN;
1244 if (bd->client.netwm.state.hidden)
1246 bd->client.netwm.update.state = 1;
1247 bd->client.netwm.state.hidden = 0;
1253 e_hints_window_shaded_set(E_Border *bd,
1256 if ((!bd->client.netwm.state.shaded) && (on))
1258 bd->client.netwm.update.state = 1;
1259 bd->client.netwm.state.shaded = 1;
1262 else if ((bd->client.netwm.state.shaded) && (!on))
1264 bd->client.netwm.update.state = 1;
1265 bd->client.netwm.state.shaded = 0;
1271 e_hints_window_shade_direction_set(E_Border *bd,
1274 ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_SHADE_DIRECTION, &dir, 1);
1278 e_hints_window_shade_direction_get(E_Border *bd)
1283 ret = ecore_x_window_prop_card32_get(bd->client.win,
1284 E_ATOM_SHADE_DIRECTION,
1289 return E_DIRECTION_UP;
1293 e_hints_window_size_set(E_Border *bd)
1295 unsigned int sizes[4];
1301 ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_BORDER_SIZE, sizes, 4);
1305 e_hints_window_size_unset(E_Border *bd)
1307 ecore_x_window_prop_property_del(bd->client.win, E_ATOM_BORDER_SIZE);
1311 e_hints_window_size_get(E_Border *bd)
1314 unsigned int sizes[4];
1316 memset(sizes, 0, sizeof(sizes));
1317 ret = ecore_x_window_prop_card32_get(bd->client.win, E_ATOM_BORDER_SIZE,
1331 e_hints_window_maximized_set(E_Border *bd,
1335 if ((horizontal) && (!bd->client.netwm.state.maximized_h))
1337 bd->client.netwm.update.state = 1;
1338 bd->client.netwm.state.maximized_h = 1;
1341 else if ((!horizontal) && (bd->client.netwm.state.maximized_h))
1343 bd->client.netwm.update.state = 1;
1344 bd->client.netwm.state.maximized_h = 0;
1347 if ((vertical) && (!bd->client.netwm.state.maximized_v))
1349 bd->client.netwm.update.state = 1;
1350 bd->client.netwm.state.maximized_v = 1;
1353 else if ((!vertical) && (bd->client.netwm.state.maximized_v))
1355 bd->client.netwm.update.state = 1;
1356 bd->client.netwm.state.maximized_v = 0;
1362 e_hints_window_fullscreen_set(E_Border *bd,
1365 if ((!bd->client.netwm.state.fullscreen) && (on))
1367 bd->client.netwm.update.state = 1;
1368 bd->client.netwm.state.fullscreen = 1;
1371 else if ((bd->client.netwm.state.fullscreen) && (!on))
1373 bd->client.netwm.update.state = 1;
1374 bd->client.netwm.state.fullscreen = 0;
1380 e_hints_window_sticky_set(E_Border *bd,
1383 if ((!bd->client.netwm.state.sticky) && (on))
1385 bd->client.netwm.update.state = 1;
1386 bd->client.netwm.state.sticky = 1;
1389 else if ((bd->client.netwm.state.sticky) && (!on))
1391 bd->client.netwm.update.state = 1;
1392 bd->client.netwm.state.sticky = 0;
1398 e_hints_window_stacking_set(E_Border *bd,
1399 E_Stacking stacking)
1401 if (bd->client.netwm.state.stacking == stacking) return;
1402 bd->client.netwm.update.state = 1;
1403 bd->client.netwm.state.stacking = stacking;
1408 e_hints_window_desktop_set(E_Border *bd)
1410 /* This function is only called when really changing desktop,
1411 * so just set the property and don't care about the roundtrip.
1413 unsigned int deskpos[2];
1415 /* if valgrind complains here it is complaining bd->client.netwm.desktop
1416 * is an uninitialised variable - but it isn't. it can't be. its part of
1417 * a calloc()'d struct and thus has to have been set to 0. hell even
1418 * e_border.c explicitly sets it to 0 on creation of the border object.
1420 deskpos[0] = bd->desk->x;
1421 deskpos[1] = bd->desk->y;
1422 ecore_x_window_prop_card32_set(bd->client.win, E_ATOM_DESK, deskpos, 2);
1424 if (strcmp(bd->desk->window_profile,
1425 e_config->desktop_default_window_profile) != 0)
1427 ecore_x_e_window_profile_set(bd->client.win,
1428 bd->desk->window_profile);
1432 ecore_x_netwm_desktop_set(bd->client.win, current);
1434 bd->client.netwm.desktop = (bd->desk->y * bd->zone->desk_x_count) + bd->desk->x;
1438 e_hints_window_e_state_get(E_Border *bd)
1440 /* Remember to update the count if we add more states! */
1441 Ecore_X_Atom state[1];
1445 memset(state, 0, sizeof(state));
1447 /* ugly, but avoids possible future overflow if more states are added */
1448 size = (sizeof(state) / sizeof(state[0]));
1451 ecore_x_window_prop_card32_get(bd->client.win, E_ATOM_WINDOW_STATE,
1455 for (i = 0; (i < num) && (i < size); i++)
1457 if (state[i] == E_ATOM_WINDOW_STATE_CENTERED)
1458 bd->client.e.state.centered = 1;
1463 e_hints_window_e_state_set(E_Border *bd __UNUSED__)
1469 e_hints_window_qtopia_soft_menu_get(E_Border *bd)
1473 if (ecore_x_window_prop_card32_get(bd->client.win, ATM__QTOPIA_SOFT_MENU, &val, 1))
1474 bd->client.qtopia.soft_menu = val;
1476 bd->client.qtopia.soft_menu = 0;
1480 e_hints_window_qtopia_soft_menus_get(E_Border *bd)
1484 if (ecore_x_window_prop_card32_get(bd->client.win, ATM__QTOPIA_SOFT_MENUS, &val, 1))
1485 bd->client.qtopia.soft_menus = val;
1487 bd->client.qtopia.soft_menus = 0;
1491 e_hints_window_virtual_keyboard_state_get(E_Border *bd)
1493 bd->client.vkbd.state = ecore_x_e_virtual_keyboard_state_get(bd->client.win);
1497 e_hints_window_virtual_keyboard_get(E_Border *bd)
1499 bd->client.vkbd.vkbd = ecore_x_e_virtual_keyboard_get(bd->client.win);
1503 e_hints_openoffice_gnome_fake(Ecore_X_Window root)
1505 const char *string = "ATM_GNOME_SM_PROXY";
1507 ecore_x_window_prop_property_set(root, ATM_GNOME_SM_PROXY, ECORE_X_ATOM_STRING,
1508 8, (void *)string, strlen(string));
1512 e_hints_openoffice_kde_fake(Ecore_X_Window root)
1514 Ecore_X_Window win2;
1516 win2 = ecore_x_window_new(root, -20, -20, 1, 1);
1517 ecore_x_netwm_wm_identify(root, win2, "KWin");
1521 e_hints_scale_update(void)
1523 Ecore_X_Window *roots = NULL;
1527 roots = ecore_x_window_root_list(&num);
1530 scale = e_scale * 1000;
1531 for (i = 0; i < num; i++)
1532 ecore_x_window_prop_card32_set(roots[i], ATM_ENLIGHTENMENT_SCALE, &scale, 1);