/[rdesktop]/sourceforge.net/branches/seamlessrdp-branch/rdesktop/ewmhints.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Contents of /sourceforge.net/branches/seamlessrdp-branch/rdesktop/ewmhints.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1137 - (show annotations)
Wed Mar 15 15:10:21 2006 UTC (18 years, 2 months ago) by ossman_
File MIME type: text/plain
File size: 9685 byte(s)
Handle popups without a parent by setting the window manager hints
_NET_WM_STATE_SKIP_TASKBAR and _NET_WM_STATE_SKIP_PAGER. That seems to get
the behaviour we want.

1 /* -*- c-basic-offset: 8 -*-
2 rdesktop: A Remote Desktop Protocol client.
3
4 Support functions for Extended Window Manager Hints,
5 http://www.freedesktop.org/wiki/Standards_2fwm_2dspec
6
7 Copyright (C) Peter Astrand <astrand@cendio.se> 2005
8
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2 of the License, or
12 (at your option) any later version.
13
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
18
19 You should have received a copy of the GNU General Public License
20 along with this program; if not, write to the Free Software
21 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 */
23
24 #include <X11/Xlib.h>
25 #include <X11/Xatom.h>
26 #include "rdesktop.h"
27
28 #define _NET_WM_STATE_REMOVE 0 /* remove/unset property */
29 #define _NET_WM_STATE_ADD 1 /* add/set property */
30 #define _NET_WM_STATE_TOGGLE 2 /* toggle property */
31
32 extern Display *g_display;
33 static Atom g_net_wm_state_maximized_vert_atom, g_net_wm_state_maximized_horz_atom,
34 g_net_wm_state_hidden_atom, g_net_wm_state_skip_taskbar_atom,
35 g_net_wm_state_skip_pager_atom;
36 Atom g_net_wm_state_atom, g_net_wm_desktop_atom;
37
38 /*
39 Get window property value (32 bit format)
40 Returns zero on success, -1 on error
41 */
42 static int
43 get_property_value(Window wnd, char *propname, long max_length,
44 unsigned long *nitems_return, unsigned char **prop_return)
45 {
46 int result;
47 Atom property;
48 Atom actual_type_return;
49 int actual_format_return;
50 unsigned long bytes_after_return;
51
52 property = XInternAtom(g_display, propname, True);
53 if (property == None)
54 {
55 fprintf(stderr, "Atom %s does not exist\n", propname);
56 return (-1);
57 }
58
59 result = XGetWindowProperty(g_display, wnd, property, 0, /* long_offset */
60 max_length, /* long_length */
61 False, /* delete */
62 AnyPropertyType, /* req_type */
63 &actual_type_return,
64 &actual_format_return,
65 nitems_return, &bytes_after_return, prop_return);
66
67 if (result != Success)
68 {
69 fprintf(stderr, "XGetWindowProperty failed\n");
70 return (-1);
71 }
72
73 if (actual_type_return == None || actual_format_return == 0)
74 {
75 fprintf(stderr, "Window is missing property %s\n", propname);
76 return (-1);
77 }
78
79 if (bytes_after_return)
80 {
81 fprintf(stderr, "%s is too big for me\n", propname);
82 return (-1);
83 }
84
85 if (actual_format_return != 32)
86 {
87 fprintf(stderr, "%s has bad format\n", propname);
88 return (-1);
89 }
90
91 return (0);
92 }
93
94 /*
95 Get current desktop number
96 Returns -1 on error
97 */
98 static int
99 get_current_desktop(void)
100 {
101 unsigned long nitems_return;
102 unsigned char *prop_return;
103 int current_desktop;
104
105 if (get_property_value
106 (DefaultRootWindow(g_display), "_NET_CURRENT_DESKTOP", 1, &nitems_return,
107 &prop_return) < 0)
108 return (-1);
109
110 if (nitems_return != 1)
111 {
112 fprintf(stderr, "_NET_CURRENT_DESKTOP has bad length\n");
113 return (-1);
114 }
115
116 current_desktop = *prop_return;
117 XFree(prop_return);
118 return current_desktop;
119 }
120
121 /*
122 Get workarea geometry
123 Returns zero on success, -1 on error
124 */
125
126 int
127 get_current_workarea(uint32 * x, uint32 * y, uint32 * width, uint32 * height)
128 {
129 int current_desktop;
130 unsigned long nitems_return;
131 unsigned char *prop_return;
132 uint32 *return_words;
133 const uint32 net_workarea_x_offset = 0;
134 const uint32 net_workarea_y_offset = 1;
135 const uint32 net_workarea_width_offset = 2;
136 const uint32 net_workarea_height_offset = 3;
137 const uint32 max_prop_length = 32 * 4; /* Max 32 desktops */
138
139 if (get_property_value
140 (DefaultRootWindow(g_display), "_NET_WORKAREA", max_prop_length, &nitems_return,
141 &prop_return) < 0)
142 return (-1);
143
144 if (nitems_return % 4)
145 {
146 fprintf(stderr, "_NET_WORKAREA has odd length\n");
147 return (-1);
148 }
149
150 current_desktop = get_current_desktop();
151
152 if (current_desktop < 0)
153 return -1;
154
155 return_words = (uint32 *) prop_return;
156
157 *x = return_words[current_desktop * 4 + net_workarea_x_offset];
158 *y = return_words[current_desktop * 4 + net_workarea_y_offset];
159 *width = return_words[current_desktop * 4 + net_workarea_width_offset];
160 *height = return_words[current_desktop * 4 + net_workarea_height_offset];
161
162 XFree(prop_return);
163
164 return (0);
165
166 }
167
168
169
170 void
171 ewmh_init()
172 {
173 g_net_wm_state_maximized_vert_atom =
174 XInternAtom(g_display, "_NET_WM_STATE_MAXIMIZED_VERT", False);
175 g_net_wm_state_maximized_horz_atom =
176 XInternAtom(g_display, "_NET_WM_STATE_MAXIMIZED_HORZ", False);
177 g_net_wm_state_hidden_atom = XInternAtom(g_display, "_NET_WM_STATE_HIDDEN", False);
178 g_net_wm_state_skip_taskbar_atom =
179 XInternAtom(g_display, "_NET_WM_STATE_SKIP_TASKBAR", False);
180 g_net_wm_state_skip_pager_atom = XInternAtom(g_display, "_NET_WM_STATE_SKIP_PAGER", False);
181 g_net_wm_state_atom = XInternAtom(g_display, "_NET_WM_STATE", False);
182 g_net_wm_desktop_atom = XInternAtom(g_display, "_NET_WM_DESKTOP", False);
183 }
184
185
186 /*
187 Get the window state: normal/minimized/maximized.
188 */
189 #ifndef MAKE_PROTO
190 int
191 ewmh_get_window_state(Window w)
192 {
193 unsigned long nitems_return;
194 unsigned char *prop_return;
195 uint32 *return_words;
196 unsigned long item;
197 BOOL maximized_vert, maximized_horz, hidden;
198
199 maximized_vert = maximized_horz = hidden = False;
200
201 if (get_property_value(w, "_NET_WM_STATE", 64, &nitems_return, &prop_return) < 0)
202 return SEAMLESSRDP_NORMAL;
203
204 return_words = (uint32 *) prop_return;
205
206 for (item = 0; item < nitems_return; item++)
207 {
208 if (return_words[item] == g_net_wm_state_maximized_vert_atom)
209 maximized_vert = True;
210 if (return_words[item] == g_net_wm_state_maximized_horz_atom)
211 maximized_horz = True;
212 if (return_words[item] == g_net_wm_state_hidden_atom)
213 hidden = True;
214 }
215
216 XFree(prop_return);
217
218 if (maximized_vert && maximized_horz)
219 return SEAMLESSRDP_MAXIMIZED;
220 else if (hidden)
221 return SEAMLESSRDP_MINIMIZED;
222 else
223 return SEAMLESSRDP_NORMAL;
224 }
225
226 /*
227 Set the window state: normal/minimized/maximized.
228 Returns -1 on failure.
229 */
230 int
231 ewmh_change_state(Window wnd, int state)
232 {
233 Status status;
234 XEvent xevent;
235
236 /*
237 * Deal with the hidden atom
238 */
239 xevent.type = ClientMessage;
240 xevent.xclient.window = wnd;
241 xevent.xclient.message_type = g_net_wm_state_atom;
242 xevent.xclient.format = 32;
243 if (state == SEAMLESSRDP_MINIMIZED)
244 xevent.xclient.data.l[0] = _NET_WM_STATE_ADD;
245 else
246 xevent.xclient.data.l[0] = _NET_WM_STATE_REMOVE;
247 xevent.xclient.data.l[1] = g_net_wm_state_hidden_atom;
248 xevent.xclient.data.l[2] = 0;
249 xevent.xclient.data.l[3] = 0;
250 xevent.xclient.data.l[4] = 0;
251 status = XSendEvent(g_display, DefaultRootWindow(g_display), False,
252 SubstructureNotifyMask | SubstructureRedirectMask, &xevent);
253 if (!status)
254 return -1;
255
256
257 /*
258 * Deal with the max atoms
259 */
260 xevent.type = ClientMessage;
261 xevent.xclient.window = wnd;
262 xevent.xclient.message_type = g_net_wm_state_atom;
263 xevent.xclient.format = 32;
264 if (state == SEAMLESSRDP_MAXIMIZED)
265 xevent.xclient.data.l[0] = _NET_WM_STATE_ADD;
266 else
267 xevent.xclient.data.l[0] = _NET_WM_STATE_REMOVE;
268 xevent.xclient.data.l[1] = g_net_wm_state_maximized_vert_atom;
269 xevent.xclient.data.l[2] = g_net_wm_state_maximized_horz_atom;
270 xevent.xclient.data.l[3] = 0;
271 xevent.xclient.data.l[4] = 0;
272 status = XSendEvent(g_display, DefaultRootWindow(g_display), False,
273 SubstructureNotifyMask | SubstructureRedirectMask, &xevent);
274 if (!status)
275 return -1;
276
277 return 0;
278 }
279
280
281 int
282 ewmh_get_window_desktop(Window wnd)
283 {
284 unsigned long nitems_return;
285 unsigned char *prop_return;
286 int desktop;
287
288 if (get_property_value(wnd, "_NET_WM_DESKTOP", 1, &nitems_return, &prop_return) < 0)
289 return (-1);
290
291 if (nitems_return != 1)
292 {
293 fprintf(stderr, "_NET_WM_DESKTOP has bad length\n");
294 return (-1);
295 }
296
297 desktop = *prop_return;
298 XFree(prop_return);
299 return desktop;
300 }
301
302
303 int
304 ewmh_move_to_desktop(Window wnd, unsigned int desktop)
305 {
306 Status status;
307 XEvent xevent;
308
309 xevent.type = ClientMessage;
310 xevent.xclient.window = wnd;
311 xevent.xclient.message_type = g_net_wm_desktop_atom;
312 xevent.xclient.format = 32;
313 xevent.xclient.data.l[0] = desktop;
314 xevent.xclient.data.l[1] = 0;
315 xevent.xclient.data.l[2] = 0;
316 xevent.xclient.data.l[3] = 0;
317 xevent.xclient.data.l[4] = 0;
318 status = XSendEvent(g_display, DefaultRootWindow(g_display), False,
319 SubstructureNotifyMask | SubstructureRedirectMask, &xevent);
320 if (!status)
321 return -1;
322
323 return 0;
324 }
325
326 void
327 ewmh_set_window_popup(Window wnd)
328 {
329 Atom atoms[2] = { g_net_wm_state_skip_taskbar_atom, g_net_wm_state_skip_pager_atom };
330
331 XChangeProperty(g_display, wnd, g_net_wm_state_atom, XA_ATOM, 32, PropModeAppend, atoms, 2);
332 }
333
334 #endif /* MAKE_PROTO */
335
336
337 #if 0
338
339 /* FIXME: _NET_MOVERESIZE_WINDOW is for pagers, not for
340 applications. We should implement _NET_WM_MOVERESIZE instead */
341
342 int
343 ewmh_net_moveresize_window(Window wnd, int x, int y, int width, int height)
344 {
345 Status status;
346 XEvent xevent;
347 Atom moveresize;
348
349 moveresize = XInternAtom(g_display, "_NET_MOVERESIZE_WINDOW", False);
350 if (!moveresize)
351 {
352 return -1;
353 }
354
355 xevent.type = ClientMessage;
356 xevent.xclient.window = wnd;
357 xevent.xclient.message_type = moveresize;
358 xevent.xclient.format = 32;
359 xevent.xclient.data.l[0] = StaticGravity | (1 << 8) | (1 << 9) | (1 << 10) | (1 << 11);
360 xevent.xclient.data.l[1] = x;
361 xevent.xclient.data.l[2] = y;
362 xevent.xclient.data.l[3] = width;
363 xevent.xclient.data.l[4] = height;
364
365 status = XSendEvent(g_display, DefaultRootWindow(g_display), False,
366 SubstructureNotifyMask | SubstructureRedirectMask, &xevent);
367 if (!status)
368 return -1;
369 return 0;
370 }
371
372 #endif

  ViewVC Help
Powered by ViewVC 1.1.26