fluxbox/src/Screen.cc

2467 lines
86 KiB
C++
Raw Normal View History

// Screen.cc for Fluxbox Window Manager
2003-04-14 15:01:55 +00:00
// Copyright (c) 2001 - 2003 Henrik Kinnunen (fluxgen at users.sourceforge.net)
//
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.
2003-05-11 17:14:41 +00:00
// $Id: Screen.cc,v 1.156 2003/05/11 17:11:59 fluxgen Exp $
2002-10-25 20:58:14 +00:00
#include "Screen.hh"
#include "i18n.hh"
#include "fluxbox.hh"
2002-11-30 20:18:35 +00:00
#include "ImageControl.hh"
2002-10-25 20:58:14 +00:00
#include "Toolbar.hh"
#include "Window.hh"
#include "Workspace.hh"
#include "StringUtil.hh"
2002-11-17 11:29:06 +00:00
#include "Netizen.hh"
#include "DirHelper.hh"
#include "WinButton.hh"
#include "SimpleCommand.hh"
#include "FbWinFrameTheme.hh"
2003-01-07 02:06:06 +00:00
#include "MenuTheme.hh"
#include "RootTheme.hh"
#include "WinButtonTheme.hh"
#include "FbCommands.hh"
#include "BoolMenuItem.hh"
#include "IntResMenuItem.hh"
#include "MacroCommand.hh"
#include "XLayerItem.hh"
#include "MultLayers.hh"
#include "FbMenu.hh"
#include "LayerMenu.hh"
2003-04-14 15:01:55 +00:00
#include "WinClient.hh"
2003-04-20 13:46:18 +00:00
#include "Subject.hh"
//use GNU extensions
2001-12-11 20:47:02 +00:00
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif // _GNU_SOURCE
2002-10-13 22:30:18 +00:00
#ifdef HAVE_CONFIG_H
#include "config.h"
2001-12-11 20:47:02 +00:00
#endif // HAVE_CONFIG_H
2002-10-13 22:30:18 +00:00
#ifdef SLIT
2001-12-11 20:47:02 +00:00
#include "Slit.hh"
#endif // SLIT
#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_CTYPE_H
2002-10-25 20:58:14 +00:00
#include <ctype.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_CTYPE_H
#ifdef HAVE_DIRENT_H
2002-10-25 20:58:14 +00:00
#include <dirent.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_DIRENT_H
#ifdef HAVE_LOCALE_H
2002-10-25 20:58:14 +00:00
#include <locale.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_LOCALE_H
#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
#ifdef HAVE_SYS_STAT_H
2002-10-25 20:58:14 +00:00
#include <sys/stat.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_SYS_STAT_H
#ifdef HAVE_STDARG_H
2002-10-25 20:58:14 +00:00
#include <stdarg.h>
2001-12-11 20:47:02 +00:00
#endif // HAVE_STDARG_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
#include <sys/time.h>
#else // !HAVE_SYS_TIME_H
#include <time.h>
#endif // HAVE_SYS_TIME_H
#endif // TIME_WITH_SYS_TIME
#ifndef MAXPATHLEN
2001-12-11 20:47:02 +00:00
#define MAXPATHLEN 255
#endif // MAXPATHLEN
2002-01-27 12:45:32 +00:00
#include <X11/Xatom.h>
#include <X11/keysym.h>
2001-12-11 20:47:02 +00:00
#include <iostream>
2002-01-20 02:17:23 +00:00
#include <memory>
2002-02-08 13:35:20 +00:00
#include <algorithm>
2003-04-14 15:01:55 +00:00
#include <functional>
2002-01-20 02:17:23 +00:00
2001-12-11 20:47:02 +00:00
using namespace std;
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 *) {
2002-12-01 13:42:15 +00:00
fprintf(stderr,
I18n::instance()->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenAnotherWMRunning,
"BScreen::BScreen: an error occured while querying the X server.\n"
" another window manager already running on display %s.\n"),
DisplayString(display));
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
2002-12-01 13:42:15 +00:00
return(-1);
2001-12-11 20:47:02 +00:00
}
FbTk::Menu *createMenuFromScreen(BScreen &screen) {
FbTk::Menu *menu = new FbMenu(*screen.menuTheme(),
2003-02-03 13:56:12 +00:00
screen.getScreenNumber(),
*screen.getImageControl(),
*screen.layerManager().getLayer(Fluxbox::instance()->getMenuLayer()));
return menu;
}
2003-02-16 15:12:08 +00:00
class FocusModelMenuItem : public FbTk::MenuItem {
2003-02-16 15:12:08 +00:00
public:
FocusModelMenuItem(const char *label, BScreen &screen, Fluxbox::FocusModel model, FbTk::RefCount<FbTk::Command> &cmd):
FbTk::MenuItem(label, cmd), m_screen(screen), m_focusmodel(model) {
2003-02-16 15:12:08 +00:00
}
bool isEnabled() const { return m_screen.getFocusModel() != m_focusmodel; }
2003-02-16 15:12:08 +00:00
void click(int button, int time) {
m_screen.saveFocusModel(m_focusmodel);
FbTk::MenuItem::click(button, time);
2003-02-16 15:12:08 +00:00
}
private:
BScreen &m_screen;
Fluxbox::FocusModel m_focusmodel;
2002-10-25 20:58:14 +00:00
};
2002-10-25 20:58:14 +00:00
2003-02-16 15:12:08 +00:00
}; // End anonymous namespace
2002-01-20 02:17:23 +00:00
template<>
void Resource<Toolbar::Placement>::
setFromString(const char *strval) {
2002-12-01 13:42:15 +00:00
if (strcasecmp(strval, "TopLeft")==0)
m_value = Toolbar::TOPLEFT;
else if (strcasecmp(strval, "BottomLeft")==0)
m_value = Toolbar::BOTTOMLEFT;
else if (strcasecmp(strval, "TopCenter")==0)
m_value = Toolbar::TOPCENTER;
else if (strcasecmp(strval, "BottomCenter")==0)
m_value = Toolbar::BOTTOMCENTER;
else if (strcasecmp(strval, "TopRight")==0)
m_value = Toolbar::TOPRIGHT;
else if (strcasecmp(strval, "BottomRight")==0)
m_value = Toolbar::BOTTOMRIGHT;
2002-12-04 22:43:18 +00:00
else if (strcasecmp(strval, "LeftTop") == 0)
m_value = Toolbar::LEFTTOP;
else if (strcasecmp(strval, "LeftCenter") == 0)
m_value = Toolbar::LEFTCENTER;
else if (strcasecmp(strval, "LeftBottom") == 0)
m_value = Toolbar::LEFTBOTTOM;
else if (strcasecmp(strval, "RightTop") == 0)
m_value = Toolbar::RIGHTTOP;
else if (strcasecmp(strval, "RightCenter") == 0)
m_value = Toolbar::RIGHTCENTER;
else if (strcasecmp(strval, "RightBottom") == 0)
m_value = Toolbar::RIGHTBOTTOM;
2002-12-01 13:42:15 +00:00
else
setDefaultValue();
2002-01-20 02:17:23 +00:00
}
template<>
void Resource<Slit::Placement>::
setFromString(const char *strval) {
if (strcasecmp(strval, "TopLeft")==0)
m_value = Slit::TOPLEFT;
else if (strcasecmp(strval, "CenterLeft")==0)
m_value = Slit::CENTERLEFT;
else if (strcasecmp(strval, "BottomLeft")==0)
m_value = Slit::BOTTOMLEFT;
else if (strcasecmp(strval, "TopCenter")==0)
m_value = Slit::TOPCENTER;
else if (strcasecmp(strval, "BottomCenter")==0)
m_value = Slit::BOTTOMCENTER;
else if (strcasecmp(strval, "TopRight")==0)
m_value = Slit::TOPRIGHT;
else if (strcasecmp(strval, "CenterRight")==0)
m_value = Slit::CENTERRIGHT;
else if (strcasecmp(strval, "BottomRight")==0)
m_value = Slit::BOTTOMRIGHT;
else
setDefaultValue();
}
2003-03-03 21:51:13 +00:00
template<>
void Resource<ToolbarHandler::ToolbarMode>::
setFromString(const char *strval) {
if (strcasecmp(strval, "Off") == 0)
m_value = ToolbarHandler::OFF;
else if (strcasecmp(strval, "None") == 0)
m_value = ToolbarHandler::NONE;
else if (strcasecmp(strval, "Icons") == 0)
m_value = ToolbarHandler::ICONS;
else if (strcasecmp(strval, "WorkspaceIcons") == 0)
m_value = ToolbarHandler::WORKSPACEICONS;
else if (strcasecmp(strval, "Workspace") == 0)
m_value = ToolbarHandler::WORKSPACE;
else if (strcasecmp(strval, "AllWindows") == 0)
m_value = ToolbarHandler::ALLWINDOWS;
else
setDefaultValue();
}
2002-01-20 02:17:23 +00:00
template<>
void Resource<Slit::Direction>::
setFromString(const char *strval) {
if (strcasecmp(strval, "Vertical") == 0)
m_value = Slit::VERTICAL;
else if (strcasecmp(strval, "Horizontal") == 0)
m_value = Slit::HORIZONTAL;
else
setDefaultValue();
}
2002-01-20 02:17:23 +00:00
string Resource<Toolbar::Placement>::
getString() {
2002-12-01 13:42:15 +00:00
switch (m_value) {
case Toolbar::TOPLEFT:
return string("TopLeft");
break;
case Toolbar::BOTTOMLEFT:
return string("BottomLeft");
break;
case Toolbar::TOPCENTER:
return string("TopCenter");
break;
case Toolbar::BOTTOMCENTER:
return string("BottomCenter");
break;
case Toolbar::TOPRIGHT:
return string("TopRight");
break;
case Toolbar::BOTTOMRIGHT:
return string("BottomRight");
break;
2002-12-04 22:43:18 +00:00
case Toolbar::LEFTTOP:
return string("LeftTop");
break;
case Toolbar::LEFTCENTER:
return string("LeftCenter");
break;
case Toolbar::LEFTBOTTOM:
return string("LeftBottom");
break;
case Toolbar::RIGHTTOP:
return string("RightTop");
break;
case Toolbar::RIGHTCENTER:
return string("RightCenter");
break;
case Toolbar::RIGHTBOTTOM:
return string("RightBottom");
break;
2002-12-01 13:42:15 +00:00
}
//default string
return string("BottomCenter");
2002-01-20 02:17:23 +00:00
}
string Resource<Slit::Placement>::
getString() {
switch (m_value) {
case Slit::TOPLEFT:
return string("TopLeft");
break;
case Slit::CENTERLEFT:
return string("CenterLeft");
break;
case Slit::BOTTOMLEFT:
return string("BottomLeft");
break;
case Slit::TOPCENTER:
return string("TopCenter");
break;
case Slit::BOTTOMCENTER:
return string("BottomCenter");
break;
case Slit::TOPRIGHT:
return string("TopRight");
break;
case Slit::CENTERRIGHT:
return string("CenterRight");
break;
case Slit::BOTTOMRIGHT:
return string("BottomRight");
break;
}
//default string
return string("BottomRight");
}
2003-03-03 21:51:13 +00:00
template<>
string Resource<ToolbarHandler::ToolbarMode>::
getString() {
switch (m_value) {
case ToolbarHandler::OFF:
return string("Off");
break;
case ToolbarHandler::NONE:
return string("None");
break;
case ToolbarHandler::LASTMODE:
case ToolbarHandler::ICONS:
return string("Icons");
break;
case ToolbarHandler::WORKSPACEICONS:
return string("Workspace");
break;
case ToolbarHandler::WORKSPACE:
return string("Workspace");
break;
case ToolbarHandler::ALLWINDOWS:
return string("AllWindows");
break;
}
// default string
return string("Icons");
}
template<>
string Resource<Slit::Direction>::
getString() {
switch (m_value) {
case Slit::VERTICAL:
return string("Vertical");
break;
case Slit::HORIZONTAL:
return string("Horizontal");
break;
}
// default string
return string("Vertical");
}
namespace {
class AddWorkspaceCmd:public FbTk::Command {
public:
explicit AddWorkspaceCmd(BScreen &scrn):m_screen(scrn) { }
void execute() {
m_screen.addWorkspace();
}
private:
BScreen &m_screen;
};
class RemoveLastWorkspaceCmd:public FbTk::Command {
public:
explicit RemoveLastWorkspaceCmd(BScreen &scrn):m_screen(scrn) { }
void execute() {
m_screen.removeLastWorkspace();
}
private:
BScreen &m_screen;
};
class ReloadStyleCmd: public FbTk::Command {
public:
void execute() {
FbCommands::SetStyleCmd cmd(Fluxbox::instance()->getStyleFilename());
cmd.execute();
}
};
void setupWorkspacemenu(BScreen &scr, FbTk::Menu &menu) {
menu.removeAll(); // clear all items
using namespace FbTk;
menu.setLabel("Workspace");
RefCount<Command> new_workspace(new AddWorkspaceCmd(scr));
RefCount<Command> remove_last(new RemoveLastWorkspaceCmd(scr));
menu.insert("New Workspace", new_workspace);
menu.insert("Remove Last", remove_last);
// for each workspace add workspace name and it's menu to our workspace menu
for (size_t workspace = 0; workspace < scr.getCount(); ++workspace) {
Workspace *wkspc = scr.getWorkspace(workspace);
menu.insert(wkspc->name().c_str(), &wkspc->menu());
}
// update graphics
menu.update();
}
};
2002-01-20 02:17:23 +00:00
template <>
void FbTk::ThemeItem<std::string>::load() { }
template <>
void FbTk::ThemeItem<std::string>::setDefaultValue() {
*(*this) = "";
}
template <>
void FbTk::ThemeItem<std::string>::setFromString(const char *str) {
*(*this) = (str ? str : "");
}
template <>
void FbTk::ThemeItem<int>::load() { }
template <>
void FbTk::ThemeItem<int>::setDefaultValue() {
*(*this) = 0;
}
template <>
void FbTk::ThemeItem<int>::setFromString(const char *str) {
if (str == 0)
return;
sscanf(str, "%d", &m_value);
}
2002-01-20 02:17:23 +00:00
BScreen::ScreenResource::ScreenResource(ResourceManager &rm,
const std::string &scrname,
const std::string &altscrname):
2002-12-01 13:42:15 +00:00
toolbar_auto_hide(rm, false, scrname+".toolbar.autoHide", altscrname+".Toolbar.AutoHide"),
image_dither(rm, false, scrname+".imageDither", altscrname+".ImageDither"),
opaque_move(rm, false, "session.opaqueMove", "Session.OpaqueMove"),
full_max(rm, true, scrname+".fullMaximization", altscrname+".FullMaximization"),
max_over_slit(rm, true, scrname+".maxOverSlit",altscrname+".MaxOverSlit"),
2003-04-15 12:22:52 +00:00
sloppy_window_grouping(rm, true,
scrname+".sloppywindowgrouping", altscrname+".SloppyWindowGrouping"),
2002-12-01 13:42:15 +00:00
workspace_warping(rm, true, scrname+".workspacewarping", altscrname+".WorkspaceWarping"),
desktop_wheeling(rm, true, scrname+".desktopwheeling", altscrname+".DesktopWheeling"),
show_window_pos(rm, true, scrname+".showwindowposition", altscrname+".ShowWindowPosition"),
focus_last(rm, true, scrname+".focusLastWindow", altscrname+".FocusLastWindow"),
focus_new(rm, true, scrname+".focusNewWindows", altscrname+".FocusNewWindows"),
antialias(rm, false, scrname+".antialias", altscrname+".Antialias"),
auto_raise(rm, false, scrname+".autoRaise", altscrname+".AutoRaise"),
click_raises(rm, true, scrname+".clickRaises", altscrname+".ClickRaises"),
2002-12-01 13:42:15 +00:00
rootcommand(rm, "", scrname+".rootCommand", altscrname+".RootCommand"),
focus_model(rm, Fluxbox::CLICKTOFOCUS, scrname+".focusModel", altscrname+".FocusModel"),
2002-12-01 13:42:15 +00:00
workspaces(rm, 1, scrname+".workspaces", altscrname+".Workspaces"),
2003-04-15 12:22:52 +00:00
toolbar_width_percent(rm, 65,
scrname+".toolbar.widthPercent", altscrname+".Toolbar.WidthPercent"),
2002-12-01 13:42:15 +00:00
edge_snap_threshold(rm, 0, scrname+".edgeSnapThreshold", altscrname+".EdgeSnapThreshold"),
2003-04-20 13:46:18 +00:00
menu_alpha(rm, 255, scrname+".menuAlpha", altscrname+".MenuAlpha"),
2003-04-15 12:22:52 +00:00
slit_layernum(rm, Fluxbox::Layer(Fluxbox::instance()->getDockLayer()),
scrname+".slit.layer", altscrname+".Slit.Layer"),
toolbar_layernum(rm, Fluxbox::Layer(Fluxbox::instance()->getDesktopLayer()),
scrname+".toolbar.layer", altscrname+".Toolbar.Layer"),
2003-03-03 21:51:13 +00:00
toolbar_mode(rm, ToolbarHandler::ICONS, scrname+".toolbar.mode", altscrname+".Toolbar.Mode"),
2003-02-23 00:56:09 +00:00
toolbar_on_head(rm, 0, scrname+".toolbar.onhead", altscrname+".Toolbar.onHead"),
2003-04-15 12:22:52 +00:00
toolbar_placement(rm, Toolbar::BOTTOMCENTER,
scrname+".toolbar.placement", altscrname+".Toolbar.Placement"),
slit_auto_hide(rm, false, scrname+".slit.autoHide", altscrname+".Slit.AutoHide"),
slit_placement(rm, Slit::BOTTOMRIGHT,
scrname+".slit.placement", altscrname+".Slit.Placement"),
slit_direction(rm, Slit::VERTICAL, scrname+".slit.direction", altscrname+".Slit.Direction")
2002-01-20 02:17:23 +00:00
{
};
2002-10-25 20:58:14 +00:00
BScreen::BScreen(ResourceManager &rm,
2002-12-01 13:42:15 +00:00
const string &screenname, const string &altscreenname,
2003-04-28 00:38:42 +00:00
int scrn, int num_layers) :
ScreenInfo(scrn),
m_clientlist_sig(*this), // client signal
m_workspacecount_sig(*this), // workspace count signal
m_workspacenames_sig(*this), // workspace names signal
m_currentworkspace_sig(*this), // current workspace signal
m_layermanager(num_layers),
cycling_focus(false),
cycling_last(0),
m_windowtheme(new FbWinFrameTheme(scrn)),
// 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_winbutton_theme(new WinButtonTheme(scrn)),
2003-04-28 00:38:42 +00:00
m_menutheme(new FbTk::MenuTheme(scrn)),
m_root_theme(new
RootTheme(scrn,
*resource.rootcommand)),
2003-05-10 22:59:32 +00:00
m_root_window(scrn),
resource(rm, screenname, altscreenname),
2003-04-28 00:38:42 +00:00
m_toolbarhandler(0) {
Display *disp = FbTk::App::instance()->display();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
event_mask = ColormapChangeMask | EnterWindowMask | PropertyChangeMask |
SubstructureRedirectMask | KeyPressMask | KeyReleaseMask |
ButtonPressMask | ButtonReleaseMask| SubstructureNotifyMask;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
XErrorHandler old = XSetErrorHandler((XErrorHandler) anotherWMRunning);
2003-05-10 22:59:32 +00:00
rootWindow().setEventMask(event_mask);
2002-12-01 13:42:15 +00:00
XSync(disp, False);
XSetErrorHandler((XErrorHandler) old);
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
managed = running;
if (! managed)
return;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
I18n *i18n = I18n::instance();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenManagingScreen,
"BScreen::BScreen: managing screen %d "
"using visual 0x%lx, depth %d\n"),
2003-05-10 22:59:32 +00:00
getScreenNumber(), XVisualIDFromVisual(rootWindow().visual()),
rootWindow().depth());
2002-12-01 13:42:15 +00:00
Fluxbox * const fluxbox = Fluxbox::instance();
2002-10-15 10:54:40 +00:00
#ifdef HAVE_GETPID
2002-12-01 13:42:15 +00:00
pid_t bpid = getpid();
2001-12-11 20:47:02 +00:00
2003-05-10 22:59:32 +00:00
XChangeProperty(disp, rootWindow().window(),
2002-12-01 13:42:15 +00:00
Fluxbox::instance()->getFluxboxPidAtom(), XA_CARDINAL,
sizeof(pid_t) * 8, PropModeReplace,
(unsigned char *) &bpid, 1);
2002-10-15 10:54:40 +00:00
#endif // HAVE_GETPID
2001-12-11 20:47:02 +00:00
2003-04-20 13:46:18 +00:00
cycling_window = focused_list.end();
2001-12-11 20:47:02 +00:00
2003-05-10 22:59:32 +00:00
XDefineCursor(disp, rootWindow().window(), fluxbox->getSessionCursor());
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
image_control =
2003-01-10 00:35:19 +00:00
new FbTk::ImageControl(scrn, true, fluxbox->colorsPerChannel(),
fluxbox->getCacheLife(), fluxbox->getCacheMax());
2002-12-01 13:42:15 +00:00
image_control->installRootColormap();
root_colormap_installed = true;
2001-12-11 20:47:02 +00:00
2003-04-15 12:22:52 +00:00
fluxbox->load_rc(*this);
2003-04-20 13:46:18 +00:00
FbTk::Menu::setAlpha(*resource.menu_alpha);
2002-12-01 13:42:15 +00:00
image_control->setDither(*resource.image_dither);
2002-01-11 10:21:44 +00:00
// setup windowtheme, toolbartheme for antialias
2003-05-07 11:36:43 +00:00
// before we load the theme
winFrameTheme().font().setAntialias(*resource.antialias);
menuTheme()->titleFont().setAntialias(*resource.antialias);
menuTheme()->frameFont().setAntialias(*resource.antialias);
2003-05-07 11:36:43 +00:00
// load database for theme engine
FbTk::ThemeManager::instance().load(fluxbox->getStyleFilename().c_str());
2003-04-25 17:39:00 +00:00
const char *s = i18n->getMessage(FBNLS::ScreenSet, FBNLS::ScreenPositionLength,
"W: 0000 x H: 0000");
2002-01-20 02:17:23 +00:00
2002-12-01 13:42:15 +00:00
int l = strlen(s);
2001-12-11 20:47:02 +00:00
geom_h = winFrameTheme().font().height();
geom_w = winFrameTheme().font().textWidth(s, l);
geom_w += m_root_theme->bevelWidth()*2;
geom_h += m_root_theme->bevelWidth()*2;
2002-12-01 13:42:15 +00:00
XSetWindowAttributes attrib;
unsigned long mask = CWBorderPixel | CWColormap | CWSaveUnder;
attrib.border_pixel = m_root_theme->borderColor().pixel();
2003-05-10 22:59:32 +00:00
attrib.colormap = rootWindow().colormap();
2002-12-01 13:42:15 +00:00
attrib.save_under = true;
2003-05-07 11:36:43 +00:00
geom_window =
2003-05-10 22:59:32 +00:00
XCreateWindow(disp, rootWindow().window(),
0, 0, geom_w, geom_h, rootTheme().borderWidth(), rootWindow().depth(),
InputOutput, rootWindow().visual(), mask, &attrib);
2002-12-01 13:42:15 +00:00
geom_visible = false;
if (winFrameTheme().labelFocusTexture().type() & FbTk::Texture::PARENTRELATIVE) {
if (winFrameTheme().titleFocusTexture().type() ==
2002-12-01 13:42:15 +00:00
(FbTk::Texture::FLAT | FbTk::Texture::SOLID)) {
geom_pixmap = None;
geom_window.setBackgroundColor(winFrameTheme().titleFocusTexture().color());
2002-12-01 13:42:15 +00:00
} else {
geom_pixmap = image_control->renderImage(geom_w, geom_h,
winFrameTheme().titleFocusTexture());
geom_window.setBackgroundPixmap(geom_pixmap);
2002-12-01 13:42:15 +00:00
}
} else {
if (winFrameTheme().labelFocusTexture().type() ==
2002-12-01 13:42:15 +00:00
(FbTk::Texture::FLAT | FbTk::Texture::SOLID)) {
geom_pixmap = None;
geom_window.setBackgroundColor(winFrameTheme().labelFocusTexture().color());
2002-12-01 13:42:15 +00:00
} else {
geom_pixmap = image_control->renderImage(geom_w, geom_h,
winFrameTheme().labelFocusTexture());
geom_window.setBackgroundPixmap(geom_pixmap);
2002-12-01 13:42:15 +00:00
}
}
workspacemenu.reset(createMenuFromScreen(*this));
2002-12-01 13:42:15 +00:00
if (*resource.workspaces != 0) {
for (int i = 0; i < *resource.workspaces; ++i) {
Workspace *wkspc = new Workspace(*this, m_layermanager, workspacesList.size());
2002-12-01 13:42:15 +00:00
workspacesList.push_back(wkspc);
}
} else { // create at least one workspace
Workspace *wkspc = new Workspace(*this, m_layermanager, workspacesList.size());
2002-12-01 13:42:15 +00:00
workspacesList.push_back(wkspc);
}
current_workspace = workspacesList.front();
2001-12-11 20:47:02 +00:00
2002-10-15 10:54:40 +00:00
#ifdef SLIT
2003-05-08 01:04:05 +00:00
m_slit.reset(new Slit(*this, *layerManager().getLayer(getSlitLayerNum()),
Fluxbox::instance()->getSlitlistFilename().c_str()));
2002-10-15 10:54:40 +00:00
#endif // SLIT
2001-12-11 20:47:02 +00:00
2003-03-03 21:51:13 +00:00
m_toolbarhandler = new ToolbarHandler(*this, getToolbarMode());
setupWorkspacemenu(*this, *workspacemenu);
m_configmenu.reset(createMenuFromScreen(*this));
setupConfigmenu(*m_configmenu.get());
2003-04-18 12:51:14 +00:00
m_configmenu->setInternalMenu();
workspacemenu->setItemSelected(2, true);
if (getToolbar()) {
getToolbar()->setPlacement(*resource.toolbar_placement);
getToolbar()->theme().font().setAntialias(*resource.antialias);
2003-03-03 21:51:13 +00:00
getToolbar()->reconfigure();
}
2002-12-13 20:19:05 +00:00
initMenu(); // create and initiate rootmenu
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
//update menus
2002-12-13 20:19:05 +00:00
m_rootmenu->update();
m_configmenu->update();
#ifdef SLIT
2002-12-01 13:42:15 +00:00
if (m_slit.get())
m_slit->reconfigure();
#endif // SLIT
2002-10-15 10:54:40 +00:00
2002-12-01 13:42:15 +00:00
// start with workspace 0
changeWorkspaceID(0);
updateNetizenWorkspaceCount();
2002-12-01 13:42:15 +00:00
int i;
unsigned int nchild;
Window r, p, *children;
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 (i = 0; i < (int) nchild; i++) {
2002-12-01 13:42:15 +00:00
if (children[i] == None) continue;
2003-04-25 16:07:46 +00:00
XWMHints *wmhints = XGetWMHints(FbTk::App::instance()->display(),
2002-12-01 13:42:15 +00:00
children[i]);
if (wmhints) {
if ((wmhints->flags & IconWindowHint) &&
(wmhints->icon_window != children[i]))
for (int j = 0; j < (int) nchild; j++) {
if (children[j] == wmhints->icon_window) {
children[j] = None;
break;
}
}
XFree(wmhints);
}
}
// manage shown windows
for (i = 0; i < (int) nchild; ++i) {
if (children[i] == None || (! fluxbox->validateWindow(children[i])))
continue;
XWindowAttributes attrib;
if (XGetWindowAttributes(disp, children[i],
&attrib)) {
if (attrib.override_redirect)
continue;
if (attrib.map_state != IsUnmapped) {
FluxboxWindow *win = createWindow(children[i]);
2002-12-01 13:42:15 +00:00
if (win) {
XMapRequestEvent mre;
mre.window = children[i];
win->mapRequestEvent(mre);
2002-12-01 13:42:15 +00:00
}
}
}
}
2003-03-03 21:51:13 +00:00
if (! isSloppyFocus() && getToolbar() != 0) {
2003-04-23 10:51:17 +00:00
XSetInputFocus(disp, getToolbar()->window().window(),
2002-12-01 13:42:15 +00:00
RevertToParent, CurrentTime);
}
2003-03-03 21:51:13 +00:00
// set the toolbarhandler after the windows are setup, so it catches their state properly
2002-12-01 13:42:15 +00:00
XFree(children);
XFlush(disp);
2002-02-08 13:35:20 +00:00
}
2001-12-11 20:47:02 +00:00
BScreen::~BScreen() {
2002-12-01 13:42:15 +00:00
if (! managed)
return;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
if (geom_pixmap != None)
image_control->removeImage(geom_pixmap);
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
removeWorkspaceNames();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
Workspaces::iterator w_it = workspacesList.begin();
Workspaces::iterator w_it_end = workspacesList.end();
for(; w_it != w_it_end; ++w_it) {
delete (*w_it);
}
workspacesList.clear();
2002-05-17 16:30:24 +00:00
2002-12-01 13:42:15 +00:00
Icons::iterator i_it = iconList.begin();
Icons::iterator i_it_end = iconList.end();
for(; i_it != i_it_end; ++i_it) {
delete (*i_it);
}
iconList.clear();
2002-05-17 16:30:24 +00:00
2002-12-01 13:42:15 +00:00
Netizens::iterator n_it = netizenList.begin();
Netizens::iterator n_it_end = netizenList.end();
for(; n_it != n_it_end; ++n_it) {
delete (*n_it);
}
netizenList.clear();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
delete image_control;
2001-12-11 20:47:02 +00:00
}
2003-03-03 21:51:13 +00:00
const FbTk::Menu &BScreen::getToolbarModemenu() const {
return m_toolbarhandler->getModeMenu();
}
FbTk::Menu &BScreen::getToolbarModemenu() {
return m_toolbarhandler->getModeMenu();
}
unsigned int BScreen::getCurrentWorkspaceID() const {
return current_workspace->workspaceID();
}
2003-04-25 11:56:13 +00:00
Pixmap BScreen::rootPixmap() const {
Pixmap root_pm = 0;
Display *disp = FbTk::App::instance()->display();
Atom real_type;
int real_format;
unsigned long items_read, items_left;
unsigned int *data;
2003-05-10 22:59:32 +00:00
if (XGetWindowProperty(disp, rootWindow().window(),
2003-04-25 11:56:13 +00:00
XInternAtom(disp, "_XROOTPMAP_ID", false),
0L, 1L,
false, XA_PIXMAP, &real_type,
&real_format, &items_read, &items_left,
(unsigned char **) &data) == Success &&
items_read) {
root_pm = (Pixmap) (*data);
XFree(data);
}
return root_pm;
}
/// TODO
unsigned int BScreen::getMaxLeft() const {
return 0;
}
///!! TODO
unsigned int BScreen::getMaxRight() const {
return getWidth();
}
///!! TODO
unsigned int BScreen::getMaxTop() const {
return 0;
}
///!! TODO
unsigned int BScreen::getMaxBottom() const {
return getHeight();
}
void BScreen::reconfigure() {
2002-10-15 10:54:40 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): BScreen::reconfigure"<<endl;
2002-10-15 10:54:40 +00:00
#endif // DEBUG
2003-04-20 13:46:18 +00:00
FbTk::Menu::setAlpha(*resource.menu_alpha);
2003-04-15 12:22:52 +00:00
Fluxbox::instance()->loadRootCommand(*this);
// setup windowtheme, toolbartheme for antialias
winFrameTheme().font().setAntialias(*resource.antialias);
m_menutheme->titleFont().setAntialias(*resource.antialias);
m_menutheme->frameFont().setAntialias(*resource.antialias);
if (getToolbar() && getToolbar()->theme().font().isAntialias() != *resource.antialias)
getToolbar()->theme().font().setAntialias(*resource.antialias);
// load theme
std::string theme_filename(Fluxbox::instance()->getStyleFilename());
FbTk::ThemeManager::instance().load(theme_filename.c_str());
2002-12-01 13:42:15 +00:00
I18n *i18n = I18n::instance();
const char *s = i18n->getMessage(FBNLS::ScreenSet,
FBNLS::ScreenPositionLength,
"W: 0000 x H: 0000");
2002-12-01 13:42:15 +00:00
int l = strlen(s);
//TODO: repeated from somewhere else?
geom_h = winFrameTheme().font().height();
geom_w = winFrameTheme().font().textWidth(s, l);
geom_w += m_root_theme->bevelWidth()*2;
geom_h += m_root_theme->bevelWidth()*2;
2002-12-01 13:42:15 +00:00
Pixmap tmp = geom_pixmap;
if (winFrameTheme().labelFocusTexture().type() & FbTk::Texture::PARENTRELATIVE) {
if (winFrameTheme().titleFocusTexture().type() ==
2002-12-01 13:42:15 +00:00
(FbTk::Texture::FLAT | FbTk::Texture::SOLID)) {
geom_pixmap = None;
geom_window.setBackgroundColor(winFrameTheme().titleFocusTexture().color());
2002-12-01 13:42:15 +00:00
} else {
geom_pixmap = image_control->renderImage(geom_w, geom_h,
winFrameTheme().titleFocusTexture());
geom_window.setBackgroundPixmap(geom_pixmap);
2002-12-01 13:42:15 +00:00
}
} else {
if (winFrameTheme().labelFocusTexture().type() ==
2002-12-01 13:42:15 +00:00
(FbTk::Texture::FLAT | FbTk::Texture::SOLID)) {
geom_pixmap = None;
geom_window.setBackgroundColor(winFrameTheme().labelFocusTexture().color());
2002-12-01 13:42:15 +00:00
} else {
geom_pixmap = image_control->renderImage(geom_w, geom_h,
winFrameTheme().labelFocusTexture());
geom_window.setBackgroundPixmap(geom_pixmap);
2002-12-01 13:42:15 +00:00
}
}
if (tmp)
image_control->removeImage(tmp);
2002-12-01 13:42:15 +00:00
geom_window.setBorderWidth(m_root_theme->borderWidth());
geom_window.setBorderColor(m_root_theme->borderColor());
2002-12-01 13:42:15 +00:00
//reconfigure menus
workspacemenu->reconfigure();
m_configmenu->reconfigure();
2002-04-28 19:54:10 +00:00
initMenu();
m_rootmenu->reconfigure();
2001-12-11 20:47:02 +00:00
if (getToolbar()) {
getToolbar()->setPlacement(*resource.toolbar_placement);
getToolbar()->reconfigure();
2003-03-03 21:51:13 +00:00
}
#ifdef SLIT
2003-02-23 00:56:09 +00:00
if (m_slit.get()) {
m_slit->setPlacement(static_cast<Slit::Placement>(getSlitPlacement()));
m_slit->setDirection(static_cast<Slit::Direction>(getSlitDirection()));
2002-12-01 13:42:15 +00:00
m_slit->reconfigure();
2003-02-23 00:56:09 +00:00
}
#endif // SLIT
2002-12-01 13:42:15 +00:00
//reconfigure workspaces
2003-04-14 15:01:55 +00:00
for_each(workspacesList.begin(),
workspacesList.end(),
mem_fun(&Workspace::reconfigure));
2002-12-01 13:42:15 +00:00
//reconfigure Icons
2003-04-14 15:01:55 +00:00
for_each(iconList.begin(),
iconList.end(),
mem_fun(&FluxboxWindow::reconfigure));
2002-12-01 13:42:15 +00:00
image_control->timeout();
2001-12-11 20:47:02 +00:00
}
void BScreen::rereadMenu() {
2002-12-01 13:42:15 +00:00
initMenu();
2001-12-11 20:47:02 +00:00
2002-12-13 20:19:05 +00:00
m_rootmenu->reconfigure();
2001-12-11 20:47:02 +00:00
}
void BScreen::removeWorkspaceNames() {
2002-12-01 13:42:15 +00:00
workspaceNames.erase(workspaceNames.begin(), workspaceNames.end());
2001-12-11 20:47:02 +00:00
}
void BScreen::updateWorkspaceNamesAtom() {
2002-12-01 13:42:15 +00:00
m_workspacenames_sig.notify();
2001-12-11 20:47:02 +00:00
}
void BScreen::addIcon(FluxboxWindow *w) {
2002-12-01 13:42:15 +00:00
if (! w) return;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
w->setWindowNumber(iconList.size());
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
iconList.push_back(w);
2001-12-11 20:47:02 +00:00
}
void BScreen::removeIcon(FluxboxWindow *w) {
2002-12-01 13:42:15 +00:00
if (! w)
return;
2002-02-20 22:41:13 +00:00
2003-04-14 15:01:55 +00:00
Icons::iterator erase_it = remove_if(iconList.begin(),
iconList.end(),
bind2nd(equal_to<FluxboxWindow *>(), w));
if (erase_it != iconList.end())
iconList.erase(erase_it);
2003-03-03 21:51:13 +00:00
2002-12-01 13:42:15 +00:00
Icons::iterator it = iconList.begin();
Icons::iterator it_end = iconList.end();
for (int i = 0; it != it_end; ++it, ++i) {
(*it)->setWindowNumber(i);
}
2001-12-11 20:47:02 +00:00
}
void BScreen::removeWindow(FluxboxWindow *win) {
if (win->isIconic())
removeIcon(win);
else
getWorkspace(win->getWorkspaceNumber())->removeWindow(win);
}
2001-12-11 20:47:02 +00:00
void BScreen::removeClient(WinClient &client) {
WinClient *cyc = *cycling_window;
2003-05-07 13:50:41 +00:00
FluxboxWindow *focused = Fluxbox::instance()->getFocusedWindow();
focused_list.remove(&client);
if (cyc == &client) {
cycling_window = focused_list.end();
} else if (focused && &focused->winClient() == &client) {
// if we are focused, then give our focus to our transient parent
// or revert normally
if (client.transientFor() && client.transientFor()->fbwindow())
client.transientFor()->fbwindow()->setInputFocus();
else
2003-05-11 15:26:34 +00:00
Fluxbox::instance()->revertFocus(focused->screen());
}
}
2002-03-23 15:14:45 +00:00
FluxboxWindow *BScreen::getIcon(unsigned int index) {
2002-12-01 13:42:15 +00:00
if (index < iconList.size())
return iconList[index];
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
return 0;
2001-12-11 20:47:02 +00:00
}
2002-10-15 10:54:40 +00:00
void BScreen::setAntialias(bool value) {
2002-12-01 13:42:15 +00:00
if (*resource.antialias == value)
return;
resource.antialias = value;
reconfigure();
2002-10-15 10:54:40 +00:00
}
2001-12-11 20:47:02 +00:00
int BScreen::addWorkspace() {
Workspace *wkspc = new Workspace(*this, m_layermanager, workspacesList.size());
2002-12-01 13:42:15 +00:00
workspacesList.push_back(wkspc);
addWorkspaceName(wkspc->name().c_str()); // update names
//add workspace to workspacemenu
workspacemenu->insert(wkspc->name().c_str(), &wkspc->menu(),
wkspc->workspaceID() + 2); //+2 so we add it after "remove last"
2002-02-28 15:46:01 +00:00
2002-12-01 13:42:15 +00:00
workspacemenu->update();
saveWorkspaces(workspacesList.size());
2003-03-03 21:51:13 +00:00
if (getToolbar() != 0)
getToolbar()->reconfigure();
2002-12-01 13:42:15 +00:00
updateNetizenWorkspaceCount();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
return workspacesList.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() {
2002-12-01 13:42:15 +00:00
if (workspacesList.size() <= 1)
return 0;
Workspace *wkspc = workspacesList.back();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
if (current_workspace->workspaceID() == wkspc->workspaceID())
changeWorkspaceID(current_workspace->workspaceID() - 1);
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
wkspc->removeAll();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
workspacemenu->remove(wkspc->workspaceID()+2); // + 2 is where workspaces starts
workspacemenu->update();
2002-10-25 20:58:14 +00:00
2002-12-01 13:42:15 +00:00
//remove last workspace
workspacesList.pop_back();
delete wkspc;
2001-12-11 20:47:02 +00:00
2003-03-03 21:51:13 +00:00
if (getToolbar() != 0)
getToolbar()->reconfigure();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
updateNetizenWorkspaceCount();
saveWorkspaces(workspacesList.size());
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
return workspacesList.size();
2001-12-11 20:47:02 +00:00
}
2002-03-23 15:14:45 +00:00
void BScreen::changeWorkspaceID(unsigned int id) {
2003-05-11 11:47:19 +00:00
if (! current_workspace || id >= workspacesList.size() ||
id == current_workspace->workspaceID())
2002-12-01 13:42:15 +00:00
return;
2003-05-11 11:47:19 +00:00
XSync(FbTk::App::instance()->display(), true);
FluxboxWindow *focused = Fluxbox::instance()->getFocusedWindow();
2002-10-11 10:22:06 +00:00
#ifdef DEBUG
2003-05-11 11:47:19 +00:00
cerr<<__FILE__<<"("<<__FUNCTION__<<"): focused = "<<focused<<endl;
2002-10-11 10:22:06 +00:00
#endif // DEBUG
2003-05-11 11:47:19 +00:00
if (focused && focused->isMoving()) {
if (doOpaqueMove())
reassociateWindow(focused, id, true);
// don't reassociate if not opaque moving
focused->pauseMoving();
}
2003-05-11 11:47:19 +00:00
// reassociate all windows that are stuck to the new workspace
Workspace *wksp = getCurrentWorkspace();
2003-05-11 17:14:41 +00:00
Workspace::Windows wins = wksp->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
}
2003-05-11 11:47:19 +00:00
current_workspace->hideAll();
2001-12-11 20:47:02 +00:00
2003-05-11 11:47:19 +00:00
workspacemenu->setItemSelected(current_workspace->workspaceID() + 2, false);
2001-12-11 20:47:02 +00:00
2003-05-11 11:47:19 +00:00
// set new workspace
current_workspace = getWorkspace(id);
2001-12-11 20:47:02 +00:00
2003-05-11 11:47:19 +00:00
workspacemenu->setItemSelected(current_workspace->workspaceID() + 2, true);
if (getToolbar() != 0)
getToolbar()->redrawWorkspaceLabel(true);
2001-12-11 20:47:02 +00:00
2003-05-11 11:47:19 +00:00
current_workspace->showAll();
2001-12-11 20:47:02 +00:00
2003-05-11 11:47:19 +00:00
if (focused && (focused->isStuck() || focused->isMoving())) {
focused->setInputFocus();
} else
2003-05-11 15:26:34 +00:00
Fluxbox::instance()->revertFocus(*this);
2003-05-11 11:47:19 +00:00
if (focused && focused->isMoving()) {
focused->resumeMoving();
2002-12-01 13:42:15 +00:00
}
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
updateNetizenCurrentWorkspace();
2001-12-11 20:47:02 +00:00
}
void BScreen::sendToWorkspace(unsigned int id, FluxboxWindow *win, bool changeWS) {
2002-12-01 13:42:15 +00:00
if (! current_workspace || id >= workspacesList.size())
return;
2002-12-01 13:42:15 +00:00
if (!win)
win = Fluxbox::instance()->getFocusedWindow();
2002-12-01 13:42:15 +00:00
if (id != current_workspace->workspaceID()) {
2003-04-25 16:07:46 +00:00
XSync(FbTk::App::instance()->display(), True);
2003-05-11 13:36:12 +00:00
if (win && &win->screen() == this &&
2002-12-01 13:42:15 +00:00
(! win->isStuck())) {
2002-12-01 13:42:15 +00:00
if (win->isIconic()) {
win->deiconify();
}
2002-09-10 10:59:57 +00:00
2002-12-01 13:42:15 +00:00
win->withdraw();
reassociateWindow(win, id, true);
2002-10-11 10:22:06 +00:00
2002-12-01 13:42:15 +00:00
// change workspace ?
if (changeWS) {
changeWorkspaceID(id);
win->setInputFocus();
}
2002-10-11 10:22:06 +00:00
#ifdef DEBUG
cerr<<__FILE__<<": Sending to id = "<<id<<endl;
cerr<<__FILE__<<": win->workspaceId="<<win->getWorkspaceNumber()<<endl;
2002-10-11 10:22:06 +00:00
#endif //DEBUG
2002-12-01 13:42:15 +00:00
}
2002-10-11 10:22:06 +00:00
2002-12-01 13:42:15 +00:00
}
}
2001-12-11 20:47:02 +00:00
2003-05-11 17:14:41 +00:00
void BScreen::addNetizen(Window win) {
Netizen *net = new Netizen(*this, win);
netizenList.push_back(net);
2001-12-11 20:47:02 +00:00
2003-05-11 17:14:41 +00:00
net->sendWorkspaceCount();
net->sendCurrentWorkspace();
2001-12-11 20:47:02 +00:00
2003-05-11 17:14:41 +00:00
// send all windows to netizen
2002-12-01 13:42:15 +00:00
Workspaces::iterator it = workspacesList.begin();
Workspaces::iterator it_end = workspacesList.end();
for (; it != it_end; ++it) {
2003-05-11 17:14:41 +00:00
Workspace::Windows::iterator win_it = (*it)->windowList().begin();
Workspace::Windows::iterator win_it_end = (*it)->windowList().end();
for (; win_it != win_it_end; ++win_it) {
net->sendWindowAdd((*win_it)->getClientWindow(),
2002-12-01 13:42:15 +00:00
(*it)->workspaceID());
}
}
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
Window f = ((Fluxbox::instance()->getFocusedWindow()) ?
2002-10-25 20:58:14 +00:00
Fluxbox::instance()->getFocusedWindow()->getClientWindow() : None);
2003-05-11 17:14:41 +00:00
net->sendWindowFocus(f);
2001-12-11 20:47:02 +00:00
}
void BScreen::removeNetizen(Window w) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
2003-05-10 18:15:23 +00:00
if ((*it)->window() == w) {
2002-12-01 13:42:15 +00:00
Netizen *n = *it;
delete n;
netizenList.erase(it);
break;
}
}
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenCurrentWorkspace() {
2002-12-01 13:42:15 +00:00
m_currentworkspace_sig.notify();
2003-04-14 15:01:55 +00:00
for_each(netizenList.begin(),
netizenList.end(),
mem_fun(&Netizen::sendCurrentWorkspace));
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWorkspaceCount() {
2003-04-14 15:01:55 +00:00
for_each(netizenList.begin(),
netizenList.end(),
mem_fun(&Netizen::sendWorkspaceCount));
2002-09-07 20:22:08 +00:00
2002-12-01 13:42:15 +00:00
m_workspacecount_sig.notify();
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWindowFocus() {
2002-02-08 13:35:20 +00:00
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
Window f = ((Fluxbox::instance()->getFocusedWindow()) ?
Fluxbox::instance()->getFocusedWindow()->getClientWindow() : None);
for (; it != it_end; ++it) {
(*it)->sendWindowFocus(f);
}
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWindowAdd(Window w, unsigned long p) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
(*it)->sendWindowAdd(w, p);
}
2002-09-07 20:22:08 +00:00
2002-12-01 13:42:15 +00:00
m_clientlist_sig.notify();
2002-02-20 22:41:13 +00:00
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWindowDel(Window w) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
(*it)->sendWindowDel(w);
}
2002-09-07 20:22:08 +00:00
2002-12-01 13:42:15 +00:00
m_clientlist_sig.notify();
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWindowRaise(Window w) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
(*it)->sendWindowRaise(w);
}
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenWindowLower(Window w) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
(*it)->sendWindowLower(w);
}
2001-12-11 20:47:02 +00:00
}
void BScreen::updateNetizenConfigNotify(XEvent *e) {
2002-12-01 13:42:15 +00:00
Netizens::iterator it = netizenList.begin();
Netizens::iterator it_end = netizenList.end();
for (; it != it_end; ++it) {
(*it)->sendConfigNotify(e);
}
2001-12-11 20:47:02 +00:00
}
FluxboxWindow *BScreen::createWindow(Window client) {
2003-04-14 15:01:55 +00:00
FluxboxWindow *win = new FluxboxWindow(client, *this,
2003-02-03 13:56:12 +00:00
winFrameTheme(), *menuTheme(),
2003-02-09 14:11:14 +00:00
*layerManager().getLayer(Fluxbox::instance()->getNormalLayer()));
#ifdef SLIT
if (win->initialState() == WithdrawnState)
getSlit()->addClient(win->getClientWindow());
#endif // SLIT
if (!win->isManaged()) {
delete win;
return 0;
} else {
// always put on end of focused list, if it gets focused it'll get pushed up
// there is only the one win client at this stage
focused_list.push_back(&win->winClient());
//TODO: is next line needed?
Fluxbox::instance()->saveWindowSearch(client, win);
setupWindowActions(*win);
Fluxbox::instance()->attachSignals(*win);
2003-02-16 17:57:54 +00:00
}
if (win->getWorkspaceNumber() == getCurrentWorkspaceID() || win->isStuck()) {
win->show();
2003-02-16 17:57:54 +00:00
}
XSync(FbTk::App::instance()->display(), False);
return win;
}
2003-04-14 15:01:55 +00:00
FluxboxWindow *BScreen::createWindow(WinClient &client) {
FluxboxWindow *win = new FluxboxWindow(client, *this,
winFrameTheme(), *menuTheme(),
*layerManager().getLayer(Fluxbox::instance()->getNormalLayer()));
#ifdef SLIT
if (win->initialState() == WithdrawnState)
getSlit()->addClient(win->getClientWindow());
#endif // SLIT
if (!win->isManaged()) {
delete win;
return 0;
}
// don't add to focused_list, as it should already be in there (since the
// WinClient already exists).
2003-04-14 15:01:55 +00:00
Fluxbox::instance()->saveWindowSearch(client.window(), win);
setupWindowActions(*win);
Fluxbox::instance()->attachSignals(*win);
2003-04-14 15:01:55 +00:00
if (win->getWorkspaceNumber() == getCurrentWorkspaceID() || win->isStuck()) {
win->show();
}
return win;
}
void BScreen::setupWindowActions(FluxboxWindow &win) {
FbWinFrame &frame = win.frame();
typedef FbTk::RefCount<FbTk::Command> CommandRef;
using namespace FbTk;
2003-01-07 02:06:06 +00:00
typedef RefCount<Command> CommandRef;
typedef SimpleCommand<FluxboxWindow> WindowCmd;
CommandRef iconify_cmd(new WindowCmd(win, &FluxboxWindow::iconify));
CommandRef maximize_cmd(new WindowCmd(win, &FluxboxWindow::maximize));
CommandRef maximize_vert_cmd(new WindowCmd(win, &FluxboxWindow::maximizeVertical));
CommandRef maximize_horiz_cmd(new WindowCmd(win, &FluxboxWindow::maximizeHorizontal));
CommandRef close_cmd(new WindowCmd(win, &FluxboxWindow::close));
CommandRef shade_cmd(new WindowCmd(win, &FluxboxWindow::shade));
CommandRef raise_cmd(new WindowCmd(win, &FluxboxWindow::raise));
2003-02-16 17:57:54 +00:00
CommandRef lower_cmd(new WindowCmd(win, &FluxboxWindow::lower));
CommandRef raise_and_focus_cmd(new WindowCmd(win, &FluxboxWindow::raiseAndFocus));
2003-01-07 02:06:06 +00:00
CommandRef stick_cmd(new WindowCmd(win, &FluxboxWindow::stick));
CommandRef show_menu_cmd(new WindowCmd(win, &FluxboxWindow::popupMenu));
// clear old buttons from frame
frame.removeAllButtons();
//!! TODO: fix this ugly hack
// get titlebar configuration
const vector<Fluxbox::Titlebar> *dir = &Fluxbox::instance()->getTitlebarLeft();
for (char c=0; c<2; c++) {
for (size_t i=0; i< dir->size(); ++i) {
//create new buttons
FbTk::Button *newbutton = 0;
if (win.isIconifiable() && (*dir)[i] == Fluxbox::MINIMIZE) {
newbutton = new WinButton(win, *m_winbutton_theme.get(),
2003-04-28 00:38:42 +00:00
WinButton::MINIMIZE,
frame.titlebar(),
0, 0, 10, 10);
newbutton->setOnClick(iconify_cmd);
} else if (win.isMaximizable() && (*dir)[i] == Fluxbox::MAXIMIZE) {
newbutton = new WinButton(win, *m_winbutton_theme.get(),
2003-04-28 00:38:42 +00:00
WinButton::MAXIMIZE,
frame.titlebar(),
0, 0, 10, 10);
newbutton->setOnClick(maximize_cmd, 1);
newbutton->setOnClick(maximize_horiz_cmd, 3);
newbutton->setOnClick(maximize_vert_cmd, 2);
} else if (win.isClosable() && (*dir)[i] == Fluxbox::CLOSE) {
newbutton = new WinButton(win, *m_winbutton_theme.get(),
2003-04-28 00:38:42 +00:00
WinButton::CLOSE,
frame.titlebar(),
0, 0, 10, 10);
2003-01-07 02:06:06 +00:00
newbutton->setOnClick(close_cmd);
#ifdef DEBUG
cerr<<__FILE__<<": Creating close button"<<endl;
#endif // DEBUG
} else if ((*dir)[i] == Fluxbox::STICK) {
WinButton *winbtn = new WinButton(win, *m_winbutton_theme.get(),
2003-04-28 00:38:42 +00:00
WinButton::STICK,
2003-04-26 18:27:56 +00:00
frame.titlebar(),
0, 0, 10, 10);
2003-04-25 17:39:00 +00:00
win.stateSig().attach(winbtn);
winbtn->setOnClick(stick_cmd);
newbutton = winbtn;
} else if ((*dir)[i] == Fluxbox::SHADE) {
WinButton *winbtn = new WinButton(win, *m_winbutton_theme.get(),
2003-04-28 00:38:42 +00:00
WinButton::SHADE,
2003-04-26 18:27:56 +00:00
frame.titlebar(),
0, 0, 10, 10);
2003-04-25 17:39:00 +00:00
winbtn->setOnClick(shade_cmd);
}
if (newbutton != 0) {
newbutton->show();
if (c == 0)
frame.addLeftButton(newbutton);
else
frame.addRightButton(newbutton);
}
} //end for i
dir = &Fluxbox::instance()->getTitlebarRight();
} // end for c
2003-04-14 15:01:55 +00:00
frame.reconfigure();
2003-01-07 02:06:06 +00:00
// setup titlebar
frame.setOnClickTitlebar(raise_and_focus_cmd, 1, false, true); // on press with button 1
2003-01-07 02:06:06 +00:00
frame.setOnClickTitlebar(shade_cmd, 1, true); // doubleclick with button 1
frame.setOnClickTitlebar(show_menu_cmd, 3); // on release with button 3
frame.setOnClickTitlebar(lower_cmd, 2); // on release with button 2
2003-01-07 02:06:06 +00:00
frame.setDoubleClickTime(Fluxbox::instance()->getDoubleClickInterval());
// setup menu
FbTk::Menu &menu = win.getWindowmenu();
menu.removeAll(); // clear old items
menu.disableTitle(); // not titlebar
2003-01-07 02:06:06 +00:00
// set new menu items
menu.insert("Shade", shade_cmd);
menu.insert("Stick", stick_cmd);
menu.insert("Maximize", maximize_cmd);
menu.insert("Maximize Vertical", maximize_vert_cmd);
menu.insert("Maximize Horizontal", maximize_horiz_cmd);
menu.insert("Iconify", iconify_cmd);
menu.insert("Raise", raise_cmd);
menu.insert("Lower", lower_cmd);
2003-04-14 15:01:55 +00:00
menu.insert("Layer...", &win.getLayermenu());
CommandRef next_client_cmd(new WindowCmd(win, &FluxboxWindow::nextClient));
CommandRef prev_client_cmd(new WindowCmd(win, &FluxboxWindow::prevClient));
menu.insert("Next Client", next_client_cmd);
menu.insert("Prev Client", prev_client_cmd);
2003-01-07 02:06:06 +00:00
menu.insert("<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>");
menu.insert("Close", close_cmd);
menu.reconfigure(); // update graphics
}
2001-12-11 20:47:02 +00:00
void BScreen::saveStrftimeFormat(const char *format) {
2002-12-01 13:42:15 +00:00
//make sure std::string don't get 0 string
resource.strftime_format = (format ? format : "");
2001-12-11 20:47:02 +00:00
}
void BScreen::addWorkspaceName(const char *name) {
2002-12-01 13:42:15 +00:00
workspaceNames.push_back(name);
2001-12-11 20:47:02 +00:00
}
2002-08-11 22:28:18 +00:00
string BScreen::getNameOfWorkspace(unsigned int workspace) const {
2002-12-01 13:42:15 +00:00
if (workspace < workspaceNames.size()) {
return workspaceNames[workspace];
} else {
return "";
}
2001-12-11 20:47:02 +00:00
}
2003-04-28 00:38:42 +00:00
void BScreen::reassociateWindow(FluxboxWindow *w, unsigned int wkspc_id,
bool ignore_sticky) {
2003-04-14 15:01:55 +00:00
if (w == 0)
return;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
if (wkspc_id >= getCount()) {
wkspc_id = current_workspace->workspaceID();
2002-10-11 10:22:06 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): wkspc_id >= getCount()"<<endl;
2002-10-11 10:22:06 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
}
2003-03-03 21:51:13 +00:00
if (!w->isIconic() && w->getWorkspaceNumber() == 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()) {
getWorkspace(w->getWorkspaceNumber())->removeWindow(w);
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
}
void BScreen::nextFocus(int opts) {
2002-12-01 13:42:15 +00:00
bool have_focused = false;
int focused_window_number = -1;
FluxboxWindow *focused = Fluxbox::instance()->getFocusedWindow();
2003-05-11 17:14:41 +00:00
const int num_windows = getCurrentWorkspace()->numberOfWindows();
2002-12-01 13:42:15 +00:00
if (focused != 0) {
2003-05-11 13:36:12 +00:00
if (focused->screen().getScreenNumber() ==
2002-12-01 13:42:15 +00:00
getScreenNumber()) {
have_focused = true;
focused_window_number = focused->getWindowNumber();
}
}
if (num_windows >= 1) {
if (!(opts & CYCLELINEAR)) {
if (!cycling_focus) {
cycling_focus = True;
cycling_window = focused_list.begin();
cycling_last = 0;
} else {
// already cycling, so restack to put windows back in their proper order
m_layermanager.restack();
}
// if it is stacked, we want the highest window in the focused list
// that is on the same workspace
FocusedWindows::iterator it = cycling_window;
FocusedWindows::iterator it_end = focused_list.end();
while (true) {
++it;
if (it == it_end) {
it = focused_list.begin();
}
// give up [do nothing] if we reach the current focused again
if ((*it) == (*cycling_window)) {
break;
}
2002-12-01 13:42:15 +00:00
FluxboxWindow *fbwin = (*it)->m_win;
if (fbwin && !fbwin->isIconic() &&
(fbwin->isStuck()
|| fbwin->getWorkspaceNumber() == getCurrentWorkspaceID())) {
// either on this workspace, or stuck
// keep track of the originally selected window in a set
WinClient &last_client = fbwin->winClient();
if (! (doSkipWindow(fbwin, opts) || !fbwin->setCurrentClient(**it)) ) {
// moved onto a new fbwin
if (!cycling_last || cycling_last->fbwindow() != fbwin) {
if (cycling_last)
// set back to orig current Client in that fbwin
cycling_last->fbwindow()->setCurrentClient(*cycling_last, false);
cycling_last = &last_client;
}
fbwin->tempRaise();
break;
}
}
}
cycling_window = it;
} else { // not stacked cycling
Workspace *wksp = getCurrentWorkspace();
2003-05-11 17:14:41 +00:00
Workspace::Windows &wins = wksp->windowList();
Workspace::Windows::iterator it = wins.begin();
if (!have_focused) {
focused = (*it);
} else {
for (; (*it) != focused; ++it) //get focused window iterator
continue;
}
do {
++it;
if (it == wins.end())
it = wins.begin();
// see if the window should be skipped
if (! (doSkipWindow((*it), opts) || !(*it)->setInputFocus()) )
break;
} while ((*it) != focused);
if ((*it) != focused && it != wins.end())
(*it)->raise();
2002-12-01 13:42:15 +00:00
}
}
2001-12-11 20:47:02 +00:00
}
void BScreen::prevFocus(int opts) {
2002-12-01 13:42:15 +00:00
bool have_focused = false;
int focused_window_number = -1;
FluxboxWindow *focused;
2003-05-11 17:14:41 +00:00
int num_windows = getCurrentWorkspace()->numberOfWindows();
2002-12-01 13:42:15 +00:00
if ((focused = Fluxbox::instance()->getFocusedWindow())) {
2003-05-11 13:36:12 +00:00
if (focused->screen().getScreenNumber() ==
2002-12-01 13:42:15 +00:00
getScreenNumber()) {
have_focused = true;
focused_window_number = focused->getWindowNumber();
}
}
if (num_windows >= 1) {
if (!(opts & CYCLELINEAR)) {
if (!cycling_focus) {
2003-05-11 17:14:41 +00:00
cycling_focus = true;
cycling_window = focused_list.end();
cycling_last = 0;
} else {
// already cycling, so restack to put windows back in their proper order
m_layermanager.restack();
}
// if it is stacked, we want the highest window in the focused list
// that is on the same workspace
FocusedWindows::iterator it = cycling_window;
FocusedWindows::iterator it_end = focused_list.end();
while (true) {
--it;
if (it == it_end) {
it = focused_list.end();
--it;
}
// give up [do nothing] if we reach the current focused again
if ((*it) == (*cycling_window)) {
break;
}
2002-12-01 13:42:15 +00:00
FluxboxWindow *fbwin = (*it)->m_win;
if (fbwin && !fbwin->isIconic() &&
(fbwin->isStuck()
|| fbwin->getWorkspaceNumber() == getCurrentWorkspaceID())) {
// either on this workspace, or stuck
// keep track of the originally selected window in a set
WinClient &last_client = fbwin->winClient();
if (! (doSkipWindow(fbwin, opts) || !fbwin->setCurrentClient(**it)) ) {
// moved onto a new fbwin
if (!cycling_last || cycling_last->fbwindow() != fbwin) {
if (cycling_last)
// set back to orig current Client in that fbwin
cycling_last->fbwindow()->setCurrentClient(*cycling_last, false);
cycling_last = &last_client;
}
fbwin->tempRaise();
break;
}
}
}
cycling_window = it;
} else { // not stacked cycling
Workspace *wksp = getCurrentWorkspace();
2003-05-11 17:14:41 +00:00
Workspace::Windows &wins = wksp->windowList();
Workspace::Windows::iterator it = wins.begin();
if (!have_focused) {
focused = (*it);
} else {
for (; (*it) != focused; ++it) //get focused window iterator
continue;
}
do {
if (it == wins.begin())
it = wins.end();
--it;
// see if the window should be skipped
if (! (doSkipWindow((*it), opts) || !(*it)->setInputFocus()) )
break;
} while ((*it) != focused);
if ((*it) != focused && it != wins.end())
(*it)->raise();
2002-12-01 13:42:15 +00:00
}
}
2001-12-11 20:47:02 +00:00
}
void BScreen::raiseFocus() {
2002-12-01 13:42:15 +00:00
bool have_focused = false;
int focused_window_number = -1;
Fluxbox * const fb = Fluxbox::instance();
2002-10-25 20:58:14 +00:00
2002-12-01 13:42:15 +00:00
if (fb->getFocusedWindow())
2003-05-11 13:36:12 +00:00
if (fb->getFocusedWindow()->screen().getScreenNumber() ==
2002-12-01 13:42:15 +00:00
getScreenNumber()) {
have_focused = true;
focused_window_number = fb->getFocusedWindow()->getWindowNumber();
}
2003-05-11 17:14:41 +00:00
if ((getCurrentWorkspace()->numberOfWindows() > 1) && have_focused)
2003-02-09 14:11:14 +00:00
fb->getFocusedWindow()->raise();
2001-12-11 20:47:02 +00:00
}
void BScreen::setFocusedWindow(WinClient &winclient) {
// raise newly focused window to the top of the focused list
if (!cycling_focus) { // don't change the order if we're cycling
focused_list.remove(&winclient);
focused_list.push_front(&winclient);
cycling_window = focused_list.begin();
}
}
void BScreen::dirFocus(FluxboxWindow &win, FocusDir dir) {
// change focus to the window in direction dir from the given window
// we scan through the list looking for the window that is "closest"
// in the given direction
FluxboxWindow *foundwin = 0;
int weight = 999999, exposure = 0; // extreme values
int borderW = m_root_theme->borderWidth(),
top = win.getYFrame(),
2003-05-11 13:36:12 +00:00
bottom = win.getYFrame() + win.height() + 2*borderW,
left = win.getXFrame(),
2003-05-11 13:36:12 +00:00
right = win.getXFrame() + win.width() + 2*borderW;
2003-05-11 17:14:41 +00:00
Workspace::Windows &wins = getCurrentWorkspace()->windowList();
Workspace::Windows::iterator it = wins.begin();
for (; it != wins.end(); ++it) {
if ((*it) == &win) continue; // skip self
// we check things against an edge, and within the bounds (draw a picture)
int edge=0, upper=0, lower=0, oedge=0, oupper=0, olower=0;
int otop = (*it)->getYFrame(),
2003-05-11 13:36:12 +00:00
obottom = (*it)->getYFrame() + (*it)->height() + 2*borderW,
oleft = (*it)->getXFrame(),
2003-05-11 13:36:12 +00:00
oright = (*it)->getXFrame() + (*it)->width() + 2*borderW;
// check if they intersect
switch (dir) {
case FOCUSUP:
edge = obottom;
oedge = bottom;
upper = left;
oupper = oleft;
lower = right;
olower = oright;
break;
case FOCUSDOWN:
edge = top;
oedge = otop;
upper = left;
oupper = oleft;
lower = right;
olower = oright;
break;
case FOCUSLEFT:
edge = oright;
oedge = right;
upper = top;
oupper = otop;
lower = bottom;
olower = obottom;
break;
case FOCUSRIGHT:
edge = left;
oedge = oleft;
upper = top;
oupper = otop;
lower = bottom;
olower = obottom;
break;
}
if (oedge < edge) continue; // not in the right direction
if (olower <= upper || oupper >= lower) {
// outside our horz bounds, get a heavy weight penalty
int myweight = 100000 + oedge - edge + abs(upper-oupper)+abs(lower-olower);
if (myweight < weight) {
foundwin = *it;
exposure = 0;
weight = myweight;
}
} else if ((oedge - edge) < weight) {
foundwin = *it;
weight = oedge - edge;
exposure = ((lower < olower)?lower:olower) - ((upper > oupper)?upper:oupper);
} else if (foundwin && oedge - edge == weight) {
int myexp = ((lower < olower)?lower:olower) - ((upper > oupper)?upper:oupper);
if (myexp > exposure) {
foundwin = *it;
// weight is same
exposure = myexp;
}
} // else not improvement
}
if (foundwin)
foundwin->setInputFocus();
}
void BScreen::initMenu() {
2002-12-01 13:42:15 +00:00
I18n *i18n = I18n::instance();
2001-12-11 20:47:02 +00:00
2002-12-13 20:19:05 +00:00
if (m_rootmenu.get()) {
2003-04-26 15:00:25 +00:00
// since all menus in root is submenus in m_rootmenu
// just remove every item in m_rootmenu and then clear rootmenuList
2002-12-13 20:19:05 +00:00
while (m_rootmenu->numberOfItems())
2003-04-18 12:51:14 +00:00
m_rootmenu->remove(0);
2003-04-26 15:00:25 +00:00
rootmenuList.clear();
2002-12-01 13:42:15 +00:00
} else
m_rootmenu.reset(createMenuFromScreen(*this));
2002-12-01 13:42:15 +00:00
bool defaultMenu = true;
Fluxbox * const fb = Fluxbox::instance();
if (fb->getMenuFilename()) {
ifstream menu_file(fb->getMenuFilename());
if (!menu_file.fail()) {
if (! menu_file.eof()) {
string line;
int row = 0;
while (getline(menu_file, line) && ! menu_file.eof()) {
row++;
if (line[0] != '#') {
string key;
int pos=0;
2003-04-26 18:27:56 +00:00
int err = FbTk::StringUtil::
getStringBetween(key,
line.c_str(),
'[', ']');
2002-01-27 12:45:32 +00:00
2002-12-01 13:42:15 +00:00
if (key == "begin") {
pos += err;
string label;
2003-04-26 18:27:56 +00:00
err = FbTk::StringUtil::
getStringBetween(label,
line.c_str()+pos,
'(', ')');
2002-12-01 13:42:15 +00:00
if (err>0) {
2002-12-13 20:19:05 +00:00
m_rootmenu->setLabel(label.c_str());
defaultMenu = parseMenuFile(menu_file, *m_rootmenu.get(), row);
2002-12-01 13:42:15 +00:00
} else
cerr<<"Error in menufile. Line("<<row<<")"<<endl;
break;
}
}
}
} else {
fprintf(stderr,
i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenEmptyMenuFile,
"%s: Empty menu file"),
2002-12-01 13:42:15 +00:00
fb->getMenuFilename());
}
menu_file.close();
} else
perror(fb->getMenuFilename());
}
if (defaultMenu) {
FbTk::RefCount<FbTk::Command> restart_fb(new FbCommands::RestartFluxboxCmd());
FbTk::RefCount<FbTk::Command> exit_fb(new FbCommands::ExitFluxboxCmd());
2003-04-28 00:38:42 +00:00
FbTk::RefCount<FbTk::Command> execute_xterm(new FbCommands::ExecuteCmd("xterm", getScreenNumber()));
2002-12-13 20:19:05 +00:00
m_rootmenu->setInternalMenu();
m_rootmenu->insert(i18n->getMessage(
FBNLS::ScreenSet, FBNLS::Screenxterm,
"xterm"),
execute_xterm);
2002-12-13 20:19:05 +00:00
m_rootmenu->insert(i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenRestart,
"Restart"),
restart_fb);
2002-12-13 20:19:05 +00:00
m_rootmenu->insert(i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenExit,
"Exit"),
exit_fb);
2002-12-01 13:42:15 +00:00
} else
fb->saveMenuFilename(fb->getMenuFilename());
2001-12-11 20:47:02 +00:00
}
/// looks through a menufile and adds correct items to the root-menu.
bool BScreen::parseMenuFile(ifstream &file, FbTk::Menu &menu, int &row) {
2002-01-27 12:45:32 +00:00
2002-12-01 13:42:15 +00:00
string line;
FbTk::RefCount<FbTk::Command>
hide_menu(new FbTk::SimpleCommand<FbTk::Menu>(menu, &FbTk::Menu::hide));
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
while (! file.eof()) {
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
if (getline(file, line)) {
row++;
if (line[0] != '#') { //the line is commented
int parse_pos = 0, err = 0;
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
std::string str_key, str_label, str_cmd;
2002-01-27 12:45:32 +00:00
2003-04-26 18:27:56 +00:00
err = FbTk::StringUtil::
getStringBetween(str_key,
line.c_str(),
'[', ']');
2002-12-01 13:42:15 +00:00
if (err > 0 ) {
parse_pos += err;
2003-04-26 18:27:56 +00:00
err = FbTk::StringUtil::
getStringBetween(str_label,
line.c_str() + parse_pos,
'(', ')');
2002-12-01 13:42:15 +00:00
if (err>0) {
parse_pos += err;
2003-04-26 18:27:56 +00:00
FbTk::StringUtil::
getStringBetween(str_cmd,
line.c_str() + parse_pos,
'{', '}');
2002-12-01 13:42:15 +00:00
}
} else
continue; //read next line
2002-01-27 12:45:32 +00:00
2002-12-01 13:42:15 +00:00
if (!str_key.size())
continue; //read next line
I18n *i18n = I18n::instance();
if (str_key == "end") {
return ((menu.numberOfItems() == 0) ? true : false);
2002-12-01 13:42:15 +00:00
} else if (str_key == "nop") {
menu.insert(str_label.c_str());
2002-12-01 13:42:15 +00:00
} else if (str_key == "exec") { // exec
if (!(str_label.size() && str_cmd.size())) {
fprintf(stderr,
i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenEXECError,
"BScreen::parseMenuFile: [exec] error, "
"no menu label and/or command defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else {
2003-04-28 00:38:42 +00:00
FbTk::RefCount<FbTk::Command> exec_cmd(new FbCommands::ExecuteCmd(str_cmd, getScreenNumber()));
FbTk::MacroCommand *exec_and_hide = new FbTk::MacroCommand();
exec_and_hide->add(hide_menu);
exec_and_hide->add(exec_cmd);
FbTk::RefCount<FbTk::Command> exec_and_hide_cmd(exec_and_hide);
menu.insert(str_label.c_str(), exec_and_hide_cmd);
}
2002-12-01 13:42:15 +00:00
} else if (str_key == "exit") { // exit
if (!str_label.size()) {
fprintf(stderr,
i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenEXITError,
"BScreen::parseMenuFile: [exit] error, "
"no menu label defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else {
FbTk::RefCount<FbTk::Command> exit_fb_cmd(new FbCommands::ExitFluxboxCmd());
menu.insert(str_label.c_str(), exit_fb_cmd);
2002-12-01 13:42:15 +00:00
}
} else if (str_key == "style") { // style
if (!( str_label.size() && str_cmd.size())) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenSTYLEError,
"BScreen::parseMenuFile: [style] error, "
"no menu label and/or filename defined\n"));
cerr<<"Row: "<<row<<endl;
} else {
// perform shell style ~ home directory expansion
// and insert style
FbTk::RefCount<FbTk::Command>
setstyle_cmd(new FbCommands::
2003-04-26 18:27:56 +00:00
SetStyleCmd(FbTk::StringUtil::
expandFilename(str_cmd)));
menu.insert(str_label.c_str(), setstyle_cmd);
}
} else if (str_key == "config") {
if (! str_label.size()) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenCONFIGError,
"BScreen::parseMenufile: [config] error, "
"no label defined"));
cerr<<"Row: "<<row<<endl;
} else {
2003-04-26 15:00:25 +00:00
#ifdef DEBUG
cerr<<__FILE__<<"("<<__FUNCTION__<<
"): inserts configmenu: "<<m_configmenu.get()<<endl;
#endif // DEBUG
menu.insert(str_label.c_str(), m_configmenu.get());
}
} // end of config
2002-12-01 13:42:15 +00:00
else if ( str_key == "include") { // include
if (!str_label.size()) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenINCLUDEError,
"BScreen::parseMenuFile: [include] error, "
"no filename defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else { // start of else 'x'
// perform shell style ~ home directory expansion
2003-04-26 18:27:56 +00:00
string newfile(FbTk::StringUtil::expandFilename(str_label));
2002-12-01 13:42:15 +00:00
if (newfile.size() != 0) {
FILE *submenufile = fopen(newfile.c_str(), "r");
if (submenufile) {
struct stat buf;
if (fstat(fileno(submenufile), &buf) ||
(! S_ISREG(buf.st_mode))) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet,
FBNLS::ScreenINCLUDEErrorReg,
"BScreen::parseMenuFile: [include] error: "
"'%s' is not a regular file\n"),
newfile.c_str());
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
}
2002-01-27 12:45:32 +00:00
2002-12-01 13:42:15 +00:00
if (! feof(submenufile)) {
fclose(submenufile);
ifstream subfile(newfile.c_str());
if (! parseMenuFile(subfile, menu, row))
Fluxbox::instance()->saveMenuFilename(newfile.c_str());
}
} else
perror(newfile.c_str());
}
} // end of else 'x'
} // end of include
else if (str_key == "submenu") { // sub
if (!str_label.size()) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenSUBMENUError,
"BScreen::parseMenuFile: [submenu] error, "
"no menu label defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else {
FbTk::Menu *submenu = createMenuFromScreen(*this);
2002-12-01 13:42:15 +00:00
if (str_cmd.size())
submenu->setLabel(str_cmd.c_str());
else
submenu->setLabel(str_label.c_str());
parseMenuFile(file, *submenu, row);
2002-12-01 13:42:15 +00:00
submenu->update();
menu.insert(str_label.c_str(), submenu);
// save to list so we can delete it later
2002-12-01 13:42:15 +00:00
rootmenuList.push_back(submenu);
2002-12-01 13:42:15 +00:00
}
} // end of sub
else if (str_key == "restart") {
if (!str_label.size()) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenRESTARTError,
"BScreen::parseMenuFile: [restart] error, "
"no menu label defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else {
FbTk::RefCount<FbTk::Command> restart_fb(new FbCommands::RestartFluxboxCmd());
menu.insert(str_label.c_str(), restart_fb);
2002-12-01 13:42:15 +00:00
}
} // end of restart
else if (str_key == "reconfig") { // reconf
if (!str_label.c_str()) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenRECONFIGError,
"BScreen::parseMenuFile: [reconfig] error, "
"no menu label defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else {
FbTk::RefCount<FbTk::Command>
reconfig_fb_cmd(new FbCommands::ReconfigureFluxboxCmd());
menu.insert(str_label.c_str(), reconfig_fb_cmd);
}
} else if (str_key == "stylesdir" || str_key == "stylesmenu") {
bool newmenu = (str_key == "stylesmenu");
if (!( str_label.size() && str_cmd.size()) && newmenu) {
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenSTYLESDIRError,
"BScreen::parseMenuFile: [stylesdir/stylesmenu]"
" error, no directory defined\n"));
cerr<<"Row: "<<row<<endl;
} else {
createStyleMenu(menu, str_label.c_str(),
newmenu ? str_cmd.c_str() : str_label.c_str());
}
} // end of stylesdir
2002-12-01 13:42:15 +00:00
else if (str_key == "workspaces") {
if (!str_label.size()) {
fprintf(stderr,
i18n->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenWORKSPACESError,
"BScreen:parseMenuFile: [workspaces] error, "
"no menu label defined\n"));
2002-12-01 13:42:15 +00:00
cerr<<"Row: "<<row<<endl;
} else
menu.insert(str_label.c_str(), workspacemenu.get());
2002-12-01 13:42:15 +00:00
} // end of work
}
}
}
return ((menu.numberOfItems() == 0) ? true : false);
2001-12-11 20:47:02 +00:00
}
void BScreen::setupConfigmenu(FbTk::Menu &menu) {
I18n *i18n = I18n::instance();
using namespace FBNLS;
FbTk::MacroCommand *s_a_reconf_macro = new FbTk::MacroCommand();
FbTk::RefCount<FbTk::Command> saverc_cmd(new FbTk::SimpleCommand<Fluxbox>(*Fluxbox::instance(),
&Fluxbox::save_rc));
FbTk::RefCount<FbTk::Command> reconf_cmd(new FbCommands::ReconfigureFluxboxCmd());
s_a_reconf_macro->add(saverc_cmd);
s_a_reconf_macro->add(reconf_cmd);
FbTk::RefCount<FbTk::Command> save_and_reconfigure(s_a_reconf_macro);
// create focus menu
2003-04-26 15:00:25 +00:00
// we don't set this to internal menu so will
// be deleted toghether with the parent
FbTk::Menu *focus_menu = createMenuFromScreen(*this);
focus_menu->insert(new FocusModelMenuItem(i18n->getMessage(
ConfigmenuSet,
ConfigmenuClickToFocus,
"Click To Focus"),
*this,
Fluxbox::CLICKTOFOCUS,
save_and_reconfigure));
focus_menu->insert(new FocusModelMenuItem(i18n->getMessage(
ConfigmenuSet,
ConfigmenuSloppyFocus,
"Sloppy Focus"),
*this,
Fluxbox::SLOPPYFOCUS,
save_and_reconfigure));
focus_menu->insert(new FocusModelMenuItem(i18n->getMessage(
ConfigmenuSet,
ConfigmenuSemiSloppyFocus,
"Semi Sloppy Focus"),
*this,
Fluxbox::SEMISLOPPYFOCUS,
save_and_reconfigure));
focus_menu->insert(new BoolMenuItem(i18n->getMessage(
ConfigmenuSet,
ConfigmenuAutoRaise,
"Auto Raise"),
*resource.auto_raise,
save_and_reconfigure));
focus_menu->update();
menu.insert(i18n->getMessage(
ConfigmenuSet, ConfigmenuFocusModel,
"Focus Model"),
focus_menu);
#ifdef SLIT
2003-04-26 15:00:25 +00:00
if (getSlit() != 0) {
getSlit()->menu().setInternalMenu();
menu.insert("Slit", &getSlit()->menu());
2003-04-26 15:00:25 +00:00
}
#endif // SLIT
menu.insert(i18n->getMessage(
ToolbarSet, ToolbarToolbarTitle,
"Toolbar"), &m_toolbarhandler->getToolbarMenu());
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuImageDithering,
"Image Dithering"),
*resource.image_dither, save_and_reconfigure));
menu.insert(new
BoolMenuItem(
i18n->getMessage(
ConfigmenuSet, ConfigmenuOpaqueMove,
"Opaque Window Moving"),
*resource.opaque_move, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuFullMax,
"Full Maximization"),
*resource.full_max, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuFocusNew,
"Focus New Windows"),
*resource.focus_new, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuFocusLast,
"Focus Last Window on Workspace"),
*resource.focus_last, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuMaxOverSlit,
"Maximize Over Slit"),
*resource.max_over_slit, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuWorkspaceWarping,
"Workspace Warping"),
*resource.workspace_warping, save_and_reconfigure));
menu.insert(new
BoolMenuItem(i18n->getMessage(
ConfigmenuSet, ConfigmenuDesktopWheeling,
"Desktop MouseWheel Switching"),
*resource.desktop_wheeling, save_and_reconfigure));
menu.insert(new BoolMenuItem("Click Raises",
*resource.click_raises,
2003-04-20 13:46:18 +00:00
save_and_reconfigure));
// setup antialias cmd to reload style and save resource on toggle
menu.insert(new BoolMenuItem("antialias", *resource.antialias,
save_and_reconfigure));
2003-04-20 13:46:18 +00:00
FbTk::MenuItem *menu_alpha_item = new IntResMenuItem("Menu Alpha", resource.menu_alpha,
0, 255);
menu_alpha_item->setCommand(saverc_cmd);
menu.insert(menu_alpha_item);
// finaly update menu
menu.update();
}
void BScreen::createStyleMenu(FbTk::Menu &menu,
const char *label, const char *directory) {
2002-12-01 13:42:15 +00:00
// perform shell style ~ home directory expansion
2003-04-26 18:27:56 +00:00
string stylesdir(FbTk::StringUtil::expandFilename(directory ? directory : ""));
2002-12-01 13:42:15 +00:00
I18n *i18n = I18n::instance();
struct stat statbuf;
2002-02-17 18:56:14 +00:00
if (! stat(stylesdir.c_str(), &statbuf)) {
if (S_ISDIR(statbuf.st_mode)) { // is a directory?
2002-02-17 18:56:14 +00:00
DirHelper d(stylesdir.c_str());
// create a vector of all the filenames in the directory
// add sort it
std::vector<std::string> filelist(d.entries());
for (size_t file_index = 0; file_index < d.entries(); ++file_index)
filelist[file_index] = d.readFilename();
std::sort(filelist.begin(), filelist.end(), less<string>());
2002-12-01 13:42:15 +00:00
int slen = stylesdir.size();
// for each file in directory add filename and path to menu
for (size_t file_index = 0; file_index < d.entries(); file_index++) {
int nlen = filelist[file_index].size();
2002-12-01 13:42:15 +00:00
char style[MAXPATHLEN + 1];
2002-12-01 13:42:15 +00:00
strncpy(style, stylesdir.c_str(), slen);
*(style + slen) = '/';
strncpy(style + slen + 1, filelist[file_index].c_str(), nlen + 1);
if ( !stat(style, &statbuf) && S_ISREG(statbuf.st_mode)) {
FbTk::RefCount<FbTk::Command> setstyle_cmd(new FbCommands::
SetStyleCmd(style));
menu.insert(filelist[file_index].c_str(), setstyle_cmd);
}
2002-12-01 13:42:15 +00:00
}
// update menu graphics
menu.update();
2002-12-01 13:42:15 +00:00
Fluxbox::instance()->saveMenuFilename(stylesdir.c_str());
} else { // no directory
2002-12-01 13:42:15 +00:00
fprintf(stderr,
i18n->
getMessage(
FBNLS::ScreenSet, FBNLS::ScreenSTYLESDIRErrorNotDir,
"BScreen::parseMenuFile:"
" [stylesdir/stylesmenu] error, %s is not a"
" directory\n"), stylesdir.c_str());
} // end of directory check
} else { // stat failed
2002-12-01 13:42:15 +00:00
fprintf(stderr,
2002-02-17 18:56:14 +00:00
i18n->
getMessage(
2002-12-01 13:42:15 +00:00
FBNLS::ScreenSet, FBNLS::ScreenSTYLESDIRErrorNoExist,
"BScreen::parseMenuFile: [stylesdir/stylesmenu]"
" error, %s does not exist\n"), stylesdir.c_str());
} // end of stat
2002-02-17 18:56:14 +00:00
}
2001-12-11 20:47:02 +00:00
void BScreen::shutdown() {
2003-04-14 15:01:55 +00:00
Display *disp = FbTk::App::instance()->display();
2003-05-10 22:59:32 +00:00
rootWindow().setEventMask(NoEventMask);
2003-04-14 15:01:55 +00:00
XSync(disp, False);
2003-04-14 15:01:55 +00:00
for_each(workspacesList.begin(),
workspacesList.end(),
mem_fun(&Workspace::shutdown));
#ifdef SLIT
2002-12-01 13:42:15 +00:00
if (m_slit.get())
m_slit->shutdown();
#endif // SLIT
2001-12-11 20:47:02 +00:00
}
void BScreen::showPosition(int x, int y) {
2002-12-01 13:42:15 +00:00
if (! geom_visible) {
2002-03-19 14:30:43 +00:00
#ifdef XINERAMA
2002-12-01 13:42:15 +00:00
unsigned int head = hasXinerama() ? getCurrHead() : 0;
2002-03-19 14:30:43 +00:00
geom_window.moveResize(getHeadX(head) + (getHeadWidth(head) - geom_w) / 2,
getHeadY(head) + (getHeadHeight(head) - geom_h) / 2,
geom_w, geom_h);
2002-03-19 14:30:43 +00:00
#else // !XINERMA
geom_window.moveResize((getWidth() - geom_w) / 2,
(getHeight() - geom_h) / 2, geom_w, geom_h);
2002-03-19 14:30:43 +00:00
#endif // XINERAMA
geom_window.show();
geom_window.raise();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
geom_visible = true;
}
char label[256];
2001-12-11 20:47:02 +00:00
sprintf(label,
2002-12-01 13:42:15 +00:00
I18n::instance()->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenPositionFormat,
"X: %4d x Y: %4d"), x, y);
geom_window.clear();
winFrameTheme().font().drawText(
geom_window.window(),
getScreenNumber(),
winFrameTheme().labelTextFocusGC(),
label, strlen(label),
m_root_theme->bevelWidth(),
m_root_theme->bevelWidth() +
winFrameTheme().font().ascent());
2001-12-11 20:47:02 +00:00
}
void BScreen::showGeometry(unsigned int gx, unsigned int gy) {
2002-12-01 13:42:15 +00:00
if (! geom_visible) {
2002-03-19 14:30:43 +00:00
#ifdef XINERAMA
2002-12-01 13:42:15 +00:00
unsigned int head = hasXinerama() ? getCurrHead() : 0;
2002-03-19 14:30:43 +00:00
geom_window.moveResize(getHeadX(head) + (getHeadWidth(head) - geom_w) / 2,
getHeadY(head) + (getHeadHeight(head) - geom_h) / 2,
geom_w, geom_h);
2002-03-19 14:30:43 +00:00
#else // !XINERMA
geom_window.moveResize((getWidth() - geom_w) / 2,
(getHeight() - geom_h) / 2, geom_w, geom_h);
2002-03-19 14:30:43 +00:00
#endif // XINERAMA
geom_window.show();
geom_window.raise();
2001-12-11 20:47:02 +00:00
2002-12-01 13:42:15 +00:00
geom_visible = true;
}
2001-12-11 20:47:02 +00:00
char label[256];
2002-12-01 13:42:15 +00:00
sprintf(label,
I18n::instance()->getMessage(
FBNLS::ScreenSet, FBNLS::ScreenGeometryFormat,
"W: %4d x H: %4d"), gx, gy);
2002-12-01 13:42:15 +00:00
geom_window.clear();
2002-12-01 13:42:15 +00:00
//TODO: geom window again?! repeated
winFrameTheme().font().drawText(geom_window.window(),
getScreenNumber(),
winFrameTheme().labelTextFocusGC(),
label, strlen(label),
m_root_theme->bevelWidth(),
m_root_theme->bevelWidth() +
winFrameTheme().font().ascent());
2001-12-11 20:47:02 +00:00
}
void BScreen::hideGeometry() {
2002-12-01 13:42:15 +00:00
if (geom_visible) {
geom_window.hide();
2002-12-01 13:42:15 +00:00
geom_visible = false;
}
2001-12-11 20:47:02 +00:00
}
void BScreen::setLayer(FbTk::XLayerItem &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(const int delta) {
2003-05-11 17:14:41 +00:00
changeWorkspaceID( (getCurrentWorkspaceID() + delta) % getCount());
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(const int delta) {
2003-05-11 17:14:41 +00:00
changeWorkspaceID( (getCurrentWorkspaceID() - delta + getCount()) % getCount());
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(const int delta) {
2002-12-01 13:42:15 +00:00
if (getCurrentWorkspaceID()+delta < getCount())
changeWorkspaceID(getCurrentWorkspaceID()+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(const int delta) {
2002-12-01 13:42:15 +00:00
if (getCurrentWorkspaceID() >= static_cast<unsigned int>(delta))
changeWorkspaceID(getCurrentWorkspaceID()-delta);
2002-02-02 19:51:15 +00:00
}
2002-02-07 14:46:23 +00:00
2002-11-27 21:55:36 +00:00
/**
@return true if the windows should be skiped else false
*/
bool BScreen::doSkipWindow(const FluxboxWindow *w, int opts) {
2002-12-01 13:42:15 +00:00
return ((opts & CYCLESKIPSTUCK) != 0 && w->isStuck() || // skip if stuck
/* (opts & CYCLESKIPLOWERTABS) != 0 && w->isLowerTab() || // skip if lower tab
*/
2002-12-01 13:42:15 +00:00
(opts & CYCLESKIPSHADED) != 0 && w->isShaded()); // skip if shaded
}
/**
Called when a set of watched modifiers has been released
*/
void BScreen::notifyReleasedKeys(XKeyEvent &ke) {
if (cycling_focus) {
cycling_focus = false;
cycling_last = 0;
// put currently focused window to top
WinClient *client = *cycling_window;
focused_list.erase(cycling_window);
focused_list.push_front(client);
client->fbwindow()->raise();
}
}
2003-05-04 23:38:06 +00:00
/**
* Used to find out which window was last focused on the given workspace
* If workspace is outside the ID range, then the absolute last focused window
* is given.
*/
WinClient *BScreen::getLastFocusedWindow(int workspace) {
if (focused_list.empty()) return 0;
if (workspace < 0 || workspace >= (int) getCount())
return focused_list.front();
FocusedWindows::iterator it = focused_list.begin();
FocusedWindows::iterator it_end = focused_list.end();
for (; it != it_end; ++it)
if ((*it)->fbwindow() &&
(((int)(*it)->fbwindow()->getWorkspaceNumber()) == workspace
&& (!(*it)->fbwindow()->isStuck() || (*it)->fbwindow()->isFocused())))
// only give focus to a stuck window if it is currently focused
// otherwise they tend to override normal workspace focus
2003-05-04 23:38:06 +00:00
return *it;
return 0;
}
2002-11-27 21:55:36 +00:00
/**
Access and clear the auto-group window
*/
FluxboxWindow* BScreen::useAutoGroupWindow() {
2002-12-01 13:42:15 +00:00
Window w = auto_group_window;
auto_group_window = 0;
return w ? Fluxbox::instance()->searchWindow(w) : 0;
}