openbox/src/client.cc

1309 lines
36 KiB
C++
Raw Normal View History

2002-11-06 07:01:34 +00:00
// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
#ifdef HAVE_CONFIG_H
# include "../config.h"
#endif
2002-11-06 07:01:34 +00:00
#include "client.hh"
2002-12-04 00:26:45 +00:00
#include "frame.hh"
2002-12-04 01:04:31 +00:00
#include "screen.hh"
2002-11-06 07:01:34 +00:00
#include "openbox.hh"
#include "otk/display.hh"
#include "otk/property.hh"
extern "C" {
#include <X11/Xlib.h>
#include <X11/Xutil.h>
2003-01-16 08:44:52 +00:00
#include <X11/Xatom.h>
2002-11-06 07:01:34 +00:00
#include <assert.h>
#include "gettext.h"
#define _(str) gettext(str)
}
namespace ob {
2003-01-11 19:42:43 +00:00
Client::Client(int screen, Window window)
: otk::EventHandler(),
WidgetBase(WidgetBase::Type_Client),
frame(0), _screen(screen), _window(window)
2002-11-06 07:01:34 +00:00
{
assert(screen >= 0);
2002-11-06 07:01:34 +00:00
assert(window);
ignore_unmaps = 0;
2002-11-06 07:01:34 +00:00
// update EVERYTHING the first time!!
// the state is kinda assumed to be normal. is this right? XXX
2003-01-03 18:21:28 +00:00
_wmstate = NormalState; _iconic = false;
2002-11-06 11:31:50 +00:00
// no default decors or functions, each has to be enabled
_decorations = _functions = 0;
// start unfocused
_focused = false;
2003-01-05 01:40:38 +00:00
// not a transient by default of course
_transient_for = 0;
// pick a layer to start from
_layer = Layer_Normal;
getArea();
getDesktop();
2003-01-05 01:40:38 +00:00
updateTransientFor();
getType();
getMwmHints();
setupDecorAndFunctions();
getState();
getShaped();
2002-11-06 10:05:56 +00:00
updateProtocols();
getGravity(); // get the attribute gravity
updateNormalHints(); // this may override the attribute gravity
updateWMHints();
updateTitle();
2002-11-07 08:22:27 +00:00
updateIconTitle();
updateClass();
updateStrut();
changeState();
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
Client::~Client()
2002-11-06 07:01:34 +00:00
{
2003-01-12 18:29:17 +00:00
// clean up childrens' references
while (!_transients.empty()) {
_transients.front()->_transient_for = 0;
_transients.pop_front();
}
// clean up parents reference to this
if (_transient_for)
_transient_for->_transients.remove(this); // remove from old parent
if (openbox->state() != Openbox::State_Exiting) {
2003-01-03 23:57:00 +00:00
// these values should not be persisted across a window unmapping/mapping
2003-01-16 08:44:52 +00:00
otk::Property::erase(_window, otk::Property::atoms.net_wm_desktop);
otk::Property::erase(_window, otk::Property::atoms.net_wm_state);
2003-01-03 23:57:00 +00:00
}
}
void Client::getGravity()
{
XWindowAttributes wattrib;
Status ret;
ret = XGetWindowAttributes(**otk::display, _window, &wattrib);
assert(ret != BadWindow);
_gravity = wattrib.win_gravity;
}
2003-01-17 02:52:26 +00:00
2003-01-11 19:42:43 +00:00
void Client::getDesktop()
{
// defaults to the current desktop
_desktop = openbox->screen(_screen)->desktop();
2003-01-16 08:44:52 +00:00
if (!otk::Property::get(_window, otk::Property::atoms.net_wm_desktop,
otk::Property::atoms.cardinal,
(long unsigned*)&_desktop)) {
2003-01-07 01:54:26 +00:00
// make sure the hint exists
2003-01-16 08:44:52 +00:00
otk::Property::set(_window, otk::Property::atoms.net_wm_desktop,
otk::Property::atoms.cardinal, (unsigned)_desktop);
2003-01-07 01:54:26 +00:00
}
}
2003-01-11 19:42:43 +00:00
void Client::getType()
{
_type = (WindowType) -1;
unsigned long *val;
unsigned long num = (unsigned) -1;
2003-01-16 08:44:52 +00:00
if (otk::Property::get(_window, otk::Property::atoms.net_wm_window_type,
otk::Property::atoms.atom, &num, &val)) {
// use the first value that we know about in the array
for (unsigned long i = 0; i < num; ++i) {
2003-01-16 08:44:52 +00:00
if (val[i] == otk::Property::atoms.net_wm_window_type_desktop)
_type = Type_Desktop;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_dock)
_type = Type_Dock;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_toolbar)
_type = Type_Toolbar;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_menu)
_type = Type_Menu;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_utility)
_type = Type_Utility;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_splash)
_type = Type_Splash;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_dialog)
_type = Type_Dialog;
2003-01-16 08:44:52 +00:00
else if (val[i] == otk::Property::atoms.net_wm_window_type_normal)
_type = Type_Normal;
2003-01-16 08:44:52 +00:00
// XXX: make this work again
// else if (val[i] == otk::Property::atoms.kde_net_wm_window_type_override)
// mwm_decorations = 0; // prevent this window from getting any decor
2003-01-06 04:44:00 +00:00
if (_type != (WindowType) -1)
break; // grab the first known type
}
delete val;
}
if (_type == (WindowType) -1) {
/*
* the window type hint was not set, which means we either classify ourself
* as a normal window or a dialog, depending on if we are a transient.
*/
2003-01-05 01:40:38 +00:00
if (_transient_for)
_type = Type_Dialog;
else
_type = Type_Normal;
}
2002-11-06 11:31:50 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::setupDecorAndFunctions()
2002-11-06 11:31:50 +00:00
{
// start with everything
_decorations = Decor_Titlebar | Decor_Handle | Decor_Border |
Decor_Iconify | Decor_Maximize;
_functions = Func_Resize | Func_Move | Func_Iconify | Func_Maximize;
switch (_type) {
case Type_Normal:
// normal windows retain all of the possible decorations and
// functionality
2002-11-06 11:31:50 +00:00
case Type_Dialog:
// dialogs cannot be maximized
_decorations &= ~Decor_Maximize;
_functions &= ~Func_Maximize;
break;
2002-11-06 11:31:50 +00:00
case Type_Menu:
case Type_Toolbar:
case Type_Utility:
// these windows get less functionality
_decorations &= ~(Decor_Iconify | Decor_Handle);
_functions &= ~(Func_Iconify | Func_Resize);
break;
case Type_Desktop:
case Type_Dock:
case Type_Splash:
// none of these windows are manipulated by the window manager
_decorations = 0;
_functions = 0;
break;
2002-11-06 11:31:50 +00:00
}
// Mwm Hints are applied subtractively to what has already been chosen for
// decor and functionality
if (_mwmhints.flags & MwmFlag_Decorations) {
if (! (_mwmhints.decorations & MwmDecor_All)) {
if (! (_mwmhints.decorations & MwmDecor_Border))
2002-11-06 11:31:50 +00:00
_decorations &= ~Decor_Border;
if (! (_mwmhints.decorations & MwmDecor_Handle))
2002-11-06 11:31:50 +00:00
_decorations &= ~Decor_Handle;
if (! (_mwmhints.decorations & MwmDecor_Title))
2002-11-06 11:31:50 +00:00
_decorations &= ~Decor_Titlebar;
if (! (_mwmhints.decorations & MwmDecor_Iconify))
2002-11-06 11:31:50 +00:00
_decorations &= ~Decor_Iconify;
if (! (_mwmhints.decorations & MwmDecor_Maximize))
2002-11-06 11:31:50 +00:00
_decorations &= ~Decor_Maximize;
}
}
if (_mwmhints.flags & MwmFlag_Functions) {
if (! (_mwmhints.functions & MwmFunc_All)) {
if (! (_mwmhints.functions & MwmFunc_Resize))
_functions &= ~Func_Resize;
if (! (_mwmhints.functions & MwmFunc_Move))
_functions &= ~Func_Move;
if (! (_mwmhints.functions & MwmFunc_Iconify))
_functions &= ~Func_Iconify;
if (! (_mwmhints.functions & MwmFunc_Maximize))
_functions &= ~Func_Maximize;
// dont let mwm hints kill the close button
//if (! (_mwmhints.functions & MwmFunc_Close))
// _functions &= ~Func_Close;
2002-11-06 11:31:50 +00:00
}
}
changeAllowedActions();
}
2003-01-11 19:42:43 +00:00
void Client::getMwmHints()
{
unsigned long num = MwmHints::elements;
unsigned long *hints;
_mwmhints.flags = 0; // default to none
2003-01-16 08:44:52 +00:00
if (!otk::Property::get(_window, otk::Property::atoms.motif_wm_hints,
otk::Property::atoms.motif_wm_hints, &num,
(unsigned long **)&hints))
return;
if (num >= MwmHints::elements) {
// retrieved the hints
_mwmhints.flags = hints[0];
_mwmhints.functions = hints[1];
_mwmhints.decorations = hints[2];
}
2002-11-06 11:31:50 +00:00
delete [] hints;
}
2003-01-11 19:42:43 +00:00
void Client::getArea()
{
XWindowAttributes wattrib;
Status ret;
2003-01-13 09:04:57 +00:00
ret = XGetWindowAttributes(**otk::display, _window, &wattrib);
assert(ret != BadWindow);
_area.setRect(wattrib.x, wattrib.y, wattrib.width, wattrib.height);
2002-11-06 10:05:56 +00:00
_border_width = wattrib.border_width;
}
2003-01-11 19:42:43 +00:00
void Client::getState()
{
2003-01-03 18:21:28 +00:00
_modal = _shaded = _max_horz = _max_vert = _fullscreen = _above = _below =
2003-01-03 23:51:06 +00:00
_skip_taskbar = _skip_pager = false;
unsigned long *state;
unsigned long num = (unsigned) -1;
2003-01-16 08:44:52 +00:00
if (otk::Property::get(_window, otk::Property::atoms.net_wm_state,
otk::Property::atoms.atom, &num, &state)) {
for (unsigned long i = 0; i < num; ++i) {
2003-01-16 08:44:52 +00:00
if (state[i] == otk::Property::atoms.net_wm_state_modal)
_modal = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_shaded) {
_shaded = true;
_wmstate = IconicState;
2003-01-16 08:44:52 +00:00
} else if (state[i] == otk::Property::atoms.net_wm_state_skip_taskbar)
2003-01-03 23:51:06 +00:00
_skip_taskbar = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_skip_pager)
2003-01-03 23:51:06 +00:00
_skip_pager = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_fullscreen)
_fullscreen = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_maximized_vert)
_max_vert = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_maximized_horz)
_max_horz = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_above)
2003-01-03 18:21:28 +00:00
_above = true;
2003-01-16 08:44:52 +00:00
else if (state[i] == otk::Property::atoms.net_wm_state_below)
2003-01-03 18:21:28 +00:00
_below = true;
}
delete [] state;
}
}
2003-01-11 19:42:43 +00:00
void Client::getShaped()
{
_shaped = false;
#ifdef SHAPE
2003-01-13 09:04:57 +00:00
if (otk::display->shape()) {
int foo;
unsigned int ufoo;
2002-11-10 05:54:12 +00:00
int s;
2003-01-13 09:04:57 +00:00
XShapeSelectInput(**otk::display, _window, ShapeNotifyMask);
2002-11-10 05:54:12 +00:00
2003-01-13 09:04:57 +00:00
XShapeQueryExtents(**otk::display, _window, &s, &foo,
&foo, &ufoo, &ufoo, &foo, &foo, &foo, &ufoo, &ufoo);
2002-11-10 05:54:12 +00:00
_shaped = (s != 0);
}
#endif // SHAPE
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::calcLayer() {
StackLayer l;
if (_iconic) l = Layer_Icon;
else if (_fullscreen) l = Layer_Fullscreen;
else if (_type == Type_Desktop) l = Layer_Desktop;
else if (_type == Type_Dock) {
if (!_below) l = Layer_Top;
else l = Layer_Normal;
}
else if (_above) l = Layer_Above;
else if (_below) l = Layer_Below;
else l = Layer_Normal;
if (l != _layer) {
_layer = l;
if (frame) {
/*
if we don't have a frame, then we aren't mapped yet (and this would
SIGSEGV :)
*/
openbox->screen(_screen)->raiseWindow(this);
}
}
2003-01-03 18:21:28 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::updateProtocols()
2002-11-06 11:31:50 +00:00
{
2002-11-06 10:05:56 +00:00
Atom *proto;
int num_return = 0;
_focus_notify = false;
2002-11-07 08:22:27 +00:00
_decorations &= ~Decor_Close;
_functions &= ~Func_Close;
2002-11-06 10:05:56 +00:00
2003-01-13 09:04:57 +00:00
if (XGetWMProtocols(**otk::display, _window, &proto, &num_return)) {
2002-11-06 10:05:56 +00:00
for (int i = 0; i < num_return; ++i) {
2003-01-16 08:44:52 +00:00
if (proto[i] == otk::Property::atoms.wm_delete_window) {
_decorations |= Decor_Close;
_functions |= Func_Close;
if (frame)
frame->adjustSize(); // update the decorations
2003-01-16 08:44:52 +00:00
} else if (proto[i] == otk::Property::atoms.wm_take_focus)
2002-11-06 10:05:56 +00:00
// if this protocol is requested, then the window will be notified
// by the window manager whenever it receives focus
_focus_notify = true;
}
XFree(proto);
}
}
2003-01-11 19:42:43 +00:00
void Client::updateNormalHints()
2002-11-06 07:01:34 +00:00
{
XSizeHints size;
long ret;
int oldgravity = _gravity;
2002-11-06 07:01:34 +00:00
// defaults
2002-12-04 00:26:45 +00:00
_size_inc.setPoint(1, 1);
_base_size.setPoint(0, 0);
_min_size.setPoint(0, 0);
_max_size.setPoint(INT_MAX, INT_MAX);
2002-11-06 07:01:34 +00:00
2002-11-06 11:31:50 +00:00
// XXX: might want to cancel any interactive resizing of the window at this
// point..
2002-11-06 07:01:34 +00:00
// get the hints from the window
2003-01-13 09:04:57 +00:00
if (XGetWMNormalHints(**otk::display, _window, &size, &ret)) {
_positioned = (size.flags & (PPosition|USPosition));
if (size.flags & PWinGravity) {
2002-11-06 07:01:34 +00:00
_gravity = size.win_gravity;
// if the client has a frame, i.e. has already been mapped and is
// changing its gravity
if (frame && _gravity != oldgravity) {
// move our idea of the client's position based on its new gravity
int x, y;
frame->frameGravity(x, y);
_area.setPos(x, y);
}
}
2002-12-04 00:26:45 +00:00
if (size.flags & PMinSize)
_min_size.setPoint(size.min_width, size.min_height);
2002-12-04 00:26:45 +00:00
if (size.flags & PMaxSize)
_max_size.setPoint(size.max_width, size.max_height);
2002-12-04 00:26:45 +00:00
if (size.flags & PBaseSize)
_base_size.setPoint(size.base_width, size.base_height);
2002-12-04 00:26:45 +00:00
if (size.flags & PResizeInc)
_size_inc.setPoint(size.width_inc, size.height_inc);
2002-11-06 07:01:34 +00:00
}
}
2003-01-11 19:42:43 +00:00
void Client::updateWMHints()
2002-11-06 07:01:34 +00:00
{
XWMHints *hints;
// assume a window takes input if it doesnt specify
_can_focus = true;
_urgent = false;
2003-01-13 09:04:57 +00:00
if ((hints = XGetWMHints(**otk::display, _window)) != NULL) {
2002-11-06 07:01:34 +00:00
if (hints->flags & InputHint)
_can_focus = hints->input;
2002-11-06 07:01:34 +00:00
if (hints->flags & XUrgencyHint)
_urgent = true;
if (hints->flags & WindowGroupHint) {
2002-11-06 07:01:34 +00:00
if (hints->window_group != _group) {
// XXX: remove from the old group if there was one
_group = hints->window_group;
// XXX: do stuff with the group
}
} else // no group!
_group = None;
2002-11-06 07:01:34 +00:00
XFree(hints);
}
}
2003-01-11 19:42:43 +00:00
void Client::updateTitle()
2002-11-06 07:01:34 +00:00
{
_title = "";
// try netwm
2003-01-16 08:44:52 +00:00
if (!otk::Property::get(_window, otk::Property::atoms.net_wm_name,
otk::Property::utf8, &_title)) {
2002-11-06 07:01:34 +00:00
// try old x stuff
2003-01-16 08:44:52 +00:00
otk::Property::get(_window, otk::Property::atoms.wm_name,
otk::Property::ascii, &_title);
2002-11-06 07:01:34 +00:00
}
if (_title.empty())
_title = _("Unnamed Window");
if (frame)
frame->setTitle(_title);
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::updateIconTitle()
2002-11-07 08:22:27 +00:00
{
_icon_title = "";
// try netwm
2003-01-16 08:44:52 +00:00
if (!otk::Property::get(_window, otk::Property::atoms.net_wm_icon_name,
otk::Property::utf8, &_icon_title)) {
2002-11-07 08:22:27 +00:00
// try old x stuff
2003-01-16 08:44:52 +00:00
otk::Property::get(_window, otk::Property::atoms.wm_icon_name,
otk::Property::ascii, &_icon_title);
2002-11-07 08:22:27 +00:00
}
if (_title.empty())
_icon_title = _("Unnamed Window");
}
2003-01-11 19:42:43 +00:00
void Client::updateClass()
2002-11-06 07:01:34 +00:00
{
// set the defaults
2003-01-04 19:09:52 +00:00
_app_name = _app_class = _role = "";
2002-11-06 07:01:34 +00:00
2003-01-11 19:42:43 +00:00
otk::Property::StringVect v;
2002-11-06 07:01:34 +00:00
unsigned long num = 2;
2003-01-16 08:44:52 +00:00
if (otk::Property::get(_window, otk::Property::atoms.wm_class,
otk::Property::ascii, &num, &v)) {
2003-01-13 05:54:40 +00:00
if (num > 0) _app_name = v[0].c_str();
if (num > 1) _app_class = v[1].c_str();
2003-01-04 19:09:52 +00:00
}
2002-11-06 07:01:34 +00:00
2003-01-04 19:09:52 +00:00
v.clear();
num = 1;
2003-01-16 08:44:52 +00:00
if (otk::Property::get(_window, otk::Property::atoms.wm_window_role,
otk::Property::ascii, &num, &v)) {
2003-01-13 05:54:40 +00:00
if (num > 0) _role = v[0].c_str();
2003-01-04 19:09:52 +00:00
}
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::updateStrut()
{
unsigned long num = 4;
unsigned long *data;
2003-01-16 08:44:52 +00:00
if (!otk::Property::get(_window, otk::Property::atoms.net_wm_strut,
otk::Property::atoms.cardinal, &num, &data))
return;
if (num == 4) {
_strut.left = data[0];
_strut.right = data[1];
_strut.top = data[2];
_strut.bottom = data[3];
openbox->screen(_screen)->updateStrut();
}
delete [] data;
}
2003-01-11 19:42:43 +00:00
void Client::updateTransientFor()
2003-01-05 01:40:38 +00:00
{
Window t = 0;
2003-01-11 19:42:43 +00:00
Client *c = 0;
2003-01-05 01:40:38 +00:00
2003-01-13 09:04:57 +00:00
if (XGetTransientForHint(**otk::display, _window, &t) &&
2003-01-05 01:40:38 +00:00
t != _window) { // cant be transient to itself!
c = openbox->findClient(t);
2003-01-05 01:40:38 +00:00
assert(c != this); // if this happens then we need to check for it
if (!c /*XXX: && _group*/) {
// not transient to a client, see if it is transient for a group
if (//t == _group->leader() ||
t == None ||
2003-01-13 09:04:57 +00:00
t == otk::display->screenInfo(_screen)->rootWindow()) {
2003-01-05 01:40:38 +00:00
// window is a transient for its group!
// XXX: for now this is treated as non-transient.
// this needs to be fixed!
}
}
}
// if anything has changed...
if (c != _transient_for) {
if (_transient_for)
_transient_for->_transients.remove(this); // remove from old parent
_transient_for = c;
if (_transient_for)
_transient_for->_transients.push_back(this); // add to new parent
// XXX: change decor status?
}
}
2003-01-11 19:42:43 +00:00
void Client::propertyHandler(const XPropertyEvent &e)
2002-11-06 07:01:34 +00:00
{
2003-01-11 19:42:43 +00:00
otk::EventHandler::propertyHandler(e);
2002-12-02 22:32:38 +00:00
// compress changes to a single property into a single change
XEvent ce;
2003-01-13 09:04:57 +00:00
while (XCheckTypedEvent(**otk::display, e.type, &ce)) {
2002-12-02 22:32:38 +00:00
// XXX: it would be nice to compress ALL changes to a property, not just
// changes in a row without other props between.
if (ce.xproperty.atom != e.atom) {
2003-01-13 09:04:57 +00:00
XPutBackEvent(**otk::display, &ce);
2002-12-02 22:32:38 +00:00
break;
}
}
2002-11-06 07:01:34 +00:00
if (e.atom == XA_WM_NORMAL_HINTS)
updateNormalHints();
else if (e.atom == XA_WM_HINTS)
updateWMHints();
else if (e.atom == XA_WM_TRANSIENT_FOR) {
2003-01-05 01:40:38 +00:00
updateTransientFor();
getType();
calcLayer(); // type may have changed, so update the layer
setupDecorAndFunctions();
frame->adjustSize(); // this updates the frame for any new decor settings
}
2003-01-16 08:44:52 +00:00
else if (e.atom == otk::Property::atoms.net_wm_name ||
e.atom == otk::Property::atoms.wm_name)
2002-11-06 07:01:34 +00:00
updateTitle();
2003-01-16 08:44:52 +00:00
else if (e.atom == otk::Property::atoms.net_wm_icon_name ||
e.atom == otk::Property::atoms.wm_icon_name)
2002-11-07 08:22:27 +00:00
updateIconTitle();
2003-01-16 08:44:52 +00:00
else if (e.atom == otk::Property::atoms.wm_class)
2002-11-06 07:01:34 +00:00
updateClass();
2003-01-16 08:44:52 +00:00
else if (e.atom == otk::Property::atoms.wm_protocols)
2002-11-06 10:05:56 +00:00
updateProtocols();
2003-01-16 08:44:52 +00:00
else if (e.atom == otk::Property::atoms.net_wm_strut)
updateStrut();
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::setWMState(long state)
2002-11-06 07:01:34 +00:00
{
if (state == _wmstate) return; // no change
2002-11-06 07:01:34 +00:00
2003-01-05 01:40:38 +00:00
_wmstate = state;
switch (_wmstate) {
2002-11-06 07:01:34 +00:00
case IconicState:
// XXX: cause it to iconify
break;
case NormalState:
// XXX: cause it to uniconify
break;
}
}
2003-01-11 19:42:43 +00:00
void Client::setDesktop(long target)
2002-11-06 07:01:34 +00:00
{
if (target == _desktop) return;
2003-01-04 02:03:30 +00:00
printf("Setting desktop %ld\n", target);
2003-01-07 01:54:26 +00:00
if (!(target >= 0 || target == (signed)0xffffffff)) return;
2002-11-06 07:01:34 +00:00
_desktop = target;
2003-01-07 01:54:26 +00:00
2003-01-16 08:44:52 +00:00
otk::Property::set(_window, otk::Property::atoms.net_wm_desktop,
otk::Property::atoms.cardinal, (unsigned)_desktop);
2003-01-07 01:54:26 +00:00
// 'move' the window to the new desktop
if (_desktop == openbox->screen(_screen)->desktop() ||
_desktop == (signed)0xffffffff)
frame->show();
else
frame->hide();
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::setState(StateAction action, long data1, long data2)
2002-11-06 07:01:34 +00:00
{
bool shadestate = _shaded;
2002-11-06 07:01:34 +00:00
if (!(action == State_Add || action == State_Remove ||
action == State_Toggle))
return; // an invalid action was passed to the client message, ignore it
for (int i = 0; i < 2; ++i) {
Atom state = i == 0 ? data1 : data2;
if (! state) continue;
// if toggling, then pick whether we're adding or removing
if (action == State_Toggle) {
2003-01-16 08:44:52 +00:00
if (state == otk::Property::atoms.net_wm_state_modal)
2002-11-06 07:01:34 +00:00
action = _modal ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_maximized_vert)
2002-11-06 07:01:34 +00:00
action = _max_vert ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_maximized_horz)
2002-11-06 07:01:34 +00:00
action = _max_horz ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_shaded)
2002-11-06 07:01:34 +00:00
action = _shaded ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_skip_taskbar)
2003-01-03 23:51:06 +00:00
action = _skip_taskbar ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_skip_pager)
2003-01-03 23:51:06 +00:00
action = _skip_pager ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_fullscreen)
2002-11-06 07:01:34 +00:00
action = _fullscreen ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_above)
2003-01-03 18:21:28 +00:00
action = _above ? State_Remove : State_Add;
2003-01-16 08:44:52 +00:00
else if (state == otk::Property::atoms.net_wm_state_below)
2003-01-03 18:21:28 +00:00
action = _below ? State_Remove : State_Add;
2002-11-06 07:01:34 +00:00
}
if (action == State_Add) {
2003-01-16 08:44:52 +00:00
if (state == otk::Property::atoms.net_wm_state_modal) {
2002-11-06 07:01:34 +00:00
if (_modal) continue;
_modal = true;
// XXX: give it focus if another window has focus that shouldnt now
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_maximized_vert) {
2002-11-06 07:01:34 +00:00
if (_max_vert) continue;
_max_vert = true;
// XXX: resize the window etc
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_maximized_horz) {
2002-11-06 07:01:34 +00:00
if (_max_horz) continue;
_max_horz = true;
// XXX: resize the window etc
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_shaded) {
2002-11-06 07:01:34 +00:00
if (_shaded) continue;
2003-01-04 02:49:43 +00:00
// shade when we're all thru here
shadestate = true;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_skip_taskbar) {
2003-01-03 23:51:06 +00:00
_skip_taskbar = true;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_skip_pager) {
2003-01-03 23:51:06 +00:00
_skip_pager = true;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_fullscreen) {
2002-11-06 07:01:34 +00:00
if (_fullscreen) continue;
_fullscreen = true;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_above) {
2003-01-03 18:21:28 +00:00
if (_above) continue;
_above = true;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_below) {
2003-01-03 18:21:28 +00:00
if (_below) continue;
_below = true;
2002-11-06 07:01:34 +00:00
}
} else { // action == State_Remove
2003-01-16 08:44:52 +00:00
if (state == otk::Property::atoms.net_wm_state_modal) {
2002-11-06 07:01:34 +00:00
if (!_modal) continue;
_modal = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_maximized_vert) {
2002-11-06 07:01:34 +00:00
if (!_max_vert) continue;
_max_vert = false;
// XXX: resize the window etc
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_maximized_horz) {
2002-11-06 07:01:34 +00:00
if (!_max_horz) continue;
_max_horz = false;
// XXX: resize the window etc
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_shaded) {
2002-11-06 07:01:34 +00:00
if (!_shaded) continue;
2003-01-04 02:49:43 +00:00
// unshade when we're all thru here
shadestate = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_skip_taskbar) {
2003-01-03 23:51:06 +00:00
_skip_taskbar = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_skip_pager) {
2003-01-03 23:51:06 +00:00
_skip_pager = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_fullscreen) {
2002-11-06 07:01:34 +00:00
if (!_fullscreen) continue;
_fullscreen = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_above) {
2003-01-03 18:21:28 +00:00
if (!_above) continue;
_above = false;
2003-01-16 08:44:52 +00:00
} else if (state == otk::Property::atoms.net_wm_state_below) {
2003-01-03 18:21:28 +00:00
if (!_below) continue;
_below = false;
2002-11-06 07:01:34 +00:00
}
}
}
2003-01-04 02:49:43 +00:00
if (shadestate != _shaded)
shade(shadestate);
calcLayer();
2002-11-06 07:01:34 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::toggleClientBorder(bool addborder)
{
// adjust our idea of where the client is, based on its border. When the
// border is removed, the client should now be considered to be in a
// different position.
// when re-adding the border to the client, the same operation needs to be
// reversed.
int x = _area.x(), y = _area.y();
switch(_gravity) {
default:
case NorthWestGravity:
case WestGravity:
case SouthWestGravity:
break;
case NorthEastGravity:
case EastGravity:
case SouthEastGravity:
if (addborder) x -= _border_width * 2;
else x += _border_width * 2;
break;
case NorthGravity:
case SouthGravity:
case CenterGravity:
case ForgetGravity:
case StaticGravity:
if (addborder) x -= _border_width;
else x += _border_width;
break;
}
switch(_gravity) {
default:
case NorthWestGravity:
case NorthGravity:
case NorthEastGravity:
break;
case SouthWestGravity:
case SouthGravity:
case SouthEastGravity:
if (addborder) y -= _border_width * 2;
else y += _border_width * 2;
break;
case WestGravity:
case EastGravity:
case CenterGravity:
case ForgetGravity:
case StaticGravity:
if (addborder) y -= _border_width;
else y += _border_width;
break;
}
_area.setPos(x, y);
if (addborder) {
2003-01-13 09:04:57 +00:00
XSetWindowBorderWidth(**otk::display, _window, _border_width);
// move the client so it is back it the right spot _with_ its border!
2003-01-13 09:04:57 +00:00
XMoveWindow(**otk::display, _window, x, y);
} else
2003-01-13 09:04:57 +00:00
XSetWindowBorderWidth(**otk::display, _window, 0);
}
2003-01-11 19:42:43 +00:00
void Client::clientMessageHandler(const XClientMessageEvent &e)
2002-11-06 07:01:34 +00:00
{
2003-01-11 19:42:43 +00:00
otk::EventHandler::clientMessageHandler(e);
2002-11-06 07:01:34 +00:00
if (e.format != 32) return;
2003-01-16 08:44:52 +00:00
if (e.message_type == otk::Property::atoms.wm_change_state) {
// compress changes into a single change
bool compress = false;
XEvent ce;
2003-01-13 09:04:57 +00:00
while (XCheckTypedEvent(**otk::display, e.type, &ce)) {
2002-12-02 22:36:43 +00:00
// XXX: it would be nice to compress ALL messages of a type, not just
// messages in a row without other message types between.
if (ce.xclient.message_type != e.message_type) {
2003-01-13 09:04:57 +00:00
XPutBackEvent(**otk::display, &ce);
2002-12-02 22:36:43 +00:00
break;
}
compress = true;
2002-12-02 22:36:43 +00:00
}
if (compress)
2002-12-02 22:36:43 +00:00
setWMState(ce.xclient.data.l[0]); // use the found event
else
setWMState(e.data.l[0]); // use the original event
2003-01-16 08:44:52 +00:00
} else if (e.message_type == otk::Property::atoms.net_wm_desktop) {
// compress changes into a single change
bool compress = false;
XEvent ce;
2003-01-13 09:04:57 +00:00
while (XCheckTypedEvent(**otk::display, e.type, &ce)) {
2002-12-02 22:36:43 +00:00
// XXX: it would be nice to compress ALL messages of a type, not just
// messages in a row without other message types between.
if (ce.xclient.message_type != e.message_type) {
2003-01-13 09:04:57 +00:00
XPutBackEvent(**otk::display, &ce);
2002-12-02 22:36:43 +00:00
break;
}
compress = true;
2002-12-02 22:36:43 +00:00
}
if (compress)
setDesktop(e.data.l[0]); // use the found event
else
setDesktop(e.data.l[0]); // use the original event
2003-01-16 08:44:52 +00:00
} else if (e.message_type == otk::Property::atoms.net_wm_state) {
// can't compress these
#ifdef DEBUG
printf("net_wm_state %s %ld %ld for 0x%lx\n",
(e.data.l[0] == 0 ? "Remove" : e.data.l[0] == 1 ? "Add" :
e.data.l[0] == 2 ? "Toggle" : "INVALID"),
e.data.l[1], e.data.l[2], _window);
#endif
2002-11-06 07:01:34 +00:00
setState((StateAction)e.data.l[0], e.data.l[1], e.data.l[2]);
2003-01-16 08:44:52 +00:00
} else if (e.message_type == otk::Property::atoms.net_close_window) {
#ifdef DEBUG
printf("net_close_window for 0x%lx\n", _window);
#endif
close();
2003-01-16 08:44:52 +00:00
} else if (e.message_type == otk::Property::atoms.net_active_window) {
#ifdef DEBUG
printf("net_active_window for 0x%lx\n", _window);
#endif
if (_shaded)
shade(false);
// XXX: deiconify
focus();
openbox->screen(_screen)->raiseWindow(this);
}
2002-11-06 07:01:34 +00:00
}
#if defined(SHAPE)
2003-01-11 19:42:43 +00:00
void Client::shapeHandler(const XShapeEvent &e)
2002-11-10 05:54:12 +00:00
{
2003-01-11 19:42:43 +00:00
otk::EventHandler::shapeHandler(e);
if (e.kind == ShapeBounding) {
_shaped = e.shaped;
frame->adjustShape();
}
2002-11-10 05:54:12 +00:00
}
#endif
void Client::resize(Corner anchor, int w, int h)
{
if (!(_functions & Func_Resize)) return;
internal_resize(anchor, w, h);
}
void Client::internal_resize(Corner anchor, int w, int h, int x, int y)
2002-12-04 00:26:45 +00:00
{
w -= _base_size.x();
h -= _base_size.y();
// for interactive resizing. have to move half an increment in each
// direction.
w += _size_inc.x() / 2;
h += _size_inc.y() / 2;
2002-12-04 00:26:45 +00:00
// is the window resizable? if it is not, then don't check its sizes, the
// client can do what it wants and the user can't change it anyhow
if (_min_size.x() <= _max_size.x() && _min_size.y() <= _max_size.y()) {
// smaller than min size or bigger than max size?
if (w < _min_size.x()) w = _min_size.x();
else if (w > _max_size.x()) w = _max_size.x();
if (h < _min_size.y()) h = _min_size.y();
else if (h > _max_size.y()) h = _max_size.y();
}
// keep to the increments
w /= _size_inc.x();
h /= _size_inc.y();
// you cannot resize to nothing
if (w < 1) w = 1;
if (h < 1) h = 1;
2002-12-04 00:26:45 +00:00
// store the logical size
_logical_size.setPoint(w, h);
w *= _size_inc.x();
h *= _size_inc.y();
w += _base_size.x();
h += _base_size.y();
2003-01-04 07:24:40 +00:00
if (x == INT_MIN || y == INT_MIN) {
x = _area.x();
y = _area.y();
switch (anchor) {
case TopLeft:
break;
case TopRight:
x -= w - _area.width();
break;
case BottomLeft:
y -= h - _area.height();
break;
case BottomRight:
x -= w - _area.width();
y -= h - _area.height();
break;
}
2002-12-04 00:26:45 +00:00
}
_area.setSize(w, h);
2003-01-04 07:24:40 +00:00
2003-01-13 09:04:57 +00:00
XResizeWindow(**otk::display, _window, w, h);
2002-12-04 00:26:45 +00:00
// resize the frame to match the request
frame->adjustSize();
internal_move(x, y);
2002-12-04 00:26:45 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::move(int x, int y)
{
if (!(_functions & Func_Move)) return;
internal_move(x, y);
}
void Client::internal_move(int x, int y)
{
2002-12-04 00:26:45 +00:00
_area.setPos(x, y);
2003-01-04 07:24:40 +00:00
2002-12-04 00:26:45 +00:00
// move the frame to be in the requested position
if (frame) { // this can be called while mapping, before frame exists
frame->adjustPosition();
// send synthetic configure notify (we don't need to if we aren't mapped
// yet)
XEvent event;
event.type = ConfigureNotify;
2003-01-13 09:04:57 +00:00
event.xconfigure.display = **otk::display;
event.xconfigure.event = _window;
event.xconfigure.window = _window;
event.xconfigure.x = x;
event.xconfigure.y = y;
event.xconfigure.width = _area.width();
event.xconfigure.height = _area.height();
event.xconfigure.border_width = _border_width;
event.xconfigure.above = frame->window();
event.xconfigure.override_redirect = False;
XSendEvent(event.xconfigure.display, event.xconfigure.window, False,
StructureNotifyMask, &event);
}
}
2002-12-04 00:26:45 +00:00
2003-01-11 19:42:43 +00:00
void Client::close()
{
XEvent ce;
if (!(_functions & Func_Close)) return;
// XXX: itd be cool to do timeouts and shit here for killing the client's
// process off
2003-01-04 07:26:57 +00:00
// like... if the window is around after 5 seconds, then the close button
// turns a nice red, and if this function is called again, the client is
// explicitly killed.
ce.xclient.type = ClientMessage;
2003-01-16 08:44:52 +00:00
ce.xclient.message_type = otk::Property::atoms.wm_protocols;
2003-01-13 09:04:57 +00:00
ce.xclient.display = **otk::display;
ce.xclient.window = _window;
ce.xclient.format = 32;
2003-01-16 08:44:52 +00:00
ce.xclient.data.l[0] = otk::Property::atoms.wm_delete_window;
ce.xclient.data.l[1] = CurrentTime;
ce.xclient.data.l[2] = 0l;
ce.xclient.data.l[3] = 0l;
ce.xclient.data.l[4] = 0l;
2003-01-13 09:04:57 +00:00
XSendEvent(**otk::display, _window, false, NoEventMask, &ce);
}
2003-01-11 19:42:43 +00:00
void Client::changeState()
2003-01-03 23:51:06 +00:00
{
unsigned long state[2];
state[0] = _wmstate;
state[1] = None;
2003-01-16 08:44:52 +00:00
otk::Property::set(_window, otk::Property::atoms.wm_state,
otk::Property::atoms.wm_state, state, 2);
2003-01-03 23:51:06 +00:00
Atom netstate[10];
int num = 0;
if (_modal)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_modal;
2003-01-03 23:51:06 +00:00
if (_shaded)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_shaded;
2003-01-03 23:51:06 +00:00
if (_iconic)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_hidden;
2003-01-03 23:51:06 +00:00
if (_skip_taskbar)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_skip_taskbar;
2003-01-03 23:51:06 +00:00
if (_skip_pager)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_skip_pager;
2003-01-03 23:51:06 +00:00
if (_fullscreen)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_fullscreen;
2003-01-03 23:51:06 +00:00
if (_max_vert)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_maximized_vert;
2003-01-03 23:51:06 +00:00
if (_max_horz)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_maximized_horz;
2003-01-03 23:51:06 +00:00
if (_above)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_above;
2003-01-03 23:51:06 +00:00
if (_below)
2003-01-16 08:44:52 +00:00
netstate[num++] = otk::Property::atoms.net_wm_state_below;
otk::Property::set(_window, otk::Property::atoms.net_wm_state,
otk::Property::atoms.atom, netstate, num);
calcLayer();
2003-01-03 23:51:06 +00:00
}
2003-01-04 08:41:42 +00:00
void Client::changeAllowedActions(void)
{
Atom actions[7];
int num = 0;
actions[num++] = otk::Property::atoms.net_wm_action_shade;
actions[num++] = otk::Property::atoms.net_wm_action_change_desktop;
if (_functions & Func_Close)
actions[num++] = otk::Property::atoms.net_wm_action_close;
if (_functions & Func_Move)
actions[num++] = otk::Property::atoms.net_wm_action_move;
if (_functions & Func_Resize)
actions[num++] = otk::Property::atoms.net_wm_action_resize;
if (_functions & Func_Maximize) {
actions[num++] = otk::Property::atoms.net_wm_action_maximize_horz;
actions[num++] = otk::Property::atoms.net_wm_action_maximize_vert;
}
otk::Property::set(_window, otk::Property::atoms.net_wm_allowed_actions,
otk::Property::atoms.atom, actions, num);
}
2003-01-11 19:42:43 +00:00
void Client::shade(bool shade)
2003-01-03 23:51:06 +00:00
{
if (shade == _shaded) return; // already done
_wmstate = shade ? IconicState : NormalState;
_shaded = shade;
changeState();
frame->adjustSize();
}
2003-01-11 19:42:43 +00:00
bool Client::focus() const
{
// won't try focus if the client doesn't want it, or if the window isn't
// visible on the screen
if (!(frame->isVisible() && (_can_focus || _focus_notify))) return false;
if (_focused) return true;
if (_can_focus)
2003-01-13 09:04:57 +00:00
XSetInputFocus(**otk::display, _window,
RevertToNone, CurrentTime);
if (_focus_notify) {
XEvent ce;
ce.xclient.type = ClientMessage;
2003-01-16 08:44:52 +00:00
ce.xclient.message_type = otk::Property::atoms.wm_protocols;
2003-01-13 09:04:57 +00:00
ce.xclient.display = **otk::display;
ce.xclient.window = _window;
ce.xclient.format = 32;
2003-01-16 08:44:52 +00:00
ce.xclient.data.l[0] = otk::Property::atoms.wm_take_focus;
ce.xclient.data.l[1] = openbox->lastTime();
ce.xclient.data.l[2] = 0l;
ce.xclient.data.l[3] = 0l;
ce.xclient.data.l[4] = 0l;
2003-01-13 09:04:57 +00:00
XSendEvent(**otk::display, _window, False, NoEventMask, &ce);
}
return true;
}
2003-01-11 19:42:43 +00:00
void Client::unfocus() const
2002-12-26 18:15:23 +00:00
{
if (!_focused) return;
assert(openbox->focusedClient() == this);
openbox->setFocusedClient(0);
2002-12-26 18:15:23 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::focusHandler(const XFocusChangeEvent &e)
{
#ifdef DEBUG
// printf("FocusIn for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::focusHandler(e);
frame->focus();
_focused = true;
openbox->setFocusedClient(this);
}
2003-01-11 19:42:43 +00:00
void Client::unfocusHandler(const XFocusChangeEvent &e)
{
#ifdef DEBUG
// printf("FocusOut for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::unfocusHandler(e);
frame->unfocus();
_focused = false;
if (openbox->focusedClient() == this)
openbox->setFocusedClient(0);
}
2003-01-11 19:42:43 +00:00
void Client::configureRequestHandler(const XConfigureRequestEvent &e)
2002-12-04 00:26:45 +00:00
{
#ifdef DEBUG
printf("ConfigureRequest for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::configureRequestHandler(e);
2002-12-04 00:26:45 +00:00
// XXX: if we are iconic (or shaded? (fvwm does that)) ignore the event
if (e.value_mask & CWBorderWidth)
_border_width = e.border_width;
2003-01-03 18:21:28 +00:00
// resize, then move, as specified in the EWMH section 7.7
2002-12-04 00:26:45 +00:00
if (e.value_mask & (CWWidth | CWHeight)) {
int w = (e.value_mask & CWWidth) ? e.width : _area.width();
int h = (e.value_mask & CWHeight) ? e.height : _area.height();
Corner corner;
switch (_gravity) {
case NorthEastGravity:
case EastGravity:
corner = TopRight;
break;
case SouthWestGravity:
case SouthGravity:
corner = BottomLeft;
break;
case SouthEastGravity:
corner = BottomRight;
break;
default: // NorthWest, Static, etc
corner = TopLeft;
}
2003-01-04 07:24:40 +00:00
// if moving AND resizing ...
if (e.value_mask & (CWX | CWY)) {
int x = (e.value_mask & CWX) ? e.x : _area.x();
int y = (e.value_mask & CWY) ? e.y : _area.y();
internal_resize(corner, w, h, x, y);
2003-01-04 07:24:40 +00:00
} else // if JUST resizing...
internal_resize(corner, w, h);
2003-01-04 07:24:40 +00:00
} else if (e.value_mask & (CWX | CWY)) { // if JUST moving...
2002-12-04 00:26:45 +00:00
int x = (e.value_mask & CWX) ? e.x : _area.x();
int y = (e.value_mask & CWY) ? e.y : _area.y();
internal_move(x, y);
2002-12-04 00:26:45 +00:00
}
if (e.value_mask & CWStackMode) {
switch (e.detail) {
case Below:
case BottomIf:
openbox->screen(_screen)->lowerWindow(this);
2002-12-04 00:26:45 +00:00
break;
case Above:
case TopIf:
default:
openbox->screen(_screen)->raiseWindow(this);
2002-12-04 00:26:45 +00:00
break;
}
}
}
2003-01-11 19:42:43 +00:00
void Client::unmapHandler(const XUnmapEvent &e)
2002-12-04 01:04:31 +00:00
{
if (ignore_unmaps) {
2002-12-04 01:04:31 +00:00
#ifdef DEBUG
printf("Ignored UnmapNotify for 0x%lx (event 0x%lx)\n", e.window, e.event);
2002-12-04 01:04:31 +00:00
#endif // DEBUG
2002-12-04 01:29:01 +00:00
ignore_unmaps--;
return;
}
#ifdef DEBUG
printf("UnmapNotify for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::unmapHandler(e);
2002-12-04 01:04:31 +00:00
// this deletes us etc
openbox->screen(_screen)->unmanageWindow(this);
2002-12-04 01:04:31 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::destroyHandler(const XDestroyWindowEvent &e)
2002-12-04 01:04:31 +00:00
{
#ifdef DEBUG
printf("DestroyNotify for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::destroyHandler(e);
2002-12-04 01:04:31 +00:00
// this deletes us etc
openbox->screen(_screen)->unmanageWindow(this);
2002-12-04 01:04:31 +00:00
}
2003-01-11 19:42:43 +00:00
void Client::reparentHandler(const XReparentEvent &e)
2002-12-27 08:52:16 +00:00
{
2002-12-27 10:07:57 +00:00
// this is when the client is first taken captive in the frame
2003-01-06 04:58:07 +00:00
if (e.parent == frame->plate()) return;
2002-12-27 10:07:57 +00:00
2002-12-27 08:52:16 +00:00
#ifdef DEBUG
printf("ReparentNotify for 0x%lx\n", e.window);
#endif // DEBUG
2003-01-11 19:42:43 +00:00
otk::EventHandler::reparentHandler(e);
2002-12-27 08:52:16 +00:00
/*
This event is quite rare and is usually handled in unmapHandler.
However, if the window is unmapped when the reparent event occurs,
the window manager never sees it because an unmap event is not sent
to an already unmapped window.
*/
// we don't want the reparent event, put it back on the stack for the X
// server to deal with after we unmanage the window
XEvent ev;
ev.xreparent = e;
2003-01-13 09:04:57 +00:00
XPutBackEvent(**otk::display, &ev);
2002-12-27 08:52:16 +00:00
// this deletes us etc
openbox->screen(_screen)->unmanageWindow(this);
2002-12-27 08:52:16 +00:00
}
2002-11-06 07:01:34 +00:00
}