fluxbox/src/WinClient.cc

668 lines
20 KiB
C++
Raw Normal View History

2003-04-14 12:13:36 +00:00
// WinClient.cc for Fluxbox - an X11 Window manager
// Copyright (c) 2003 Henrik Kinnunen (fluxgen(at)users.sourceforge.net)
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
// $Id: WinClient.cc,v 1.33 2003/12/15 20:20:20 rathnor Exp $
2003-04-14 12:13:36 +00:00
#include "WinClient.hh"
#include "Window.hh"
#include "fluxbox.hh"
#include "Screen.hh"
2003-06-12 14:51:59 +00:00
#include "I18n.hh"
2003-04-15 12:12:29 +00:00
#include "FbAtoms.hh"
2003-04-25 11:21:17 +00:00
#include "EventManager.hh"
2003-06-22 12:35:03 +00:00
#include "Xutil.hh"
2003-04-14 12:13:36 +00:00
#include <iostream>
#include <algorithm>
#include <iterator>
2003-04-27 02:26:21 +00:00
#include <cassert>
2003-04-14 12:13:36 +00:00
using namespace std;
WinClient::WinClient(Window win, BScreen &screen, FluxboxWindow *fbwin):FbTk::FbWindow(win),
2003-04-14 12:13:36 +00:00
transient_for(0),
window_group(0),
x(0), y(0), old_bw(0),
min_width(1), min_height(1),
max_width(0), max_height(0),
2003-04-14 12:13:36 +00:00
width_inc(1), height_inc(1),
min_aspect_x(1), min_aspect_y(1),
max_aspect_x(1), max_aspect_y(1),
base_width(1), base_height(1),
initial_state(0),
normal_hint_flags(0),
wm_hint_flags(0),
m_win(fbwin),
m_modal(0),
send_focus_message(false),
2003-09-21 12:49:48 +00:00
send_close_message(false),
2003-09-11 19:55:27 +00:00
m_win_gravity(0),
2003-04-14 12:13:36 +00:00
m_title(""), m_icon_title(""),
m_class_name(""), m_instance_name(""),
m_blackbox_hint(0),
m_mwm_hint(0),
m_focus_mode(F_PASSIVE),
m_diesig(*this), m_screen(screen),
m_strut(0) {
2003-12-15 11:55:58 +00:00
updateWMProtocols();
updateBlackboxHints();
updateMWMHints();
updateWMHints();
updateWMNormalHints();
updateWMClassHint();
updateTitle();
updateIconTitle();
Fluxbox::instance()->saveWindowSearch(win, this);
if (window_group != None)
Fluxbox::instance()->saveGroupSearch(window_group, this);
}
2003-04-14 12:13:36 +00:00
WinClient::~WinClient() {
#ifdef DEBUG
cerr<<__FILE__<<"(~"<<__FUNCTION__<<")[this="<<this<<"]"<<endl;
#endif // DEBUG
FbTk::EventManager::instance()->remove(window());
clearStrut();
if (m_win != 0)
m_win->removeClient(*this);
// this takes care of any focus issues
2003-04-14 12:13:36 +00:00
m_diesig.notify();
Fluxbox *fluxbox = Fluxbox::instance();
if (transient_for != 0) {
assert(transient_for != this);
transient_for->transientList().remove(this);
transient_for = 0;
2003-04-14 12:13:36 +00:00
}
while (!transients.empty()) {
transients.back()->transient_for = 0;
2003-04-14 12:13:36 +00:00
transients.pop_back();
}
2003-06-27 15:05:19 +00:00
screen().removeNetizen(window());
2003-04-14 12:13:36 +00:00
if (window_group != 0) {
fluxbox->removeGroupSearch(window_group);
window_group = 0;
}
if (m_mwm_hint != 0)
XFree(m_mwm_hint);
2003-04-14 12:13:36 +00:00
if (m_blackbox_hint != 0)
XFree(m_blackbox_hint);
2003-04-14 12:13:36 +00:00
if (window())
fluxbox->removeWindowSearch(window());
m_win = 0;
}
void WinClient::updateRect(int x, int y,
unsigned int width, unsigned int height) {
Display *disp = FbTk::App::instance()->display();
XEvent event;
event.type = ConfigureNotify;
event.xconfigure.display = disp;
event.xconfigure.event = window();
event.xconfigure.window = window();
event.xconfigure.x = x;
event.xconfigure.y = y;
event.xconfigure.width = width;
event.xconfigure.height = height;
//!! TODO
event.xconfigure.border_width = 1;//client.old_bw;
//!! TODO
event.xconfigure.above = None; //m_frame.window().window();
event.xconfigure.override_redirect = false;
XSendEvent(disp, window(), False, StructureNotifyMask, &event);
}
2003-07-21 15:26:57 +00:00
bool WinClient::sendFocus() {
if (!send_focus_message)
2003-07-21 15:26:57 +00:00
return false;
2003-04-14 12:13:36 +00:00
Display *disp = FbTk::App::instance()->display();
// setup focus msg
XEvent ce;
ce.xclient.type = ClientMessage;
ce.xclient.message_type = FbAtoms::instance()->getWMProtocolsAtom();
ce.xclient.display = disp;
ce.xclient.window = window();
ce.xclient.format = 32;
ce.xclient.data.l[0] = FbAtoms::instance()->getWMTakeFocusAtom();
ce.xclient.data.l[1] = Fluxbox::instance()->getLastTime();
ce.xclient.data.l[2] = 0l;
ce.xclient.data.l[3] = 0l;
ce.xclient.data.l[4] = 0l;
// send focus msg
XSendEvent(disp, window(), false, NoEventMask, &ce);
2003-07-21 15:26:57 +00:00
return true;
2003-04-14 12:13:36 +00:00
}
void WinClient::sendClose(bool forceful) {
2003-09-21 12:49:48 +00:00
if (forceful || !send_close_message)
XKillClient(FbTk::App::instance()->display(), window());
else {
// send WM_DELETE message
Display *disp = FbTk::App::instance()->display();
// fill in XClientMessage structure for delete message
XEvent ce;
ce.xclient.type = ClientMessage;
ce.xclient.message_type = FbAtoms::instance()->getWMProtocolsAtom();
ce.xclient.display = disp;
ce.xclient.window = window();
ce.xclient.format = 32;
ce.xclient.data.l[0] = FbAtoms::instance()->getWMDeleteAtom();
ce.xclient.data.l[1] = CurrentTime;
ce.xclient.data.l[2] = 0l;
ce.xclient.data.l[3] = 0l;
ce.xclient.data.l[4] = 0l;
// send event delete message to client window
XSendEvent(disp, window(), false, NoEventMask, &ce);
}
2003-04-14 12:13:36 +00:00
}
void WinClient::reparent(Window win, int x, int y) {
XReparentWindow(FbTk::App::instance()->display(), window(), win, x, y);
}
bool WinClient::getAttrib(XWindowAttributes &attr) const {
return XGetWindowAttributes(FbTk::App::instance()->display(), window(), &attr);
}
bool WinClient::getWMName(XTextProperty &textprop) const {
return XGetWMName(FbTk::App::instance()->display(), window(), &textprop);
}
bool WinClient::getWMIconName(XTextProperty &textprop) const {
return XGetWMName(FbTk::App::instance()->display(), window(), &textprop);
}
2003-06-15 18:36:16 +00:00
const std::string &WinClient::getWMClassName() const {
return m_instance_name;
}
const std::string &WinClient::getWMClassClass() const {
return m_class_name;
}
void WinClient::updateWMClassHint() {
2003-06-12 15:13:23 +00:00
XClassHint ch;
if (XGetClassHint(FbTk::App::instance()->display(), window(), &ch) == 0) {
#ifdef DEBUG
2003-06-15 18:36:16 +00:00
cerr<<"WinClient: Failed to read class hint!"<<endl;
2003-06-12 15:13:23 +00:00
#endif //DEBUG
2003-06-15 18:36:16 +00:00
} else {
2003-06-12 15:13:23 +00:00
if (ch.res_name != 0) {
m_instance_name = const_cast<char *>(ch.res_name);
XFree(ch.res_name);
2003-06-15 18:36:16 +00:00
ch.res_name = 0;
2003-06-12 15:13:23 +00:00
} else
m_instance_name = "";
if (ch.res_class != 0) {
m_class_name = const_cast<char *>(ch.res_class);
XFree(ch.res_class);
2003-06-15 18:36:16 +00:00
ch.res_class = 0;
2003-06-12 15:13:23 +00:00
} else
m_class_name = "";
}
}
2003-04-14 12:13:36 +00:00
void WinClient::updateTransientInfo() {
if (m_win == 0)
return;
// remove us from parent
if (transientFor() != 0) {
transientFor()->transientList().remove(this);
2003-04-14 12:13:36 +00:00
}
transient_for = 0;
Display *disp = FbTk::App::instance()->display();
// determine if this is a transient window
Window win;
if (!XGetTransientForHint(disp, window(), &win))
return;
// we can't be transient to ourself
if (win == window())
return;
2003-05-10 22:59:32 +00:00
if (win != None && m_win->screen().rootWindow() == win) {
// transient for root window... = transient for group
// I don't think we are group-aware yet
return;
2003-04-14 12:13:36 +00:00
}
transient_for = Fluxbox::instance()->searchWindow(win);
2003-04-14 12:13:36 +00:00
// make sure we don't have deadlock loop in transient chain
for (WinClient *w = this; w != 0; w = w->transient_for) {
if (w == w->transient_for) {
w->transient_for = 0;
2003-04-14 12:13:36 +00:00
break;
}
}
if (transientFor() != 0) {
// we need to add ourself to the right client in
// the transientFor() window so we search client
transient_for->transientList().push_back(this);
if (transientFor()->fbwindow() && transientFor()->fbwindow()->isStuck())
m_win->stick();
2003-04-14 12:13:36 +00:00
}
}
void WinClient::updateTitle() {
2003-06-22 12:35:03 +00:00
m_title = Xutil::getWMName(window());
2003-04-14 12:13:36 +00:00
}
void WinClient::updateIconTitle() {
XTextProperty text_prop;
char **list = 0;
int num = 0;
2003-04-14 12:13:36 +00:00
if (getWMIconName(text_prop)) {
if (text_prop.value && text_prop.nitems > 0) {
if (text_prop.encoding != XA_STRING) {
text_prop.nitems = strlen((char *) text_prop.value);
if (XmbTextPropertyToTextList(FbTk::App::instance()->display(), &text_prop,
&list, &num) == Success &&
num > 0 && *list) {
m_icon_title = (char *)*list;
XFreeStringList(list);
} else
2003-06-22 12:35:03 +00:00
m_icon_title = text_prop.value ? (char *)text_prop.value : "";
2003-04-14 12:13:36 +00:00
} else
2003-06-22 12:35:03 +00:00
m_icon_title = text_prop.value ? (char *)text_prop.value : "";
2003-04-14 12:13:36 +00:00
2003-06-22 12:35:03 +00:00
if (text_prop.value)
XFree((char *) text_prop.value);
2003-04-14 12:13:36 +00:00
} else
m_icon_title = title();
} else
m_icon_title = title();
}
void WinClient::saveBlackboxAttribs(FluxboxWindow::BlackboxAttributes &blackbox_attribs) {
changeProperty(FbAtoms::instance()->getFluxboxAttributesAtom(),
PropModeReplace, XA_CARDINAL, 32,
(unsigned char *)&blackbox_attribs,
FluxboxWindow::PropBlackboxAttributesElements
);
}
void WinClient::updateBlackboxHints() {
int format;
Atom atom_return;
unsigned long num, len;
FbAtoms *atoms = FbAtoms::instance();
if (m_blackbox_hint) {
XFree(m_blackbox_hint);
m_blackbox_hint = 0;
}
if (property(atoms->getFluxboxHintsAtom(), 0,
2003-07-10 11:58:13 +00:00
PropBlackboxHintsElements, False,
atoms->getFluxboxHintsAtom(), &atom_return,
&format, &num, &len,
(unsigned char **) &m_blackbox_hint) &&
m_blackbox_hint) {
2003-07-10 11:58:13 +00:00
if (num != (unsigned)PropBlackboxHintsElements) {
XFree(m_blackbox_hint);
m_blackbox_hint = 0;
}
}
}
void WinClient::updateMWMHints() {
int format;
Atom atom_return;
unsigned long num = 0, len = 0;
if (m_mwm_hint) {
XFree(m_mwm_hint);
m_mwm_hint = 0;
}
Atom motif_wm_hints = FbAtoms::instance()->getMWMHintsAtom();
if (!(property(motif_wm_hints, 0,
PropMwmHintsElements, false,
motif_wm_hints, &atom_return,
&format, &num, &len,
(unsigned char **) &m_mwm_hint) &&
m_mwm_hint)) {
if (num != static_cast<unsigned int>(PropMwmHintsElements)) {
XFree(m_mwm_hint);
m_mwm_hint = 0;
return;
}
}
}
void WinClient::updateWMHints() {
XWMHints *wmhint = XGetWMHints(FbTk::App::instance()->display(), window());
if (! wmhint) {
m_focus_mode = F_PASSIVE;
window_group = None;
initial_state = NormalState;
} else {
wm_hint_flags = wmhint->flags;
if (wmhint->flags & InputHint) {
if (wmhint->input) {
if (send_focus_message)
m_focus_mode = F_LOCALLYACTIVE;
else
m_focus_mode = F_PASSIVE;
} else {
if (send_focus_message)
m_focus_mode = F_GLOBALLYACTIVE;
else
m_focus_mode = F_NOINPUT;
}
2003-12-15 11:55:58 +00:00
} else // not present => false (check?).
// note that mozilla has no value, and has send_focus
// and requires globally active
if (send_focus_message)
m_focus_mode = F_GLOBALLYACTIVE;
else // lots of apps have no hint, but need focus
m_focus_mode = F_PASSIVE;
if (wmhint->flags & StateHint)
initial_state = wmhint->initial_state;
else
initial_state = NormalState;
if (wmhint->flags & WindowGroupHint) {
if (! window_group)
window_group = wmhint->window_group;
} else
window_group = None;
XFree(wmhint);
}
}
void WinClient::updateWMNormalHints() {
long icccm_mask;
XSizeHints sizehint;
if (! XGetWMNormalHints(FbTk::App::instance()->display(), window(), &sizehint, &icccm_mask)) {
min_width = min_height =
base_width = base_height =
width_inc = height_inc = 1;
max_width = 0; // unbounded
max_height = 0;
min_aspect_x = min_aspect_y =
max_aspect_x = max_aspect_y = 1;
2003-09-11 19:55:27 +00:00
m_win_gravity = NorthWestGravity;
} else {
normal_hint_flags = sizehint.flags;
if (sizehint.flags & PMinSize) {
min_width = sizehint.min_width;
min_height = sizehint.min_height;
if (!(sizehint.flags & PBaseSize)) {
base_width = min_width;
base_height = min_height;
}
} else {
min_width = min_height = 1;
base_width = base_height = 0;
}
if (sizehint.flags & PBaseSize) {
base_width = sizehint.base_width;
base_height = sizehint.base_height;
if (!(sizehint.flags & PMinSize)) {
min_width = base_width;
min_height = base_height;
}
} // default set in PMinSize
if (sizehint.flags & PMaxSize) {
max_width = sizehint.max_width;
max_height = sizehint.max_height;
} else {
max_width = 0; // unbounded
max_height = 0;
}
if (sizehint.flags & PResizeInc) {
width_inc = sizehint.width_inc;
height_inc = sizehint.height_inc;
} else
width_inc = height_inc = 1;
if (sizehint.flags & PAspect) {
min_aspect_x = sizehint.min_aspect.x;
min_aspect_y = sizehint.min_aspect.y;
max_aspect_x = sizehint.max_aspect.x;
max_aspect_y = sizehint.max_aspect.y;
} else
min_aspect_x = min_aspect_y =
max_aspect_x = max_aspect_y = 1;
if (sizehint.flags & PWinGravity)
2003-09-11 19:55:27 +00:00
m_win_gravity = sizehint.win_gravity;
else
2003-09-11 19:55:27 +00:00
m_win_gravity = NorthWestGravity;
}
}
Window WinClient::getGroupLeftWindow() const {
int format;
Atom atom_return;
unsigned long num = 0, len = 0;
Atom group_left_hint = XInternAtom(FbTk::App::instance()->display(), "_FLUXBOX_GROUP_LEFT", False);
Window *data = 0;
if (property(group_left_hint, 0,
1, false,
XA_WINDOW, &atom_return,
&format, &num, &len,
(unsigned char **) &data) &&
data) {
if (num != 1) {
XFree(data);
return None;
} else {
Window ret = *data;
XFree(data);
return ret;
}
}
return None;
}
void WinClient::setGroupLeftWindow(Window win) {
Atom group_left_hint = XInternAtom(FbTk::App::instance()->display(), "_FLUXBOX_GROUP_LEFT", False);
changeProperty(group_left_hint, XA_WINDOW, 32,
PropModeReplace, (unsigned char *) &win, 1);
}
bool WinClient::hasGroupLeftWindow() const {
2003-07-28 18:30:02 +00:00
// try to find _FLUXBOX_GROUP_LEFT atom in window
// if we have one then we have a group left window
int format;
Atom atom_return;
unsigned long num = 0, len = 0;
Atom group_left_hint = XInternAtom(FbTk::App::instance()->display(), "_FLUXBOX_GROUP_LEFT", False);
Window *data = 0;
if (property(group_left_hint, 0,
1, false,
XA_WINDOW, &atom_return,
&format, &num, &len,
(unsigned char **) &data) &&
data) {
XFree(data);
2003-07-28 18:30:02 +00:00
if (num != 1)
return false;
else
return true;
}
2003-07-28 18:30:02 +00:00
return false;
}
void WinClient::addModal() {
++m_modal;
if (transient_for)
transient_for->addModal();
}
void WinClient::removeModal() {
--m_modal;
if (transient_for)
transient_for->removeModal();
}
bool WinClient::validateClient() const {
Display *display = FbTk::App::instance()->display();
2003-12-04 21:31:02 +00:00
FbTk::App::instance()->sync(false);
XEvent e;
if (( XCheckTypedWindowEvent(display, window(), DestroyNotify, &e) ||
XCheckTypedWindowEvent(display, window(), UnmapNotify, &e))
&& XPutBackEvent(display, &e)) {
Fluxbox::instance()->ungrab();
return false;
}
return true;
}
void WinClient::setStrut(Strut *strut) {
clearStrut();
m_strut = strut;
}
void WinClient::clearStrut() {
if (m_strut != 0) {
screen().clearStrut(m_strut);
m_strut = 0;
}
}
bool WinClient::focus() {
if (m_win == 0)
return false;
else
return m_win->setCurrentClient(*this, true);
}
2003-07-28 15:46:00 +00:00
void WinClient::updateWMProtocols() {
Atom *proto = 0;
int num_return = 0;
FbAtoms *fbatoms = FbAtoms::instance();
if (XGetWMProtocols(FbTk::App::instance()->display(), window(), &proto, &num_return)) {
2003-09-21 12:49:48 +00:00
// defaults
send_focus_message = false;
send_close_message = false;
// could be added to netizens twice...
for (int i = 0; i < num_return; ++i) {
if (proto[i] == fbatoms->getWMDeleteAtom())
2003-09-21 12:49:48 +00:00
send_close_message = true;
else if (proto[i] == fbatoms->getWMTakeFocusAtom())
send_focus_message = true;
else if (proto[i] == fbatoms->getFluxboxStructureMessagesAtom())
screen().addNetizen(window());
}
XFree(proto);
2003-07-28 16:29:25 +00:00
if (m_win)
m_win->updateFunctions();
} else {
cerr<<"Warning: Failed to read WM Protocols. "<<endl;
}
}
/**
* Changes width and height to the nearest (lower) value
* that conforms to it's size hints.
*
* display_* give the values that would be displayed
* to the user when resizing.
* We use pointers for display_* since they are optional.
*
* See ICCCM section 4.1.2.3
*/
void WinClient::applySizeHints(int &width, int &height,
int *display_width, int *display_height) {
int i = width, j = height;
// Check minimum size
if (width < 0 || width < static_cast<signed>(min_width))
width = min_width;
if (height < 0 || height < static_cast<signed>(min_height))
height = min_height;
// Check maximum size
if (max_width > 0 && width > static_cast<signed>(max_width))
width = max_width;
if (max_height > 0 && height > static_cast<signed>(max_height))
height = max_height;
// enforce incremental size limits, wrt base size
// only calculate this if we really need to
i = (width - base_width) / width_inc;
width = i*width_inc + base_width;
j = (height - base_height) / height_inc;
height = j*height_inc + base_height;
if (display_width)
*display_width = i;
if (display_height)
*display_height = j;
}