fluxbox/src/Screen.cc

1841 lines
55 KiB
C++
Raw Normal View History

// Screen.cc for Fluxbox Window Manager
2006-02-16 06:53:05 +00:00
// Copyright (c) 2001 - 2006 Henrik Kinnunen (fluxgen at fluxbox dot org)
//
2001-12-11 20:47:02 +00:00
// Screen.cc for Blackbox - an X11 Window manager
2002-12-13 20:19:05 +00:00
// Copyright (c) 1997 - 2000 Brad Hughes (bhughes at tcac.net)
2001-12-11 20:47:02 +00:00
//
// 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,
2002-05-08 10:14:51 +00:00
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2001-12-11 20:47:02 +00:00
// 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.
2002-10-25 20:58:14 +00:00
#include "Screen.hh"
#include "fluxbox.hh"
#include "Keys.hh"
2002-10-25 20:58:14 +00:00
#include "Window.hh"
2007-10-15 18:05:28 +00:00
#include "WindowCmd.hh"
2002-10-25 20:58:14 +00:00
#include "Workspace.hh"
2006-03-16 18:25:33 +00:00
#include "Layer.hh"
#include "FocusControl.hh"
#include "ScreenPlacement.hh"
// menus
#include "ConfigMenu.hh"
#include "FbMenu.hh"
#include "LayerMenu.hh"
#include "MenuCreator.hh"
2003-04-14 15:01:55 +00:00
#include "WinClient.hh"
2003-05-15 11:17:29 +00:00
#include "FbWinFrame.hh"
2003-06-18 13:42:21 +00:00
#include "Strut.hh"
#include "FbTk/CommandParser.hh"
#include "AtomHandler.hh"
#include "HeadArea.hh"
#include "RectangleUtil.hh"
#include "FbCommands.hh"
#ifdef USE_SYSTRAY
#include "SystemTray.hh"
#endif
#include "Debug.hh"
2004-06-07 11:46:05 +00:00
#include "FbTk/I18n.hh"
#include "FbTk/FbWindow.hh"
#include "FbTk/SimpleCommand.hh"
#include "FbTk/MultLayers.hh"
#include "FbTk/LayerItem.hh"
#include "FbTk/MacroCommand.hh"
#include "FbTk/StringUtil.hh"
#include "FbTk/ImageControl.hh"
#include "FbTk/EventManager.hh"
#include "FbTk/Transparent.hh"
2005-05-02 12:10:01 +00:00
#include "FbTk/Select2nd.hh"
#include "FbTk/Compose.hh"
#include "FbTk/FbString.hh"
2007-12-28 06:22:38 +00:00
#include "FbTk/STLUtil.hh"
2008-01-02 08:31:18 +00:00
#include "FbTk/KeyUtil.hh"
#include "FbTk/Util.hh"
#ifdef USE_SLIT
2001-12-11 20:47:02 +00:00
#include "Slit.hh"
#include "SlitClient.hh"
2003-10-28 17:39:59 +00:00
#else
// fill it in
class Slit {};
#endif // USE_SLIT
2001-12-11 20:47:02 +00:00
#ifdef USE_TOOLBAR
#include "Toolbar.hh"
#else
class Toolbar {};
#endif // USE_TOOLBAR
#ifdef STDC_HEADERS
2002-10-25 20:58:14 +00:00
#include <sys/types.h>
2001-12-11 20:47:02 +00:00
#endif // STDC_HEADERS
#ifdef HAVE_UNISTD_H
2002-10-25 20:58:14 +00:00
#include <sys/types.h>
#include <unistd.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_UNISTD_H
2002-11-27 21:55:36 +00:00
#ifdef TIME_WITH_SYS_TIME
#include <sys/time.h>
#include <time.h>
#else // !TIME_WITH_SYS_TIME
#ifdef HAVE_SYS_TIME_H
2002-11-27 21:55:36 +00:00
#include <sys/time.h>
#else // !HAVE_SYS_TIME_H
#include <time.h>
#endif // HAVE_SYS_TIME_H
#endif // TIME_WITH_SYS_TIME
2002-01-27 12:45:32 +00:00
#include <X11/Xatom.h>
#include <X11/keysym.h>
2003-06-12 15:24:37 +00:00
#include <X11/cursorfont.h>
2002-01-27 12:45:32 +00:00
#ifdef XINERAMA
extern "C" {
#include <X11/extensions/Xinerama.h>
}
#endif // XINERAMA
#if defined(HAVE_RANDR) || defined(HAVE_RANDR1_2)
#include <X11/extensions/Xrandr.h>
#endif // HAVE_RANDR
2001-12-11 20:47:02 +00:00
#include <iostream>
2002-02-08 13:35:20 +00:00
#include <algorithm>
2003-04-14 15:01:55 +00:00
#include <functional>
#include <stack>
#include <cstdarg>
#include <cstring>
2002-01-20 02:17:23 +00:00
using std::cerr;
using std::endl;
using std::string;
using std::make_pair;
using std::pair;
using std::list;
using std::vector;
using std::mem_fun;
using std::bind2nd;
using std::equal_to;
using std::hex;
using std::dec;
2001-12-11 20:47:02 +00:00
2002-04-04 14:28:54 +00:00
static bool running = true;
2002-10-25 20:58:14 +00:00
namespace {
2001-12-11 20:47:02 +00:00
2002-10-25 20:58:14 +00:00
int anotherWMRunning(Display *display, XErrorEvent *) {
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
cerr<<_FB_CONSOLETEXT(Screen, AnotherWMRunning,
2004-06-07 11:46:05 +00:00
"BScreen::BScreen: an error occured while querying the X server.\n"
" another window manager already running on display ",
"Message when another WM is found already active on all screens")
<<DisplayString(display)<<endl;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
running = false;
2001-12-11 20:47:02 +00:00
2003-05-17 11:05:33 +00:00
return -1;
2001-12-11 20:47:02 +00:00
}
int calcSquareDistance(int x1, int y1, int x2, int y2) {
return (x2-x1)*(x2-x1) + (y2-y1)*(y2-y1);
}
void clampMenuDelay(int& delay) {
delay = FbTk::Util::clamp(delay, 0, 5000);
}
Atom atom_fbcmd = 0;
Atom atom_wm_check = 0;
Atom atom_net_desktop = 0;
Atom atom_utf8_string = 0;
Atom atom_kde_systray = 0;
Atom atom_kwm1 = 0;
void initAtoms(Display* dpy) {
atom_wm_check = XInternAtom(dpy, "_NET_SUPPORTING_WM_CHECK", False);
atom_net_desktop = XInternAtom(dpy, "_NET_CURRENT_DESKTOP", False);
atom_fbcmd = XInternAtom(dpy, "_FLUXBOX_ACTION", False);
atom_utf8_string = XInternAtom(dpy, "UTF8_STRING", False);
atom_kde_systray = XInternAtom(dpy, "_KDE_NET_WM_SYSTEM_TRAY_WINDOW_FOR", False);
atom_kwm1 = XInternAtom(dpy, "KWM_DOCKWINDOW", False);
}
} // end anonymous namespace
2002-01-20 02:17:23 +00:00
2006-03-22 12:23:17 +00:00
2002-01-20 02:17:23 +00:00
BScreen::BScreen(FbTk::ResourceManager &rm,
const string &screenname,
const string &altscreenname,
int scrn, int num_layers,
unsigned int opts) :
2003-04-28 00:38:42 +00:00
m_layermanager(num_layers),
root_colormap_installed(false),
m_current_workspace(0),
m_focused_windowtheme(new FbWinFrameTheme(scrn, ".focus", ".Focus")),
m_unfocused_windowtheme(new FbWinFrameTheme(scrn, ".unfocus", ".Unfocus")),
// the order of windowtheme and winbutton theme is important
// because winbutton need to rescale the pixmaps in winbutton theme
// after fbwinframe have resized them
m_focused_winbutton_theme(new WinButtonTheme(scrn, "", "", *m_focused_windowtheme)),
m_unfocused_winbutton_theme(new WinButtonTheme(scrn, ".unfocus", ".Unfocus", *m_unfocused_windowtheme)),
m_pressed_winbutton_theme(new WinButtonTheme(scrn, ".pressed", ".Pressed", *m_focused_windowtheme)),
2007-12-28 07:16:17 +00:00
m_menutheme(new FbTk::MenuTheme(scrn)),
2003-05-10 22:59:32 +00:00
m_root_window(scrn),
m_geom_window(new OSDWindow(m_root_window, *this, *m_focused_windowtheme)),
m_pos_window(new OSDWindow(m_root_window, *this, *m_focused_windowtheme)),
m_tooltip_window(new TooltipWindow(m_root_window, *this, *m_focused_windowtheme)),
m_dummy_window(scrn, -1, -1, 1, 1, 0, true, false, CopyFromParent, InputOnly),
resource(rm, screenname, altscreenname),
m_resource_manager(rm),
2003-06-20 01:30:08 +00:00
m_name(screenname),
m_altname(altscreenname),
m_focus_control(new FocusControl(*this)),
m_placement_strategy(new ScreenPlacement(*this)),
m_opts(opts) {
2003-04-28 00:38:42 +00:00
2003-12-29 01:23:04 +00:00
m_state.cycling = false;
m_state.restart = false;
m_state.shutdown = false;
m_state.managed = false;
Fluxbox *fluxbox = Fluxbox::instance();
Display *disp = fluxbox->display();
initAtoms(disp);
// Create the first one, initXinerama will expand this if needed.
m_head_areas.resize(1);
m_head_areas[0] = new HeadArea();
2003-05-19 15:32:47 +00:00
initXinerama();
2003-06-20 01:48:06 +00:00
// setup error handler to catch "screen already managed by other wm"
2002-12-01 13:42:15 +00:00
XErrorHandler old = XSetErrorHandler((XErrorHandler) anotherWMRunning);
2003-06-12 15:24:37 +00:00
rootWindow().setEventMask(ColormapChangeMask | EnterWindowMask | PropertyChangeMask |
SubstructureRedirectMask | KeyPressMask | KeyReleaseMask |
ButtonPressMask | ButtonReleaseMask| SubstructureNotifyMask);
fluxbox->sync(false);
2003-06-12 15:24:37 +00:00
2002-12-01 13:42:15 +00:00
XSetErrorHandler((XErrorHandler) old);
2001-12-11 20:47:02 +00:00
m_state.managed = running;
if (!m_state.managed) {
2006-04-22 15:37:04 +00:00
delete m_placement_strategy; m_placement_strategy = 0;
delete m_focus_control; m_focus_control = 0;
2002-12-01 13:42:15 +00:00
return;
2006-04-22 15:37:04 +00:00
}
// we're going to manage the screen, so now add our pid
#ifdef HAVE_GETPID
unsigned long bpid = static_cast<unsigned long>(getpid());
rootWindow().changeProperty(fluxbox->getFluxboxPidAtom(), XA_CARDINAL,
sizeof(pid_t) * 8, PropModeReplace,
(unsigned char *) &bpid, 1);
#endif // HAVE_GETPID
// check if we're the first EWMH compliant window manager on this screen
union { Atom atom; unsigned long ul; int i; } ignore;
unsigned char *ret_prop;
if (rootWindow().property(atom_wm_check, 0l, 1l,
False, XA_WINDOW, &ignore.atom, &ignore.i, &ignore.ul,
&ignore.ul, &ret_prop) ) {
m_state.restart = (ret_prop != NULL);
XFree(ret_prop);
}
// setup RANDR for this screens root window
#if defined(HAVE_RANDR)
int randr_mask = RRScreenChangeNotifyMask;
# ifdef RRCrtcChangeNotifyMask
randr_mask |= RRCrtcChangeNotifyMask;
# endif
# ifdef RROutputChangeNotifyMask
randr_mask |= RROutputChangeNotifyMask;
# endif
# ifdef RROutputPropertyNotifyMask
randr_mask |= RROutputPropertyNotifyMask;
# endif
XRRSelectInput(disp, rootWindow().window(), randr_mask);
#endif // HAVE_RANDR
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
2008-04-25 13:01:43 +00:00
#ifdef DEBUG
fprintf(stderr, _FB_CONSOLETEXT(Screen, ManagingScreen,
2004-06-07 11:46:05 +00:00
"BScreen::BScreen: managing screen %d "
"using visual 0x%lx, depth %d\n",
"informational message saying screen number (%d), visual (%lx), and colour depth (%d)").c_str(),
2003-05-15 12:00:46 +00:00
screenNumber(), XVisualIDFromVisual(rootWindow().visual()),
rootWindow().maxDepth());
2008-04-25 13:01:43 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
FbTk::EventManager *evm = FbTk::EventManager::instance();
evm->add(*this, rootWindow());
Keys *keys = fluxbox->keys();
2007-10-13 21:51:37 +00:00
if (keys)
keys->registerWindow(rootWindow().window(), *this,
2007-10-13 21:51:37 +00:00
Keys::GLOBAL|Keys::ON_DESKTOP);
2003-06-12 15:24:37 +00:00
rootWindow().setCursor(XCreateFontCursor(disp, XC_left_ptr));
2001-12-11 20:47:02 +00:00
// load this screens resources
fluxbox->load_rc(*this);
2003-06-20 01:48:06 +00:00
// setup image cache engine
2008-06-26 19:41:43 +00:00
m_image_control.reset(new FbTk::ImageControl(scrn,
fluxbox->colorsPerChannel(),
2003-06-12 15:24:37 +00:00
fluxbox->getCacheLife(), fluxbox->getCacheMax()));
2003-05-15 23:30:07 +00:00
imageControl().installRootColormap();
2002-12-01 13:42:15 +00:00
root_colormap_installed = true;
2001-12-11 20:47:02 +00:00
2007-03-10 18:06:09 +00:00
m_root_theme.reset(new RootTheme(imageControl()));
m_root_theme->reconfigTheme();
focusedWinFrameTheme()->setAlpha(*resource.focused_alpha);
unfocusedWinFrameTheme()->setAlpha(*resource.unfocused_alpha);
m_menutheme->setAlpha(*resource.menu_alpha);
clampMenuDelay(*resource.menu_delay);
2003-12-18 15:27:21 +00:00
m_menutheme->setDelay(*resource.menu_delay);
m_tracker.join(focusedWinFrameTheme()->reconfigSig(),
FbTk::MemFun(*this, &BScreen::focusedWinFrameThemeReconfigured));
2003-08-25 13:15:53 +00:00
renderGeomWindow();
renderPosWindow();
m_tooltip_window->setDelay(*resource.tooltip_delay);
2003-06-20 01:48:06 +00:00
// setup workspaces and workspace menu
int nr_ws = *resource.workspaces;
addWorkspace(); // at least one
for (int i = 1; i < nr_ws; ++i) {
addWorkspace();
2002-12-01 13:42:15 +00:00
}
2003-05-19 22:45:51 +00:00
m_current_workspace = m_workspaces_list.front();
m_windowmenu.reset(MenuCreator::createMenu("", *this));
m_windowmenu->setInternalMenu();
m_windowmenu->setReloadHelper(new FbTk::AutoReloadHelper());
m_windowmenu->reloadHelper()->setReloadCmd(FbTk::RefCount<FbTk::Command<void> >(new FbTk::SimpleCommand<BScreen>(*this, &BScreen::rereadWindowMenu)));
m_rootmenu.reset(MenuCreator::createMenu("", *this));
m_rootmenu->setReloadHelper(new FbTk::AutoReloadHelper());
m_rootmenu->reloadHelper()->setReloadCmd(FbTk::RefCount<FbTk::Command<void> >(new FbTk::SimpleCommand<BScreen>(*this, &BScreen::rereadMenu)));
m_configmenu.reset(MenuCreator::createMenu(_FB_XTEXT(Menu, Configuration,
"Configuration", "Title of configuration menu"), *this));
2003-04-18 12:51:14 +00:00
m_configmenu->setInternalMenu();
setupConfigmenu(*m_configmenu.get());
2007-01-14 19:05:34 +00:00
// check which desktop we should start on
2015-01-21 09:02:38 +00:00
int first_desktop = 0;
if (m_state.restart) {
bool exists;
2015-01-21 09:02:38 +00:00
int ret = (rootWindow().cardinalProperty(atom_net_desktop, &exists));
if (exists) {
2015-01-21 09:02:38 +00:00
first_desktop = FbTk::Util::clamp<int>(ret, 0, nr_ws);
2007-01-14 19:05:34 +00:00
}
}
changeWorkspaceID(first_desktop);
2003-12-31 00:38:40 +00:00
#ifdef USE_SLIT
if (opts & Fluxbox::OPT_SLIT) {
Slit* slit = new Slit(*this, *layerManager().getLayer(ResourceLayer::DESKTOP), fluxbox->getSlitlistFilename().c_str());
m_slit.reset(slit);
}
#endif // USE_SLIT
2004-06-07 21:22:42 +00:00
rm.unlock();
XFlush(disp);
}
BScreen::~BScreen() {
if (!m_state.managed)
return;
m_toolbar.reset(0);
m_toolButtonMap.reset(0);
FbTk::EventManager *evm = FbTk::EventManager::instance();
evm->remove(rootWindow());
2007-10-13 21:51:37 +00:00
Keys *keys = Fluxbox::instance()->keys();
if (keys)
keys->unregisterWindow(rootWindow().window());
if (m_rootmenu.get() != 0)
m_rootmenu->removeAll();
// Since workspacemenu holds client list menus (from workspace)
// we need to destroy it before we destroy workspaces
m_workspacemenu.reset(0);
removeWorkspaceNames();
2007-12-28 06:22:38 +00:00
using namespace FbTk::STLUtil;
destroyAndClear(m_workspaces_list);
destroyAndClear(m_managed_resources);
//why not destroyAndClear(m_icon_list); ?
//problem with that: a delete FluxboxWindow* calls m_diesig.notify()
//which leads to screen.removeWindow() which leads to removeIcon(win)
//which would modify the m_icon_list anyways...
Icons tmp;
tmp = m_icon_list;
while(!tmp.empty()) {
removeWindow(tmp.back());
tmp.back()->restore(true);
delete (tmp.back());
tmp.pop_back();
}
if (hasXinerama()) {
m_xinerama.heads.clear();
}
// slit must be destroyed before headAreas (Struts)
m_slit.reset(0);
m_windowmenu.reset(0);
m_rootmenu.reset(0);
m_workspacemenu.reset(0);
m_configmenu.reset(0);
// TODO fluxgen: check if this is the right place
2008-01-28 09:46:05 +00:00
for (size_t i = 0; i < m_head_areas.size(); i++)
delete m_head_areas[i];
2006-02-19 08:11:10 +00:00
delete m_focus_control;
delete m_placement_strategy;
}
bool BScreen::isRestart() {
return Fluxbox::instance()->isStartup() && m_state.restart;
}
void BScreen::initWindows() {
#ifdef USE_TOOLBAR
if (m_opts & Fluxbox::OPT_TOOLBAR) {
m_toolButtonMap.reset(new ToolButtonMap());
Toolbar* tb = new Toolbar(*this, *layerManager().getLayer(::ResourceLayer::NORMAL));
m_toolbar.reset(tb);
}
#endif // USE_TOOLBAR
2002-12-01 13:42:15 +00:00
unsigned int nchild;
Window r, p, *children;
Fluxbox* fluxbox = Fluxbox::instance();
Display* disp = fluxbox->display();
2003-05-10 22:59:32 +00:00
XQueryTree(disp, rootWindow().window(), &r, &p, &children, &nchild);
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
// preen the window list of all icon windows... for better dockapp support
for (unsigned int i = 0; i < nchild; i++) {
if (children[i] == None)
continue;
2002-12-01 13:42:15 +00:00
XWMHints *wmhints = XGetWMHints(disp, children[i]);
2002-12-01 13:42:15 +00:00
if (wmhints) {
if ((wmhints->flags & IconWindowHint) &&
(wmhints->icon_window != children[i]))
for (unsigned int j = 0; j < nchild; j++) {
2002-12-01 13:42:15 +00:00
if (children[j] == wmhints->icon_window) {
fbdbg<<"BScreen::initWindows(): children[j] = 0x"<<hex<<children[j]<<dec<<endl;
fbdbg<<"BScreen::initWindows(): = icon_window"<<endl;
2002-12-01 13:42:15 +00:00
children[j] = None;
break;
}
}
XFree(wmhints);
}
}
2002-12-01 13:42:15 +00:00
// manage shown windows
Window transient_for = 0;
bool safety_flag = false;
unsigned int num_transients = 0;
for (unsigned int i = 0; i <= nchild; ++i) {
if (i == nchild) {
if (num_transients) {
if (num_transients == nchild)
safety_flag = true;
nchild = num_transients;
i = num_transients = 0;
} else
break;
}
if (children[i] == None)
continue;
else if (!fluxbox->validateWindow(children[i])) {
fbdbg<<"BScreen::initWindows(): not valid window = "<<hex<<children[i]<<dec<<endl;
children[i] = None;
continue;
}
2002-12-01 13:42:15 +00:00
// if we have a transient_for window and it isn't created yet...
// postpone creation of this window until after all others
if (XGetTransientForHint(disp, children[i], &transient_for) &&
fluxbox->searchWindow(transient_for) == 0 && !safety_flag) {
// add this window back to the beginning of the list of children
children[num_transients] = children[i];
num_transients++;
fbdbg<<"BScreen::initWindows(): postpone creation of 0x"<<hex<<children[i]<<dec<<endl;
fbdbg<<"BScreen::initWindows(): transient_for = 0x"<<hex<<transient_for<<dec<<endl;
continue;
}
XWindowAttributes attrib;
if (XGetWindowAttributes(disp, children[i],
&attrib)) {
if (attrib.override_redirect) {
children[i] = None; // we dont need this anymore, since we already created a window for it
continue;
}
if (attrib.map_state != IsUnmapped)
createWindow(children[i]);
2002-12-01 13:42:15 +00:00
}
children[i] = None; // we dont need this anymore, since we already created a window for it
2002-12-01 13:42:15 +00:00
}
XFree(children);
// now, show slit and toolbar
#ifdef USE_SLIT
if (slit())
slit()->show();
#endif // USE_SLIT
2001-12-11 20:47:02 +00:00
}
unsigned int BScreen::currentWorkspaceID() const {
return m_current_workspace->workspaceID();
}
2003-04-25 11:56:13 +00:00
const Strut* BScreen::availableWorkspaceArea(int head) const {
if (head > numHeads()) {
/* May this ever happen? */
static Strut whole(-1 /* should never be used */, 0, width(), 0, height());
return &whole;
}
return m_head_areas[head ? head-1 : 0]->availableWorkspaceArea();
}
unsigned int BScreen::maxLeft(int head) const {
// we ignore strut if we're doing full maximization
if (hasXinerama())
return doFullMax() ? getHeadX(head) :
getHeadX(head) + availableWorkspaceArea(head)->left();
else
return doFullMax() ? 0 : availableWorkspaceArea(head)->left();
}
unsigned int BScreen::maxRight(int head) const {
// we ignore strut if we're doing full maximization
if (hasXinerama())
return doFullMax() ? getHeadX(head) + getHeadWidth(head) :
getHeadX(head) + getHeadWidth(head) - availableWorkspaceArea(head)->right();
else
return doFullMax() ? width() : width() - availableWorkspaceArea(head)->right();
}
unsigned int BScreen::maxTop(int head) const {
// we ignore strut if we're doing full maximization
if (hasXinerama())
return doFullMax() ? getHeadY(head) : getHeadY(head) + availableWorkspaceArea(head)->top();
else
return doFullMax() ? 0 : availableWorkspaceArea(head)->top();
}
2003-05-19 22:45:51 +00:00
unsigned int BScreen::maxBottom(int head) const {
// we ignore strut if we're doing full maximization
if (hasXinerama())
return doFullMax() ? getHeadY(head) + getHeadHeight(head) :
getHeadY(head) + getHeadHeight(head) - availableWorkspaceArea(head)->bottom();
else
return doFullMax() ? height() : height() - availableWorkspaceArea(head)->bottom();
}
void BScreen::focusedWinFrameThemeReconfigured() {
2003-08-25 13:15:53 +00:00
renderGeomWindow();
renderPosWindow();
2006-06-26 11:45:14 +00:00
Fluxbox *fluxbox = Fluxbox::instance();
2007-12-21 07:35:36 +00:00
const std::list<Focusable *> winlist =
focusControl().focusedOrderWinList().clientList();
std::list<Focusable *>::const_iterator it = winlist.begin();
std::list<Focusable *>::const_iterator it_end = winlist.end();
2006-06-26 11:45:14 +00:00
for (; it != it_end; ++it)
2007-12-21 07:35:36 +00:00
fluxbox->updateFrameExtents(*(*it)->fbwindow());
2003-08-25 13:15:53 +00:00
}
void BScreen::propertyNotify(Atom atom) {
if (allowRemoteActions() && atom == atom_fbcmd) {
Atom xa_ret_type;
int ret_format;
unsigned long ret_nitems, ret_bytes_after;
char *str;
if (rootWindow().property(atom_fbcmd, 0l, 64l,
True, XA_STRING, &xa_ret_type, &ret_format, &ret_nitems,
&ret_bytes_after, (unsigned char **)&str) && str) {
if (ret_bytes_after) {
XFree(str);
long len = 64 + (ret_bytes_after + 3)/4;
rootWindow().property(atom_fbcmd, 0l, len,
True, XA_STRING, &xa_ret_type, &ret_format, &ret_nitems,
&ret_bytes_after, (unsigned char **)&str);
}
static std::unique_ptr<FbTk::Command<void> > cmd;
cmd.reset(FbTk::CommandParser<void>::instance().parse(str, false));
if (cmd.get()) {
cmd->execute();
}
XFree(str);
}
// TODO: this doesn't belong in FbPixmap
} else if (FbTk::FbPixmap::rootwinPropertyNotify(screenNumber(), atom))
m_bg_change_sig.emit(*this);
}
void BScreen::keyPressEvent(XKeyEvent &ke) {
if (Fluxbox::instance()->keys()->doAction(ke.type, ke.state, ke.keycode,
Keys::GLOBAL|(ke.subwindow ? 0 : Keys::ON_DESKTOP))) {
// re-grab keyboard, so we don't pass KeyRelease to clients
// also for catching invalid keys in the middle of keychains
FbTk::EventManager::instance()->grabKeyboard(rootWindow().window());
XAllowEvents(Fluxbox::instance()->display(), SyncKeyboard, CurrentTime);
} else {
XAllowEvents(Fluxbox::instance()->display(), ReplayKeyboard, CurrentTime);
}
}
void BScreen::keyReleaseEvent(XKeyEvent &ke) {
if (m_state.cycling) {
unsigned int state = FbTk::KeyUtil::instance().cleanMods(ke.state);
state &= ~FbTk::KeyUtil::instance().keycodeToModmask(ke.keycode);
if (state) // still cycling
return;
m_state.cycling = false;
focusControl().stopCyclingFocus();
}
if (!Fluxbox::instance()->keys()->inKeychain())
FbTk::EventManager::instance()->ungrabKeyboard();
}
void BScreen::buttonPressEvent(XButtonEvent &be) {
if (be.button == 1 && !isRootColormapInstalled())
imageControl().installRootColormap();
Keys *keys = Fluxbox::instance()->keys();
2007-12-09 20:47:41 +00:00
keys->doAction(be.type, be.state, be.button, Keys::GLOBAL|Keys::ON_DESKTOP,
2008-02-11 18:17:45 +00:00
0, be.time);
}
2007-10-13 21:51:37 +00:00
void BScreen::cycleFocus(int options, const ClientPattern *pat, bool reverse) {
// get modifiers from event that causes this for focus order cycling
XEvent ev = Fluxbox::instance()->lastEvent();
unsigned int mods = 0;
if (ev.type == KeyPress)
mods = FbTk::KeyUtil::instance().cleanMods(ev.xkey.state);
else if (ev.type == ButtonPress)
mods = FbTk::KeyUtil::instance().cleanMods(ev.xbutton.state);
if (!m_state.cycling && mods) {
m_state.cycling = true;
FbTk::EventManager::instance()->grabKeyboard(rootWindow().window());
}
if (mods == 0) // can't stacked cycle unless there is a mod to grab
options |= FocusableList::STATIC_ORDER;
2007-10-13 21:51:37 +00:00
const FocusableList *win_list =
FocusableList::getListFromOptions(*this, options);
2007-10-13 21:51:37 +00:00
focusControl().cycleFocus(*win_list, pat, reverse);
}
void BScreen::reconfigure() {
Fluxbox *fluxbox = Fluxbox::instance();
focusedWinFrameTheme()->setAlpha(*resource.focused_alpha);
unfocusedWinFrameTheme()->setAlpha(*resource.unfocused_alpha);
m_menutheme->setAlpha(*resource.menu_alpha);
2003-12-18 15:27:21 +00:00
clampMenuDelay(*resource.menu_delay);
2003-12-18 15:27:21 +00:00
m_menutheme->setDelay(*resource.menu_delay);
2003-12-12 18:18:49 +00:00
// provide the number of workspaces from the init-file
const unsigned int nr_ws = *resource.workspaces;
if (nr_ws > m_workspaces_list.size()) {
while(nr_ws != m_workspaces_list.size()) {
addWorkspace();
}
} else if (nr_ws < m_workspaces_list.size()) {
while(nr_ws != m_workspaces_list.size()) {
removeLastWorkspace();
}
}
// update menu filenames
m_rootmenu->reloadHelper()->setMainFile(fluxbox->getMenuFilename());
m_windowmenu->reloadHelper()->setMainFile(windowMenuFilename());
2003-08-04 12:57:23 +00:00
// reconfigure workspaces
2003-05-19 22:45:51 +00:00
for_each(m_workspaces_list.begin(),
m_workspaces_list.end(),
2003-04-14 15:01:55 +00:00
mem_fun(&Workspace::reconfigure));
2003-08-04 12:57:23 +00:00
// reconfigure Icons
2003-05-19 22:45:51 +00:00
for_each(m_icon_list.begin(),
m_icon_list.end(),
2003-04-14 15:01:55 +00:00
mem_fun(&FluxboxWindow::reconfigure));
imageControl().cleanCache();
2003-07-01 12:39:09 +00:00
// notify objects that the screen is reconfigured
m_reconfigure_sig.emit(*this);
2004-03-23 09:21:29 +00:00
// Reload style
FbTk::ThemeManager::instance().load(fluxbox->getStyleFilename(),
fluxbox->getStyleOverlayFilename(),
m_root_theme->screenNum());
2006-03-22 12:23:17 +00:00
reconfigureTabs();
reconfigureStruts();
2006-03-22 12:23:17 +00:00
}
void BScreen::reconfigureTabs() {
2007-12-21 07:35:36 +00:00
const std::list<Focusable *> winlist =
focusControl().focusedOrderWinList().clientList();
std::list<Focusable *>::const_iterator it = winlist.begin(),
it_end = winlist.end();
for (; it != it_end; ++it)
(*it)->fbwindow()->applyDecorations();
2001-12-11 20:47:02 +00:00
}
static void parseStruts(const std::string &s, int &l, int &r, int &t, int &b) {
std::list<std::string> v;
FbTk::StringUtil::stringtok(v, s, " ,");
std::list<std::string>::iterator it = v.begin();
if (it != v.end()) l = std::max(0, atoi(it->c_str()));
if (++it != v.end()) r = std::max(0, atoi(it->c_str()));
if (++it != v.end()) t = std::max(0, atoi(it->c_str()));
if (++it != v.end()) b = std::max(0, atoi(it->c_str()));
}
void BScreen::reconfigureStruts() {
for (std::vector<Strut*>::iterator it = m_head_struts.begin(),
end = m_head_struts.end(); it != end; ++it) {
clearStrut(*it);
}
m_head_struts.clear();
int gl = 0, gr = 0, gt = 0, gb = 0;
parseStruts(FbTk::Resource<std::string>(resourceManager(), "",
name() + ".struts",
altName() + ".Struts"), gl, gr, gt, gb);
const int nh = std::max(1, numHeads());
for (int i = 1; i <= nh; ++i) {
int l = gl, r = gr, t = gt, b = gb;
char ai[16];
sprintf(ai, "%d", i);
parseStruts(FbTk::Resource<std::string>(resourceManager(), "",
name() + ".struts." + ai,
altName() + ".Struts." + ai), l, r, t, b);
if (l+t+r+b)
m_head_struts.push_back(requestStrut(i, l, r, t, b));
}
updateAvailableWorkspaceArea();
}
void BScreen::updateWorkspaceName(unsigned int w) {
Workspace *space = getWorkspace(w);
if (space) {
m_workspace_names[w] = space->name();
m_workspacenames_sig.emit(*this);
Fluxbox::instance()->save_rc();
}
}
2001-12-11 20:47:02 +00:00
void BScreen::removeWorkspaceNames() {
2003-08-24 11:19:45 +00:00
m_workspace_names.clear();
2001-12-11 20:47:02 +00:00
}
void BScreen::addIcon(FluxboxWindow *w) {
if (w == 0)
return;
2001-12-11 20:47:02 +00:00
// make sure we have a unique list
if (find(iconList().begin(), iconList().end(), w) != iconList().end())
return;
2008-09-22 05:44:40 +00:00
iconList().push_back(w);
// notify listeners
2008-09-22 05:44:40 +00:00
iconListSig().emit(*this);
2001-12-11 20:47:02 +00:00
}
void BScreen::removeIcon(FluxboxWindow *w) {
if (w == 0)
2002-12-01 13:42:15 +00:00
return;
Icons::iterator erase_it = find_if(iconList().begin(),
iconList().end(),
bind2nd(equal_to<FluxboxWindow *>(), w));
// no need to send iconlist signal if we didn't
// change the iconlist
if (erase_it != m_icon_list.end()) {
iconList().erase(erase_it);
2008-09-22 05:44:40 +00:00
iconListSig().emit(*this);
}
2003-07-28 18:28:03 +00:00
}
void BScreen::removeWindow(FluxboxWindow *win) {
fbdbg<<"BScreen::removeWindow("<<win<<")"<<endl;
// extra precaution, if for some reason, the
// icon list should be out of sync
removeIcon(win);
// remove from workspace
Workspace *space = getWorkspace(win->workspaceNumber());
if (space != 0)
space->removeWindow(win, false);
}
2001-12-11 20:47:02 +00:00
void BScreen::removeClient(WinClient &client) {
2003-07-28 18:28:03 +00:00
focusControl().removeClient(client);
2007-10-13 21:51:37 +00:00
if (client.fbwindow() && client.fbwindow()->isIconic())
iconListSig().emit(*this);
2005-05-02 12:10:01 +00:00
using namespace FbTk;
// remove any grouping this is expecting
2005-05-02 12:10:01 +00:00
Groupables::iterator erase_it = find_if(m_expecting_groups.begin(),
m_expecting_groups.end(),
Compose(bind2nd(equal_to<WinClient *>(), &client),
Select2nd<Groupables::value_type>()));
if (erase_it != m_expecting_groups.end())
m_expecting_groups.erase(erase_it);
2003-07-28 18:28:03 +00:00
// the client could be on icon menu so we update it
//!! TODO: check this with the new icon menu
// updateIconMenu();
}
int BScreen::addWorkspace() {
bool save_name = getNameOfWorkspace(m_workspaces_list.size()) == "";
std::string name = getNameOfWorkspace(m_workspaces_list.size());
Workspace *ws = new Workspace(*this, name, m_workspaces_list.size());
m_workspaces_list.push_back(ws);
if (save_name) {
addWorkspaceName(ws->name().c_str());
m_workspacenames_sig.emit(*this);
}
2003-05-19 22:45:51 +00:00
saveWorkspaces(m_workspaces_list.size());
workspaceCountSig().emit( *this );
2003-05-19 22:45:51 +00:00
return m_workspaces_list.size();
2001-12-11 20:47:02 +00:00
}
2002-10-23 17:32:30 +00:00
/// removes last workspace
/// @return number of desktops left
int BScreen::removeLastWorkspace() {
2003-05-19 22:45:51 +00:00
if (m_workspaces_list.size() <= 1)
2002-12-01 13:42:15 +00:00
return 0;
2003-05-19 22:45:51 +00:00
Workspace *wkspc = m_workspaces_list.back();
2001-12-11 20:47:02 +00:00
2003-05-19 22:45:51 +00:00
if (m_current_workspace->workspaceID() == wkspc->workspaceID())
changeWorkspaceID(m_current_workspace->workspaceID() - 1);
2001-12-11 20:47:02 +00:00
2006-07-22 07:15:24 +00:00
wkspc->removeAll(wkspc->workspaceID()-1);
2001-12-11 20:47:02 +00:00
Icons::iterator it = iconList().begin();
const Icons::iterator it_end = iconList().end();
for (; it != it_end; ++it) {
if ((*it)->workspaceNumber() == wkspc->workspaceID())
(*it)->setWorkspace(wkspc->workspaceID()-1);
}
m_clientlist_sig.emit(*this);
2002-12-01 13:42:15 +00:00
//remove last workspace
m_workspaces_list.pop_back();
2004-12-30 14:33:38 +00:00
2003-05-19 22:45:51 +00:00
saveWorkspaces(m_workspaces_list.size());
workspaceCountSig().emit( *this );
2004-12-30 14:33:38 +00:00
// must be deleted after we send notify!!
// so we dont get bad pointers somewhere
// while processing the notify signal
delete wkspc;
2001-12-11 20:47:02 +00:00
2003-05-19 22:45:51 +00:00
return m_workspaces_list.size();
2001-12-11 20:47:02 +00:00
}
void BScreen::changeWorkspaceID(unsigned int id, bool revert) {
2003-09-14 10:13:54 +00:00
2003-05-19 22:45:51 +00:00
if (! m_current_workspace || id >= m_workspaces_list.size() ||
id == m_current_workspace->workspaceID())
2002-12-01 13:42:15 +00:00
return;
2003-05-11 11:47:19 +00:00
/* Ignore all EnterNotify events until the pointer actually moves */
this->focusControl().ignoreAtPointer();
FbTk::App::instance()->sync(false);
XGrabServer(Fluxbox::instance()->display());
FluxboxWindow *focused = FocusControl::focusedFbWindow();
if (focused && focused->isMoving() && doOpaqueMove())
// don't reassociate if not opaque moving
reassociateWindow(focused, id, true);
2007-11-04 17:36:40 +00:00
// set new workspace
Workspace *old = currentWorkspace();
m_current_workspace = getWorkspace(id);
// we show new workspace first in order to appear faster
currentWorkspace()->showAll();
2003-05-11 11:47:19 +00:00
// reassociate all windows that are stuck to the new workspace
Workspace::Windows wins = old->windowList();
2003-05-11 11:47:19 +00:00
Workspace::Windows::iterator it = wins.begin();
for (; it != wins.end(); ++it) {
if ((*it)->isStuck()) {
reassociateWindow(*it, id, true);
2002-12-01 13:42:15 +00:00
}
2003-05-11 11:47:19 +00:00
}
// change workspace ID of stuck iconified windows, too
Icons::iterator icon_it = iconList().begin();
for (; icon_it != iconList().end(); ++icon_it) {
if ((*icon_it)->isStuck())
(*icon_it)->setWorkspace(id);
}
if (focused && focused->isMoving() && doOpaqueMove())
2007-10-13 21:51:37 +00:00
focused->focus();
else if (revert)
2007-01-26 17:18:14 +00:00
FocusControl::revertFocus(*this);
2001-12-11 20:47:02 +00:00
2007-11-04 17:36:40 +00:00
old->hideAll(false);
XUngrabServer(Fluxbox::instance()->display());
2004-01-19 18:29:43 +00:00
FbTk::App::instance()->sync(false);
2003-09-14 10:13:54 +00:00
m_currentworkspace_sig.emit(*this);
2007-10-13 21:51:37 +00:00
// do this after atom handlers, so scripts can access new workspace number
Fluxbox::instance()->keys()->doAction(FocusIn, 0, 0, Keys::ON_DESKTOP);
2001-12-11 20:47:02 +00:00
}
void BScreen::sendToWorkspace(unsigned int id, FluxboxWindow *win, bool changeWS) {
2003-12-18 15:27:21 +00:00
if (! m_current_workspace || id >= m_workspaces_list.size())
2002-12-01 13:42:15 +00:00
return;
if (!win)
win = FocusControl::focusedFbWindow();
if (!win || &win->screen() != this || win->isStuck())
return;
2003-12-04 00:08:55 +00:00
2007-10-13 21:51:37 +00:00
FbTk::App::instance()->sync(false);
2003-12-04 00:08:55 +00:00
windowMenu().hide();
reassociateWindow(win, id, true);
2003-12-18 15:27:21 +00:00
// change workspace ?
2007-10-13 21:51:37 +00:00
if (changeWS)
changeWorkspaceID(id, false);
2007-10-13 21:51:37 +00:00
// if the window is on current workspace, show it; else hide it.
if (id == currentWorkspace()->workspaceID() && !win->isIconic())
2007-11-23 00:45:56 +00:00
win->show();
2007-10-13 21:51:37 +00:00
else {
win->hide(true);
FocusControl::revertFocus(*this);
}
2002-10-11 10:22:06 +00:00
// send all the transients too
FluxboxWindow::ClientList::iterator client_it = win->clientList().begin();
FluxboxWindow::ClientList::iterator client_it_end = win->clientList().end();
for (; client_it != client_it_end; ++client_it) {
WinClient::TransientList::const_iterator it = (*client_it)->transientList().begin();
WinClient::TransientList::const_iterator it_end = (*client_it)->transientList().end();
for (; it != it_end; ++it) {
if ((*it)->fbwindow())
sendToWorkspace(id, (*it)->fbwindow(), false);
}
2002-12-01 13:42:15 +00:00
}
2003-12-04 00:08:55 +00:00
}
2001-12-11 20:47:02 +00:00
bool BScreen::isKdeDockapp(Window client) const {
//Check and see if client is KDE dock applet.
bool iskdedockapp = false;
Atom ajunk;
int ijunk;
2006-08-09 04:54:58 +00:00
unsigned long *data = 0, uljunk;
Display *disp = FbTk::App::instance()->display();
// Check if KDE v2.x dock applet
if (XGetWindowProperty(disp, client, atom_kde_systray,
0l, 1l, False,
XA_WINDOW, &ajunk, &ijunk, &uljunk,
2006-08-09 04:54:58 +00:00
&uljunk, (unsigned char **) &data) == Success) {
if (data)
iskdedockapp = true;
XFree((void *) data);
data = 0;
}
2003-07-28 20:11:55 +00:00
// Check if KDE v1.x dock applet
if (!iskdedockapp) {
2003-07-28 20:11:55 +00:00
if (XGetWindowProperty(disp, client,
atom_kwm1, 0l, 1l, False,
atom_kwm1, &ajunk, &ijunk, &uljunk,
2006-08-09 04:54:58 +00:00
&uljunk, (unsigned char **) &data) == Success && data) {
iskdedockapp = (data && data[0] != 0);
2003-07-28 20:11:55 +00:00
XFree((void *) data);
data = 0;
}
}
2003-07-28 20:11:55 +00:00
return iskdedockapp;
}
bool BScreen::addKdeDockapp(Window client) {
XSelectInput(FbTk::App::instance()->display(), client, StructureNotifyMask);
FbTk::EventHandler *evh = 0;
FbTk::EventManager *evm = FbTk::EventManager::instance();
AtomHandler* handler = 0;
#if USE_SYSTRAY
handler = Fluxbox::instance()->getAtomHandler(SystemTray::getNetSystemTrayAtom(screenNumber()));
#endif
if (handler == 0) {
#ifdef USE_SLIT
if (slit() != 0 && slit()->acceptKdeDockapp())
slit()->addClient(client);
else
#endif // USE_SLIT
return false;
} else {
// this handler is a special case
// so we call setupClient in it
WinClient winclient(client, *this);
handler->setupClient(winclient);
// we need to save old handler and re-add it later
evh = evm->find(client);
}
if (evh != 0) // re-add handler
evm->add(*evh, client);
return true;
}
FluxboxWindow *BScreen::createWindow(Window client) {
Fluxbox* fluxbox = Fluxbox::instance();
fluxbox->sync(false);
2003-07-28 20:11:55 +00:00
2005-04-29 02:52:36 +00:00
if (isKdeDockapp(client) && addKdeDockapp(client)) {
return 0; // dont create a FluxboxWindow for this one
}
2003-07-28 20:11:55 +00:00
WinClient *winclient = new WinClient(client, *this);
if (winclient->initial_state == WithdrawnState ||
winclient->getWMClassClass() == "DockApp") {
2003-06-27 15:05:19 +00:00
delete winclient;
#ifdef USE_SLIT
if (slit() && !isKdeDockapp(client))
slit()->addClient(client);
#endif // USE_SLIT
2003-05-17 11:05:33 +00:00
return 0;
}
// check if it should be grouped with something else
WinClient* other = findGroupLeft(*winclient);
FluxboxWindow* win = other ? other->fbwindow() : 0;
if (other && win) {
win->attachClient(*winclient);
fluxbox->attachSignals(*winclient);
} else {
fluxbox->attachSignals(*winclient);
if (winclient->fbwindow()) { // may have been set in an atomhandler
win = winclient->fbwindow();
Workspace *workspace = getWorkspace(win->workspaceNumber());
if (workspace)
workspace->updateClientmenu();
} else {
win = new FluxboxWindow(*winclient);
if (!win->isManaged()) {
delete win;
return 0;
}
}
}
// add the window to the focus list
2007-01-27 18:06:28 +00:00
// always add to front on startup to keep the focus order the same
if (win->isFocused() || fluxbox->isStartup())
focusControl().addFocusFront(*winclient);
else
focusControl().addFocusBack(*winclient);
// we also need to check if another window expects this window to the left
// and if so, then join it.
if ((other = findGroupRight(*winclient)) && other->fbwindow() != win)
win->attachClient(*other);
else if (other) // should never happen
win->moveClientRightOf(*other, *winclient);
m_clientlist_sig.emit(*this);
2004-01-19 18:29:43 +00:00
fluxbox->sync(false);
return win;
}
2003-04-14 15:01:55 +00:00
FluxboxWindow *BScreen::createWindow(WinClient &client) {
if (isKdeDockapp(client.window()) && addKdeDockapp(client.window())) {
return 0;
}
FluxboxWindow *win = new FluxboxWindow(client);
2003-04-14 15:01:55 +00:00
#ifdef SLIT
if (slit() != 0) {
if (win->initialState() == WithdrawnState) {
slit()->addClient(client.window());
} else if (client->getWMClassClass() == "DockApp") {
slit()->addClient(client.window());
}
}
2003-04-14 15:01:55 +00:00
#endif // SLIT
2003-04-14 15:01:55 +00:00
if (!win->isManaged()) {
delete win;
return 0;
}
2003-12-14 01:09:00 +00:00
2007-01-27 18:06:28 +00:00
win->show();
// don't ask me why, but client doesn't seem to keep focus in new window
// and we don't seem to get a FocusIn event from setInputFocus
if ((focusControl().focusNew() || FocusControl::focusedWindow() == &client)
2007-10-13 21:51:37 +00:00
&& win->focus())
2006-07-09 06:51:28 +00:00
FocusControl::setFocusedWindow(&client);
m_clientlist_sig.emit(*this);
2004-01-19 18:29:43 +00:00
2003-04-14 15:01:55 +00:00
return win;
}
Strut *BScreen::requestStrut(int head, int left, int right, int top, int bottom) {
if (head > numHeads() && head != 1) {
// head does not exist (if head == 1, then numHeads() == 0,
// which means no xinerama, but there's a head after all
head = numHeads();
}
2003-06-18 13:42:21 +00:00
int begin = head-1;
int end = head;
2003-06-18 13:42:21 +00:00
if (head == 0) { // all heads (or no xinerama)
begin = 0;
end = (numHeads() ? numHeads() : 1);
2003-06-18 13:42:21 +00:00
}
Strut* next = 0;
for (int i = begin; i != end; i++) {
next = m_head_areas[i]->requestStrut(i+1, left, right, top, bottom, next);
}
return next;
}
void BScreen::clearStrut(Strut *str) {
if (str->next())
clearStrut(str->next());
int head = str->head() ? str->head() - 1 : 0;
/* The number of heads may have changed, be careful. */
if (head < (numHeads() ? numHeads() : 1))
m_head_areas[head]->clearStrut(str);
// str is invalid now
}
2003-06-18 13:42:21 +00:00
void BScreen::updateAvailableWorkspaceArea() {
size_t n = (numHeads() ? numHeads() : 1);
bool updated = false;
2003-06-18 13:42:21 +00:00
for (size_t i = 0; i < n; i++) {
updated = m_head_areas[i]->updateAvailableWorkspaceArea() || updated;
}
2004-01-19 18:29:43 +00:00
if (updated)
m_workspace_area_sig.emit(*this);
2003-06-18 13:42:21 +00:00
}
2001-12-11 20:47:02 +00:00
void BScreen::addWorkspaceName(const char *name) {
m_workspace_names.push_back(FbTk::FbStringUtil::LocaleStrToFb(name));
Workspace *wkspc = getWorkspace(m_workspace_names.size()-1);
if (wkspc)
wkspc->setName(m_workspace_names.back());
2001-12-11 20:47:02 +00:00
}
2002-08-11 22:28:18 +00:00
string BScreen::getNameOfWorkspace(unsigned int workspace) const {
2003-12-14 01:09:00 +00:00
if (workspace < m_workspace_names.size())
2003-05-19 22:45:51 +00:00
return m_workspace_names[workspace];
2003-12-14 01:09:00 +00:00
else
2002-12-01 13:42:15 +00:00
return "";
2001-12-11 20:47:02 +00:00
}
void BScreen::reassociateWindow(FluxboxWindow *w, unsigned int wkspc_id,
2003-04-28 00:38:42 +00:00
bool ignore_sticky) {
2003-04-14 15:01:55 +00:00
if (w == 0)
return;
2001-12-11 20:47:02 +00:00
if (wkspc_id >= numberOfWorkspaces())
2003-05-19 22:45:51 +00:00
wkspc_id = currentWorkspace()->workspaceID();
2002-12-01 13:42:15 +00:00
2003-05-15 11:17:29 +00:00
if (!w->isIconic() && w->workspaceNumber() == wkspc_id)
2002-12-01 13:42:15 +00:00
return;
2003-02-16 17:57:54 +00:00
2002-12-01 13:42:15 +00:00
if (w->isIconic()) {
removeIcon(w);
2003-04-14 15:01:55 +00:00
getWorkspace(wkspc_id)->addWindow(*w);
2002-12-01 13:42:15 +00:00
} else if (ignore_sticky || ! w->isStuck()) {
// fresh windows have workspaceNumber == -1, which leads to
// an invalid workspace (unsigned int)
2011-02-24 15:56:53 +00:00
Workspace* ws = getWorkspace(w->workspaceNumber());
if (ws)
ws->removeWindow(w, true);
2003-04-14 15:01:55 +00:00
getWorkspace(wkspc_id)->addWindow(*w);
2002-12-01 13:42:15 +00:00
}
2001-12-11 20:47:02 +00:00
}
#if USE_TOOLBAR
void BScreen::clearToolButtonMap() {
m_toolButtonMap->clear();
}
void BScreen::mapToolButton(std::string name, FbTk::TextButton *button) {
m_toolButtonMap->insert(std::pair<std::string, FbTk::TextButton*>(name, button));
}
bool BScreen::relabelToolButton(std::string button, std::string label) {
ToolButtonMap::const_iterator it = m_toolButtonMap->find(button);
if (it != m_toolButtonMap->end() && it->second) {
it->second->setText(label);
m_toolbar->relayout();
return true;
}
return false;
}
#endif
void BScreen::initMenus() {
m_workspacemenu.reset(MenuCreator::createMenuType("workspacemenu", screenNumber()));
m_rootmenu->reloadHelper()->setMainFile(Fluxbox::instance()->getMenuFilename());
m_windowmenu->reloadHelper()->setMainFile(windowMenuFilename());
}
void BScreen::rereadMenu() {
m_rootmenu->removeAll();
m_rootmenu->setLabel(FbTk::BiDiString(""));
2002-12-01 13:42:15 +00:00
Fluxbox * const fb = Fluxbox::instance();
if (!fb->getMenuFilename().empty())
MenuCreator::createFromFile(fb->getMenuFilename(), *m_rootmenu,
m_rootmenu->reloadHelper());
2002-12-01 13:42:15 +00:00
if (m_rootmenu->numberOfItems() == 0) {
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
m_rootmenu->setLabel(_FB_XTEXT(Menu, DefaultRootMenu, "Fluxbox default menu", "Title of fallback root menu"));
FbTk::RefCount<FbTk::Command<void> > restart_fb(FbTk::CommandParser<void>::instance().parse("restart"));
FbTk::RefCount<FbTk::Command<void> > exit_fb(FbTk::CommandParser<void>::instance().parse("exit"));
FbTk::RefCount<FbTk::Command<void> > execute_xterm(FbTk::CommandParser<void>::instance().parse("exec xterm"));
2002-12-13 20:19:05 +00:00
m_rootmenu->setInternalMenu();
m_rootmenu->insertCommand("xterm", execute_xterm);
m_rootmenu->insertCommand(_FB_XTEXT(Menu, Restart, "Restart", "Restart command"),
restart_fb);
m_rootmenu->insertCommand(_FB_XTEXT(Menu, Exit, "Exit", "Exit command"),
exit_fb);
}
2001-12-11 20:47:02 +00:00
}
const std::string BScreen::windowMenuFilename() const {
2015-02-01 07:07:26 +00:00
std::string name = *resource.windowmenufile;
if (name.empty()) {
name = Fluxbox::instance()->getDefaultDataFilename("windowmenu");
}
return name;
}
void BScreen::rereadWindowMenu() {
m_windowmenu->removeAll();
if (!windowMenuFilename().empty())
MenuCreator::createFromFile(windowMenuFilename(), *m_windowmenu,
m_windowmenu->reloadHelper());
}
void BScreen::addConfigMenu(const FbTk::FbString &label, FbTk::Menu &menu) {
FbTk::Menu* cm = m_configmenu.get();
if (cm) {
int pos = cm->findSubmenuIndex(&menu);
if (pos == -1) { // not found? add
cm->insertSubmenu(label, &menu, pos);
}
}
}
void BScreen::removeConfigMenu(FbTk::Menu &menu) {
2004-06-14 12:25:31 +00:00
FbTk::Menu* cm = m_configmenu.get();
if (cm) {
int pos = cm->findSubmenuIndex(&menu);
if (pos > -1) {
cm->remove(pos);
}
}
}
void BScreen::addManagedResource(FbTk::Resource_base *resource) {
m_managed_resources.push_back(resource);
}
int BScreen::getGap(int head, const char type) {
return type == 'w' ? getXGap(head) : getYGap(head);
}
int BScreen::calRelativeSize(int head, int i, char type) {
// return floor(i * getGap(head, type) / 100 + 0.5);
return FbTk::RelCalcHelper::calPercentageValueOf(i, getGap(head, type));
}
int BScreen::calRelativeWidth(int head, int i) {
return calRelativeSize(head, i, 'w');
}
int BScreen::calRelativeHeight(int head, int i) {
return calRelativeSize(head, i, 'h');
}
int BScreen::calRelativePosition(int head, int i, char type) {
int max = type == 'w' ? maxLeft(head) : maxTop(head);
// return floor((i - min) / getGap(head, type) * 100 + 0.5);
return FbTk::RelCalcHelper::calPercentageOf((i - max), getGap(head, type));
}
// returns a pixel, which is relative to the width of the screen
// screen starts from 0, 1000 px width, if i is 10 then it should return 100
int BScreen::calRelativePositionWidth(int head, int i) {
return calRelativePosition(head, i, 'w');
}
// returns a pixel, which is relative to the height of th escreen
// screen starts from 0, 1000 px height, if i is 10 then it should return 100
int BScreen::calRelativePositionHeight(int head, int i) {
return calRelativePosition(head, i, 'h');
}
int BScreen::calRelativeDimension(int head, int i, char type) {
// return floor(i / getGap(head, type) * 100 + 0.5);
return FbTk::RelCalcHelper::calPercentageOf(i, getGap(head, type));
}
int BScreen::calRelativeDimensionWidth(int head, int i) {
return calRelativeDimension(head, i, 'w');
}
int BScreen::calRelativeDimensionHeight(int head, int i) {
return calRelativeDimension(head, i, 'h');
}
float BScreen::getXGap(int head) {
return maxRight(head) - maxLeft(head);
}
float BScreen::getYGap(int head) {
return maxBottom(head) - maxTop(head);
}
void BScreen::setupConfigmenu(FbTk::Menu &menu) {
2007-10-13 21:51:37 +00:00
2015-01-21 08:53:54 +00:00
struct ConfigMenu::SetupHelper sh(*this, m_resource_manager, resource);
menu.removeAll();
ConfigMenu::setup(menu, sh);
menu.updateMenu();
}
2001-12-11 20:47:02 +00:00
void BScreen::shutdown() {
2003-05-10 22:59:32 +00:00
rootWindow().setEventMask(NoEventMask);
2003-12-04 21:31:02 +00:00
FbTk::App::instance()->sync(false);
m_state.shutdown = true;
m_focus_control->shutdown();
2003-05-19 22:45:51 +00:00
for_each(m_workspaces_list.begin(),
m_workspaces_list.end(),
2003-04-14 15:01:55 +00:00
mem_fun(&Workspace::shutdown));
2001-12-11 20:47:02 +00:00
}
void BScreen::showPosition(int x, int y) {
if (!doShowWindowPos())
2003-12-10 22:28:07 +00:00
return;
char buf[256];
2015-01-21 09:02:38 +00:00
snprintf(buf, sizeof(buf), "X:%5d x Y:%5d", x, y);
FbTk::BiDiString label(buf);
m_pos_window->showText(label);
2001-12-11 20:47:02 +00:00
}
void BScreen::hidePosition() {
m_pos_window->hide();
}
void BScreen::showGeometry(unsigned int gx, unsigned int gy) {
2003-12-10 22:28:07 +00:00
if (!doShowWindowPos())
return;
char buf[256];
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
2002-12-01 13:42:15 +00:00
2015-01-21 09:02:38 +00:00
snprintf(buf, sizeof(buf),
_FB_XTEXT(Screen, GeometryFormat,
2004-06-07 11:46:05 +00:00
"W: %4d x H: %4d",
"Format for width and height window, %4d for width, and %4d for height").c_str(),
2004-06-07 11:46:05 +00:00
gx, gy);
FbTk::BiDiString label(buf);
m_geom_window->showText(label);
2001-12-11 20:47:02 +00:00
}
void BScreen::showTooltip(const FbTk::BiDiString &text) {
if (*resource.tooltip_delay >= 0)
m_tooltip_window->showText(text);
}
void BScreen::hideTooltip() {
if (*resource.tooltip_delay >= 0)
m_tooltip_window->hide();
}
void BScreen::hideGeometry() {
m_geom_window->hide();
2001-12-11 20:47:02 +00:00
}
void BScreen::setLayer(FbTk::LayerItem &item, int layernum) {
2003-02-03 13:56:12 +00:00
m_layermanager.moveToLayer(item, layernum);
}
2002-11-27 21:55:36 +00:00
/**
Goes to the workspace "right" of the current
*/
void BScreen::nextWorkspace(int delta) {
changeWorkspaceID( (currentWorkspaceID() + delta) % numberOfWorkspaces());
2001-12-11 20:47:02 +00:00
}
2002-11-27 21:55:36 +00:00
/**
Goes to the workspace "left" of the current
*/
void BScreen::prevWorkspace(int delta) {
changeWorkspaceID( (static_cast<signed>(numberOfWorkspaces()) + currentWorkspaceID() - (delta % numberOfWorkspaces())) % numberOfWorkspaces());
2001-12-11 20:47:02 +00:00
}
2002-02-02 19:51:15 +00:00
2002-11-27 21:55:36 +00:00
/**
Goes to the workspace "right" of the current
*/
void BScreen::rightWorkspace(int delta) {
if (currentWorkspaceID()+delta < numberOfWorkspaces())
2003-05-15 12:00:46 +00:00
changeWorkspaceID(currentWorkspaceID()+delta);
2002-02-02 19:51:15 +00:00
}
2002-11-27 21:55:36 +00:00
/**
Goes to the workspace "left" of the current
*/
void BScreen::leftWorkspace(int delta) {
2003-05-15 12:00:46 +00:00
if (currentWorkspaceID() >= static_cast<unsigned int>(delta))
changeWorkspaceID(currentWorkspaceID()-delta);
2002-02-02 19:51:15 +00:00
}
2002-02-07 14:46:23 +00:00
2003-06-12 15:24:37 +00:00
void BScreen::renderGeomWindow() {
char buf[256];
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
2003-08-25 13:15:53 +00:00
const std::string msg = _FB_XTEXT(Screen, GeometrySpacing,
"W: %04d x H: %04d", "Representative maximum sized text for width and height dialog");
2015-01-21 09:02:38 +00:00
const int n = snprintf(buf, sizeof(buf), msg.c_str(), 0, 0);
FbTk::BiDiString label(std::string(buf, n));
m_geom_window->resizeForText(label);
m_geom_window->reconfigTheme();
2003-06-12 15:24:37 +00:00
}
void BScreen::renderPosWindow() {
m_pos_window->resizeForText(FbTk::BiDiString("0:00000 x 0:00000"));
m_pos_window->reconfigTheme();
}
void BScreen::updateSize() {
// update xinerama layout
initXinerama();
// check if window geometry has changed
if (rootWindow().updateGeometry()) {
// reset background
m_root_theme->reset();
// send resize notify
m_resize_sig.emit(*this);
m_workspace_area_sig.emit(*this);
// move windows out of inactive heads
clearHeads();
}
}
2003-05-19 15:32:47 +00:00
/**
* Find the winclient to this window's left
* So, we check the leftgroup hint, and see if we know any windows
*/
WinClient *BScreen::findGroupLeft(WinClient &winclient) {
Window w = winclient.getGroupLeftWindow();
if (w == None)
return 0;
WinClient *have_client = Fluxbox::instance()->searchWindow(w);
if (!have_client) {
// not found, add it to expecting
m_expecting_groups[w] = &winclient;
} else if (&have_client->screen() != &winclient.screen())
// something is not consistent
return 0;
return have_client;
}
WinClient *BScreen::findGroupRight(WinClient &winclient) {
Groupables::iterator it = m_expecting_groups.find(winclient.window());
if (it == m_expecting_groups.end())
return 0;
// yay, this'll do.
WinClient *other = it->second;
m_expecting_groups.erase(it); // don't expect it anymore
// forget about it if it isn't the left-most client in the group
Window leftwin = other->getGroupLeftWindow();
if (leftwin != None && leftwin != winclient.window())
return 0;
return other;
}
2008-02-10 17:16:50 +00:00
void BScreen::clearXinerama() {
fbdbg<<"BScreen::initXinerama(): dont have Xinerama"<<endl;
m_xinerama.avail = false;
m_xinerama.heads.clear();
2008-02-10 17:16:50 +00:00
}
2003-05-19 15:32:47 +00:00
void BScreen::initXinerama() {
#ifdef XINERAMA
Display* display = FbTk::App::instance()->display();
int number = 0;
XineramaScreenInfo *si = XineramaQueryScreens(display, &number);
if (!si && number == 0) {
2008-02-10 17:16:50 +00:00
clearXinerama();
return;
}
m_xinerama.avail = true;
fbdbg<<"BScreen::initXinerama(): have Xinerama"<<endl;
/* The call may have actually failed. If this is the first time we init
* Xinerama, fall back to turning it off. If not, pretend nothing
* happened -- another event will tell us and it will work then. */
if (!si) {
if (m_xinerama.heads.empty())
2008-02-10 17:16:50 +00:00
clearXinerama();
return;
}
m_xinerama.heads.resize(number);
for (int i = 0; i < number; i++) {
m_xinerama.heads[i]._x = si[i].x_org;
m_xinerama.heads[i]._y = si[i].y_org;
m_xinerama.heads[i]._width = si[i].width;
m_xinerama.heads[i]._height = si[i].height;
}
XFree(si);
fbdbg<<"BScreen::initXinerama(): number of heads ="<<number<<endl;
/* Reallocate to the new number of heads. */
2010-05-01 12:29:41 +00:00
int ha_num = numHeads() ? numHeads() : 1;
int ha_oldnum = m_head_areas.size();
if (ha_num > ha_oldnum) {
m_head_areas.resize(ha_num);
for (int i = ha_oldnum; i < ha_num; i++)
m_head_areas[i] = new HeadArea();
} else if (ha_num < ha_oldnum) {
for (int i = ha_num; i < ha_oldnum; i++)
delete m_head_areas[i];
m_head_areas.resize(ha_num);
}
2003-05-19 15:32:47 +00:00
#else // XINERAMA
m_xinerama.avail = false;
2003-05-19 15:32:47 +00:00
#endif // XINERAMA
reconfigureStruts();
}
/* Move windows out of inactive heads */
void BScreen::clearHeads() {
if (!hasXinerama()) return;
for (Workspaces::iterator i = m_workspaces_list.begin();
i != m_workspaces_list.end(); ++i) {
for (Workspace::Windows::iterator win = (*i)->windowList().begin();
win != (*i)->windowList().end(); ++win) {
FluxboxWindow& w = *(*win);
// check if the window is invisible
bool invisible = true;
int j;
for (j = 0; j < numHeads(); ++j) {
XineramaHeadInfo& hi = m_xinerama.heads[j];
if (RectangleUtil::overlapRectangles(hi, w)) {
invisible = false;
break;
}
}
if (invisible) { // get closest head and replace the (now invisible) cwindow
int closest_head = getHead(w.fbWindow());
if (closest_head == 0) {
closest_head = 1; // first head is a safe bet here
}
w.placeWindow(closest_head);
}
}
}
}
int BScreen::getHead(int x, int y) const {
2010-05-01 12:29:41 +00:00
#ifdef XINERAMA
if (hasXinerama()) {
for (int i = 0; i < numHeads(); i++) {
if (RectangleUtil::insideBorder(m_xinerama.heads[i], x, y, 0)) {
2010-05-01 12:29:41 +00:00
return i+1;
}
}
}
2003-05-19 15:32:47 +00:00
#endif // XINERAMA
return 0;
}
2007-03-31 22:00:39 +00:00
int BScreen::getHead(const FbTk::FbWindow &win) const {
int head = 0; // whole screen
2010-08-22 18:11:59 +00:00
#ifdef XINERAMA
if (hasXinerama()) {
// cast needed to prevent win.x() become "unsigned int" which is bad
2013-05-23 07:16:19 +00:00
// since it might be negative
int cx = win.x() + static_cast<int>(win.width() / 2);
int cy = win.y() + static_cast<int>(win.height() / 2);
head = getHead(cx, cy);
if ( head == 0 ) {
// if the center of the window is not on any head then select
// the head which center is nearest to the window center
long dist = -1;
int i;
for (i = 0; i < numHeads(); ++i) {
const XineramaHeadInfo& hi = m_xinerama.heads[i];
long d = calcSquareDistance(cx, cy,
hi.x() + (hi.width() / 2), hi.y() + (hi.height() / 2));
if (dist == -1 || d < dist) { // found a closer head
head = i + 1;
dist = d;
}
}
}
}
#endif
return head;
}
int BScreen::getCurrHead() const {
if (!hasXinerama()) return 0;
2003-05-19 15:32:47 +00:00
int root_x = 0, root_y = 0;
#ifdef XINERAMA
union { int i; unsigned int ui; Window w; } ignore;
XQueryPointer(FbTk::App::instance()->display(),
rootWindow().window(), &ignore.w,
&ignore.w, &root_x, &root_y,
&ignore.i, &ignore.i, &ignore.ui);
2003-05-19 15:32:47 +00:00
#endif // XINERAMA
return getHead(root_x, root_y);
}
int BScreen::getHeadX(int head) const {
2003-05-19 15:32:47 +00:00
#ifdef XINERAMA
if (head == 0 || head > numHeads()) return 0;
return m_xinerama.heads[head-1].x();
2003-05-19 15:32:47 +00:00
#else
return 0;
#endif // XINERAMA
}
int BScreen::getHeadY(int head) const {
2003-05-19 15:32:47 +00:00
#ifdef XINERAMA
if (head == 0 || head > numHeads()) return 0;
return m_xinerama.heads[head-1].y();
2003-05-19 15:32:47 +00:00
#else
return 0;
#endif // XINERAMA
}
int BScreen::getHeadWidth(int head) const {
2003-05-19 15:32:47 +00:00
#ifdef XINERAMA
if (head == 0 || head > numHeads()) return width();
return m_xinerama.heads[head-1].width();
2003-05-19 15:32:47 +00:00
#else
return width();
#endif // XINERAMA
}
int BScreen::getHeadHeight(int head) const {
2003-05-19 15:32:47 +00:00
#ifdef XINERAMA
if (head == 0 || head > numHeads()) return height();
return m_xinerama.heads[head-1].height();
2003-05-19 15:32:47 +00:00
#else
return height();
#endif // XINERAMA
}
pair<int,int> BScreen::clampToHead(int head, int x, int y, int w, int h) const {
// if there are multiple heads, head=0 is not valid
// a better way would be to search the closest head
if (head == 0 && numHeads() != 0)
2010-05-01 12:29:41 +00:00
head = 1;
int hx = getHeadX(head);
int hy = getHeadY(head);
int hw = getHeadWidth(head);
int hh = getHeadHeight(head);
x = FbTk::Util::clamp(x, hx, hx + hw - w);
y = FbTk::Util::clamp(y, hy, hy + hh - h);
return make_pair(x,y);
}