2002-05-25 00:29:44 +00:00
|
|
|
// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
|
|
|
|
// blackbox.cc for Blackbox - an X11 Window manager
|
|
|
|
// Copyright (c) 2001 - 2002 Sean 'Shaleh' Perry <shaleh@debian.org>
|
|
|
|
// Copyright (c) 1997 - 2000 Brad Hughes (bhughes@tcac.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
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
|
|
#include <X11/Xresource.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
|
|
|
|
}
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <string>
|
|
|
|
using std::string;
|
|
|
|
|
|
|
|
#include "i18n.hh"
|
|
|
|
#include "blackbox.hh"
|
|
|
|
#include "Basemenu.hh"
|
|
|
|
#include "Clientmenu.hh"
|
|
|
|
#include "GCCache.hh"
|
|
|
|
#include "Image.hh"
|
|
|
|
#include "Rootmenu.hh"
|
|
|
|
#include "Screen.hh"
|
|
|
|
#include "Slit.hh"
|
|
|
|
#include "Toolbar.hh"
|
|
|
|
#include "Util.hh"
|
|
|
|
#include "Window.hh"
|
|
|
|
#include "Workspace.hh"
|
|
|
|
#include "Workspacemenu.hh"
|
|
|
|
|
|
|
|
|
|
|
|
// X event scanner for enter/leave notifies - adapted from twm
|
|
|
|
struct scanargs {
|
|
|
|
Window w;
|
|
|
|
bool leave, inferior, enter;
|
|
|
|
};
|
|
|
|
|
|
|
|
static Bool queueScanner(Display *, XEvent *e, char *args) {
|
|
|
|
scanargs *scan = (scanargs *) args;
|
|
|
|
if ((e->type == LeaveNotify) &&
|
|
|
|
(e->xcrossing.window == scan->w) &&
|
|
|
|
(e->xcrossing.mode == NotifyNormal)) {
|
|
|
|
scan->leave = True;
|
|
|
|
scan->inferior = (e->xcrossing.detail == NotifyInferior);
|
|
|
|
} else if ((e->type == EnterNotify) && (e->xcrossing.mode == NotifyUngrab)) {
|
|
|
|
scan->enter = True;
|
|
|
|
}
|
|
|
|
|
|
|
|
return False;
|
|
|
|
}
|
|
|
|
|
|
|
|
Blackbox *blackbox;
|
|
|
|
|
|
|
|
|
2002-05-25 02:24:30 +00:00
|
|
|
Blackbox::Blackbox(char **m_argv, char *dpy_name, char *rc, char *menu)
|
2002-05-25 00:29:44 +00:00
|
|
|
: BaseDisplay(m_argv[0], dpy_name) {
|
|
|
|
if (! XSupportsLocale())
|
|
|
|
fprintf(stderr, "X server does not support locale\n");
|
|
|
|
|
|
|
|
if (XSetLocaleModifiers("") == NULL)
|
|
|
|
fprintf(stderr, "cannot set locale modifiers\n");
|
|
|
|
|
|
|
|
::blackbox = this;
|
|
|
|
argv = m_argv;
|
2002-05-25 02:24:30 +00:00
|
|
|
if (! rc) rc = "~/.openbox/rc";
|
2002-05-25 00:29:44 +00:00
|
|
|
rc_file = expandTilde(rc);
|
2002-05-25 02:24:30 +00:00
|
|
|
if (! menu) menu = "~/.openbox/menu";
|
|
|
|
menu_file = expandTilde(menu);
|
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;
|
|
|
|
focused_window = (BlackboxWindow *) 0;
|
|
|
|
|
|
|
|
XrmInitialize();
|
|
|
|
load_rc();
|
|
|
|
|
|
|
|
init_icccm();
|
|
|
|
|
|
|
|
cursor.session = XCreateFontCursor(getXDisplay(), XC_left_ptr);
|
|
|
|
cursor.move = XCreateFontCursor(getXDisplay(), XC_fleur);
|
|
|
|
cursor.ll_angle = XCreateFontCursor(getXDisplay(), XC_ll_angle);
|
|
|
|
cursor.lr_angle = XCreateFontCursor(getXDisplay(), XC_lr_angle);
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < getNumberOfScreens(); i++) {
|
|
|
|
BScreen *screen = new BScreen(this, i);
|
|
|
|
|
|
|
|
if (! screen->isScreenManaged()) {
|
|
|
|
delete screen;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
screenList.push_back(screen);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (screenList.empty()) {
|
|
|
|
fprintf(stderr,
|
|
|
|
i18n(blackboxSet, blackboxNoManagableScreens,
|
|
|
|
"Blackbox::Blackbox: no managable screens found, aborting.\n"));
|
|
|
|
::exit(3);
|
|
|
|
}
|
|
|
|
|
|
|
|
// set the screen with mouse to the first managed screen
|
|
|
|
active_screen = screenList.front();
|
|
|
|
setFocusedWindow(0);
|
|
|
|
|
|
|
|
XSynchronize(getXDisplay(), False);
|
|
|
|
XSync(getXDisplay(), False);
|
|
|
|
|
|
|
|
reconfigure_wait = reread_menu_wait = False;
|
|
|
|
|
|
|
|
timer = new BTimer(this, this);
|
|
|
|
timer->setTimeout(0l);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Blackbox::~Blackbox(void) {
|
|
|
|
std::for_each(screenList.begin(), screenList.end(), PointerAssassin());
|
|
|
|
|
|
|
|
std::for_each(menuTimestamps.begin(), menuTimestamps.end(),
|
|
|
|
PointerAssassin());
|
|
|
|
|
|
|
|
delete timer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::process_event(XEvent *e) {
|
|
|
|
switch (e->type) {
|
|
|
|
case ButtonPress: {
|
|
|
|
// strip the lock key modifiers
|
|
|
|
e->xbutton.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
|
|
|
|
|
|
|
last_time = e->xbutton.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
Toolbar *tbar = (Toolbar *) 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 ((menu = searchMenu(e->xbutton.window))) {
|
|
|
|
menu->buttonPressEvent(&e->xbutton);
|
|
|
|
} else if ((slit = searchSlit(e->xbutton.window))) {
|
|
|
|
slit->buttonPressEvent(&e->xbutton);
|
|
|
|
} else if ((tbar = searchToolbar(e->xbutton.window))) {
|
|
|
|
tbar->buttonPressEvent(&e->xbutton);
|
|
|
|
} 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
|
|
|
|
e->xbutton.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
|
|
|
|
|
|
|
last_time = e->xbutton.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
Toolbar *tbar = (Toolbar *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xbutton.window)))
|
|
|
|
win->buttonReleaseEvent(&e->xbutton);
|
|
|
|
else if ((menu = searchMenu(e->xbutton.window)))
|
|
|
|
menu->buttonReleaseEvent(&e->xbutton);
|
|
|
|
else if ((tbar = searchToolbar(e->xbutton.window)))
|
|
|
|
tbar->buttonReleaseEvent(&e->xbutton);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ConfigureRequest: {
|
|
|
|
// compress configure requests...
|
|
|
|
XEvent realevent;
|
|
|
|
unsigned int i = 0;
|
|
|
|
while(XCheckTypedWindowEvent(getXDisplay(), e->xconfigurerequest.window,
|
|
|
|
ConfigureRequest, &realevent)) {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if ( i > 0 )
|
|
|
|
e = &realevent;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xconfigurerequest.window))) {
|
|
|
|
win->configureRequestEvent(&e->xconfigurerequest);
|
|
|
|
} else if ((slit = searchSlit(e->xconfigurerequest.window))) {
|
|
|
|
slit->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;
|
|
|
|
|
|
|
|
XConfigureWindow(getXDisplay(), e->xconfigurerequest.window,
|
|
|
|
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);
|
|
|
|
|
|
|
|
if (! win) {
|
|
|
|
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;
|
|
|
|
if (! XGetWindowAttributes(getXDisplay(), e->xmaprequest.window,
|
|
|
|
&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;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xunmap.window))) {
|
|
|
|
win->unmapNotifyEvent(&e->xunmap);
|
|
|
|
} else if ((slit = searchSlit(e->xunmap.window))) {
|
|
|
|
slit->unmapNotifyEvent(&e->xunmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DestroyNotify: {
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
BWindowGroup *group = (BWindowGroup *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xdestroywindow.window))) {
|
|
|
|
win->destroyNotifyEvent(&e->xdestroywindow);
|
|
|
|
} else if ((slit = searchSlit(e->xdestroywindow.window))) {
|
|
|
|
slit->removeClient(e->xdestroywindow.window, False);
|
|
|
|
} else if ((group = searchGroup(e->xdestroywindow.window))) {
|
|
|
|
delete group;
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
if (win) {
|
|
|
|
win->reparentNotifyEvent(&e->xreparent);
|
|
|
|
} else {
|
|
|
|
Slit *slit = searchSlit(e->xreparent.window);
|
|
|
|
if (slit && slit->getWindowID() != e->xreparent.parent)
|
|
|
|
slit->removeClient(e->xreparent.window, True);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case MotionNotify: {
|
|
|
|
// motion notify compression...
|
|
|
|
XEvent realevent;
|
|
|
|
unsigned int i = 0;
|
|
|
|
while (XCheckTypedWindowEvent(getXDisplay(), e->xmotion.window,
|
|
|
|
MotionNotify, &realevent)) {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we have compressed some motion events, use the last one
|
|
|
|
if ( i > 0 )
|
|
|
|
e = &realevent;
|
|
|
|
|
|
|
|
// strip the lock key modifiers
|
|
|
|
e->xbutton.state &= ~(NumLockMask | ScrollLockMask | LockMask);
|
|
|
|
|
|
|
|
last_time = e->xmotion.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xmotion.window)))
|
|
|
|
win->motionNotifyEvent(&e->xmotion);
|
|
|
|
else if ((menu = searchMenu(e->xmotion.window)))
|
|
|
|
menu->motionNotifyEvent(&e->xmotion);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case PropertyNotify: {
|
|
|
|
last_time = e->xproperty.time;
|
|
|
|
|
|
|
|
if (e->xproperty.state != PropertyDelete) {
|
|
|
|
BlackboxWindow *win = searchWindow(e->xproperty.window);
|
|
|
|
|
|
|
|
if (win)
|
|
|
|
win->propertyNotifyEvent(e->xproperty.atom);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case EnterNotify: {
|
|
|
|
last_time = e->xcrossing.time;
|
|
|
|
|
|
|
|
BScreen *screen = (BScreen *) 0;
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
Toolbar *tbar = (Toolbar *) 0;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
|
|
|
|
if (e->xcrossing.mode == NotifyGrab) break;
|
|
|
|
|
|
|
|
XEvent dummy;
|
|
|
|
scanargs sa;
|
|
|
|
sa.w = e->xcrossing.window;
|
|
|
|
sa.enter = sa.leave = False;
|
|
|
|
XCheckIfEvent(getXDisplay(), &dummy, queueScanner, (char *) &sa);
|
|
|
|
|
|
|
|
if ((e->xcrossing.window == e->xcrossing.root) &&
|
|
|
|
(screen = searchScreen(e->xcrossing.window))) {
|
|
|
|
screen->getImageControl()->installRootColormap();
|
|
|
|
} else if ((win = searchWindow(e->xcrossing.window))) {
|
|
|
|
if (win->getScreen()->isSloppyFocus() &&
|
|
|
|
(! win->isFocused()) && (! no_focus)) {
|
|
|
|
if (((! sa.leave) || sa.inferior) && win->isVisible()) {
|
|
|
|
if (win->setInputFocus())
|
|
|
|
win->installColormap(True); // XXX: shouldnt we honour no install?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if ((menu = searchMenu(e->xcrossing.window))) {
|
|
|
|
menu->enterNotifyEvent(&e->xcrossing);
|
|
|
|
} else if ((tbar = searchToolbar(e->xcrossing.window))) {
|
|
|
|
tbar->enterNotifyEvent(&e->xcrossing);
|
|
|
|
} else if ((slit = searchSlit(e->xcrossing.window))) {
|
|
|
|
slit->enterNotifyEvent(&e->xcrossing);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LeaveNotify: {
|
|
|
|
last_time = e->xcrossing.time;
|
|
|
|
|
|
|
|
BlackboxWindow *win = (BlackboxWindow *) 0;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
Toolbar *tbar = (Toolbar *) 0;
|
|
|
|
Slit *slit = (Slit *) 0;
|
|
|
|
|
|
|
|
if ((menu = searchMenu(e->xcrossing.window)))
|
|
|
|
menu->leaveNotifyEvent(&e->xcrossing);
|
|
|
|
else if ((win = searchWindow(e->xcrossing.window)))
|
|
|
|
win->installColormap(False);
|
|
|
|
else if ((tbar = searchToolbar(e->xcrossing.window)))
|
|
|
|
tbar->leaveNotifyEvent(&e->xcrossing);
|
|
|
|
else if ((slit = searchSlit(e->xcrossing.window)))
|
|
|
|
slit->leaveNotifyEvent(&e->xcrossing);
|
|
|
|
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;
|
|
|
|
while (XCheckTypedWindowEvent(getXDisplay(), e->xexpose.window,
|
|
|
|
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;
|
|
|
|
Basemenu *menu = (Basemenu *) 0;
|
|
|
|
Toolbar *tbar = (Toolbar *) 0;
|
|
|
|
|
|
|
|
if ((win = searchWindow(e->xexpose.window)))
|
|
|
|
win->exposeEvent(&e->xexpose);
|
|
|
|
else if ((menu = searchMenu(e->xexpose.window)))
|
|
|
|
menu->exposeEvent(&e->xexpose);
|
|
|
|
else if ((tbar = searchToolbar(e->xexpose.window)))
|
|
|
|
tbar->exposeEvent(&e->xexpose);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case KeyPress: {
|
|
|
|
Toolbar *tbar = searchToolbar(e->xkey.window);
|
|
|
|
|
|
|
|
if (tbar && tbar->isEditing())
|
|
|
|
tbar->keyPressEvent(&e->xkey);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ColormapNotify: {
|
|
|
|
BScreen *screen = searchScreen(e->xcolormap.window);
|
|
|
|
|
|
|
|
if (screen)
|
|
|
|
screen->setRootColormapInstalled((e->xcolormap.state ==
|
|
|
|
ColormapInstalled) ? True : False);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case FocusIn: {
|
|
|
|
if (e->xfocus.detail != NotifyNonlinear) {
|
|
|
|
/*
|
|
|
|
don't process FocusIns when:
|
|
|
|
1. 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) {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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).
|
|
|
|
*/
|
|
|
|
if (XCheckTypedEvent(getXDisplay(), FocusIn, &event)) {
|
|
|
|
|
|
|
|
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;
|
|
|
|
XGetInputFocus(getXDisplay(), &w, &revert);
|
|
|
|
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) {
|
|
|
|
if (e->xclient.message_type == getWMChangeStateAtom()) {
|
|
|
|
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();
|
|
|
|
} else if(e->xclient.message_type == getBlackboxChangeWorkspaceAtom()) {
|
|
|
|
BScreen *screen = searchScreen(e->xclient.window);
|
|
|
|
|
|
|
|
if (screen && e->xclient.data.l[0] >= 0 &&
|
|
|
|
e->xclient.data.l[0] <
|
|
|
|
static_cast<signed>(screen->getWorkspaceCount()))
|
|
|
|
screen->changeWorkspaceID(e->xclient.data.l[0]);
|
|
|
|
} else if (e->xclient.message_type == getBlackboxChangeWindowFocusAtom()) {
|
|
|
|
BlackboxWindow *win = searchWindow(e->xclient.window);
|
|
|
|
|
|
|
|
if (win && win->isVisible() && win->setInputFocus())
|
|
|
|
win->installColormap(True);
|
|
|
|
} else if (e->xclient.message_type == getBlackboxCycleWindowFocusAtom()) {
|
|
|
|
BScreen *screen = searchScreen(e->xclient.window);
|
|
|
|
|
|
|
|
if (screen) {
|
|
|
|
if (! e->xclient.data.l[0])
|
|
|
|
screen->prevFocus();
|
|
|
|
else
|
|
|
|
screen->nextFocus();
|
|
|
|
}
|
|
|
|
} else if (e->xclient.message_type == getBlackboxChangeAttributesAtom()) {
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
if (win)
|
|
|
|
win->shapeEvent(shape_event);
|
|
|
|
}
|
|
|
|
#endif // SHAPE
|
|
|
|
}
|
|
|
|
} // switch
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Blackbox::handleSignal(int sig) {
|
|
|
|
switch (sig) {
|
|
|
|
case SIGHUP:
|
|
|
|
reconfigure();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGUSR1:
|
|
|
|
reload_rc();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGUSR2:
|
|
|
|
rereadMenu();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGPIPE:
|
|
|
|
case SIGSEGV:
|
|
|
|
case SIGFPE:
|
|
|
|
case SIGINT:
|
|
|
|
case SIGTERM:
|
|
|
|
shutdown();
|
|
|
|
|
|
|
|
default:
|
|
|
|
return False;
|
|
|
|
}
|
|
|
|
|
|
|
|
return True;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::init_icccm(void) {
|
|
|
|
xa_wm_colormap_windows =
|
|
|
|
XInternAtom(getXDisplay(), "WM_COLORMAP_WINDOWS", False);
|
|
|
|
xa_wm_protocols = XInternAtom(getXDisplay(), "WM_PROTOCOLS", False);
|
|
|
|
xa_wm_state = XInternAtom(getXDisplay(), "WM_STATE", False);
|
|
|
|
xa_wm_change_state = XInternAtom(getXDisplay(), "WM_CHANGE_STATE", False);
|
|
|
|
xa_wm_delete_window = XInternAtom(getXDisplay(), "WM_DELETE_WINDOW", False);
|
|
|
|
xa_wm_take_focus = XInternAtom(getXDisplay(), "WM_TAKE_FOCUS", False);
|
|
|
|
motif_wm_hints = XInternAtom(getXDisplay(), "_MOTIF_WM_HINTS", False);
|
|
|
|
|
|
|
|
blackbox_hints = XInternAtom(getXDisplay(), "_BLACKBOX_HINTS", False);
|
|
|
|
blackbox_attributes =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_ATTRIBUTES", False);
|
|
|
|
blackbox_change_attributes =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_CHANGE_ATTRIBUTES", False);
|
|
|
|
blackbox_structure_messages =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_STRUCTURE_MESSAGES", False);
|
|
|
|
blackbox_notify_startup =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_STARTUP", False);
|
|
|
|
blackbox_notify_window_add =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WINDOW_ADD", False);
|
|
|
|
blackbox_notify_window_del =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WINDOW_DEL", False);
|
|
|
|
blackbox_notify_current_workspace =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_CURRENT_WORKSPACE", False);
|
|
|
|
blackbox_notify_workspace_count =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WORKSPACE_COUNT", False);
|
|
|
|
blackbox_notify_window_focus =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WINDOW_FOCUS", False);
|
|
|
|
blackbox_notify_window_raise =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WINDOW_RAISE", False);
|
|
|
|
blackbox_notify_window_lower =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_NOTIFY_WINDOW_LOWER", False);
|
|
|
|
blackbox_change_workspace =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_CHANGE_WORKSPACE", False);
|
|
|
|
blackbox_change_window_focus =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_CHANGE_WINDOW_FOCUS", False);
|
|
|
|
blackbox_cycle_window_focus =
|
|
|
|
XInternAtom(getXDisplay(), "_BLACKBOX_CYCLE_WINDOW_FOCUS", False);
|
|
|
|
|
|
|
|
#ifdef NEWWMSPEC
|
|
|
|
net_supported = XInternAtom(getXDisplay(), "_NET_SUPPORTED", False);
|
|
|
|
net_client_list = XInternAtom(getXDisplay(), "_NET_CLIENT_LIST", False);
|
|
|
|
net_client_list_stacking =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_CLIENT_LIST_STACKING", False);
|
|
|
|
net_number_of_desktops =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_NUMBER_OF_DESKTOPS", False);
|
|
|
|
net_desktop_geometry =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_DESKTOP_GEOMETRY", False);
|
|
|
|
net_desktop_viewport =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_DESKTOP_VIEWPORT", False);
|
|
|
|
net_current_desktop =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_CURRENT_DESKTOP", False);
|
|
|
|
net_desktop_names = XInternAtom(getXDisplay(), "_NET_DESKTOP_NAMES", False);
|
|
|
|
net_active_window = XInternAtom(getXDisplay(), "_NET_ACTIVE_WINDOW", False);
|
|
|
|
net_workarea = XInternAtom(getXDisplay(), "_NET_WORKAREA", False);
|
|
|
|
net_supporting_wm_check =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_SUPPORTING_WM_CHECK", False);
|
|
|
|
net_virtual_roots = XInternAtom(getXDisplay(), "_NET_VIRTUAL_ROOTS", False);
|
|
|
|
net_close_window = XInternAtom(getXDisplay(), "_NET_CLOSE_WINDOW", False);
|
|
|
|
net_wm_moveresize = XInternAtom(getXDisplay(), "_NET_WM_MOVERESIZE", False);
|
|
|
|
net_properties = XInternAtom(getXDisplay(), "_NET_PROPERTIES", False);
|
|
|
|
net_wm_name = XInternAtom(getXDisplay(), "_NET_WM_NAME", False);
|
|
|
|
net_wm_desktop = XInternAtom(getXDisplay(), "_NET_WM_DESKTOP", False);
|
|
|
|
net_wm_window_type =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_WM_WINDOW_TYPE", False);
|
|
|
|
net_wm_state = XInternAtom(getXDisplay(), "_NET_WM_STATE", False);
|
|
|
|
net_wm_strut = XInternAtom(getXDisplay(), "_NET_WM_STRUT", False);
|
|
|
|
net_wm_icon_geometry =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_WM_ICON_GEOMETRY", False);
|
|
|
|
net_wm_icon = XInternAtom(getXDisplay(), "_NET_WM_ICON", False);
|
|
|
|
net_wm_pid = XInternAtom(getXDisplay(), "_NET_WM_PID", False);
|
|
|
|
net_wm_handled_icons =
|
|
|
|
XInternAtom(getXDisplay(), "_NET_WM_HANDLED_ICONS", False);
|
|
|
|
net_wm_ping = XInternAtom(getXDisplay(), "_NET_WM_PING", False);
|
|
|
|
#endif // NEWWMSPEC
|
|
|
|
|
|
|
|
#ifdef HAVE_GETPID
|
|
|
|
blackbox_pid = XInternAtom(getXDisplay(), "_BLACKBOX_PID", False);
|
|
|
|
#endif // HAVE_GETPID
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Blackbox::validateWindow(Window window) {
|
|
|
|
XEvent event;
|
|
|
|
if (XCheckTypedWindowEvent(getXDisplay(), window, DestroyNotify, &event)) {
|
|
|
|
XPutBackEvent(getXDisplay(), &event);
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Basemenu *Blackbox::searchMenu(Window window) {
|
|
|
|
MenuLookup::iterator it = menuSearchList.find(window);
|
|
|
|
if (it != menuSearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (Basemenu*) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Toolbar *Blackbox::searchToolbar(Window window) {
|
|
|
|
ToolbarLookup::iterator it = toolbarSearchList.find(window);
|
|
|
|
if (it != toolbarSearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (Toolbar*) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Slit *Blackbox::searchSlit(Window window) {
|
|
|
|
SlitLookup::iterator it = slitSearchList.find(window);
|
|
|
|
if (it != slitSearchList.end())
|
|
|
|
return it->second;
|
|
|
|
|
|
|
|
return (Slit*) 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveWindowSearch(Window window, BlackboxWindow *data) {
|
|
|
|
windowSearchList.insert(WindowLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveGroupSearch(Window window, BWindowGroup *data) {
|
|
|
|
groupSearchList.insert(GroupLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveMenuSearch(Window window, Basemenu *data) {
|
|
|
|
menuSearchList.insert(MenuLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveToolbarSearch(Window window, Toolbar *data) {
|
|
|
|
toolbarSearchList.insert(ToolbarLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveSlitSearch(Window window, Slit *data) {
|
|
|
|
slitSearchList.insert(SlitLookupPair(window, data));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeWindowSearch(Window window) {
|
|
|
|
windowSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeGroupSearch(Window window) {
|
|
|
|
groupSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeMenuSearch(Window window) {
|
|
|
|
menuSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeToolbarSearch(Window window) {
|
|
|
|
toolbarSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::removeSlitSearch(Window window) {
|
|
|
|
slitSearchList.erase(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::restart(const char *prog) {
|
|
|
|
shutdown();
|
|
|
|
|
|
|
|
if (prog) {
|
|
|
|
execlp(prog, prog, NULL);
|
|
|
|
perror(prog);
|
|
|
|
}
|
|
|
|
|
|
|
|
// fall back in case the above execlp doesn't work
|
|
|
|
execvp(argv[0], argv);
|
|
|
|
string name = basename(argv[0]);
|
|
|
|
execvp(name.c_str(), argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::shutdown(void) {
|
|
|
|
BaseDisplay::shutdown();
|
|
|
|
|
|
|
|
XSetInputFocus(getXDisplay(), PointerRoot, None, CurrentTime);
|
|
|
|
|
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::shutdown));
|
|
|
|
|
|
|
|
XSync(getXDisplay(), False);
|
|
|
|
|
|
|
|
save_rc();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::save_rc(void) {
|
|
|
|
XrmDatabase new_blackboxrc = (XrmDatabase) 0;
|
|
|
|
char rc_string[1024];
|
|
|
|
|
|
|
|
load_rc();
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.colorsPerChannel: %d",
|
|
|
|
resource.colors_per_channel);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.doubleClickInterval: %lu",
|
|
|
|
resource.double_click_interval);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.autoRaiseDelay: %lu",
|
|
|
|
((resource.auto_raise_delay.tv_sec * 1000) +
|
|
|
|
(resource.auto_raise_delay.tv_usec / 1000)));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.cacheLife: %lu", resource.cache_life / 60000);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.cacheMax: %lu", resource.cache_max);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
ScreenList::iterator it = screenList.begin();
|
|
|
|
for (; it != screenList.end(); ++it) {
|
|
|
|
BScreen *screen = *it;
|
|
|
|
int screen_number = screen->getScreenNumber();
|
|
|
|
|
|
|
|
char *placement = (char *) 0;
|
|
|
|
|
|
|
|
switch (screen->getSlitPlacement()) {
|
|
|
|
case Slit::TopLeft: placement = "TopLeft"; break;
|
|
|
|
case Slit::CenterLeft: placement = "CenterLeft"; break;
|
|
|
|
case Slit::BottomLeft: placement = "BottomLeft"; break;
|
|
|
|
case Slit::TopCenter: placement = "TopCenter"; break;
|
|
|
|
case Slit::BottomCenter: placement = "BottomCenter"; break;
|
|
|
|
case Slit::TopRight: placement = "TopRight"; break;
|
|
|
|
case Slit::BottomRight: placement = "BottomRight"; break;
|
|
|
|
case Slit::CenterRight: default: placement = "CenterRight"; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.slit.placement: %s", screen_number,
|
|
|
|
placement);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.slit.direction: %s", screen_number,
|
|
|
|
((screen->getSlitDirection() == Slit::Horizontal) ? "Horizontal" :
|
|
|
|
"Vertical"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.slit.onTop: %s", screen_number,
|
|
|
|
((screen->getSlit()->isOnTop()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.slit.autoHide: %s", screen_number,
|
|
|
|
((screen->getSlit()->doAutoHide()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.opaqueMove: %s",
|
|
|
|
((screen->doOpaqueMove()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.imageDither: %s",
|
|
|
|
((screen->getImageControl()->doDither()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.fullMaximization: %s", screen_number,
|
|
|
|
((screen->doFullMax()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.focusNewWindows: %s", screen_number,
|
|
|
|
((screen->doFocusNew()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.focusLastWindow: %s", screen_number,
|
|
|
|
((screen->doFocusLast()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.rowPlacementDirection: %s",
|
|
|
|
screen_number,
|
|
|
|
((screen->getRowPlacementDirection() == BScreen::LeftRight) ?
|
|
|
|
"LeftToRight" : "RightToLeft"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.colPlacementDirection: %s",
|
|
|
|
screen_number,
|
|
|
|
((screen->getColPlacementDirection() == BScreen::TopBottom) ?
|
|
|
|
"TopToBottom" : "BottomToTop"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
switch (screen->getPlacementPolicy()) {
|
|
|
|
case BScreen::CascadePlacement:
|
|
|
|
placement = "CascadePlacement";
|
|
|
|
break;
|
|
|
|
case BScreen::ColSmartPlacement:
|
|
|
|
placement = "ColSmartPlacement";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BScreen::RowSmartPlacement:
|
|
|
|
default:
|
|
|
|
placement = "RowSmartPlacement";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sprintf(rc_string, "session.screen%d.windowPlacement: %s", screen_number,
|
|
|
|
placement);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
string fmodel;
|
|
|
|
if (screen->isSloppyFocus()) {
|
|
|
|
fmodel = "SloppyFocus";
|
|
|
|
if (screen->doAutoRaise()) fmodel += " AutoRaise";
|
|
|
|
if (screen->doClickRaise()) fmodel += " ClickRaise";
|
|
|
|
} else {
|
|
|
|
fmodel = "ClickToFocus";
|
|
|
|
}
|
|
|
|
sprintf(rc_string, "session.screen%d.focusModel: %s", screen_number,
|
|
|
|
fmodel.c_str());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.workspaces: %d", screen_number,
|
|
|
|
screen->getWorkspaceCount());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.toolbar.onTop: %s", screen_number,
|
|
|
|
((screen->getToolbar()->isOnTop()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.toolbar.autoHide: %s",
|
|
|
|
screen_number,
|
|
|
|
((screen->getToolbar()->doAutoHide()) ? "True" : "False"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
switch (screen->getToolbarPlacement()) {
|
|
|
|
case Toolbar::TopLeft: placement = "TopLeft"; break;
|
|
|
|
case Toolbar::BottomLeft: placement = "BottomLeft"; break;
|
|
|
|
case Toolbar::TopCenter: placement = "TopCenter"; break;
|
|
|
|
case Toolbar::TopRight: placement = "TopRight"; break;
|
|
|
|
case Toolbar::BottomRight: placement = "BottomRight"; break;
|
|
|
|
case Toolbar::BottomCenter: default: placement = "BottomCenter"; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.toolbar.placement: %s",
|
|
|
|
screen_number, placement);
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
load_rc(screen);
|
|
|
|
|
|
|
|
// these are static, but may not be saved in the users .blackboxrc,
|
|
|
|
// writing these resources will allow the user to edit them at a later
|
|
|
|
// time... but loading the defaults before saving allows us to rewrite the
|
|
|
|
// users changes...
|
|
|
|
|
|
|
|
#ifdef HAVE_STRFTIME
|
|
|
|
sprintf(rc_string, "session.screen%d.strftimeFormat: %s", screen_number,
|
|
|
|
screen->getStrftimeFormat());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
#else // !HAVE_STRFTIME
|
|
|
|
sprintf(rc_string, "session.screen%d.dateFormat: %s", screen_number,
|
|
|
|
((screen->getDateFormat() == B_EuropeanDate) ?
|
|
|
|
"European" : "American"));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.clockFormat: %d", screen_number,
|
|
|
|
((screen->isClock24Hour()) ? 24 : 12));
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
#endif // HAVE_STRFTIME
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.edgeSnapThreshold: %d",
|
|
|
|
screen_number, screen->getEdgeSnapThreshold());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
sprintf(rc_string, "session.screen%d.toolbar.widthPercent: %d",
|
|
|
|
screen_number, screen->getToolbarWidthPercent());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, rc_string);
|
|
|
|
|
|
|
|
// write out the user's workspace names
|
|
|
|
|
|
|
|
string save_string = screen->getWorkspace(0)->getName();
|
|
|
|
for (unsigned int i = 1; i < screen->getWorkspaceCount(); ++i) {
|
|
|
|
save_string += ',';
|
|
|
|
save_string += screen->getWorkspace(i)->getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
char *resource_string = new char[save_string.length() + 48];
|
|
|
|
sprintf(resource_string, "session.screen%d.workspaceNames: %s",
|
|
|
|
screen_number, save_string.c_str());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, resource_string);
|
|
|
|
|
|
|
|
delete [] resource_string;
|
|
|
|
}
|
|
|
|
|
|
|
|
XrmDatabase old_blackboxrc = XrmGetFileDatabase(rc_file.c_str());
|
|
|
|
|
|
|
|
XrmMergeDatabases(new_blackboxrc, &old_blackboxrc);
|
|
|
|
XrmPutFileDatabase(old_blackboxrc, rc_file.c_str());
|
|
|
|
XrmDestroyDatabase(old_blackboxrc);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::load_rc(void) {
|
|
|
|
XrmDatabase database = (XrmDatabase) 0;
|
|
|
|
|
|
|
|
database = XrmGetFileDatabase(rc_file.c_str());
|
|
|
|
|
|
|
|
XrmValue value;
|
|
|
|
char *value_type;
|
|
|
|
int int_value;
|
|
|
|
unsigned long long_value;
|
|
|
|
|
|
|
|
resource.colors_per_channel = 4;
|
|
|
|
if (XrmGetResource(database, "session.colorsPerChannel",
|
|
|
|
"Session.ColorsPerChannel", &value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%d", &int_value) == 1) {
|
|
|
|
resource.colors_per_channel = int_value;
|
|
|
|
if (resource.colors_per_channel < 2) resource.colors_per_channel = 2;
|
|
|
|
if (resource.colors_per_channel > 6) resource.colors_per_channel = 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (XrmGetResource(database, "session.styleFile", "Session.StyleFile",
|
|
|
|
&value_type, &value))
|
|
|
|
resource.style_file = expandTilde(value.addr);
|
|
|
|
else
|
|
|
|
resource.style_file = DEFAULTSTYLE;
|
|
|
|
|
|
|
|
resource.double_click_interval = 250;
|
|
|
|
if (XrmGetResource(database, "session.doubleClickInterval",
|
|
|
|
"Session.DoubleClickInterval", &value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%lu", &long_value) == 1) {
|
|
|
|
resource.double_click_interval = long_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
resource.auto_raise_delay.tv_usec = 400;
|
|
|
|
if (XrmGetResource(database, "session.autoRaiseDelay",
|
|
|
|
"Session.AutoRaiseDelay", &value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%lu", &long_value) == 1) {
|
|
|
|
resource.auto_raise_delay.tv_usec = long_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
resource.cache_life = 5l;
|
|
|
|
if (XrmGetResource(database, "session.cacheLife", "Session.CacheLife",
|
|
|
|
&value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%lu", &long_value) == 1) {
|
|
|
|
resource.cache_life = long_value;
|
|
|
|
}
|
|
|
|
resource.cache_life *= 60000;
|
|
|
|
|
|
|
|
resource.cache_max = 200;
|
|
|
|
if (XrmGetResource(database, "session.cacheMax", "Session.CacheMax",
|
|
|
|
&value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%lu", &long_value) == 1) {
|
|
|
|
resource.cache_max = long_value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::load_rc(BScreen *screen) {
|
|
|
|
XrmDatabase database = (XrmDatabase) 0;
|
|
|
|
|
|
|
|
database = XrmGetFileDatabase(rc_file.c_str());
|
|
|
|
|
|
|
|
XrmValue value;
|
|
|
|
char *value_type, name_lookup[1024], class_lookup[1024];
|
|
|
|
int screen_number = screen->getScreenNumber();
|
|
|
|
int int_value;
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.fullMaximization", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.FullMaximization", screen_number);
|
|
|
|
screen->saveFullMax(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveFullMax(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.focusNewWindows", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.FocusNewWindows", screen_number);
|
|
|
|
screen->saveFocusNew(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveFocusNew(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.focusLastWindow", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.focusLastWindow", screen_number);
|
|
|
|
screen->saveFocusLast(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveFocusLast(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.rowPlacementDirection",
|
|
|
|
screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.RowPlacementDirection",
|
|
|
|
screen_number);
|
|
|
|
screen->saveRowPlacementDirection(BScreen::LeftRight);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "righttoleft", value.size)) {
|
|
|
|
screen->saveRowPlacementDirection(BScreen::RightLeft);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.colPlacementDirection",
|
|
|
|
screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.ColPlacementDirection",
|
|
|
|
screen_number);
|
|
|
|
screen->saveColPlacementDirection(BScreen::TopBottom);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "bottomtotop", value.size)) {
|
|
|
|
screen->saveColPlacementDirection(BScreen::BottomTop);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.workspaces", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Workspaces", screen_number);
|
|
|
|
screen->saveWorkspaces(1);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup,
|
|
|
|
&value_type, &value) &&
|
|
|
|
sscanf(value.addr, "%d", &int_value) == 1 &&
|
|
|
|
int_value > 0 && int_value < 128) {
|
|
|
|
screen->saveWorkspaces(int_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.toolbar.widthPercent",
|
|
|
|
screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Toolbar.WidthPercent",
|
|
|
|
screen_number);
|
|
|
|
screen->saveToolbarWidthPercent(66);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
sscanf(value.addr, "%d", &int_value) == 1 &&
|
|
|
|
int_value > 0 && int_value <= 100) {
|
|
|
|
screen->saveToolbarWidthPercent(int_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.toolbar.placement", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Toolbar.Placement", screen_number);
|
|
|
|
screen->saveToolbarPlacement(Toolbar::BottomCenter);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
if (! strncasecmp(value.addr, "TopLeft", value.size))
|
|
|
|
screen->saveToolbarPlacement(Toolbar::TopLeft);
|
|
|
|
else if (! strncasecmp(value.addr, "BottomLeft", value.size))
|
|
|
|
screen->saveToolbarPlacement(Toolbar::BottomLeft);
|
|
|
|
else if (! strncasecmp(value.addr, "TopCenter", value.size))
|
|
|
|
screen->saveToolbarPlacement(Toolbar::TopCenter);
|
|
|
|
else if (! strncasecmp(value.addr, "TopRight", value.size))
|
|
|
|
screen->saveToolbarPlacement(Toolbar::TopRight);
|
|
|
|
else if (! strncasecmp(value.addr, "BottomRight", value.size))
|
|
|
|
screen->saveToolbarPlacement(Toolbar::BottomRight);
|
|
|
|
}
|
|
|
|
screen->removeWorkspaceNames();
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.workspaceNames", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.WorkspaceNames", screen_number);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
string search = value.addr;
|
|
|
|
string::const_iterator it = search.begin(),
|
|
|
|
end = search.end();
|
|
|
|
while (1) {
|
|
|
|
string::const_iterator tmp = it; // current string.begin()
|
|
|
|
it = std::find(tmp, end, ','); // look for comma between tmp and end
|
|
|
|
screen->addWorkspaceName(string(tmp, it)); // string = search[tmp:it]
|
|
|
|
if (it == end) break;
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.toolbar.onTop", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Toolbar.OnTop", screen_number);
|
|
|
|
screen->saveToolbarOnTop(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveToolbarOnTop(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.toolbar.autoHide", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Toolbar.autoHide", screen_number);
|
|
|
|
screen->saveToolbarAutoHide(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveToolbarAutoHide(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.focusModel", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.FocusModel", screen_number);
|
|
|
|
screen->saveSloppyFocus(True);
|
|
|
|
screen->saveAutoRaise(False);
|
|
|
|
screen->saveClickRaise(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
string fmodel = value.addr;
|
|
|
|
|
|
|
|
if (fmodel.find("ClickToFocus") != string::npos) {
|
|
|
|
screen->saveSloppyFocus(False);
|
|
|
|
} else {
|
|
|
|
// must be sloppy
|
|
|
|
|
|
|
|
if (fmodel.find("AutoRaise") != string::npos)
|
|
|
|
screen->saveAutoRaise(True);
|
|
|
|
if (fmodel.find("ClickRaise") != string::npos)
|
|
|
|
screen->saveClickRaise(True);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.windowPlacement", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.WindowPlacement", screen_number);
|
|
|
|
screen->savePlacementPolicy(BScreen::RowSmartPlacement);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
if (! strncasecmp(value.addr, "RowSmartPlacement", value.size))
|
|
|
|
/* pass */;
|
|
|
|
else if (! strncasecmp(value.addr, "ColSmartPlacement", value.size))
|
|
|
|
screen->savePlacementPolicy(BScreen::ColSmartPlacement);
|
|
|
|
else if (! strncasecmp(value.addr, "CascadePlacement", value.size))
|
|
|
|
screen->savePlacementPolicy(BScreen::CascadePlacement);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.slit.placement", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Slit.Placement", screen_number);
|
|
|
|
screen->saveSlitPlacement(Slit::CenterRight);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
if (! strncasecmp(value.addr, "TopLeft", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::TopLeft);
|
|
|
|
else if (! strncasecmp(value.addr, "CenterLeft", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::CenterLeft);
|
|
|
|
else if (! strncasecmp(value.addr, "BottomLeft", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::BottomLeft);
|
|
|
|
else if (! strncasecmp(value.addr, "TopCenter", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::TopCenter);
|
|
|
|
else if (! strncasecmp(value.addr, "BottomCenter", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::BottomCenter);
|
|
|
|
else if (! strncasecmp(value.addr, "TopRight", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::TopRight);
|
|
|
|
else if (! strncasecmp(value.addr, "BottomRight", value.size))
|
|
|
|
screen->saveSlitPlacement(Slit::BottomRight);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.slit.direction", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Slit.Direction", screen_number);
|
|
|
|
screen->saveSlitDirection(Slit::Vertical);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "Horizontal", value.size)) {
|
|
|
|
screen->saveSlitDirection(Slit::Horizontal);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.slit.onTop", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Slit.OnTop", screen_number);
|
|
|
|
screen->saveSlitOnTop(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "True", value.size)) {
|
|
|
|
screen->saveSlitOnTop(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.slit.autoHide", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.Slit.AutoHide", screen_number);
|
|
|
|
screen->saveSlitAutoHide(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
! strncasecmp(value.addr, "true", value.size)) {
|
|
|
|
screen->saveSlitAutoHide(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef HAVE_STRFTIME
|
|
|
|
sprintf(name_lookup, "session.screen%d.strftimeFormat", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.StrftimeFormat", screen_number);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
screen->saveStrftimeFormat(value.addr);
|
|
|
|
} else {
|
|
|
|
screen->saveStrftimeFormat("%I:%M %p");
|
|
|
|
}
|
|
|
|
#else // HAVE_STRFTIME
|
|
|
|
sprintf(name_lookup, "session.screen%d.dateFormat", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.DateFormat", screen_number);
|
|
|
|
screen->saveDateFormat(B_AmericanDate);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value)) {
|
|
|
|
if (! strncasecmp(value.addr, "european", value.size))
|
|
|
|
screen->saveDateFormat(B_EuropeanDate);
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.clockFormat", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.ClockFormat", screen_number);
|
|
|
|
screen->saveClock24Hour(False);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
sscanf(value.addr, "%d", &int_value) == 1 && int_value == 24) {
|
|
|
|
screen->saveClock24Hour(True);
|
|
|
|
}
|
|
|
|
#endif // HAVE_STRFTIME
|
|
|
|
|
|
|
|
sprintf(name_lookup, "session.screen%d.edgeSnapThreshold", screen_number);
|
|
|
|
sprintf(class_lookup, "Session.Screen%d.EdgeSnapThreshold", screen_number);
|
|
|
|
if (XrmGetResource(database, name_lookup, class_lookup, &value_type,
|
|
|
|
&value) &&
|
|
|
|
sscanf(value.addr, "%d", &int_value) == 1) {
|
|
|
|
screen->saveEdgeSnapThreshold(int_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
screen->saveImageDither(True);
|
|
|
|
if (XrmGetResource(database, "session.imageDither", "Session.ImageDither",
|
|
|
|
&value_type, &value) &&
|
|
|
|
! strncasecmp("false", value.addr, value.size)) {
|
|
|
|
screen->saveImageDither(False);
|
|
|
|
}
|
|
|
|
|
|
|
|
screen->saveOpaqueMove(False);
|
|
|
|
if (XrmGetResource(database, "session.opaqueMove", "Session.OpaqueMove",
|
|
|
|
&value_type, &value) &&
|
|
|
|
! strncasecmp("true", value.addr, value.size)) {
|
|
|
|
screen->saveOpaqueMove(True);
|
|
|
|
}
|
|
|
|
|
|
|
|
XrmDestroyDatabase(database);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::reload_rc(void) {
|
|
|
|
load_rc();
|
|
|
|
reconfigure();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::reconfigure(void) {
|
|
|
|
reconfigure_wait = True;
|
|
|
|
|
|
|
|
if (! timer->isTiming()) timer->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::real_reconfigure(void) {
|
|
|
|
XrmDatabase new_blackboxrc = (XrmDatabase) 0;
|
|
|
|
char *style = new char[resource.style_file.length() + 20];
|
|
|
|
|
|
|
|
sprintf(style, "session.styleFile: %s", getStyleFilename());
|
|
|
|
XrmPutLineResource(&new_blackboxrc, style);
|
|
|
|
|
|
|
|
delete [] style;
|
|
|
|
|
|
|
|
XrmDatabase old_blackboxrc = XrmGetFileDatabase(rc_file.c_str());
|
|
|
|
|
|
|
|
XrmMergeDatabases(new_blackboxrc, &old_blackboxrc);
|
|
|
|
XrmPutFileDatabase(old_blackboxrc, rc_file.c_str());
|
|
|
|
if (old_blackboxrc) XrmDestroyDatabase(old_blackboxrc);
|
|
|
|
|
|
|
|
std::for_each(menuTimestamps.begin(), menuTimestamps.end(),
|
|
|
|
PointerAssassin());
|
|
|
|
menuTimestamps.clear();
|
|
|
|
|
|
|
|
gcCache()->purge();
|
|
|
|
|
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::reconfigure));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::checkMenu(void) {
|
|
|
|
bool reread = False;
|
|
|
|
MenuTimestampList::iterator it = menuTimestamps.begin();
|
|
|
|
for(; it != menuTimestamps.end(); ++it) {
|
|
|
|
MenuTimestamp *tmp = *it;
|
|
|
|
struct stat buf;
|
|
|
|
|
|
|
|
if (! stat(tmp->filename.c_str(), &buf)) {
|
|
|
|
if (tmp->timestamp != buf.st_ctime)
|
|
|
|
reread = True;
|
|
|
|
} else {
|
|
|
|
reread = True;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (reread) rereadMenu();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::rereadMenu(void) {
|
|
|
|
reread_menu_wait = True;
|
|
|
|
|
|
|
|
if (! timer->isTiming()) timer->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::real_rereadMenu(void) {
|
|
|
|
std::for_each(menuTimestamps.begin(), menuTimestamps.end(),
|
|
|
|
PointerAssassin());
|
|
|
|
menuTimestamps.clear();
|
|
|
|
|
|
|
|
std::for_each(screenList.begin(), screenList.end(),
|
|
|
|
std::mem_fun(&BScreen::rereadMenu));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::saveStyleFilename(const string& filename) {
|
|
|
|
assert(! filename.empty());
|
|
|
|
resource.style_file = filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-25 02:24:30 +00:00
|
|
|
void Blackbox::addMenuTimestamp(const string& filename) {
|
2002-05-25 00:29:44 +00:00
|
|
|
assert(! filename.empty());
|
|
|
|
bool found = False;
|
|
|
|
|
|
|
|
MenuTimestampList::iterator it = menuTimestamps.begin();
|
|
|
|
for (; it != menuTimestamps.end() && !found; ++it) {
|
|
|
|
if ((*it)->filename == filename) found = True;
|
|
|
|
}
|
|
|
|
if (! found) {
|
|
|
|
struct stat buf;
|
|
|
|
|
|
|
|
if (! stat(filename.c_str(), &buf)) {
|
|
|
|
MenuTimestamp *ts = new MenuTimestamp;
|
|
|
|
|
|
|
|
ts->filename = filename;
|
|
|
|
ts->timestamp = buf.st_ctime;
|
|
|
|
|
|
|
|
menuTimestamps.push_back(ts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::timeout(void) {
|
|
|
|
if (reconfigure_wait)
|
|
|
|
real_reconfigure();
|
|
|
|
|
|
|
|
if (reread_menu_wait)
|
|
|
|
real_rereadMenu();
|
|
|
|
|
|
|
|
reconfigure_wait = reread_menu_wait = False;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
XSetInputFocus(getXDisplay(),
|
|
|
|
active_screen->getToolbar()->getWindowID(),
|
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
} else {
|
|
|
|
// set input focus to the toolbar of the first managed screen
|
|
|
|
XSetInputFocus(getXDisplay(),
|
|
|
|
screenList.front()->getToolbar()->getWindowID(),
|
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// set input focus to the toolbar of the last screen
|
|
|
|
XSetInputFocus(getXDisplay(), old_screen->getToolbar()->getWindowID(),
|
|
|
|
RevertToPointerRoot, CurrentTime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (active_screen && active_screen->isScreenManaged()) {
|
|
|
|
active_screen->getToolbar()->redrawWindowLabel(True);
|
|
|
|
active_screen->updateNetizenWindowFocus();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_screen && old_screen != active_screen) {
|
|
|
|
old_screen->getToolbar()->redrawWindowLabel(True);
|
|
|
|
old_screen->updateNetizenWindowFocus();
|
|
|
|
}
|
|
|
|
}
|