openbox/util/epist/window.cc
Dana Jansens 6595476d81 keep track of window dimentions
use them for window moving
2002-07-20 20:04:27 +00:00

377 lines
10 KiB
C++

// -*- mode: C++; indent-tabs-mode: nil; -*-
// window.cc for Epistophy - a key handler for NETWM/EWMH window managers.
// Copyright (c) 2002 - 2002 Ben Jansens <ben at orodu.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.
#ifdef HAVE_CONFIG_H
# include "../../config.h"
#endif // HAVE_CONFIG_H
#include <iostream>
using std::cout;
using std::endl;
using std::hex;
using std::dec;
#include "epist.hh"
#include "screen.hh"
#include "window.hh"
#include "../../src/XAtom.hh"
XWindow::XWindow(epist *epist, screen *screen, Window window)
: _epist(epist), _screen(screen), _xatom(epist->xatom()), _window(window) {
_unmapped = false;
XSelectInput(_epist->getXDisplay(), _window,
PropertyChangeMask | StructureNotifyMask);
updateDimentions();
updateGravity();
updateState();
updateDesktop();
updateTitle();
updateClass();
_epist->addWindow(this);
}
XWindow::~XWindow() {
if (! _unmapped)
XSelectInput(_epist->getXDisplay(), _window, None);
_epist->removeWindow(this);
}
void XWindow::updateDimentions() {
Window root, child;
int x, y;
unsigned int w, h, b, d;
if (XGetGeometry(_epist->getXDisplay(), _window, &root, &x, &y, &w, &h,
&b, &d) &&
XTranslateCoordinates(_epist->getXDisplay(), _window, root, x, y,
&x, &y, &child))
_rect.setRect(x, y, w, h);
else
_rect.setRect(0, 0, 1, 1);
}
void XWindow::updateGravity() {
XSizeHints size;
long ret;
if (XGetWMNormalHints(_epist->getXDisplay(), _window, &size, &ret) &&
(size.flags & PWinGravity))
_gravity = size.win_gravity;
else
_gravity = NorthWestGravity;
}
void XWindow::updateState() {
// set the defaults
_shaded = _iconic = _max_vert = _max_horz = false;
unsigned long num = (unsigned) -1;
Atom *state;
if (! _xatom->getValue(_window, XAtom::net_wm_state, XAtom::atom,
num, &state))
return;
for (unsigned long i = 0; i < num; ++i) {
if (state[i] == _xatom->getAtom(XAtom::net_wm_state_maximized_vert))
_max_vert = true;
if (state[i] == _xatom->getAtom(XAtom::net_wm_state_maximized_horz))
_max_horz = true;
if (state[i] == _xatom->getAtom(XAtom::net_wm_state_shaded))
_shaded = true;
if (state[i] == _xatom->getAtom(XAtom::net_wm_state_hidden))
_iconic = true;
}
delete [] state;
}
void XWindow::updateDesktop() {
if (! _xatom->getValue(_window, XAtom::net_wm_desktop, XAtom::cardinal,
static_cast<unsigned long>(_desktop)))
_desktop = 0;
}
void XWindow::updateTitle() {
_title = "";
// try netwm
if (! _xatom->getValue(_window, XAtom::net_wm_name, XAtom::utf8, _title)) {
// try old x stuff
_xatom->getValue(_window, XAtom::wm_name, XAtom::ansi, _title);
}
if (_title.empty())
_title = "Unnamed";
}
void XWindow::updateClass() {
// set the defaults
_app_name = _app_class = "";
XAtom::StringVect v;
unsigned long num = 2;
if (! _xatom->getValue(_window, XAtom::wm_class, XAtom::ansi, num, v))
return;
if (num > 0) _app_name = v[0];
if (num > 1) _app_class = v[1];
}
void XWindow::processEvent(const XEvent &e) {
assert(e.xany.window == _window);
switch (e.type) {
case ConfigureNotify:
updateDimentions();
break;
case PropertyNotify:
if (e.xproperty.atom == XA_WM_NORMAL_HINTS)
updateGravity();
else if (e.xproperty.atom == _xatom->getAtom(XAtom::net_wm_state))
updateState();
else if (e.xproperty.atom == _xatom->getAtom(XAtom::net_wm_desktop))
updateDesktop();
else if (e.xproperty.atom == _xatom->getAtom(XAtom::net_wm_name) ||
e.xproperty.atom == _xatom->getAtom(XAtom::wm_name))
updateTitle();
else if (e.xproperty.atom == _xatom->getAtom(XAtom::wm_class))
updateClass();
break;
case DestroyNotify:
case UnmapNotify:
_unmapped = true;
break;
}
}
void XWindow::shade(const bool sh) const {
_xatom->sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, (sh ? 1 : 0),
_xatom->getAtom(XAtom::net_wm_state_shaded));
}
void XWindow::close() const {
_xatom->sendClientMessage(_screen->rootWindow(), XAtom::net_close_window,
_window);
}
void XWindow::raise() const {
XRaiseWindow(_epist->getXDisplay(), _window);
}
void XWindow::lower() const {
XLowerWindow(_epist->getXDisplay(), _window);
}
void XWindow::iconify() const {
_xatom->sendClientMessage(_screen->rootWindow(), XAtom::wm_change_state,
_window, IconicState);
}
void XWindow::focus() const {
// this will also unshade the window..
_xatom->sendClientMessage(_screen->rootWindow(), XAtom::net_active_window,
_window);
}
void XWindow::sendTo(unsigned int dest) const {
_xatom->sendClientMessage(_screen->rootWindow(), XAtom::net_wm_desktop,
_window, dest);
}
void XWindow::move(int x, int y) const {
// get the window's decoration sizes (margins)
Strut margins;
Window win = _window, parent, root, last = None;
Window *children = 0;
unsigned int nchildren;
XWindowAttributes wattr;
while (XQueryTree(_epist->getXDisplay(), win, &root, &parent, &children,
&nchildren)) {
if (children && nchildren > 0)
XFree(children); // don't care about the children
if (! parent) // no parent!?
return;
// if the parent window is the root window, stop here
if (parent == root)
break;
last = win;
win = parent;
}
if (! (XTranslateCoordinates(_epist->getXDisplay(), last, win, 0, 0,
(int *) &margins.left,
(int *) &margins.top,
&parent) &&
XGetWindowAttributes(_epist->getXDisplay(), win, &wattr)))
return;
margins.right = wattr.width - _rect.width() - margins.left;
margins.bottom = wattr.height - _rect.height() - margins.top;
margins.left += wattr.border_width;
margins.right += wattr.border_width;
margins.top += wattr.border_width;
margins.bottom += wattr.border_width;
// this makes things work. why? i don't know. but you need them.
margins.right -= 2;
margins.bottom -= 2;
// find the frame's reference position based on the window's gravity
switch (_gravity) {
case NorthWestGravity:
x -= margins.left;
y -= margins.top;
break;
case NorthGravity:
x += (margins.left + margins.right) / 2;
y -= margins.top;
break;
case NorthEastGravity:
x += margins.right;
y -= margins.top;
case WestGravity:
x -= margins.left;
y += (margins.top + margins.bottom) / 2;
break;
case CenterGravity:
x += (margins.left + margins.right) / 2;
y += (margins.top + margins.bottom) / 2;
break;
case EastGravity:
x += margins.right;
y += (margins.top + margins.bottom) / 2;
case SouthWestGravity:
x -= margins.left;
y += margins.bottom;
break;
case SouthGravity:
x += (margins.left + margins.right) / 2;
y += margins.bottom;
break;
case SouthEastGravity:
x += margins.right;
y += margins.bottom;
break;
default:
break;
}
XMoveWindow(_epist->getXDisplay(), _window, x, y);
}
void XWindow::resize(unsigned int width, unsigned int height) const {
XResizeWindow(_epist->getXDisplay(), _window, width, height);
}
void XWindow::toggleMaximize(Max max) const {
switch (max) {
case Max_Full:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, (_max_vert == _max_horz ? 2 : 1),
_xatom->getAtom(XAtom::net_wm_state_maximized_horz),
_xatom->getAtom(XAtom::net_wm_state_maximized_vert));
break;
case Max_Horz:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 2,
_xatom->getAtom(XAtom::net_wm_state_maximized_horz));
break;
case Max_Vert:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 2,
_xatom->getAtom(XAtom::net_wm_state_maximized_vert));
break;
case Max_None:
assert(false); // you should not do this. it is pointless and probly a bug
break;
}
}
void XWindow::maximize(Max max) const {
switch (max) {
case Max_None:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 0,
_xatom->getAtom(XAtom::net_wm_state_maximized_horz),
_xatom->getAtom(XAtom::net_wm_state_maximized_vert));
break;
case Max_Full:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 1,
_xatom->getAtom(XAtom::net_wm_state_maximized_horz),
_xatom->getAtom(XAtom::net_wm_state_maximized_vert));
break;
case Max_Horz:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 1,
_xatom->getAtom(XAtom::net_wm_state_maximized_horz));
break;
case Max_Vert:
_xatom->
sendClientMessage(_screen->rootWindow(), XAtom::net_wm_state,
_window, 1,
_xatom->getAtom(XAtom::net_wm_state_maximized_vert));
break;
}
}