when creating an OBClient class, get all the info on that class immediately.
print the info to stdout for now too!
This commit is contained in:
parent
7b343db222
commit
9b23dff16c
2 changed files with 234 additions and 17 deletions
236
src/client.cc
236
src/client.cc
|
@ -23,22 +23,218 @@ OBClient::OBClient(Window window)
|
||||||
{
|
{
|
||||||
assert(window);
|
assert(window);
|
||||||
|
|
||||||
// initialize vars to false/invalid values
|
|
||||||
_group = 0;
|
|
||||||
_gravity = _base_x = _base_y = _inc_x = _inc_y = _max_x = _max_y = _min_x =
|
|
||||||
_min_y = -1;
|
|
||||||
_state = -1;
|
|
||||||
_type = Type_Normal;
|
|
||||||
_desktop = 0xffffffff - 1;
|
|
||||||
_can_focus = _urgent = _focus_notify = _shaped = _modal = _shaded =
|
|
||||||
_max_horz = _max_vert = _fullscreen = _floating = false;
|
|
||||||
|
|
||||||
|
|
||||||
// update EVERYTHING the first time!!
|
// update EVERYTHING the first time!!
|
||||||
|
|
||||||
|
// the state is kinda assumed to be normal. is this right? XXX
|
||||||
|
_wmstate = NormalState;
|
||||||
|
|
||||||
|
getArea();
|
||||||
|
getDesktop();
|
||||||
|
getType();
|
||||||
|
getState();
|
||||||
|
getShaped();
|
||||||
|
|
||||||
|
updateNormalHints();
|
||||||
|
updateWMHints();
|
||||||
|
// XXX: updateTransientFor();
|
||||||
|
updateTitle();
|
||||||
|
updateClass();
|
||||||
|
|
||||||
|
#ifdef DEBUG
|
||||||
|
printf("Mapped window: 0x%lx\n"
|
||||||
|
" title: %s\n"
|
||||||
|
" icon title: %s\n"
|
||||||
|
" app name: %s\n"
|
||||||
|
" class: %s\n"
|
||||||
|
" position: %d, %d\n"
|
||||||
|
" size: %d, %d\n"
|
||||||
|
" desktop: %lu\n"
|
||||||
|
" group: 0x%lx\n"
|
||||||
|
" type: %d\n"
|
||||||
|
" min size %d, %d\n"
|
||||||
|
" base size %d, %d\n"
|
||||||
|
" max size %d, %d\n"
|
||||||
|
" size incr %d, %d\n"
|
||||||
|
" gravity %d\n"
|
||||||
|
" wm state %ld\n"
|
||||||
|
" can be focused: %s\n"
|
||||||
|
" notify focus: %s\n"
|
||||||
|
" urgent: %s\n"
|
||||||
|
" shaped: %s\n"
|
||||||
|
" modal: %s\n"
|
||||||
|
" shaded: %s\n"
|
||||||
|
" iconic: %s\n"
|
||||||
|
" vert maximized: %s\n"
|
||||||
|
" horz maximized: %s\n"
|
||||||
|
" fullscreen: %s\n"
|
||||||
|
" floating: %s\n",
|
||||||
|
_window,
|
||||||
|
_title.c_str(),
|
||||||
|
_icon_title.c_str(),
|
||||||
|
_app_name.c_str(),
|
||||||
|
_app_class.c_str(),
|
||||||
|
_area.x(), _area.y(),
|
||||||
|
_area.width(), _area.height(),
|
||||||
|
_desktop,
|
||||||
|
_group,
|
||||||
|
_type,
|
||||||
|
_min_x, _min_y,
|
||||||
|
_base_x, _base_y,
|
||||||
|
_max_x, _max_y,
|
||||||
|
_inc_x, _inc_y,
|
||||||
|
_gravity,
|
||||||
|
_wmstate,
|
||||||
|
_can_focus ? "yes" : "no",
|
||||||
|
_focus_notify ? "yes" : "no",
|
||||||
|
_urgent ? "yes" : "no",
|
||||||
|
_shaped ? "yes" : "no",
|
||||||
|
_modal ? "yes" : "no",
|
||||||
|
_shaded ? "yes" : "no",
|
||||||
|
_iconic ? "yes" : "no",
|
||||||
|
_max_vert ? "yes" : "no",
|
||||||
|
_max_horz ? "yes" : "no",
|
||||||
|
_fullscreen ? "yes" : "no",
|
||||||
|
_floating ? "yes" : "no");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
OBClient::~OBClient()
|
OBClient::~OBClient()
|
||||||
{
|
{
|
||||||
|
const otk::OBProperty *property = Openbox::instance->property();
|
||||||
|
|
||||||
|
// these values should not be persisted across a window unmapping/mapping
|
||||||
|
property->erase(_window, otk::OBProperty::net_wm_desktop);
|
||||||
|
property->erase(_window, otk::OBProperty::net_wm_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::getDesktop()
|
||||||
|
{
|
||||||
|
const otk::OBProperty *property = Openbox::instance->property();
|
||||||
|
|
||||||
|
// defaults to the current desktop
|
||||||
|
_desktop = 0; // XXX: change this to the current desktop!
|
||||||
|
|
||||||
|
property->get(_window, otk::OBProperty::net_wm_desktop,
|
||||||
|
otk::OBProperty::Atom_Cardinal,
|
||||||
|
&_desktop);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::getType()
|
||||||
|
{
|
||||||
|
const otk::OBProperty *property = Openbox::instance->property();
|
||||||
|
|
||||||
|
_type = (WindowType) -1;
|
||||||
|
|
||||||
|
unsigned long *val;
|
||||||
|
unsigned long num = (unsigned) -1;
|
||||||
|
if (property->get(_window, otk::OBProperty::net_wm_window_type,
|
||||||
|
otk::OBProperty::Atom_Atom,
|
||||||
|
&num, &val)) {
|
||||||
|
// use the first value that we know about in the array
|
||||||
|
for (unsigned long i = 0; i < num; ++i) {
|
||||||
|
if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_desktop))
|
||||||
|
_type = Type_Desktop;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_dock))
|
||||||
|
_type = Type_Dock;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_toolbar))
|
||||||
|
_type = Type_Toolbar;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_menu))
|
||||||
|
_type = Type_Menu;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_utility))
|
||||||
|
_type = Type_Utility;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_splash))
|
||||||
|
_type = Type_Splash;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_dialog))
|
||||||
|
_type = Type_Dialog;
|
||||||
|
else if (val[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_window_type_normal))
|
||||||
|
_type = Type_Normal;
|
||||||
|
// else if (val[i] ==
|
||||||
|
// property->atom(otk::OBProperty::kde_net_wm_window_type_override))
|
||||||
|
// mwm_decorations = 0; // prevent this window from getting any decor
|
||||||
|
// XXX: make this work again
|
||||||
|
}
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
// XXX: make this code work!
|
||||||
|
//if (isTransient())
|
||||||
|
// _type = Type_Dialog;
|
||||||
|
//else
|
||||||
|
_type = Type_Normal;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::getArea()
|
||||||
|
{
|
||||||
|
XWindowAttributes wattrib;
|
||||||
|
assert(XGetWindowAttributes(otk::OBDisplay::display, _window, &wattrib));
|
||||||
|
|
||||||
|
_area.setRect(wattrib.x, wattrib.y, wattrib.width, wattrib.height);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::getState()
|
||||||
|
{
|
||||||
|
const otk::OBProperty *property = Openbox::instance->property();
|
||||||
|
|
||||||
|
_modal = _shaded = _max_horz = _max_vert = _fullscreen = _floating = false;
|
||||||
|
|
||||||
|
unsigned long *state;
|
||||||
|
unsigned long num = (unsigned) -1;
|
||||||
|
|
||||||
|
if (property->get(_window, otk::OBProperty::net_wm_state,
|
||||||
|
otk::OBProperty::Atom_Atom, &num, &state)) {
|
||||||
|
for (unsigned long i = 0; i < num; ++i) {
|
||||||
|
if (state[i] == property->atom(otk::OBProperty::net_wm_state_modal))
|
||||||
|
_modal = true;
|
||||||
|
else if (state[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_state_shaded))
|
||||||
|
_shaded = true;
|
||||||
|
else if (state[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_state_fullscreen))
|
||||||
|
_fullscreen = true;
|
||||||
|
else if (state[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_state_maximized_vert))
|
||||||
|
_max_vert = true;
|
||||||
|
else if (state[i] ==
|
||||||
|
property->atom(otk::OBProperty::net_wm_state_maximized_horz))
|
||||||
|
_max_horz = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete [] state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::getShaped()
|
||||||
|
{
|
||||||
|
_shaped = false;
|
||||||
|
#ifdef SHAPE
|
||||||
|
if (otk::OBDisplay::shape()) {
|
||||||
|
int foo;
|
||||||
|
unsigned int ufoo;
|
||||||
|
|
||||||
|
XShapeQueryExtents(otk::OBDisplay::display, client.window, &_shaped, &foo,
|
||||||
|
&foo, &ufoo, &ufoo, &foo, &foo, &foo, &ufoo, &ufoo);
|
||||||
|
}
|
||||||
|
#endif // SHAPE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -79,14 +275,19 @@ void OBClient::updateWMHints()
|
||||||
if ((hints = XGetWMHints(otk::OBDisplay::display, _window)) != NULL) {
|
if ((hints = XGetWMHints(otk::OBDisplay::display, _window)) != NULL) {
|
||||||
if (hints->flags & InputHint)
|
if (hints->flags & InputHint)
|
||||||
_can_focus = hints->input;
|
_can_focus = hints->input;
|
||||||
|
|
||||||
if (hints->flags & XUrgencyHint)
|
if (hints->flags & XUrgencyHint)
|
||||||
_urgent = true;
|
_urgent = true;
|
||||||
if (hints->flags & WindowGroupHint)
|
|
||||||
|
if (hints->flags & WindowGroupHint) {
|
||||||
if (hints->window_group != _group) {
|
if (hints->window_group != _group) {
|
||||||
// XXX: remove from the old group if there was one
|
// XXX: remove from the old group if there was one
|
||||||
_group = hints->window_group;
|
_group = hints->window_group;
|
||||||
// XXX: do stuff with the group
|
// XXX: do stuff with the group
|
||||||
}
|
}
|
||||||
|
} else // no group!
|
||||||
|
_group = None;
|
||||||
|
|
||||||
XFree(hints);
|
XFree(hints);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -145,12 +346,13 @@ void OBClient::update(const XPropertyEvent &e)
|
||||||
updateTitle();
|
updateTitle();
|
||||||
else if (e.atom == property->atom(otk::OBProperty::wm_class))
|
else if (e.atom == property->atom(otk::OBProperty::wm_class))
|
||||||
updateClass();
|
updateClass();
|
||||||
|
// XXX: transient for hint
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void OBClient::setWMState(long state)
|
void OBClient::setWMState(long state)
|
||||||
{
|
{
|
||||||
if (state == _state) return; // no change
|
if (state == _wmstate) return; // no change
|
||||||
|
|
||||||
switch (state) {
|
switch (state) {
|
||||||
case IconicState:
|
case IconicState:
|
||||||
|
@ -160,7 +362,7 @@ void OBClient::setWMState(long state)
|
||||||
// XXX: cause it to uniconify
|
// XXX: cause it to uniconify
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
_state = state;
|
_wmstate = state;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -288,4 +490,10 @@ void OBClient::update(const XClientMessageEvent &e)
|
||||||
setState((StateAction)e.data.l[0], e.data.l[1], e.data.l[2]);
|
setState((StateAction)e.data.l[0], e.data.l[1], e.data.l[2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void OBClient::setArea(const otk::Rect &area)
|
||||||
|
{
|
||||||
|
_area = area;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -71,7 +71,7 @@ private:
|
||||||
// XXX: transient_for, transients
|
// XXX: transient_for, transients
|
||||||
|
|
||||||
//! The desktop on which the window resides (0xffffffff for all desktops)
|
//! The desktop on which the window resides (0xffffffff for all desktops)
|
||||||
unsigned int _desktop;
|
unsigned long _desktop;
|
||||||
|
|
||||||
//! Normal window title
|
//! Normal window title
|
||||||
std::string _title; // XXX: Have to keep track if this string is Utf8 or not
|
std::string _title; // XXX: Have to keep track if this string is Utf8 or not
|
||||||
|
@ -101,7 +101,7 @@ private:
|
||||||
|
|
||||||
//! The state of the window, one of WithdrawnState, IconicState, or
|
//! The state of the window, one of WithdrawnState, IconicState, or
|
||||||
//! NormalState
|
//! NormalState
|
||||||
long _state;
|
long _wmstate;
|
||||||
|
|
||||||
//! Can the window receive input focus?
|
//! Can the window receive input focus?
|
||||||
bool _can_focus;
|
bool _can_focus;
|
||||||
|
@ -131,12 +131,19 @@ private:
|
||||||
|
|
||||||
// XXX: motif decoration hints!
|
// XXX: motif decoration hints!
|
||||||
|
|
||||||
|
void getDesktop();
|
||||||
|
void getType();
|
||||||
|
void getArea();
|
||||||
|
void getState();
|
||||||
|
void getShaped();
|
||||||
|
|
||||||
void setWMState(long state);
|
void setWMState(long state);
|
||||||
void setDesktop(long desktop);
|
void setDesktop(long desktop);
|
||||||
void setState(StateAction action, long data1, long data2);
|
void setState(StateAction action, long data1, long data2);
|
||||||
|
|
||||||
void updateNormalHints();
|
void updateNormalHints();
|
||||||
void updateWMHints();
|
void updateWMHints();
|
||||||
|
// XXX: updateTransientFor();
|
||||||
void updateTitle();
|
void updateTitle();
|
||||||
void updateClass();
|
void updateClass();
|
||||||
|
|
||||||
|
@ -147,7 +154,7 @@ public:
|
||||||
inline Window window() const { return _window; }
|
inline Window window() const { return _window; }
|
||||||
|
|
||||||
inline WindowType type() const { return _type; }
|
inline WindowType type() const { return _type; }
|
||||||
inline unsigned int desktop() const { return _desktop; }
|
inline unsigned long desktop() const { return _desktop; }
|
||||||
inline const std::string &title() const { return _title; }
|
inline const std::string &title() const { return _title; }
|
||||||
inline const std::string &iconTitle() const { return _title; }
|
inline const std::string &iconTitle() const { return _title; }
|
||||||
inline const std::string &appName() const { return _app_name; }
|
inline const std::string &appName() const { return _app_name; }
|
||||||
|
@ -180,6 +187,8 @@ public:
|
||||||
|
|
||||||
void update(const XPropertyEvent &e);
|
void update(const XPropertyEvent &e);
|
||||||
void update(const XClientMessageEvent &e);
|
void update(const XClientMessageEvent &e);
|
||||||
|
|
||||||
|
void setArea(const otk::Rect &area);
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue