2 * client.c - client management
4 * Copyright © 2007 Julien Danjou <julien@danjou.info>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 #include <X11/Xatom.h>
24 #include <X11/Xutil.h>
25 #include <X11/extensions/shape.h>
32 #include "layouts/floating.h"
35 extern Client
*clients
, *sel
, *stack
; /* global client list and stack */
37 /** Attach client stack to clients stacks
41 attachstack(Client
* c
)
47 /** Detach client from stack
51 detachstack(Client
* c
)
55 for(tc
= &stack
; *tc
&& *tc
!= c
; tc
= &(*tc
)->snext
);
59 /** Grab or ungrab buttons when a client is focused
61 * \param focused True if client is focused
62 * \param modkey Mod key mask
63 * \param numlockmask Numlock mask
66 grabbuttons(Client
* c
, Bool focused
, KeySym modkey
, unsigned int numlockmask
)
68 XUngrabButton(c
->display
, AnyButton
, AnyModifier
, c
->win
);
72 XGrabButton(c
->display
, Button1
, modkey
, c
->win
, False
, BUTTONMASK
,
73 GrabModeAsync
, GrabModeSync
, None
, None
);
74 XGrabButton(c
->display
, Button1
, modkey
| LockMask
, c
->win
, False
,
75 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
76 XGrabButton(c
->display
, Button1
, modkey
| numlockmask
, c
->win
, False
,
77 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
78 XGrabButton(c
->display
, Button1
, modkey
| numlockmask
| LockMask
,
79 c
->win
, False
, BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
81 XGrabButton(c
->display
, Button2
, modkey
, c
->win
, False
, BUTTONMASK
,
82 GrabModeAsync
, GrabModeSync
, None
, None
);
83 XGrabButton(c
->display
, Button2
, modkey
| LockMask
, c
->win
, False
,
84 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
85 XGrabButton(c
->display
, Button2
, modkey
| numlockmask
, c
->win
, False
,
86 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
87 XGrabButton(c
->display
, Button2
, modkey
| numlockmask
| LockMask
,
88 c
->win
, False
, BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
90 XGrabButton(c
->display
, Button3
, modkey
, c
->win
, False
, BUTTONMASK
,
91 GrabModeAsync
, GrabModeSync
, None
, None
);
92 XGrabButton(c
->display
, Button3
, modkey
| LockMask
, c
->win
, False
,
93 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
94 XGrabButton(c
->display
, Button3
, modkey
| numlockmask
, c
->win
, False
,
95 BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
96 XGrabButton(c
->display
, Button3
, modkey
| numlockmask
| LockMask
,
97 c
->win
, False
, BUTTONMASK
, GrabModeAsync
, GrabModeSync
, None
, None
);
99 XUngrabButton(c
->display
, AnyButton
, AnyModifier
, DefaultRootWindow(c
->display
));
103 XGrabButton(c
->display
, AnyButton
, AnyModifier
, c
->win
, False
, BUTTONMASK
,
104 GrabModeAsync
, GrabModeSync
, None
, None
);
105 XGrabButton(c
->display
, Button4
, NoSymbol
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
106 GrabModeAsync
, GrabModeSync
, None
, None
);
107 XGrabButton(c
->display
, Button4
, LockMask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
108 GrabModeAsync
, GrabModeSync
, None
, None
);
109 XGrabButton(c
->display
, Button4
, numlockmask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
110 GrabModeAsync
, GrabModeSync
, None
, None
);
111 XGrabButton(c
->display
, Button4
, numlockmask
| LockMask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
112 GrabModeAsync
, GrabModeSync
, None
, None
);
114 XGrabButton(c
->display
, Button5
, NoSymbol
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
115 GrabModeAsync
, GrabModeSync
, None
, None
);
116 XGrabButton(c
->display
, Button5
, LockMask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
117 GrabModeAsync
, GrabModeSync
, None
, None
);
118 XGrabButton(c
->display
, Button5
, numlockmask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
119 GrabModeAsync
, GrabModeSync
, None
, None
);
120 XGrabButton(c
->display
, Button5
, numlockmask
| LockMask
, DefaultRootWindow(c
->display
), False
, BUTTONMASK
,
121 GrabModeAsync
, GrabModeSync
, None
, None
);
126 /** Check if client supports protocol WM_DELETE_WINDOW
127 * \param c the client
128 * \return True if client has WM_DELETE_WINDOW
131 isprotodel(Client
* c
)
137 if(XGetWMProtocols(c
->display
, c
->win
, &protocols
, &n
))
139 for(i
= 0; !ret
&& i
< n
; i
++)
140 if(protocols
[i
] == XInternAtom(c
->display
, "WM_DELETE_WINDOW", False
))
147 /** Set client WM_STATE property
148 * \param c the client
149 * \param state no idea
152 setclientstate(Client
* c
, long state
)
154 long data
[] = { state
, None
};
156 XChangeProperty(c
->display
, c
->win
, XInternAtom(c
->display
, "WM_STATE", False
),
157 XInternAtom(c
->display
, "WM_STATE", False
), 32,
158 PropModeReplace
, (unsigned char *) data
, 2);
161 /** Set client transparency using composite
163 * \param opacity opacity percentage
166 setclienttrans(Client
*c
, double opacity
)
168 unsigned int real_opacity
= 0xffffffff;
170 if(opacity
>= 0 && opacity
<= 100)
172 real_opacity
= ((opacity
/ 100.0) * 0xffffffff);
173 XChangeProperty(c
->display
, c
->win
,
174 XInternAtom(c
->display
, "_NET_WM_WINDOW_OPACITY", False
),
175 XA_CARDINAL
, 32, PropModeReplace
, (unsigned char *) &real_opacity
, 1L);
178 XDeleteProperty(c
->display
, c
->win
, XInternAtom(c
->display
, "_NET_WM_WINDOW_OPACITY", False
));
180 XSync(c
->display
, False
);
183 /** Attach client to the beginning of the clients stack
184 * \param c the client
196 updatetitle(Client
* c
)
198 if(!xgettextprop(c
->display
, c
->win
, XInternAtom(c
->display
, "_NET_WM_NAME", False
), c
->name
, sizeof c
->name
))
199 xgettextprop(c
->display
, c
->win
, XInternAtom(c
->display
, "WM_NAME", False
), c
->name
, sizeof c
->name
);
203 * \param c the client
210 XUnmapWindow(c
->display
, c
->win
);
211 setclientstate(c
, IconicState
);
217 * \param c the client
220 configure(Client
* c
)
224 ce
.type
= ConfigureNotify
;
225 ce
.display
= c
->display
;
232 ce
.border_width
= c
->border
;
234 ce
.override_redirect
= False
;
235 XSendEvent(c
->display
, c
->win
, False
, StructureNotifyMask
, (XEvent
*) & ce
);
242 c
->prev
->next
= c
->next
;
244 c
->next
->prev
= c
->prev
;
247 c
->next
= c
->prev
= NULL
;
250 /** Give focus to client, or to first client if c is NULL
251 * \param disp Display ref
252 * \param drawcontext drawcontext ref
254 * \param selscreen True if current screen is selected
255 * \param awesomeconf awesome config
258 focus(Display
*disp
, DC
*drawcontext
, Client
* c
, Bool selscreen
, awesome_config
*awesomeconf
)
260 /* if c is NULL or invisible, take next client in the stack */
261 if((!c
&& selscreen
) || (c
&& !isvisible(c
, awesomeconf
->selected_tags
, awesomeconf
->ntags
)))
262 for(c
= stack
; c
&& !isvisible(c
, awesomeconf
->selected_tags
, awesomeconf
->ntags
); c
= c
->snext
);
264 /* if a client was selected but it's not the current client, unfocus it */
267 grabbuttons(sel
, False
, awesomeconf
->modkey
, awesomeconf
->numlockmask
);
268 XSetWindowBorder(sel
->display
, sel
->win
, drawcontext
->norm
[ColBorder
]);
269 setclienttrans(sel
, awesomeconf
->opacity_unfocused
);
275 grabbuttons(c
, True
, awesomeconf
->modkey
, awesomeconf
->numlockmask
);
280 drawstatus(disp
, drawcontext
, awesomeconf
);
283 XSetWindowBorder(sel
->display
, sel
->win
, drawcontext
->sel
[ColBorder
]);
284 XSetInputFocus(sel
->display
, sel
->win
, RevertToPointerRoot
, CurrentTime
);
285 for(c
= stack
; c
; c
= c
->snext
)
287 setclienttrans(c
, awesomeconf
->opacity_unfocused
);
288 setclienttrans(sel
, -1);
291 XSetInputFocus(disp
, DefaultRootWindow(disp
), RevertToPointerRoot
, CurrentTime
);
294 /** Kill selected client
295 * \param disp Display ref
296 * \param awesomeconf awesome config
298 * \ingroup ui_callback
301 uicb_killclient(Display
*disp
__attribute__ ((unused
)),
302 DC
*drawcontext
__attribute__ ((unused
)),
303 awesome_config
*awesomeconf
__attribute__ ((unused
)),
304 const char *arg
__attribute__ ((unused
)))
312 ev
.type
= ClientMessage
;
313 ev
.xclient
.window
= sel
->win
;
314 ev
.xclient
.message_type
= XInternAtom(disp
, "WM_PROTOCOLS", False
);
315 ev
.xclient
.format
= 32;
316 ev
.xclient
.data
.l
[0] = XInternAtom(disp
, "WM_DELETE_WINDOW", False
);
317 ev
.xclient
.data
.l
[1] = CurrentTime
;
318 XSendEvent(sel
->display
, sel
->win
, False
, NoEventMask
, &ev
);
321 XKillClient(sel
->display
, sel
->win
);
325 loadprops(Client
* c
, int ntags
)
331 prop
= p_new(char, ntags
+ 2);
333 if(xgettextprop(c
->display
, c
->win
, AWESOMEPROPS_ATOM(c
->display
), prop
, ntags
+ 2))
335 for(i
= 0; i
< ntags
&& prop
[i
]; i
++)
336 if((c
->tags
[i
] = prop
[i
] == '1'))
338 if(i
<= ntags
&& prop
[i
])
339 c
->isfloating
= prop
[i
] == '1';
348 manage(Display
* disp
, DC
*drawcontext
, Window w
, XWindowAttributes
* wa
, awesome_config
*awesomeconf
)
351 Client
*c
, *t
= NULL
;
356 c
= p_new(Client
, 1);
357 c
->tags
= p_new(Bool
, awesomeconf
->ntags
);
360 c
->x
= c
->rw
= wa
->x
;
361 c
->y
= c
->ry
= wa
->y
;
362 c
->w
= c
->rw
= wa
->width
;
363 c
->h
= c
->rh
= wa
->height
;
364 c
->oldborder
= wa
->border_width
;
366 if(c
->w
== DisplayWidth(disp
, DefaultScreen(disp
))
367 && c
->h
== DisplayHeight(disp
, DefaultScreen(disp
)))
371 c
->border
= wa
->border_width
;
375 ScreenInfo
*si
= get_display_info(disp
, awesomeconf
->statusbar
);
377 if(c
->x
+ c
->w
+ 2 * c
->border
> si
->x_org
+ si
->width
)
378 c
->x
= c
->rx
= si
->x_org
+ si
->width
- c
->w
- 2 * c
->border
;
379 if(c
->y
+ c
->h
+ 2 * c
->border
> si
->y_org
+ si
->height
)
380 c
->y
= c
->ry
= si
->y_org
+ si
->height
- c
->h
- 2 * c
->border
;
382 c
->x
= c
->rx
= si
->x_org
;
384 c
->y
= c
->ry
= si
->y_org
;
385 c
->border
= awesomeconf
->borderpx
;
388 wc
.border_width
= c
->border
;
389 XConfigureWindow(disp
, w
, CWBorderWidth
, &wc
);
390 XSetWindowBorder(disp
, w
, drawcontext
->norm
[ColBorder
]);
391 configure(c
); /* propagates border_width, if size doesn't change */
393 XSelectInput(disp
, w
, StructureNotifyMask
| PropertyChangeMask
| EnterWindowMask
);
394 if(awesomeconf
->have_shape
)
396 XShapeSelectInput(disp
, w
, ShapeNotifyMask
);
399 grabbuttons(c
, False
, awesomeconf
->modkey
, awesomeconf
->numlockmask
);
401 if((rettrans
= XGetTransientForHint(disp
, w
, &trans
) == Success
))
402 for(t
= clients
; t
&& t
->win
!= trans
; t
= t
->next
);
404 for(i
= 0; i
< awesomeconf
->ntags
; i
++)
405 c
->tags
[i
] = t
->tags
[i
];
406 if(!loadprops(c
, awesomeconf
->ntags
))
407 applyrules(c
, awesomeconf
);
409 c
->isfloating
= (rettrans
== Success
) || c
->isfixed
;
410 saveprops(c
, awesomeconf
->ntags
);
413 XMoveResizeWindow(disp
, c
->win
, c
->x
, c
->y
, c
->w
, c
->h
); /* some windows require this */
415 arrange(disp
, drawcontext
, awesomeconf
);
419 resize(Client
* c
, int x
, int y
, int w
, int h
, Bool sizehints
)
421 double dx
, dy
, max
, min
, ratio
;
426 if(c
->minay
> 0 && c
->maxay
> 0 && (h
- c
->baseh
) > 0 && (w
- c
->basew
) > 0)
428 dx
= (double) (w
- c
->basew
);
429 dy
= (double) (h
- c
->baseh
);
430 min
= (double) (c
->minax
) / (double) (c
->minay
);
431 max
= (double) (c
->maxax
) / (double) (c
->maxay
);
433 if(max
> 0 && min
> 0 && ratio
> 0)
437 dy
= (dx
* min
+ dy
) / (min
* min
+ 1);
439 w
= (int) dx
+ c
->basew
;
440 h
= (int) dy
+ c
->baseh
;
444 dy
= (dx
* min
+ dy
) / (max
* max
+ 1);
446 w
= (int) dx
+ c
->basew
;
447 h
= (int) dy
+ c
->baseh
;
451 if(c
->minw
&& w
< c
->minw
)
453 if(c
->minh
&& h
< c
->minh
)
455 if(c
->maxw
&& w
> c
->maxw
)
457 if(c
->maxh
&& h
> c
->maxh
)
460 w
-= (w
- c
->basew
) % c
->incw
;
462 h
-= (h
- c
->baseh
) % c
->inch
;
466 /* offscreen appearance fixes */
467 if(x
> DisplayWidth(c
->display
, DefaultScreen(c
->display
)))
468 x
= DisplayWidth(c
->display
, DefaultScreen(c
->display
)) - w
- 2 * c
->border
;
469 if(y
> DisplayHeight(c
->display
, DefaultScreen(c
->display
)))
470 y
= DisplayHeight(c
->display
, DefaultScreen(c
->display
)) - h
- 2 * c
->border
;
471 if(x
+ w
+ 2 * c
->border
< 0)
473 if(y
+ h
+ 2 * c
->border
< 0)
475 if(c
->x
!= x
|| c
->y
!= y
|| c
->w
!= w
|| c
->h
!= h
)
480 c
->h
= wc
.height
= h
;
481 wc
.border_width
= c
->border
;
482 XConfigureWindow(c
->display
, c
->win
, CWX
| CWY
| CWWidth
| CWHeight
| CWBorderWidth
, &wc
);
484 XSync(c
->display
, False
);
489 uicb_moveresize(Display
*disp
__attribute__ ((unused
)),
490 DC
*drawcontext
__attribute__ ((unused
)),
491 awesome_config
*awesomeconf
,
494 int x
, y
, w
, h
, nx
, ny
, nw
, nh
, ox
, oy
, ow
, oh
;
495 char xabs
, yabs
, wabs
, habs
;
496 int mx
, my
, dx
, dy
, nmx
, nmy
;
500 if(!IS_ARRANGE(floating
))
501 if(!sel
|| !sel
->isfloating
|| sel
->isfixed
|| !arg
)
503 if(sscanf(arg
, "%d%c %d%c %d%c %d%c", &x
, &xabs
, &y
, &yabs
, &w
, &wabs
, &h
, &habs
) != 8)
505 nx
= xabs
== 'x' ? sel
->x
+ x
: x
;
506 ny
= yabs
== 'y' ? sel
->y
+ y
: y
;
507 nw
= wabs
== 'w' ? sel
->w
+ w
: w
;
508 nh
= habs
== 'h' ? sel
->h
+ h
: h
;
515 Bool xqp
= XQueryPointer(sel
->display
, DefaultRootWindow(sel
->display
), &dummy
, &dummy
, &mx
, &my
, &dx
, &dy
, &dui
);
516 resize(sel
, nx
, ny
, nw
, nh
, True
);
517 if (xqp
&& ox
<= mx
&& (ox
+ ow
) >= mx
&& oy
<= my
&& (oy
+ oh
) >= my
)
519 nmx
= mx
-ox
+sel
->w
-ow
-1 < 0 ? 0 : mx
-ox
+sel
->w
-ow
-1;
520 nmy
= my
-oy
+sel
->h
-oh
-1 < 0 ? 0 : my
-oy
+sel
->h
-oh
-1;
521 XWarpPointer(sel
->display
, None
, sel
->win
, 0, 0, 0, 0, nmx
, nmy
);
526 saveprops(Client
* c
, int ntags
)
531 prop
= p_new(char, ntags
+ 2);
533 for(i
= 0; i
< ntags
; i
++)
534 prop
[i
] = c
->tags
[i
] ? '1' : '0';
537 prop
[i
] = c
->isfloating
? '1' : '0';
541 XChangeProperty(c
->display
, c
->win
, AWESOMEPROPS_ATOM(c
->display
), XA_STRING
, 8,
542 PropModeReplace
, (unsigned char *) prop
, i
);
552 XMapWindow(c
->display
, c
->win
);
553 setclientstate(c
, NormalState
);
558 unmanage(Client
* c
, DC
*drawcontext
, long state
, awesome_config
*awesomeconf
)
562 wc
.border_width
= c
->oldborder
;
563 /* The server grab construct avoids race conditions. */
564 XGrabServer(c
->display
);
565 XConfigureWindow(c
->display
, c
->win
, CWBorderWidth
, &wc
); /* restore border */
569 focus(c
->display
, drawcontext
, NULL
, True
, awesomeconf
);
570 XUngrabButton(c
->display
, AnyButton
, AnyModifier
, c
->win
);
571 setclientstate(c
, state
);
572 XSync(c
->display
, False
);
573 XSetErrorHandler(xerror
);
574 XUngrabServer(c
->display
);
575 if(state
!= NormalState
)
576 arrange(c
->display
, drawcontext
, awesomeconf
);
582 updatesizehints(Client
* c
)
587 if(!XGetWMNormalHints(c
->display
, c
->win
, &size
, &msize
) || !size
.flags
)
589 c
->flags
= size
.flags
;
590 if(c
->flags
& PBaseSize
)
592 c
->basew
= size
.base_width
;
593 c
->baseh
= size
.base_height
;
595 else if(c
->flags
& PMinSize
)
597 c
->basew
= size
.min_width
;
598 c
->baseh
= size
.min_height
;
601 c
->basew
= c
->baseh
= 0;
602 if(c
->flags
& PResizeInc
)
604 c
->incw
= size
.width_inc
;
605 c
->inch
= size
.height_inc
;
608 c
->incw
= c
->inch
= 0;
610 if(c
->flags
& PMaxSize
)
612 c
->maxw
= size
.max_width
;
613 c
->maxh
= size
.max_height
;
616 c
->maxw
= c
->maxh
= 0;
618 if(c
->flags
& PMinSize
)
620 c
->minw
= size
.min_width
;
621 c
->minh
= size
.min_height
;
623 else if(c
->flags
& PBaseSize
)
625 c
->minw
= size
.base_width
;
626 c
->minh
= size
.base_height
;
629 c
->minw
= c
->minh
= 0;
631 if(c
->flags
& PAspect
)
633 c
->minax
= size
.min_aspect
.x
;
634 c
->maxax
= size
.max_aspect
.x
;
635 c
->minay
= size
.min_aspect
.y
;
636 c
->maxay
= size
.max_aspect
.y
;
639 c
->minax
= c
->maxax
= c
->minay
= c
->maxay
= 0;
641 c
->isfixed
= (c
->maxw
&& c
->minw
&& c
->maxh
&& c
->minh
642 && c
->maxw
== c
->minw
&& c
->maxh
== c
->minh
);
649 int i
, b
; unsigned int u
; /* dummies */
650 /* Logic to decide if we have a shaped window cribbed from fvwm-2.5.10. */
651 if (XShapeQueryExtents(c
->display
, c
->win
, &bounding_shaped
, &i
, &i
,
652 &u
, &u
, &b
, &i
, &i
, &u
, &u
) && bounding_shaped
)
653 XShapeCombineShape(c
->display
, DefaultRootWindow(c
->display
), ShapeBounding
, 0, 0, c
->win
, ShapeBounding
, ShapeSet
);
657 uicb_settrans(Display
*disp
__attribute__ ((unused
)),
658 DC
*drawcontext
__attribute__ ((unused
)),
659 awesome_config
*awesomeconf
__attribute__ ((unused
)),
662 double delta
= 100.0, current_opacity
= 0.0;
666 unsigned long n
, left
;
667 unsigned int current_opacity_raw
= 0;
673 XGetWindowProperty(sel
->display
, sel
->win
, XInternAtom(sel
->display
, "_NET_WM_WINDOW_OPACITY", False
),
674 0L, 1L, False
, XA_CARDINAL
, &actual
, &format
, &n
, &left
,
675 (unsigned char **) &data
);
678 memcpy(¤t_opacity_raw
, data
, sizeof(unsigned int));
680 current_opacity
= (current_opacity_raw
* 100.0) / 0xffffffff;
685 delta
= compute_new_value_from_arg(arg
, current_opacity
);
689 else if(delta
> 100.0)
695 if(delta
== 100.0 && !set_prop
)
696 setclienttrans(sel
, -1);
698 setclienttrans(sel
, delta
);