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/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 "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"
|
2002-05-28 14:32:42 +00:00
|
|
|
#include "XAtom.hh"
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
// 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-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-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-26 20:25:38 +00:00
|
|
|
config.setFile(rc_file);
|
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;
|
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-06-21 20:40:14 +00:00
|
|
|
xatom = new XAtom(getXDisplay());
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
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);
|
2002-06-21 20:40:14 +00:00
|
|
|
cursor.ul_angle = XCreateFontCursor(getXDisplay(), XC_ul_angle);
|
|
|
|
cursor.ur_angle = XCreateFontCursor(getXDisplay(), XC_ur_angle);
|
2002-05-25 00:29:44 +00:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
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());
|
|
|
|
|
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
|
|
|
|
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);
|
|
|
|
|
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
|
|
|
|
|
|
|
if (focus && (win->isTransient() || win->getScreen()->doFocusNew()))
|
|
|
|
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;
|
|
|
|
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;
|
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);
|
|
|
|
} else if ((slit = searchSlit(e->xunmap.window))) {
|
|
|
|
slit->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;
|
|
|
|
Slit *slit = (Slit *) 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 ((slit = searchSlit(e->xdestroywindow.window))) {
|
|
|
|
slit->removeClient(e->xdestroywindow.window, False);
|
|
|
|
} 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);
|
|
|
|
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;
|
|
|
|
|
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;
|
|
|
|
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() &&
|
2002-07-16 08:00:59 +00:00
|
|
|
(! win->isFocused()) && (! no_focus) &&
|
|
|
|
win->isNormal()) { // don't focus non-normal windows with mouseover
|
2002-05-25 00:29:44 +00:00
|
|
|
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: {
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
2002-05-30 06:51:43 +00:00
|
|
|
if (e->xclient.message_type == xatom->getAtom(XAtom::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 ==
|
|
|
|
xatom->getAtom(XAtom::blackbox_change_workspace) ||
|
|
|
|
e->xclient.message_type ==
|
|
|
|
xatom->getAtom(XAtom::net_current_desktop)) {
|
|
|
|
// 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-07-16 04:15:24 +00:00
|
|
|
xatom->getAtom(XAtom::blackbox_change_window_focus)) {
|
|
|
|
// 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-06-21 20:40:14 +00:00
|
|
|
xatom->getAtom(XAtom::net_active_window)) {
|
|
|
|
// 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())
|
|
|
|
win->deiconify(False, True);
|
2002-07-11 23:44:04 +00:00
|
|
|
if (win->isShaded())
|
|
|
|
win->shade();
|
2002-07-16 04:18:13 +00:00
|
|
|
if (win->getWorkspaceNumber() != screen->getCurrentWorkspaceID())
|
|
|
|
screen->changeWorkspaceID(win->getWorkspaceNumber());
|
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 ==
|
|
|
|
xatom->getAtom(XAtom::blackbox_cycle_window_focus)) {
|
|
|
|
// 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 ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_desktop)) {
|
|
|
|
// 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 ==
|
|
|
|
xatom->getAtom(XAtom::blackbox_change_attributes)) {
|
|
|
|
// 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 ==
|
|
|
|
xatom->getAtom(XAtom::net_number_of_desktops)) {
|
|
|
|
// 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 ==
|
|
|
|
xatom->getAtom(XAtom::net_close_window)) {
|
|
|
|
// 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 ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize)) {
|
|
|
|
// 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] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize_move)) {
|
|
|
|
win->beginMove(x_root, y_root);
|
|
|
|
} else {
|
|
|
|
if ((Atom) e->xclient.data.l[2] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize_size_topleft))
|
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::TopLeft);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize_size_topright))
|
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::TopRight);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize_size_bottomleft))
|
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::BottomLeft);
|
|
|
|
else if ((Atom) e->xclient.data.l[2] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_moveresize_size_bottomright))
|
|
|
|
win->beginResize(x_root, y_root, BlackboxWindow::BottomRight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (e->xclient.message_type ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state)) {
|
|
|
|
// 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
|
|
|
|
if (state[i] == xatom->getAtom(XAtom::net_wm_state_modal)) {
|
|
|
|
win->setModal(True);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_vert)) {
|
|
|
|
if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else if (! win->isMaximized()) {
|
|
|
|
win->maximize(2); // vert
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_horz)) {
|
|
|
|
if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(1); // full
|
|
|
|
} else if (! win->isMaximized()) {
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_shaded)) {
|
|
|
|
if (! win->isShaded())
|
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_taskbar)) {
|
|
|
|
win->setSkipTaskbar(True);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_pager)) {
|
|
|
|
win->setSkipPager(True);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_fullscreen)) {
|
|
|
|
win->setFullscreen(True);
|
|
|
|
}
|
|
|
|
} else if (action == 0) {
|
|
|
|
// REMOVE
|
|
|
|
if (state[i] == xatom->getAtom(XAtom::net_wm_state_modal)) {
|
|
|
|
win->setModal(False);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_vert)) {
|
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(3); // horiz
|
|
|
|
} else if (win->isMaximizedVert()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_horz)) {
|
|
|
|
if (win->isMaximizedFull()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
win->maximize(2); // vert
|
|
|
|
} else if (win->isMaximizedHoriz()) {
|
|
|
|
win->maximize(0); // unmaximize
|
|
|
|
}
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_shaded)) {
|
|
|
|
if (win->isShaded())
|
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_taskbar)) {
|
|
|
|
win->setSkipTaskbar(False);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_pager)) {
|
|
|
|
win->setSkipPager(False);
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_fullscreen)) {
|
|
|
|
win->setFullscreen(False);
|
|
|
|
}
|
|
|
|
} else if (action == 2) {
|
|
|
|
// TOGGLE
|
|
|
|
if (state[i] == xatom->getAtom(XAtom::net_wm_state_modal)) {
|
|
|
|
win->setModal(! win->isModal());
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_vert)) {
|
|
|
|
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] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_maximized_horz)) {
|
|
|
|
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] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_shaded)) {
|
|
|
|
win->shade();
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_taskbar)) {
|
|
|
|
win->setSkipTaskbar(! win->skipTaskbar());
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_skip_pager)) {
|
|
|
|
win->setSkipPager(! win->skipPager());
|
|
|
|
} else if (state[i] ==
|
|
|
|
xatom->getAtom(XAtom::net_wm_state_fullscreen)) {
|
|
|
|
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);
|
|
|
|
|
|
|
|
if (win)
|
|
|
|
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:
|
|
|
|
rereadMenu();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SIGPIPE:
|
|
|
|
case SIGSEGV:
|
|
|
|
case SIGFPE:
|
|
|
|
case SIGINT:
|
|
|
|
case SIGTERM:
|
|
|
|
shutdown();
|
|
|
|
|
|
|
|
default:
|
|
|
|
return False;
|
|
|
|
}
|
|
|
|
|
|
|
|
return True;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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::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) {
|
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);
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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-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-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::reconfigure(void) {
|
|
|
|
reconfigure_wait = True;
|
|
|
|
|
|
|
|
if (! timer->isTiming()) timer->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Blackbox::real_reconfigure(void) {
|
2002-05-26 20:25:38 +00:00
|
|
|
load_rc();
|
|
|
|
|
2002-05-25 00:29:44 +00:00
|
|
|
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-26 20:25:38 +00:00
|
|
|
config.setValue("session.styleFile", resource.style_file);
|
2002-05-25 00:29:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
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();
|
2002-05-26 20:25:38 +00:00
|
|
|
for (; it != menuTimestamps.end() && ! found; ++it) {
|
2002-05-25 00:29:44 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
|
XSetInputFocus(getXDisplay(),
|
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
|
|
|
|
XSetInputFocus(getXDisplay(),
|
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-05-28 10:15:08 +00:00
|
|
|
XSetInputFocus(getXDisplay(), old_screen->getRootWindow(),
|
2002-05-25 00:29:44 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
}
|