2002-05-25 00:29:44 +00:00
|
|
|
// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "../config.h"
|
|
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
#include <X11/cursorfont.h>
|
|
|
|
#include <X11/keysym.h>
|
|
|
|
|
|
|
|
#ifdef SHAPE
|
|
|
|
#include <X11/extensions/shape.h>
|
|
|
|
#endif // SHAPE
|
|
|
|
|
|
|
|
#ifdef HAVE_STDIO_H
|
|
|
|
# include <stdio.h>
|
|
|
|
#endif // HAVE_STDIO_H
|
|
|
|
|
|
|
|
#ifdef HAVE_STDLIB_H
|
|
|
|
# include <stdlib.h>
|
|
|
|
#endif // HAVE_STDLIB_H
|
|
|
|
|
|
|
|
#ifdef HAVE_STRING_H
|
|
|
|
# include <string.h>
|
|
|
|
#endif // HAVE_STRING_H
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
#endif // HAVE_UNISTD_H
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_PARAM_H
|
|
|
|
# include <sys/param.h>
|
|
|
|
#endif // HAVE_SYS_PARAM_H
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_SELECT_H
|
|
|
|
# include <sys/select.h>
|
|
|
|
#endif // HAVE_SYS_SELECT_H
|
|
|
|
|
|
|
|
#ifdef HAVE_SIGNAL_H
|
|
|
|
# include <signal.h>
|
|
|
|
#endif // HAVE_SIGNAL_H
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_SIGNAL_H
|
|
|
|
# include <sys/signal.h>
|
|
|
|
#endif // HAVE_SYS_SIGNAL_H
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <sys/stat.h>
|
|
|
|
#endif // HAVE_SYS_STAT_H
|
|
|
|
|
|
|
|
#ifdef TIME_WITH_SYS_TIME
|
|
|
|
# include <sys/time.h>
|
|
|
|
# include <time.h>
|
|
|
|
#else // !TIME_WITH_SYS_TIME
|
|
|
|
# ifdef HAVE_SYS_TIME_H
|
|
|
|
# include <sys/time.h>
|
|
|
|
# else // !HAVE_SYS_TIME_H
|
|
|
|
# include <time.h>
|
|
|
|
# endif // HAVE_SYS_TIME_H
|
|
|
|
#endif // TIME_WITH_SYS_TIME
|
|
|
|
|
|
|
|
#ifdef HAVE_LIBGEN_H
|
|
|
|
# include <libgen.h>
|
|
|
|
#endif // HAVE_LIBGEN_H
|
|
|
|
}
|
|
|
|
|
2002-06-01 17:54:32 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
#include <algorithm>
|
|
|
|
#include <string>
|
|
|
|
using std::string;
|
|
|
|
|
|
|
|
#include "blackbox.hh"
|
2002-11-03 11:46:05 +00:00
|
|
|
#include "otk/gccache.hh"
|
|
|
|
#include "otk/image.hh"
|
|
|
|
#include "otk/assassin.hh"
|
2002-10-31 06:54:56 +00:00
|
|
|
#include "screen.hh"
|
|
|
|
#include "util.hh"
|
2002-11-04 09:40:56 +00:00
|
|
|
#include "bbwindow.hh"
|
2002-10-31 06:54:56 +00:00
|
|
|
#include "workspace.hh"
|
2002-11-05 07:48:26 +00:00
|
|
|
#include "atom.hh"
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
namespace ob {
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
Blackbox *blackbox;
|
|
|
|
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
Blackbox::Blackbox(int argc, char **m_argv, char *rc)
|
|
|
|
: Openbox(argc, m_argv) {
|
2002-11-03 05:56:15 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
if (! XSupportsLocale())
|
|
|
|
fprintf(stderr, "X server does not support locale\n");
|
|
|
|
|
|
|
|
if (XSetLocaleModifiers("") == NULL)
|
|
|
|
fprintf(stderr, "cannot set locale modifiers\n");
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
ob::blackbox = this;
|
2002-05-25 00:29:44 +00:00
|
|
|
argv = m_argv;
|
2002-07-07 10:26:42 +00:00
|
|
|
|
|
|
|
// try to make sure the ~/.openbox directory exists
|
|
|
|
mkdir(expandTilde("~/.openbox").c_str(), S_IREAD | S_IWRITE | S_IEXEC |
|
|
|
|
S_IRGRP | S_IWGRP | S_IXGRP |
|
|
|
|
S_IROTH | S_IWOTH | S_IXOTH);
|
|
|
|
|
2002-11-03 12:55:30 +00:00
|
|
|
if (! rc) rc = "~/.openbox/rc3";
|
2002-05-25 00:29:44 +00:00
|
|
|
rc_file = expandTilde(rc);
|
2002-05-26 20:25:38 +00:00
|
|
|
config.setFile(rc_file);
|
2002-10-26 17:06:00 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
no_focus = False;
|
|
|
|
|
|
|
|
resource.auto_raise_delay.tv_sec = resource.auto_raise_delay.tv_usec = 0;
|
|
|
|
|
|
|
|
active_screen = 0;
|
2002-06-21 20:40:14 +00:00
|
|
|
focused_window = changing_window = (BlackboxWindow *) 0;
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
load_rc();
|
|
|
|
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom = new OBAtom();
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
cursor.session = XCreateFontCursor(otk::OBDisplay::display, XC_left_ptr);
|
|
|
|
cursor.move = XCreateFontCursor(otk::OBDisplay::display, XC_fleur);
|
|
|
|
cursor.ll_angle = XCreateFontCursor(otk::OBDisplay::display, XC_ll_angle);
|
|
|
|
cursor.lr_angle = XCreateFontCursor(otk::OBDisplay::display, XC_lr_angle);
|
|
|
|
cursor.ul_angle = XCreateFontCursor(otk::OBDisplay::display, XC_ul_angle);
|
|
|
|
cursor.ur_angle = XCreateFontCursor(otk::OBDisplay::display, XC_ur_angle);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
for (int i = 0; i < ScreenCount(otk::OBDisplay::display); i++) {
|
2002-05-25 00:29:44 +00:00
|
|
|
BScreen *screen = new BScreen(this, i);
|
|
|
|
|
|
|
|
if (! screen->isScreenManaged()) {
|
|
|
|
delete screen;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
screenList.push_back(screen);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (screenList.empty()) {
|
|
|
|
fprintf(stderr,
|
2002-10-31 08:42:45 +00:00
|
|
|
"Blackbox::Blackbox: no managable screens found, aborting.\n");
|
2002-05-25 00:29:44 +00:00
|
|
|
::exit(3);
|
|
|
|
}
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
// save current settings and default values
|
|
|
|
save_rc();
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
// set the screen with mouse to the first managed screen
|
|
|
|
active_screen = screenList.front();
|
|
|
|
setFocusedWindow(0);
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
XSynchronize(otk::OBDisplay::display, False);
|
|
|
|
XSync(otk::OBDisplay::display, False);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-10-31 08:42:45 +00:00
|
|
|
reconfigure_wait = False;
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 14:29:34 +00:00
|
|
|
timer = new otk::OBTimer(Openbox::instance->timerManager(),
|
|
|
|
(otk::OBTimeoutHandler)timeout,
|
|
|
|
this);
|
2002-05-25 00:29:44 +00:00
|
|
|
timer->setTimeout(0l);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Blackbox::~Blackbox(void) {
|
2002-11-03 11:46:05 +00:00
|
|
|
std::for_each(screenList.begin(), screenList.end(), otk::PointerAssassin());
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-05-28 14:32:42 +00:00
|
|
|
delete xatom;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
delete timer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::process_event(XEvent *e) {
|
|
|
|
switch (e->type) {
|
|
|
|
case ButtonPress: {
|
|
|
|
// strip the lock key modifiers
|
2002-11-03 11:46:05 +00:00
|
|
|
//e->xbutton.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
last_time = e->xbutton.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
BScreen *scrn = (BScreen *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xbutton.window))) {
|
|
|
|
win->buttonPressEvent(&e->xbutton);
|
|
|
|
|
|
|
|
/* XXX: is this sane on low colour desktops? */
|
|
|
|
if (e->xbutton.button == 1)
|
|
|
|
win->installColormap(True);
|
|
|
|
} else if ((scrn = searchScreen(e->xbutton.window))) {
|
|
|
|
scrn->buttonPressEvent(&e->xbutton);
|
|
|
|
if (active_screen != scrn) {
|
|
|
|
active_screen = scrn;
|
|
|
|
// first, set no focus window on the old screen
|
|
|
|
setFocusedWindow(0);
|
|
|
|
// and move focus to this screen
|
|
|
|
setFocusedWindow(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ButtonRelease: {
|
|
|
|
// strip the lock key modifiers
|
2002-11-03 11:46:05 +00:00
|
|
|
//e->xbutton.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
last_time = e->xbutton.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xbutton.window)))
|
|
|
|
win->buttonReleaseEvent(&e->xbutton);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ConfigureRequest: {
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xconfigurerequest.window))) {
|
|
|
|
win->configureRequestEvent(&e->xconfigurerequest);
|
|
|
|
} else {
|
|
|
|
if (validateWindow(e->xconfigurerequest.window)) {
|
|
|
|
XWindowChanges xwc;
|
|
|
|
|
|
|
|
xwc.x = e->xconfigurerequest.x;
|
|
|
|
xwc.y = e->xconfigurerequest.y;
|
|
|
|
xwc.width = e->xconfigurerequest.width;
|
|
|
|
xwc.height = e->xconfigurerequest.height;
|
|
|
|
xwc.border_width = e->xconfigurerequest.border_width;
|
|
|
|
xwc.sibling = e->xconfigurerequest.above;
|
|
|
|
xwc.stack_mode = e->xconfigurerequest.detail;
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
XConfigureWindow(otk::OBDisplay::display, e->xconfigurerequest.window,
|
2002-05-25 00:29:44 +00:00
|
|
|
e->xconfigurerequest.value_mask, &xwc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case MapRequest: {
|
|
|
|
#ifdef DEBUG
|
|
|
|
fprintf(stderr, "Blackbox::process_event(): MapRequest for 0x%lx\n",
|
|
|
|
e->xmaprequest.window);
|
|
|
|
#endif // DEBUG
|
|
|
|
|
|
|
|
BlackboxWindow *win = searchWindow(e->xmaprequest.window);
|
|
|
|
|
2002-07-05 01:24:32 +00:00
|
|
|
if (win) {
|
2002-07-10 22:24:48 +00:00
|
|
|
bool focus = False;
|
2002-07-05 01:24:32 +00:00
|
|
|
if (win->isIconic()) {
|
|
|
|
win->deiconify();
|
2002-07-10 22:24:48 +00:00
|
|
|
focus = True;
|
|
|
|
}
|
|
|
|
if (win->isShaded()) {
|
|
|
|
win->shade();
|
|
|
|
focus = True;
|
2002-07-05 01:24:32 +00:00
|
|
|
}
|
2002-07-10 22:24:48 +00:00
|
|
|
|
2002-07-30 07:17:07 +00:00
|
|
|
if (focus && (win->isTransient() || win->getScreen()->doFocusNew()) &&
|
|
|
|
win->isVisible())
|
2002-07-10 22:24:48 +00:00
|
|
|
win->setInputFocus();
|
2002-07-05 01:24:32 +00:00
|
|
|
} else {
|
2002-05-25 00:29:44 +00:00
|
|
|
BScreen *screen = searchScreen(e->xmaprequest.parent);
|
|
|
|
|
|
|
|
if (! screen) {
|
|
|
|
/*
|
|
|
|
we got a map request for a window who's parent isn't root. this
|
|
|
|
can happen in only one circumstance:
|
|
|
|
|
|
|
|
a client window unmapped a managed window, and then remapped it
|
|
|
|
somewhere between unmapping the client window and reparenting it
|
|
|
|
to root.
|
|
|
|
|
|
|
|
regardless of how it happens, we need to find the screen that
|
|
|
|
the window is on
|
|
|
|
*/
|
|
|
|
XWindowAttributes wattrib;
|
2002-11-03 11:46:05 +00:00
|
|
|
if (! XGetWindowAttributes(otk::OBDisplay::display, e->xmaprequest.window,
|
2002-05-25 00:29:44 +00:00
|
|
|
&wattrib)) {
|
|
|
|
// failed to get the window attributes, perhaps the window has
|
|
|
|
// now been destroyed?
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
screen = searchScreen(wattrib.root);
|
|
|
|
assert(screen != 0); // this should never happen
|
|
|
|
}
|
|
|
|
|
|
|
|
screen->manageWindow(e->xmaprequest.window);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case UnmapNotify: {
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
2002-06-21 20:40:14 +00:00
|
|
|
BScreen *screen = (BScreen *) 0;
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
if ((win = searchWindow(e->xunmap.window))) {
|
|
|
|
win->unmapNotifyEvent(&e->xunmap);
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if ((screen = searchSystrayWindow(e->xunmap.window))) {
|
|
|
|
screen->removeSystrayWindow(e->xunmap.window);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DestroyNotify: {
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
2002-06-21 20:40:14 +00:00
|
|
|
BScreen *screen = (BScreen *) 0;
|
2002-05-25 00:29:44 +00:00
|
|
|
BWindowGroup *group = (BWindowGroup *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xdestroywindow.window))) {
|
|
|
|
win->destroyNotifyEvent(&e->xdestroywindow);
|
|
|
|
} else if ((group = searchGroup(e->xdestroywindow.window))) {
|
|
|
|
delete group;
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if ((screen = searchSystrayWindow(e->xunmap.window))) {
|
|
|
|
screen->removeSystrayWindow(e->xunmap.window);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ReparentNotify: {
|
|
|
|
/*
|
|
|
|
this event is quite rare and is usually handled in unmapNotify
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
BlackboxWindow *win = searchWindow(e->xreparent.window);
|
2002-10-31 08:42:45 +00:00
|
|
|
if (win)
|
2002-05-25 00:29:44 +00:00
|
|
|
win->reparentNotifyEvent(&e->xreparent);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case MotionNotify: {
|
|
|
|
// motion notify compression...
|
|
|
|
XEvent realevent;
|
|
|
|
unsigned int i = 0;
|
2002-11-03 11:46:05 +00:00
|
|
|
while (XCheckTypedWindowEvent(otk::OBDisplay::display, e->xmotion.window,
|
2002-05-25 00:29:44 +00:00
|
|
|
MotionNotify, &realevent)) {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we have compressed some motion events, use the last one
|
|
|
|
if ( i > 0 )
|
|
|
|
e = &realevent;
|
|
|
|
|
2002-07-23 06:04:08 +00:00
|
|
|
// the pointer is on the wrong screen
|
2002-07-23 06:03:36 +00:00
|
|
|
if (! e->xmotion.same_screen)
|
|
|
|
break;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
// strip the lock key modifiers
|
2002-11-03 11:46:05 +00:00
|
|
|
//e->xmotion.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
last_time = e->xmotion.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xmotion.window)))
|
|
|
|
win->motionNotifyEvent(&e->xmotion);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case PropertyNotify: {
|
|
|
|
last_time = e->xproperty.time;
|
|
|
|
|
2002-07-14 20:54:03 +00:00
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
BScreen *screen = (BScreen *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xproperty.window)))
|
2002-07-05 01:24:32 +00:00
|
|
|
win->propertyNotifyEvent(&e->xproperty);
|
2002-07-14 20:54:03 +00:00
|
|
|
else if ((screen = searchScreen(e->xproperty.window)))
|
|
|
|
screen->propertyNotifyEvent(&e->xproperty);
|
2002-05-25 00:29:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case EnterNotify: {
|
|
|
|
last_time = e->xcrossing.time;
|
|
|
|
|
|
|
|
BScreen *screen = (BScreen *) 0;
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
if (e->xcrossing.mode == NotifyGrab) break;
|
|
|
|
|
|
|
|
if ((e->xcrossing.window == e->xcrossing.root) &&
|
|
|
|
(screen = searchScreen(e->xcrossing.window))) {
|
|
|
|
screen->getImageControl()->installRootColormap();
|
|
|
|
} else if ((win = searchWindow(e->xcrossing.window))) {
|
2002-08-07 00:24:58 +00:00
|
|
|
if (! no_focus)
|
|
|
|
win->enterNotifyEvent(&e->xcrossing);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LeaveNotify: {
|
|
|
|
last_time = e->xcrossing.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
2002-10-31 08:42:45 +00:00
|
|
|
if ((win = searchWindow(e->xcrossing.window)))
|
2002-08-07 00:24:58 +00:00
|
|
|
win->leaveNotifyEvent(&e->xcrossing);
|
2002-05-25 00:29:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Expose: {
|
|
|
|
// compress expose events
|
|
|
|
XEvent realevent;
|
|
|
|
unsigned int i = 0;
|
|
|
|
int ex1, ey1, ex2, ey2;
|
|
|
|
ex1 = e->xexpose.x;
|
|
|
|
ey1 = e->xexpose.y;
|
|
|
|
ex2 = ex1 + e->xexpose.width - 1;
|
|
|
|
ey2 = ey1 + e->xexpose.height - 1;
|
2002-11-03 11:46:05 +00:00
|
|
|
while (XCheckTypedWindowEvent(otk::OBDisplay::display, e->xexpose.window,
|
2002-05-25 00:29:44 +00:00
|
|
|
Expose, &realevent)) {
|
|
|
|
i++;
|
|
|
|
|
|
|
|
// merge expose area
|
|
|
|
ex1 = std::min(realevent.xexpose.x, ex1);
|
|
|
|
ey1 = std::min(realevent.xexpose.y, ey1);
|
|
|
|
ex2 = std::max(realevent.xexpose.x + realevent.xexpose.width - 1, ex2);
|
|
|
|
ey2 = std::max(realevent.xexpose.y + realevent.xexpose.height - 1, ey2);
|
|
|
|
}
|
|
|
|
if ( i > 0 )
|
|
|
|
e = &realevent;
|
|
|
|
|
|
|
|
// use the merged area
|
|
|
|
e->xexpose.x = ex1;
|
|
|
|
e->xexpose.y = ey1;
|
|
|
|
e->xexpose.width = ex2 - ex1 + 1;
|
|
|
|
e->xexpose.height = ey2 - ey1 + 1;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xexpose.window)))
|
|
|
|
win->exposeEvent(&e->xexpose);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case KeyPress: {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ColormapNotify: {
|
|
|
|
BScreen *screen = searchScreen(e->xcolormap.window);
|
|
|
|
|
|
|
|
if (screen)
|
|
|
|
screen->setRootColormapInstalled((e->xcolormap.state ==
|
|
|
|
ColormapInstalled) ? True : False);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case FocusIn: {
|
2002-05-28 19:34:23 +00:00
|
|
|
if (e->xfocus.detail != NotifyNonlinear &&
|
|
|
|
e->xfocus.detail != NotifyAncestor) {
|
2002-05-25 00:29:44 +00:00
|
|
|
/*
|
|
|
|
don't process FocusIns when:
|
|
|
|
1. the new focus window isn't an ancestor or inferior of the old
|
|
|
|
focus window (NotifyNonlinear)
|
2002-05-28 19:34:23 +00:00
|
|
|
make sure to allow the FocusIn when the old focus window was an
|
|
|
|
ancestor but didn't have a parent, such as root (NotifyAncestor)
|
2002-05-25 00:29:44 +00:00
|
|
|
*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
BlackboxWindow *win = searchWindow(e->xfocus.window);
|
|
|
|
if (win) {
|
|
|
|
if (! win->isFocused())
|
|
|
|
win->setFocusFlag(True);
|
|
|
|
|
|
|
|
/*
|
|
|
|
set the event window to None. when the FocusOut event handler calls
|
|
|
|
this function recursively, it uses this as an indication that focus
|
|
|
|
has moved to a known window.
|
|
|
|
*/
|
|
|
|
e->xfocus.window = None;
|
2002-08-16 04:25:13 +00:00
|
|
|
|
|
|
|
no_focus = False; // focusing is back on
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case FocusOut: {
|
|
|
|
if (e->xfocus.detail != NotifyNonlinear) {
|
|
|
|
/*
|
|
|
|
don't process FocusOuts when:
|
|
|
|
2. the new focus window isn't an ancestor or inferior of the old
|
|
|
|
focus window (NotifyNonlinear)
|
|
|
|
*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
BlackboxWindow *win = searchWindow(e->xfocus.window);
|
|
|
|
if (win && win->isFocused()) {
|
|
|
|
/*
|
|
|
|
before we mark "win" as unfocused, we need to verify that focus is
|
|
|
|
going to a known location, is in a known location, or set focus
|
|
|
|
to a known location.
|
|
|
|
*/
|
|
|
|
|
|
|
|
XEvent event;
|
|
|
|
// don't check the current focus if FocusOut was generated during a grab
|
|
|
|
bool check_focus = (e->xfocus.mode == NotifyNormal);
|
|
|
|
|
|
|
|
/*
|
|
|
|
First, check if there is a pending FocusIn event waiting. if there
|
|
|
|
is, process it and determine if focus has moved to another window
|
|
|
|
(the FocusIn event handler sets the window in the event
|
|
|
|
structure to None to indicate this).
|
|
|
|
*/
|
2002-11-03 11:46:05 +00:00
|
|
|
if (XCheckTypedEvent(otk::OBDisplay::display, FocusIn, &event)) {
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
process_event(&event);
|
|
|
|
if (event.xfocus.window == None) {
|
|
|
|
// focus has moved
|
|
|
|
check_focus = False;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (check_focus) {
|
|
|
|
/*
|
|
|
|
Second, we query the X server for the current input focus.
|
|
|
|
to make sure that we keep a consistent state.
|
|
|
|
*/
|
|
|
|
BlackboxWindow *focus;
|
|
|
|
Window w;
|
|
|
|
int revert;
|
2002-11-03 11:46:05 +00:00
|
|
|
XGetInputFocus(otk::OBDisplay::display, &w, &revert);
|
2002-05-25 00:29:44 +00:00
|
|
|
focus = searchWindow(w);
|
|
|
|
if (focus) {
|
|
|
|
/*
|
|
|
|
focus got from "win" to "focus" under some very strange
|
|
|
|
circumstances, and we need to make sure that the focus indication
|
|
|
|
is correct.
|
|
|
|
*/
|
|
|
|
setFocusedWindow(focus);
|
|
|
|
} else {
|
|
|
|
// we have no idea where focus went... so we set it to somewhere
|
|
|
|
setFocusedWindow(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ClientMessage: {
|
|
|
|
if (e->xclient.format == 32) {
|
2002-11-05 07:48:26 +00:00
|
|
|
if (e->xclient.message_type == xatom->getAtom(OBAtom::wm_change_state)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// WM_CHANGE_STATE message
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
if (! win || ! win->validateClient()) return;
|
|
|
|
|
|
|
|
if (e->xclient.data.l[0] == IconicState)
|
|
|
|
win->iconify();
|
|
|
|
if (e->xclient.data.l[0] == NormalState)
|
|
|
|
win->deiconify();
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::blackbox_change_workspace) ||
|
2002-06-21 20:40:14 +00:00
|
|
|
e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_current_desktop)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_CURRENT_DESKTOP message
|
2002-05-25 00:29:44 +00:00
|
|
|
BScreen *screen = searchScreen(e->xclient.window);
|
|
|
|
|
2002-06-05 04:20:08 +00:00
|
|
|
unsigned int workspace = e->xclient.data.l[0];
|
|
|
|
if (screen && workspace < screen->getWorkspaceCount())
|
|
|
|
screen->changeWorkspaceID(workspace);
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::blackbox_change_window_focus)) {
|
2002-07-16 04:15:24 +00:00
|
|
|
// TEMP HACK TO KEEP BBKEYS WORKING
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
|
|
|
|
if (win && win->isVisible() && win->setInputFocus())
|
|
|
|
win->installColormap(True);
|
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_active_window)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_ACTIVE_WINDOW
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win) {
|
2002-07-16 04:18:13 +00:00
|
|
|
BScreen *screen = win->getScreen();
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isIconic())
|
2002-08-06 22:21:12 +00:00
|
|
|
win->deiconify(False, False);
|
2002-08-02 14:57:30 +00:00
|
|
|
if (! win->isStuck() &&
|
2002-08-16 04:25:13 +00:00
|
|
|
(win->getWorkspaceNumber() != screen->getCurrentWorkspaceID())) {
|
|
|
|
no_focus = True;
|
2002-07-16 04:18:13 +00:00
|
|
|
screen->changeWorkspaceID(win->getWorkspaceNumber());
|
2002-08-16 04:25:13 +00:00
|
|
|
}
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isVisible() && win->setInputFocus()) {
|
2002-07-11 23:44:04 +00:00
|
|
|
win->getScreen()->getWorkspace(win->getWorkspaceNumber())->
|
|
|
|
raiseWindow(win);
|
2002-06-21 20:40:14 +00:00
|
|
|
win->installColormap(True);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::blackbox_cycle_window_focus)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// BLACKBOX_CYCLE_WINDOW_FOCUS
|
2002-05-25 00:29:44 +00:00
|
|
|
BScreen *screen = searchScreen(e->xclient.window);
|
|
|
|
|
|
|
|
if (screen) {
|
|
|
|
if (! e->xclient.data.l[0])
|
|
|
|
screen->prevFocus();
|
|
|
|
else
|
|
|
|
screen->nextFocus();
|
|
|
|
}
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_desktop)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_WM_DESKTOP
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
|
|
|
|
if (win) {
|
|
|
|
BScreen *screen = win->getScreen();
|
|
|
|
unsigned long wksp = (unsigned) e->xclient.data.l[0];
|
|
|
|
if (wksp < screen->getWorkspaceCount()) {
|
|
|
|
if (win->isIconic()) win->deiconify(False, True);
|
|
|
|
if (win->isStuck()) win->stick();
|
|
|
|
if (wksp != screen->getCurrentWorkspaceID())
|
|
|
|
win->withdraw();
|
|
|
|
else
|
|
|
|
win->show();
|
|
|
|
screen->reassociateWindow(win, wksp, True);
|
2002-07-19 14:38:10 +00:00
|
|
|
} else if (wksp == 0xfffffffe || // XXX: BUG, BUT DOING THIS SO KDE WORKS FOR NOW!!
|
|
|
|
wksp == 0xffffffff) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isIconic()) win->deiconify(False, True);
|
|
|
|
if (! win->isStuck()) win->stick();
|
|
|
|
if (! win->isVisible()) win->show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::blackbox_change_attributes)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// BLACKBOX_CHANGE_ATTRIBUTES
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
|
|
|
|
if (win && win->validateClient()) {
|
|
|
|
BlackboxHints net;
|
|
|
|
net.flags = e->xclient.data.l[0];
|
|
|
|
net.attrib = e->xclient.data.l[1];
|
|
|
|
net.workspace = e->xclient.data.l[2];
|
|
|
|
net.stack = e->xclient.data.l[3];
|
|
|
|
net.decoration = e->xclient.data.l[4];
|
|
|
|
|
|
|
|
win->changeBlackboxHints(&net);
|
|
|
|
}
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_number_of_desktops)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_NUMBER_OF_DESKTOPS
|
|
|
|
BScreen *screen = searchScreen(e->xclient.window);
|
|
|
|
|
2002-07-19 14:38:10 +00:00
|
|
|
if (e->xclient.data.l[0] > 0)
|
|
|
|
screen->changeWorkspaceCount((unsigned) e->xclient.data.l[0]);
|
2002-06-21 20:40:14 +00:00
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_close_window)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_CLOSE_WINDOW
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
if (win && win->validateClient())
|
|
|
|
win->close(); // could this be smarter?
|
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_WM_MOVERESIZE
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
if (win && win->validateClient()) {
|
|
|
|
int x_root = e->xclient.data.l[0],
|
|
|
|
y_root = e->xclient.data.l[1];
|
|
|
|
if ((Atom) e->xclient.data.l[2] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize_move)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->beginMove(x_root, y_root);
|
|
|
|
} else {
|
|
|
|
if ((Atom) e->xclient.data.l[2] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize_size_topleft))
|
2002-06-21 20:40:14 +00:00
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::TopLeft);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize_size_topright))
|
2002-06-21 20:40:14 +00:00
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::TopRight);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize_size_bottomleft))
|
2002-06-21 20:40:14 +00:00
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::BottomLeft);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_moveresize_size_bottomright))
|
2002-06-21 20:40:14 +00:00
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::BottomRight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (e->xclient.message_type ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
// NET_WM_STATE
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
if (win && win->validateClient()) {
|
|
|
|
const Atom action = (Atom) e->xclient.data.l[0];
|
|
|
|
const Atom state[] = { (Atom) e->xclient.data.l[1],
|
|
|
|
(Atom) e->xclient.data.l[2] };
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
|
|
if (! state[i])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ((Atom) e->xclient.data.l[0] == 1) {
|
|
|
|
// ADD
|
2002-11-05 07:48:26 +00:00
|
|
|
if (state[i] == xatom->getAtom(OBAtom::net_wm_state_modal)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setModal(True);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_vert)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else if (! win->isMaximized()) {
|
|
|
|
win->maximize(2); // vert
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_horz)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else if (! win->isMaximized()) {
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_shaded)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (! win->isShaded())
|
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_taskbar)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipTaskbar(True);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_pager)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipPager(True);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_fullscreen)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setFullscreen(True);
|
|
|
|
}
|
|
|
|
} else if (action == 0) {
|
|
|
|
// REMOVE
|
2002-11-05 07:48:26 +00:00
|
|
|
if (state[i] == xatom->getAtom(OBAtom::net_wm_state_modal)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setModal(False);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_vert)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
} else if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_horz)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(2); // vert
|
|
|
|
} else if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_shaded)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isShaded())
|
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_taskbar)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipTaskbar(False);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_pager)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipPager(False);
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_fullscreen)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setFullscreen(False);
|
|
|
|
}
|
|
|
|
} else if (action == 2) {
|
|
|
|
// TOGGLE
|
2002-11-05 07:48:26 +00:00
|
|
|
if (state[i] == xatom->getAtom(OBAtom::net_wm_state_modal)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setModal(! win->isModal());
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_vert)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
} else if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
} else if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else {
|
|
|
|
win->maximize(2); // vert
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_maximized_horz)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(2); // vert
|
|
|
|
} else if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
} else if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else {
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_shaded)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_taskbar)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipTaskbar(! win->skipTaskbar());
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_skip_pager)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setSkipPager(! win->skipPager());
|
|
|
|
} else if (state[i] ==
|
2002-11-05 07:48:26 +00:00
|
|
|
xatom->getAtom(OBAtom::net_wm_state_fullscreen)) {
|
2002-06-21 20:40:14 +00:00
|
|
|
win->setFullscreen(! win->isFullscreen());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NoExpose:
|
|
|
|
case ConfigureNotify:
|
|
|
|
case MapNotify:
|
|
|
|
break; // not handled, just ignore
|
|
|
|
|
|
|
|
default: {
|
|
|
|
#ifdef SHAPE
|
|
|
|
if (e->type == getShapeEventBase()) {
|
|
|
|
XShapeEvent *shape_event = (XShapeEvent *) e;
|
|
|
|
BlackboxWindow *win = searchWindow(e->xany.window);
|
|
|
|
|
2002-08-27 08:18:04 +00:00
|
|
|
if (win && shape_event->kind == ShapeBounding)
|
2002-05-25 00:29:44 +00:00
|
|
|
win->shapeEvent(shape_event);
|
|
|
|
}
|
|
|
|
#endif // SHAPE
|
|
|
|
}
|
|
|
|
} // switch
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Blackbox::handleSignal(int sig) {
|
|
|
|
switch (sig) {
|
|
|
|
case SIGHUP:
|
2002-07-19 19:04:28 +00:00
|
|
|
reconfigure();
|
2002-07-10 22:29:57 +00:00
|
|
|
break;
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
case SIGUSR1:
|
2002-07-19 19:04:28 +00:00
|
|
|
restart();
|
2002-05-25 00:29:44 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGUSR2:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGPIPE:
|
|
|
|
case SIGSEGV:
|
|
|
|
case SIGFPE:
|
|
|
|
case SIGINT:
|
|
|
|
case SIGTERM:
|
|
|
|
shutdown();
|
|
|
|
|
|
|
|
default:
|
|
|
|
return False;
|
|
|
|
}
|
|
|
|
|
|
|
|
return True;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Blackbox::validateWindow(Window window) {
|
|
|
|
XEvent event;
|
2002-11-03 11:46:05 +00:00
|
|
|
if (XCheckTypedWindowEvent(otk::OBDisplay::display, window, DestroyNotify, &event)) {
|
|
|
|
XPutBackEvent(otk::OBDisplay::display, &event);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
return False;
|
|
|
|
}
|
|
|
|
|
|
|
|
return True;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BScreen *Blackbox::searchScreen(Window window) {
|
|
|
|
ScreenList::iterator it = screenList.begin();
|
|
|
|
|
|
|
|
for (; it != screenList.end(); ++it) {
|
|
|
|
BScreen *s = *it;
|
|
|
|
if (s->getRootWindow() == window)
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (BScreen *) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
BScreen *Blackbox::searchSystrayWindow(Window window) {
|
|
|
|
WindowScreenLookup::iterator it = systraySearchList.find(window);
|
|
|
|
if (it != systraySearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (BScreen*) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
BlackboxWindow *Blackbox::searchWindow(Window window) {
|
|
|
|
WindowLookup::iterator it = windowSearchList.find(window);
|
|
|
|
if (it != windowSearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (BlackboxWindow*) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BWindowGroup *Blackbox::searchGroup(Window window) {
|
|
|
|
GroupLookup::iterator it = groupSearchList.find(window);
|
|
|
|
if (it != groupSearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (BWindowGroup *) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
void Blackbox::saveSystrayWindowSearch(Window window, BScreen *screen) {
|
|
|
|
systraySearchList.insert(WindowScreenLookupPair(window, screen));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Blackbox::saveWindowSearch(Window window, BlackboxWindow *data) {
|
|
|
|
windowSearchList.insert(WindowLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveGroupSearch(Window window, BWindowGroup *data) {
|
|
|
|
groupSearchList.insert(GroupLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
void Blackbox::removeSystrayWindowSearch(Window window) {
|
|
|
|
systraySearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Blackbox::removeWindowSearch(Window window) {
|
|
|
|
windowSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeGroupSearch(Window window) {
|
|
|
|
groupSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::restart(const char *prog) {
|
|
|
|
shutdown();
|
|
|
|
|
|
|
|
if (prog) {
|
2002-06-30 09:53:52 +00:00
|
|
|
putenv(const_cast<char *>(screenList.front()->displayString().c_str()));
|
2002-05-25 00:29:44 +00:00
|
|
|
execlp(prog, prog, NULL);
|
|
|
|
perror(prog);
|
|
|
|
}
|
|
|
|
|
|
|
|
// fall back in case the above execlp doesn't work
|
|
|
|
execvp(argv[0], argv);
|
2002-11-03 11:46:05 +00:00
|
|
|
string name = ::basename(argv[0]);
|
2002-05-25 00:29:44 +00:00
|
|
|
execvp(name.c_str(), argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::shutdown(void) {
|
2002-11-03 11:46:05 +00:00
|
|
|
Openbox::shutdown();
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
XSetInputFocus(otk::OBDisplay::display, PointerRoot, None, CurrentTime);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::shutdown));
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
XSync(otk::OBDisplay::display, False);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-22 04:29:40 +00:00
|
|
|
#ifdef XINERAMA
|
|
|
|
void Blackbox::saveXineramaPlacement(bool x) {
|
|
|
|
resource.xinerama_placement = x;
|
|
|
|
config.setValue("session.xineramaSupport.windowPlacement",
|
|
|
|
resource.xinerama_placement);
|
2002-11-03 11:46:05 +00:00
|
|
|
reconfigure(); // make sure all screens get this
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
void Blackbox::saveXineramaMaximizing(bool x) {
|
|
|
|
resource.xinerama_maximize = x;
|
|
|
|
config.setValue("session.xineramaSupport.windowMaximizing",
|
|
|
|
resource.xinerama_maximize);
|
|
|
|
reconfigure(); // make sure all screens get this change
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveXineramaSnapping(bool x) {
|
|
|
|
resource.xinerama_snap = x;
|
|
|
|
config.setValue("session.xineramaSupport.windowSnapping",
|
|
|
|
resource.xinerama_snap);
|
|
|
|
reconfigure(); // make sure all screens get this change
|
|
|
|
}
|
|
|
|
#endif // XINERAMA
|
|
|
|
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
/*
|
|
|
|
* Save all values as they are so that the defaults will be written to the rc
|
|
|
|
* file
|
|
|
|
*/
|
2002-05-25 00:29:44 +00:00
|
|
|
void Blackbox::save_rc(void) {
|
2002-05-26 20:25:38 +00:00
|
|
|
config.setAutoSave(false);
|
|
|
|
|
|
|
|
config.setValue("session.colorsPerChannel", resource.colors_per_channel);
|
|
|
|
config.setValue("session.doubleClickInterval",
|
|
|
|
resource.double_click_interval);
|
|
|
|
config.setValue("session.autoRaiseDelay",
|
|
|
|
((resource.auto_raise_delay.tv_sec * 1000) +
|
|
|
|
(resource.auto_raise_delay.tv_usec / 1000)));
|
|
|
|
config.setValue("session.cacheLife", resource.cache_life / 60000);
|
|
|
|
config.setValue("session.cacheMax", resource.cache_max);
|
|
|
|
config.setValue("session.styleFile", resource.style_file);
|
2002-05-27 03:12:21 +00:00
|
|
|
config.setValue("session.titlebarLayout", resource.titlebar_layout);
|
2002-08-10 18:21:10 +00:00
|
|
|
|
|
|
|
string s;
|
|
|
|
if (resource.mod_mask & Mod1Mask) s += "Mod1-";
|
|
|
|
if (resource.mod_mask & Mod2Mask) s += "Mod2-";
|
|
|
|
if (resource.mod_mask & Mod3Mask) s += "Mod3-";
|
|
|
|
if (resource.mod_mask & Mod4Mask) s += "Mod4-";
|
|
|
|
if (resource.mod_mask & Mod5Mask) s += "Mod5-";
|
|
|
|
if (resource.mod_mask & ShiftMask) s += "Shift-";
|
|
|
|
if (resource.mod_mask & ControlMask) s += "Control-";
|
|
|
|
s.resize(s.size() - 1); // drop the last '-'
|
|
|
|
config.setValue("session.modifierMask", s);
|
2002-05-26 20:25:38 +00:00
|
|
|
|
2002-07-22 04:29:40 +00:00
|
|
|
#ifdef XINERAMA
|
|
|
|
saveXineramaPlacement(resource.xinerama_placement);
|
|
|
|
saveXineramaMaximizing(resource.xinerama_maximize);
|
|
|
|
saveXineramaSnapping(resource.xinerama_snap);
|
|
|
|
#endif // XINERAMA
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::save_rc));
|
|
|
|
|
|
|
|
config.setAutoSave(true);
|
|
|
|
config.save();
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::load_rc(void) {
|
2002-05-26 20:25:38 +00:00
|
|
|
if (! config.load())
|
2002-07-07 20:38:39 +00:00
|
|
|
config.create();
|
2002-05-26 20:25:38 +00:00
|
|
|
|
|
|
|
string s;
|
|
|
|
|
|
|
|
if (! config.getValue("session.colorsPerChannel",
|
|
|
|
resource.colors_per_channel))
|
|
|
|
resource.colors_per_channel = 4;
|
|
|
|
if (resource.colors_per_channel < 2) resource.colors_per_channel = 2;
|
|
|
|
else if (resource.colors_per_channel > 6) resource.colors_per_channel = 6;
|
|
|
|
|
|
|
|
if (config.getValue("session.styleFile", s))
|
|
|
|
resource.style_file = expandTilde(s);
|
2002-05-25 00:29:44 +00:00
|
|
|
else
|
|
|
|
resource.style_file = DEFAULTSTYLE;
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
if (! config.getValue("session.doubleClickInterval",
|
|
|
|
resource.double_click_interval));
|
|
|
|
resource.double_click_interval = 250;
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
if (! config.getValue("session.autoRaiseDelay",
|
|
|
|
resource.auto_raise_delay.tv_usec))
|
|
|
|
resource.auto_raise_delay.tv_usec = 400;
|
2002-05-25 00:29:44 +00:00
|
|
|
resource.auto_raise_delay.tv_sec = resource.auto_raise_delay.tv_usec / 1000;
|
|
|
|
resource.auto_raise_delay.tv_usec -=
|
|
|
|
(resource.auto_raise_delay.tv_sec * 1000);
|
|
|
|
resource.auto_raise_delay.tv_usec *= 1000;
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
if (! config.getValue("session.cacheLife", resource.cache_life))
|
|
|
|
resource.cache_life = 5;
|
2002-05-25 00:29:44 +00:00
|
|
|
resource.cache_life *= 60000;
|
|
|
|
|
2002-05-26 20:25:38 +00:00
|
|
|
if (! config.getValue("session.cacheMax", resource.cache_max))
|
|
|
|
resource.cache_max = 200;
|
2002-05-27 03:12:21 +00:00
|
|
|
|
|
|
|
if (! config.getValue("session.titlebarLayout", resource.titlebar_layout))
|
|
|
|
resource.titlebar_layout = "ILMC";
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
#ifdef XINERAMA
|
|
|
|
if (! config.getValue("session.xineramaSupport.windowPlacement",
|
|
|
|
resource.xinerama_placement))
|
2002-07-22 06:12:20 +00:00
|
|
|
resource.xinerama_placement = false;
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
if (! config.getValue("session.xineramaSupport.windowMaximizing",
|
|
|
|
resource.xinerama_maximize))
|
2002-07-22 06:12:20 +00:00
|
|
|
resource.xinerama_maximize = false;
|
2002-07-22 04:29:40 +00:00
|
|
|
|
|
|
|
if (! config.getValue("session.xineramaSupport.windowSnapping",
|
|
|
|
resource.xinerama_snap))
|
2002-07-22 06:12:20 +00:00
|
|
|
resource.xinerama_snap = false;
|
2002-07-22 04:29:40 +00:00
|
|
|
#endif // XINERAMA
|
2002-08-10 18:21:10 +00:00
|
|
|
|
|
|
|
resource.mod_mask = 0;
|
|
|
|
if (config.getValue("session.modifierMask", s)) {
|
|
|
|
if (s.find("Mod1") != string::npos)
|
|
|
|
resource.mod_mask |= Mod1Mask;
|
|
|
|
if (s.find("Mod2") != string::npos)
|
|
|
|
resource.mod_mask |= Mod2Mask;
|
|
|
|
if (s.find("Mod3") != string::npos)
|
|
|
|
resource.mod_mask |= Mod3Mask;
|
|
|
|
if (s.find("Mod4") != string::npos)
|
|
|
|
resource.mod_mask |= Mod4Mask;
|
|
|
|
if (s.find("Mod5") != string::npos)
|
|
|
|
resource.mod_mask |= Mod5Mask;
|
|
|
|
if (s.find("Shift") != string::npos)
|
|
|
|
resource.mod_mask |= ShiftMask;
|
|
|
|
if (s.find("Control") != string::npos)
|
|
|
|
resource.mod_mask |= ControlMask;
|
|
|
|
}
|
|
|
|
if (! resource.mod_mask)
|
|
|
|
resource.mod_mask = Mod1Mask;
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::reconfigure(void) {
|
2002-07-22 04:29:40 +00:00
|
|
|
// don't reconfigure while saving the initial rc file, it's a waste and it
|
|
|
|
// breaks somethings (workspace names)
|
2002-11-03 11:46:05 +00:00
|
|
|
if (state() == Openbox::State_Starting) return;
|
2002-07-22 04:29:40 +00:00
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
reconfigure_wait = True;
|
|
|
|
|
2002-11-04 06:06:13 +00:00
|
|
|
if (! timer->timing()) timer->start();
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::real_reconfigure(void) {
|
2002-05-26 20:25:38 +00:00
|
|
|
load_rc();
|
|
|
|
|
2002-11-03 11:46:05 +00:00
|
|
|
otk::OBDisplay::gcCache()->purge();
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::reconfigure));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveStyleFilename(const string& filename) {
|
|
|
|
assert(! filename.empty());
|
|
|
|
resource.style_file = filename;
|
2002-05-26 20:25:38 +00:00
|
|
|
config.setValue("session.styleFile", resource.style_file);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-03 14:29:34 +00:00
|
|
|
void Blackbox::timeout(Blackbox *t) {
|
|
|
|
if (t->reconfigure_wait)
|
|
|
|
t->real_reconfigure();
|
2002-05-25 00:29:44 +00:00
|
|
|
|
2002-11-03 14:29:34 +00:00
|
|
|
t->reconfigure_wait = False;
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-21 20:40:14 +00:00
|
|
|
void Blackbox::setChangingWindow(BlackboxWindow *win) {
|
|
|
|
// make sure one of the two is null and the other isn't
|
|
|
|
assert((! changing_window && win) || (! win && changing_window));
|
|
|
|
changing_window = win;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
void Blackbox::setFocusedWindow(BlackboxWindow *win) {
|
|
|
|
if (focused_window && focused_window == win) // nothing to do
|
|
|
|
return;
|
|
|
|
|
|
|
|
BScreen *old_screen = 0;
|
|
|
|
|
|
|
|
if (focused_window) {
|
|
|
|
focused_window->setFocusFlag(False);
|
|
|
|
old_screen = focused_window->getScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (win && ! win->isIconic()) {
|
|
|
|
// the active screen is the one with the last focused window...
|
|
|
|
// this will keep focus on this screen no matter where the mouse goes,
|
|
|
|
// so multihead keybindings will continue to work on that screen until the
|
|
|
|
// user focuses a window on a different screen.
|
|
|
|
active_screen = win->getScreen();
|
|
|
|
focused_window = win;
|
|
|
|
} else {
|
|
|
|
focused_window = 0;
|
|
|
|
if (! old_screen) {
|
|
|
|
if (active_screen) {
|
|
|
|
// set input focus to the toolbar of the screen with mouse
|
2002-11-03 11:46:05 +00:00
|
|
|
XSetInputFocus(otk::OBDisplay::display,
|
2002-05-28 10:15:08 +00:00
|
|
|
active_screen->getRootWindow(),
|
2002-05-25 00:29:44 +00:00
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
} else {
|
|
|
|
// set input focus to the toolbar of the first managed screen
|
2002-11-03 11:46:05 +00:00
|
|
|
XSetInputFocus(otk::OBDisplay::display,
|
2002-05-28 10:15:08 +00:00
|
|
|
screenList.front()->getRootWindow(),
|
2002-05-25 00:29:44 +00:00
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// set input focus to the toolbar of the last screen
|
2002-11-03 11:46:05 +00:00
|
|
|
XSetInputFocus(otk::OBDisplay::display, old_screen->getRootWindow(),
|
2002-05-25 00:29:44 +00:00
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (active_screen && active_screen->isScreenManaged()) {
|
|
|
|
active_screen->updateNetizenWindowFocus();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_screen && old_screen != active_screen) {
|
|
|
|
old_screen->updateNetizenWindowFocus();
|
|
|
|
}
|
|
|
|
}
|
2002-11-03 11:46:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|